27using NetId = uint32_t;
28using want_t = int_fast8_t;
30OPENDHT_PUBLIC
const char* version();
34using Sp = std::shared_ptr<T>;
36template<
typename Key,
typename Item,
typename Condition>
38erase_if(std::map<Key, Item>& map,
const Condition& condition)
40 for (
auto it = map.begin(); it != map.end();) {
49template<
typename... Args>
53 static_assert((std::is_constructible_v<std::string_view, Args&&> && ...));
55 s.reserve((std::string_view {args}.size() + ...));
56 (s.append(std::forward<Args>(args)), ...);
63OPENDHT_PUBLIC std::pair<std::string_view, std::string_view>
splitPort(std::string_view s);
65class OPENDHT_PUBLIC DhtException :
public std::runtime_error
68 DhtException(
const std::string& str =
"")
69 : std::runtime_error(
"DhtException occurred: " + str)
73class OPENDHT_PUBLIC SocketException :
public DhtException
76 SocketException(
int err)
77 : DhtException(strerror(err))
83using clock = std::chrono::steady_clock;
84using system_clock = std::chrono::system_clock;
85using time_point = clock::time_point;
86using duration = clock::duration;
88time_point from_time_t(std::time_t t);
89std::time_t to_time_t(time_point t);
95 if (d < std::chrono::seconds(0)) {
96 return "-" + print_duration(-d);
97 }
else if (d < std::chrono::milliseconds(1)) {
98 return fmt::format(
"{:.3g} us",
99 std::chrono::duration_cast<std::chrono::duration<double, std::micro>>(d).count());
100 }
else if (d < std::chrono::seconds(1)) {
101 return fmt::format(
"{:.3g} ms",
102 std::chrono::duration_cast<std::chrono::duration<double, std::milli>>(d).count());
103 }
else if (d < std::chrono::minutes(1)) {
104 return fmt::format(
"{:.3g} s", std::chrono::duration_cast<std::chrono::duration<double>>(d).count());
105 }
else if (d < std::chrono::hours(1)) {
106 return fmt::format(
"{:.3g} min",
107 std::chrono::duration_cast<std::chrono::duration<
double, std::ratio<60>>>(d).count());
108 }
else if (d < std::chrono::hours(72)) {
109 return fmt::format(
"{:.3g} h",
110 std::chrono::duration_cast<std::chrono::duration<
double, std::ratio<3600>>>(d).count());
112 return fmt::format(
"{:.3g} days",
113 std::chrono::duration_cast<std::chrono::duration<
double, std::ratio<86400>>>(d).count());
117template<
class TimePo
int>
119print_time_relative(TimePoint now, TimePoint d)
121 using namespace std::literals;
122 if (d == TimePoint::min())
126 return (d > now) ? concat(
"in "sv, print_duration(d - now)) : concat(print_duration(now - d),
" ago"sv);
129template<
typename Duration = duration>
130class uniform_duration_distribution :
public std::uniform_int_distribution<typename Duration::rep>
132 using Base = std::uniform_int_distribution<typename Duration::rep>;
133 using param_type =
typename Base::param_type;
136 uniform_duration_distribution(Duration min, Duration max)
137 : Base(min.count(), max.count())
139 template<
class Generator>
140 Duration operator()(Generator&& g)
142 return Duration(Base::operator()(g));
144 template<
class Generator>
145 Duration operator()(Generator&& g,
const param_type& params)
147 return Duration(Base::operator()(g, params));
156using Blob = std::vector<uint8_t>;
163template<
typename Type>
165packMsg(
const Type& t)
167 msgpack::sbuffer buffer;
168 msgpack::packer<msgpack::sbuffer> pk(&buffer);
170 return {buffer.data(), buffer.data() + buffer.size()};
173template<
typename Type>
175unpackMsg(
const Blob& b)
177 msgpack::unpacked msg_res = msgpack::unpack((
const char*) b.data(), b.size());
178 return msg_res.get().as<Type>();
181inline msgpack::unpacked
182unpackMsg(
const Blob& b)
184 return msgpack::unpack((
const char*) b.data(), b.size());
187msgpack::object* findMapValue(
const msgpack::object& map,
const char* key,
size_t length);
189inline msgpack::object*
190findMapValue(
const msgpack::object& map,
const char* key)
192 return findMapValue(map, key, strlen(key));
194inline msgpack::object*
195findMapValue(
const msgpack::object& map, std::string_view key)
197 return findMapValue(map, key.data(), key.size());
OPENDHT_PUBLIC Blob unpackBlob(const msgpack::object &o)
OPENDHT_PUBLIC std::pair< std::string_view, std::string_view > splitPort(std::string_view s)
std::vector< uint8_t > Blob