5#include "../DataBinder/UnicodeConverter.hpp"
6#include "../SqlLogger.hpp"
8#include "MFCStringLike.hpp"
9#include "Primitives.hpp"
11#include "SqlDateTime.hpp"
12#include "SqlFixedString.hpp"
14#include "SqlNullValue.hpp"
17#include "StdString.hpp"
18#include "StdStringView.hpp"
27struct overloaded: Ts...
29 using Ts::operator()...;
33overloaded(Ts...) -> overloaded<Ts...>;
91 value(std::move(other))
96 template <std::
size_t N,
typename T =
char, SqlFixedStringMode Mode>
98 value { std::string_view { other.data(), other.size() } }
103 template <std::
size_t TextSize>
104 constexpr LIGHTWEIGHT_FORCE_INLINE
SqlVariant(
char const (&text)[TextSize]):
105 value { std::string_view { text, TextSize - 1 } }
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 } }
117 template <
typename T>
118 LIGHTWEIGHT_FORCE_INLINE
SqlVariant(std::optional<T>
const& other):
133 value = std::move(other);
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) }
145 template <detail::HasSqlViewHelper StringViewLike>
148 value = std::string_view(newValue->GetString(), newValue->GetLength());
153 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
bool IsNull() const noexcept
155 return std::holds_alternative<SqlNullType>(value);
159 template <
typename T>
160 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
bool Is() const noexcept
162 return std::holds_alternative<T>(value);
166 template <
typename T>
167 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
decltype(
auto)
Get() noexcept
169 if constexpr (IsSpecializationOf<std::optional, T>)
172 return T { std::nullopt };
174 return T { std::get<typename T::value_type>(value) };
177 return std::get<T>(value);
181 template <
typename T>
182 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE T
ValueOr(T&& defaultValue)
const noexcept
184 if constexpr (std::is_integral_v<T>)
185 return TryGetIntegral<T>().value_or(std::forward<T>(defaultValue));
188 return std::forward<T>(defaultValue);
190 return std::get<T>(value);
195 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<bool>
TryGetBool() const noexcept {
return TryGetIntegral<bool>(); }
197 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<int8_t>
TryGetInt8() const noexcept {
return TryGetIntegral<int8_t>(); }
199 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<short>
TryGetShort() const noexcept {
return TryGetIntegral<short>(); }
201 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<unsigned short>
TryGetUShort() const noexcept {
return TryGetIntegral<unsigned short>(); }
203 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<int>
TryGetInt() const noexcept {
return TryGetIntegral<int>(); }
205 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<unsigned int>
TryGetUInt() const noexcept {
return TryGetIntegral<unsigned int>(); }
207 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<long long>
TryGetLongLong() const noexcept {
return TryGetIntegral<long long>(); }
209 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<unsigned long long>
TryGetULongLong() const noexcept {
return TryGetIntegral<unsigned long long>(); }
214 template <
typename ResultType>
215 [[nodiscard]] std::optional<ResultType> TryGetIntegral() const noexcept
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(); }
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(); }
245 [[nodiscard]] std::optional<std::u16string_view> TryGetUtf16StringView() const noexcept
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(); }
260 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<SqlDate>
TryGetDate()
const
265 if (
auto const* date = std::get_if<SqlDate>(&value))
268 if (
auto const* datetime = std::get_if<SqlDateTime>(&value))
270 return SqlDate { std::chrono::year(datetime->sqlValue.year),
271 std::chrono::month(datetime->sqlValue.month),
272 std::chrono::day(datetime->sqlValue.day) };
275 throw std::bad_variant_access();
278 [[nodiscard]]
bool operator==(
SqlVariant const& other)
const noexcept
280 return ToString() == other.ToString();
283 [[nodiscard]]
bool operator!=(
SqlVariant const& other)
const noexcept
285 return !(*
this == other);
289 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<SqlTime>
TryGetTime()
const
294 if (
auto const* time = std::get_if<SqlTime>(&value))
297 if (
auto const* datetime = std::get_if<SqlDateTime>(&value))
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) };
305 throw std::bad_variant_access();
309 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<SqlDateTime>
TryGetDateTime()
const
314 if (
auto const* dateTime = std::get_if<SqlDateTime>(&value))
317 throw std::bad_variant_access();
321 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::optional<SqlGuid>
TryGetGuid()
const
326 if (
auto const* guid = std::get_if<SqlGuid>(&value))
329 throw std::bad_variant_access();
333 [[nodiscard]] LIGHTWEIGHT_API std::string
ToString()
const;
337using SqlVariantRow = std::vector<SqlVariant>;
340struct LIGHTWEIGHT_API std::formatter<SqlVariant>: formatter<string>
342 auto format(
SqlVariant const& value, format_context& ctx)
const -> format_context::iterator
344 return std::formatter<string>::format(value.ToString(), ctx);
349struct LIGHTWEIGHT_API SqlDataBinder<
SqlVariant>
351 static SQLRETURN InputParameter(SQLHSTMT stmt,
356 static SQLRETURN GetColumn(SQLHSTMT stmt,
362 static LIGHTWEIGHT_FORCE_INLINE std::string Inspect(
SqlVariant const& value)
noexcept
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.