Lightweight 0.1.0
Loading...
Searching...
No Matches
SqlNumeric.hpp
1// SPDX-License-Identifier: Apache-2.0
2
3#pragma once
4
5#include "../SqlColumnTypeDefinitions.hpp"
6#include "../SqlError.hpp"
7#include "../SqlLogger.hpp"
8#include "Primitives.hpp"
9
10#include <cmath>
11#include <compare>
12#include <concepts>
13#include <cstring>
14#include <print>
15#include <source_location>
16
17
18// clang-cl doesn't support __int128_t but defines __SIZEOF_INT128__
19// and also since it pretends to be MSVC, it also defines _MSC_VER
20// clang-format off
21#if defined(__SIZEOF_INT128__) && !defined(_MSC_VER)
22 #define LIGHTWEIGHT_INT128_T __int128_t
23 static_assert(sizeof(__int128_t) == sizeof(SQL_NUMERIC_STRUCT::val));
24#endif
25// clang-format on
26
27/// Represents a fixed-point number with a given precision and scale.
28///
29/// Precision is *exactly* the total number of digits in the number,
30/// including the digits after the decimal point.
31///
32/// Scale is the number of digits after the decimal point.
33///
34/// @ingroup DataTypes
35template <std::size_t ThePrecision, std::size_t TheScale>
37{
38 static constexpr auto ColumnType = SqlColumnTypeDefinitions::Decimal { .precision = ThePrecision, .scale = TheScale };
39
40 /// Number of total digits
41 static constexpr auto Precision = ThePrecision;
42
43 /// Number of digits after the decimal point
44 static constexpr auto Scale = TheScale;
45
46 static_assert(TheScale < SQL_MAX_NUMERIC_LEN);
47 static_assert(Scale <= Precision);
48
49 /// The value is stored as a string to avoid floating point precision issues.
50 SQL_NUMERIC_STRUCT sqlValue {};
51
52 constexpr SqlNumeric() noexcept = default;
53 constexpr SqlNumeric(SqlNumeric&&) noexcept = default;
54 constexpr SqlNumeric& operator=(SqlNumeric&&) noexcept = default;
55 constexpr SqlNumeric(SqlNumeric const&) noexcept = default;
56 constexpr SqlNumeric& operator=(SqlNumeric const&) noexcept = default;
57 constexpr ~SqlNumeric() noexcept = default;
58
59 /// Constructs a numeric from a floating point value.
60 constexpr SqlNumeric(std::floating_point auto value) noexcept
61 {
62 assign(value);
63 }
64
65 /// Constructs a numeric from a SQL_NUMERIC_STRUCT.
66 constexpr explicit SqlNumeric(SQL_NUMERIC_STRUCT const& value) noexcept:
67 sqlValue(value)
68 {
69 }
70
71 // For encoding/decoding purposes, we assume little-endian.
72 static_assert(std::endian::native == std::endian::little);
73
74 /// Assigns a value to the numeric.
75 LIGHTWEIGHT_FORCE_INLINE constexpr void assign(std::floating_point auto value) noexcept
76 {
77#if defined(LIGHTWEIGHT_INT128_T)
78 auto const num = static_cast<LIGHTWEIGHT_INT128_T>(value * std::pow(10, Scale));
79 *((LIGHTWEIGHT_INT128_T*) sqlValue.val) = num;
80#else
81 auto const num = static_cast<int64_t>(value * std::pow(10, Scale));
82 std::memset(sqlValue.val, 0, sizeof(sqlValue.val));
83 *((int64_t*) sqlValue.val) = num;
84#endif
85
86 sqlValue.sign = num >= 0; // 1 == positive, 0 == negative
87 sqlValue.precision = Precision;
88 sqlValue.scale = Scale;
89 }
90
91 /// Assigns a floating point value to the numeric.
92 LIGHTWEIGHT_FORCE_INLINE constexpr SqlNumeric& operator=(std::floating_point auto value) noexcept
93 {
94 assign(value);
95 return *this;
96 }
97
98 /// Converts the numeric to an unscaled integer value.
99 [[nodiscard]] constexpr LIGHTWEIGHT_FORCE_INLINE auto ToUnscaledValue() const noexcept
100 {
101 auto const sign = sqlValue.sign ? 1 : -1;
102#if defined(LIGHTWEIGHT_INT128_T)
103 return sign * *reinterpret_cast<LIGHTWEIGHT_INT128_T const*>(sqlValue.val);
104#else
105 return sign * *reinterpret_cast<int64_t const*>(sqlValue.val);
106#endif
107 }
108
109 /// Converts the numeric to a floating point value.
110 [[nodiscard]] constexpr LIGHTWEIGHT_FORCE_INLINE float ToFloat() const noexcept
111 {
112 return float(ToUnscaledValue()) / std::powf(10, Scale);
113 }
114
115 /// Converts the numeric to a double precision floating point value.
116 [[nodiscard]] constexpr LIGHTWEIGHT_FORCE_INLINE double ToDouble() const noexcept
117 {
118 return double(ToUnscaledValue()) / std::pow(10, Scale);
119 }
120
121 /// Converts the numeric to a long double precision floating point value.
122 [[nodiscard]] constexpr LIGHTWEIGHT_FORCE_INLINE long double ToLongDouble() const noexcept
123 {
124 return static_cast<long double>(ToUnscaledValue()) / std::pow(10, Scale);
125 }
126
127 /// Converts the numeric to a floating point value.
128 [[nodiscard]] constexpr LIGHTWEIGHT_FORCE_INLINE explicit operator float() const noexcept
129 {
130 return ToFloat();
131 }
132
133 /// Converts the numeric to a double precision floating point value.
134 [[nodiscard]] constexpr LIGHTWEIGHT_FORCE_INLINE explicit operator double() const noexcept
135 {
136 return ToDouble();
137 }
138
139 /// Converts the numeric to a long double precision floating point value.
140 [[nodiscard]] constexpr LIGHTWEIGHT_FORCE_INLINE explicit operator long double() const noexcept
141 {
142 return ToLongDouble();
143 }
144
145 /// Converts the numeric to a string.
146 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::string ToString() const
147 {
148 return std::format("{:.{}f}", ToFloat(), Scale);
149 }
150
151 [[nodiscard]] constexpr LIGHTWEIGHT_FORCE_INLINE std::weak_ordering operator<=>(
152 SqlNumeric const& other) const noexcept
153 {
154 return ToDouble() <=> other.ToDouble();
155 }
156
157 template <std::size_t OtherPrecision, std::size_t OtherScale>
158 [[nodiscard]] constexpr LIGHTWEIGHT_FORCE_INLINE bool operator==(
159 SqlNumeric<OtherPrecision, OtherScale> const& other) const noexcept
160 {
161 return ToFloat() == other.ToFloat();
162 }
163};
164
165// clang-format off
166template <std::size_t Precision, std::size_t Scale>
167struct SqlDataBinder<SqlNumeric<Precision, Scale>>
168{
169 using ValueType = SqlNumeric<Precision, Scale>;
170
171 static constexpr auto ColumnType = SqlColumnTypeDefinitions::Decimal { .precision = Precision, .scale = Scale };
172
173 static void RequireSuccess(SQLHSTMT stmt, SQLRETURN error, std::source_location sourceLocation = std::source_location::current())
174 {
175 if (SQL_SUCCEEDED(error))
176 return;
177
178 throw SqlException(SqlErrorInfo::fromStatementHandle(stmt), sourceLocation);
179 }
180
181 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN InputParameter(SQLHSTMT stmt, SQLUSMALLINT column, ValueType const& value, SqlDataBinderCallback& /*cb*/) noexcept
182 {
183 auto* mut = const_cast<ValueType*>(&value);
184 mut->sqlValue.precision = Precision;
185 mut->sqlValue.scale = Scale;
186 RequireSuccess(stmt, SQLBindParameter(stmt, column, SQL_PARAM_INPUT, SQL_C_NUMERIC, SQL_NUMERIC, Precision, Scale, (SQLPOINTER) &value.sqlValue, 0, nullptr));
187 return SQL_SUCCESS;
188 }
189
190 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN OutputColumn(SQLHSTMT stmt, SQLUSMALLINT column, ValueType* result, SQLLEN* indicator, SqlDataBinderCallback& /*unused*/) noexcept
191 {
192 SQLHDESC hDesc {};
193 RequireSuccess(stmt, SQLGetStmtAttr(stmt, SQL_ATTR_APP_ROW_DESC, (SQLPOINTER) &hDesc, 0, nullptr));
194 RequireSuccess(stmt, SQLSetDescField(hDesc, (SQLSMALLINT) column, SQL_DESC_PRECISION, (SQLPOINTER) Precision, 0));
195 RequireSuccess(stmt, SQLSetDescField(hDesc, (SQLSMALLINT) column, SQL_DESC_SCALE, (SQLPOINTER) Scale, 0));
196
197 return SQLBindCol(stmt, column, SQL_C_NUMERIC, &result->sqlValue, sizeof(ValueType), indicator);
198 }
199
200 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN GetColumn(SQLHSTMT stmt, SQLUSMALLINT column, ValueType* result, SQLLEN* indicator, SqlDataBinderCallback const& /*cb*/) noexcept
201 {
202 SQLHDESC hDesc {};
203 RequireSuccess(stmt, SQLGetStmtAttr(stmt, SQL_ATTR_APP_ROW_DESC, (SQLPOINTER) &hDesc, 0, nullptr));
204 RequireSuccess(stmt, SQLSetDescField(hDesc, (SQLSMALLINT) column, SQL_DESC_PRECISION, (SQLPOINTER) Precision, 0));
205 RequireSuccess(stmt, SQLSetDescField(hDesc, (SQLSMALLINT) column, SQL_DESC_SCALE, (SQLPOINTER) Scale, 0));
206
207 return SQLGetData(stmt, column, SQL_C_NUMERIC, &result->sqlValue, sizeof(ValueType), indicator);
208 }
209
210 static LIGHTWEIGHT_FORCE_INLINE std::string Inspect(ValueType const& value) noexcept
211 {
212 return value.ToString();
213 }
214};
215// clang-format off
216
217
218template <std::size_t Precision, std::size_t Scale>
219struct std::formatter<SqlNumeric<Precision, Scale>>: std::formatter<double>
220{
221 template <typename FormatContext>
222 auto format(SqlNumeric<Precision, Scale> const& value, FormatContext& ctx)
223 {
224 return formatter<double>::format(value.ToDouble(), ctx);
225 }
226};
static SqlErrorInfo fromStatementHandle(SQLHSTMT hStmt)
Constructs an ODBC error info object from the given ODBC statement handle.
Definition SqlError.hpp:41
constexpr LIGHTWEIGHT_FORCE_INLINE auto ToUnscaledValue() const noexcept
Converts the numeric to an unscaled integer value.
SQL_NUMERIC_STRUCT sqlValue
The value is stored as a string to avoid floating point precision issues.
constexpr LIGHTWEIGHT_FORCE_INLINE long double ToLongDouble() const noexcept
Converts the numeric to a long double precision floating point value.
static constexpr auto Precision
Number of total digits.
LIGHTWEIGHT_FORCE_INLINE constexpr SqlNumeric & operator=(std::floating_point auto value) noexcept
Assigns a floating point value to the numeric.
static constexpr auto Scale
Number of digits after the decimal point.
constexpr LIGHTWEIGHT_FORCE_INLINE float ToFloat() const noexcept
Converts the numeric to a floating point value.
LIGHTWEIGHT_FORCE_INLINE std::string ToString() const
Converts the numeric to a string.
constexpr SqlNumeric(SQL_NUMERIC_STRUCT const &value) noexcept
Constructs a numeric from a SQL_NUMERIC_STRUCT.
constexpr LIGHTWEIGHT_FORCE_INLINE double ToDouble() const noexcept
Converts the numeric to a double precision floating point value.
LIGHTWEIGHT_FORCE_INLINE constexpr void assign(std::floating_point auto value) noexcept
Assigns a value to the numeric.