Lightweight 0.1.0
Loading...
Searching...
No Matches
SqlGuid.hpp
1// SPDX-License-Identifier: Apache-2.0
2
3#pragma once
4
5#include "../SqlColumnTypeDefinitions.hpp"
6#include "Core.hpp"
7
8#include <format>
9#include <optional>
10#include <string>
11
12/// Represents a GUID (Globally Unique Identifier).
13///
14/// @ingroup DataTypes
15struct LIGHTWEIGHT_API SqlGuid
16{
17 uint8_t data[16] {};
18
19 /// Creates a new non-empty GUID.
20 static SqlGuid Create() noexcept;
21
22 /// Parses a GUID from a string.
23 static std::optional<SqlGuid> TryParse(std::string_view const& text) noexcept;
24
25 /// Parses a GUID from a string. Use with caution and always prefer TryParse at all cost.
26 static SqlGuid constexpr UnsafeParse(std::string_view const& text) noexcept;
27
28 constexpr std::weak_ordering operator<=>(SqlGuid const& other) const noexcept = default;
29
30 constexpr bool operator==(SqlGuid const& other) const noexcept
31 {
32 return (*this <=> other) == std::weak_ordering::equivalent;
33 }
34
35 constexpr bool operator!=(SqlGuid const& other) const noexcept
36 {
37 return !(*this == other);
38 }
39
40 // Test if the GUID is non-empty.
41 constexpr explicit operator bool() const noexcept
42 {
43 return *this != SqlGuid {};
44 }
45
46 // Test if the GUID is empty.
47 constexpr bool operator!() const noexcept
48 {
49 return !static_cast<bool>(*this);
50 }
51};
52
53constexpr SqlGuid SqlGuid::UnsafeParse(std::string_view const& text) noexcept
54{
55 SqlGuid guid {};
56
57 // UUID format: xxxxxxxx-xxxx-Mxxx-Nxxx-xxxxxxxxxxxx
58 // M is the version and N is the variant
59
60 // Check for length
61 if (text.size() != 36)
62 return { "\x01" };
63
64 // Check for dashes
65 if (text[8] != '-' || text[13] != '-' || text[18] != '-' || text[23] != '-')
66 return { "\x02" };
67
68 // Version must be 1, 2, 3, 4, or 5
69 auto const version = text[14];
70 if (!('1' <= version && version <= '5'))
71 return { "\x03" };
72
73 // Variant must be 8, 9, A, or B
74 auto const variant = text[21];
75 if (variant != '8' && variant != '9' && variant != 'A' && variant != 'B' && variant != 'a' && variant != 'b')
76 return { "\x04" };
77
78 // clang-format off
79 size_t i = 0;
80 for (auto const index: { 0, 2, 4, 6,
81 9, 11,
82 14, 16,
83 21, 19,
84 24, 26, 28, 30, 32, 34 })
85 {
86 if (std::from_chars(text.data() + index, text.data() + index + 2, guid.data[i], 16).ec != std::errc())
87 return { "\x05" };
88 i++;
89 }
90 // clang-format on
91
92 return guid;
93}
94
95template <>
96struct std::formatter<SqlGuid>: std::formatter<std::string>
97{
98 LIGHTWEIGHT_FORCE_INLINE auto format(SqlGuid const& guid, format_context& ctx) const -> format_context::iterator
99 {
100 // clang-format off
101 return formatter<std::string>::format(std::format(
102 "{:08X}-{:04X}-{:04X}-{:04X}-{:012X}",
103 (uint32_t) guid.data[3] | (uint32_t) guid.data[2] << 8 |
104 (uint32_t) guid.data[1] << 16 | (uint32_t) guid.data[0] << 24,
105 (uint16_t) guid.data[5] | (uint16_t) guid.data[4] << 8,
106 (uint16_t) guid.data[7] | (uint16_t) guid.data[6] << 8,
107 (uint16_t) guid.data[8] | (uint16_t) guid.data[9] << 8,
108 (uint64_t) guid.data[15] | (uint64_t) guid.data[14] << 8 |
109 (uint64_t) guid.data[13] << 16 | (uint64_t) guid.data[12] << 24 |
110 (uint64_t) guid.data[11] << 32 | (uint64_t) guid.data[10] << 40
111 ),
112 ctx
113 );
114 // clang-format on
115 }
116};
117
118inline LIGHTWEIGHT_FORCE_INLINE std::string to_string(SqlGuid const& guid)
119{
120 return std::format("{}", guid);
121}
122
123template <>
124struct LIGHTWEIGHT_API SqlDataBinder<SqlGuid>
125{
126 static constexpr auto ColumnType = SqlColumnTypeDefinitions::Guid {};
127
128 static SQLRETURN InputParameter(SQLHSTMT stmt,
129 SQLUSMALLINT column,
130 SqlGuid const& value,
131 SqlDataBinderCallback& cb) noexcept;
132
133 static SQLRETURN OutputColumn(
134 SQLHSTMT stmt, SQLUSMALLINT column, SqlGuid* result, SQLLEN* indicator, SqlDataBinderCallback& cb) noexcept;
135
136 static SQLRETURN GetColumn(SQLHSTMT stmt,
137 SQLUSMALLINT column,
138 SqlGuid* result,
139 SQLLEN* indicator,
140 SqlDataBinderCallback const& cb) noexcept;
141
142 static LIGHTWEIGHT_FORCE_INLINE std::string Inspect(SqlGuid const& value) noexcept
143 {
144 return std::format("{}", value);
145 }
146};
static SqlGuid constexpr UnsafeParse(std::string_view const &text) noexcept
Parses a GUID from a string. Use with caution and always prefer TryParse at all cost.
Definition SqlGuid.hpp:53
static SqlGuid Create() noexcept
Creates a new non-empty GUID.