rx: add StaticString utility

This commit is contained in:
DH 2025-10-04 14:25:55 +03:00
parent 825c92b135
commit 593297153a

View file

@ -0,0 +1,226 @@
#pragma once
#include "format-base.hpp"
#include <cassert>
#include <compare>
#include <cstddef>
#include <cstring>
#include <stdexcept>
#include <string_view>
#include <type_traits>
namespace rx {
template <std::size_t N> class StaticString {
private:
char m_data[N]{};
std::size_t m_size = 0;
public:
// Reserve space for null terminator
static constexpr std::size_t capacity = N - 1;
constexpr StaticString() noexcept = default;
constexpr StaticString(std::string_view sv) noexcept { assign(sv); }
template <std::size_t OtherN>
requires(OtherN <= N)
constexpr StaticString(const StaticString<OtherN> &other) noexcept
: m_size(other.m_size) {
std::memcpy(m_data, other.m_data, m_size + 1); // +1 for null terminator
}
template <std::size_t OtherN>
requires(OtherN <= N)
constexpr StaticString &
operator=(const StaticString<OtherN> &other) noexcept {
if (this != &other) {
m_size = other.m_size;
std::memcpy(m_data, other.m_data, m_size + 1);
}
return *this;
}
constexpr StaticString &operator=(std::string_view sv) noexcept {
assign(sv);
return *this;
}
constexpr void assign(std::string_view sv) noexcept {
if (sv.size() > capacity) {
// Truncate if too long
sv = sv.substr(0, capacity);
}
m_size = sv.size();
sv.copy(m_data, m_size);
m_data[m_size] = '\0';
}
constexpr void append(std::string_view sv) noexcept {
std::size_t available = capacity - m_size;
if (sv.size() > available) {
sv = sv.substr(0, available);
}
sv.copy(m_data + m_size, sv.size());
m_size += sv.size();
m_data[m_size] = '\0';
}
constexpr StaticString &operator+=(std::string_view sv) noexcept {
append(sv);
return *this;
}
constexpr void remove_suffix(std::size_t n) {
m_size -= n;
m_data[m_size] = 0;
}
constexpr void clear() noexcept {
m_size = 0;
m_data[0] = '\0';
}
[[nodiscard]] constexpr const char *data() const noexcept { return m_data; }
[[nodiscard]] constexpr char *data() noexcept { return m_data; }
[[nodiscard]] constexpr const char *c_str() const noexcept { return m_data; }
[[nodiscard]] constexpr std::size_t size() const noexcept { return m_size; }
[[nodiscard]] constexpr std::size_t length() const noexcept { return m_size; }
[[nodiscard]] constexpr bool empty() const noexcept { return m_size == 0; }
[[nodiscard]] constexpr std::size_t max_size() const noexcept {
return capacity;
}
constexpr operator std::string_view() const noexcept {
return std::string_view(m_data, m_size);
}
[[nodiscard]] constexpr const char *begin() const noexcept { return m_data; }
[[nodiscard]] constexpr const char *end() const noexcept {
return m_data + m_size;
}
[[nodiscard]] constexpr char *begin() noexcept { return m_data; }
[[nodiscard]] constexpr char *end() noexcept { return m_data + m_size; }
[[nodiscard]] constexpr const char *cbegin() const noexcept { return m_data; }
[[nodiscard]] constexpr const char *cend() const noexcept {
return m_data + m_size;
}
constexpr char &operator[](std::size_t pos) noexcept { return m_data[pos]; }
constexpr const char &operator[](std::size_t pos) const noexcept {
return m_data[pos];
}
[[nodiscard]] char &at(std::size_t pos) {
if (pos >= m_size) {
throw std::out_of_range("StaticString::at: index out of range");
}
return m_data[pos];
}
[[nodiscard]] constexpr char &at(std::size_t pos) const {
if (pos >= m_size) {
throw std::out_of_range("StaticString::at: index out of range");
}
return m_data[pos];
}
constexpr std::size_t find(char c, std::size_t pos = 0) const noexcept {
return std::string_view(*this).find(c, pos);
}
constexpr std::size_t find(std::string_view sv,
std::size_t pos = 0) const noexcept {
return std::string_view(*this).find(sv, pos);
}
[[nodiscard]] constexpr std::string_view
substr(std::size_t pos = 0, std::size_t len = std::string_view::npos) const {
return std::string_view(*this).substr(pos, len);
}
template <typename... Args>
[[nodiscard]] constexpr static StaticString format(format_string<Args...> fmt,
Args &&...args) {
StaticString result;
auto [ptr, size] = format_to_n(result.m_data, result.capacity, fmt,
std::forward<Args>(args)...);
result.m_data[size] = '\0';
return result;
}
void resize(std::size_t size, char c = ' ') {
assert(size <= capacity);
if (size < m_size) {
m_data[size] = 0;
m_size = size;
return;
}
for (std::size_t i = m_size; i < size; ++i) {
m_data[i] = c;
}
m_size = size;
}
[[nodiscard]] constexpr static StaticString vformat(std::string_view fmt,
format_args args) {
StaticString result;
auto [ptr, size] =
vformat_to(result.m_data, result.capacity, fmt, std::move(args));
result.m_data[size] = '\0';
return result;
}
template <typename... Args>
constexpr void assignFormat(format_string<Args...> fmt, Args &&...args) {
auto [ptr, size] =
format_to_n(m_data, capacity, fmt, std::forward<Args>(args)...);
m_data[size] = '\0';
}
constexpr void assignVFormat(std::string_view fmt, format_args args) {
auto [ptr, size] = vformat_to(m_data, capacity, fmt, std::move(args));
m_data[size] = '\0';
}
};
template <std::size_t LN, std::size_t RN>
constexpr bool operator==(const StaticString<LN> &lhs,
const StaticString<RN> &rhs) noexcept {
return std::string_view(lhs) == std::string_view(rhs);
}
template <std::size_t N>
constexpr bool operator==(const StaticString<N> &lhs,
std::string_view sv) noexcept {
return std::string_view(lhs) == sv;
}
template <std::size_t LN, std::size_t RN>
constexpr auto operator<=>(const StaticString<LN> &lhs,
const StaticString<RN> &rhs) noexcept {
return std::string_view(lhs) <=> std::string_view(rhs);
}
template <std::size_t N>
constexpr auto operator<=>(const StaticString<N> &lhs,
std::string_view sv) noexcept {
return std::string_view(lhs) <=> sv;
}
template <std::size_t N> StaticString(const char (&)[N]) -> StaticString<N>;
template <std::size_t N = 32, typename... Args>
auto formatStatic(format_string<Args...> fmt, Args &&...args) {
return StaticString<N>::format(fmt, std::forward<Args>(args)...);
}
} // namespace rx
template <std::size_t N>
struct rx::formatter<rx::StaticString<N>> : rx::formatter<std::string_view> {
auto format(const rx::StaticString<N> &str, rx::format_context &ctx) const {
return rx::formatter<std::string_view>::format(std::string_view(str), ctx);
}
};