Lightweight 0.1.0
Loading...
Searching...
No Matches
SqlConnection.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 "Api.hpp"
10#include "SqlConnectInfo.hpp"
11#include "SqlError.hpp"
12#include "SqlLogger.hpp"
13#include "SqlServerType.hpp"
14
15#include <atomic>
16#include <chrono>
17#include <expected>
18#include <format>
19#include <functional>
20#include <memory>
21#include <optional>
22#include <string>
23#include <string_view>
24#include <system_error>
25
26#include <sql.h>
27#include <sqlext.h>
28#include <sqlspi.h>
29#include <sqltypes.h>
30
31class SqlQueryBuilder;
34
35/// @brief Represents a connection to a SQL database.
36class LIGHTWEIGHT_API SqlConnection final
37{
38 public:
39 /// @brief Constructs a new SQL connection to the default connection.
40 ///
41 /// The default connection is set via SetDefaultConnectInfo.
42 /// In case the default connection is not set, the connection will fail.
43 /// And in case the connection fails, the last error will be set.
45
46 /// @brief Constructs a new SQL connection to the given connect informaton.
47 ///
48 /// @param connectInfo The connection information to use. If not provided,
49 /// no connection will be established.
50 explicit SqlConnection(std::optional<SqlConnectionString> connectInfo);
51
52 SqlConnection(SqlConnection&& /*other*/) noexcept;
53 SqlConnection& operator=(SqlConnection&& /*other*/) noexcept;
54 SqlConnection(SqlConnection const& /*other*/) = delete;
55 SqlConnection& operator=(SqlConnection const& /*other*/) = delete;
56
57 /// Destructs this SQL connection object,
58 ~SqlConnection() noexcept;
59
60 /// Retrieves the default connection information.
61 static SqlConnectionString const& DefaultConnectionString() noexcept;
62
63 /// Sets the default connection information.
64 ///
65 /// @param connectionString The connection information to use.
66 static void SetDefaultConnectionString(SqlConnectionString const& connectionString) noexcept;
67
68 /// Sets the default connection information as SqlConnectionDataSource.
69 static void SetDefaultDataSource(SqlConnectionDataSource const& dataSource) noexcept;
70
71 /// Sets a callback to be called after each connection being established.
72 static void SetPostConnectedHook(std::function<void(SqlConnection&)> hook);
73
74 /// Resets the post connected hook.
75 static void ResetPostConnectedHook();
76
77 /// @brief Retrieves the connection ID.
78 ///
79 /// This is a unique identifier for the connection, which is useful for debugging purposes.
80 /// Note, this ID will not change if the connection is moved nor when it is reused via the connection pool.
81 [[nodiscard]] uint64_t ConnectionId() const noexcept
82 {
83 return m_connectionId;
84 }
85
86 /// Closes the connection (attempting to put it back into the connect[[ion pool).
87 void Close() noexcept;
88
89 /// Connects to the given database with the given username and password.
90 ///
91 /// @retval true if the connection was successful.
92 /// @retval false if the connection failed. Use LastError() to retrieve the error information.
93 bool Connect(SqlConnectionDataSource const& info) noexcept;
94
95 /// Connects to the given database with the given username and password.
96 ///
97 /// @retval true if the connection was successful.
98 /// @retval false if the connection failed. Use LastError() to retrieve the error information.
99 bool Connect(SqlConnectionString sqlConnectionString) noexcept;
100
101 /// Retrieves the last error information with respect to this SQL connection handle.
102 [[nodiscard]] SqlErrorInfo LastError() const;
103
104 /// Retrieves the name of the database in use.
105 [[nodiscard]] std::string DatabaseName() const;
106
107 /// Retrieves the name of the user.
108 [[nodiscard]] std::string UserName() const;
109
110 /// Retrieves the name of the server.
111 [[nodiscard]] std::string ServerName() const;
112
113 /// Retrieves the reported server version.
114 [[nodiscard]] std::string ServerVersion() const;
115
116 /// Retrieves the type of the server.
117 [[nodiscard]] SqlServerType ServerType() const noexcept;
118
119 /// Retrieves a query formatter suitable for the SQL server being connected.
120 [[nodiscard]] SqlQueryFormatter const& QueryFormatter() const noexcept;
121
122 /// Creates a new query builder for the given table, compatible with the current connection.
123 ///
124 /// @param table The table to query.
125 /// If not provided, the query will be a generic query builder.
126 [[nodiscard]] SqlQueryBuilder Query(std::string_view const& table = {}) const;
127
128 /// Creates a new query builder for the given table with an alias, compatible with the current connection.
129 ///
130 /// @param table The table to query.
131 /// @param tableAlias The alias to use for the table.
132 [[nodiscard]] SqlQueryBuilder QueryAs(std::string_view const& table, std::string_view const& tableAlias) const;
133
134 /// Creates a new migration query builder, compatible the current connection.
135 [[nodiscard]] SqlMigrationQueryBuilder Migration() const;
136
137 /// Tests if a transaction is active.
138 [[nodiscard]] bool TransactionActive() const noexcept;
139
140 /// Tests if transactions are allowed.
141 [[nodiscard]] bool TransactionsAllowed() const noexcept;
142
143 /// Tests if the connection is still active.
144 [[nodiscard]] bool IsAlive() const noexcept;
145
146 /// Retrieves the connection information.
147 [[nodiscard]] SqlConnectionString const& ConnectionString() const noexcept;
148
149 /// Retrieves the native handle.
150 [[nodiscard]] SQLHDBC NativeHandle() const noexcept
151 {
152 return m_hDbc;
153 }
154
155 /// Retrieves the last time the connection was used.
156 [[nodiscard]] std::chrono::steady_clock::time_point LastUsed() const noexcept;
157
158 /// Sets the last time the connection was used.
159 void SetLastUsed(std::chrono::steady_clock::time_point lastUsed) noexcept;
160
161 /// Checks the result of an SQL operation, and throws an exception if it is not successful.
162 void RequireSuccess(SQLRETURN sqlResult,
163 std::source_location sourceLocation = std::source_location::current()) const;
164
165 private:
166 void PostConnect();
167
168 // Private data members
169 SQLHENV m_hEnv {};
170 SQLHDBC m_hDbc {};
171 uint64_t m_connectionId;
172 SqlServerType m_serverType = SqlServerType::UNKNOWN;
173 SqlQueryFormatter const* m_queryFormatter {};
174
175 struct Data;
176 Data* m_data {};
177};
178
179inline SqlServerType SqlConnection::ServerType() const noexcept
180{
181 return m_serverType;
182}
183
185{
186 return *m_queryFormatter;
187}
Represents a connection to a SQL database.
SqlConnection()
Constructs a new SQL connection to the default connection.
SqlServerType ServerType() const noexcept
Retrieves the type of the server.
SqlConnection(std::optional< SqlConnectionString > connectInfo)
Constructs a new SQL connection to the given connect informaton.
SqlQueryBuilder QueryAs(std::string_view const &table, std::string_view const &tableAlias) const
bool TransactionActive() const noexcept
Tests if a transaction is active.
std::chrono::steady_clock::time_point LastUsed() const noexcept
Retrieves the last time the connection was used.
SqlMigrationQueryBuilder Migration() const
Creates a new migration query builder, compatible the current connection.
SqlQueryFormatter const & QueryFormatter() const noexcept
Retrieves a query formatter suitable for the SQL server being connected.
void Close() noexcept
Closes the connection (attempting to put it back into the connect[[ion pool).
Query builder for building SQL migration queries.
Definition Migrate.hpp:184
API Entry point for building SQL queries.
Definition SqlQuery.hpp:26
API to format SQL queries for different SQL dialects.
Represents a connection data source as a DSN, username, password, and timeout.
Represents an ODBC connection string.
Represents an ODBC SQL error.
Definition SqlError.hpp:29