Lightweight 0.1.0
Loading...
Searching...
No Matches
SqlDynamicString.hpp
1// SPDX-License-Identifier: Apache-2.0
2
3#pragma once
4
5#include "../SqlColumnTypeDefinitions.hpp"
6#include "../Utils.hpp"
7#include "Core.hpp"
8#include "UnicodeConverter.hpp"
9
10#include <format>
11#include <limits>
12#include <string>
13
14constexpr size_t SqlMaxColumnSize = (std::numeric_limits<uint32_t>::max)();
15
16/// SQL dynamic-capacity string that mimmicks standard library string.
17///
18/// The underlying memory is allocated dynamically and the string can grow up to the maximum size of a SQL column.
19///
20/// @ingroup DataTypes
21template <std::size_t N, typename T = char>
23{
24 public:
25 static constexpr std::size_t DynamicCapacity = N;
26 using value_type = T;
27 using string_type = std::basic_string<T>;
28
29 /// Constructs a fixed-size string from a string literal.
30 template <std::size_t SourceSize>
31 constexpr LIGHTWEIGHT_FORCE_INLINE SqlDynamicString(T const (&text)[SourceSize]):
32 _value { text, SourceSize - 1 }
33 {
34 static_assert(SourceSize <= N + 1, "RHS string size must not exceed target string's capacity.");
35 }
36
37 /// Constructs a fixed-size string from a string pointer and end pointer.
38 LIGHTWEIGHT_FORCE_INLINE constexpr SqlDynamicString(T const* s, T const* e) noexcept:
39 _value { s, e }
40 {
41 }
42
43 LIGHTWEIGHT_FORCE_INLINE constexpr SqlDynamicString() noexcept = default;
44 LIGHTWEIGHT_FORCE_INLINE constexpr SqlDynamicString(SqlDynamicString const&) noexcept = default;
45 LIGHTWEIGHT_FORCE_INLINE constexpr SqlDynamicString& operator=(SqlDynamicString const&) noexcept = default;
46 LIGHTWEIGHT_FORCE_INLINE constexpr SqlDynamicString(SqlDynamicString&&) noexcept = default;
47 LIGHTWEIGHT_FORCE_INLINE constexpr SqlDynamicString& operator=(SqlDynamicString&&) noexcept = default;
48 LIGHTWEIGHT_FORCE_INLINE constexpr ~SqlDynamicString() noexcept = default;
49
50 /// Constructs a fixed-size string from a string view.
51 LIGHTWEIGHT_FORCE_INLINE constexpr SqlDynamicString(std::basic_string_view<T> s) noexcept:
52 _value { s }
53 {
54 }
55
56 /// Constructs a fixed-size string from a string.
57 LIGHTWEIGHT_FORCE_INLINE constexpr SqlDynamicString(std::basic_string<T> const& s) noexcept:
58 _value { s }
59 {
60 }
61
62 /// Retrieves the string's inner value (std::basic_string<T>).
63 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE string_type const& value() const noexcept
64 {
65 return _value;
66 }
67
68 /// Retrieves the string's inner value (std::basic_string<T>).
69 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE string_type& value() noexcept
70 {
71 return _value;
72 }
73
74 /// Retrieves the string's inner value (as T const*).
75 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE T const* data() const noexcept
76 {
77 return _value.data();
78 }
79
80 /// Retrieves the string's inner value (as T*).
81 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE T* data() noexcept
82 {
83 return _value.data();
84 }
85
86 /// Retrieves the string's capacity.
87 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::size_t capacity() const noexcept
88 {
89 return N;
90 }
91
92 /// Retrieves the string's size.
93 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE std::size_t size() const noexcept
94 {
95 return _value.size();
96 }
97
98 /// Tests if the string is empty.
99 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr bool empty() const noexcept
100 {
101 return _value.empty();
102 }
103
104 /// Clears the string.
105 LIGHTWEIGHT_FORCE_INLINE constexpr void clear() noexcept
106 {
107 _value.clear();
108 }
109
110 /// Resizes the string.
111 LIGHTWEIGHT_FORCE_INLINE constexpr void resize(std::size_t n) noexcept
112 {
113 _value.resize(n);
114 }
115
116 /// Retrieves a string view of the string.
117 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr std::basic_string_view<T> ToStringView() const noexcept
118 {
119 return { _value.data(), _value.size() };
120 }
121
122 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr T& operator[](std::size_t index) noexcept
123 {
124 return _value[index];
125 }
126
127 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr T const& operator[](std::size_t index) const noexcept
128 {
129 return _value[index];
130 }
131
132 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr explicit operator std::basic_string_view<T>() const noexcept
133 {
134 return ToStringView();
135 }
136
137 template <std::size_t OtherSize>
138 LIGHTWEIGHT_FORCE_INLINE std::weak_ordering operator<=>(SqlDynamicString<OtherSize, T> const& other) const noexcept
139 {
140 return _value <=> other._value;
141 }
142
143 template <std::size_t OtherSize>
144 LIGHTWEIGHT_FORCE_INLINE constexpr bool operator==(SqlDynamicString<OtherSize, T> const& other) const noexcept
145 {
146 return (*this <=> other) == std::weak_ordering::equivalent;
147 }
148
149 template <std::size_t OtherSize>
150 LIGHTWEIGHT_FORCE_INLINE constexpr bool operator!=(SqlDynamicString<OtherSize, T> const& other) const noexcept
151 {
152 return !(*this == other);
153 }
154
155 LIGHTWEIGHT_FORCE_INLINE constexpr bool operator==(std::basic_string_view<T> other) const noexcept
156 {
157 return (ToStringView() <=> other) == std::weak_ordering::equivalent;
158 }
159
160 LIGHTWEIGHT_FORCE_INLINE constexpr bool operator!=(std::basic_string_view<T> other) const noexcept
161 {
162 return !(*this == other);
163 }
164
165 private:
166 string_type _value;
167};
168
169template <std::size_t N, typename CharT>
170struct detail::SqlViewHelper<SqlDynamicString<N, CharT>>
171{
172 static LIGHTWEIGHT_FORCE_INLINE std::basic_string_view<CharT> View(SqlDynamicString<N, CharT> const& str) noexcept
173 {
174 return { str.data(), str.size() };
175 }
176};
177
178namespace detail
179{
180
181template <typename>
182struct IsSqlDynamicStringImpl: std::false_type
183{
184};
185
186template <std::size_t N, typename T>
187struct IsSqlDynamicStringImpl<SqlDynamicString<N, T>>: std::true_type
188{
189};
190
191} // namespace detail
192
193template <typename T>
194constexpr bool IsSqlDynamicString = detail::IsSqlDynamicStringImpl<T>::value;
195
196/// Fixed-size string of element type `char` with a capacity of `N` characters.
197///
198/// @ingroup DataTypes
199template <std::size_t N>
201
202/// Fixed-size string of element type `char16_t` with a capacity of `N` characters.
203///
204/// @ingroup DataTypes
205template <std::size_t N>
207
208/// Fixed-size string of element type `char32_t` with a capacity of `N` characters.
209///
210/// @ingroup DataTypes
211template <std::size_t N>
213
214/// Fixed-size string of element type `wchar_t` with a capacity of `N` characters.
215///
216/// @ingroup DataTypes
217template <std::size_t N>
219
220template <std::size_t N, typename T>
221struct std::formatter<SqlDynamicString<N, T>>: std::formatter<std::string>
222{
223 using value_type = SqlDynamicString<N, T>;
224
225 auto format(value_type const& text, format_context& ctx) const
226 {
227 if constexpr (detail::OneOf<T, wchar_t, char32_t, char16_t>)
228 return std::formatter<std::string>::format((char const*) ToUtf8(text.ToStringView()).c_str(), ctx);
229 else
230 return std::formatter<std::string>::format((char const*) text.data(), ctx);
231 }
232};
233
234template <std::size_t N, typename T>
235struct SqlBasicStringOperations<SqlDynamicString<N, T>>
236{
237 using CharType = T;
238 using ValueType = SqlDynamicString<N, CharType>;
239
240 static constexpr SqlColumnTypeDefinition ColumnType = []() constexpr {
241 if constexpr (std::same_as<CharType, char>)
242 return SqlColumnTypeDefinitions::Varchar { N };
243 else
244 return SqlColumnTypeDefinitions::NVarchar { N };
245 }();
246
247 static CharType const* Data(ValueType const* str) noexcept
248 {
249 return str->data();
250 }
251
252 static CharType* Data(ValueType* str) noexcept
253 {
254 return str->data();
255 }
256
257 static SQLULEN Size(ValueType const* str) noexcept
258 {
259 return str->size();
260 }
261
262 static void Clear(ValueType* str) noexcept
263 {
264 str->value().clear();
265 }
266
267 static void Reserve(ValueType* str, size_t capacity) noexcept
268 {
269 str->value().resize((std::min)(N, capacity));
270 }
271
272 static void Resize(ValueType* str, SQLLEN indicator) noexcept
273 {
274 str->value().resize(indicator);
275 }
276};
LIGHTWEIGHT_FORCE_INLINE std::size_t capacity() const noexcept
Retrieves the string's capacity.
LIGHTWEIGHT_FORCE_INLINE T * data() noexcept
Retrieves the string's inner value (as T*).
LIGHTWEIGHT_FORCE_INLINE string_type & value() noexcept
Retrieves the string's inner value (std::basic_string<T>).
LIGHTWEIGHT_FORCE_INLINE T const * data() const noexcept
Retrieves the string's inner value (as T const*).
LIGHTWEIGHT_FORCE_INLINE string_type const & value() const noexcept
Retrieves the string's inner value (std::basic_string<T>).
LIGHTWEIGHT_FORCE_INLINE constexpr void resize(std::size_t n) noexcept
Resizes the string.
LIGHTWEIGHT_FORCE_INLINE constexpr SqlDynamicString(T const *s, T const *e) noexcept
Constructs a fixed-size string from a string pointer and end pointer.
LIGHTWEIGHT_FORCE_INLINE constexpr void clear() noexcept
Clears the string.
LIGHTWEIGHT_FORCE_INLINE std::size_t size() const noexcept
Retrieves the string's size.
LIGHTWEIGHT_FORCE_INLINE constexpr std::basic_string_view< T > ToStringView() const noexcept
Retrieves a string view of the string.
constexpr LIGHTWEIGHT_FORCE_INLINE SqlDynamicString(T const (&text)[SourceSize])
Constructs a fixed-size string from a string literal.
LIGHTWEIGHT_FORCE_INLINE constexpr bool empty() const noexcept
Tests if the string is empty.
LIGHTWEIGHT_FORCE_INLINE constexpr SqlDynamicString(std::basic_string< T > const &s) noexcept
Constructs a fixed-size string from a string.
LIGHTWEIGHT_API std::u8string ToUtf8(std::u32string_view u32InputString)