#pragma once // This header is here to provide just the basic bt_value/bt_dict/bt_list definitions without // needing to include the full bt_serialize.h header. #include #include #include #include #include #include namespace oxenc { struct bt_value; /// The type used to store dictionaries inside bt_value. using bt_dict = std::map; // NB: unordered_map doesn't work because it can't be used with a predeclared type /// The type used to store list items inside bt_value. using bt_list = std::list; /// The basic variant that can hold anything (recursively). using bt_variant = std::variant< std::string, std::string_view, int64_t, uint64_t, bt_list, bt_dict >; #ifdef __cpp_lib_remove_cvref // C++20 using std::remove_cvref_t; #else template using remove_cvref_t = std::remove_cv_t>; #endif template struct has_alternative; template struct has_alternative> : std::bool_constant<(std::is_same_v || ...)> {}; template constexpr bool has_alternative_v = has_alternative::value; namespace detail { template bt_list tuple_to_list(const Tuple& tuple, std::index_sequence) { return {{bt_value{std::get(tuple)}...}}; } template constexpr bool is_tuple = false; template constexpr bool is_tuple> = true; template constexpr bool is_tuple> = true; } /// Recursive generic type that can fully represent everything valid for a BT serialization. /// This is basically just an empty wrapper around the std::variant, except we add some extra /// converting constructors: /// - integer constructors so that any unsigned value goes to the uint64_t and any signed value goes /// to the int64_t. /// - std::tuple and std::pair constructors that build a bt_list out of the tuple/pair elements. struct bt_value : bt_variant { using bt_variant::bt_variant; using bt_variant::operator=; template , std::enable_if_t && std::is_unsigned_v, int> = 0> bt_value(T&& uint) : bt_variant{static_cast(uint)} {} template , std::enable_if_t && std::is_signed_v, int> = 0> bt_value(T&& sint) : bt_variant{static_cast(sint)} {} template bt_value(const std::tuple& tuple) : bt_variant{detail::tuple_to_list(tuple, std::index_sequence_for{})} {} template bt_value(const std::pair& pair) : bt_variant{detail::tuple_to_list(pair, std::index_sequence_for{})} {} template , std::enable_if_t && !detail::is_tuple, int> = 0> bt_value(T&& v) : bt_variant{std::forward(v)} {} bt_value(const char* s) : bt_value{std::string_view{s}} {} }; }