Crow  0.3
A C++ microframework for the web
routing.h
1#pragma once
2
3#include <cstdint>
4#include <utility>
5#include <tuple>
6#include <unordered_map>
7#include <memory>
8#include <boost/lexical_cast.hpp>
9#include <vector>
10
11#include "crow/common.h"
12#include "crow/http_response.h"
13#include "crow/http_request.h"
14#include "crow/utility.h"
15#include "crow/logging.h"
16#include "crow/websocket.h"
17
18namespace crow
19{
20 /// A base class for all rules.
21
22 /// Used to provide a common interface for code dealing with different types of rules.
23 /// A Rule provides a URL, allowed HTTP methods, and handlers.
25 {
26 public:
27 BaseRule(std::string rule)
28 : rule_(std::move(rule))
29 {
30 }
31
32 virtual ~BaseRule()
33 {
34 }
35
36 virtual void validate() = 0;
37 std::unique_ptr<BaseRule> upgrade()
38 {
39 if (rule_to_upgrade_)
40 return std::move(rule_to_upgrade_);
41 return {};
42 }
43
44 virtual void handle(const request&, response&, const routing_params&) = 0;
45 virtual void handle_upgrade(const request&, response& res, SocketAdaptor&&)
46 {
47 res = response(404);
48 res.end();
49 }
50#ifdef CROW_ENABLE_SSL
51 virtual void handle_upgrade(const request&, response& res, SSLAdaptor&&)
52 {
53 res = response(404);
54 res.end();
55 }
56#endif
57
58 uint32_t get_methods()
59 {
60 return methods_;
61 }
62
63 template <typename F>
64 void foreach_method(F f)
65 {
66 for(uint32_t method = 0, method_bit = 1; method < static_cast<uint32_t>(HTTPMethod::InternalMethodCount); method++, method_bit<<=1)
67 {
68 if (methods_ & method_bit)
69 f(method);
70 }
71 }
72
73 const std::string& rule() { return rule_; }
74
75 protected:
76 uint32_t methods_{1<<static_cast<int>(HTTPMethod::Get)};
77
78 std::string rule_;
79 std::string name_;
80
81 std::unique_ptr<BaseRule> rule_to_upgrade_;
82
83 friend class Router;
84 template <typename T>
85 friend struct RuleParameterTraits;
86 };
87
88
89 namespace detail
90 {
91 namespace routing_handler_call_helper
92 {
93 template <typename T, int Pos>
94 struct call_pair
95 {
96 using type = T;
97 static const int pos = Pos;
98 };
99
100 template <typename H1>
102 {
103 H1& handler;
104 const routing_params& params;
105 const request& req;
106 response& res;
107 };
108
109 template <typename F, int NInt, int NUint, int NDouble, int NString, typename S1, typename S2>
110 struct call
111 {
112 };
113
114 template <typename F, int NInt, int NUint, int NDouble, int NString, typename ... Args1, typename ... Args2>
115 struct call<F, NInt, NUint, NDouble, NString, black_magic::S<int64_t, Args1...>, black_magic::S<Args2...>>
116 {
117 void operator()(F cparams)
118 {
119 using pushed = typename black_magic::S<Args2...>::template push_back<call_pair<int64_t, NInt>>;
120 call<F, NInt+1, NUint, NDouble, NString,
121 black_magic::S<Args1...>, pushed>()(cparams);
122 }
123 };
124
125 template <typename F, int NInt, int NUint, int NDouble, int NString, typename ... Args1, typename ... Args2>
126 struct call<F, NInt, NUint, NDouble, NString, black_magic::S<uint64_t, Args1...>, black_magic::S<Args2...>>
127 {
128 void operator()(F cparams)
129 {
130 using pushed = typename black_magic::S<Args2...>::template push_back<call_pair<uint64_t, NUint>>;
131 call<F, NInt, NUint+1, NDouble, NString,
132 black_magic::S<Args1...>, pushed>()(cparams);
133 }
134 };
135
136 template <typename F, int NInt, int NUint, int NDouble, int NString, typename ... Args1, typename ... Args2>
137 struct call<F, NInt, NUint, NDouble, NString, black_magic::S<double, Args1...>, black_magic::S<Args2...>>
138 {
139 void operator()(F cparams)
140 {
141 using pushed = typename black_magic::S<Args2...>::template push_back<call_pair<double, NDouble>>;
142 call<F, NInt, NUint, NDouble+1, NString,
143 black_magic::S<Args1...>, pushed>()(cparams);
144 }
145 };
146
147 template <typename F, int NInt, int NUint, int NDouble, int NString, typename ... Args1, typename ... Args2>
148 struct call<F, NInt, NUint, NDouble, NString, black_magic::S<std::string, Args1...>, black_magic::S<Args2...>>
149 {
150 void operator()(F cparams)
151 {
152 using pushed = typename black_magic::S<Args2...>::template push_back<call_pair<std::string, NString>>;
153 call<F, NInt, NUint, NDouble, NString+1,
154 black_magic::S<Args1...>, pushed>()(cparams);
155 }
156 };
157
158 template <typename F, int NInt, int NUint, int NDouble, int NString, typename ... Args1>
159 struct call<F, NInt, NUint, NDouble, NString, black_magic::S<>, black_magic::S<Args1...>>
160 {
161 void operator()(F cparams)
162 {
163 cparams.handler(
164 cparams.req,
165 cparams.res,
166 cparams.params.template get<typename Args1::type>(Args1::pos)...
167 );
168 }
169 };
170
171 template <typename Func, typename ... ArgsWrapped>
172 struct Wrapped
173 {
174 template <typename ... Args>
175 void set_(Func f, typename std::enable_if<
176 !std::is_same<typename std::tuple_element<0, std::tuple<Args..., void>>::type, const request&>::value
177 , int>::type = 0)
178 {
179 handler_ = (
180#ifdef CROW_CAN_USE_CPP14
181 [f = std::move(f)]
182#else
183 [f]
184#endif
185 (const request&, response& res, Args... args){
186 res = response(f(args...));
187 res.end();
188 });
189 }
190
191 template <typename Req, typename ... Args>
193 {
194 req_handler_wrapper(Func f)
195 : f(std::move(f))
196 {
197 }
198
199 void operator()(const request& req, response& res, Args... args)
200 {
201 res = response(f(req, args...));
202 res.end();
203 }
204
205 Func f;
206 };
207
208 template <typename ... Args>
209 void set_(Func f, typename std::enable_if<
210 std::is_same<typename std::tuple_element<0, std::tuple<Args..., void>>::type, const request&>::value &&
211 !std::is_same<typename std::tuple_element<1, std::tuple<Args..., void, void>>::type, response&>::value
212 , int>::type = 0)
213 {
214 handler_ = req_handler_wrapper<Args...>(std::move(f));
215 /*handler_ = (
216 [f = std::move(f)]
217 (const request& req, response& res, Args... args){
218 res = response(f(req, args...));
219 res.end();
220 });*/
221 }
222
223 template <typename ... Args>
224 void set_(Func f, typename std::enable_if<
225 std::is_same<typename std::tuple_element<0, std::tuple<Args..., void>>::type, const request&>::value &&
226 std::is_same<typename std::tuple_element<1, std::tuple<Args..., void, void>>::type, response&>::value
227 , int>::type = 0)
228 {
229 handler_ = std::move(f);
230 }
231
232 template <typename ... Args>
234 {
235 using type = std::function<void(const crow::request&, crow::response&, Args...)>;
237 };
238
239 template <typename ... Args>
240 struct handler_type_helper<const request&, Args...>
241 {
242 using type = std::function<void(const crow::request&, crow::response&, Args...)>;
244 };
245
246 template <typename ... Args>
247 struct handler_type_helper<const request&, response&, Args...>
248 {
249 using type = std::function<void(const crow::request&, crow::response&, Args...)>;
251 };
252
253 typename handler_type_helper<ArgsWrapped...>::type handler_;
254
255 void operator()(const request& req, response& res, const routing_params& params)
256 {
259 decltype(handler_)>,
260 0, 0, 0, 0,
263 >()(
265 decltype(handler_)>
266 {handler_, params, req, res}
267 );
268 }
269 };
270
271 }
272 }
273
274
276 {
277 public:
278 CatchallRule(){}
279
280 template <typename Func>
281 typename std::enable_if<black_magic::CallHelper<Func, black_magic::S<>>::value, void>::type
282 operator()(Func&& f)
283 {
284 static_assert(!std::is_same<void, decltype(f())>::value,
285 "Handler function cannot have void return type; valid return types: string, int, crow::response, crow::returnable");
286
287 handler_ = (
288#ifdef CROW_CAN_USE_CPP14
289 [f = std::move(f)]
290#else
291 [f]
292#endif
293 (const request&, response& res){
294 res = response(f());
295 res.end();
296 });
297
298 }
299
300 template <typename Func>
301 typename std::enable_if<
304 void>::type
305 operator()(Func&& f)
306 {
307 static_assert(!std::is_same<void, decltype(f(std::declval<crow::request>()))>::value,
308 "Handler function cannot have void return type; valid return types: string, int, crow::response, crow::returnable");
309
310 handler_ = (
311#ifdef CROW_CAN_USE_CPP14
312 [f = std::move(f)]
313#else
314 [f]
315#endif
316 (const crow::request& req, crow::response& res){
317 res = response(f(req));
318 res.end();
319 });
320 }
321
322 template <typename Func>
323 typename std::enable_if<
327 void>::type
328 operator()(Func&& f)
329 {
330 static_assert(std::is_same<void, decltype(f(std::declval<crow::response&>()))>::value,
331 "Handler function with response argument should have void return type");
332 handler_ = (
333#ifdef CROW_CAN_USE_CPP14
334 [f = std::move(f)]
335#else
336 [f]
337#endif
338 (const crow::request&, crow::response& res){
339 f(res);
340 });
341 }
342
343 template <typename Func>
344 typename std::enable_if<
348 void>::type
349 operator()(Func&& f)
350 {
351 static_assert(std::is_same<void, decltype(f(std::declval<crow::request>(), std::declval<crow::response&>()))>::value,
352 "Handler function with response argument should have void return type");
353
354 handler_ = std::move(f);
355 }
356
357 bool has_handler()
358 {
359 return (handler_ != nullptr);
360 }
361
362 protected:
363 friend class Router;
364 private:
365 std::function<void(const crow::request&, crow::response&)> handler_;
366 };
367
368
369 /// A rule dealing with websockets.
370
371 /// Provides the interface for the user to put in the necessary handlers for a websocket to work.
372 ///
373 class WebSocketRule : public BaseRule
374 {
375 using self_t = WebSocketRule;
376 public:
377 WebSocketRule(std::string rule)
378 : BaseRule(std::move(rule))
379 {
380 }
381
382 void validate() override
383 {
384 }
385
386 void handle(const request&, response& res, const routing_params&) override
387 {
388 res = response(404);
389 res.end();
390 }
391
392 void handle_upgrade(const request& req, response&, SocketAdaptor&& adaptor) override
393 {
394 new crow::websocket::Connection<SocketAdaptor>(req, std::move(adaptor), open_handler_, message_handler_, close_handler_, error_handler_, accept_handler_);
395 }
396#ifdef CROW_ENABLE_SSL
397 void handle_upgrade(const request& req, response&, SSLAdaptor&& adaptor) override
398 {
399 new crow::websocket::Connection<SSLAdaptor>(req, std::move(adaptor), open_handler_, message_handler_, close_handler_, error_handler_, accept_handler_);
400 }
401#endif
402
403 template <typename Func>
404 self_t& onopen(Func f)
405 {
406 open_handler_ = f;
407 return *this;
408 }
409
410 template <typename Func>
411 self_t& onmessage(Func f)
412 {
413 message_handler_ = f;
414 return *this;
415 }
416
417 template <typename Func>
418 self_t& onclose(Func f)
419 {
420 close_handler_ = f;
421 return *this;
422 }
423
424 template <typename Func>
425 self_t& onerror(Func f)
426 {
427 error_handler_ = f;
428 return *this;
429 }
430
431 template <typename Func>
432 self_t& onaccept(Func f)
433 {
434 accept_handler_ = f;
435 return *this;
436 }
437
438 protected:
439 std::function<void(crow::websocket::connection&)> open_handler_;
440 std::function<void(crow::websocket::connection&, const std::string&, bool)> message_handler_;
441 std::function<void(crow::websocket::connection&, const std::string&)> close_handler_;
442 std::function<void(crow::websocket::connection&)> error_handler_;
443 std::function<bool(const crow::request&)> accept_handler_;
444 };
445
446 /// Allows the user to assign parameters using functions.
447 ///
448 /// `rule.name("name").methods(HTTPMethod::POST)`
449 template <typename T>
451 {
452 using self_t = T;
453 WebSocketRule& websocket()
454 {
455 auto p =new WebSocketRule(static_cast<self_t*>(this)->rule_);
456 static_cast<self_t*>(this)->rule_to_upgrade_.reset(p);
457 return *p;
458 }
459
460 self_t& name(std::string name) noexcept
461 {
462 static_cast<self_t*>(this)->name_ = std::move(name);
463 return static_cast<self_t&>(*this);
464 }
465
466 self_t& methods(HTTPMethod method)
467 {
468 static_cast<self_t*>(this)->methods_ = 1 << static_cast<int>(method);
469 return static_cast<self_t&>(*this);
470 }
471
472 template <typename ... MethodArgs>
473 self_t& methods(HTTPMethod method, MethodArgs ... args_method)
474 {
475 methods(args_method...);
476 static_cast<self_t*>(this)->methods_ |= 1 << static_cast<int>(method);
477 return static_cast<self_t&>(*this);
478 }
479
480 };
481
482 /// A rule that can change its parameters during runtime.
483 class DynamicRule : public BaseRule, public RuleParameterTraits<DynamicRule>
484 {
485 public:
486
487 DynamicRule(std::string rule)
488 : BaseRule(std::move(rule))
489 {
490 }
491
492 void validate() override
493 {
494 if (!erased_handler_)
495 {
496 throw std::runtime_error(name_ + (!name_.empty() ? ": " : "") + "no handler for url " + rule_);
497 }
498 }
499
500 void handle(const request& req, response& res, const routing_params& params) override
501 {
502 erased_handler_(req, res, params);
503 }
504
505 template <typename Func>
506 void operator()(Func f)
507 {
508#ifdef CROW_MSVC_WORKAROUND
509 using function_t = utility::function_traits<decltype(&Func::operator())>;
510#else
511 using function_t = utility::function_traits<Func>;
512#endif
513 erased_handler_ = wrap(std::move(f), black_magic::gen_seq<function_t::arity>());
514 }
515
516 // enable_if Arg1 == request && Arg2 == response
517 // enable_if Arg1 == request && Arg2 != resposne
518 // enable_if Arg1 != request
519#ifdef CROW_MSVC_WORKAROUND
520 template <typename Func, size_t ... Indices>
521#else
522 template <typename Func, unsigned ... Indices>
523#endif
524 std::function<void(const request&, response&, const routing_params&)>
525 wrap(Func f, black_magic::seq<Indices...>)
526 {
527#ifdef CROW_MSVC_WORKAROUND
528 using function_t = utility::function_traits<decltype(&Func::operator())>;
529#else
530 using function_t = utility::function_traits<Func>;
531#endif
532 if (!black_magic::is_parameter_tag_compatible(
533 black_magic::get_parameter_tag_runtime(rule_.c_str()),
535 typename function_t::template arg<Indices>...>::value))
536 {
537 throw std::runtime_error("route_dynamic: Handler type is mismatched with URL parameters: " + rule_);
538 }
540 ret.template set_<
541 typename function_t::template arg<Indices>...
542 >(std::move(f));
543 return ret;
544 }
545
546 template <typename Func>
547 void operator()(std::string name, Func&& f)
548 {
549 name_ = std::move(name);
550 (*this).template operator()<Func>(std::forward(f));
551 }
552 private:
553 std::function<void(const request&, response&, const routing_params&)> erased_handler_;
554
555 };
556
557 /// Default rule created when CROW_ROUTE is called.
558 template <typename ... Args>
559 class TaggedRule : public BaseRule, public RuleParameterTraits<TaggedRule<Args...>>
560 {
561 public:
562 using self_t = TaggedRule<Args...>;
563
564 TaggedRule(std::string rule)
565 : BaseRule(std::move(rule))
566 {
567 }
568
569 void validate() override
570 {
571 if (!handler_)
572 {
573 throw std::runtime_error(name_ + (!name_.empty() ? ": " : "") + "no handler for url " + rule_);
574 }
575 }
576
577 template <typename Func>
578 typename std::enable_if<black_magic::CallHelper<Func, black_magic::S<Args...>>::value, void>::type
579 operator()(Func&& f)
580 {
581 static_assert(black_magic::CallHelper<Func, black_magic::S<Args...>>::value ||
583 "Handler type is mismatched with URL parameters");
584 static_assert(!std::is_same<void, decltype(f(std::declval<Args>()...))>::value,
585 "Handler function cannot have void return type; valid return types: string, int, crow::response, crow::returnable");
586
587 handler_ = (
588#ifdef CROW_CAN_USE_CPP14
589 [f = std::move(f)]
590#else
591 [f]
592#endif
593 (const request&, response& res, Args ... args){
594 res = response(f(args...));
595 res.end();
596 });
597 }
598
599 template <typename Func>
600 typename std::enable_if<
601 !black_magic::CallHelper<Func, black_magic::S<Args...>>::value &&
603 void>::type
604 operator()(Func&& f)
605 {
606 static_assert(black_magic::CallHelper<Func, black_magic::S<Args...>>::value ||
608 "Handler type is mismatched with URL parameters");
609 static_assert(!std::is_same<void, decltype(f(std::declval<crow::request>(), std::declval<Args>()...))>::value,
610 "Handler function cannot have void return type; valid return types: string, int, crow::response, crow::returnable");
611
612 handler_ = (
613#ifdef CROW_CAN_USE_CPP14
614 [f = std::move(f)]
615#else
616 [f]
617#endif
618 (const crow::request& req, crow::response& res, Args ... args){
619 res = response(f(req, args...));
620 res.end();
621 });
622 }
623
624 template <typename Func>
625 typename std::enable_if<
626 !black_magic::CallHelper<Func, black_magic::S<Args...>>::value &&
627 !black_magic::CallHelper<Func, black_magic::S<crow::request, Args...>>::value &&
629 void>::type
630 operator()(Func&& f)
631 {
632 static_assert(black_magic::CallHelper<Func, black_magic::S<Args...>>::value ||
634 ,
635 "Handler type is mismatched with URL parameters");
636 static_assert(std::is_same<void, decltype(f(std::declval<crow::response&>(), std::declval<Args>()...))>::value,
637 "Handler function with response argument should have void return type");
638 handler_ = (
639#ifdef CROW_CAN_USE_CPP14
640 [f = std::move(f)]
641#else
642 [f]
643#endif
644 (const crow::request&, crow::response& res, Args ... args){
645 f(res, args...);
646 });
647 }
648
649 template <typename Func>
650 typename std::enable_if<
651 !black_magic::CallHelper<Func, black_magic::S<Args...>>::value &&
652 !black_magic::CallHelper<Func, black_magic::S<crow::request, Args...>>::value &&
653 !black_magic::CallHelper<Func, black_magic::S<crow::response&, Args...>>::value,
654 void>::type
655 operator()(Func&& f)
656 {
657 static_assert(black_magic::CallHelper<Func, black_magic::S<Args...>>::value ||
658 black_magic::CallHelper<Func, black_magic::S<crow::request, Args...>>::value ||
660 ,
661 "Handler type is mismatched with URL parameters");
662 static_assert(std::is_same<void, decltype(f(std::declval<crow::request>(), std::declval<crow::response&>(), std::declval<Args>()...))>::value,
663 "Handler function with response argument should have void return type");
664
665 handler_ = std::move(f);
666 }
667
668 template <typename Func>
669 void operator()(std::string name, Func&& f)
670 {
671 name_ = std::move(name);
672 (*this).template operator()<Func>(std::forward(f));
673 }
674
675 void handle(const request& req, response& res, const routing_params& params) override
676 {
679 decltype(handler_)>,
680 0, 0, 0, 0,
683 >()(
685 decltype(handler_)>
686 {handler_, params, req, res}
687 );
688 }
689
690 private:
691 std::function<void(const crow::request&, crow::response&, Args...)> handler_;
692
693 };
694
695 const int RULE_SPECIAL_REDIRECT_SLASH = 1;
696
697 /// A search tree.
698 class Trie
699 {
700 public:
701 struct Node
702 {
703 unsigned rule_index{};
704 std::array<unsigned, static_cast<int>(ParamType::MAX)> param_childrens{};
705 std::unordered_map<std::string, unsigned> children;
706
707 bool IsSimpleNode() const
708 {
709 return
710 !rule_index &&
711 std::all_of(
712 std::begin(param_childrens),
713 std::end(param_childrens),
714 [](unsigned x){ return !x; });
715 }
716 };
717
718 Trie() : nodes_(1)
719 {
720 }
721
722 ///Check whether or not the trie is empty.
723 bool is_empty()
724 {
725 return nodes_.size() > 1;
726 }
727
728 private:
729 void optimizeNode(Node* node)
730 {
731 for(auto x : node->param_childrens)
732 {
733 if (!x)
734 continue;
735 Node* child = &nodes_[x];
736 optimizeNode(child);
737 }
738 if (node->children.empty())
739 return;
740 bool mergeWithChild = true;
741 for(auto& kv : node->children)
742 {
743 Node* child = &nodes_[kv.second];
744 if (!child->IsSimpleNode())
745 {
746 mergeWithChild = false;
747 break;
748 }
749 }
750 if (mergeWithChild)
751 {
752 decltype(node->children) merged;
753 for(auto& kv : node->children)
754 {
755 Node* child = &nodes_[kv.second];
756 for(auto& child_kv : child->children)
757 {
758 merged[kv.first + child_kv.first] = child_kv.second;
759 }
760 }
761 node->children = std::move(merged);
762 optimizeNode(node);
763 }
764 else
765 {
766 for(auto& kv : node->children)
767 {
768 Node* child = &nodes_[kv.second];
769 optimizeNode(child);
770 }
771 }
772 }
773
774 void optimize()
775 {
776 optimizeNode(head());
777 }
778
779 public:
780 void validate()
781 {
782 if (!head()->IsSimpleNode())
783 throw std::runtime_error("Internal error: Trie header should be simple!");
784 optimize();
785 }
786
787 std::pair<unsigned, routing_params> find(const std::string& req_url, const Node* node = nullptr, unsigned pos = 0, routing_params* params = nullptr) const
788 {
789 routing_params empty;
790 if (params == nullptr)
791 params = &empty;
792
793 unsigned found{};
794 routing_params match_params;
795
796 if (node == nullptr)
797 node = head();
798 if (pos == req_url.size())
799 return {node->rule_index, *params};
800
801 auto update_found = [&found, &match_params](std::pair<unsigned, routing_params>& ret)
802 {
803 if (ret.first && (!found || found > ret.first))
804 {
805 found = ret.first;
806 match_params = std::move(ret.second);
807 }
808 };
809
810 if (node->param_childrens[static_cast<int>(ParamType::INT)])
811 {
812 char c = req_url[pos];
813 if ((c >= '0' && c <= '9') || c == '+' || c == '-')
814 {
815 char* eptr;
816 errno = 0;
817 long long int value = strtoll(req_url.data()+pos, &eptr, 10);
818 if (errno != ERANGE && eptr != req_url.data()+pos)
819 {
820 params->int_params.push_back(value);
821 auto ret = find(req_url, &nodes_[node->param_childrens[static_cast<int>(ParamType::INT)]], eptr - req_url.data(), params);
822 update_found(ret);
823 params->int_params.pop_back();
824 }
825 }
826 }
827
828 if (node->param_childrens[static_cast<int>(ParamType::UINT)])
829 {
830 char c = req_url[pos];
831 if ((c >= '0' && c <= '9') || c == '+')
832 {
833 char* eptr;
834 errno = 0;
835 unsigned long long int value = strtoull(req_url.data()+pos, &eptr, 10);
836 if (errno != ERANGE && eptr != req_url.data()+pos)
837 {
838 params->uint_params.push_back(value);
839 auto ret = find(req_url, &nodes_[node->param_childrens[static_cast<int>(ParamType::UINT)]], eptr - req_url.data(), params);
840 update_found(ret);
841 params->uint_params.pop_back();
842 }
843 }
844 }
845
846 if (node->param_childrens[static_cast<int>(ParamType::DOUBLE)])
847 {
848 char c = req_url[pos];
849 if ((c >= '0' && c <= '9') || c == '+' || c == '-' || c == '.')
850 {
851 char* eptr;
852 errno = 0;
853 double value = strtod(req_url.data()+pos, &eptr);
854 if (errno != ERANGE && eptr != req_url.data()+pos)
855 {
856 params->double_params.push_back(value);
857 auto ret = find(req_url, &nodes_[node->param_childrens[static_cast<int>(ParamType::DOUBLE)]], eptr - req_url.data(), params);
858 update_found(ret);
859 params->double_params.pop_back();
860 }
861 }
862 }
863
864 if (node->param_childrens[static_cast<int>(ParamType::STRING)])
865 {
866 size_t epos = pos;
867 for(; epos < req_url.size(); epos ++)
868 {
869 if (req_url[epos] == '/')
870 break;
871 }
872
873 if (epos != pos)
874 {
875 params->string_params.push_back(req_url.substr(pos, epos-pos));
876 auto ret = find(req_url, &nodes_[node->param_childrens[static_cast<int>(ParamType::STRING)]], epos, params);
877 update_found(ret);
878 params->string_params.pop_back();
879 }
880 }
881
882 if (node->param_childrens[static_cast<int>(ParamType::PATH)])
883 {
884 size_t epos = req_url.size();
885
886 if (epos != pos)
887 {
888 params->string_params.push_back(req_url.substr(pos, epos-pos));
889 auto ret = find(req_url, &nodes_[node->param_childrens[static_cast<int>(ParamType::PATH)]], epos, params);
890 update_found(ret);
891 params->string_params.pop_back();
892 }
893 }
894
895 for(auto& kv : node->children)
896 {
897 const std::string& fragment = kv.first;
898 const Node* child = &nodes_[kv.second];
899
900 if (req_url.compare(pos, fragment.size(), fragment) == 0)
901 {
902 auto ret = find(req_url, child, pos + fragment.size(), params);
903 update_found(ret);
904 }
905 }
906
907 return {found, match_params};
908 }
909
910 void add(const std::string& url, unsigned rule_index)
911 {
912 unsigned idx{0};
913
914 for(unsigned i = 0; i < url.size(); i ++)
915 {
916 char c = url[i];
917 if (c == '<')
918 {
919 static struct ParamTraits
920 {
921 ParamType type;
922 std::string name;
923 } paramTraits[] =
924 {
925 { ParamType::INT, "<int>" },
926 { ParamType::UINT, "<uint>" },
927 { ParamType::DOUBLE, "<float>" },
928 { ParamType::DOUBLE, "<double>" },
929 { ParamType::STRING, "<str>" },
930 { ParamType::STRING, "<string>" },
931 { ParamType::PATH, "<path>" },
932 };
933
934 for(auto& x:paramTraits)
935 {
936 if (url.compare(i, x.name.size(), x.name) == 0)
937 {
938 if (!nodes_[idx].param_childrens[static_cast<int>(x.type)])
939 {
940 auto new_node_idx = new_node();
941 nodes_[idx].param_childrens[static_cast<int>(x.type)] = new_node_idx;
942 }
943 idx = nodes_[idx].param_childrens[static_cast<int>(x.type)];
944 i += x.name.size();
945 break;
946 }
947 }
948
949 i --;
950 }
951 else
952 {
953 std::string piece(&c, 1);
954 if (!nodes_[idx].children.count(piece))
955 {
956 auto new_node_idx = new_node();
957 nodes_[idx].children.emplace(piece, new_node_idx);
958 }
959 idx = nodes_[idx].children[piece];
960 }
961 }
962 if (nodes_[idx].rule_index)
963 throw std::runtime_error("handler already exists for " + url);
964 nodes_[idx].rule_index = rule_index;
965 }
966 private:
967 void debug_node_print(Node* n, int level)
968 {
969 for(int i = 0; i < static_cast<int>(ParamType::MAX); i ++)
970 {
971 if (n->param_childrens[i])
972 {
973 CROW_LOG_DEBUG << std::string(2*level, ' ') /*<< "("<<n->param_childrens[i]<<") "*/;
974 switch(static_cast<ParamType>(i))
975 {
976 case ParamType::INT:
977 CROW_LOG_DEBUG << "<int>";
978 break;
979 case ParamType::UINT:
980 CROW_LOG_DEBUG << "<uint>";
981 break;
982 case ParamType::DOUBLE:
983 CROW_LOG_DEBUG << "<float>";
984 break;
985 case ParamType::STRING:
986 CROW_LOG_DEBUG << "<str>";
987 break;
988 case ParamType::PATH:
989 CROW_LOG_DEBUG << "<path>";
990 break;
991 default:
992 CROW_LOG_DEBUG << "<ERROR>";
993 break;
994 }
995
996 debug_node_print(&nodes_[n->param_childrens[i]], level+1);
997 }
998 }
999 for(auto& kv : n->children)
1000 {
1001 CROW_LOG_DEBUG << std::string(2*level, ' ') /*<< "(" << kv.second << ") "*/ << kv.first;
1002 debug_node_print(&nodes_[kv.second], level+1);
1003 }
1004 }
1005
1006 public:
1007 void debug_print()
1008 {
1009 debug_node_print(head(), 0);
1010 }
1011
1012 private:
1013 const Node* head() const
1014 {
1015 return &nodes_.front();
1016 }
1017
1018 Node* head()
1019 {
1020 return &nodes_.front();
1021 }
1022
1023 unsigned new_node()
1024 {
1025 nodes_.resize(nodes_.size()+1);
1026 return nodes_.size() - 1;
1027 }
1028
1029 std::vector<Node> nodes_;
1030 };
1031
1032
1033 /// Handles matching requests to existing rules and upgrade requests.
1035 {
1036 public:
1037 Router()
1038 {
1039 }
1040
1041 DynamicRule& new_rule_dynamic(const std::string& rule)
1042 {
1043 auto ruleObject = new DynamicRule(rule);
1044 all_rules_.emplace_back(ruleObject);
1045
1046 return *ruleObject;
1047 }
1048
1049 template <uint64_t N>
1050 typename black_magic::arguments<N>::type::template rebind<TaggedRule>& new_rule_tagged(const std::string& rule)
1051 {
1052 using RuleT = typename black_magic::arguments<N>::type::template rebind<TaggedRule>;
1053
1054 auto ruleObject = new RuleT(rule);
1055 all_rules_.emplace_back(ruleObject);
1056
1057 return *ruleObject;
1058 }
1059
1060 CatchallRule& catchall_rule()
1061 {
1062 return catchall_rule_;
1063 }
1064
1065 void internal_add_rule_object(const std::string& rule, BaseRule* ruleObject)
1066 {
1067 bool has_trailing_slash = false;
1068 std::string rule_without_trailing_slash;
1069 if (rule.size() > 1 && rule.back() == '/')
1070 {
1071 has_trailing_slash = true;
1072 rule_without_trailing_slash = rule;
1073 rule_without_trailing_slash.pop_back();
1074 }
1075
1076 ruleObject->foreach_method([&](int method)
1077 {
1078 per_methods_[method].rules.emplace_back(ruleObject);
1079 per_methods_[method].trie.add(rule, per_methods_[method].rules.size() - 1);
1080
1081 // directory case:
1082 // request to '/about' url matches '/about/' rule
1083 if (has_trailing_slash)
1084 {
1085 per_methods_[method].trie.add(rule_without_trailing_slash, RULE_SPECIAL_REDIRECT_SLASH);
1086 }
1087 });
1088
1089 }
1090
1091 void validate()
1092 {
1093 for(auto& rule:all_rules_)
1094 {
1095 if (rule)
1096 {
1097 auto upgraded = rule->upgrade();
1098 if (upgraded)
1099 rule = std::move(upgraded);
1100 rule->validate();
1101 internal_add_rule_object(rule->rule(), rule.get());
1102 }
1103 }
1104 for(auto& per_method:per_methods_)
1105 {
1106 per_method.trie.validate();
1107 }
1108 }
1109
1110 //TODO maybe add actual_method
1111 template <typename Adaptor>
1112 void handle_upgrade(const request& req, response& res, Adaptor&& adaptor)
1113 {
1114 if (req.method >= HTTPMethod::InternalMethodCount)
1115 return;
1116
1117 auto& per_method = per_methods_[static_cast<int>(req.method)];
1118 auto& rules = per_method.rules;
1119 unsigned rule_index = per_method.trie.find(req.url).first;
1120
1121 if (!rule_index)
1122 {
1123 for (auto& per_method: per_methods_)
1124 {
1125 if (per_method.trie.find(req.url).first)
1126 {
1127 CROW_LOG_DEBUG << "Cannot match method " << req.url << " " << method_name(req.method);
1128 res = response(405);
1129 res.end();
1130 return;
1131 }
1132 }
1133
1134 CROW_LOG_INFO << "Cannot match rules " << req.url;
1135 res = response(404);
1136 res.end();
1137 return;
1138 }
1139
1140 if (rule_index >= rules.size())
1141 throw std::runtime_error("Trie internal structure corrupted!");
1142
1143 if (rule_index == RULE_SPECIAL_REDIRECT_SLASH)
1144 {
1145 CROW_LOG_INFO << "Redirecting to a url with trailing slash: " << req.url;
1146 res = response(301);
1147
1148 // TODO absolute url building
1149 if (req.get_header_value("Host").empty())
1150 {
1151 res.add_header("Location", req.url + "/");
1152 }
1153 else
1154 {
1155 res.add_header("Location", "http://" + req.get_header_value("Host") + req.url + "/");
1156 }
1157 res.end();
1158 return;
1159 }
1160
1161 CROW_LOG_DEBUG << "Matched rule (upgrade) '" << rules[rule_index]->rule_ << "' " << static_cast<uint32_t>(req.method) << " / " << rules[rule_index]->get_methods();
1162
1163 // any uncaught exceptions become 500s
1164 try
1165 {
1166 rules[rule_index]->handle_upgrade(req, res, std::move(adaptor));
1167 }
1168 catch(std::exception& e)
1169 {
1170 CROW_LOG_ERROR << "An uncaught exception occurred: " << e.what();
1171 res = response(500);
1172 res.end();
1173 return;
1174 }
1175 catch(...)
1176 {
1177 CROW_LOG_ERROR << "An uncaught exception occurred. The type was unknown so no information was available.";
1178 res = response(500);
1179 res.end();
1180 return;
1181 }
1182 }
1183
1184 void handle(const request& req, response& res)
1185 {
1186 HTTPMethod method_actual = req.method;
1187 if (req.method >= HTTPMethod::InternalMethodCount)
1188 return;
1189 else if (req.method == HTTPMethod::Head)
1190 {
1191 method_actual = HTTPMethod::Get;
1192 res.is_head_response = true;
1193 }
1194 else if (req.method == HTTPMethod::Options)
1195 {
1196 std::string allow = "OPTIONS, HEAD, ";
1197
1198 if (req.url == "/*")
1199 {
1200 for(int i = 0; i < static_cast<int>(HTTPMethod::InternalMethodCount); i ++)
1201 {
1202 if (per_methods_[i].trie.is_empty())
1203 {
1204 allow += method_name(static_cast<HTTPMethod>(i)) + ", ";
1205 }
1206 }
1207 allow = allow.substr(0, allow.size()-2);
1208 res = response(204);
1209 res.set_header("Allow", allow);
1210 res.manual_length_header = true;
1211 res.end();
1212 return;
1213 }
1214 else
1215 {
1216 for(int i = 0; i < static_cast<int>(HTTPMethod::InternalMethodCount); i ++)
1217 {
1218 if (per_methods_[i].trie.find(req.url).first)
1219 {
1220 allow += method_name(static_cast<HTTPMethod>(i)) + ", ";
1221 }
1222 }
1223 if (allow != "OPTIONS, HEAD, ")
1224 {
1225 allow = allow.substr(0, allow.size()-2);
1226 res = response(204);
1227 res.set_header("Allow", allow);
1228 res.manual_length_header = true;
1229 res.end();
1230 return;
1231 }
1232 else
1233 {
1234 CROW_LOG_DEBUG << "Cannot match rules " << req.url;
1235 res = response(404);
1236 res.end();
1237 return;
1238 }
1239 }
1240 }
1241
1242 auto& per_method = per_methods_[static_cast<int>(method_actual)];
1243 auto& trie = per_method.trie;
1244 auto& rules = per_method.rules;
1245
1246 auto found = trie.find(req.url);
1247
1248 unsigned rule_index = found.first;
1249
1250 if (!rule_index)
1251 {
1252 for (auto& per_method: per_methods_)
1253 {
1254 if (per_method.trie.find(req.url).first)
1255 {
1256 CROW_LOG_DEBUG << "Cannot match method " << req.url << " " << method_name(method_actual);
1257 res = response(405);
1258 res.end();
1259 return;
1260 }
1261 }
1262
1263 if (catchall_rule_.has_handler())
1264 {
1265 CROW_LOG_DEBUG << "Cannot match rules " << req.url << ". Redirecting to Catchall rule";
1266 catchall_rule_.handler_(req, res);
1267 }
1268 else
1269 {
1270 CROW_LOG_DEBUG << "Cannot match rules " << req.url;
1271 res = response(404);
1272 }
1273 res.end();
1274 return;
1275 }
1276
1277 if (rule_index >= rules.size())
1278 throw std::runtime_error("Trie internal structure corrupted!");
1279
1280 if (rule_index == RULE_SPECIAL_REDIRECT_SLASH)
1281 {
1282 CROW_LOG_INFO << "Redirecting to a url with trailing slash: " << req.url;
1283 res = response(301);
1284
1285 // TODO absolute url building
1286 if (req.get_header_value("Host").empty())
1287 {
1288 res.add_header("Location", req.url + "/");
1289 }
1290 else
1291 {
1292 res.add_header("Location", "http://" + req.get_header_value("Host") + req.url + "/");
1293 }
1294 res.end();
1295 return;
1296 }
1297
1298 CROW_LOG_DEBUG << "Matched rule '" << rules[rule_index]->rule_ << "' " << static_cast<uint32_t>(req.method) << " / " << rules[rule_index]->get_methods();
1299
1300 // any uncaught exceptions become 500s
1301 try
1302 {
1303 rules[rule_index]->handle(req, res, found.second);
1304 }
1305 catch(std::exception& e)
1306 {
1307 CROW_LOG_ERROR << "An uncaught exception occurred: " << e.what();
1308 res = response(500);
1309 res.end();
1310 return;
1311 }
1312 catch(...)
1313 {
1314 CROW_LOG_ERROR << "An uncaught exception occurred. The type was unknown so no information was available.";
1315 res = response(500);
1316 res.end();
1317 return;
1318 }
1319 }
1320
1321 void debug_print()
1322 {
1323 for(int i = 0; i < static_cast<int>(HTTPMethod::InternalMethodCount); i ++)
1324 {
1325 CROW_LOG_DEBUG << method_name(static_cast<HTTPMethod>(i));
1326 per_methods_[i].trie.debug_print();
1327 }
1328 }
1329
1330 private:
1331 CatchallRule catchall_rule_;
1332
1333 struct PerMethod
1334 {
1335 std::vector<BaseRule*> rules;
1336 Trie trie;
1337
1338 // rule index 0, 1 has special meaning; preallocate it to avoid duplication.
1339 PerMethod() : rules(2) {}
1340 };
1341 std::array<PerMethod, static_cast<int>(HTTPMethod::InternalMethodCount)> per_methods_;
1342 std::vector<std::unique_ptr<BaseRule>> all_rules_;
1343
1344 };
1345}
A base class for all rules.
Definition: routing.h:25
Definition: routing.h:276
A rule that can change its parameters during runtime.
Definition: routing.h:484
Handles matching requests to existing rules and upgrade requests.
Definition: routing.h:1035
Default rule created when CROW_ROUTE is called.
Definition: routing.h:560
A search tree.
Definition: routing.h:699
bool is_empty()
Check whether or not the trie is empty.
Definition: routing.h:723
A rule dealing with websockets.
Definition: routing.h:374
A websocket connection.
Definition: websocket.h:60
Definition: routing.h:451
A wrapper for the asio::ip::tcp::socket and asio::ssl::stream.
Definition: socket_adaptors.h:19
Definition: routing.h:702
Definition: utility.h:252
Definition: utility.h:243
Definition: utility.h:306
Definition: utility.h:349
Definition: utility.h:334
An HTTP request.
Definition: http_request.h:27
std::string url
The endpoint without any parameters.
Definition: http_request.h:30
HTTP response.
Definition: http_response.h:24
void add_header(std::string key, std::string value)
Add a new header to the response.
Definition: http_response.h:46
bool is_head_response
Whether this is a response to a HEAD request.
Definition: http_response.h:35
bool manual_length_header
Whether Crow should automatically add a "Content-Length" header.
Definition: http_response.h:36
void set_header(std::string key, std::string value)
Set the value of an existing header in the response.
Definition: http_response.h:39
void end()
Set the response completion flag and call the handler (to send the response).
Definition: http_response.h:151
Definition: common.h:84
Definition: utility.h:465
A base class for websocket connection.
Definition: websocket.h:23