5#include "../SqlColumnTypeDefinitions.hpp"
7#include "UnicodeConverter.hpp"
13enum class SqlFixedStringMode : uint8_t
16 FIXED_SIZE_RIGHT_TRIMMED,
27template <std::
size_t N,
typename T =
char, SqlFixedStringMode Mode = SqlFixedStringMode::FIXED_SIZE>
32 std::size_t _size = 0;
37 using const_iterator = T
const*;
38 using pointer_type = T*;
39 using const_pointer_type = T
const*;
41 static constexpr std::size_t Capacity = N;
42 static constexpr SqlFixedStringMode PostRetrieveOperation = Mode;
45 template <std::
size_t SourceSize>
46 constexpr LIGHTWEIGHT_FORCE_INLINE
SqlFixedString(T
const (&text)[SourceSize]):
47 _size { SourceSize - 1 }
49 static_assert(SourceSize <= N + 1,
"RHS string size must not exceed target string's capacity.");
50 std::copy_n(text, SourceSize, _data);
53 LIGHTWEIGHT_FORCE_INLINE
constexpr SqlFixedString() noexcept = default;
58 LIGHTWEIGHT_FORCE_INLINE constexpr ~
SqlFixedString() noexcept = default;
61 LIGHTWEIGHT_FORCE_INLINE constexpr
SqlFixedString(std::basic_string_view<T> s) noexcept:
62 _size { (std::min)(N, s.size()) }
64 std::copy_n(s.data(), _size, _data);
68 LIGHTWEIGHT_FORCE_INLINE
constexpr SqlFixedString(std::basic_string<T>
const& s)
noexcept:
69 _size { (std::min)(N, s.size()) }
71 std::copy_n(s.data(), _size, _data);
75 LIGHTWEIGHT_FORCE_INLINE
constexpr SqlFixedString(T
const* s, std::size_t len)
noexcept:
76 _size { (std::min)(N, len) }
78 std::copy_n(s, _size, _data);
82 LIGHTWEIGHT_FORCE_INLINE
constexpr SqlFixedString(T
const* s, T
const* e)
noexcept:
83 _size { (std::min)(N,
static_cast<std::size_t
>(e - s)) }
85 std::copy(s, e, _data);
88 LIGHTWEIGHT_FORCE_INLINE
void reserve(std::size_t
capacity)
91 throw std::length_error(
92 std::format(
"SqlFixedString: capacity {} exceeds maximum capacity {}",
capacity, N));
96 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr bool empty() const noexcept
102 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr std::size_t
size() const noexcept
107 LIGHTWEIGHT_FORCE_INLINE
void setsize(std::size_t n)
noexcept
109 auto const newSize = (std::min)(n, N);
111 _data[newSize] =
'\0';
118 LIGHTWEIGHT_FORCE_INLINE
constexpr void resize(std::size_t n)
noexcept
120 auto const newSize = (std::min)(n, N);
122 _data[newSize] =
'\0';
126 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr std::size_t
capacity() const noexcept
132 LIGHTWEIGHT_FORCE_INLINE
constexpr void clear() noexcept
138 template <std::
size_t SourceSize>
139 LIGHTWEIGHT_FORCE_INLINE
constexpr void assign(T
const (&source)[SourceSize])
noexcept
141 static_assert(SourceSize <= N + 1,
"Source string must not overflow the target string's capacity.");
142 _size = SourceSize - 1;
143 std::copy_n(source, SourceSize, _data);
147 LIGHTWEIGHT_FORCE_INLINE
constexpr void assign(std::string_view s)
noexcept
149 _size = (std::min)(N, s.size());
150 std::copy_n(s.data(), _size, _data);
154 LIGHTWEIGHT_FORCE_INLINE
constexpr void push_back(T c)
noexcept
164 LIGHTWEIGHT_FORCE_INLINE
constexpr void pop_back() noexcept
171 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr std::basic_string_view<T>
substr(
172 std::size_t offset = 0, std::size_t count = (std::numeric_limits<std::size_t>::max)()) const noexcept
176 if (count == (std::numeric_limits<std::size_t>::max)())
177 return std::basic_string_view<T>(_data + offset, _size - offset);
178 if (offset + count > _size)
179 return std::basic_string_view<T>(_data + offset, _size - offset);
180 return std::basic_string_view<T>(_data + offset, count);
184 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr std::basic_string_view<T>
str() const noexcept
186 return std::basic_string_view<T> { _data, _size };
190 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr pointer_type c_str() noexcept { _data[_size] =
'\0';
return _data; }
191 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr pointer_type data() noexcept {
return _data; }
192 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr iterator begin() noexcept {
return _data; }
193 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr iterator end() noexcept {
return _data +
size(); }
194 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr T& at(std::size_t i)
noexcept {
return _data[i]; }
195 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr T& operator[](std::size_t i)
noexcept {
return _data[i]; }
197 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr const_pointer_type c_str() const noexcept {
return _data; }
198 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr const_pointer_type data() const noexcept {
return _data; }
199 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr const_iterator begin() const noexcept {
return _data; }
200 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr const_iterator end() const noexcept {
return _data +
size(); }
201 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr T
const& at(std::size_t i)
const noexcept {
return _data[i]; }
202 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr T
const& operator[](std::size_t i)
const noexcept {
return _data[i]; }
205 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr std::basic_string_view<T> ToStringView() const noexcept
207 return { _data, _size };
210 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
constexpr explicit operator std::basic_string_view<T>() const noexcept
212 return ToStringView();
215 template <std::
size_t OtherSize, SqlFixedStringMode OtherMode>
216 LIGHTWEIGHT_FORCE_INLINE std::weak_ordering operator<=>(
219 if ((
void*)
this == (
void*) &other) [[unlikely]]
220 return std::weak_ordering::equivalent;
222 for (std::size_t i = 0; i < (std::min)(
size(), other.size()); ++i)
223 if (
auto const cmp = _data[i] <=> other._data[i]; cmp != std::weak_ordering::equivalent) [[unlikely]]
225 return size() <=> other.size();
228 template <std::
size_t OtherSize, SqlFixedStringMode OtherMode>
229 LIGHTWEIGHT_FORCE_INLINE
constexpr bool operator==(
232 return (*this <=> other) == std::weak_ordering::equivalent;
235 template <std::
size_t OtherSize, SqlFixedStringMode OtherMode>
236 LIGHTWEIGHT_FORCE_INLINE
constexpr bool operator!=(
239 return !(*
this == other);
242 LIGHTWEIGHT_FORCE_INLINE
constexpr bool operator==(std::basic_string_view<T> other)
const noexcept
244 return (
substr() <=> other) == std::weak_ordering::equivalent;
247 LIGHTWEIGHT_FORCE_INLINE
constexpr bool operator!=(std::basic_string_view<T> other)
const noexcept
249 return !(*
this == other);
253template <std::
size_t N,
typename CharT, SqlFixedStringMode Mode>
256 static LIGHTWEIGHT_FORCE_INLINE std::basic_string_view<CharT> View(
259 return { str.data(), str.size() };
267struct IsSqlFixedStringTypeImpl: std::false_type
271template <std::
size_t N,
typename T, SqlFixedStringMode Mode>
272struct IsSqlFixedStringTypeImpl<
SqlFixedString<N, T, Mode>>: std::true_type
279constexpr bool IsSqlFixedString = detail::IsSqlFixedStringTypeImpl<T>::value;
284template <std::
size_t N>
290template <std::
size_t N>
296template <std::
size_t N>
302template <std::
size_t N>
308template <std::
size_t N,
typename T =
char>
311template <std::
size_t N,
typename T =
char>
314template <std::
size_t N,
typename T, SqlFixedStringMode Mode>
319 static constexpr auto ColumnType = []()
constexpr -> SqlColumnTypeDefinition {
320 if constexpr (std::same_as<CharType, char>)
322 if constexpr (Mode == SqlFixedStringMode::VARIABLE_SIZE)
323 return SqlColumnTypeDefinitions::Varchar { N };
325 return SqlColumnTypeDefinitions::Char { N };
329 if constexpr (Mode == SqlFixedStringMode::VARIABLE_SIZE)
330 return SqlColumnTypeDefinitions::NVarchar { N };
332 return SqlColumnTypeDefinitions::NChar { N };
336 static CharType
const* Data(ValueType
const* str)
noexcept
341 static CharType* Data(ValueType* str)
noexcept
346 static SQLULEN Size(ValueType
const* str)
noexcept
351 static void Clear(ValueType* str)
noexcept
356 static void Reserve(ValueType* str,
size_t capacity)
noexcept
358 str->reserve((std::min)(N, capacity));
359 str->resize((std::min)(N, capacity));
362 static void Resize(ValueType* str, SQLLEN indicator)
noexcept
364 str->resize(indicator);
367 static void PostProcessOutputColumn(ValueType* result, SQLLEN indicator)
378 auto const len = (std::min)(N,
static_cast<std::size_t
>(indicator) /
sizeof(CharType));
379 result->setsize(len);
381 if constexpr (Mode == SqlFixedStringMode::FIXED_SIZE_RIGHT_TRIMMED)
383 TrimRight(result, indicator);
390 LIGHTWEIGHT_FORCE_INLINE
static void TrimRight(ValueType* boundOutputString, SQLLEN indicator)
noexcept
392 size_t n = (std::min)((
size_t) indicator, N - 1);
393 while (n > 0 && std::isspace((*boundOutputString)[n - 1]))
395 boundOutputString->setsize(n);
399template <std::
size_t N,
typename T, SqlFixedStringMode P>
400struct std::formatter<
SqlFixedString<N, T, P>>: std::formatter<std::string>
403 auto format(value_type
const& text, format_context& ctx)
const -> format_context::iterator
405 if constexpr (std::same_as<T, wchar_t>)
406 return std::formatter<std::string>::format((
char const*)
ToUtf8(text.ToStringView()).c_str(), ctx);
408 return std::formatter<std::string>::format(text.c_str(), ctx);
LIGHTWEIGHT_FORCE_INLINE constexpr std::basic_string_view< T > substr(std::size_t offset=0, std::size_t count=(std::numeric_limits< std::size_t >::max)()) const noexcept
Returns a sub string of the string.
LIGHTWEIGHT_FORCE_INLINE constexpr void assign(T const (&source)[SourceSize]) noexcept
Assigns a string literal to the string.
LIGHTWEIGHT_FORCE_INLINE constexpr void resize(std::size_t n) noexcept
LIGHTWEIGHT_FORCE_INLINE constexpr bool empty() const noexcept
Tests if the string is empty.
LIGHTWEIGHT_FORCE_INLINE constexpr std::basic_string_view< T > str() const noexcept
Returns a string view of the string.
LIGHTWEIGHT_FORCE_INLINE constexpr void clear() noexcept
Clears the string.
LIGHTWEIGHT_FORCE_INLINE constexpr std::size_t size() const noexcept
Returns the size of the string.
LIGHTWEIGHT_FORCE_INLINE constexpr void assign(std::string_view s) noexcept
Assigns a string view to the string.
LIGHTWEIGHT_FORCE_INLINE constexpr void push_back(T c) noexcept
Appends a character to the string.
constexpr LIGHTWEIGHT_FORCE_INLINE SqlFixedString(T const (&text)[SourceSize])
Constructs a fixed-size string from a string literal.
LIGHTWEIGHT_FORCE_INLINE constexpr void pop_back() noexcept
Removes the last character from the string.
LIGHTWEIGHT_FORCE_INLINE constexpr SqlFixedString(T const *s, T const *e) noexcept
Constructs a fixed-size string from a string pointer and end pointer.
LIGHTWEIGHT_FORCE_INLINE constexpr std::size_t capacity() const noexcept
Returns the maximum capacity of the string.
LIGHTWEIGHT_FORCE_INLINE constexpr SqlFixedString(std::basic_string< T > const &s) noexcept
Constructs a fixed-size string from a string.
LIGHTWEIGHT_FORCE_INLINE constexpr SqlFixedString(T const *s, std::size_t len) noexcept
Constructs a fixed-size string from a string pointer and length.
LIGHTWEIGHT_API std::u8string ToUtf8(std::u32string_view u32InputString)