12 #include <unordered_map>
16 #include "crow/settings.h"
18 #if defined(CROW_CAN_USE_CPP17) && !defined(CROW_FILESYSTEM_IS_EXPERIMENTAL)
23 #if defined(__GNUG__) || defined(__clang__)
24 #define CROW_LIKELY(X) __builtin_expect(!!(X), 1)
25 #define CROW_UNLIKELY(X) __builtin_expect(!!(X), 0)
27 #define CROW_LIKELY(X) (X)
28 #define CROW_UNLIKELY(X) (X)
36 #ifndef CROW_MSVC_WORKAROUND
40 OutOfRange(
unsigned ,
unsigned ) {}
43 constexpr
unsigned requires_in_range(
unsigned i,
unsigned len)
45 return i >= len ?
throw OutOfRange(i, len) : i;
51 const char*
const begin_;
56 constexpr const_str(
const char (&arr)[N]):
57 begin_(arr), size_(N - 1)
59 static_assert(N >= 1,
"not a string literal");
61 constexpr
char operator[](
unsigned i)
const
63 return requires_in_range(i, size_), begin_[i];
66 constexpr
operator const char*()
const
71 constexpr
const char* begin()
const {
return begin_; }
72 constexpr
const char* end()
const {
return begin_ + size_; }
74 constexpr
unsigned size()
const
80 constexpr
unsigned find_closing_tag(const_str s,
unsigned p)
82 return s[p] ==
'>' ? p : find_closing_tag(s, p + 1);
86 constexpr
bool is_valid(const_str s,
unsigned i = 0,
int f = 0)
88 return i == s.size() ? f == 0 :
89 f < 0 || f >= 2 ? false :
90 s[i] ==
'<' ? is_valid(s, i + 1, f + 1) :
91 s[i] ==
'>' ? is_valid(s, i + 1, f - 1) :
92 is_valid(s, i + 1, f);
95 constexpr
bool is_equ_p(
const char* a,
const char* b,
unsigned n)
97 return *a == 0 && *b == 0 && n == 0 ? true :
98 (*a == 0 || *b == 0) ?
false :
101 is_equ_p(a + 1, b + 1, n - 1);
104 constexpr
bool is_equ_n(const_str a,
unsigned ai, const_str b,
unsigned bi,
unsigned n)
106 return ai + n > a.size() || bi + n > b.size() ? false :
108 a[ai] != b[bi] ? false :
109 is_equ_n(a, ai + 1, b, bi + 1, n - 1);
112 constexpr
bool is_int(const_str s,
unsigned i)
114 return is_equ_n(s, i,
"<int>", 0, 5);
117 constexpr
bool is_uint(const_str s,
unsigned i)
119 return is_equ_n(s, i,
"<uint>", 0, 6);
122 constexpr
bool is_float(const_str s,
unsigned i)
124 return is_equ_n(s, i,
"<float>", 0, 7) ||
125 is_equ_n(s, i,
"<double>", 0, 8);
128 constexpr
bool is_str(const_str s,
unsigned i)
130 return is_equ_n(s, i,
"<str>", 0, 5) ||
131 is_equ_n(s, i,
"<string>", 0, 8);
134 constexpr
bool is_path(const_str s,
unsigned i)
136 return is_equ_n(s, i,
"<path>", 0, 6);
142 static const int value = 0;
144 #define CROW_INTERNAL_PARAMETER_TAG(t, i) \
146 struct parameter_tag<t> \
148 static const int value = i; \
150 CROW_INTERNAL_PARAMETER_TAG(
int, 1);
151 CROW_INTERNAL_PARAMETER_TAG(
char, 1);
152 CROW_INTERNAL_PARAMETER_TAG(
short, 1);
153 CROW_INTERNAL_PARAMETER_TAG(
long, 1);
154 CROW_INTERNAL_PARAMETER_TAG(
long long, 1);
155 CROW_INTERNAL_PARAMETER_TAG(
unsigned int, 2);
156 CROW_INTERNAL_PARAMETER_TAG(
unsigned char, 2);
157 CROW_INTERNAL_PARAMETER_TAG(
unsigned short, 2);
158 CROW_INTERNAL_PARAMETER_TAG(
unsigned long, 2);
159 CROW_INTERNAL_PARAMETER_TAG(
unsigned long long, 2);
160 CROW_INTERNAL_PARAMETER_TAG(
double, 3);
161 CROW_INTERNAL_PARAMETER_TAG(std::string, 4);
162 #undef CROW_INTERNAL_PARAMETER_TAG
163 template<
typename... Args>
164 struct compute_parameter_tag_from_args_list;
167 struct compute_parameter_tag_from_args_list<>
169 static const int value = 0;
172 template<
typename Arg,
typename... Args>
173 struct compute_parameter_tag_from_args_list<Arg, Args...>
175 static const int sub_value =
176 compute_parameter_tag_from_args_list<Args...>::value;
177 static const int value =
178 parameter_tag<typename std::decay<Arg>::type>::value ? sub_value * 6 + parameter_tag<typename std::decay<Arg>::type>::value : sub_value;
181 static inline bool is_parameter_tag_compatible(uint64_t a, uint64_t b)
193 return is_parameter_tag_compatible(a / 6, b / 6);
196 static inline unsigned find_closing_tag_runtime(
const char* s,
unsigned p)
198 return s[p] == 0 ?
throw std::runtime_error(
"unmatched tag <") :
200 find_closing_tag_runtime(s, p + 1);
203 static inline uint64_t get_parameter_tag_runtime(
const char* s,
unsigned p = 0)
205 return s[p] == 0 ? 0 :
207 std::strncmp(s + p,
"<int>", 5) == 0 ? get_parameter_tag_runtime(s, find_closing_tag_runtime(s, p)) * 6 + 1 :
208 std::strncmp(s + p,
"<uint>", 6) == 0 ? get_parameter_tag_runtime(s, find_closing_tag_runtime(s, p)) * 6 + 2 :
209 (std::strncmp(s + p,
"<float>", 7) == 0 ||
210 std::strncmp(s + p,
"<double>", 8) == 0) ?
211 get_parameter_tag_runtime(s, find_closing_tag_runtime(s, p)) * 6 + 3 :
212 (std::strncmp(s + p,
"<str>", 5) == 0 ||
213 std::strncmp(s + p,
"<string>", 8) == 0) ?
214 get_parameter_tag_runtime(s, find_closing_tag_runtime(s, p)) * 6 + 4 :
215 std::strncmp(s + p,
"<path>", 6) == 0 ? get_parameter_tag_runtime(s, find_closing_tag_runtime(s, p)) * 6 + 5 :
216 throw std::runtime_error(
"invalid parameter type")) :
217 get_parameter_tag_runtime(s, p + 1);
219 #ifndef CROW_MSVC_WORKAROUND
220 constexpr uint64_t get_parameter_tag(const_str s,
unsigned p = 0)
222 return p == s.size() ? 0 :
224 is_int(s, p) ? get_parameter_tag(s, find_closing_tag(s, p)) * 6 + 1 :
225 is_uint(s, p) ? get_parameter_tag(s, find_closing_tag(s, p)) * 6 + 2 :
226 is_float(s, p) ? get_parameter_tag(s, find_closing_tag(s, p)) * 6 + 3 :
227 is_str(s, p) ? get_parameter_tag(s, find_closing_tag(s, p)) * 6 + 4 :
228 is_path(s, p) ? get_parameter_tag(s, find_closing_tag(s, p)) * 6 + 5 :
229 throw std::runtime_error(
"invalid parameter type")) :
230 get_parameter_tag(s, p + 1);
234 template<
typename... T>
238 using push = S<U, T...>;
240 using push_back = S<T..., U>;
241 template<
template<
typename... Args>
class U>
242 using rebind = U<T...>;
246 template<
typename F,
typename Set>
248 template<
typename F,
typename... Args>
249 struct CallHelper<F, S<Args...>>
251 template<
typename F1,
typename... Args1,
typename = decltype(std::declval<F1>()(std::declval<Args1>()...))>
252 static char __test(
int);
254 template<
typename...>
255 static int __test(...);
257 static constexpr
bool value =
sizeof(__test<F, Args...>(0)) ==
sizeof(char);
261 template<
typename T,
typename Tuple>
265 struct has_type<T, std::tuple<>> : std::false_type
268 template<
typename T,
typename U,
typename... Ts>
269 struct has_type<T, std::tuple<U, Ts...>> : has_type<T, std::tuple<Ts...>>
272 template<
typename T,
typename... Ts>
273 struct has_type<T, std::tuple<T, Ts...>> : std::true_type
277 template<
class T,
class Tuple>
280 template<
class T,
class... Types>
281 struct tuple_index<T, std::tuple<T, Types...>>
283 static const int value = 0;
286 template<
class T,
class U,
class... Types>
287 struct tuple_index<T, std::tuple<U, Types...>>
289 static const int value = 1 + tuple_index<T, std::tuple<Types...>>::value;
293 #ifdef CROW_CAN_USE_CPP14
294 template<
typename T,
typename Tup>
295 typename std::enable_if<has_type<T&, Tup>::value,
typename std::decay<T>::type&&>::type
296 tuple_extract(Tup& tup)
298 return std::move(std::get<T&>(tup));
301 template<
typename T,
typename Tup>
302 typename std::enable_if<has_type<T&, Tup>::value, T&&>::type
303 tuple_extract(Tup& tup)
305 return std::move(std::get<tuple_index<T&, Tup>::value>(tup));
309 template<
typename T,
typename Tup>
310 typename std::enable_if<!has_type<T&, Tup>::value, T>::type
320 using all_true = std::is_same<bool_pack<bs...,
true>, bool_pack<
true, bs...>>;
323 struct single_tag_to_type
327 struct single_tag_to_type<1>
329 using type = int64_t;
333 struct single_tag_to_type<2>
335 using type = uint64_t;
339 struct single_tag_to_type<3>
345 struct single_tag_to_type<4>
347 using type = std::string;
351 struct single_tag_to_type<5>
353 using type = std::string;
357 template<u
int64_t Tag>
360 using subarguments =
typename arguments<Tag / 6>::type;
362 typename subarguments::template push<typename single_tag_to_type<Tag % 6>::type>;
371 template<
typename... T>
372 struct last_element_type
374 using type =
typename std::tuple_element<
sizeof...(T) - 1, std::tuple<T...>>::type;
379 struct last_element_type<>
385 using Invoke =
typename T::type;
387 template<
unsigned...>
393 template<
class S1,
class S2>
396 template<
unsigned... I1,
unsigned... I2>
397 struct concat<seq<I1...>, seq<I2...>> : seq<I1..., (sizeof...(I1) + I2)...>
400 template<
class S1,
class S2>
401 using Concat = Invoke<concat<S1, S2>>;
406 using GenSeq = Invoke<gen_seq<N>>;
409 struct gen_seq : Concat<GenSeq<N / 2>, GenSeq<N - N / 2>>
413 struct gen_seq<0> : seq<>
416 struct gen_seq<1> : seq<0>
419 template<
typename Seq,
typename Tuple>
420 struct pop_back_helper;
422 template<
unsigned... N,
typename Tuple>
423 struct pop_back_helper<seq<N...>, Tuple>
425 template<
template<
typename... Args>
class U>
426 using rebind = U<typename std::tuple_element<N, Tuple>::type...>;
429 template<
typename... T>
432 template<
template<
typename... Args>
class U>
433 using rebind =
typename pop_back_helper<
typename gen_seq<
sizeof...(T) - 1>::type, std::tuple<T...>>::
template rebind<U>;
439 template<
template<
typename... Args>
class U>
444 template<
typename Tp,
typename... List>
445 struct contains : std::true_type
448 template<
typename Tp,
typename Head,
typename... Rest>
449 struct contains<Tp, Head, Rest...> : std::conditional<std::is_same<Tp, Head>::value, std::true_type, contains<Tp, Rest...>>::type
452 template<
typename Tp>
453 struct contains<Tp> : std::false_type
466 #define CROW_INTERNAL_PROMOTE_TYPE(t1, t2) \
473 CROW_INTERNAL_PROMOTE_TYPE(
char, int64_t);
474 CROW_INTERNAL_PROMOTE_TYPE(
short, int64_t);
475 CROW_INTERNAL_PROMOTE_TYPE(
int, int64_t);
476 CROW_INTERNAL_PROMOTE_TYPE(
long, int64_t);
477 CROW_INTERNAL_PROMOTE_TYPE(
long long, int64_t);
478 CROW_INTERNAL_PROMOTE_TYPE(
unsigned char, uint64_t);
479 CROW_INTERNAL_PROMOTE_TYPE(
unsigned short, uint64_t);
480 CROW_INTERNAL_PROMOTE_TYPE(
unsigned int, uint64_t);
481 CROW_INTERNAL_PROMOTE_TYPE(
unsigned long, uint64_t);
482 CROW_INTERNAL_PROMOTE_TYPE(
unsigned long long, uint64_t);
483 CROW_INTERNAL_PROMOTE_TYPE(
float,
double);
484 #undef CROW_INTERNAL_PROMOTE_TYPE
487 using promote_t =
typename promote<T>::type;
494 template<
class T, std::size_t N,
class... Args>
495 struct get_index_of_element_from_tuple_by_type_impl
497 static constexpr
auto value = N;
500 template<
class T, std::size_t N,
class... Args>
501 struct get_index_of_element_from_tuple_by_type_impl<T, N, T, Args...>
503 static constexpr
auto value = N;
506 template<
class T, std::size_t N,
class U,
class... Args>
507 struct get_index_of_element_from_tuple_by_type_impl<T, N, U, Args...>
509 static constexpr
auto value = get_index_of_element_from_tuple_by_type_impl<T, N + 1, Args...>::value;
515 template<
class T,
class... Args>
516 T& get_element_by_type(std::tuple<Args...>& t)
518 return std::get<detail::get_index_of_element_from_tuple_by_type_impl<T, 0, Args...>::value>(t);
522 struct function_traits;
524 #ifndef CROW_MSVC_WORKAROUND
526 struct function_traits :
public function_traits<decltype(&T::operator())>
528 using parent_t = function_traits<decltype(&T::operator())>;
529 static const size_t arity = parent_t::arity;
530 using result_type =
typename parent_t::result_type;
532 using arg =
typename parent_t::template arg<i>;
536 template<
typename ClassType,
typename R,
typename... Args>
537 struct function_traits<R (ClassType::*)(Args...) const>
539 static const size_t arity =
sizeof...(Args);
541 typedef R result_type;
544 using arg =
typename std::tuple_element<i, std::tuple<Args...>>::type;
547 template<
typename ClassType,
typename R,
typename... Args>
548 struct function_traits<R (ClassType::*)(Args...)>
550 static const size_t arity =
sizeof...(Args);
552 typedef R result_type;
555 using arg =
typename std::tuple_element<i, std::tuple<Args...>>::type;
558 template<
typename R,
typename... Args>
559 struct function_traits<std::function<R(Args...)>>
561 static const size_t arity =
sizeof...(Args);
563 typedef R result_type;
566 using arg =
typename std::tuple_element<i, std::tuple<Args...>>::type;
570 inline static std::string base64encode(
const unsigned char* data,
size_t size,
const char* key =
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/")
573 ret.resize((size + 2) / 3 * 4);
574 auto it = ret.begin();
577 *it++ = key[(
static_cast<unsigned char>(*data) & 0xFC) >> 2];
578 unsigned char h = (
static_cast<unsigned char>(*data++) & 0x03) << 4;
579 *it++ = key[h | ((
static_cast<unsigned char>(*data) & 0xF0) >> 4)];
580 h = (
static_cast<unsigned char>(*data++) & 0x0F) << 2;
581 *it++ = key[h | ((
static_cast<unsigned char>(*data) & 0xC0) >> 6)];
582 *it++ = key[
static_cast<unsigned char>(*data++) & 0x3F];
588 *it++ = key[(
static_cast<unsigned char>(*data) & 0xFC) >> 2];
589 unsigned char h = (
static_cast<unsigned char>(*data++) & 0x03) << 4;
596 *it++ = key[(
static_cast<unsigned char>(*data) & 0xFC) >> 2];
597 unsigned char h = (
static_cast<unsigned char>(*data++) & 0x03) << 4;
598 *it++ = key[h | ((
static_cast<unsigned char>(*data) & 0xF0) >> 4)];
599 h = (
static_cast<unsigned char>(*data++) & 0x0F) << 2;
606 inline static std::string base64encode(std::string data,
size_t size,
const char* key =
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/")
608 return base64encode((
const unsigned char*)data.c_str(), size, key);
611 inline static std::string base64encode_urlsafe(
const unsigned char* data,
size_t size)
613 return base64encode(data, size,
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_");
616 inline static std::string base64encode_urlsafe(std::string data,
size_t size)
618 return base64encode((
const unsigned char*)data.c_str(), size,
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_");
621 inline static std::string base64decode(
const char* data,
size_t size)
627 auto key = [](
char c) ->
unsigned char {
628 if ((c >=
'A') && (c <=
'Z'))
return c -
'A';
629 if ((c >=
'a') && (c <=
'z'))
return c -
'a' + 26;
630 if ((c >=
'0') && (c <=
'9'))
return c -
'0' + 52;
631 if ((c ==
'+') || (c ==
'-'))
return 62;
632 if ((c ==
'/') || (c ==
'_'))
return 63;
638 size = (size / 4 * 3) + 1;
639 else if (size % 4 == 3)
640 size = (size / 4 * 3) + 2;
643 else if (data[size - 2] ==
'=')
644 size = (size / 4 * 3) - 2;
645 else if (data[size - 1] ==
'=')
646 size = (size / 4 * 3) - 1;
654 auto it = ret.begin();
666 *it++ = (odd << 2) | ((even & 0x30) >> 4);
669 *it++ = ((even & 0x0F) << 4) | ((odd & 0x3C) >> 2);
672 *it++ = ((odd & 0x03) << 6) | (even);
681 *it++ = (odd << 2) | ((even & 0x30) >> 4);
684 *it++ = ((even & 0x0F) << 4) | ((odd & 0x3C) >> 2);
691 *it++ = (odd << 2) | ((even & 0x30) >> 4);
696 inline static std::string base64decode(
const std::string& data,
size_t size)
698 return base64decode(data.data(), size);
701 inline static std::string base64decode(
const std::string& data)
703 return base64decode(data.data(), data.length());
706 inline static std::string normalize_path(
const std::string& directoryPath)
708 std::string normalizedPath = directoryPath;
709 std::replace(normalizedPath.begin(), normalizedPath.end(),
'\\',
'/');
710 if (!normalizedPath.empty() && normalizedPath.back() !=
'/')
711 normalizedPath +=
'/';
712 return normalizedPath;
715 inline static void sanitize_filename(std::string& data,
char replacement =
'_')
717 if (data.length() > 255)
720 static const auto toUpper = [](
char c) {
721 return ((c >=
'a') && (c <=
'z')) ? (c - (
'a' -
'A')) : c;
726 auto sanitizeSpecialFile = [](std::string& source,
unsigned ofs,
const char* pattern,
bool includeNumber,
char replacement) {
728 size_t len = source.length();
729 const char* p = pattern;
732 if (i >= len)
return;
733 if (toUpper(source[i]) != *p)
return;
739 if ((i >= len) || (source[i] <
'1') || (source[i] >
'9'))
return;
742 if ((i >= len) || (source[i] ==
'.') || (source[i] ==
':') || (source[i] ==
'/') || (source[i] ==
'\\'))
744 source.erase(ofs + 1, (i - ofs) - 1);
745 source[ofs] = replacement;
748 bool checkForSpecialEntries =
true;
749 for (
unsigned i = 0; i < data.length(); ++i)
752 if (checkForSpecialEntries)
754 checkForSpecialEntries =
false;
755 switch (toUpper(data[i]))
758 sanitizeSpecialFile(data, i,
"AUX",
false, replacement);
761 sanitizeSpecialFile(data, i,
"CON",
false, replacement);
762 sanitizeSpecialFile(data, i,
"COM",
true, replacement);
765 sanitizeSpecialFile(data, i,
"LPT",
true, replacement);
768 sanitizeSpecialFile(data, i,
"NUL",
false, replacement);
771 sanitizeSpecialFile(data, i,
"PRN",
false, replacement);
774 sanitizeSpecialFile(data, i,
"..",
false, replacement);
780 unsigned char c = data[i];
781 if ((c <
' ') || ((c >= 0x80) && (c <= 0x9F)) || (c ==
'?') || (c ==
'<') || (c ==
'>') || (c ==
':') || (c ==
'*') || (c ==
'|') || (c ==
'\"'))
783 data[i] = replacement;
785 else if ((c ==
'/') || (c ==
'\\'))
787 if (CROW_UNLIKELY(i == 0))
789 data[i] = replacement;
793 checkForSpecialEntries =
true;
799 inline static std::string random_alphanum(std::size_t size)
801 static const char alphabet[] =
"0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ";
802 std::random_device dev;
803 std::mt19937 rng(dev());
804 std::uniform_int_distribution<std::mt19937::result_type> dist(0,
sizeof(alphabet) - 2);
807 for (std::size_t i = 0; i < size; i++)
808 out.push_back(alphabet[dist(rng)]);
812 inline static std::string join_path(std::string path,
const std::string& fname)
814 #if defined(CROW_CAN_USE_CPP17) && !defined(CROW_FILESYSTEM_IS_EXPERIMENTAL)
815 return (std::filesystem::path(path) / fname).string();
817 if (!(path.back() ==
'/' || path.back() ==
'\\'))
829 inline static bool string_equals(
const std::string& l,
const std::string& r,
bool case_sensitive =
false)
831 if (l.length() != r.length())
834 for (
size_t i = 0; i < l.length(); i++)
843 if (std::toupper(l[i]) != std::toupper(r[i]))
851 template<
typename T,
typename U>
852 inline static T lexical_cast(
const U& v)
854 std::stringstream stream;
864 inline static T lexical_cast(
const char* v,
size_t count)
866 std::stringstream stream;
869 stream.write(v, count);
878 inline static std::string trim(
const std::string& v)
883 size_t begin = 0, end = v.length();
886 for (i = 0; i < v.length(); i++)
888 if (!std::isspace(v[i]))
898 for (i = v.length(); i > 0; i--)
900 if (!std::isspace(v[i - 1]))
907 return v.substr(begin, end - begin);
The main namespace of the library. In this namespace is defined the most important classes and functi...