14#include <fmt/format.h>
16#include <userver/compiler/demangle.hpp>
17#include <userver/utils/assert.hpp>
18#include <userver/utils/string_literal.hpp>
20USERVER_NAMESPACE_BEGIN
26constexpr bool HasUppercaseAscii(std::string_view value)
noexcept {
27 for (
auto c : value) {
28 if (
'A' <= c && c <=
'Z')
return true;
34constexpr bool ICaseEqualLowercase(std::string_view lowercase, std::string_view y)
noexcept {
35 const auto size = lowercase.size();
37 constexpr char kLowerToUpperMask =
static_cast<
char>(~
unsigned{32});
38 for (std::size_t i = 0; i < size; ++i) {
39 const auto lowercase_c = lowercase[i];
40 UASSERT(!(
'A' <= lowercase_c && lowercase_c <=
'Z'));
41 if (lowercase_c != y[i]) {
42 if (!(
'a' <= lowercase_c && lowercase_c <=
'z') || (lowercase_c & kLowerToUpperMask) != y[i]) {
52 constexpr explicit Found(std::size_t value)
noexcept {
UASSERT(value == 0); }
54 constexpr explicit operator std::size_t()
const noexcept {
return 0; }
57template <
typename Key,
typename Value,
typename Enabled =
void>
58class SearchState
final {
60 constexpr explicit SearchState(Key key)
noexcept : key_or_result_(std::in_place_index<0>, key) {}
62 constexpr bool IsFound()
const noexcept {
return key_or_result_.index() != 0; }
64 constexpr Key GetKey()
const noexcept {
66 return std::get<0>(key_or_result_);
69 constexpr void SetValue(Value value)
noexcept {
70 key_or_result_ = std::variant<Key, Value>(std::in_place_index<1>, value);
73 [[nodiscard]]
constexpr std::optional<Value> Extract()
noexcept {
74 if (key_or_result_.index() == 1) {
75 return std::get<1>(key_or_result_);
82 std::variant<Key, Value> key_or_result_;
85inline constexpr std::size_t kInvalidSize = std::numeric_limits<std::size_t>::max();
87template <
typename Payload>
88inline constexpr bool kFitsInStringOrPayload =
89 sizeof(Payload) <=
sizeof(
const char*) &&
90 (std::is_integral_v<Payload> || std::is_enum_v<Payload> || std::is_same_v<Payload, Found>);
93template <
typename Payload>
94class StringOrPayload
final {
96 constexpr explicit StringOrPayload(std::string_view string)
noexcept
97 : data_or_payload_(string.data()), size_(string.size()) {
99 __builtin_assume(size_ != kInvalidSize);
100#elif defined(__GNUC__)
101 if (size_ == kInvalidSize) __builtin_unreachable();
105 constexpr explicit StringOrPayload(Payload payload)
noexcept : data_or_payload_(payload), size_(kInvalidSize) {}
107 constexpr bool HasPayload()
const noexcept {
return size_ == kInvalidSize; }
109 constexpr const char* GetStringPointer()
const noexcept {
111 UASSERT(data_or_payload_.data);
112 return data_or_payload_.data;
115 constexpr std::size_t GetStringSize()
const noexcept {
117 UASSERT(data_or_payload_.data);
121 constexpr Payload GetPayload()
const noexcept {
123 return data_or_payload_.payload;
127 static_assert(kFitsInStringOrPayload<Payload>);
129 union DataOrPayload {
130 constexpr explicit DataOrPayload(
const char* data)
noexcept : data(data) {}
132 constexpr explicit DataOrPayload(Payload payload)
noexcept : payload(payload) {}
138 DataOrPayload data_or_payload_;
142template <
typename Value>
143class SearchState<std::string_view, Value, std::enable_if_t<kFitsInStringOrPayload<Value>>> {
145 constexpr explicit SearchState(std::string_view key)
noexcept : state_(key) {}
147 constexpr bool IsFound()
const noexcept {
return state_.HasPayload(); }
149 constexpr std::string_view GetKey()
const noexcept {
150 return std::string_view{state_.GetStringPointer(), state_.GetStringSize()};
153 constexpr void SetValue(Value value)
noexcept { state_ = StringOrPayload<Value>{value}; }
155 [[nodiscard]]
constexpr std::optional<Value> Extract()
noexcept {
156 return IsFound() ? std::optional{state_.GetPayload()} : std::nullopt;
160 StringOrPayload<Value> state_;
163template <
typename Value>
164class SearchState<
NullTerminatedView, Value, std::enable_if_t<kFitsInStringOrPayload<Value>>>
final
165 :
public SearchState<std::string_view, Value> {};
167template <
typename Value>
168class SearchState<
StringLiteral, Value, std::enable_if_t<kFitsInStringOrPayload<Value>>>
final
169 :
public SearchState<std::string_view, Value> {};
171template <
typename Key>
172class SearchState<Key, std::string_view, std::enable_if_t<kFitsInStringOrPayload<Key>>>
final {
174 constexpr explicit SearchState(Key key)
noexcept : state_(key) {}
176 constexpr bool IsFound()
const noexcept {
return !state_.HasPayload(); }
178 constexpr Key GetKey()
const noexcept {
return state_.GetPayload(); }
180 constexpr void SetValue(std::string_view value)
noexcept { state_ = StringOrPayload<Key>{value}; }
182 [[nodiscard]]
constexpr std::optional<std::string_view> Extract()
noexcept {
183 return IsFound() ? std::optional{std::string_view{state_.GetStringPointer(), state_.GetStringSize()}}
188 StringOrPayload<Key> state_;
191template <
typename Key>
192class SearchState<Key,
NullTerminatedView, std::enable_if_t<kFitsInStringOrPayload<Key>>>
final {
194 constexpr explicit SearchState(Key key)
noexcept : state_(key) {}
196 constexpr bool IsFound()
const noexcept {
return !state_.HasPayload(); }
198 constexpr Key GetKey()
const noexcept {
return state_.GetPayload(); }
200 constexpr void SetValue(
NullTerminatedView value)
noexcept { state_ = StringOrPayload<Key>{value}; }
209 StringOrPayload<Key> state_;
212template <
typename Key>
213class SearchState<Key,
StringLiteral, std::enable_if_t<kFitsInStringOrPayload<Key>>>
final {
215 constexpr explicit SearchState(Key key)
noexcept : state_(key) {}
217 constexpr bool IsFound()
const noexcept {
return !state_.HasPayload(); }
219 constexpr Key GetKey()
const noexcept {
return state_.GetPayload(); }
221 constexpr void SetValue(
StringLiteral value)
noexcept { state_ = StringOrPayload<Key>{value}; }
223 [[nodiscard]]
constexpr std::optional<
StringLiteral> Extract()
noexcept {
229 StringOrPayload<Key> state_;
232template <
typename First,
typename Second>
233class SwitchByFirst
final {
235 constexpr explicit SwitchByFirst(First search)
noexcept : state_(search) {}
237 constexpr SwitchByFirst& Case(First first, Second second)
noexcept {
238 if (!state_.IsFound() && state_.GetKey() == first) {
239 state_.SetValue(second);
244 template <
typename T,
typename U =
void>
245 constexpr SwitchByFirst& Type() {
249 [[nodiscard]]
constexpr std::optional<Second> Extract()
noexcept {
return state_.Extract(); }
252 SearchState<First, Second> state_;
255template <
typename First>
256class SwitchByFirst<First,
void>
final {
258 constexpr explicit SwitchByFirst(First search)
noexcept : state_(search) {}
260 constexpr SwitchByFirst& Case(First first)
noexcept {
261 if (!state_.IsFound() && state_.GetKey() == first) {
262 state_.SetValue(Found{0});
267 template <
typename T,
typename U =
void>
268 constexpr SwitchByFirst& Type() {
272 [[nodiscard]]
constexpr bool Extract()
noexcept {
return state_.IsFound(); }
275 SearchState<First, Found> state_;
278template <
typename Second>
279class SwitchByFirstICase
final {
281 constexpr explicit SwitchByFirstICase(std::string_view search)
noexcept : state_(search) {}
283 constexpr SwitchByFirstICase& Case(std::string_view first, Second second)
noexcept {
285 !impl::HasUppercaseAscii(first),
286 fmt::format(
"String literal '{}' in utils::Switch*::Case() should be in lower case", first)
288 if (!state_.IsFound() && state_.GetKey().size() == first.size() &&
289 impl::ICaseEqualLowercase(first, state_.GetKey())) {
290 state_.SetValue(second);
295 template <
typename T,
typename U>
296 constexpr SwitchByFirstICase& Type() {
300 [[nodiscard]]
constexpr std::optional<Second> Extract()
noexcept {
return state_.Extract(); }
303 SearchState<std::string_view, Second> state_;
307class SwitchByFirstICase<
void>
final {
309 constexpr explicit SwitchByFirstICase(std::string_view search)
noexcept : state_(search) {}
311 constexpr SwitchByFirstICase& Case(std::string_view first)
noexcept {
313 !impl::HasUppercaseAscii(first),
314 fmt::format(
"String literal '{}' in utils::Switch*::Case() should be in lower case", first)
316 if (!state_.IsFound() && state_.GetKey().size() == first.size() &&
317 impl::ICaseEqualLowercase(first, state_.GetKey())) {
318 state_.SetValue(Found{0});
323 template <
typename T,
typename U>
324 constexpr SwitchByFirstICase& Type() {
328 [[nodiscard]]
constexpr bool Extract()
const noexcept {
return state_.IsFound(); }
331 SearchState<std::string_view, Found> state_;
334template <
typename First>
335class SwitchBySecondICase
final {
337 constexpr explicit SwitchBySecondICase(std::string_view search)
noexcept : state_(search) {}
339 constexpr SwitchBySecondICase& Case(First first, std::string_view second)
noexcept {
341 !impl::HasUppercaseAscii(second),
342 fmt::format(
"String literal '{}' in utils::Switch*::Case() should be in lower case", second)
344 if (!state_.IsFound() && state_.GetKey().size() == second.size() &&
345 impl::ICaseEqualLowercase(second, state_.GetKey())) {
346 state_.SetValue(first);
351 template <
typename T,
typename U>
352 constexpr SwitchBySecondICase& Type() {
356 [[nodiscard]]
constexpr std::optional<First> Extract()
noexcept {
return state_.Extract(); }
359 SearchState<std::string_view, First> state_;
362template <
typename First,
typename Second>
363class SwitchBySecond
final {
365 constexpr explicit SwitchBySecond(Second search)
noexcept : state_(search) {}
367 constexpr SwitchBySecond& Case(First first, Second second)
noexcept {
368 if (!state_.IsFound() && state_.GetKey() == second) {
369 state_.SetValue(first);
374 template <
typename T,
typename U>
375 constexpr SwitchBySecond& Type() {
379 [[nodiscard]]
constexpr std::optional<First> Extract()
noexcept {
return state_.Extract(); }
382 SearchState<Second, First> state_;
385template <
typename First,
typename Second>
386class SwitchTypesDetected
final {
388 using first_type = First;
389 using second_type = Second;
391 constexpr SwitchTypesDetected& Case(First, Second)
noexcept {
return *
this; }
394template <
typename First>
395class SwitchTypesDetected<First,
void>
final {
397 using first_type = First;
398 using second_type =
void;
400 constexpr SwitchTypesDetected& Case(First)
noexcept {
return *
this; }
403class SwitchTypesDetector
final {
406 using DetectType = std::conditional_t<
412 std::conditional_t<std::is_convertible_v<T, std::string_view>, std::string_view, T>>>;
414 constexpr SwitchTypesDetector& operator()()
noexcept {
return *
this; }
416 template <
typename First,
typename Second>
417 constexpr auto Case(First, Second)
noexcept {
418 return Type<First, Second>();
421 template <
typename First>
422 constexpr auto Case(First)
noexcept {
423 return Type<First,
void>();
426 template <
typename First,
typename Second =
void>
427 constexpr auto Type()
noexcept {
428 return SwitchTypesDetected<DetectType<First>, DetectType<Second>>{};
432class CaseCounter
final {
434 template <
typename First,
typename Second>
435 constexpr CaseCounter& Case(First, Second)
noexcept {
440 template <
typename First>
441 constexpr CaseCounter& Case(First)
noexcept {
446 template <
typename T,
typename U>
447 constexpr CaseCounter& Type() {
451 [[nodiscard]]
constexpr std::size_t Extract()
const noexcept {
return count_; }
454 std::size_t count_{0};
457class CaseDescriber
final {
459 template <
typename First,
typename Second>
460 CaseDescriber& Case(First first, Second second)
noexcept {
461 if (!description_.empty()) {
462 description_ +=
", ";
465 description_ += fmt::format(
"('{}', '{}')", first, second);
470 template <
typename T,
typename U>
471 constexpr CaseDescriber& Type() {
475 [[nodiscard]] std::string Extract() &&
noexcept {
return std::move(description_); }
478 std::string description_{};
481class CaseFirstDescriber
final {
483 template <
typename First>
484 CaseFirstDescriber& Case(First first)
noexcept {
485 if (!description_.empty()) {
486 description_ +=
", ";
489 description_ += fmt::format(
"'{}'", first);
494 template <
typename First,
typename Second>
495 CaseFirstDescriber& Case(First first, Second )
noexcept {
499 template <
typename T,
typename U =
void>
500 constexpr CaseFirstDescriber& Type() {
504 [[nodiscard]] std::string Extract() &&
noexcept {
return std::move(description_); }
507 std::string description_{};
510class CaseSecondDescriber
final {
512 template <
typename First,
typename Second>
513 CaseSecondDescriber& Case(First , Second second)
noexcept {
514 if (!description_.empty()) {
515 description_ +=
", ";
518 description_ += fmt::format(
"'{}'", second);
523 template <
typename T,
typename U>
524 constexpr CaseSecondDescriber& Type() {
528 [[nodiscard]] std::string Extract() &&
noexcept {
return std::move(description_); }
531 std::string description_{};
534template <
typename First,
typename Second>
535class CaseGetValuesByIndex
final {
537 explicit constexpr CaseGetValuesByIndex(std::size_t search_index) : index_(search_index + 1) {}
539 constexpr CaseGetValuesByIndex& Case(First first, Second second)
noexcept {
544 lazy_ = Lazy{Storage{first, second}};
551 template <
typename T,
typename U>
552 constexpr CaseGetValuesByIndex& Type() {
556 [[nodiscard]]
constexpr First GetFirst()
noexcept {
return std::move(lazy_.storage.first); }
558 [[nodiscard]]
constexpr Second GetSecond()
noexcept {
return std::move(lazy_.storage.second); }
570 constexpr Lazy()
noexcept : empty{} {}
571 constexpr Lazy(Storage s)
noexcept : storage{s} {}
579template <
typename First>
580class CaseFirstIndexer
final {
582 constexpr explicit CaseFirstIndexer(First search_value)
noexcept : state_(search_value) {}
584 constexpr CaseFirstIndexer& Case(First first)
noexcept {
585 if (!state_.IsFound() && state_.GetKey() == first) {
586 state_.SetValue(index_);
592 template <
typename T,
typename U =
void>
593 constexpr CaseFirstIndexer& Type() {
597 [[nodiscard]]
constexpr std::optional<std::size_t> Extract() &&
noexcept {
return state_.Extract(); }
600 SearchState<First, std::size_t> state_;
601 std::size_t index_ = 0;
604template <
typename First>
605class CaseFirstIndexerICase
final {
607 constexpr explicit CaseFirstIndexerICase(First search_value)
noexcept : state_(search_value) {}
609 constexpr CaseFirstIndexerICase& Case(First first)
noexcept {
610 if (!state_.IsFound() && state_.GetKey().size() == first.size() &&
611 impl::ICaseEqualLowercase(first, state_.GetKey())) {
612 state_.SetValue(index_);
618 template <
typename T,
typename U =
void>
619 constexpr CaseFirstIndexerICase& Type() {
623 [[nodiscard]]
constexpr std::optional<std::size_t> Extract() &&
noexcept {
return state_.Extract(); }
626 SearchState<First, std::size_t> state_;
627 std::size_t index_ = 0;
634using DecayToStringView =
664template <
typename BuilderFunc>
665class TrivialBiMap
final {
666 using TypesPair = std::invoke_result_t<
const BuilderFunc&, impl::SwitchTypesDetector>;
669 using First =
typename TypesPair::first_type;
670 using Second =
typename TypesPair::second_type;
680 using MappedTypeFor = std::conditional_t<std::is_convertible_v<T, DecayToStringView<First>>, Second, First>;
682 constexpr TrivialBiMap(BuilderFunc&& func)
noexcept : func_(std::move(func)) {
683 static_assert(std::is_empty_v<BuilderFunc>,
"Mapping function should not capture variables");
684 static_assert(std::is_trivially_copyable_v<First>,
"First type in Case must be trivially copyable");
686 !std::is_void_v<Second>,
687 "If second type in Case is missing, use utils::TrivialSet instead of utils::TrivialBiMap"
689 static_assert(std::is_trivially_copyable_v<Second>,
"Second type in Case must be trivially copyable");
692 constexpr std::optional<Second> TryFindByFirst(DecayToStringView<First> value)
const noexcept {
693 return func_([value]() {
return impl::SwitchByFirst<DecayToStringView<First>, Second>{value}; }).Extract();
696 constexpr std::optional<First> TryFindBySecond(DecayToStringView<Second> value)
const noexcept {
697 return func_([value]() {
return impl::SwitchBySecond<First, DecayToStringView<Second>>{value}; }).Extract();
701 constexpr std::optional<MappedTypeFor<T>> TryFind(T value)
const noexcept {
702 if constexpr (std::is_convertible_v<T, DecayToStringView<First>>) {
704 !std::is_convertible_v<T, DecayToStringView<Second>>,
705 "Ambiguous conversion, use TryFindByFirst/TryFindBySecond instead"
707 return TryFindByFirst(value);
709 return TryFindBySecond(value);
718 return func_([value]() {
return impl::SwitchByFirstICase<Second>{value}; }).Extract();
726 return func_([value]() {
return impl::SwitchBySecondICase<First>{value}; }).Extract();
731 constexpr std::optional<MappedTypeFor<std::string_view>>
TryFindICase(std::string_view value)
const noexcept {
732 if constexpr (std::is_convertible_v<std::string_view, DecayToStringView<First>>) {
734 !std::is_convertible_v<std::string_view, DecayToStringView<Second>>,
735 "Ambiguous conversion, use TryFindICaseByFirst/TryFindICaseBySecond"
731 constexpr std::optional<MappedTypeFor<std::string_view>>
TryFindICase(std::string_view value)
const noexcept {
…}
744 constexpr std::size_t
size()
const noexcept {
745 return func_([]() {
return impl::CaseCounter{}; }).Extract();
744 constexpr std::size_t
size()
const noexcept {
…}
754 return func_([]() {
return impl::CaseDescriber{}; }).Extract();
764 return func_([]() {
return impl::CaseFirstDescriber{}; }).Extract();
774 return func_([]() {
return impl::CaseSecondDescriber{}; }).Extract();
783 template <
typename T>
785 if constexpr (std::is_convertible_v<T, DecayToStringView<First>>) {
792 constexpr value_type GetValuesByIndex(std::size_t index)
const {
794 auto result = func_([index]() {
return impl::CaseGetValuesByIndex<First, Second>{index}; });
795 return value_type{result.GetFirst(), result.GetSecond()};
800 using iterator_category = std::input_iterator_tag;
801 using difference_type = std::ptrdiff_t;
803 explicit constexpr iterator(
const TrivialBiMap& map, std::size_t position) : map_{map}, position_{position} {}
805 constexpr bool operator==(
iterator other)
const {
return position_ == other.position_; }
807 constexpr bool operator!=(
iterator other)
const {
return position_ != other.position_; }
814 constexpr iterator operator++(
int) {
820 constexpr value_type operator*()
const {
return map_.GetValuesByIndex(position_); }
823 const TrivialBiMap& map_;
824 std::size_t position_;
829 constexpr iterator cbegin()
const {
return begin(); }
830 constexpr iterator cend()
const {
return end(); }
833 const BuilderFunc func_;
665class TrivialBiMap
final {
…};
836template <
typename BuilderFunc>
837TrivialBiMap(BuilderFunc) -> TrivialBiMap<BuilderFunc>;
845template <
typename BuilderFunc>
846class TrivialSet
final {
847 using TypesPair = std::invoke_result_t<
const BuilderFunc&, impl::SwitchTypesDetector>;
850 using First =
typename TypesPair::first_type;
851 using Second =
typename TypesPair::second_type;
853 constexpr TrivialSet(BuilderFunc&& func)
noexcept : func_(std::move(func)) {
854 static_assert(std::is_empty_v<BuilderFunc>,
"Mapping function should not capture variables");
855 static_assert(std::is_trivially_copyable_v<First>,
"First type in Case must be trivially copyable");
856 static_assert(std::is_void_v<Second>,
"Second type in Case should be skipped in utils::TrivialSet");
859 constexpr bool Contains(DecayToStringView<First> value)
const noexcept {
860 return func_([value]() {
return impl::SwitchByFirst<DecayToStringView<First>, Second>{value}; }).Extract();
863 constexpr bool ContainsICase(std::string_view value)
const noexcept {
865 std::is_convertible_v<DecayToStringView<First>, std::string_view>,
866 "ContainsICase works only with std::string_view"
869 return func_([value]() {
return impl::SwitchByFirstICase<
void>{value}; }).Extract();
872 constexpr std::size_t size()
const noexcept {
873 return func_([]() {
return impl::CaseCounter{}; }).Extract();
882 return func_([]() {
return impl::CaseFirstDescriber{}; }).Extract();
887 constexpr std::optional<std::size_t>
GetIndex(DecayToStringView<First> value)
const {
888 return func_([value]() {
return impl::CaseFirstIndexer{value}; }).Extract();
887 constexpr std::optional<std::size_t>
GetIndex(DecayToStringView<First> value)
const {
…}
893 constexpr std::optional<std::size_t>
GetIndexICase(std::string_view value)
const {
894 return func_([value]() {
return impl::CaseFirstIndexerICase{value}; }).Extract();
893 constexpr std::optional<std::size_t>
GetIndexICase(std::string_view value)
const {
…}
898 const BuilderFunc func_;
846class TrivialSet
final {
…};
901template <
typename BuilderFunc>
902TrivialSet(BuilderFunc) -> TrivialSet<BuilderFunc>;
909template <
typename ExceptionType =
void,
typename Value,
typename BuilderFunc>
911 if constexpr (!std::is_void_v<ExceptionType>) {
912 if (!value.IsString()) {
913 throw ExceptionType(fmt::format(
"Invalid value at '{}': expected a string", value.GetPath()));
917 const auto string = value.
template As<std::string>();
918 const auto parsed = map.TryFind(string);
919 if (parsed)
return *parsed;
921 using Exception = std::conditional_t<std::is_void_v<ExceptionType>,
typename Value::Exception, ExceptionType>;
923 throw Exception(fmt::format(
924 "Invalid value of {} at '{}': '{}' is not one of {}",
925 compiler::GetTypeName<std::decay_t<
decltype(*parsed)>>(),
928 map.
template DescribeByType<std::string_view>()
937template <
typename Enum,
typename BuilderFunc>
938std::string_view EnumToStringView(Enum value, TrivialBiMap<BuilderFunc> map) {
939 static_assert(std::is_enum_v<Enum>);
940 if (
const auto string = map.TryFind(value))
return *string;
945 "Invalid value of enum {}: {}",
947 static_cast<std::underlying_type_t<Enum>>(value)
952template <
typename Selector,
class Keys,
typename Values, std::size_t... Indices>
954TrivialBiMapMultiCase(Selector selector,
const Keys& keys,
const Values& values, std::index_sequence<0, Indices...>) {
955 auto selector2 = selector.Case(std::data(keys)[0], std::data(values)[0]);
956 ((selector2 = selector2.Case(std::data(keys)[Indices], std::data(values)[Indices])), ...);
960template <
const auto& Keys,
const auto& Values>
961struct TrivialBiMapMultiCaseDispatch {
962 template <
class Selector>
963 constexpr auto operator()(Selector selector)
const {
964 constexpr auto kKeysSize = std::size(Keys);
965 return impl::TrivialBiMapMultiCase(selector(), Keys, Values, std::make_index_sequence<kKeysSize>{});
969template <
typename Selector,
class Values, std::size_t... Indices>
970constexpr auto TrivialSetMultiCase(Selector selector,
const Values& values, std::index_sequence<0, Indices...>) {
971 auto selector2 = selector.Case(std::data(values)[0]);
972 ((selector2 = selector2.Case(std::data(values)[Indices])), ...);
976template <
const auto& Values>
977struct TrivialSetMultiCaseDispatch {
978 template <
class Selector>
979 constexpr auto operator()(Selector selector)
const {
980 constexpr auto kValuesSize = std::size(Values);
981 return impl::TrivialSetMultiCase(selector(), Values, std::make_index_sequence<kValuesSize>{});
988template <
const auto& Keys,
const auto& Values>
990 static_assert(std::size(Keys) == std::size(Values));
991 static_assert(std::size(Keys) >= 1);
992 return TrivialBiMap(impl::TrivialBiMapMultiCaseDispatch<Keys, Values>{});
995template <
const auto& Values>
996constexpr auto MakeTrivialSet() {
997 return TrivialSet(impl::TrivialSetMultiCaseDispatch<Values>{});
1002USERVER_NAMESPACE_END