42std::string
ltrim(
const std::string &s);
50std::string
rtrim(
const std::string &s);
58std::string
trim(
const std::string &s);
88std::string
get_env(
const std::string &name);
137template <
typename T,
typename S>
140 if (T *
const converted =
dynamic_cast<T *
>(p.get())) {
141 std::move(p).release();
142 return std::unique_ptr<T>{converted};
155 const std::string &separator,
const std::string &input);
170std::vector<std::string>
split(
171 std::string str, std::string_view delimiter,
bool skip_empty =
true);
183template <
typename T,
typename F>
void erase_if(std::vector<T> &v, F &&f)
185 v.erase(std::remove_if(v.begin(), v.end(), std::forward<F>(f)), v.end());
196 const std::vector<std::string> &toks, std::string_view delimiter);
206template <
typename... Args>
207std::string
join(std::string_view delimiter, Args... args)
209 std::vector<std::string> coll{args...};
212 return s.find_first_not_of(
" \t") == std::string::npos;
215 return fmt::format(
"{}", fmt::join(coll, delimiter));
226std::string
abbreviate(
const std::string &s,
unsigned int max_length);
239 const std::string &input, std::tuple<std::string, size_t, size_t> &result);
248bool replace_all(std::string &input,
const std::string &pattern,
249 const std::string &replace_with);
266template <
typename T>
void append(std::vector<T> &l,
const std::vector<T> &r)
268 l.insert(l.end(), r.begin(), r.end());
279template <
typename T>
bool starts_with(
const T &col,
const T &prefix)
281 if (prefix.size() > col.size())
284 return std::search(col.begin(), col.end(), prefix.begin(), prefix.end()) ==
288template <>
bool starts_with(
const std::string &s,
const std::string &prefix);
290template <
typename T>
bool ends_with(
const T &value,
const T &suffix);
292template <>
bool ends_with(
const std::string &value,
const std::string &suffix);
295bool ends_with(
const std::vector<T> &col,
const std::vector<T> &suffix)
297 if (suffix.size() > col.size())
300 return std::vector<std::string>(suffix.rbegin(), suffix.rend()) ==
301 std::vector<std::string>(col.rbegin(), col.rbegin() + suffix.size());
317 col = std::vector<T>(col.begin() + prefix.size(), col.end());
329template <
typename T,
typename E>
332 if constexpr (std::is_pointer_v<E>) {
333 return std::find_if(container.begin(), container.end(),
334 [&element](
const auto &e) { return *e == *element; }) !=
337 else if constexpr (std::is_same_v<std::remove_cv_t<T>, std::string>) {
338 return container.find(element) != std::string::npos;
341 return std::find(container.begin(), container.end(), element) !=
346template <
typename T,
typename F>
void for_each(
const T &collection, F &&func)
348 std::for_each(std::begin(collection), std::end(collection),
349 std::forward<
decltype(func)>(func));
352template <
typename R,
typename T,
typename F>
353std::vector<R>
map(
const std::vector<T> &in, F &&f)
357 in.cbegin(), in.cend(), std::back_inserter(out), std::forward<F>(f));
361template <
typename T,
typename F,
typename FElse>
364 if (pointer !=
nullptr) {
365 std::forward<F>(func)(pointer);
368 std::forward<FElse>(func_else)();
372template <
typename T,
typename F>
void if_not_null(
const T *pointer, F &&func)
374 if_not_null(pointer, std::forward<F>(func), []() {});
377template <
typename F,
typename FElse>
378void _if(
const bool condition, F &&func, FElse &&func_else)
381 std::forward<F>(func)();
384 std::forward<FElse>(func_else)();
388template <
typename F>
void _if(
const bool condition, F &&func)
390 _if(condition, std::forward<F>(func), []() {});
395 std::sort(coll.begin(), coll.end());
396 coll.erase(std::unique(coll.begin(), coll.end()), coll.end());
416std::string
path_to_url(
const std::filesystem::path &p);
429 const std::filesystem::path &root = std::filesystem::current_path());
440 const std::filesystem::path &parent,
const std::filesystem::path &child);
446 const std::map<std::string, std::string> &m,
const std::string &prefix);