11#include <boost/algorithm/string.hpp>
13#include "crow/settings.h"
19#ifndef CROW_MSVC_WORKAROUND
24 constexpr unsigned requires_in_range(
unsigned i,
unsigned len )
26 return i >= len ?
throw OutOfRange(i, len) : i;
32 const char *
const begin_;
36 template<
unsigned N >
37 constexpr const_str(
const char(&arr)[N] ) : begin_(arr), size_(N - 1) {
38 static_assert( N >= 1,
"not a string literal");
40 constexpr char operator[](
unsigned i )
const {
41 return requires_in_range(i, size_), begin_[i];
44 constexpr operator const char *()
const {
48 constexpr const char* begin()
const {
return begin_; }
49 constexpr const char* end()
const {
return begin_ + size_; }
51 constexpr unsigned size()
const {
56 constexpr unsigned find_closing_tag(
const_str s,
unsigned p)
58 return s[p] ==
'>' ? p : find_closing_tag(s, p+1);
61 constexpr bool is_valid(const_str s,
unsigned i = 0,
int f = 0)
69 ? is_valid(s, i+1, f+1) :
71 ? is_valid(s, i+1, f-1) :
75 constexpr bool is_equ_p(
const char* a,
const char* b,
unsigned n)
78 *a == 0 && *b == 0 && n == 0
86 is_equ_p(a+1, b+1, n-1);
89 constexpr bool is_equ_n(const_str a,
unsigned ai, const_str b,
unsigned bi,
unsigned n)
92 ai + n > a.size() || bi + n > b.size()
98 is_equ_n(a,ai+1,b,bi+1,n-1);
101 constexpr bool is_int(const_str s,
unsigned i)
103 return is_equ_n(s, i,
"<int>", 0, 5);
106 constexpr bool is_uint(const_str s,
unsigned i)
108 return is_equ_n(s, i,
"<uint>", 0, 6);
111 constexpr bool is_float(const_str s,
unsigned i)
113 return is_equ_n(s, i,
"<float>", 0, 7) ||
114 is_equ_n(s, i,
"<double>", 0, 8);
117 constexpr bool is_str(const_str s,
unsigned i)
119 return is_equ_n(s, i,
"<str>", 0, 5) ||
120 is_equ_n(s, i,
"<string>", 0, 8);
123 constexpr bool is_path(const_str s,
unsigned i)
125 return is_equ_n(s, i,
"<path>", 0, 6);
128 template <
typename T>
131 static const int value = 0;
133#define CROW_INTERNAL_PARAMETER_TAG(t, i) \
135struct parameter_tag<t> \
137 static const int value = i; \
139 CROW_INTERNAL_PARAMETER_TAG(
int, 1);
140 CROW_INTERNAL_PARAMETER_TAG(
char, 1);
141 CROW_INTERNAL_PARAMETER_TAG(
short, 1);
142 CROW_INTERNAL_PARAMETER_TAG(
long, 1);
143 CROW_INTERNAL_PARAMETER_TAG(
long long, 1);
144 CROW_INTERNAL_PARAMETER_TAG(
unsigned int, 2);
145 CROW_INTERNAL_PARAMETER_TAG(
unsigned char, 2);
146 CROW_INTERNAL_PARAMETER_TAG(
unsigned short, 2);
147 CROW_INTERNAL_PARAMETER_TAG(
unsigned long, 2);
148 CROW_INTERNAL_PARAMETER_TAG(
unsigned long long, 2);
149 CROW_INTERNAL_PARAMETER_TAG(
double, 3);
150 CROW_INTERNAL_PARAMETER_TAG(std::string, 4);
151#undef CROW_INTERNAL_PARAMETER_TAG
152 template <
typename ... Args>
158 static const int value = 0;
161 template <
typename Arg,
typename ... Args>
164 static const int sub_value =
166 static const int value =
172 static inline bool is_parameter_tag_compatible(uint64_t a, uint64_t b)
184 return is_parameter_tag_compatible(a/6, b/6);
187 static inline unsigned find_closing_tag_runtime(
const char* s,
unsigned p)
191 ?
throw std::runtime_error(
"unmatched tag <") :
193 ? p : find_closing_tag_runtime(s, p + 1);
196 static inline uint64_t get_parameter_tag_runtime(
const char* s,
unsigned p = 0)
202 std::strncmp(s+p,
"<int>", 5) == 0
203 ? get_parameter_tag_runtime(s, find_closing_tag_runtime(s, p)) * 6 + 1 :
204 std::strncmp(s+p,
"<uint>", 6) == 0
205 ? get_parameter_tag_runtime(s, find_closing_tag_runtime(s, p)) * 6 + 2 :
206 (std::strncmp(s+p,
"<float>", 7) == 0 ||
207 std::strncmp(s+p,
"<double>", 8) == 0)
208 ? get_parameter_tag_runtime(s, find_closing_tag_runtime(s, p)) * 6 + 3 :
209 (std::strncmp(s+p,
"<str>", 5) == 0 ||
210 std::strncmp(s+p,
"<string>", 8) == 0)
211 ? get_parameter_tag_runtime(s, find_closing_tag_runtime(s, p)) * 6 + 4 :
212 std::strncmp(s+p,
"<path>", 6) == 0
213 ? get_parameter_tag_runtime(s, find_closing_tag_runtime(s, p)) * 6 + 5 :
214 throw std::runtime_error(
"invalid parameter type")
216 get_parameter_tag_runtime(s, p+1);
218#ifndef CROW_MSVC_WORKAROUND
219 constexpr uint64_t get_parameter_tag(const_str s,
unsigned p = 0)
226 ? get_parameter_tag(s, find_closing_tag(s, p)) * 6 + 1 :
228 ? get_parameter_tag(s, find_closing_tag(s, p)) * 6 + 2 :
230 ? get_parameter_tag(s, find_closing_tag(s, p)) * 6 + 3 :
232 ? get_parameter_tag(s, find_closing_tag(s, p)) * 6 + 4 :
234 ? get_parameter_tag(s, find_closing_tag(s, p)) * 6 + 5 :
235 throw std::runtime_error(
"invalid parameter type")
237 get_parameter_tag(s, p+1);
241 template <
typename ... T>
244 template <
typename U>
245 using push =
S<U, T...>;
246 template <
typename U>
248 template <
template<
typename ... Args>
class U>
249 using rebind = U<T...>;
251template <
typename F,
typename Set>
253 template <
typename F,
typename ...Args>
256 template <
typename F1,
typename ...Args1,
typename =
257 decltype(std::declval<F1>()(std::declval<Args1>()...))
259 static char __test(
int);
261 template <
typename ...>
262 static int __test(...);
264 static constexpr bool value =
sizeof(__test<F, Args...>(0)) ==
sizeof(char);
276 using type = int64_t;
282 using type = uint64_t;
294 using type = std::string;
300 using type = std::string;
304 template <u
int64_t Tag>
307 using subarguments =
typename arguments<Tag/6>::type;
309 typename subarguments::template push<typename single_tag_to_type<Tag%6>::type>;
318 template <
typename ... T>
321 using type =
typename std::tuple_element<
sizeof...(T)-1, std::tuple<T...>>::type;
332 template<
class T>
using Invoke =
typename T::type;
336 template<
class S1,
class S2>
struct concat;
338 template<
unsigned... I1,
unsigned... I2>
340 :
seq<I1..., (sizeof...(I1)+I2)...>{};
342 template<
class S1,
class S2>
343 using Concat = Invoke<concat<S1, S2>>;
345 template<
unsigned N>
struct gen_seq;
346 template<
unsigned N>
using GenSeq = Invoke<gen_seq<N>>;
349 struct gen_seq : Concat<GenSeq<N/2>, GenSeq<N - N/2>>{};
354 template <
typename Seq,
typename Tuple>
357 template <
unsigned ... N,
typename Tuple>
360 template <
template <
typename ... Args>
class U>
361 using rebind = U<typename std::tuple_element<N, Tuple>::type...>;
364 template <
typename ... T>
367 template <
template <
typename ... Args>
class U>
368 using rebind =
typename pop_back_helper<
typename gen_seq<
sizeof...(T)-1>::type, std::tuple<T...>>::template rebind<U>;
374 template <
template <
typename ... Args>
class U>
379 template <
typename Tp,
typename... List >
382 template <
typename Tp,
typename Head,
typename... Rest >
384 : std::conditional< std::is_same<Tp, Head>::value,
386 contains<Tp, Rest...>
389 template <
typename Tp >
392 template <
typename T>
397 template <
typename T>
403#define CROW_INTERNAL_PROMOTE_TYPE(t1, t2) \
410 CROW_INTERNAL_PROMOTE_TYPE(
char, int64_t);
411 CROW_INTERNAL_PROMOTE_TYPE(
short, int64_t);
412 CROW_INTERNAL_PROMOTE_TYPE(
int, int64_t);
413 CROW_INTERNAL_PROMOTE_TYPE(
long, int64_t);
414 CROW_INTERNAL_PROMOTE_TYPE(
long long, int64_t);
415 CROW_INTERNAL_PROMOTE_TYPE(
unsigned char, uint64_t);
416 CROW_INTERNAL_PROMOTE_TYPE(
unsigned short, uint64_t);
417 CROW_INTERNAL_PROMOTE_TYPE(
unsigned int, uint64_t);
418 CROW_INTERNAL_PROMOTE_TYPE(
unsigned long, uint64_t);
419 CROW_INTERNAL_PROMOTE_TYPE(
unsigned long long, uint64_t);
420 CROW_INTERNAL_PROMOTE_TYPE(
float,
double);
421#undef CROW_INTERNAL_PROMOTE_TYPE
423 template <
typename T>
424 using promote_t =
typename promote<T>::type;
431 template <
class T, std::size_t N,
class... Args>
434 static constexpr auto value = N;
437 template <
class T, std::size_t N,
class... Args>
440 static constexpr auto value = N;
443 template <
class T, std::size_t N,
class U,
class... Args>
453 template <
class T,
class... Args>
454 T& get_element_by_type(std::tuple<Args...>& t)
460 struct function_traits;
462#ifndef CROW_MSVC_WORKAROUND
467 static const size_t arity = parent_t::arity;
468 using result_type =
typename parent_t::result_type;
470 using arg =
typename parent_t::template arg<i>;
475 template<
typename ClassType,
typename R,
typename ...Args>
478 static const size_t arity =
sizeof...(Args);
480 typedef R result_type;
483 using arg =
typename std::tuple_element<i, std::tuple<Args...>>::type;
486 template<
typename ClassType,
typename R,
typename ...Args>
489 static const size_t arity =
sizeof...(Args);
491 typedef R result_type;
494 using arg =
typename std::tuple_element<i, std::tuple<Args...>>::type;
497 template<
typename R,
typename ...Args>
500 static const size_t arity =
sizeof...(Args);
502 typedef R result_type;
505 using arg =
typename std::tuple_element<i, std::tuple<Args...>>::type;
508 inline static std::string base64encode(
const char* data,
size_t size,
const char* key =
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/")
511 ret.resize((size+2) / 3 * 4);
512 auto it = ret.begin();
515 *it++ = key[(
static_cast<unsigned char>(*data)&0xFC)>>2];
516 unsigned char h = (
static_cast<unsigned char>(*data++) & 0x03) << 4;
517 *it++ = key[h|((
static_cast<unsigned char>(*data)&0xF0)>>4)];
518 h = (
static_cast<unsigned char>(*data++) & 0x0F) << 2;
519 *it++ = key[h|((
static_cast<unsigned char>(*data)&0xC0)>>6)];
520 *it++ = key[
static_cast<unsigned char>(*data++)&0x3F];
526 *it++ = key[(
static_cast<unsigned char>(*data)&0xFC)>>2];
527 unsigned char h = (
static_cast<unsigned char>(*data++) & 0x03) << 4;
534 *it++ = key[(
static_cast<unsigned char>(*data)&0xFC)>>2];
535 unsigned char h = (
static_cast<unsigned char>(*data++) & 0x03) << 4;
536 *it++ = key[h|((
static_cast<unsigned char>(*data)&0xF0)>>4)];
537 h = (
static_cast<unsigned char>(*data++) & 0x0F) << 2;
544 inline static std::string base64encode_urlsafe(
const char* data,
size_t size)
546 return base64encode(data, size,
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_");
549 inline static void sanitize_filename(std::string& data,
char replacement =
'_')
551 unsigned char i = 0, length_limit;
553 length_limit = data.length() < 255 ? data.length() : 255;
554 data = data.substr(0, length_limit);
556 for (; i < length_limit; i++)
558 switch ((
unsigned char)data[i])
637 data[i] = replacement;
644 std::string str_replacement(1, replacement);
646 boost::ireplace_all(data,
"..", str_replacement);
648 boost::ireplace_all(data,
"CON", str_replacement);
649 boost::ireplace_all(data,
"PRN", str_replacement);
650 boost::ireplace_all(data,
"AUX", str_replacement);
651 boost::ireplace_all(data,
"NUL", str_replacement);
652 boost::ireplace_all(data,
"COM1", str_replacement);
653 boost::ireplace_all(data,
"COM2", str_replacement);
654 boost::ireplace_all(data,
"COM3", str_replacement);
655 boost::ireplace_all(data,
"COM4", str_replacement);
656 boost::ireplace_all(data,
"COM5", str_replacement);
657 boost::ireplace_all(data,
"COM6", str_replacement);
658 boost::ireplace_all(data,
"COM7", str_replacement);
659 boost::ireplace_all(data,
"COM8", str_replacement);
660 boost::ireplace_all(data,
"COM9", str_replacement);
661 boost::ireplace_all(data,
"LPT1", str_replacement);
662 boost::ireplace_all(data,
"LPT2", str_replacement);
663 boost::ireplace_all(data,
"LPT3", str_replacement);
664 boost::ireplace_all(data,
"LPT4", str_replacement);
665 boost::ireplace_all(data,
"LPT5", str_replacement);
666 boost::ireplace_all(data,
"LPT6", str_replacement);
667 boost::ireplace_all(data,
"LPT7", str_replacement);
668 boost::ireplace_all(data,
"LPT8", str_replacement);
669 boost::ireplace_all(data,
"LPT9", str_replacement);
A constant string implementation.
Definition: utility.h:31
Definition: utility.h:252
Definition: utility.h:243
Definition: utility.h:306
Definition: utility.h:153
Definition: utility.h:336
Definition: utility.h:380
Definition: utility.h:394
Definition: utility.h:349
Definition: utility.h:320
Definition: utility.h:130
Definition: utility.h:355
Definition: utility.h:366
Definition: utility.h:334
Definition: utility.h:275
Definition: utility.h:281
Definition: utility.h:287
Definition: utility.h:293
Definition: utility.h:299
Definition: utility.h:270
Definition: utility.h:433
Definition: utility.h:465