Lightweight 0.1.0
Loading...
Searching...
No Matches
SqlVariant.hpp
1// SPDX-License-Identifier: Apache-2.0
2
3#pragma once
4
5#include "../DataBinder/UnicodeConverter.hpp"
6#include "../SqlLogger.hpp"
7#include "Core.hpp"
8#include "MFCStringLike.hpp"
9#include "Primitives.hpp"
10#include "SqlDate.hpp"
11#include "SqlDateTime.hpp"
12#include "SqlFixedString.hpp"
13#include "SqlGuid.hpp"
14#include "SqlNullValue.hpp"
15#include "SqlText.hpp"
16#include "SqlTime.hpp"
17#include "StdString.hpp"
18#include "StdStringView.hpp"
19
20#include <format>
21#include <print>
22#include <variant>
23
24namespace detail
25{
26template <class... Ts>
27struct overloaded: Ts... // NOLINT(readability-identifier-naming)
28{
29 using Ts::operator()...;
30};
31
32template <class... Ts>
33overloaded(Ts...) -> overloaded<Ts...>;
34
35} // namespace detail
36
37/// @brief Represents a value that can be any of the supported SQL data types.
38///
39/// Use this class with care. Always prefer native types when possible, in order to avoid any unnecessary overhead.
40///
41/// @ingroup DataTypes
43{
44 /// @brief The inner type of the variant.
45 ///
46 /// This type is a variant of all the supported SQL data types.
47 using InnerType = std::variant<SqlNullType,
48 SqlGuid,
49 bool,
50 int8_t,
51 short,
52 unsigned short,
53 int,
54 unsigned int,
55 long long,
56 unsigned long long,
57 float,
58 double,
59 std::string,
60 std::string_view,
61 std::u16string,
62 std::u16string_view,
63 SqlText,
64 SqlDate,
65 SqlTime,
67
68 InnerType value;
69
70 /// @brief Default construct a new SqlVariant.
71 SqlVariant() = default;
72 /// @brief Copy construct a new SqlVariant from another.
73 SqlVariant(SqlVariant const&) = default;
74 /// @brief Move construct a new SqlVariant from another.
75 SqlVariant(SqlVariant&&) noexcept = default;
76 /// @brief Copy assign a new SqlVariant from another.
77 SqlVariant& operator=(SqlVariant const&) = default;
78 /// @brief Move assign a new SqlVariant from another.
79 SqlVariant& operator=(SqlVariant&&) noexcept = default;
80 /// @brief Destructor for SqlVariant.
81 ~SqlVariant() = default;
82
83 /// @brief Copy constructor of a SqlVariant from one of the supported types.
84 LIGHTWEIGHT_FORCE_INLINE SqlVariant(InnerType const& other):
85 value(other)
86 {
87 }
88
89 /// @brief Move constructor of a SqlVariant from one of the supported types.
90 LIGHTWEIGHT_FORCE_INLINE SqlVariant(InnerType&& other) noexcept:
91 value(std::move(other))
92 {
93 }
94
95 /// @brief Construct a new SqlVariant from a SqlFixedString.
96 template <std::size_t N, typename T = char, SqlFixedStringMode Mode>
97 constexpr LIGHTWEIGHT_FORCE_INLINE SqlVariant(SqlFixedString<N, T, Mode> const& other):
98 value { std::string_view { other.data(), other.size() } }
99 {
100 }
101
102 /// @brief Copy constructor of a SqlVariant from a char array.
103 template <std::size_t TextSize>
104 constexpr LIGHTWEIGHT_FORCE_INLINE SqlVariant(char const (&text)[TextSize]):
105 value { std::string_view { text, TextSize - 1 } }
106 {
107 }
108
109 /// @brief Copy constructor of a SqlVariant from a char16_t array.
110 template <std::size_t TextSize>
111 constexpr LIGHTWEIGHT_FORCE_INLINE SqlVariant(char16_t const (&text)[TextSize]):
112 value { std::u16string_view { text, TextSize - 1 } }
113 {
114 }
115
116 /// @brief Copy constructor of a SqlVariant from an optional of one of the supported types.
117 template <typename T>
118 LIGHTWEIGHT_FORCE_INLINE SqlVariant(std::optional<T> const& other):
119 value { other ? InnerType { *other } : InnerType { SqlNullValue } }
120 {
121 }
122
123 /// @brief Assignment operator of a SqlVariant from one of the supported types.
124 LIGHTWEIGHT_FORCE_INLINE SqlVariant& operator=(InnerType const& other)
125 {
126 value = other;
127 return *this;
128 }
129
130 /// @brief Assignment operator of a SqlVariant from one of the supported types.
131 LIGHTWEIGHT_FORCE_INLINE SqlVariant& operator=(InnerType&& other) noexcept
132 {
133 value = std::move(other);
134 return *this;
135 }
136
137 /// @brief Construct from an string-like object that implements an SqlViewHelper<>.
138 template <detail::HasSqlViewHelper StringViewLike>
139 LIGHTWEIGHT_FORCE_INLINE explicit SqlVariant(StringViewLike const* newValue):
140 value { detail::SqlViewHelper<std::remove_cv_t<decltype(*newValue)>>::View(*newValue) }
141 {
142 }
143
144 /// @brief Assign from an string-like object that implements an SqlViewHelper<>.
145 template <detail::HasSqlViewHelper StringViewLike>
146 LIGHTWEIGHT_FORCE_INLINE SqlVariant& operator=(StringViewLike const* newValue) noexcept
147 {
148 value = std::string_view(newValue->GetString(), newValue->GetLength());
149 return *this;
150 }
151
152 /// @brief Check if the value is NULL.
153 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE bool IsNull() const noexcept
154 {
155 return std::holds_alternative<SqlNullType>(value);
156 }
157
158 /// @brief Check if the value is of the specified type.
159 template <typename T>
160 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE bool Is() const noexcept
161 {
162 return std::holds_alternative<T>(value);
163 }
164
165 /// @brief Retrieve the value as the specified type.
166 template <typename T>
167 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE decltype(auto) Get() noexcept
168 {
169 if constexpr (IsSpecializationOf<std::optional, T>)
170 {
171 if (IsNull())
172 return T { std::nullopt };
173 else
174 return T { std::get<typename T::value_type>(value) };
175 }
176 else
177 return std::get<T>(value);
178 }
179
180 /// @brief Retrieve the value as the specified type, or return the default value if the value is NULL.
181 template <typename T>
182 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE T ValueOr(T&& defaultValue) const noexcept
183 {
184 if constexpr (std::is_integral_v<T>)
185 return TryGetIntegral<T>().value_or(std::forward<T>(defaultValue));
186
187 if (IsNull())
188 return std::forward<T>(defaultValue);
189
190 return std::get<T>(value);
191 }
192
193 // clang-format off
194 /// @brief Retrieve the bool from the variant or std::nullopt
195 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<bool> TryGetBool() const noexcept { return TryGetIntegral<bool>(); }
196 /// @brief Retrieve the int8_t from the variant or std::nullopt
197 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<int8_t> TryGetInt8() const noexcept { return TryGetIntegral<int8_t>(); }
198 /// @brief Retrieve the unsigned short from the variant or std::nullopt
199 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<short> TryGetShort() const noexcept { return TryGetIntegral<short>(); }
200 /// @brief Retrieve the unsigned short from the variant or std::nullopt
201 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<unsigned short> TryGetUShort() const noexcept { return TryGetIntegral<unsigned short>(); }
202 /// @brief Retrieve the int from the variant or std::nullopt
203 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<int> TryGetInt() const noexcept { return TryGetIntegral<int>(); }
204 /// @brief Retrieve the unsigned int from the variant or std::nullopt
205 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<unsigned int> TryGetUInt() const noexcept { return TryGetIntegral<unsigned int>(); }
206 /// @brief Retrieve the long long from the variant or std::nullopt
207 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<long long> TryGetLongLong() const noexcept { return TryGetIntegral<long long>(); }
208 /// @brief Retrieve the unsigned long long from the variant or std::nullopt
209 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<unsigned long long> TryGetULongLong() const noexcept { return TryGetIntegral<unsigned long long>(); }
210 // clang-format on
211
212 private:
213 /// @brief template that is used to get integral types
214 template <typename ResultType>
215 [[nodiscard]] std::optional<ResultType> TryGetIntegral() const noexcept
216 {
217 if (IsNull())
218 return std::nullopt;
219
220 // clang-format off
221 return std::visit(detail::overloaded {
222 []<typename T>(T v) -> ResultType requires(std::is_integral_v<T>) { return static_cast<ResultType>(v); },
223 [](auto) -> ResultType { throw std::bad_variant_access(); }
224 }, value);
225 // clang-format on
226 }
227
228 public:
229 /// @brief function to get string_view from SqlVariant or std::nullopt
230 [[nodiscard]] std::optional<std::string_view> TryGetStringView() const noexcept
231 {
232 if (IsNull())
233 return std::nullopt;
234
235 // clang-format off
236 return std::visit(detail::overloaded {
237 [](std::string_view v) { return v; },
238 [](std::string const& v) { return std::string_view(v.data(), v.size()); },
239 [](SqlText const& v) { return std::string_view(v.value.data(), v.value.size()); },
240 [](auto const&) -> std::string_view { throw std::bad_variant_access(); }
241 }, value);
242 // clang-format on
243 }
244
245 [[nodiscard]] std::optional<std::u16string_view> TryGetUtf16StringView() const noexcept
246 {
247 if (IsNull())
248 return std::nullopt;
249
250 // clang-format off
251 return std::visit(detail::overloaded {
252 [](std::u16string_view v) { return v; },
253 [](std::u16string const& v) { return std::u16string_view(v.data(), v.size()); },
254 [](auto const&) -> std::u16string_view { throw std::bad_variant_access(); }
255 }, value);
256 // clang-format on
257 }
258
259 /// @brief function to get SqlDate from SqlVariant or std::nullopt
260 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<SqlDate> TryGetDate() const
261 {
262 if (IsNull())
263 return std::nullopt;
264
265 if (auto const* date = std::get_if<SqlDate>(&value))
266 return *date;
267
268 if (auto const* datetime = std::get_if<SqlDateTime>(&value))
269 {
270 return SqlDate { std::chrono::year(datetime->sqlValue.year),
271 std::chrono::month(datetime->sqlValue.month),
272 std::chrono::day(datetime->sqlValue.day) };
273 }
274
275 throw std::bad_variant_access();
276 }
277
278 [[nodiscard]] bool operator==(SqlVariant const& other) const noexcept
279 {
280 return ToString() == other.ToString();
281 }
282
283 [[nodiscard]] bool operator!=(SqlVariant const& other) const noexcept
284 {
285 return !(*this == other);
286 }
287
288 /// @brief function to get SqlTime from SqlVariant or std::nullopt
289 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<SqlTime> TryGetTime() const
290 {
291 if (IsNull())
292 return std::nullopt;
293
294 if (auto const* time = std::get_if<SqlTime>(&value))
295 return *time;
296
297 if (auto const* datetime = std::get_if<SqlDateTime>(&value))
298 {
299 return SqlTime { std::chrono::hours(datetime->sqlValue.hour),
300 std::chrono::minutes(datetime->sqlValue.minute),
301 std::chrono::seconds(datetime->sqlValue.second),
302 std::chrono::microseconds(datetime->sqlValue.fraction) };
303 }
304
305 throw std::bad_variant_access();
306 }
307
308 /// @brief function to get SqlDateTime from SqlVariant or std::nullopt
309 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<SqlDateTime> TryGetDateTime() const
310 {
311 if (IsNull())
312 return std::nullopt;
313
314 if (auto const* dateTime = std::get_if<SqlDateTime>(&value))
315 return *dateTime;
316
317 throw std::bad_variant_access();
318 }
319
320 /// @brief Retrieve the GUID from the variant or std::nullopt if the value is NULL.
321 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<SqlGuid> TryGetGuid() const
322 {
323 if (IsNull())
324 return std::nullopt;
325
326 if (auto const* guid = std::get_if<SqlGuid>(&value))
327 return *guid;
328
329 throw std::bad_variant_access();
330 }
331
332 /// @brief Create string representation of the variant. Can be used for debug purposes
333 [[nodiscard]] LIGHTWEIGHT_API std::string ToString() const;
334};
335
336/// @brief Represents a row of data from the database using SqlVariant as the column data type.
337using SqlVariantRow = std::vector<SqlVariant>;
338
339template <>
340struct LIGHTWEIGHT_API std::formatter<SqlVariant>: formatter<string>
341{
342 auto format(SqlVariant const& value, format_context& ctx) const -> format_context::iterator
343 {
344 return std::formatter<string>::format(value.ToString(), ctx);
345 }
346};
347
348template <>
349struct LIGHTWEIGHT_API SqlDataBinder<SqlVariant>
350{
351 static SQLRETURN InputParameter(SQLHSTMT stmt,
352 SQLUSMALLINT column,
353 SqlVariant const& variantValue,
354 SqlDataBinderCallback& cb) noexcept;
355
356 static SQLRETURN GetColumn(SQLHSTMT stmt,
357 SQLUSMALLINT column,
358 SqlVariant* result,
359 SQLLEN* indicator,
360 SqlDataBinderCallback const& cb) noexcept;
361
362 static LIGHTWEIGHT_FORCE_INLINE std::string Inspect(SqlVariant const& value) noexcept
363 {
364 return value.ToString();
365 }
366};
constexpr auto SqlNullValue
Represents a value that can be any of the supported SQL data types.
std::optional< std::string_view > TryGetStringView() const noexcept
function to get string_view from SqlVariant or std::nullopt
SqlVariant()=default
Default construct a new SqlVariant.
LIGHTWEIGHT_FORCE_INLINE SqlVariant(std::optional< T > const &other)
Copy constructor of a SqlVariant from an optional of one of the supported types.
LIGHTWEIGHT_FORCE_INLINE std::optional< unsigned short > TryGetUShort() const noexcept
Retrieve the unsigned short from the variant or std::nullopt.
SqlVariant(SqlVariant const &)=default
Copy construct a new SqlVariant from another.
LIGHTWEIGHT_FORCE_INLINE decltype(auto) Get() noexcept
Retrieve the value as the specified type.
LIGHTWEIGHT_FORCE_INLINE std::optional< unsigned long long > TryGetULongLong() const noexcept
Retrieve the unsigned long long from the variant or std::nullopt.
LIGHTWEIGHT_FORCE_INLINE SqlVariant & operator=(StringViewLike const *newValue) noexcept
Assign from an string-like object that implements an SqlViewHelper<>.
LIGHTWEIGHT_FORCE_INLINE bool Is() const noexcept
Check if the value is of the specified type.
LIGHTWEIGHT_FORCE_INLINE std::optional< SqlDate > TryGetDate() const
function to get SqlDate from SqlVariant or std::nullopt
LIGHTWEIGHT_FORCE_INLINE std::optional< int8_t > TryGetInt8() const noexcept
Retrieve the int8_t from the variant or std::nullopt.
LIGHTWEIGHT_FORCE_INLINE SqlVariant(InnerType &&other) noexcept
Move constructor of a SqlVariant from one of the supported types.
LIGHTWEIGHT_FORCE_INLINE T ValueOr(T &&defaultValue) const noexcept
Retrieve the value as the specified type, or return the default value if the value is NULL.
LIGHTWEIGHT_FORCE_INLINE SqlVariant & operator=(InnerType const &other)
Assignment operator of a SqlVariant from one of the supported types.
LIGHTWEIGHT_FORCE_INLINE SqlVariant(StringViewLike const *newValue)
Construct from an string-like object that implements an SqlViewHelper<>.
LIGHTWEIGHT_FORCE_INLINE std::optional< SqlTime > TryGetTime() const
function to get SqlTime from SqlVariant or std::nullopt
LIGHTWEIGHT_FORCE_INLINE bool IsNull() const noexcept
Check if the value is NULL.
LIGHTWEIGHT_FORCE_INLINE std::optional< SqlDateTime > TryGetDateTime() const
function to get SqlDateTime from SqlVariant or std::nullopt
std::variant< SqlNullType, SqlGuid, bool, int8_t, short, unsigned short, int, unsigned int, long long, unsigned long long, float, double, std::string, std::string_view, std::u16string, std::u16string_view, SqlText, SqlDate, SqlTime, SqlDateTime > InnerType
The inner type of the variant.
LIGHTWEIGHT_FORCE_INLINE SqlVariant & operator=(InnerType &&other) noexcept
Assignment operator of a SqlVariant from one of the supported types.
LIGHTWEIGHT_FORCE_INLINE std::optional< long long > TryGetLongLong() const noexcept
Retrieve the long long from the variant or std::nullopt.
constexpr LIGHTWEIGHT_FORCE_INLINE SqlVariant(char16_t const (&text)[TextSize])
Copy constructor of a SqlVariant from a char16_t array.
LIGHTWEIGHT_FORCE_INLINE std::optional< unsigned int > TryGetUInt() const noexcept
Retrieve the unsigned int from the variant or std::nullopt.
LIGHTWEIGHT_FORCE_INLINE std::optional< int > TryGetInt() const noexcept
Retrieve the int from the variant or std::nullopt.
constexpr LIGHTWEIGHT_FORCE_INLINE SqlVariant(char const (&text)[TextSize])
Copy constructor of a SqlVariant from a char array.
LIGHTWEIGHT_API std::string ToString() const
Create string representation of the variant. Can be used for debug purposes.
LIGHTWEIGHT_FORCE_INLINE std::optional< bool > TryGetBool() const noexcept
Retrieve the bool from the variant or std::nullopt.
constexpr LIGHTWEIGHT_FORCE_INLINE SqlVariant(SqlFixedString< N, T, Mode > const &other)
Construct a new SqlVariant from a SqlFixedString.
SqlVariant(SqlVariant &&) noexcept=default
Move construct a new SqlVariant from another.
LIGHTWEIGHT_FORCE_INLINE std::optional< short > TryGetShort() const noexcept
Retrieve the unsigned short from the variant or std::nullopt.
LIGHTWEIGHT_FORCE_INLINE std::optional< SqlGuid > TryGetGuid() const
Retrieve the GUID from the variant or std::nullopt if the value is NULL.