Crow  0.3
A C++ microframework for the web
utility.h
1#pragma once
2
3#include <cstdint>
4#include <stdexcept>
5#include <tuple>
6#include <type_traits>
7#include <cstring>
8#include <functional>
9#include <string>
10
11#include <boost/algorithm/string.hpp>
12
13#include "crow/settings.h"
14
15namespace crow
16{
17 namespace black_magic
18 {
19#ifndef CROW_MSVC_WORKAROUND
21 {
22 OutOfRange(unsigned /*pos*/, unsigned /*length*/) {}
23 };
24 constexpr unsigned requires_in_range( unsigned i, unsigned len )
25 {
26 return i >= len ? throw OutOfRange(i, len) : i;
27 }
28
29 /// A constant string implementation.
31 {
32 const char * const begin_;
33 unsigned size_;
34
35 public:
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");
39 }
40 constexpr char operator[]( unsigned i ) const {
41 return requires_in_range(i, size_), begin_[i];
42 }
43
44 constexpr operator const char *() const {
45 return begin_;
46 }
47
48 constexpr const char* begin() const { return begin_; }
49 constexpr const char* end() const { return begin_ + size_; }
50
51 constexpr unsigned size() const {
52 return size_;
53 }
54 };
55
56 constexpr unsigned find_closing_tag(const_str s, unsigned p)
57 {
58 return s[p] == '>' ? p : find_closing_tag(s, p+1);
59 }
60
61 constexpr bool is_valid(const_str s, unsigned i = 0, int f = 0)
62 {
63 return
64 i == s.size()
65 ? f == 0 :
66 f < 0 || f >= 2
67 ? false :
68 s[i] == '<'
69 ? is_valid(s, i+1, f+1) :
70 s[i] == '>'
71 ? is_valid(s, i+1, f-1) :
72 is_valid(s, i+1, f);
73 }
74
75 constexpr bool is_equ_p(const char* a, const char* b, unsigned n)
76 {
77 return
78 *a == 0 && *b == 0 && n == 0
79 ? true :
80 (*a == 0 || *b == 0)
81 ? false :
82 n == 0
83 ? true :
84 *a != *b
85 ? false :
86 is_equ_p(a+1, b+1, n-1);
87 }
88
89 constexpr bool is_equ_n(const_str a, unsigned ai, const_str b, unsigned bi, unsigned n)
90 {
91 return
92 ai + n > a.size() || bi + n > b.size()
93 ? false :
94 n == 0
95 ? true :
96 a[ai] != b[bi]
97 ? false :
98 is_equ_n(a,ai+1,b,bi+1,n-1);
99 }
100
101 constexpr bool is_int(const_str s, unsigned i)
102 {
103 return is_equ_n(s, i, "<int>", 0, 5);
104 }
105
106 constexpr bool is_uint(const_str s, unsigned i)
107 {
108 return is_equ_n(s, i, "<uint>", 0, 6);
109 }
110
111 constexpr bool is_float(const_str s, unsigned i)
112 {
113 return is_equ_n(s, i, "<float>", 0, 7) ||
114 is_equ_n(s, i, "<double>", 0, 8);
115 }
116
117 constexpr bool is_str(const_str s, unsigned i)
118 {
119 return is_equ_n(s, i, "<str>", 0, 5) ||
120 is_equ_n(s, i, "<string>", 0, 8);
121 }
122
123 constexpr bool is_path(const_str s, unsigned i)
124 {
125 return is_equ_n(s, i, "<path>", 0, 6);
126 }
127#endif
128 template <typename T>
130 {
131 static const int value = 0;
132 };
133#define CROW_INTERNAL_PARAMETER_TAG(t, i) \
134template <> \
135struct parameter_tag<t> \
136{ \
137 static const int value = i; \
138}
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>
154
155 template <>
157 {
158 static const int value = 0;
159 };
160
161 template <typename Arg, typename ... Args>
163 {
164 static const int sub_value =
166 static const int value =
168 ? sub_value* 6 + parameter_tag<typename std::decay<Arg>::type>::value
169 : sub_value;
170 };
171
172 static inline bool is_parameter_tag_compatible(uint64_t a, uint64_t b)
173 {
174 if (a == 0)
175 return b == 0;
176 if (b == 0)
177 return a == 0;
178 int sa = a%6;
179 int sb = a%6;
180 if (sa == 5) sa = 4;
181 if (sb == 5) sb = 4;
182 if (sa != sb)
183 return false;
184 return is_parameter_tag_compatible(a/6, b/6);
185 }
186
187 static inline unsigned find_closing_tag_runtime(const char* s, unsigned p)
188 {
189 return
190 s[p] == 0
191 ? throw std::runtime_error("unmatched tag <") :
192 s[p] == '>'
193 ? p : find_closing_tag_runtime(s, p + 1);
194 }
195
196 static inline uint64_t get_parameter_tag_runtime(const char* s, unsigned p = 0)
197 {
198 return
199 s[p] == 0
200 ? 0 :
201 s[p] == '<' ? (
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")
215 ) :
216 get_parameter_tag_runtime(s, p+1);
217 }
218#ifndef CROW_MSVC_WORKAROUND
219 constexpr uint64_t get_parameter_tag(const_str s, unsigned p = 0)
220 {
221 return
222 p == s.size()
223 ? 0 :
224 s[p] == '<' ? (
225 is_int(s, p)
226 ? get_parameter_tag(s, find_closing_tag(s, p)) * 6 + 1 :
227 is_uint(s, p)
228 ? get_parameter_tag(s, find_closing_tag(s, p)) * 6 + 2 :
229 is_float(s, p)
230 ? get_parameter_tag(s, find_closing_tag(s, p)) * 6 + 3 :
231 is_str(s, p)
232 ? get_parameter_tag(s, find_closing_tag(s, p)) * 6 + 4 :
233 is_path(s, p)
234 ? get_parameter_tag(s, find_closing_tag(s, p)) * 6 + 5 :
235 throw std::runtime_error("invalid parameter type")
236 ) :
237 get_parameter_tag(s, p+1);
238 }
239#endif
240
241 template <typename ... T>
242 struct S
243 {
244 template <typename U>
245 using push = S<U, T...>;
246 template <typename U>
247 using push_back = S<T..., U>;
248 template <template<typename ... Args> class U>
249 using rebind = U<T...>;
250 };
251template <typename F, typename Set>
253 template <typename F, typename ...Args>
254 struct CallHelper<F, S<Args...>>
255 {
256 template <typename F1, typename ...Args1, typename =
257 decltype(std::declval<F1>()(std::declval<Args1>()...))
258 >
259 static char __test(int);
260
261 template <typename ...>
262 static int __test(...);
263
264 static constexpr bool value = sizeof(__test<F, Args...>(0)) == sizeof(char);
265 };
266
267
268 template <int N>
270 {
271 };
272
273 template <>
275 {
276 using type = int64_t;
277 };
278
279 template <>
281 {
282 using type = uint64_t;
283 };
284
285 template <>
287 {
288 using type = double;
289 };
290
291 template <>
293 {
294 using type = std::string;
295 };
296
297 template <>
299 {
300 using type = std::string;
301 };
302
303
304 template <uint64_t Tag>
306 {
307 using subarguments = typename arguments<Tag/6>::type;
308 using type =
309 typename subarguments::template push<typename single_tag_to_type<Tag%6>::type>;
310 };
311
312 template <>
313 struct arguments<0>
314 {
315 using type = S<>;
316 };
317
318 template <typename ... T>
320 {
321 using type = typename std::tuple_element<sizeof...(T)-1, std::tuple<T...>>::type;
322 };
323
324
325 template <>
327 {
328 };
329
330
331 // from http://stackoverflow.com/questions/13072359/c11-compile-time-array-with-logarithmic-evaluation-depth
332 template<class T> using Invoke = typename T::type;
333
334 template<unsigned...> struct seq{ using type = seq; };
335
336 template<class S1, class S2> struct concat;
337
338 template<unsigned... I1, unsigned... I2>
339 struct concat<seq<I1...>, seq<I2...>>
340 : seq<I1..., (sizeof...(I1)+I2)...>{};
341
342 template<class S1, class S2>
343 using Concat = Invoke<concat<S1, S2>>;
344
345 template<unsigned N> struct gen_seq;
346 template<unsigned N> using GenSeq = Invoke<gen_seq<N>>;
347
348 template<unsigned N>
349 struct gen_seq : Concat<GenSeq<N/2>, GenSeq<N - N/2>>{};
350
351 template<> struct gen_seq<0> : seq<>{};
352 template<> struct gen_seq<1> : seq<0>{};
353
354 template <typename Seq, typename Tuple>
356
357 template <unsigned ... N, typename Tuple>
358 struct pop_back_helper<seq<N...>, Tuple>
359 {
360 template <template <typename ... Args> class U>
361 using rebind = U<typename std::tuple_element<N, Tuple>::type...>;
362 };
363
364 template <typename ... T>
365 struct pop_back //: public pop_back_helper<typename gen_seq<sizeof...(T)-1>::type, std::tuple<T...>>
366 {
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>;
369 };
370
371 template <>
372 struct pop_back<>
373 {
374 template <template <typename ... Args> class U>
375 using rebind = U<>;
376 };
377
378 // from http://stackoverflow.com/questions/2118541/check-if-c0x-parameter-pack-contains-a-type
379 template < typename Tp, typename... List >
380 struct contains : std::true_type {};
381
382 template < typename Tp, typename Head, typename... Rest >
383 struct contains<Tp, Head, Rest...>
384 : std::conditional< std::is_same<Tp, Head>::value,
385 std::true_type,
386 contains<Tp, Rest...>
387 >::type {};
388
389 template < typename Tp >
390 struct contains<Tp> : std::false_type {};
391
392 template <typename T>
394 {
395 };
396
397 template <typename T>
398 struct promote
399 {
400 using type = T;
401 };
402
403#define CROW_INTERNAL_PROMOTE_TYPE(t1, t2) \
404 template<> \
405 struct promote<t1> \
406 { \
407 using type = t2; \
408 }
409
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
422
423 template <typename T>
424 using promote_t = typename promote<T>::type;
425
426 } // namespace black_magic
427
428 namespace detail
429 {
430
431 template <class T, std::size_t N, class... Args>
433 {
434 static constexpr auto value = N;
435 };
436
437 template <class T, std::size_t N, class... Args>
439 {
440 static constexpr auto value = N;
441 };
442
443 template <class T, std::size_t N, class U, class... Args>
445 {
446 static constexpr auto value = get_index_of_element_from_tuple_by_type_impl<T, N + 1, Args...>::value;
447 };
448
449 } // namespace detail
450
451 namespace utility
452 {
453 template <class T, class... Args>
454 T& get_element_by_type(std::tuple<Args...>& t)
455 {
456 return std::get<detail::get_index_of_element_from_tuple_by_type_impl<T, 0, Args...>::value>(t);
457 }
458
459 template<typename T>
460 struct function_traits;
461
462#ifndef CROW_MSVC_WORKAROUND
463 template<typename T>
464 struct function_traits : public function_traits<decltype(&T::operator())>
465 {
466 using parent_t = function_traits<decltype(&T::operator())>;
467 static const size_t arity = parent_t::arity;
468 using result_type = typename parent_t::result_type;
469 template <size_t i>
470 using arg = typename parent_t::template arg<i>;
471
472 };
473#endif
474
475 template<typename ClassType, typename R, typename ...Args>
476 struct function_traits<R(ClassType::*)(Args...) const>
477 {
478 static const size_t arity = sizeof...(Args);
479
480 typedef R result_type;
481
482 template <size_t i>
483 using arg = typename std::tuple_element<i, std::tuple<Args...>>::type;
484 };
485
486 template<typename ClassType, typename R, typename ...Args>
487 struct function_traits<R(ClassType::*)(Args...)>
488 {
489 static const size_t arity = sizeof...(Args);
490
491 typedef R result_type;
492
493 template <size_t i>
494 using arg = typename std::tuple_element<i, std::tuple<Args...>>::type;
495 };
496
497 template<typename R, typename ...Args>
498 struct function_traits<std::function<R(Args...)>>
499 {
500 static const size_t arity = sizeof...(Args);
501
502 typedef R result_type;
503
504 template <size_t i>
505 using arg = typename std::tuple_element<i, std::tuple<Args...>>::type;
506 };
507
508 inline static std::string base64encode(const char* data, size_t size, const char* key = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/")
509 {
510 std::string ret;
511 ret.resize((size+2) / 3 * 4);
512 auto it = ret.begin();
513 while(size >= 3)
514 {
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];
521
522 size -= 3;
523 }
524 if (size == 1)
525 {
526 *it++ = key[(static_cast<unsigned char>(*data)&0xFC)>>2];
527 unsigned char h = (static_cast<unsigned char>(*data++) & 0x03) << 4;
528 *it++ = key[h];
529 *it++ = '=';
530 *it++ = '=';
531 }
532 else if (size == 2)
533 {
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;
538 *it++ = key[h];
539 *it++ = '=';
540 }
541 return ret;
542 }
543
544 inline static std::string base64encode_urlsafe(const char* data, size_t size)
545 {
546 return base64encode(data, size, "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_");
547 }
548
549 inline static void sanitize_filename(std::string& data, char replacement = '_')
550 {
551 unsigned char i = 0, length_limit;
552
553 length_limit = data.length() < 255 ? data.length() : 255;
554 data = data.substr(0, length_limit);
555
556 for (; i < length_limit; i++)
557 {
558 switch ((unsigned char)data[i])
559 {
560 // WARNING While I can't see how using '\' or '/' would cause a problem, it still warrants an investigation
561 //case '/':
562 case '?':
563 case '<':
564 case '>':
565 //case '\\':
566 case ':':
567 case '*':
568 case '|':
569 case '\"':
570
571 case 0x00:
572 case 0x01:
573 case 0x02:
574 case 0x03:
575 case 0x04:
576 case 0x05:
577 case 0x06:
578 case 0x07:
579 case 0x08:
580 case 0x09:
581 case 0x0a:
582 case 0x0b:
583 case 0x0c:
584 case 0x0d:
585 case 0x0e:
586 case 0x0f:
587 case 0x10:
588 case 0x11:
589 case 0x12:
590 case 0x13:
591 case 0x14:
592 case 0x15:
593 case 0x16:
594 case 0x17:
595 case 0x18:
596 case 0x19:
597 case 0x1a:
598 case 0x1b:
599 case 0x1c:
600 case 0x1d:
601 case 0x1e:
602 case 0x1f:
603
604 case 0x80:
605 case 0x81:
606 case 0x82:
607 case 0x83:
608 case 0x84:
609 case 0x85:
610 case 0x86:
611 case 0x87:
612 case 0x88:
613 case 0x89:
614 case 0x8a:
615 case 0x8b:
616 case 0x8c:
617 case 0x8d:
618 case 0x8e:
619 case 0x8f:
620 case 0x90:
621 case 0x91:
622 case 0x92:
623 case 0x93:
624 case 0x94:
625 case 0x95:
626 case 0x96:
627 case 0x97:
628 case 0x98:
629 case 0x99:
630 case 0x9a:
631 case 0x9b:
632 case 0x9c:
633 case 0x9d:
634 case 0x9e:
635 case 0x9f:
636
637 data[i] = replacement;
638 break;
639
640 default:
641 break;
642 }
643 }
644 std::string str_replacement(1, replacement);
645
646 boost::ireplace_all(data, "..", str_replacement);
647
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);
670 }
671
672 } // namespace utility
673}
A constant string implementation.
Definition: utility.h:31
Definition: utility.h:252
Definition: utility.h:21
Definition: utility.h:243
Definition: utility.h:306
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:399
Definition: utility.h:334
Definition: utility.h:270
Definition: utility.h:465