6#include <unordered_map>
8#include <boost/lexical_cast.hpp>
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"
28 : rule_(std::move(rule))
36 virtual void validate() = 0;
37 std::unique_ptr<BaseRule> upgrade()
40 return std::move(rule_to_upgrade_);
51 virtual void handle_upgrade(
const request&,
response& res, SSLAdaptor&&)
58 uint32_t get_methods()
64 void foreach_method(F f)
66 for(uint32_t method = 0, method_bit = 1; method < static_cast<uint32_t>(HTTPMethod::InternalMethodCount); method++, method_bit<<=1)
68 if (methods_ & method_bit)
73 const std::string& rule() {
return rule_; }
76 uint32_t methods_{1<<
static_cast<int>(HTTPMethod::Get)};
81 std::unique_ptr<BaseRule> rule_to_upgrade_;
91 namespace routing_handler_call_helper
93 template <
typename T,
int Pos>
97 static const int pos = Pos;
100 template <
typename H1>
109 template <
typename F,
int NInt,
int NU
int,
int NDouble,
int NString,
typename S1,
typename S2>
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...>>
117 void operator()(F cparams)
119 using pushed =
typename black_magic::S<Args2...>::template push_back<call_pair<int64_t, NInt>>;
120 call<F, NInt+1, NUint, NDouble, NString,
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...>>
128 void operator()(F cparams)
130 using pushed =
typename black_magic::S<Args2...>::template push_back<call_pair<uint64_t, NUint>>;
131 call<F, NInt, NUint+1, NDouble, NString,
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...>>
139 void operator()(F cparams)
141 using pushed =
typename black_magic::S<Args2...>::template push_back<call_pair<double, NDouble>>;
142 call<F, NInt, NUint, NDouble+1, NString,
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...>>
150 void operator()(F cparams)
152 using pushed =
typename black_magic::S<Args2...>::template push_back<call_pair<std::string, NString>>;
153 call<F, NInt, NUint, NDouble, NString+1,
158 template <
typename F,
int NInt,
int NUint,
int NDouble,
int NString,
typename ... Args1>
161 void operator()(F cparams)
166 cparams.params.template get<typename Args1::type>(Args1::pos)...
171 template <
typename Func,
typename ... ArgsWrapped>
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
180#ifdef CROW_CAN_USE_CPP14
191 template <
typename Req,
typename ... Args>
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
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
229 handler_ = std::move(f);
232 template <
typename ... Args>
239 template <
typename ... Args>
246 template <
typename ... Args>
266 {handler_, params, req, res}
280 template <
typename Func>
281 typename std::enable_if<black_magic::CallHelper<Func, black_magic::S<>>::value,
void>::type
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");
288#ifdef CROW_CAN_USE_CPP14
300 template <
typename Func>
301 typename std::enable_if<
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");
311#ifdef CROW_CAN_USE_CPP14
322 template <
typename Func>
323 typename std::enable_if<
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");
333#ifdef CROW_CAN_USE_CPP14
343 template <
typename Func>
344 typename std::enable_if<
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");
354 handler_ = std::move(f);
359 return (handler_ !=
nullptr);
382 void validate()
override
396#ifdef CROW_ENABLE_SSL
397 void handle_upgrade(
const request& req,
response&, SSLAdaptor&& adaptor)
override
403 template <
typename Func>
410 template <
typename Func>
413 message_handler_ = f;
417 template <
typename Func>
424 template <
typename Func>
431 template <
typename Func>
449 template <
typename T>
455 auto p =
new WebSocketRule(
static_cast<self_t*
>(
this)->rule_);
456 static_cast<self_t*
>(
this)->rule_to_upgrade_.reset(p);
460 self_t& name(std::string name)
noexcept
462 static_cast<self_t*
>(
this)->name_ = std::move(name);
463 return static_cast<self_t&
>(*this);
466 self_t& methods(HTTPMethod method)
468 static_cast<self_t*
>(
this)->methods_ = 1 <<
static_cast<int>(method);
469 return static_cast<self_t&
>(*this);
472 template <
typename ... MethodArgs>
473 self_t& methods(HTTPMethod method, MethodArgs ... args_method)
475 methods(args_method...);
476 static_cast<self_t*
>(
this)->methods_ |= 1 <<
static_cast<int>(method);
477 return static_cast<self_t&
>(*this);
492 void validate()
override
494 if (!erased_handler_)
496 throw std::runtime_error(name_ + (!name_.empty() ?
": " :
"") +
"no handler for url " + rule_);
502 erased_handler_(req, res, params);
505 template <
typename Func>
506 void operator()(Func f)
508#ifdef CROW_MSVC_WORKAROUND
519#ifdef CROW_MSVC_WORKAROUND
520 template <
typename Func,
size_t ... Indices>
522 template <
typename Func,
unsigned ... Indices>
527#ifdef CROW_MSVC_WORKAROUND
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))
537 throw std::runtime_error(
"route_dynamic: Handler type is mismatched with URL parameters: " + rule_);
541 typename function_t::template arg<Indices>...
546 template <
typename Func>
547 void operator()(std::string name, Func&& f)
549 name_ = std::move(name);
550 (*this).template operator()<Func>(std::forward(f));
558 template <
typename ... Args>
569 void validate()
override
573 throw std::runtime_error(name_ + (!name_.empty() ?
": " :
"") +
"no handler for url " + rule_);
577 template <
typename Func>
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");
588#ifdef CROW_CAN_USE_CPP14
599 template <
typename Func>
600 typename std::enable_if<
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");
613#ifdef CROW_CAN_USE_CPP14
624 template <
typename Func>
625 typename std::enable_if<
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");
639#ifdef CROW_CAN_USE_CPP14
649 template <
typename Func>
650 typename std::enable_if<
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");
665 handler_ = std::move(f);
668 template <
typename Func>
669 void operator()(std::string name, Func&& f)
671 name_ = std::move(name);
672 (*this).template operator()<Func>(std::forward(f));
686 {handler_, params, req, res}
695 const int RULE_SPECIAL_REDIRECT_SLASH = 1;
703 unsigned rule_index{};
704 std::array<unsigned, static_cast<int>(ParamType::MAX)> param_childrens{};
705 std::unordered_map<std::string, unsigned> children;
707 bool IsSimpleNode()
const
712 std::begin(param_childrens),
713 std::end(param_childrens),
714 [](
unsigned x){
return !x; });
725 return nodes_.size() > 1;
729 void optimizeNode(Node* node)
731 for(
auto x : node->param_childrens)
735 Node* child = &nodes_[x];
738 if (node->children.empty())
740 bool mergeWithChild =
true;
741 for(
auto& kv : node->children)
743 Node* child = &nodes_[kv.second];
744 if (!child->IsSimpleNode())
746 mergeWithChild =
false;
752 decltype(node->children) merged;
753 for(
auto& kv : node->children)
755 Node* child = &nodes_[kv.second];
756 for(
auto& child_kv : child->children)
758 merged[kv.first + child_kv.first] = child_kv.second;
761 node->children = std::move(merged);
766 for(
auto& kv : node->children)
768 Node* child = &nodes_[kv.second];
776 optimizeNode(head());
782 if (!head()->IsSimpleNode())
783 throw std::runtime_error(
"Internal error: Trie header should be simple!");
787 std::pair<unsigned, routing_params> find(
const std::string& req_url,
const Node* node =
nullptr,
unsigned pos = 0, routing_params* params =
nullptr)
const
789 routing_params empty;
790 if (params ==
nullptr)
794 routing_params match_params;
798 if (pos == req_url.size())
799 return {node->rule_index, *params};
801 auto update_found = [&found, &match_params](std::pair<unsigned, routing_params>& ret)
803 if (ret.first && (!found || found > ret.first))
806 match_params = std::move(ret.second);
810 if (node->param_childrens[
static_cast<int>(ParamType::INT)])
812 char c = req_url[pos];
813 if ((c >=
'0' && c <=
'9') || c ==
'+' || c ==
'-')
817 long long int value = strtoll(req_url.data()+pos, &eptr, 10);
818 if (errno != ERANGE && eptr != req_url.data()+pos)
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);
823 params->int_params.pop_back();
828 if (node->param_childrens[
static_cast<int>(ParamType::UINT)])
830 char c = req_url[pos];
831 if ((c >=
'0' && c <=
'9') || c ==
'+')
835 unsigned long long int value = strtoull(req_url.data()+pos, &eptr, 10);
836 if (errno != ERANGE && eptr != req_url.data()+pos)
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);
841 params->uint_params.pop_back();
846 if (node->param_childrens[
static_cast<int>(ParamType::DOUBLE)])
848 char c = req_url[pos];
849 if ((c >=
'0' && c <=
'9') || c ==
'+' || c ==
'-' || c ==
'.')
853 double value = strtod(req_url.data()+pos, &eptr);
854 if (errno != ERANGE && eptr != req_url.data()+pos)
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);
859 params->double_params.pop_back();
864 if (node->param_childrens[
static_cast<int>(ParamType::STRING)])
867 for(; epos < req_url.size(); epos ++)
869 if (req_url[epos] ==
'/')
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);
878 params->string_params.pop_back();
882 if (node->param_childrens[
static_cast<int>(ParamType::PATH)])
884 size_t epos = req_url.size();
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);
891 params->string_params.pop_back();
895 for(
auto& kv : node->children)
897 const std::string& fragment = kv.first;
898 const Node* child = &nodes_[kv.second];
900 if (req_url.compare(pos, fragment.size(), fragment) == 0)
902 auto ret = find(req_url, child, pos + fragment.size(), params);
907 return {found, match_params};
910 void add(
const std::string& url,
unsigned rule_index)
914 for(
unsigned i = 0; i < url.size(); i ++)
919 static struct ParamTraits
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>" },
934 for(
auto& x:paramTraits)
936 if (url.compare(i, x.name.size(), x.name) == 0)
938 if (!nodes_[idx].param_childrens[
static_cast<int>(x.type)])
940 auto new_node_idx = new_node();
941 nodes_[idx].param_childrens[
static_cast<int>(x.type)] = new_node_idx;
943 idx = nodes_[idx].param_childrens[
static_cast<int>(x.type)];
953 std::string piece(&c, 1);
954 if (!nodes_[idx].children.count(piece))
956 auto new_node_idx = new_node();
957 nodes_[idx].children.emplace(piece, new_node_idx);
959 idx = nodes_[idx].children[piece];
962 if (nodes_[idx].rule_index)
963 throw std::runtime_error(
"handler already exists for " + url);
964 nodes_[idx].rule_index = rule_index;
967 void debug_node_print(Node* n,
int level)
969 for(
int i = 0; i < static_cast<int>(ParamType::MAX); i ++)
971 if (n->param_childrens[i])
973 CROW_LOG_DEBUG << std::string(2*level,
' ') ;
974 switch(
static_cast<ParamType
>(i))
977 CROW_LOG_DEBUG <<
"<int>";
979 case ParamType::UINT:
980 CROW_LOG_DEBUG <<
"<uint>";
982 case ParamType::DOUBLE:
983 CROW_LOG_DEBUG <<
"<float>";
985 case ParamType::STRING:
986 CROW_LOG_DEBUG <<
"<str>";
988 case ParamType::PATH:
989 CROW_LOG_DEBUG <<
"<path>";
992 CROW_LOG_DEBUG <<
"<ERROR>";
996 debug_node_print(&nodes_[n->param_childrens[i]], level+1);
999 for(
auto& kv : n->children)
1001 CROW_LOG_DEBUG << std::string(2*level,
' ') << kv.first;
1002 debug_node_print(&nodes_[kv.second], level+1);
1009 debug_node_print(head(), 0);
1013 const Node* head()
const
1015 return &nodes_.front();
1020 return &nodes_.front();
1025 nodes_.resize(nodes_.size()+1);
1026 return nodes_.size() - 1;
1029 std::vector<Node> nodes_;
1041 DynamicRule& new_rule_dynamic(
const std::string& rule)
1044 all_rules_.emplace_back(ruleObject);
1049 template <u
int64_t N>
1054 auto ruleObject =
new RuleT(rule);
1055 all_rules_.emplace_back(ruleObject);
1062 return catchall_rule_;
1065 void internal_add_rule_object(
const std::string& rule,
BaseRule* ruleObject)
1067 bool has_trailing_slash =
false;
1068 std::string rule_without_trailing_slash;
1069 if (rule.size() > 1 && rule.back() ==
'/')
1071 has_trailing_slash =
true;
1072 rule_without_trailing_slash = rule;
1073 rule_without_trailing_slash.pop_back();
1076 ruleObject->foreach_method([&](
int method)
1078 per_methods_[method].rules.emplace_back(ruleObject);
1079 per_methods_[method].trie.add(rule, per_methods_[method].rules.size() - 1);
1083 if (has_trailing_slash)
1085 per_methods_[method].trie.add(rule_without_trailing_slash, RULE_SPECIAL_REDIRECT_SLASH);
1093 for(
auto& rule:all_rules_)
1097 auto upgraded = rule->upgrade();
1099 rule = std::move(upgraded);
1101 internal_add_rule_object(rule->rule(), rule.get());
1104 for(
auto& per_method:per_methods_)
1106 per_method.trie.validate();
1111 template <
typename Adaptor>
1112 void handle_upgrade(
const request& req,
response& res, Adaptor&& adaptor)
1114 if (req.method >= HTTPMethod::InternalMethodCount)
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;
1123 for (
auto& per_method: per_methods_)
1125 if (per_method.trie.find(req.
url).first)
1127 CROW_LOG_DEBUG <<
"Cannot match method " << req.
url <<
" " << method_name(req.method);
1134 CROW_LOG_INFO <<
"Cannot match rules " << req.
url;
1140 if (rule_index >= rules.size())
1141 throw std::runtime_error(
"Trie internal structure corrupted!");
1143 if (rule_index == RULE_SPECIAL_REDIRECT_SLASH)
1145 CROW_LOG_INFO <<
"Redirecting to a url with trailing slash: " << req.
url;
1149 if (req.get_header_value(
"Host").empty())
1155 res.
add_header(
"Location",
"http://" + req.get_header_value(
"Host") + req.
url +
"/");
1161 CROW_LOG_DEBUG <<
"Matched rule (upgrade) '" << rules[rule_index]->rule_ <<
"' " <<
static_cast<uint32_t
>(req.method) <<
" / " << rules[rule_index]->get_methods();
1166 rules[rule_index]->handle_upgrade(req, res, std::move(adaptor));
1168 catch(std::exception& e)
1170 CROW_LOG_ERROR <<
"An uncaught exception occurred: " << e.what();
1177 CROW_LOG_ERROR <<
"An uncaught exception occurred. The type was unknown so no information was available.";
1186 HTTPMethod method_actual = req.method;
1187 if (req.method >= HTTPMethod::InternalMethodCount)
1189 else if (req.method == HTTPMethod::Head)
1191 method_actual = HTTPMethod::Get;
1194 else if (req.method == HTTPMethod::Options)
1196 std::string allow =
"OPTIONS, HEAD, ";
1198 if (req.
url ==
"/*")
1200 for(
int i = 0; i < static_cast<int>(HTTPMethod::InternalMethodCount); i ++)
1202 if (per_methods_[i].trie.is_empty())
1204 allow += method_name(
static_cast<HTTPMethod
>(i)) +
", ";
1207 allow = allow.substr(0, allow.size()-2);
1216 for(
int i = 0; i < static_cast<int>(HTTPMethod::InternalMethodCount); i ++)
1218 if (per_methods_[i].trie.find(req.
url).first)
1220 allow += method_name(
static_cast<HTTPMethod
>(i)) +
", ";
1223 if (allow !=
"OPTIONS, HEAD, ")
1225 allow = allow.substr(0, allow.size()-2);
1234 CROW_LOG_DEBUG <<
"Cannot match rules " << req.
url;
1242 auto& per_method = per_methods_[
static_cast<int>(method_actual)];
1243 auto& trie = per_method.trie;
1244 auto& rules = per_method.rules;
1246 auto found = trie.find(req.
url);
1248 unsigned rule_index = found.first;
1252 for (
auto& per_method: per_methods_)
1254 if (per_method.trie.find(req.
url).first)
1256 CROW_LOG_DEBUG <<
"Cannot match method " << req.
url <<
" " << method_name(method_actual);
1263 if (catchall_rule_.has_handler())
1265 CROW_LOG_DEBUG <<
"Cannot match rules " << req.
url <<
". Redirecting to Catchall rule";
1266 catchall_rule_.handler_(req, res);
1270 CROW_LOG_DEBUG <<
"Cannot match rules " << req.
url;
1277 if (rule_index >= rules.size())
1278 throw std::runtime_error(
"Trie internal structure corrupted!");
1280 if (rule_index == RULE_SPECIAL_REDIRECT_SLASH)
1282 CROW_LOG_INFO <<
"Redirecting to a url with trailing slash: " << req.
url;
1286 if (req.get_header_value(
"Host").empty())
1292 res.
add_header(
"Location",
"http://" + req.get_header_value(
"Host") + req.
url +
"/");
1298 CROW_LOG_DEBUG <<
"Matched rule '" << rules[rule_index]->rule_ <<
"' " <<
static_cast<uint32_t
>(req.method) <<
" / " << rules[rule_index]->get_methods();
1303 rules[rule_index]->handle(req, res, found.second);
1305 catch(std::exception& e)
1307 CROW_LOG_ERROR <<
"An uncaught exception occurred: " << e.what();
1314 CROW_LOG_ERROR <<
"An uncaught exception occurred. The type was unknown so no information was available.";
1323 for(
int i = 0; i < static_cast<int>(HTTPMethod::InternalMethodCount); i ++)
1325 CROW_LOG_DEBUG << method_name(static_cast<HTTPMethod>(i));
1326 per_methods_[i].trie.debug_print();
1335 std::vector<BaseRule*> rules;
1339 PerMethod() : rules(2) {}
1341 std::array<PerMethod, static_cast<int>(HTTPMethod::InternalMethodCount)> per_methods_;
1342 std::vector<std::unique_ptr<BaseRule>> all_rules_;
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:153
Definition: utility.h:349
Definition: utility.h:334
Definition: routing.h:234
Definition: routing.h:193
Definition: routing.h:173
Definition: routing.h:102
Definition: routing.h:111
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: utility.h:465
A base class for websocket connection.
Definition: websocket.h:23