Lightweight 0.1.0
Loading...
Searching...
No Matches
SqlError.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
11#include <format>
12#include <source_location>
13#include <stdexcept>
14#include <system_error>
15
16#include <sql.h>
17#include <sqlext.h>
18#include <sqlspi.h>
19#include <sqltypes.h>
20
21/// @brief Represents an ODBC SQL error.
22///
23/// NOTE: This is a simple wrapper around the SQL return codes. It is not meant to be
24/// comprehensive, but rather to provide a simple way to convert SQL return codes to
25/// std::error_code.
26///
27/// The code below is DRAFT and may be subject to change.
29{
30 SQLINTEGER nativeErrorCode {};
31 std::string sqlState = " "; // 5 characters + null terminator
32 std::string message;
33
34 /// Constructs an ODBC error info object from the given ODBC connection handle.
36 {
37 return fromHandle(SQL_HANDLE_DBC, hDbc);
38 }
39
40 /// Constructs an ODBC error info object from the given ODBC statement handle.
41 static SqlErrorInfo fromStatementHandle(SQLHSTMT hStmt)
42 {
43 return fromHandle(SQL_HANDLE_STMT, hStmt);
44 }
45
46 /// Asserts that the given result is a success code, otherwise throws an exception.
47 static void RequireStatementSuccess(SQLRETURN result, SQLHSTMT hStmt, std::string_view message);
48
49 private:
50 static SqlErrorInfo fromHandle(SQLSMALLINT handleType, SQLHANDLE handle)
51 {
52 SqlErrorInfo info {};
53 info.message = std::string(1024, '\0');
54
55 SQLSMALLINT msgLen {};
56 SQLGetDiagRecA(handleType,
57 handle,
58 1,
59 (SQLCHAR*) info.sqlState.data(),
60 &info.nativeErrorCode,
61 (SQLCHAR*) info.message.data(),
62 (SQLSMALLINT) info.message.capacity(),
63 &msgLen);
64 info.message.resize(msgLen);
65 return info;
66 }
67};
68
69class SqlException: public std::runtime_error
70{
71 public:
72 LIGHTWEIGHT_API explicit SqlException(SqlErrorInfo info,
73 std::source_location location = std::source_location::current());
74
75 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE SqlErrorInfo const& info() const noexcept
76 {
77 return _info;
78 }
79
80 private:
81 SqlErrorInfo _info;
82};
83
84enum class SqlError : std::int16_t
85{
86 SUCCESS = SQL_SUCCESS,
87 SUCCESS_WITH_INFO = SQL_SUCCESS_WITH_INFO,
88 NODATA = SQL_NO_DATA,
89 FAILURE = SQL_ERROR,
90 INVALID_HANDLE = SQL_INVALID_HANDLE,
91 STILL_EXECUTING = SQL_STILL_EXECUTING,
92 NEED_DATA = SQL_NEED_DATA,
93 PARAM_DATA_AVAILABLE = SQL_PARAM_DATA_AVAILABLE,
94 NO_DATA_FOUND = SQL_NO_DATA_FOUND,
95 UNSUPPORTED_TYPE = 1'000,
96 INVALID_ARGUMENT = 1'001,
97 TRANSACTION_ERROR = 1'002,
98};
99
100struct SqlErrorCategory: std::error_category
101{
102 static SqlErrorCategory const& get() noexcept
103 {
104 static SqlErrorCategory const category;
105 return category;
106 }
107 [[nodiscard]] const char* name() const noexcept override
108 {
109 return "Lightweight";
110 }
111
112 [[nodiscard]] std::string message(int code) const override
113 {
114 using namespace std::string_literals;
115 switch (static_cast<SqlError>(code))
116 {
117 case SqlError::SUCCESS:
118 return "SQL_SUCCESS"s;
119 case SqlError::SUCCESS_WITH_INFO:
120 return "SQL_SUCCESS_WITH_INFO"s;
121 case SqlError::NODATA:
122 return "SQL_NO_DATA"s;
123 case SqlError::FAILURE:
124 return "SQL_ERROR"s;
125 case SqlError::INVALID_HANDLE:
126 return "SQL_INVALID_HANDLE"s;
127 case SqlError::STILL_EXECUTING:
128 return "SQL_STILL_EXECUTING"s;
129 case SqlError::NEED_DATA:
130 return "SQL_NEED_DATA"s;
131 case SqlError::PARAM_DATA_AVAILABLE:
132 return "SQL_PARAM_DATA_AVAILABLE"s;
133 case SqlError::UNSUPPORTED_TYPE:
134 return "SQL_UNSUPPORTED_TYPE"s;
135 case SqlError::INVALID_ARGUMENT:
136 return "SQL_INVALID_ARGUMENT"s;
137 case SqlError::TRANSACTION_ERROR:
138 return "SQL_TRANSACTION_ERROR"s;
139 }
140 return std::format("SQL error code {}", code);
141 }
142};
143
144// Register our enum as an error code so we can constructor error_code from it
145template <>
146struct std::is_error_code_enum<SqlError>: public std::true_type
147{
148};
149
150// Tells the compiler that MyErr pairs with MyCategory
151inline std::error_code make_error_code(SqlError e)
152{
153 return { static_cast<int>(e), SqlErrorCategory::get() };
154}
155
156template <>
157struct LIGHTWEIGHT_API std::formatter<SqlError>: formatter<std::string>
158{
159 auto format(SqlError value, format_context& ctx) const -> format_context::iterator
160 {
161 return formatter<std::string>::format(std::format("{}", SqlErrorCategory().message(static_cast<int>(value))),
162 ctx);
163 }
164};
165
166template <>
167struct LIGHTWEIGHT_API std::formatter<SqlErrorInfo>: formatter<std::string>
168{
169 auto format(SqlErrorInfo const& info, format_context& ctx) const -> format_context::iterator
170 {
171 return formatter<std::string>::format(
172 std::format("{} ({}) - {}", info.sqlState, info.nativeErrorCode, info.message), ctx);
173 }
174};
Represents an ODBC SQL error.
Definition SqlError.hpp:29
static SqlErrorInfo fromConnectionHandle(SQLHDBC hDbc)
Constructs an ODBC error info object from the given ODBC connection handle.
Definition SqlError.hpp:35
static void RequireStatementSuccess(SQLRETURN result, SQLHSTMT hStmt, std::string_view message)
Asserts that the given result is a success code, otherwise throws an exception.
static SqlErrorInfo fromStatementHandle(SQLHSTMT hStmt)
Constructs an ODBC error info object from the given ODBC statement handle.
Definition SqlError.hpp:41