blob: b2355e9540cb84ea40dff2aa81714229a1019332 [file] [log] [blame]
Ed Tanous7045c8d2017-04-03 10:04:37 -07001#pragma once
2
Tanousf00032d2018-11-05 01:18:10 -03003#include "privileges.hpp"
Ratan Gupta6f359562019-04-03 10:39:08 +05304#include "sessions.hpp"
Ed Tanous1abe55e2018-09-05 08:30:59 -07005
Tanousf00032d2018-11-05 01:18:10 -03006#include <boost/container/flat_map.hpp>
7#include <boost/container/small_vector.hpp>
Ed Tanous1abe55e2018-09-05 08:30:59 -07008#include <boost/lexical_cast.hpp>
Ed Tanouse0d918b2018-03-27 17:41:04 -07009#include <cerrno>
Ed Tanous7045c8d2017-04-03 10:04:37 -070010#include <cstdint>
Ed Tanouse0d918b2018-03-27 17:41:04 -070011#include <cstdlib>
Ed Tanous3dac7492017-08-02 13:46:20 -070012#include <limits>
Ed Tanous7045c8d2017-04-03 10:04:37 -070013#include <memory>
14#include <tuple>
Ed Tanous7045c8d2017-04-03 10:04:37 -070015#include <utility>
16#include <vector>
Ed Tanous9140a672017-04-24 17:01:32 -070017
Ed Tanousc94ad492019-10-10 15:39:33 -070018#include "common.h"
19#include "http_request.h"
20#include "http_response.h"
21#include "logging.h"
22#include "utility.h"
23#include "websocket.h"
Ed Tanous7045c8d2017-04-03 10:04:37 -070024
Ed Tanous1abe55e2018-09-05 08:30:59 -070025namespace crow
26{
Tanousf00032d2018-11-05 01:18:10 -030027
28constexpr int maxHttpVerbCount =
29 static_cast<int>(boost::beast::http::verb::unlink);
30
Ed Tanous1abe55e2018-09-05 08:30:59 -070031class BaseRule
32{
33 public:
Tanousf00032d2018-11-05 01:18:10 -030034 BaseRule(std::string thisRule) : rule(std::move(thisRule))
Ed Tanous1abe55e2018-09-05 08:30:59 -070035 {
36 }
Ed Tanous7045c8d2017-04-03 10:04:37 -070037
Ed Tanous1abe55e2018-09-05 08:30:59 -070038 virtual ~BaseRule()
39 {
40 }
Ed Tanous7045c8d2017-04-03 10:04:37 -070041
Ed Tanous1abe55e2018-09-05 08:30:59 -070042 virtual void validate() = 0;
43 std::unique_ptr<BaseRule> upgrade()
44 {
45 if (ruleToUpgrade)
46 return std::move(ruleToUpgrade);
47 return {};
48 }
Ed Tanous7045c8d2017-04-03 10:04:37 -070049
Ed Tanous1abe55e2018-09-05 08:30:59 -070050 virtual void handle(const Request&, Response&, const RoutingParams&) = 0;
Ed Tanousceac6f72018-12-02 11:58:47 -080051 virtual void handleUpgrade(const Request&, Response& res,
52 boost::asio::ip::tcp::socket&&)
Ed Tanous1abe55e2018-09-05 08:30:59 -070053 {
Ed Tanousde5c9f32019-03-26 09:17:55 -070054 res.result(boost::beast::http::status::not_found);
Ed Tanous1abe55e2018-09-05 08:30:59 -070055 res.end();
56 }
Ed Tanous55c7b7a2018-05-22 15:27:24 -070057#ifdef BMCWEB_ENABLE_SSL
Ed Tanousceac6f72018-12-02 11:58:47 -080058 virtual void
59 handleUpgrade(const Request&, Response& res,
60 boost::beast::ssl_stream<boost::asio::ip::tcp::socket>&&)
Ed Tanous1abe55e2018-09-05 08:30:59 -070061 {
Ed Tanousde5c9f32019-03-26 09:17:55 -070062 res.result(boost::beast::http::status::not_found);
Ed Tanous1abe55e2018-09-05 08:30:59 -070063 res.end();
64 }
Ed Tanous7045c8d2017-04-03 10:04:37 -070065#endif
66
Ed Tanous271584a2019-07-09 16:24:22 -070067 size_t getMethods()
Ed Tanous1abe55e2018-09-05 08:30:59 -070068 {
69 return methodsBitfield;
70 }
Ed Tanous7045c8d2017-04-03 10:04:37 -070071
Tanousf00032d2018-11-05 01:18:10 -030072 bool checkPrivileges(const redfish::Privileges& userPrivileges)
73 {
74 // If there are no privileges assigned, assume no privileges
75 // required
76 if (privilegesSet.empty())
77 {
78 return true;
79 }
80
81 for (const redfish::Privileges& requiredPrivileges : privilegesSet)
82 {
83 if (userPrivileges.isSupersetOf(requiredPrivileges))
84 {
85 return true;
86 }
87 }
88 return false;
89 }
90
Ed Tanous271584a2019-07-09 16:24:22 -070091 size_t methodsBitfield{
92 1 << static_cast<size_t>(boost::beast::http::verb::get)};
Ed Tanous7045c8d2017-04-03 10:04:37 -070093
Tanousf00032d2018-11-05 01:18:10 -030094 std::vector<redfish::Privileges> privilegesSet;
95
Ed Tanous1abe55e2018-09-05 08:30:59 -070096 std::string rule;
97 std::string nameStr;
Ed Tanous7045c8d2017-04-03 10:04:37 -070098
Ed Tanous1abe55e2018-09-05 08:30:59 -070099 std::unique_ptr<BaseRule> ruleToUpgrade;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700100
Ed Tanous1abe55e2018-09-05 08:30:59 -0700101 friend class Router;
102 template <typename T> friend struct RuleParameterTraits;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700103};
104
Ed Tanous1abe55e2018-09-05 08:30:59 -0700105namespace detail
106{
107namespace routing_handler_call_helper
108{
109template <typename T, int Pos> struct CallPair
110{
111 using type = T;
112 static const int pos = Pos;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700113};
114
Ed Tanous1abe55e2018-09-05 08:30:59 -0700115template <typename H1> struct CallParams
116{
117 H1& handler;
118 const RoutingParams& params;
119 const Request& req;
120 Response& res;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700121};
122
123template <typename F, int NInt, int NUint, int NDouble, int NString,
124 typename S1, typename S2>
Ed Tanous1abe55e2018-09-05 08:30:59 -0700125struct Call
126{
127};
Ed Tanous7045c8d2017-04-03 10:04:37 -0700128
129template <typename F, int NInt, int NUint, int NDouble, int NString,
130 typename... Args1, typename... Args2>
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700131struct Call<F, NInt, NUint, NDouble, NString, black_magic::S<int64_t, Args1...>,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700132 black_magic::S<Args2...>>
133{
134 void operator()(F cparams)
135 {
136 using pushed = typename black_magic::S<Args2...>::template push_back<
137 CallPair<int64_t, NInt>>;
138 Call<F, NInt + 1, NUint, NDouble, NString, black_magic::S<Args1...>,
139 pushed>()(cparams);
140 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700141};
142
143template <typename F, int NInt, int NUint, int NDouble, int NString,
144 typename... Args1, typename... Args2>
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700145struct Call<F, NInt, NUint, NDouble, NString,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700146 black_magic::S<uint64_t, Args1...>, black_magic::S<Args2...>>
147{
148 void operator()(F cparams)
149 {
150 using pushed = typename black_magic::S<Args2...>::template push_back<
151 CallPair<uint64_t, NUint>>;
152 Call<F, NInt, NUint + 1, NDouble, NString, black_magic::S<Args1...>,
153 pushed>()(cparams);
154 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700155};
156
157template <typename F, int NInt, int NUint, int NDouble, int NString,
158 typename... Args1, typename... Args2>
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700159struct Call<F, NInt, NUint, NDouble, NString, black_magic::S<double, Args1...>,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700160 black_magic::S<Args2...>>
161{
162 void operator()(F cparams)
163 {
164 using pushed = typename black_magic::S<Args2...>::template push_back<
165 CallPair<double, NDouble>>;
166 Call<F, NInt, NUint, NDouble + 1, NString, black_magic::S<Args1...>,
167 pushed>()(cparams);
168 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700169};
170
171template <typename F, int NInt, int NUint, int NDouble, int NString,
172 typename... Args1, typename... Args2>
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700173struct Call<F, NInt, NUint, NDouble, NString,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700174 black_magic::S<std::string, Args1...>, black_magic::S<Args2...>>
175{
176 void operator()(F cparams)
177 {
178 using pushed = typename black_magic::S<Args2...>::template push_back<
179 CallPair<std::string, NString>>;
180 Call<F, NInt, NUint, NDouble, NString + 1, black_magic::S<Args1...>,
181 pushed>()(cparams);
182 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700183};
184
185template <typename F, int NInt, int NUint, int NDouble, int NString,
186 typename... Args1>
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700187struct Call<F, NInt, NUint, NDouble, NString, black_magic::S<>,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700188 black_magic::S<Args1...>>
189{
190 void operator()(F cparams)
191 {
192 cparams.handler(
193 cparams.req, cparams.res,
194 cparams.params.template get<typename Args1::type>(Args1::pos)...);
195 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700196};
197
Ed Tanous1abe55e2018-09-05 08:30:59 -0700198template <typename Func, typename... ArgsWrapped> struct Wrapped
199{
200 template <typename... Args>
201 void set(
202 Func f,
203 typename std::enable_if<
204 !std::is_same<
205 typename std::tuple_element<0, std::tuple<Args..., void>>::type,
206 const Request&>::value,
207 int>::type = 0)
208 {
Tanousf00032d2018-11-05 01:18:10 -0300209 handler = [f = std::move(f)](const Request&, Response& res,
210 Args... args) {
Ed Tanousde5c9f32019-03-26 09:17:55 -0700211 res.result(f(args...));
Tanousf00032d2018-11-05 01:18:10 -0300212 res.end();
213 };
Ed Tanous7045c8d2017-04-03 10:04:37 -0700214 }
215
Ed Tanous1abe55e2018-09-05 08:30:59 -0700216 template <typename Req, typename... Args> struct ReqHandlerWrapper
217 {
218 ReqHandlerWrapper(Func f) : f(std::move(f))
219 {
220 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700221
Ed Tanous1abe55e2018-09-05 08:30:59 -0700222 void operator()(const Request& req, Response& res, Args... args)
223 {
Ed Tanousde5c9f32019-03-26 09:17:55 -0700224 res.result(f(req, args...));
Ed Tanous1abe55e2018-09-05 08:30:59 -0700225 res.end();
226 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700227
Ed Tanous1abe55e2018-09-05 08:30:59 -0700228 Func f;
229 };
Ed Tanous7045c8d2017-04-03 10:04:37 -0700230
Ed Tanous1abe55e2018-09-05 08:30:59 -0700231 template <typename... Args>
232 void set(
233 Func f,
234 typename std::enable_if<
235 std::is_same<
236 typename std::tuple_element<0, std::tuple<Args..., void>>::type,
237 const Request&>::value &&
238 !std::is_same<typename std::tuple_element<
239 1, std::tuple<Args..., void, void>>::type,
240 Response&>::value,
241 int>::type = 0)
242 {
243 handler = ReqHandlerWrapper<Args...>(std::move(f));
244 /*handler = (
245 [f = std::move(f)]
246 (const Request& req, Response& res, Args... args){
Ed Tanousde5c9f32019-03-26 09:17:55 -0700247 res.result(f(req, args...));
Ed Tanous1abe55e2018-09-05 08:30:59 -0700248 res.end();
249 });*/
250 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700251
Ed Tanous1abe55e2018-09-05 08:30:59 -0700252 template <typename... Args>
253 void set(
254 Func f,
255 typename std::enable_if<
256 std::is_same<
257 typename std::tuple_element<0, std::tuple<Args..., void>>::type,
258 const Request&>::value &&
259 std::is_same<typename std::tuple_element<
260 1, std::tuple<Args..., void, void>>::type,
261 Response&>::value,
262 int>::type = 0)
263 {
264 handler = std::move(f);
265 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700266
Ed Tanous1abe55e2018-09-05 08:30:59 -0700267 template <typename... Args> struct HandlerTypeHelper
268 {
269 using type =
270 std::function<void(const crow::Request&, crow::Response&, Args...)>;
271 using args_type =
272 black_magic::S<typename black_magic::promote_t<Args>...>;
273 };
Ed Tanous7045c8d2017-04-03 10:04:37 -0700274
Ed Tanous1abe55e2018-09-05 08:30:59 -0700275 template <typename... Args>
276 struct HandlerTypeHelper<const Request&, Args...>
277 {
278 using type =
279 std::function<void(const crow::Request&, crow::Response&, Args...)>;
280 using args_type =
281 black_magic::S<typename black_magic::promote_t<Args>...>;
282 };
Ed Tanous7045c8d2017-04-03 10:04:37 -0700283
Ed Tanous1abe55e2018-09-05 08:30:59 -0700284 template <typename... Args>
285 struct HandlerTypeHelper<const Request&, Response&, Args...>
286 {
287 using type =
288 std::function<void(const crow::Request&, crow::Response&, Args...)>;
289 using args_type =
290 black_magic::S<typename black_magic::promote_t<Args>...>;
291 };
292
293 typename HandlerTypeHelper<ArgsWrapped...>::type handler;
294
295 void operator()(const Request& req, Response& res,
296 const RoutingParams& params)
297 {
298 detail::routing_handler_call_helper::Call<
299 detail::routing_handler_call_helper::CallParams<decltype(handler)>,
300 0, 0, 0, 0, typename HandlerTypeHelper<ArgsWrapped...>::args_type,
301 black_magic::S<>>()(
302 detail::routing_handler_call_helper::CallParams<decltype(handler)>{
303 handler, params, req, res});
304 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700305};
Ed Tanous1abe55e2018-09-05 08:30:59 -0700306} // namespace routing_handler_call_helper
307} // namespace detail
Ed Tanous7045c8d2017-04-03 10:04:37 -0700308
Ed Tanous1abe55e2018-09-05 08:30:59 -0700309class WebSocketRule : public BaseRule
310{
311 using self_t = WebSocketRule;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700312
Ed Tanous1abe55e2018-09-05 08:30:59 -0700313 public:
314 WebSocketRule(std::string rule) : BaseRule(std::move(rule))
315 {
316 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700317
Ed Tanous1abe55e2018-09-05 08:30:59 -0700318 void validate() override
319 {
320 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700321
Ed Tanous1abe55e2018-09-05 08:30:59 -0700322 void handle(const Request&, Response& res, const RoutingParams&) override
323 {
Ed Tanousde5c9f32019-03-26 09:17:55 -0700324 res.result(boost::beast::http::status::not_found);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700325 res.end();
326 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700327
Ed Tanous1abe55e2018-09-05 08:30:59 -0700328 void handleUpgrade(const Request& req, Response&,
Ed Tanousceac6f72018-12-02 11:58:47 -0800329 boost::asio::ip::tcp::socket&& adaptor) override
Ed Tanous1abe55e2018-09-05 08:30:59 -0700330 {
Ed Tanousceac6f72018-12-02 11:58:47 -0800331 new crow::websocket::ConnectionImpl<boost::asio::ip::tcp::socket>(
Ed Tanous1b0044b2018-08-03 14:30:05 -0700332 req, std::move(adaptor), openHandler, messageHandler, closeHandler,
333 errorHandler);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700334 }
335#ifdef BMCWEB_ENABLE_SSL
336 void handleUpgrade(const Request& req, Response&,
Ed Tanousceac6f72018-12-02 11:58:47 -0800337 boost::beast::ssl_stream<boost::asio::ip::tcp::socket>&&
338 adaptor) override
Ed Tanous1abe55e2018-09-05 08:30:59 -0700339 {
Ed Tanousceac6f72018-12-02 11:58:47 -0800340 std::shared_ptr<crow::websocket::ConnectionImpl<
341 boost::beast::ssl_stream<boost::asio::ip::tcp::socket>>>
342 myConnection = std::make_shared<crow::websocket::ConnectionImpl<
343 boost::beast::ssl_stream<boost::asio::ip::tcp::socket>>>(
344 req, std::move(adaptor), openHandler, messageHandler,
345 closeHandler, errorHandler);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700346 myConnection->start();
347 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700348#endif
349
Ed Tanous1abe55e2018-09-05 08:30:59 -0700350 template <typename Func> self_t& onopen(Func f)
351 {
352 openHandler = f;
353 return *this;
354 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700355
Ed Tanous1abe55e2018-09-05 08:30:59 -0700356 template <typename Func> self_t& onmessage(Func f)
357 {
358 messageHandler = f;
359 return *this;
360 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700361
Ed Tanous1abe55e2018-09-05 08:30:59 -0700362 template <typename Func> self_t& onclose(Func f)
363 {
364 closeHandler = f;
365 return *this;
366 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700367
Ed Tanous1abe55e2018-09-05 08:30:59 -0700368 template <typename Func> self_t& onerror(Func f)
369 {
370 errorHandler = f;
371 return *this;
372 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700373
Ed Tanous1abe55e2018-09-05 08:30:59 -0700374 protected:
375 std::function<void(crow::websocket::Connection&)> openHandler;
376 std::function<void(crow::websocket::Connection&, const std::string&, bool)>
377 messageHandler;
378 std::function<void(crow::websocket::Connection&, const std::string&)>
379 closeHandler;
380 std::function<void(crow::websocket::Connection&)> errorHandler;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700381};
382
Ed Tanous1abe55e2018-09-05 08:30:59 -0700383template <typename T> struct RuleParameterTraits
384{
385 using self_t = T;
386 WebSocketRule& websocket()
387 {
Ed Tanous271584a2019-07-09 16:24:22 -0700388 self_t* self = static_cast<self_t*>(this);
389 WebSocketRule* p = new WebSocketRule(self->rule);
390 self->ruleToUpgrade.reset(p);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700391 return *p;
392 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700393
Ed Tanous1abe55e2018-09-05 08:30:59 -0700394 self_t& name(std::string name) noexcept
395 {
Ed Tanous271584a2019-07-09 16:24:22 -0700396 self_t* self = static_cast<self_t*>(this);
397 self->nameStr = std::move(name);
398 return *self;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700399 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700400
Ed Tanous1abe55e2018-09-05 08:30:59 -0700401 self_t& methods(boost::beast::http::verb method)
402 {
Ed Tanous271584a2019-07-09 16:24:22 -0700403 self_t* self = static_cast<self_t*>(this);
404 self->methodsBitfield = 1U << static_cast<size_t>(method);
405 return *self;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700406 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700407
Ed Tanous1abe55e2018-09-05 08:30:59 -0700408 template <typename... MethodArgs>
409 self_t& methods(boost::beast::http::verb method, MethodArgs... args_method)
410 {
Ed Tanous271584a2019-07-09 16:24:22 -0700411 self_t* self = static_cast<self_t*>(this);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700412 methods(args_method...);
Ed Tanous271584a2019-07-09 16:24:22 -0700413 self->methodsBitfield |= 1U << static_cast<size_t>(method);
414 return *self;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700415 }
Tanousf00032d2018-11-05 01:18:10 -0300416
417 template <typename... MethodArgs>
418 self_t& requires(std::initializer_list<const char*> l)
419 {
Ed Tanous271584a2019-07-09 16:24:22 -0700420 self_t* self = static_cast<self_t*>(this);
421 self->privilegesSet.emplace_back(l);
422 return *self;
Tanousf00032d2018-11-05 01:18:10 -0300423 }
424
425 template <typename... MethodArgs>
426 self_t& requires(const std::vector<redfish::Privileges>& p)
427 {
Ed Tanous271584a2019-07-09 16:24:22 -0700428 self_t* self = static_cast<self_t*>(this);
Tanousf00032d2018-11-05 01:18:10 -0300429 for (const redfish::Privileges& privilege : p)
430 {
Ed Tanous271584a2019-07-09 16:24:22 -0700431 self->privilegesSet.emplace_back(privilege);
Tanousf00032d2018-11-05 01:18:10 -0300432 }
Ed Tanous271584a2019-07-09 16:24:22 -0700433 return *self;
Tanousf00032d2018-11-05 01:18:10 -0300434 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700435};
436
Ed Tanous1abe55e2018-09-05 08:30:59 -0700437class DynamicRule : public BaseRule, public RuleParameterTraits<DynamicRule>
438{
439 public:
440 DynamicRule(std::string rule) : BaseRule(std::move(rule))
441 {
Ed Tanous7045c8d2017-04-03 10:04:37 -0700442 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700443
Ed Tanous1abe55e2018-09-05 08:30:59 -0700444 void validate() override
445 {
446 if (!erasedHandler)
447 {
448 throw std::runtime_error(nameStr + (!nameStr.empty() ? ": " : "") +
449 "no handler for url " + rule);
450 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700451 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700452
Ed Tanous1abe55e2018-09-05 08:30:59 -0700453 void handle(const Request& req, Response& res,
454 const RoutingParams& params) override
455 {
456 erasedHandler(req, res, params);
457 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700458
Ed Tanous1abe55e2018-09-05 08:30:59 -0700459 template <typename Func> void operator()(Func f)
460 {
461 using function_t = utility::function_traits<Func>;
462
463 erasedHandler =
464 wrap(std::move(f), black_magic::gen_seq<function_t::arity>());
465 }
466
467 // enable_if Arg1 == request && Arg2 == Response
468 // enable_if Arg1 == request && Arg2 != resposne
469 // enable_if Arg1 != request
470
471 template <typename Func, unsigned... Indices>
472
473 std::function<void(const Request&, Response&, const RoutingParams&)>
474 wrap(Func f, black_magic::Seq<Indices...>)
475 {
476 using function_t = utility::function_traits<Func>;
477
478 if (!black_magic::isParameterTagCompatible(
479 black_magic::getParameterTagRuntime(rule.c_str()),
480 black_magic::compute_parameter_tag_from_args_list<
481 typename function_t::template arg<Indices>...>::value))
482 {
483 throw std::runtime_error("routeDynamic: Handler type is mismatched "
484 "with URL parameters: " +
485 rule);
486 }
487 auto ret = detail::routing_handler_call_helper::Wrapped<
488 Func, typename function_t::template arg<Indices>...>();
489 ret.template set<typename function_t::template arg<Indices>...>(
490 std::move(f));
491 return ret;
492 }
493
494 template <typename Func> void operator()(std::string name, Func&& f)
495 {
496 nameStr = std::move(name);
497 (*this).template operator()<Func>(std::forward(f));
498 }
499
500 private:
501 std::function<void(const Request&, Response&, const RoutingParams&)>
502 erasedHandler;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700503};
504
505template <typename... Args>
506class TaggedRule : public BaseRule,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700507 public RuleParameterTraits<TaggedRule<Args...>>
508{
509 public:
510 using self_t = TaggedRule<Args...>;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700511
Ed Tanous271584a2019-07-09 16:24:22 -0700512 TaggedRule(std::string ruleIn) : BaseRule(std::move(ruleIn))
Ed Tanous1abe55e2018-09-05 08:30:59 -0700513 {
Ed Tanous7045c8d2017-04-03 10:04:37 -0700514 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700515
Ed Tanous1abe55e2018-09-05 08:30:59 -0700516 void validate() override
517 {
518 if (!handler)
519 {
520 throw std::runtime_error(nameStr + (!nameStr.empty() ? ": " : "") +
521 "no handler for url " + rule);
522 }
523 }
524
525 template <typename Func>
526 typename std::enable_if<
527 black_magic::CallHelper<Func, black_magic::S<Args...>>::value,
528 void>::type
529 operator()(Func&& f)
530 {
531 static_assert(
532 black_magic::CallHelper<Func, black_magic::S<Args...>>::value ||
533 black_magic::CallHelper<
534 Func, black_magic::S<crow::Request, Args...>>::value,
535 "Handler type is mismatched with URL parameters");
536 static_assert(
537 !std::is_same<void, decltype(f(std::declval<Args>()...))>::value,
538 "Handler function cannot have void return type; valid return "
539 "types: "
540 "string, int, crow::resposne, nlohmann::json");
541
542 handler = [f = std::move(f)](const Request&, Response& res,
543 Args... args) {
Ed Tanousde5c9f32019-03-26 09:17:55 -0700544 res.result(f(args...));
Ed Tanous1abe55e2018-09-05 08:30:59 -0700545 res.end();
546 };
547 }
548
549 template <typename Func>
550 typename std::enable_if<
551 !black_magic::CallHelper<Func, black_magic::S<Args...>>::value &&
Ed Tanous7045c8d2017-04-03 10:04:37 -0700552 black_magic::CallHelper<
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700553 Func, black_magic::S<crow::Request, Args...>>::value,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700554 void>::type
555 operator()(Func&& f)
556 {
557 static_assert(
558 black_magic::CallHelper<Func, black_magic::S<Args...>>::value ||
559 black_magic::CallHelper<
560 Func, black_magic::S<crow::Request, Args...>>::value,
561 "Handler type is mismatched with URL parameters");
562 static_assert(
563 !std::is_same<void, decltype(f(std::declval<crow::Request>(),
564 std::declval<Args>()...))>::value,
565 "Handler function cannot have void return type; valid return "
566 "types: "
567 "string, int, crow::resposne,nlohmann::json");
Ed Tanous7045c8d2017-04-03 10:04:37 -0700568
Ed Tanous1abe55e2018-09-05 08:30:59 -0700569 handler = [f = std::move(f)](const crow::Request& req,
570 crow::Response& res, Args... args) {
Ed Tanousde5c9f32019-03-26 09:17:55 -0700571 res.result(f(req, args...));
Ed Tanous1abe55e2018-09-05 08:30:59 -0700572 res.end();
573 };
574 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700575
Ed Tanous1abe55e2018-09-05 08:30:59 -0700576 template <typename Func>
577 typename std::enable_if<
578 !black_magic::CallHelper<Func, black_magic::S<Args...>>::value &&
579 !black_magic::CallHelper<
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700580 Func, black_magic::S<crow::Request, Args...>>::value,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700581 void>::type
582 operator()(Func&& f)
583 {
584 static_assert(
585 black_magic::CallHelper<Func, black_magic::S<Args...>>::value ||
586 black_magic::CallHelper<
587 Func, black_magic::S<crow::Request, Args...>>::value ||
588 black_magic::CallHelper<
589 Func, black_magic::S<crow::Request, crow::Response&,
590 Args...>>::value,
591 "Handler type is mismatched with URL parameters");
592 static_assert(
593 std::is_same<void, decltype(f(std::declval<crow::Request>(),
594 std::declval<crow::Response&>(),
595 std::declval<Args>()...))>::value,
Tanousf00032d2018-11-05 01:18:10 -0300596 "Handler function with response argument should have void "
597 "return "
Ed Tanous1abe55e2018-09-05 08:30:59 -0700598 "type");
Ed Tanous7045c8d2017-04-03 10:04:37 -0700599
Ed Tanous1abe55e2018-09-05 08:30:59 -0700600 handler = std::move(f);
601 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700602
Ed Tanous1abe55e2018-09-05 08:30:59 -0700603 template <typename Func> void operator()(std::string name, Func&& f)
604 {
605 nameStr = std::move(name);
606 (*this).template operator()<Func>(std::forward(f));
607 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700608
Ed Tanous1abe55e2018-09-05 08:30:59 -0700609 void handle(const Request& req, Response& res,
610 const RoutingParams& params) override
611 {
612 detail::routing_handler_call_helper::Call<
613 detail::routing_handler_call_helper::CallParams<decltype(handler)>,
614 0, 0, 0, 0, black_magic::S<Args...>, black_magic::S<>>()(
615 detail::routing_handler_call_helper::CallParams<decltype(handler)>{
616 handler, params, req, res});
617 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700618
Ed Tanous1abe55e2018-09-05 08:30:59 -0700619 private:
620 std::function<void(const crow::Request&, crow::Response&, Args...)> handler;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700621};
622
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700623const int ruleSpecialRedirectSlash = 1;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700624
Ed Tanous1abe55e2018-09-05 08:30:59 -0700625class Trie
626{
627 public:
628 struct Node
629 {
630 unsigned ruleIndex{};
Ed Tanous271584a2019-07-09 16:24:22 -0700631 std::array<size_t, static_cast<size_t>(ParamType::MAX)>
632 paramChildrens{};
Ed Tanous1abe55e2018-09-05 08:30:59 -0700633 boost::container::flat_map<std::string, unsigned> children;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700634
Ed Tanous1abe55e2018-09-05 08:30:59 -0700635 bool isSimpleNode() const
636 {
637 return !ruleIndex && std::all_of(std::begin(paramChildrens),
638 std::end(paramChildrens),
Ed Tanous271584a2019-07-09 16:24:22 -0700639 [](size_t x) { return !x; });
Ed Tanous7045c8d2017-04-03 10:04:37 -0700640 }
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700641 };
Ed Tanous7045c8d2017-04-03 10:04:37 -0700642
Ed Tanous1abe55e2018-09-05 08:30:59 -0700643 Trie() : nodes(1)
644 {
645 }
646
647 private:
648 void optimizeNode(Node* node)
649 {
Ed Tanous271584a2019-07-09 16:24:22 -0700650 for (size_t x : node->paramChildrens)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700651 {
652 if (!x)
653 continue;
654 Node* child = &nodes[x];
655 optimizeNode(child);
Ed Tanous7045c8d2017-04-03 10:04:37 -0700656 }
Ed Tanous1abe55e2018-09-05 08:30:59 -0700657 if (node->children.empty())
658 return;
659 bool mergeWithChild = true;
Tanousf00032d2018-11-05 01:18:10 -0300660 for (const std::pair<std::string, unsigned>& kv : node->children)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700661 {
662 Node* child = &nodes[kv.second];
663 if (!child->isSimpleNode())
664 {
665 mergeWithChild = false;
666 break;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700667 }
Ed Tanous1abe55e2018-09-05 08:30:59 -0700668 }
669 if (mergeWithChild)
670 {
671 decltype(node->children) merged;
Tanousf00032d2018-11-05 01:18:10 -0300672 for (const std::pair<std::string, unsigned>& kv : node->children)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700673 {
674 Node* child = &nodes[kv.second];
Tanousf00032d2018-11-05 01:18:10 -0300675 for (const std::pair<std::string, unsigned>& childKv :
676 child->children)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700677 {
678 merged[kv.first + childKv.first] = childKv.second;
679 }
680 }
681 node->children = std::move(merged);
682 optimizeNode(node);
683 }
684 else
685 {
Tanousf00032d2018-11-05 01:18:10 -0300686 for (const std::pair<std::string, unsigned>& kv : node->children)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700687 {
688 Node* child = &nodes[kv.second];
689 optimizeNode(child);
690 }
691 }
692 }
693
694 void optimize()
695 {
696 optimizeNode(head());
697 }
698
699 public:
700 void validate()
701 {
702 if (!head()->isSimpleNode())
703 throw std::runtime_error(
704 "Internal error: Trie header should be simple!");
705 optimize();
706 }
707
708 void findRouteIndexes(const std::string& req_url,
709 std::vector<unsigned>& route_indexes,
Tanousf00032d2018-11-05 01:18:10 -0300710 const Node* node = nullptr, unsigned pos = 0) const
Ed Tanous1abe55e2018-09-05 08:30:59 -0700711 {
712 if (node == nullptr)
713 {
714 node = head();
715 }
Tanousf00032d2018-11-05 01:18:10 -0300716 for (const std::pair<std::string, unsigned>& kv : node->children)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700717 {
718 const std::string& fragment = kv.first;
719 const Node* child = &nodes[kv.second];
720 if (pos >= req_url.size())
721 {
722 if (child->ruleIndex != 0 && fragment != "/")
723 {
724 route_indexes.push_back(child->ruleIndex);
725 }
726 findRouteIndexes(req_url, route_indexes, child,
Ed Tanous271584a2019-07-09 16:24:22 -0700727 static_cast<unsigned>(pos + fragment.size()));
Ed Tanous1abe55e2018-09-05 08:30:59 -0700728 }
729 else
730 {
731 if (req_url.compare(pos, fragment.size(), fragment) == 0)
732 {
Ed Tanous271584a2019-07-09 16:24:22 -0700733 findRouteIndexes(
734 req_url, route_indexes, child,
735 static_cast<unsigned>(pos + fragment.size()));
Ed Tanous1abe55e2018-09-05 08:30:59 -0700736 }
737 }
738 }
739 }
740
741 std::pair<unsigned, RoutingParams>
Ed Tanous39e77502019-03-04 17:35:53 -0800742 find(const std::string_view req_url, const Node* node = nullptr,
Ed Tanous271584a2019-07-09 16:24:22 -0700743 size_t pos = 0, RoutingParams* params = nullptr) const
Ed Tanous1abe55e2018-09-05 08:30:59 -0700744 {
745 RoutingParams empty;
746 if (params == nullptr)
747 params = &empty;
748
749 unsigned found{};
750 RoutingParams matchParams;
751
752 if (node == nullptr)
753 node = head();
754 if (pos == req_url.size())
755 return {node->ruleIndex, *params};
756
757 auto updateFound =
758 [&found, &matchParams](std::pair<unsigned, RoutingParams>& ret) {
759 if (ret.first && (!found || found > ret.first))
760 {
761 found = ret.first;
762 matchParams = std::move(ret.second);
763 }
764 };
765
Ed Tanous271584a2019-07-09 16:24:22 -0700766 if (node->paramChildrens[static_cast<size_t>(ParamType::INT)])
Ed Tanous1abe55e2018-09-05 08:30:59 -0700767 {
768 char c = req_url[pos];
769 if ((c >= '0' && c <= '9') || c == '+' || c == '-')
770 {
771 char* eptr;
772 errno = 0;
773 long long int value =
774 std::strtoll(req_url.data() + pos, &eptr, 10);
775 if (errno != ERANGE && eptr != req_url.data() + pos)
776 {
777 params->intParams.push_back(value);
Ed Tanous271584a2019-07-09 16:24:22 -0700778 std::pair<unsigned, RoutingParams> ret = find(
779 req_url,
780 &nodes[node->paramChildrens[static_cast<size_t>(
781 ParamType::INT)]],
782 static_cast<size_t>(eptr - req_url.data()), params);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700783 updateFound(ret);
784 params->intParams.pop_back();
785 }
786 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700787 }
788
Ed Tanous271584a2019-07-09 16:24:22 -0700789 if (node->paramChildrens[static_cast<size_t>(ParamType::UINT)])
Ed Tanous1abe55e2018-09-05 08:30:59 -0700790 {
791 char c = req_url[pos];
792 if ((c >= '0' && c <= '9') || c == '+')
793 {
794 char* eptr;
795 errno = 0;
796 unsigned long long int value =
797 std::strtoull(req_url.data() + pos, &eptr, 10);
798 if (errno != ERANGE && eptr != req_url.data() + pos)
799 {
800 params->uintParams.push_back(value);
Ed Tanous271584a2019-07-09 16:24:22 -0700801 std::pair<unsigned, RoutingParams> ret = find(
802 req_url,
803 &nodes[node->paramChildrens[static_cast<size_t>(
804 ParamType::UINT)]],
805 static_cast<size_t>(eptr - req_url.data()), params);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700806 updateFound(ret);
807 params->uintParams.pop_back();
808 }
809 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700810 }
811
Ed Tanous271584a2019-07-09 16:24:22 -0700812 if (node->paramChildrens[static_cast<size_t>(ParamType::DOUBLE)])
Ed Tanous1abe55e2018-09-05 08:30:59 -0700813 {
814 char c = req_url[pos];
815 if ((c >= '0' && c <= '9') || c == '+' || c == '-' || c == '.')
816 {
817 char* eptr;
818 errno = 0;
819 double value = std::strtod(req_url.data() + pos, &eptr);
820 if (errno != ERANGE && eptr != req_url.data() + pos)
821 {
822 params->doubleParams.push_back(value);
Tanousf00032d2018-11-05 01:18:10 -0300823 std::pair<unsigned, RoutingParams> ret = find(
Ed Tanous1abe55e2018-09-05 08:30:59 -0700824 req_url,
Ed Tanous271584a2019-07-09 16:24:22 -0700825 &nodes[node->paramChildrens[static_cast<size_t>(
826 ParamType::DOUBLE)]],
827 static_cast<size_t>(eptr - req_url.data()), params);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700828 updateFound(ret);
829 params->doubleParams.pop_back();
830 }
831 }
832 }
833
Ed Tanous271584a2019-07-09 16:24:22 -0700834 if (node->paramChildrens[static_cast<size_t>(ParamType::STRING)])
Ed Tanous1abe55e2018-09-05 08:30:59 -0700835 {
Ed Tanousb01bf292019-03-25 19:25:26 +0000836 size_t epos = pos;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700837 for (; epos < req_url.size(); epos++)
838 {
839 if (req_url[epos] == '/')
840 break;
841 }
842
843 if (epos != pos)
844 {
845 params->stringParams.emplace_back(
846 req_url.substr(pos, epos - pos));
Tanousf00032d2018-11-05 01:18:10 -0300847 std::pair<unsigned, RoutingParams> ret =
Ed Tanousb01bf292019-03-25 19:25:26 +0000848 find(req_url,
Ed Tanous271584a2019-07-09 16:24:22 -0700849 &nodes[node->paramChildrens[static_cast<size_t>(
850 ParamType::STRING)]],
Ed Tanousb01bf292019-03-25 19:25:26 +0000851 epos, params);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700852 updateFound(ret);
853 params->stringParams.pop_back();
854 }
855 }
856
Ed Tanous271584a2019-07-09 16:24:22 -0700857 if (node->paramChildrens[static_cast<size_t>(ParamType::PATH)])
Ed Tanous1abe55e2018-09-05 08:30:59 -0700858 {
Ed Tanousb01bf292019-03-25 19:25:26 +0000859 size_t epos = req_url.size();
Ed Tanous1abe55e2018-09-05 08:30:59 -0700860
861 if (epos != pos)
862 {
863 params->stringParams.emplace_back(
864 req_url.substr(pos, epos - pos));
Ed Tanous271584a2019-07-09 16:24:22 -0700865 std::pair<unsigned, RoutingParams> ret =
866 find(req_url,
867 &nodes[node->paramChildrens[static_cast<size_t>(
868 ParamType::PATH)]],
869 epos, params);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700870 updateFound(ret);
871 params->stringParams.pop_back();
872 }
873 }
874
Tanousf00032d2018-11-05 01:18:10 -0300875 for (const std::pair<std::string, unsigned>& kv : node->children)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700876 {
877 const std::string& fragment = kv.first;
878 const Node* child = &nodes[kv.second];
879
880 if (req_url.compare(pos, fragment.size(), fragment) == 0)
881 {
Tanousf00032d2018-11-05 01:18:10 -0300882 std::pair<unsigned, RoutingParams> ret =
883 find(req_url, child, pos + fragment.size(), params);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700884 updateFound(ret);
885 }
886 }
887
888 return {found, matchParams};
Ed Tanous7045c8d2017-04-03 10:04:37 -0700889 }
Ed Tanous1abe55e2018-09-05 08:30:59 -0700890
891 void add(const std::string& url, unsigned ruleIndex)
892 {
Ed Tanous271584a2019-07-09 16:24:22 -0700893 size_t idx = 0;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700894
895 for (unsigned i = 0; i < url.size(); i++)
896 {
897 char c = url[i];
898 if (c == '<')
899 {
Tanousf00032d2018-11-05 01:18:10 -0300900 const static std::array<std::pair<ParamType, std::string>, 7>
901 paramTraits = {{
902 {ParamType::INT, "<int>"},
903 {ParamType::UINT, "<uint>"},
904 {ParamType::DOUBLE, "<float>"},
905 {ParamType::DOUBLE, "<double>"},
906 {ParamType::STRING, "<str>"},
907 {ParamType::STRING, "<string>"},
908 {ParamType::PATH, "<path>"},
909 }};
Ed Tanous1abe55e2018-09-05 08:30:59 -0700910
Tanousf00032d2018-11-05 01:18:10 -0300911 for (const std::pair<ParamType, std::string>& x : paramTraits)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700912 {
Tanousf00032d2018-11-05 01:18:10 -0300913 if (url.compare(i, x.second.size(), x.second) == 0)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700914 {
Ed Tanous271584a2019-07-09 16:24:22 -0700915 size_t index = static_cast<size_t>(x.first);
916 if (!nodes[idx].paramChildrens[index])
Ed Tanous1abe55e2018-09-05 08:30:59 -0700917 {
Tanousf00032d2018-11-05 01:18:10 -0300918 unsigned newNodeIdx = newNode();
Ed Tanous271584a2019-07-09 16:24:22 -0700919 nodes[idx].paramChildrens[index] = newNodeIdx;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700920 }
Ed Tanous271584a2019-07-09 16:24:22 -0700921 idx = nodes[idx].paramChildrens[index];
922 i += static_cast<unsigned>(x.second.size());
Ed Tanous1abe55e2018-09-05 08:30:59 -0700923 break;
924 }
925 }
926
927 i--;
928 }
929 else
930 {
931 std::string piece(&c, 1);
932 if (!nodes[idx].children.count(piece))
933 {
Tanousf00032d2018-11-05 01:18:10 -0300934 unsigned newNodeIdx = newNode();
Ed Tanous1abe55e2018-09-05 08:30:59 -0700935 nodes[idx].children.emplace(piece, newNodeIdx);
936 }
937 idx = nodes[idx].children[piece];
938 }
939 }
940 if (nodes[idx].ruleIndex)
941 throw std::runtime_error("handler already exists for " + url);
942 nodes[idx].ruleIndex = ruleIndex;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700943 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700944
Ed Tanous1abe55e2018-09-05 08:30:59 -0700945 private:
Ed Tanous271584a2019-07-09 16:24:22 -0700946 void debugNodePrint(Node* n, size_t level)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700947 {
Ed Tanous271584a2019-07-09 16:24:22 -0700948 for (size_t i = 0; i < static_cast<size_t>(ParamType::MAX); i++)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700949 {
950 if (n->paramChildrens[i])
951 {
952 BMCWEB_LOG_DEBUG << std::string(
Ed Tanous271584a2019-07-09 16:24:22 -0700953 2U * level, ' ') /*<< "("<<n->paramChildrens[i]<<") "*/;
954 switch (static_cast<ParamType>(i))
Ed Tanous1abe55e2018-09-05 08:30:59 -0700955 {
956 case ParamType::INT:
957 BMCWEB_LOG_DEBUG << "<int>";
958 break;
959 case ParamType::UINT:
960 BMCWEB_LOG_DEBUG << "<uint>";
961 break;
962 case ParamType::DOUBLE:
963 BMCWEB_LOG_DEBUG << "<float>";
964 break;
965 case ParamType::STRING:
966 BMCWEB_LOG_DEBUG << "<str>";
967 break;
968 case ParamType::PATH:
969 BMCWEB_LOG_DEBUG << "<path>";
970 break;
971 default:
972 BMCWEB_LOG_DEBUG << "<ERROR>";
973 break;
974 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700975
Ed Tanous1abe55e2018-09-05 08:30:59 -0700976 debugNodePrint(&nodes[n->paramChildrens[i]], level + 1);
977 }
978 }
Tanousf00032d2018-11-05 01:18:10 -0300979 for (const std::pair<std::string, unsigned>& kv : n->children)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700980 {
981 BMCWEB_LOG_DEBUG
Ed Tanous271584a2019-07-09 16:24:22 -0700982 << std::string(2U * level, ' ') /*<< "(" << kv.second << ") "*/
Ed Tanous1abe55e2018-09-05 08:30:59 -0700983 << kv.first;
984 debugNodePrint(&nodes[kv.second], level + 1);
985 }
986 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700987
Ed Tanous1abe55e2018-09-05 08:30:59 -0700988 public:
989 void debugPrint()
990 {
Ed Tanous271584a2019-07-09 16:24:22 -0700991 debugNodePrint(head(), 0U);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700992 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700993
Ed Tanous1abe55e2018-09-05 08:30:59 -0700994 private:
995 const Node* head() const
996 {
997 return &nodes.front();
998 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700999
Ed Tanous1abe55e2018-09-05 08:30:59 -07001000 Node* head()
1001 {
1002 return &nodes.front();
1003 }
1004
1005 unsigned newNode()
1006 {
1007 nodes.resize(nodes.size() + 1);
Ed Tanous271584a2019-07-09 16:24:22 -07001008 return static_cast<unsigned>(nodes.size() - 1);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001009 }
1010
1011 std::vector<Node> nodes;
Ed Tanous7045c8d2017-04-03 10:04:37 -07001012};
1013
Ed Tanous1abe55e2018-09-05 08:30:59 -07001014class Router
1015{
1016 public:
Tanousf00032d2018-11-05 01:18:10 -03001017 Router()
Ed Tanous1abe55e2018-09-05 08:30:59 -07001018 {
Ed Tanous7045c8d2017-04-03 10:04:37 -07001019 }
1020
Ed Tanous1abe55e2018-09-05 08:30:59 -07001021 DynamicRule& newRuleDynamic(const std::string& rule)
1022 {
1023 std::unique_ptr<DynamicRule> ruleObject =
1024 std::make_unique<DynamicRule>(rule);
1025 DynamicRule* ptr = ruleObject.get();
Tanousf00032d2018-11-05 01:18:10 -03001026 allRules.emplace_back(std::move(ruleObject));
Ed Tanous7045c8d2017-04-03 10:04:37 -07001027
Ed Tanous1abe55e2018-09-05 08:30:59 -07001028 return *ptr;
Ed Tanous7045c8d2017-04-03 10:04:37 -07001029 }
1030
Ed Tanous1abe55e2018-09-05 08:30:59 -07001031 template <uint64_t N>
1032 typename black_magic::Arguments<N>::type::template rebind<TaggedRule>&
1033 newRuleTagged(const std::string& rule)
1034 {
1035 using RuleT = typename black_magic::Arguments<N>::type::template rebind<
1036 TaggedRule>;
1037 std::unique_ptr<RuleT> ruleObject = std::make_unique<RuleT>(rule);
1038 RuleT* ptr = ruleObject.get();
Tanousf00032d2018-11-05 01:18:10 -03001039 allRules.emplace_back(std::move(ruleObject));
Ed Tanous1abe55e2018-09-05 08:30:59 -07001040
1041 return *ptr;
Ed Tanous7045c8d2017-04-03 10:04:37 -07001042 }
1043
Tanousf00032d2018-11-05 01:18:10 -03001044 void internalAddRuleObject(const std::string& rule, BaseRule* ruleObject)
Ed Tanous1abe55e2018-09-05 08:30:59 -07001045 {
Tanousf00032d2018-11-05 01:18:10 -03001046 if (ruleObject == nullptr)
Ed Tanous1abe55e2018-09-05 08:30:59 -07001047 {
Tanousf00032d2018-11-05 01:18:10 -03001048 return;
1049 }
1050 for (uint32_t method = 0, method_bit = 1; method < maxHttpVerbCount;
1051 method++, method_bit <<= 1)
1052 {
1053 if (ruleObject->methodsBitfield & method_bit)
1054 {
1055 perMethods[method].rules.emplace_back(ruleObject);
1056 perMethods[method].trie.add(
Ed Tanous271584a2019-07-09 16:24:22 -07001057 rule, static_cast<unsigned>(
1058 perMethods[method].rules.size() - 1U));
Tanousf00032d2018-11-05 01:18:10 -03001059 // directory case:
1060 // request to `/about' url matches `/about/' rule
1061 if (rule.size() > 2 && rule.back() == '/')
1062 {
1063 perMethods[method].trie.add(
1064 rule.substr(0, rule.size() - 1),
Ed Tanous271584a2019-07-09 16:24:22 -07001065 static_cast<unsigned>(perMethods[method].rules.size() -
1066 1));
Tanousf00032d2018-11-05 01:18:10 -03001067 }
1068 }
Ed Tanous1abe55e2018-09-05 08:30:59 -07001069 }
Ed Tanous7045c8d2017-04-03 10:04:37 -07001070 }
1071
Ed Tanous1abe55e2018-09-05 08:30:59 -07001072 void validate()
1073 {
Tanousf00032d2018-11-05 01:18:10 -03001074 for (std::unique_ptr<BaseRule>& rule : allRules)
Ed Tanous1abe55e2018-09-05 08:30:59 -07001075 {
1076 if (rule)
1077 {
Tanousf00032d2018-11-05 01:18:10 -03001078 std::unique_ptr<BaseRule> upgraded = rule->upgrade();
Ed Tanous1abe55e2018-09-05 08:30:59 -07001079 if (upgraded)
1080 rule = std::move(upgraded);
1081 rule->validate();
Tanousf00032d2018-11-05 01:18:10 -03001082 internalAddRuleObject(rule->rule, rule.get());
Ed Tanous1abe55e2018-09-05 08:30:59 -07001083 }
1084 }
Tanousf00032d2018-11-05 01:18:10 -03001085 for (PerMethod& perMethod : perMethods)
1086 {
1087 perMethod.trie.validate();
1088 }
Ed Tanous7045c8d2017-04-03 10:04:37 -07001089 }
1090
Ed Tanous1abe55e2018-09-05 08:30:59 -07001091 template <typename Adaptor>
1092 void handleUpgrade(const Request& req, Response& res, Adaptor&& adaptor)
1093 {
Ed Tanous271584a2019-07-09 16:24:22 -07001094 if (static_cast<size_t>(req.method()) >= perMethods.size())
Tanousf00032d2018-11-05 01:18:10 -03001095 return;
1096
Ed Tanous271584a2019-07-09 16:24:22 -07001097 PerMethod& perMethod = perMethods[static_cast<size_t>(req.method())];
Tanousf00032d2018-11-05 01:18:10 -03001098 Trie& trie = perMethod.trie;
1099 std::vector<BaseRule*>& rules = perMethod.rules;
1100
1101 const std::pair<unsigned, RoutingParams>& found = trie.find(req.url);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001102 unsigned ruleIndex = found.first;
1103 if (!ruleIndex)
1104 {
1105 BMCWEB_LOG_DEBUG << "Cannot match rules " << req.url;
Ed Tanousde5c9f32019-03-26 09:17:55 -07001106 res.result(boost::beast::http::status::not_found);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001107 res.end();
1108 return;
1109 }
1110
1111 if (ruleIndex >= rules.size())
1112 throw std::runtime_error("Trie internal structure corrupted!");
1113
1114 if (ruleIndex == ruleSpecialRedirectSlash)
1115 {
1116 BMCWEB_LOG_INFO << "Redirecting to a url with trailing slash: "
1117 << req.url;
Ed Tanousde5c9f32019-03-26 09:17:55 -07001118 res.result(boost::beast::http::status::moved_permanently);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001119
1120 // TODO absolute url building
1121 if (req.getHeaderValue("Host").empty())
1122 {
1123 res.addHeader("Location", std::string(req.url) + "/");
1124 }
1125 else
1126 {
1127 res.addHeader(
1128 "Location",
1129 req.isSecure
1130 ? "https://"
1131 : "http://" + std::string(req.getHeaderValue("Host")) +
1132 std::string(req.url) + "/");
1133 }
1134 res.end();
1135 return;
1136 }
1137
Ed Tanous271584a2019-07-09 16:24:22 -07001138 if ((rules[ruleIndex]->getMethods() &
1139 (1U << static_cast<size_t>(req.method()))) == 0)
Ed Tanous1abe55e2018-09-05 08:30:59 -07001140 {
1141 BMCWEB_LOG_DEBUG << "Rule found but method mismatch: " << req.url
1142 << " with " << req.methodString() << "("
Ed Tanous271584a2019-07-09 16:24:22 -07001143 << static_cast<uint32_t>(req.method()) << ") / "
Ed Tanous1abe55e2018-09-05 08:30:59 -07001144 << rules[ruleIndex]->getMethods();
Ed Tanousde5c9f32019-03-26 09:17:55 -07001145 res.result(boost::beast::http::status::not_found);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001146 res.end();
1147 return;
1148 }
1149
1150 BMCWEB_LOG_DEBUG << "Matched rule (upgrade) '" << rules[ruleIndex]->rule
Ed Tanous271584a2019-07-09 16:24:22 -07001151 << "' " << static_cast<uint32_t>(req.method()) << " / "
Ed Tanous1abe55e2018-09-05 08:30:59 -07001152 << rules[ruleIndex]->getMethods();
1153
1154 // any uncaught exceptions become 500s
1155 try
1156 {
1157 rules[ruleIndex]->handleUpgrade(req, res, std::move(adaptor));
1158 }
1159 catch (std::exception& e)
1160 {
1161 BMCWEB_LOG_ERROR << "An uncaught exception occurred: " << e.what();
Ed Tanousde5c9f32019-03-26 09:17:55 -07001162 res.result(boost::beast::http::status::internal_server_error);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001163 res.end();
1164 return;
1165 }
1166 catch (...)
1167 {
1168 BMCWEB_LOG_ERROR
1169 << "An uncaught exception occurred. The type was unknown "
1170 "so no information was available.";
Ed Tanousde5c9f32019-03-26 09:17:55 -07001171 res.result(boost::beast::http::status::internal_server_error);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001172 res.end();
1173 return;
1174 }
Ed Tanous7045c8d2017-04-03 10:04:37 -07001175 }
1176
Ed Tanous1abe55e2018-09-05 08:30:59 -07001177 void handle(const Request& req, Response& res)
1178 {
Ed Tanous271584a2019-07-09 16:24:22 -07001179 if (static_cast<size_t>(req.method()) >= perMethods.size())
Tanousf00032d2018-11-05 01:18:10 -03001180 return;
Ed Tanous271584a2019-07-09 16:24:22 -07001181 PerMethod& perMethod = perMethods[static_cast<size_t>(req.method())];
Tanousf00032d2018-11-05 01:18:10 -03001182 Trie& trie = perMethod.trie;
1183 std::vector<BaseRule*>& rules = perMethod.rules;
1184
1185 const std::pair<unsigned, RoutingParams>& found = trie.find(req.url);
Ed Tanous7045c8d2017-04-03 10:04:37 -07001186
Ed Tanous1abe55e2018-09-05 08:30:59 -07001187 unsigned ruleIndex = found.first;
1188
1189 if (!ruleIndex)
1190 {
Ed Tanous2634dcd2019-03-26 09:28:06 -07001191 // Check to see if this url exists at any verb
1192 for (const PerMethod& p : perMethods)
1193 {
1194 const std::pair<unsigned, RoutingParams>& found =
1195 p.trie.find(req.url);
1196 if (found.first > 0)
1197 {
1198 res.result(boost::beast::http::status::method_not_allowed);
1199 res.end();
1200 return;
1201 }
1202 }
Ed Tanous1abe55e2018-09-05 08:30:59 -07001203 BMCWEB_LOG_DEBUG << "Cannot match rules " << req.url;
1204 res.result(boost::beast::http::status::not_found);
1205 res.end();
1206 return;
1207 }
1208
1209 if (ruleIndex >= rules.size())
1210 throw std::runtime_error("Trie internal structure corrupted!");
1211
1212 if (ruleIndex == ruleSpecialRedirectSlash)
1213 {
1214 BMCWEB_LOG_INFO << "Redirecting to a url with trailing slash: "
1215 << req.url;
Ed Tanousde5c9f32019-03-26 09:17:55 -07001216 res.result(boost::beast::http::status::moved_permanently);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001217
1218 // TODO absolute url building
1219 if (req.getHeaderValue("Host").empty())
1220 {
1221 res.addHeader("Location", std::string(req.url) + "/");
1222 }
1223 else
1224 {
1225 res.addHeader("Location",
1226 (req.isSecure ? "https://" : "http://") +
1227 std::string(req.getHeaderValue("Host")) +
1228 std::string(req.url) + "/");
1229 }
1230 res.end();
1231 return;
1232 }
1233
Ed Tanous271584a2019-07-09 16:24:22 -07001234 if ((rules[ruleIndex]->getMethods() &
1235 (1U << static_cast<uint32_t>(req.method()))) == 0)
Ed Tanous1abe55e2018-09-05 08:30:59 -07001236 {
1237 BMCWEB_LOG_DEBUG << "Rule found but method mismatch: " << req.url
1238 << " with " << req.methodString() << "("
Ed Tanous271584a2019-07-09 16:24:22 -07001239 << static_cast<uint32_t>(req.method()) << ") / "
Ed Tanous1abe55e2018-09-05 08:30:59 -07001240 << rules[ruleIndex]->getMethods();
Ed Tanousde5c9f32019-03-26 09:17:55 -07001241 res.result(boost::beast::http::status::method_not_allowed);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001242 res.end();
1243 return;
1244 }
1245
1246 BMCWEB_LOG_DEBUG << "Matched rule '" << rules[ruleIndex]->rule << "' "
Ed Tanous271584a2019-07-09 16:24:22 -07001247 << static_cast<uint32_t>(req.method()) << " / "
Ed Tanous1abe55e2018-09-05 08:30:59 -07001248 << rules[ruleIndex]->getMethods();
1249
Ratan Gupta6f359562019-04-03 10:39:08 +05301250 redfish::Privileges userPrivileges;
1251 if (req.session != nullptr)
1252 {
1253 // Get the user role from the session.
Ed Tanousca0c93b2019-09-19 11:53:50 -07001254 const std::string& userRole =
1255 persistent_data::UserRoleMap::getInstance().getUserRole(
1256 req.session->username);
Ratan Gupta6f359562019-04-03 10:39:08 +05301257
1258 BMCWEB_LOG_DEBUG << "USER ROLE=" << userRole;
1259
1260 // Get the user privileges from the role
1261 userPrivileges = redfish::getUserPrivileges(userRole);
1262 }
Tanousf00032d2018-11-05 01:18:10 -03001263
1264 if (!rules[ruleIndex]->checkPrivileges(userPrivileges))
1265 {
Gunnar Mills961c9d92019-06-18 07:39:22 -05001266 res.result(boost::beast::http::status::forbidden);
Tanousf00032d2018-11-05 01:18:10 -03001267 res.end();
1268 return;
1269 }
1270
Ed Tanous1abe55e2018-09-05 08:30:59 -07001271 // any uncaught exceptions become 500s
1272 try
1273 {
1274 rules[ruleIndex]->handle(req, res, found.second);
1275 }
1276 catch (std::exception& e)
1277 {
1278 BMCWEB_LOG_ERROR << "An uncaught exception occurred: " << e.what();
Ed Tanousde5c9f32019-03-26 09:17:55 -07001279 res.result(boost::beast::http::status::internal_server_error);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001280 res.end();
1281 return;
1282 }
1283 catch (...)
1284 {
1285 BMCWEB_LOG_ERROR
1286 << "An uncaught exception occurred. The type was unknown "
1287 "so no information was available.";
Ed Tanousde5c9f32019-03-26 09:17:55 -07001288 res.result(boost::beast::http::status::internal_server_error);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001289 res.end();
1290 return;
1291 }
Ed Tanous7045c8d2017-04-03 10:04:37 -07001292 }
Ed Tanous7045c8d2017-04-03 10:04:37 -07001293
Ed Tanous1abe55e2018-09-05 08:30:59 -07001294 void debugPrint()
1295 {
Ed Tanous271584a2019-07-09 16:24:22 -07001296 for (size_t i = 0; i < perMethods.size(); i++)
Tanousf00032d2018-11-05 01:18:10 -03001297 {
Ed Tanous271584a2019-07-09 16:24:22 -07001298 BMCWEB_LOG_DEBUG
1299 << methodName(static_cast<boost::beast::http::verb>(i));
Tanousf00032d2018-11-05 01:18:10 -03001300 perMethods[i].trie.debugPrint();
1301 }
Ed Tanous3dac7492017-08-02 13:46:20 -07001302 }
Ed Tanousb4a7bfa2017-04-04 17:23:00 -07001303
Ed Tanous1abe55e2018-09-05 08:30:59 -07001304 std::vector<const std::string*> getRoutes(const std::string& parent)
1305 {
Ed Tanous1abe55e2018-09-05 08:30:59 -07001306 std::vector<const std::string*> ret;
Tanousf00032d2018-11-05 01:18:10 -03001307
1308 for (const PerMethod& pm : perMethods)
Ed Tanous1abe55e2018-09-05 08:30:59 -07001309 {
Tanousf00032d2018-11-05 01:18:10 -03001310 std::vector<unsigned> x;
1311 pm.trie.findRouteIndexes(parent, x);
1312 for (unsigned index : x)
1313 {
1314 ret.push_back(&pm.rules[index]->rule);
1315 }
Ed Tanous1abe55e2018-09-05 08:30:59 -07001316 }
1317 return ret;
1318 }
1319
1320 private:
Tanousf00032d2018-11-05 01:18:10 -03001321 struct PerMethod
1322 {
1323 std::vector<BaseRule*> rules;
1324 Trie trie;
1325 // rule index 0, 1 has special meaning; preallocate it to avoid
1326 // duplication.
1327 PerMethod() : rules(2)
1328 {
1329 }
1330 };
1331 std::array<PerMethod, maxHttpVerbCount> perMethods;
1332 std::vector<std::unique_ptr<BaseRule>> allRules;
Ed Tanous7045c8d2017-04-03 10:04:37 -07001333};
Ed Tanous1abe55e2018-09-05 08:30:59 -07001334} // namespace crow