Lightweight 0.1.0
Loading...
Searching...
No Matches
SqlTransaction.hpp
1// SPDX-License-Identifier: Apache-2.0
2
3#pragma once
4
5#if defined(_WIN32) || defined(_WIN64)
6 #include <Windows.h>
7#endif
8
9#include "SqlConnection.hpp"
10#include "SqlError.hpp"
11
12#include <format>
13#include <source_location>
14#include <stdexcept>
15
16#include <sql.h>
17#include <sqlext.h>
18#include <sqlspi.h>
19#include <sqltypes.h>
20
21/// Represents the isolation level of a SQL transaction.
22///
23/// The isolation level determines the degree of isolation of data between concurrent transactions.
24/// The higher the isolation level, the less likely it is that two transactions will interfere with each other.
25///
26/// @see https://learn.microsoft.com/en-us/sql/odbc/reference/develop-app/transaction-isolation-levels
27enum class SqlIsolationMode : std::uint8_t
28{
29 /// The default isolation level of the SQL driver or DBMS
30 DriverDefault = 0,
31
32 /// Transactions are not isolated from each other, allowing to concurrently read uncommitted changes.
33 ReadUncommitted = SQL_TXN_READ_UNCOMMITTED,
34
35 /// The transaction waits until the locked writes are committed by other transactions.
36 ///
37 /// The transaction holds a read lock (if it only reads the row) or write lock (if it updates
38 /// or deletes the row) on the current row to prevent other transactions from updating or deleting it.
39 ReadCommitted = SQL_TXN_READ_COMMITTED,
40
41 /// The transaction waits until the locked writes are committed by other transactions.
42 ///
43 /// The transaction holds read locks on all rows it returns to the application and write locks
44 /// on all rows it inserts, updates, or deletes.
45 RepeatableRead = SQL_TXN_REPEATABLE_READ,
46
47 /// The transaction waits until the locked writes are committed by other transactions.
48 ///
49 /// The transaction holds a read lock (if it only reads rows) or write lock (if it can update
50 /// or delete rows) on the range of rows it affects.
51 Serializable = SQL_TXN_SERIALIZABLE,
52};
53
54/// Represents the mode of a SQL transaction to be applied, if not done so explicitly.
55enum class SqlTransactionMode : std::uint8_t
56{
57 NONE,
58 COMMIT,
59 ROLLBACK,
60};
61
62/// Represents an exception that occurred during a SQL transaction.
63///
64/// @see SqlTransaction::Commit(), SqlTransaction::Rollback()
65class SqlTransactionException: public std::runtime_error
66{
67 public:
68 explicit SqlTransactionException(const std::string& message) noexcept:
69 std::runtime_error(message)
70 {
71 }
72};
73
74/// Represents a transaction to a SQL database.
75///
76/// This class is used to control the transaction manually. It disables the auto-commit mode when constructed,
77/// and automatically commits the transaction when destructed if not done so.
78///
79/// This class is designed with RAII in mind, so that the transaction is automatically committed or rolled back
80/// when the object goes out of scope.
81///
82/// @code
83///
84/// void doSomeWork(SqlTransaction& transaction)
85/// {
86/// auto stmt = SqlStatement { transaction.Connection() };
87/// stmt.ExecuteDirect("INSERT INTO table (column) VALUES (42)"); // Do some work
88/// }
89///
90/// void main()
91/// {
92/// auto connection = SqlConnection {};
93/// auto transaction = SqlTransaction { connection };
94///
95/// doSomeWork(transaction);
96///
97/// transaction.Commit();
98/// }
99/// @endcode
101{
102 public:
103 SqlTransaction() = delete;
104 SqlTransaction(SqlTransaction const&) = delete;
105 SqlTransaction& operator=(SqlTransaction const&) = delete;
106
107 SqlTransaction(SqlTransaction&&) = default;
108 SqlTransaction& operator=(SqlTransaction&&) = default;
109
110 /// Construct a new SqlTransaction object, and disable the auto-commit mode, so that the transaction can be
111 /// controlled manually.
112 LIGHTWEIGHT_API explicit SqlTransaction(SqlConnection& connection,
113 SqlTransactionMode defaultMode = SqlTransactionMode::COMMIT,
114 SqlIsolationMode isolationMode = SqlIsolationMode::DriverDefault,
115 std::source_location location = std::source_location::current());
116
117 /// Automatically commit the transaction if not done so
118 LIGHTWEIGHT_API ~SqlTransaction() noexcept;
119
120 /// Get the connection object associated with this transaction.
121 SqlConnection& Connection() noexcept;
122
123 /// Rollback the transaction. Throws an exception if the transaction cannot be rolled back.
124 LIGHTWEIGHT_API void Rollback();
125
126 /// Try to rollback the transaction, and return true if successful, falls otherwise
127 LIGHTWEIGHT_API bool TryRollback() noexcept;
128
129 /// Commit the transaction. Throws an exception if the transaction cannot be committed.
130 LIGHTWEIGHT_API void Commit();
131
132 /// Try to commit the transaction, and return true if successful, falls otherwise
133 LIGHTWEIGHT_API bool TryCommit() noexcept;
134
135 private:
136 /// Retrieves the native handle.
137 [[nodiscard]] SQLHDBC NativeHandle() const noexcept
138 {
139 return m_connection->NativeHandle();
140 }
141
142 private:
143 SqlConnection* m_connection {};
144 SqlTransactionMode m_defaultMode {};
145 std::source_location m_location {};
146};
147
149{
150 return *m_connection;
151}
152
153template <>
154struct LIGHTWEIGHT_API std::formatter<SqlTransactionMode>: std::formatter<std::string_view>
155{
156 auto format(SqlTransactionMode value, format_context& ctx) const -> format_context::iterator
157 {
158 using namespace std::string_view_literals;
159 string_view name;
160 switch (value)
161 {
162 case SqlTransactionMode::COMMIT:
163 name = "Commit";
164 break;
165 case SqlTransactionMode::ROLLBACK:
166 name = "Rollback";
167 break;
168 case SqlTransactionMode::NONE:
169 name = "None";
170 break;
171 }
172 return std::formatter<string_view>::format(name, ctx);
173 }
174};
175
176template <>
177struct LIGHTWEIGHT_API std::formatter<SqlIsolationMode>: std::formatter<std::string_view>
178{
179 auto format(SqlIsolationMode value, format_context& ctx) const -> format_context::iterator
180 {
181 using namespace std::string_view_literals;
182 string_view name;
183 switch (value)
184 {
185 case SqlIsolationMode::DriverDefault:
186 name = "DriverDefault";
187 break;
188 case SqlIsolationMode::ReadUncommitted:
189 name = "ReadUncommitted";
190 break;
191 case SqlIsolationMode::ReadCommitted:
192 name = "ReadCommitted";
193 break;
194 case SqlIsolationMode::RepeatableRead:
195 name = "RepeatableRead";
196 break;
197 case SqlIsolationMode::Serializable:
198 name = "Serializable";
199 break;
200 }
201 return std::formatter<string_view>::format(name, ctx);
202 }
203};
Represents a connection to a SQL database.
SQLHDBC NativeHandle() const noexcept
Retrieves the native handle.
LIGHTWEIGHT_API void Rollback()
Rollback the transaction. Throws an exception if the transaction cannot be rolled back.
LIGHTWEIGHT_API SqlTransaction(SqlConnection &connection, SqlTransactionMode defaultMode=SqlTransactionMode::COMMIT, SqlIsolationMode isolationMode=SqlIsolationMode::DriverDefault, std::source_location location=std::source_location::current())
LIGHTWEIGHT_API bool TryCommit() noexcept
Try to commit the transaction, and return true if successful, falls otherwise.
LIGHTWEIGHT_API bool TryRollback() noexcept
Try to rollback the transaction, and return true if successful, falls otherwise.
SqlConnection & Connection() noexcept
Get the connection object associated with this transaction.
LIGHTWEIGHT_API ~SqlTransaction() noexcept
Automatically commit the transaction if not done so.
LIGHTWEIGHT_API void Commit()
Commit the transaction. Throws an exception if the transaction cannot be committed.