blob: 2302658607efb1eb48cb837d54c034bbb407bfa4 [file] [log] [blame]
Ed Tanous7045c8d2017-04-03 10:04:37 -07001#pragma once
2
Gunnar Mills1214b7e2020-06-04 10:11:30 -05003#include "common.h"
4#include "http_request.h"
5#include "http_response.h"
6#include "logging.h"
7#include "utility.h"
8#include "websocket.h"
9
Joseph Reynolds3bf4e632020-02-06 14:44:32 -060010#include "error_messages.hpp"
Tanousf00032d2018-11-05 01:18:10 -030011#include "privileges.hpp"
Ratan Gupta6f359562019-04-03 10:39:08 +053012#include "sessions.hpp"
Ed Tanous1abe55e2018-09-05 08:30:59 -070013
Iwona Klimaszewskac0a1c8a2019-07-12 18:26:38 +020014#include <async_resp.hpp>
Tanousf00032d2018-11-05 01:18:10 -030015#include <boost/container/flat_map.hpp>
16#include <boost/container/small_vector.hpp>
Ed Tanous1abe55e2018-09-05 08:30:59 -070017#include <boost/lexical_cast.hpp>
Gunnar Mills1214b7e2020-06-04 10:11:30 -050018
Ed Tanouse0d918b2018-03-27 17:41:04 -070019#include <cerrno>
Ed Tanous7045c8d2017-04-03 10:04:37 -070020#include <cstdint>
Ed Tanouse0d918b2018-03-27 17:41:04 -070021#include <cstdlib>
Ed Tanous3dac7492017-08-02 13:46:20 -070022#include <limits>
Ed Tanous7045c8d2017-04-03 10:04:37 -070023#include <memory>
24#include <tuple>
Ed Tanous7045c8d2017-04-03 10:04:37 -070025#include <utility>
26#include <vector>
Ed Tanous9140a672017-04-24 17:01:32 -070027
Ed Tanous1abe55e2018-09-05 08:30:59 -070028namespace crow
29{
Tanousf00032d2018-11-05 01:18:10 -030030
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))
Gunnar Mills1214b7e2020-06-04 10:11:30 -050035 {}
Ed Tanous7045c8d2017-04-03 10:04:37 -070036
Ed Tanous0c0084a2019-10-24 15:57:51 -070037 virtual ~BaseRule() = default;
Ed Tanous7045c8d2017-04-03 10:04:37 -070038
Ed Tanous1abe55e2018-09-05 08:30:59 -070039 virtual void validate() = 0;
40 std::unique_ptr<BaseRule> upgrade()
41 {
42 if (ruleToUpgrade)
43 return std::move(ruleToUpgrade);
44 return {};
45 }
Ed Tanous7045c8d2017-04-03 10:04:37 -070046
Ed Tanous1abe55e2018-09-05 08:30:59 -070047 virtual void handle(const Request&, Response&, const RoutingParams&) = 0;
Ed Tanousceac6f72018-12-02 11:58:47 -080048 virtual void handleUpgrade(const Request&, Response& res,
49 boost::asio::ip::tcp::socket&&)
Ed Tanous1abe55e2018-09-05 08:30:59 -070050 {
Ed Tanousde5c9f32019-03-26 09:17:55 -070051 res.result(boost::beast::http::status::not_found);
Ed Tanous1abe55e2018-09-05 08:30:59 -070052 res.end();
53 }
Ed Tanous55c7b7a2018-05-22 15:27:24 -070054#ifdef BMCWEB_ENABLE_SSL
Ed Tanousceac6f72018-12-02 11:58:47 -080055 virtual void
56 handleUpgrade(const Request&, Response& res,
57 boost::beast::ssl_stream<boost::asio::ip::tcp::socket>&&)
Ed Tanous1abe55e2018-09-05 08:30:59 -070058 {
Ed Tanousde5c9f32019-03-26 09:17:55 -070059 res.result(boost::beast::http::status::not_found);
Ed Tanous1abe55e2018-09-05 08:30:59 -070060 res.end();
61 }
Ed Tanous7045c8d2017-04-03 10:04:37 -070062#endif
63
Ed Tanous271584a2019-07-09 16:24:22 -070064 size_t getMethods()
Ed Tanous1abe55e2018-09-05 08:30:59 -070065 {
66 return methodsBitfield;
67 }
Ed Tanous7045c8d2017-04-03 10:04:37 -070068
Tanousf00032d2018-11-05 01:18:10 -030069 bool checkPrivileges(const redfish::Privileges& userPrivileges)
70 {
71 // If there are no privileges assigned, assume no privileges
72 // required
73 if (privilegesSet.empty())
74 {
75 return true;
76 }
77
78 for (const redfish::Privileges& requiredPrivileges : privilegesSet)
79 {
80 if (userPrivileges.isSupersetOf(requiredPrivileges))
81 {
82 return true;
83 }
84 }
85 return false;
86 }
87
Ed Tanous271584a2019-07-09 16:24:22 -070088 size_t methodsBitfield{
89 1 << static_cast<size_t>(boost::beast::http::verb::get)};
Ed Tanous7045c8d2017-04-03 10:04:37 -070090
Tanousf00032d2018-11-05 01:18:10 -030091 std::vector<redfish::Privileges> privilegesSet;
92
Ed Tanous1abe55e2018-09-05 08:30:59 -070093 std::string rule;
94 std::string nameStr;
Ed Tanous7045c8d2017-04-03 10:04:37 -070095
Ed Tanous1abe55e2018-09-05 08:30:59 -070096 std::unique_ptr<BaseRule> ruleToUpgrade;
Ed Tanous7045c8d2017-04-03 10:04:37 -070097
Ed Tanous1abe55e2018-09-05 08:30:59 -070098 friend class Router;
Gunnar Mills1214b7e2020-06-04 10:11:30 -050099 template <typename T>
100 friend struct RuleParameterTraits;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700101};
102
Ed Tanous1abe55e2018-09-05 08:30:59 -0700103namespace detail
104{
105namespace routing_handler_call_helper
106{
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500107template <typename T, int Pos>
108struct CallPair
Ed Tanous1abe55e2018-09-05 08:30:59 -0700109{
110 using type = T;
111 static const int pos = Pos;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700112};
113
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500114template <typename H1>
115struct CallParams
Ed Tanous1abe55e2018-09-05 08:30:59 -0700116{
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
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500126{};
Ed Tanous7045c8d2017-04-03 10:04:37 -0700127
128template <typename F, int NInt, int NUint, int NDouble, int NString,
129 typename... Args1, typename... Args2>
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700130struct Call<F, NInt, NUint, NDouble, NString, black_magic::S<int64_t, Args1...>,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700131 black_magic::S<Args2...>>
132{
133 void operator()(F cparams)
134 {
135 using pushed = typename black_magic::S<Args2...>::template push_back<
136 CallPair<int64_t, NInt>>;
137 Call<F, NInt + 1, NUint, NDouble, NString, black_magic::S<Args1...>,
138 pushed>()(cparams);
139 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700140};
141
142template <typename F, int NInt, int NUint, int NDouble, int NString,
143 typename... Args1, typename... Args2>
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700144struct Call<F, NInt, NUint, NDouble, NString,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700145 black_magic::S<uint64_t, Args1...>, black_magic::S<Args2...>>
146{
147 void operator()(F cparams)
148 {
149 using pushed = typename black_magic::S<Args2...>::template push_back<
150 CallPair<uint64_t, NUint>>;
151 Call<F, NInt, NUint + 1, NDouble, NString, black_magic::S<Args1...>,
152 pushed>()(cparams);
153 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700154};
155
156template <typename F, int NInt, int NUint, int NDouble, int NString,
157 typename... Args1, typename... Args2>
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700158struct Call<F, NInt, NUint, NDouble, NString, black_magic::S<double, Args1...>,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700159 black_magic::S<Args2...>>
160{
161 void operator()(F cparams)
162 {
163 using pushed = typename black_magic::S<Args2...>::template push_back<
164 CallPair<double, NDouble>>;
165 Call<F, NInt, NUint, NDouble + 1, NString, black_magic::S<Args1...>,
166 pushed>()(cparams);
167 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700168};
169
170template <typename F, int NInt, int NUint, int NDouble, int NString,
171 typename... Args1, typename... Args2>
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700172struct Call<F, NInt, NUint, NDouble, NString,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700173 black_magic::S<std::string, Args1...>, black_magic::S<Args2...>>
174{
175 void operator()(F cparams)
176 {
177 using pushed = typename black_magic::S<Args2...>::template push_back<
178 CallPair<std::string, NString>>;
179 Call<F, NInt, NUint, NDouble, NString + 1, black_magic::S<Args1...>,
180 pushed>()(cparams);
181 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700182};
183
184template <typename F, int NInt, int NUint, int NDouble, int NString,
185 typename... Args1>
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700186struct Call<F, NInt, NUint, NDouble, NString, black_magic::S<>,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700187 black_magic::S<Args1...>>
188{
189 void operator()(F cparams)
190 {
191 cparams.handler(
192 cparams.req, cparams.res,
193 cparams.params.template get<typename Args1::type>(Args1::pos)...);
194 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700195};
196
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500197template <typename Func, typename... ArgsWrapped>
198struct Wrapped
Ed Tanous1abe55e2018-09-05 08:30:59 -0700199{
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
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500216 template <typename Req, typename... Args>
217 struct ReqHandlerWrapper
Ed Tanous1abe55e2018-09-05 08:30:59 -0700218 {
Ed Tanous23a21a12020-07-25 04:45:05 +0000219 ReqHandlerWrapper(Func fIn) : f(std::move(fIn))
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500220 {}
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
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500267 template <typename... Args>
268 struct HandlerTypeHelper
Ed Tanous1abe55e2018-09-05 08:30:59 -0700269 {
270 using type =
271 std::function<void(const crow::Request&, crow::Response&, Args...)>;
272 using args_type =
273 black_magic::S<typename black_magic::promote_t<Args>...>;
274 };
Ed Tanous7045c8d2017-04-03 10:04:37 -0700275
Ed Tanous1abe55e2018-09-05 08:30:59 -0700276 template <typename... Args>
277 struct HandlerTypeHelper<const Request&, Args...>
278 {
279 using type =
280 std::function<void(const crow::Request&, crow::Response&, Args...)>;
281 using args_type =
282 black_magic::S<typename black_magic::promote_t<Args>...>;
283 };
Ed Tanous7045c8d2017-04-03 10:04:37 -0700284
Ed Tanous1abe55e2018-09-05 08:30:59 -0700285 template <typename... Args>
286 struct HandlerTypeHelper<const Request&, Response&, Args...>
287 {
288 using type =
289 std::function<void(const crow::Request&, crow::Response&, Args...)>;
290 using args_type =
291 black_magic::S<typename black_magic::promote_t<Args>...>;
292 };
293
294 typename HandlerTypeHelper<ArgsWrapped...>::type handler;
295
296 void operator()(const Request& req, Response& res,
297 const RoutingParams& params)
298 {
299 detail::routing_handler_call_helper::Call<
300 detail::routing_handler_call_helper::CallParams<decltype(handler)>,
301 0, 0, 0, 0, typename HandlerTypeHelper<ArgsWrapped...>::args_type,
302 black_magic::S<>>()(
303 detail::routing_handler_call_helper::CallParams<decltype(handler)>{
304 handler, params, req, res});
305 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700306};
Ed Tanous1abe55e2018-09-05 08:30:59 -0700307} // namespace routing_handler_call_helper
308} // namespace detail
Ed Tanous7045c8d2017-04-03 10:04:37 -0700309
Ed Tanous1abe55e2018-09-05 08:30:59 -0700310class WebSocketRule : public BaseRule
311{
312 using self_t = WebSocketRule;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700313
Ed Tanous1abe55e2018-09-05 08:30:59 -0700314 public:
Ed Tanous23a21a12020-07-25 04:45:05 +0000315 WebSocketRule(std::string ruleIn) : BaseRule(std::move(ruleIn))
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500316 {}
Ed Tanous7045c8d2017-04-03 10:04:37 -0700317
Ed Tanous1abe55e2018-09-05 08:30:59 -0700318 void validate() override
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500319 {}
Ed Tanous7045c8d2017-04-03 10:04:37 -0700320
Ed Tanous1abe55e2018-09-05 08:30:59 -0700321 void handle(const Request&, Response& res, const RoutingParams&) override
322 {
Ed Tanousde5c9f32019-03-26 09:17:55 -0700323 res.result(boost::beast::http::status::not_found);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700324 res.end();
325 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700326
Jan Sowinskiee52ae12020-01-09 16:28:32 +0000327 void handleUpgrade(const Request& req, Response&,
Ed Tanousceac6f72018-12-02 11:58:47 -0800328 boost::asio::ip::tcp::socket&& adaptor) override
Ed Tanous1abe55e2018-09-05 08:30:59 -0700329 {
Ratan Gupta02453b12019-10-22 14:43:36 +0530330 std::shared_ptr<
331 crow::websocket::ConnectionImpl<boost::asio::ip::tcp::socket>>
332 myConnection = std::make_shared<
333 crow::websocket::ConnectionImpl<boost::asio::ip::tcp::socket>>(
Jan Sowinskiee52ae12020-01-09 16:28:32 +0000334 req, std::move(adaptor), openHandler, messageHandler,
Ratan Gupta02453b12019-10-22 14:43:36 +0530335 closeHandler, errorHandler);
336 myConnection->start();
Ed Tanous1abe55e2018-09-05 08:30:59 -0700337 }
338#ifdef BMCWEB_ENABLE_SSL
Jan Sowinskiee52ae12020-01-09 16:28:32 +0000339 void handleUpgrade(const Request& req, Response&,
Ed Tanousceac6f72018-12-02 11:58:47 -0800340 boost::beast::ssl_stream<boost::asio::ip::tcp::socket>&&
341 adaptor) override
Ed Tanous1abe55e2018-09-05 08:30:59 -0700342 {
Ed Tanousceac6f72018-12-02 11:58:47 -0800343 std::shared_ptr<crow::websocket::ConnectionImpl<
344 boost::beast::ssl_stream<boost::asio::ip::tcp::socket>>>
345 myConnection = std::make_shared<crow::websocket::ConnectionImpl<
346 boost::beast::ssl_stream<boost::asio::ip::tcp::socket>>>(
Jan Sowinskiee52ae12020-01-09 16:28:32 +0000347 req, std::move(adaptor), openHandler, messageHandler,
Ed Tanousceac6f72018-12-02 11:58:47 -0800348 closeHandler, errorHandler);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700349 myConnection->start();
350 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700351#endif
352
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500353 template <typename Func>
354 self_t& onopen(Func f)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700355 {
356 openHandler = f;
357 return *this;
358 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700359
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500360 template <typename Func>
361 self_t& onmessage(Func f)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700362 {
363 messageHandler = f;
364 return *this;
365 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700366
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500367 template <typename Func>
368 self_t& onclose(Func f)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700369 {
370 closeHandler = f;
371 return *this;
372 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700373
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500374 template <typename Func>
375 self_t& onerror(Func f)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700376 {
377 errorHandler = f;
378 return *this;
379 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700380
Ed Tanous1abe55e2018-09-05 08:30:59 -0700381 protected:
Iwona Klimaszewskac0a1c8a2019-07-12 18:26:38 +0200382 std::function<void(crow::websocket::Connection&,
383 std::shared_ptr<bmcweb::AsyncResp>)>
384 openHandler;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700385 std::function<void(crow::websocket::Connection&, const std::string&, bool)>
386 messageHandler;
387 std::function<void(crow::websocket::Connection&, const std::string&)>
388 closeHandler;
389 std::function<void(crow::websocket::Connection&)> errorHandler;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700390};
391
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500392template <typename T>
393struct RuleParameterTraits
Ed Tanous1abe55e2018-09-05 08:30:59 -0700394{
395 using self_t = T;
396 WebSocketRule& websocket()
397 {
Ed Tanous271584a2019-07-09 16:24:22 -0700398 self_t* self = static_cast<self_t*>(this);
399 WebSocketRule* p = new WebSocketRule(self->rule);
400 self->ruleToUpgrade.reset(p);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700401 return *p;
402 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700403
Ed Tanous1abe55e2018-09-05 08:30:59 -0700404 self_t& name(std::string name) noexcept
405 {
Ed Tanous271584a2019-07-09 16:24:22 -0700406 self_t* self = static_cast<self_t*>(this);
407 self->nameStr = std::move(name);
408 return *self;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700409 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700410
Ed Tanous1abe55e2018-09-05 08:30:59 -0700411 self_t& methods(boost::beast::http::verb method)
412 {
Ed Tanous271584a2019-07-09 16:24:22 -0700413 self_t* self = static_cast<self_t*>(this);
414 self->methodsBitfield = 1U << static_cast<size_t>(method);
415 return *self;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700416 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700417
Ed Tanous1abe55e2018-09-05 08:30:59 -0700418 template <typename... MethodArgs>
419 self_t& methods(boost::beast::http::verb method, MethodArgs... args_method)
420 {
Ed Tanous271584a2019-07-09 16:24:22 -0700421 self_t* self = static_cast<self_t*>(this);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700422 methods(args_method...);
Ed Tanous271584a2019-07-09 16:24:22 -0700423 self->methodsBitfield |= 1U << static_cast<size_t>(method);
424 return *self;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700425 }
Tanousf00032d2018-11-05 01:18:10 -0300426
427 template <typename... MethodArgs>
Ed Tanous23a21a12020-07-25 04:45:05 +0000428 self_t& privileges(std::initializer_list<const char*> l)
Tanousf00032d2018-11-05 01:18:10 -0300429 {
Ed Tanous271584a2019-07-09 16:24:22 -0700430 self_t* self = static_cast<self_t*>(this);
431 self->privilegesSet.emplace_back(l);
432 return *self;
Tanousf00032d2018-11-05 01:18:10 -0300433 }
434
435 template <typename... MethodArgs>
Ed Tanous23a21a12020-07-25 04:45:05 +0000436 self_t& privileges(const std::vector<redfish::Privileges>& p)
Tanousf00032d2018-11-05 01:18:10 -0300437 {
Ed Tanous271584a2019-07-09 16:24:22 -0700438 self_t* self = static_cast<self_t*>(this);
Tanousf00032d2018-11-05 01:18:10 -0300439 for (const redfish::Privileges& privilege : p)
440 {
Ed Tanous271584a2019-07-09 16:24:22 -0700441 self->privilegesSet.emplace_back(privilege);
Tanousf00032d2018-11-05 01:18:10 -0300442 }
Ed Tanous271584a2019-07-09 16:24:22 -0700443 return *self;
Tanousf00032d2018-11-05 01:18:10 -0300444 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700445};
446
Ed Tanous1abe55e2018-09-05 08:30:59 -0700447class DynamicRule : public BaseRule, public RuleParameterTraits<DynamicRule>
448{
449 public:
Ed Tanous23a21a12020-07-25 04:45:05 +0000450 DynamicRule(std::string ruleIn) : BaseRule(std::move(ruleIn))
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500451 {}
Ed Tanous7045c8d2017-04-03 10:04:37 -0700452
Ed Tanous1abe55e2018-09-05 08:30:59 -0700453 void validate() override
454 {
455 if (!erasedHandler)
456 {
457 throw std::runtime_error(nameStr + (!nameStr.empty() ? ": " : "") +
458 "no handler for url " + rule);
459 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700460 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700461
Ed Tanous1abe55e2018-09-05 08:30:59 -0700462 void handle(const Request& req, Response& res,
463 const RoutingParams& params) override
464 {
465 erasedHandler(req, res, params);
466 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700467
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500468 template <typename Func>
469 void operator()(Func f)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700470 {
471 using function_t = utility::function_traits<Func>;
472
473 erasedHandler =
474 wrap(std::move(f), black_magic::gen_seq<function_t::arity>());
475 }
476
477 // enable_if Arg1 == request && Arg2 == Response
Gunnar Mills6be0e402020-07-08 13:21:51 -0500478 // enable_if Arg1 == request && Arg2 != response
Ed Tanous1abe55e2018-09-05 08:30:59 -0700479 // enable_if Arg1 != request
480
481 template <typename Func, unsigned... Indices>
482
483 std::function<void(const Request&, Response&, const RoutingParams&)>
484 wrap(Func f, black_magic::Seq<Indices...>)
485 {
486 using function_t = utility::function_traits<Func>;
487
488 if (!black_magic::isParameterTagCompatible(
489 black_magic::getParameterTagRuntime(rule.c_str()),
490 black_magic::compute_parameter_tag_from_args_list<
491 typename function_t::template arg<Indices>...>::value))
492 {
493 throw std::runtime_error("routeDynamic: Handler type is mismatched "
494 "with URL parameters: " +
495 rule);
496 }
497 auto ret = detail::routing_handler_call_helper::Wrapped<
498 Func, typename function_t::template arg<Indices>...>();
499 ret.template set<typename function_t::template arg<Indices>...>(
500 std::move(f));
501 return ret;
502 }
503
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500504 template <typename Func>
505 void operator()(std::string name, Func&& f)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700506 {
507 nameStr = std::move(name);
508 (*this).template operator()<Func>(std::forward(f));
509 }
510
511 private:
512 std::function<void(const Request&, Response&, const RoutingParams&)>
513 erasedHandler;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700514};
515
516template <typename... Args>
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500517class TaggedRule :
518 public BaseRule,
519 public RuleParameterTraits<TaggedRule<Args...>>
Ed Tanous1abe55e2018-09-05 08:30:59 -0700520{
521 public:
522 using self_t = TaggedRule<Args...>;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700523
Ed Tanous271584a2019-07-09 16:24:22 -0700524 TaggedRule(std::string ruleIn) : BaseRule(std::move(ruleIn))
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500525 {}
Ed Tanous7045c8d2017-04-03 10:04:37 -0700526
Ed Tanous1abe55e2018-09-05 08:30:59 -0700527 void validate() override
528 {
529 if (!handler)
530 {
531 throw std::runtime_error(nameStr + (!nameStr.empty() ? ": " : "") +
532 "no handler for url " + rule);
533 }
534 }
535
536 template <typename Func>
537 typename std::enable_if<
538 black_magic::CallHelper<Func, black_magic::S<Args...>>::value,
539 void>::type
540 operator()(Func&& f)
541 {
542 static_assert(
543 black_magic::CallHelper<Func, black_magic::S<Args...>>::value ||
544 black_magic::CallHelper<
545 Func, black_magic::S<crow::Request, Args...>>::value,
546 "Handler type is mismatched with URL parameters");
547 static_assert(
548 !std::is_same<void, decltype(f(std::declval<Args>()...))>::value,
549 "Handler function cannot have void return type; valid return "
550 "types: "
Gunnar Mills6be0e402020-07-08 13:21:51 -0500551 "string, int, crow::response, nlohmann::json");
Ed Tanous1abe55e2018-09-05 08:30:59 -0700552
553 handler = [f = std::move(f)](const Request&, Response& res,
554 Args... args) {
Ed Tanousde5c9f32019-03-26 09:17:55 -0700555 res.result(f(args...));
Ed Tanous1abe55e2018-09-05 08:30:59 -0700556 res.end();
557 };
558 }
559
560 template <typename Func>
561 typename std::enable_if<
562 !black_magic::CallHelper<Func, black_magic::S<Args...>>::value &&
Ed Tanous7045c8d2017-04-03 10:04:37 -0700563 black_magic::CallHelper<
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700564 Func, black_magic::S<crow::Request, Args...>>::value,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700565 void>::type
566 operator()(Func&& f)
567 {
568 static_assert(
569 black_magic::CallHelper<Func, black_magic::S<Args...>>::value ||
570 black_magic::CallHelper<
571 Func, black_magic::S<crow::Request, Args...>>::value,
572 "Handler type is mismatched with URL parameters");
573 static_assert(
574 !std::is_same<void, decltype(f(std::declval<crow::Request>(),
575 std::declval<Args>()...))>::value,
576 "Handler function cannot have void return type; valid return "
577 "types: "
Gunnar Mills6be0e402020-07-08 13:21:51 -0500578 "string, int, crow::response,nlohmann::json");
Ed Tanous7045c8d2017-04-03 10:04:37 -0700579
Ed Tanous1abe55e2018-09-05 08:30:59 -0700580 handler = [f = std::move(f)](const crow::Request& req,
581 crow::Response& res, Args... args) {
Ed Tanousde5c9f32019-03-26 09:17:55 -0700582 res.result(f(req, args...));
Ed Tanous1abe55e2018-09-05 08:30:59 -0700583 res.end();
584 };
585 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700586
Ed Tanous1abe55e2018-09-05 08:30:59 -0700587 template <typename Func>
588 typename std::enable_if<
589 !black_magic::CallHelper<Func, black_magic::S<Args...>>::value &&
590 !black_magic::CallHelper<
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700591 Func, black_magic::S<crow::Request, Args...>>::value,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700592 void>::type
593 operator()(Func&& f)
594 {
595 static_assert(
596 black_magic::CallHelper<Func, black_magic::S<Args...>>::value ||
597 black_magic::CallHelper<
598 Func, black_magic::S<crow::Request, Args...>>::value ||
599 black_magic::CallHelper<
600 Func, black_magic::S<crow::Request, crow::Response&,
601 Args...>>::value,
602 "Handler type is mismatched with URL parameters");
603 static_assert(
604 std::is_same<void, decltype(f(std::declval<crow::Request>(),
605 std::declval<crow::Response&>(),
606 std::declval<Args>()...))>::value,
Tanousf00032d2018-11-05 01:18:10 -0300607 "Handler function with response argument should have void "
608 "return "
Ed Tanous1abe55e2018-09-05 08:30:59 -0700609 "type");
Ed Tanous7045c8d2017-04-03 10:04:37 -0700610
Ed Tanous1abe55e2018-09-05 08:30:59 -0700611 handler = std::move(f);
612 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700613
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500614 template <typename Func>
615 void operator()(std::string name, Func&& f)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700616 {
617 nameStr = std::move(name);
618 (*this).template operator()<Func>(std::forward(f));
619 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700620
Ed Tanous1abe55e2018-09-05 08:30:59 -0700621 void handle(const Request& req, Response& res,
622 const RoutingParams& params) override
623 {
624 detail::routing_handler_call_helper::Call<
625 detail::routing_handler_call_helper::CallParams<decltype(handler)>,
626 0, 0, 0, 0, black_magic::S<Args...>, black_magic::S<>>()(
627 detail::routing_handler_call_helper::CallParams<decltype(handler)>{
628 handler, params, req, res});
629 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700630
Ed Tanous1abe55e2018-09-05 08:30:59 -0700631 private:
632 std::function<void(const crow::Request&, crow::Response&, Args...)> handler;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700633};
634
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700635const int ruleSpecialRedirectSlash = 1;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700636
Ed Tanous1abe55e2018-09-05 08:30:59 -0700637class Trie
638{
639 public:
640 struct Node
641 {
642 unsigned ruleIndex{};
Ed Tanous271584a2019-07-09 16:24:22 -0700643 std::array<size_t, static_cast<size_t>(ParamType::MAX)>
644 paramChildrens{};
Ed Tanous1abe55e2018-09-05 08:30:59 -0700645 boost::container::flat_map<std::string, unsigned> children;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700646
Ed Tanous1abe55e2018-09-05 08:30:59 -0700647 bool isSimpleNode() const
648 {
649 return !ruleIndex && std::all_of(std::begin(paramChildrens),
650 std::end(paramChildrens),
Ed Tanous271584a2019-07-09 16:24:22 -0700651 [](size_t x) { return !x; });
Ed Tanous7045c8d2017-04-03 10:04:37 -0700652 }
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700653 };
Ed Tanous7045c8d2017-04-03 10:04:37 -0700654
Ed Tanous1abe55e2018-09-05 08:30:59 -0700655 Trie() : nodes(1)
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500656 {}
Ed Tanous1abe55e2018-09-05 08:30:59 -0700657
658 private:
659 void optimizeNode(Node* node)
660 {
Ed Tanous271584a2019-07-09 16:24:22 -0700661 for (size_t x : node->paramChildrens)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700662 {
663 if (!x)
664 continue;
665 Node* child = &nodes[x];
666 optimizeNode(child);
Ed Tanous7045c8d2017-04-03 10:04:37 -0700667 }
Ed Tanous1abe55e2018-09-05 08:30:59 -0700668 if (node->children.empty())
669 return;
670 bool mergeWithChild = true;
Tanousf00032d2018-11-05 01:18:10 -0300671 for (const std::pair<std::string, unsigned>& kv : node->children)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700672 {
673 Node* child = &nodes[kv.second];
674 if (!child->isSimpleNode())
675 {
676 mergeWithChild = false;
677 break;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700678 }
Ed Tanous1abe55e2018-09-05 08:30:59 -0700679 }
680 if (mergeWithChild)
681 {
682 decltype(node->children) merged;
Tanousf00032d2018-11-05 01:18:10 -0300683 for (const std::pair<std::string, unsigned>& kv : node->children)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700684 {
685 Node* child = &nodes[kv.second];
Tanousf00032d2018-11-05 01:18:10 -0300686 for (const std::pair<std::string, unsigned>& childKv :
687 child->children)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700688 {
689 merged[kv.first + childKv.first] = childKv.second;
690 }
691 }
692 node->children = std::move(merged);
693 optimizeNode(node);
694 }
695 else
696 {
Tanousf00032d2018-11-05 01:18:10 -0300697 for (const std::pair<std::string, unsigned>& kv : node->children)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700698 {
699 Node* child = &nodes[kv.second];
700 optimizeNode(child);
701 }
702 }
703 }
704
705 void optimize()
706 {
707 optimizeNode(head());
708 }
709
710 public:
711 void validate()
712 {
713 if (!head()->isSimpleNode())
714 throw std::runtime_error(
715 "Internal error: Trie header should be simple!");
716 optimize();
717 }
718
719 void findRouteIndexes(const std::string& req_url,
720 std::vector<unsigned>& route_indexes,
Tanousf00032d2018-11-05 01:18:10 -0300721 const Node* node = nullptr, unsigned pos = 0) const
Ed Tanous1abe55e2018-09-05 08:30:59 -0700722 {
723 if (node == nullptr)
724 {
725 node = head();
726 }
Tanousf00032d2018-11-05 01:18:10 -0300727 for (const std::pair<std::string, unsigned>& kv : node->children)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700728 {
729 const std::string& fragment = kv.first;
730 const Node* child = &nodes[kv.second];
731 if (pos >= req_url.size())
732 {
733 if (child->ruleIndex != 0 && fragment != "/")
734 {
735 route_indexes.push_back(child->ruleIndex);
736 }
737 findRouteIndexes(req_url, route_indexes, child,
Ed Tanous271584a2019-07-09 16:24:22 -0700738 static_cast<unsigned>(pos + fragment.size()));
Ed Tanous1abe55e2018-09-05 08:30:59 -0700739 }
740 else
741 {
742 if (req_url.compare(pos, fragment.size(), fragment) == 0)
743 {
Ed Tanous271584a2019-07-09 16:24:22 -0700744 findRouteIndexes(
745 req_url, route_indexes, child,
746 static_cast<unsigned>(pos + fragment.size()));
Ed Tanous1abe55e2018-09-05 08:30:59 -0700747 }
748 }
749 }
750 }
751
752 std::pair<unsigned, RoutingParams>
Ed Tanous39e77502019-03-04 17:35:53 -0800753 find(const std::string_view req_url, const Node* node = nullptr,
Ed Tanous271584a2019-07-09 16:24:22 -0700754 size_t pos = 0, RoutingParams* params = nullptr) const
Ed Tanous1abe55e2018-09-05 08:30:59 -0700755 {
756 RoutingParams empty;
757 if (params == nullptr)
758 params = &empty;
759
760 unsigned found{};
761 RoutingParams matchParams;
762
763 if (node == nullptr)
764 node = head();
765 if (pos == req_url.size())
766 return {node->ruleIndex, *params};
767
768 auto updateFound =
769 [&found, &matchParams](std::pair<unsigned, RoutingParams>& ret) {
770 if (ret.first && (!found || found > ret.first))
771 {
772 found = ret.first;
773 matchParams = std::move(ret.second);
774 }
775 };
776
Ed Tanous271584a2019-07-09 16:24:22 -0700777 if (node->paramChildrens[static_cast<size_t>(ParamType::INT)])
Ed Tanous1abe55e2018-09-05 08:30:59 -0700778 {
779 char c = req_url[pos];
780 if ((c >= '0' && c <= '9') || c == '+' || c == '-')
781 {
782 char* eptr;
783 errno = 0;
784 long long int value =
785 std::strtoll(req_url.data() + pos, &eptr, 10);
786 if (errno != ERANGE && eptr != req_url.data() + pos)
787 {
788 params->intParams.push_back(value);
Ed Tanous271584a2019-07-09 16:24:22 -0700789 std::pair<unsigned, RoutingParams> ret = find(
790 req_url,
791 &nodes[node->paramChildrens[static_cast<size_t>(
792 ParamType::INT)]],
793 static_cast<size_t>(eptr - req_url.data()), params);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700794 updateFound(ret);
795 params->intParams.pop_back();
796 }
797 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700798 }
799
Ed Tanous271584a2019-07-09 16:24:22 -0700800 if (node->paramChildrens[static_cast<size_t>(ParamType::UINT)])
Ed Tanous1abe55e2018-09-05 08:30:59 -0700801 {
802 char c = req_url[pos];
803 if ((c >= '0' && c <= '9') || c == '+')
804 {
805 char* eptr;
806 errno = 0;
807 unsigned long long int value =
808 std::strtoull(req_url.data() + pos, &eptr, 10);
809 if (errno != ERANGE && eptr != req_url.data() + pos)
810 {
811 params->uintParams.push_back(value);
Ed Tanous271584a2019-07-09 16:24:22 -0700812 std::pair<unsigned, RoutingParams> ret = find(
813 req_url,
814 &nodes[node->paramChildrens[static_cast<size_t>(
815 ParamType::UINT)]],
816 static_cast<size_t>(eptr - req_url.data()), params);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700817 updateFound(ret);
818 params->uintParams.pop_back();
819 }
820 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700821 }
822
Ed Tanous271584a2019-07-09 16:24:22 -0700823 if (node->paramChildrens[static_cast<size_t>(ParamType::DOUBLE)])
Ed Tanous1abe55e2018-09-05 08:30:59 -0700824 {
825 char c = req_url[pos];
826 if ((c >= '0' && c <= '9') || c == '+' || c == '-' || c == '.')
827 {
828 char* eptr;
829 errno = 0;
830 double value = std::strtod(req_url.data() + pos, &eptr);
831 if (errno != ERANGE && eptr != req_url.data() + pos)
832 {
833 params->doubleParams.push_back(value);
Tanousf00032d2018-11-05 01:18:10 -0300834 std::pair<unsigned, RoutingParams> ret = find(
Ed Tanous1abe55e2018-09-05 08:30:59 -0700835 req_url,
Ed Tanous271584a2019-07-09 16:24:22 -0700836 &nodes[node->paramChildrens[static_cast<size_t>(
837 ParamType::DOUBLE)]],
838 static_cast<size_t>(eptr - req_url.data()), params);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700839 updateFound(ret);
840 params->doubleParams.pop_back();
841 }
842 }
843 }
844
Ed Tanous271584a2019-07-09 16:24:22 -0700845 if (node->paramChildrens[static_cast<size_t>(ParamType::STRING)])
Ed Tanous1abe55e2018-09-05 08:30:59 -0700846 {
Ed Tanousb01bf292019-03-25 19:25:26 +0000847 size_t epos = pos;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700848 for (; epos < req_url.size(); epos++)
849 {
850 if (req_url[epos] == '/')
851 break;
852 }
853
854 if (epos != pos)
855 {
856 params->stringParams.emplace_back(
857 req_url.substr(pos, epos - pos));
Tanousf00032d2018-11-05 01:18:10 -0300858 std::pair<unsigned, RoutingParams> ret =
Ed Tanousb01bf292019-03-25 19:25:26 +0000859 find(req_url,
Ed Tanous271584a2019-07-09 16:24:22 -0700860 &nodes[node->paramChildrens[static_cast<size_t>(
861 ParamType::STRING)]],
Ed Tanousb01bf292019-03-25 19:25:26 +0000862 epos, params);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700863 updateFound(ret);
864 params->stringParams.pop_back();
865 }
866 }
867
Ed Tanous271584a2019-07-09 16:24:22 -0700868 if (node->paramChildrens[static_cast<size_t>(ParamType::PATH)])
Ed Tanous1abe55e2018-09-05 08:30:59 -0700869 {
Ed Tanousb01bf292019-03-25 19:25:26 +0000870 size_t epos = req_url.size();
Ed Tanous1abe55e2018-09-05 08:30:59 -0700871
872 if (epos != pos)
873 {
874 params->stringParams.emplace_back(
875 req_url.substr(pos, epos - pos));
Ed Tanous271584a2019-07-09 16:24:22 -0700876 std::pair<unsigned, RoutingParams> ret =
877 find(req_url,
878 &nodes[node->paramChildrens[static_cast<size_t>(
879 ParamType::PATH)]],
880 epos, params);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700881 updateFound(ret);
882 params->stringParams.pop_back();
883 }
884 }
885
Tanousf00032d2018-11-05 01:18:10 -0300886 for (const std::pair<std::string, unsigned>& kv : node->children)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700887 {
888 const std::string& fragment = kv.first;
889 const Node* child = &nodes[kv.second];
890
891 if (req_url.compare(pos, fragment.size(), fragment) == 0)
892 {
Tanousf00032d2018-11-05 01:18:10 -0300893 std::pair<unsigned, RoutingParams> ret =
894 find(req_url, child, pos + fragment.size(), params);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700895 updateFound(ret);
896 }
897 }
898
899 return {found, matchParams};
Ed Tanous7045c8d2017-04-03 10:04:37 -0700900 }
Ed Tanous1abe55e2018-09-05 08:30:59 -0700901
902 void add(const std::string& url, unsigned ruleIndex)
903 {
Ed Tanous271584a2019-07-09 16:24:22 -0700904 size_t idx = 0;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700905
906 for (unsigned i = 0; i < url.size(); i++)
907 {
908 char c = url[i];
909 if (c == '<')
910 {
Tanousf00032d2018-11-05 01:18:10 -0300911 const static std::array<std::pair<ParamType, std::string>, 7>
912 paramTraits = {{
913 {ParamType::INT, "<int>"},
914 {ParamType::UINT, "<uint>"},
915 {ParamType::DOUBLE, "<float>"},
916 {ParamType::DOUBLE, "<double>"},
917 {ParamType::STRING, "<str>"},
918 {ParamType::STRING, "<string>"},
919 {ParamType::PATH, "<path>"},
920 }};
Ed Tanous1abe55e2018-09-05 08:30:59 -0700921
Tanousf00032d2018-11-05 01:18:10 -0300922 for (const std::pair<ParamType, std::string>& x : paramTraits)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700923 {
Tanousf00032d2018-11-05 01:18:10 -0300924 if (url.compare(i, x.second.size(), x.second) == 0)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700925 {
Ed Tanous271584a2019-07-09 16:24:22 -0700926 size_t index = static_cast<size_t>(x.first);
927 if (!nodes[idx].paramChildrens[index])
Ed Tanous1abe55e2018-09-05 08:30:59 -0700928 {
Tanousf00032d2018-11-05 01:18:10 -0300929 unsigned newNodeIdx = newNode();
Ed Tanous271584a2019-07-09 16:24:22 -0700930 nodes[idx].paramChildrens[index] = newNodeIdx;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700931 }
Ed Tanous271584a2019-07-09 16:24:22 -0700932 idx = nodes[idx].paramChildrens[index];
933 i += static_cast<unsigned>(x.second.size());
Ed Tanous1abe55e2018-09-05 08:30:59 -0700934 break;
935 }
936 }
937
938 i--;
939 }
940 else
941 {
942 std::string piece(&c, 1);
943 if (!nodes[idx].children.count(piece))
944 {
Tanousf00032d2018-11-05 01:18:10 -0300945 unsigned newNodeIdx = newNode();
Ed Tanous1abe55e2018-09-05 08:30:59 -0700946 nodes[idx].children.emplace(piece, newNodeIdx);
947 }
948 idx = nodes[idx].children[piece];
949 }
950 }
951 if (nodes[idx].ruleIndex)
952 throw std::runtime_error("handler already exists for " + url);
953 nodes[idx].ruleIndex = ruleIndex;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700954 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700955
Ed Tanous1abe55e2018-09-05 08:30:59 -0700956 private:
Ed Tanous271584a2019-07-09 16:24:22 -0700957 void debugNodePrint(Node* n, size_t level)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700958 {
Ed Tanous271584a2019-07-09 16:24:22 -0700959 for (size_t i = 0; i < static_cast<size_t>(ParamType::MAX); i++)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700960 {
961 if (n->paramChildrens[i])
962 {
963 BMCWEB_LOG_DEBUG << std::string(
Ed Tanous271584a2019-07-09 16:24:22 -0700964 2U * level, ' ') /*<< "("<<n->paramChildrens[i]<<") "*/;
965 switch (static_cast<ParamType>(i))
Ed Tanous1abe55e2018-09-05 08:30:59 -0700966 {
967 case ParamType::INT:
968 BMCWEB_LOG_DEBUG << "<int>";
969 break;
970 case ParamType::UINT:
971 BMCWEB_LOG_DEBUG << "<uint>";
972 break;
973 case ParamType::DOUBLE:
974 BMCWEB_LOG_DEBUG << "<float>";
975 break;
976 case ParamType::STRING:
977 BMCWEB_LOG_DEBUG << "<str>";
978 break;
979 case ParamType::PATH:
980 BMCWEB_LOG_DEBUG << "<path>";
981 break;
Ed Tanous23a21a12020-07-25 04:45:05 +0000982 case ParamType::MAX:
Ed Tanous1abe55e2018-09-05 08:30:59 -0700983 BMCWEB_LOG_DEBUG << "<ERROR>";
984 break;
985 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700986
Ed Tanous1abe55e2018-09-05 08:30:59 -0700987 debugNodePrint(&nodes[n->paramChildrens[i]], level + 1);
988 }
989 }
Tanousf00032d2018-11-05 01:18:10 -0300990 for (const std::pair<std::string, unsigned>& kv : n->children)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700991 {
992 BMCWEB_LOG_DEBUG
Ed Tanous271584a2019-07-09 16:24:22 -0700993 << std::string(2U * level, ' ') /*<< "(" << kv.second << ") "*/
Ed Tanous1abe55e2018-09-05 08:30:59 -0700994 << kv.first;
995 debugNodePrint(&nodes[kv.second], level + 1);
996 }
997 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700998
Ed Tanous1abe55e2018-09-05 08:30:59 -0700999 public:
1000 void debugPrint()
1001 {
Ed Tanous271584a2019-07-09 16:24:22 -07001002 debugNodePrint(head(), 0U);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001003 }
Ed Tanous7045c8d2017-04-03 10:04:37 -07001004
Ed Tanous1abe55e2018-09-05 08:30:59 -07001005 private:
1006 const Node* head() const
1007 {
1008 return &nodes.front();
1009 }
Ed Tanous7045c8d2017-04-03 10:04:37 -07001010
Ed Tanous1abe55e2018-09-05 08:30:59 -07001011 Node* head()
1012 {
1013 return &nodes.front();
1014 }
1015
1016 unsigned newNode()
1017 {
1018 nodes.resize(nodes.size() + 1);
Ed Tanous271584a2019-07-09 16:24:22 -07001019 return static_cast<unsigned>(nodes.size() - 1);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001020 }
1021
1022 std::vector<Node> nodes;
Ed Tanous7045c8d2017-04-03 10:04:37 -07001023};
1024
Ed Tanous1abe55e2018-09-05 08:30:59 -07001025class Router
1026{
1027 public:
Ed Tanous0c0084a2019-10-24 15:57:51 -07001028 Router() = default;
Ed Tanous7045c8d2017-04-03 10:04:37 -07001029
Ed Tanous1abe55e2018-09-05 08:30:59 -07001030 DynamicRule& newRuleDynamic(const std::string& rule)
1031 {
1032 std::unique_ptr<DynamicRule> ruleObject =
1033 std::make_unique<DynamicRule>(rule);
1034 DynamicRule* ptr = ruleObject.get();
Tanousf00032d2018-11-05 01:18:10 -03001035 allRules.emplace_back(std::move(ruleObject));
Ed Tanous7045c8d2017-04-03 10:04:37 -07001036
Ed Tanous1abe55e2018-09-05 08:30:59 -07001037 return *ptr;
Ed Tanous7045c8d2017-04-03 10:04:37 -07001038 }
1039
Ed Tanous1abe55e2018-09-05 08:30:59 -07001040 template <uint64_t N>
1041 typename black_magic::Arguments<N>::type::template rebind<TaggedRule>&
1042 newRuleTagged(const std::string& rule)
1043 {
1044 using RuleT = typename black_magic::Arguments<N>::type::template rebind<
1045 TaggedRule>;
1046 std::unique_ptr<RuleT> ruleObject = std::make_unique<RuleT>(rule);
1047 RuleT* ptr = ruleObject.get();
Tanousf00032d2018-11-05 01:18:10 -03001048 allRules.emplace_back(std::move(ruleObject));
Ed Tanous1abe55e2018-09-05 08:30:59 -07001049
1050 return *ptr;
Ed Tanous7045c8d2017-04-03 10:04:37 -07001051 }
1052
Tanousf00032d2018-11-05 01:18:10 -03001053 void internalAddRuleObject(const std::string& rule, BaseRule* ruleObject)
Ed Tanous1abe55e2018-09-05 08:30:59 -07001054 {
Tanousf00032d2018-11-05 01:18:10 -03001055 if (ruleObject == nullptr)
Ed Tanous1abe55e2018-09-05 08:30:59 -07001056 {
Tanousf00032d2018-11-05 01:18:10 -03001057 return;
1058 }
Ed Tanous888880a2020-08-24 13:48:50 -07001059 for (size_t method = 0, methodBit = 1; method < maxHttpVerbCount;
Ed Tanous2c70f802020-09-28 14:29:23 -07001060 method++, methodBit <<= 1)
Tanousf00032d2018-11-05 01:18:10 -03001061 {
Ed Tanous2c70f802020-09-28 14:29:23 -07001062 if (ruleObject->methodsBitfield & methodBit)
Tanousf00032d2018-11-05 01:18:10 -03001063 {
1064 perMethods[method].rules.emplace_back(ruleObject);
1065 perMethods[method].trie.add(
Ed Tanous271584a2019-07-09 16:24:22 -07001066 rule, static_cast<unsigned>(
1067 perMethods[method].rules.size() - 1U));
Tanousf00032d2018-11-05 01:18:10 -03001068 // directory case:
1069 // request to `/about' url matches `/about/' rule
1070 if (rule.size() > 2 && rule.back() == '/')
1071 {
1072 perMethods[method].trie.add(
1073 rule.substr(0, rule.size() - 1),
Ed Tanous271584a2019-07-09 16:24:22 -07001074 static_cast<unsigned>(perMethods[method].rules.size() -
1075 1));
Tanousf00032d2018-11-05 01:18:10 -03001076 }
1077 }
Ed Tanous1abe55e2018-09-05 08:30:59 -07001078 }
Ed Tanous7045c8d2017-04-03 10:04:37 -07001079 }
1080
Ed Tanous1abe55e2018-09-05 08:30:59 -07001081 void validate()
1082 {
Tanousf00032d2018-11-05 01:18:10 -03001083 for (std::unique_ptr<BaseRule>& rule : allRules)
Ed Tanous1abe55e2018-09-05 08:30:59 -07001084 {
1085 if (rule)
1086 {
Tanousf00032d2018-11-05 01:18:10 -03001087 std::unique_ptr<BaseRule> upgraded = rule->upgrade();
Ed Tanous1abe55e2018-09-05 08:30:59 -07001088 if (upgraded)
1089 rule = std::move(upgraded);
1090 rule->validate();
Tanousf00032d2018-11-05 01:18:10 -03001091 internalAddRuleObject(rule->rule, rule.get());
Ed Tanous1abe55e2018-09-05 08:30:59 -07001092 }
1093 }
Tanousf00032d2018-11-05 01:18:10 -03001094 for (PerMethod& perMethod : perMethods)
1095 {
1096 perMethod.trie.validate();
1097 }
Ed Tanous7045c8d2017-04-03 10:04:37 -07001098 }
1099
Ed Tanous1abe55e2018-09-05 08:30:59 -07001100 template <typename Adaptor>
1101 void handleUpgrade(const Request& req, Response& res, Adaptor&& adaptor)
1102 {
Ed Tanous271584a2019-07-09 16:24:22 -07001103 if (static_cast<size_t>(req.method()) >= perMethods.size())
Ed Tanous888880a2020-08-24 13:48:50 -07001104 {
1105 res.result(boost::beast::http::status::not_found);
1106 res.end();
Tanousf00032d2018-11-05 01:18:10 -03001107 return;
Ed Tanous888880a2020-08-24 13:48:50 -07001108 }
Tanousf00032d2018-11-05 01:18:10 -03001109
Ed Tanous271584a2019-07-09 16:24:22 -07001110 PerMethod& perMethod = perMethods[static_cast<size_t>(req.method())];
Tanousf00032d2018-11-05 01:18:10 -03001111 Trie& trie = perMethod.trie;
1112 std::vector<BaseRule*>& rules = perMethod.rules;
1113
1114 const std::pair<unsigned, RoutingParams>& found = trie.find(req.url);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001115 unsigned ruleIndex = found.first;
1116 if (!ruleIndex)
1117 {
1118 BMCWEB_LOG_DEBUG << "Cannot match rules " << req.url;
Ed Tanousde5c9f32019-03-26 09:17:55 -07001119 res.result(boost::beast::http::status::not_found);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001120 res.end();
1121 return;
1122 }
1123
1124 if (ruleIndex >= rules.size())
1125 throw std::runtime_error("Trie internal structure corrupted!");
1126
1127 if (ruleIndex == ruleSpecialRedirectSlash)
1128 {
1129 BMCWEB_LOG_INFO << "Redirecting to a url with trailing slash: "
1130 << req.url;
Ed Tanousde5c9f32019-03-26 09:17:55 -07001131 res.result(boost::beast::http::status::moved_permanently);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001132
1133 // TODO absolute url building
1134 if (req.getHeaderValue("Host").empty())
1135 {
1136 res.addHeader("Location", std::string(req.url) + "/");
1137 }
1138 else
1139 {
1140 res.addHeader(
1141 "Location",
1142 req.isSecure
1143 ? "https://"
1144 : "http://" + std::string(req.getHeaderValue("Host")) +
1145 std::string(req.url) + "/");
1146 }
1147 res.end();
1148 return;
1149 }
1150
Ed Tanous271584a2019-07-09 16:24:22 -07001151 if ((rules[ruleIndex]->getMethods() &
1152 (1U << static_cast<size_t>(req.method()))) == 0)
Ed Tanous1abe55e2018-09-05 08:30:59 -07001153 {
1154 BMCWEB_LOG_DEBUG << "Rule found but method mismatch: " << req.url
1155 << " with " << req.methodString() << "("
Ed Tanous271584a2019-07-09 16:24:22 -07001156 << static_cast<uint32_t>(req.method()) << ") / "
Ed Tanous1abe55e2018-09-05 08:30:59 -07001157 << rules[ruleIndex]->getMethods();
Ed Tanousde5c9f32019-03-26 09:17:55 -07001158 res.result(boost::beast::http::status::not_found);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001159 res.end();
1160 return;
1161 }
1162
1163 BMCWEB_LOG_DEBUG << "Matched rule (upgrade) '" << rules[ruleIndex]->rule
Ed Tanous271584a2019-07-09 16:24:22 -07001164 << "' " << static_cast<uint32_t>(req.method()) << " / "
Ed Tanous1abe55e2018-09-05 08:30:59 -07001165 << rules[ruleIndex]->getMethods();
1166
1167 // any uncaught exceptions become 500s
1168 try
1169 {
1170 rules[ruleIndex]->handleUpgrade(req, res, std::move(adaptor));
1171 }
1172 catch (std::exception& e)
1173 {
1174 BMCWEB_LOG_ERROR << "An uncaught exception occurred: " << e.what();
Ed Tanousde5c9f32019-03-26 09:17:55 -07001175 res.result(boost::beast::http::status::internal_server_error);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001176 res.end();
1177 return;
1178 }
1179 catch (...)
1180 {
1181 BMCWEB_LOG_ERROR
1182 << "An uncaught exception occurred. The type was unknown "
1183 "so no information was available.";
Ed Tanousde5c9f32019-03-26 09:17:55 -07001184 res.result(boost::beast::http::status::internal_server_error);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001185 res.end();
1186 return;
1187 }
Ed Tanous7045c8d2017-04-03 10:04:37 -07001188 }
1189
RAJESWARAN THILLAIGOVINDAN61dbeef2019-12-13 04:26:54 -06001190 void handle(Request& req, Response& res)
Ed Tanous1abe55e2018-09-05 08:30:59 -07001191 {
Ed Tanous271584a2019-07-09 16:24:22 -07001192 if (static_cast<size_t>(req.method()) >= perMethods.size())
Ed Tanous888880a2020-08-24 13:48:50 -07001193 {
1194 res.result(boost::beast::http::status::not_found);
1195 res.end();
Tanousf00032d2018-11-05 01:18:10 -03001196 return;
Ed Tanous888880a2020-08-24 13:48:50 -07001197 }
Ed Tanous271584a2019-07-09 16:24:22 -07001198 PerMethod& perMethod = perMethods[static_cast<size_t>(req.method())];
Tanousf00032d2018-11-05 01:18:10 -03001199 Trie& trie = perMethod.trie;
1200 std::vector<BaseRule*>& rules = perMethod.rules;
1201
1202 const std::pair<unsigned, RoutingParams>& found = trie.find(req.url);
Ed Tanous7045c8d2017-04-03 10:04:37 -07001203
Ed Tanous1abe55e2018-09-05 08:30:59 -07001204 unsigned ruleIndex = found.first;
1205
1206 if (!ruleIndex)
1207 {
Ed Tanous2634dcd2019-03-26 09:28:06 -07001208 // Check to see if this url exists at any verb
1209 for (const PerMethod& p : perMethods)
1210 {
Ed Tanous23a21a12020-07-25 04:45:05 +00001211 const std::pair<unsigned, RoutingParams>& found2 =
Ed Tanous2634dcd2019-03-26 09:28:06 -07001212 p.trie.find(req.url);
Ed Tanous23a21a12020-07-25 04:45:05 +00001213 if (found2.first > 0)
Ed Tanous2634dcd2019-03-26 09:28:06 -07001214 {
1215 res.result(boost::beast::http::status::method_not_allowed);
1216 res.end();
1217 return;
1218 }
1219 }
Ed Tanous1abe55e2018-09-05 08:30:59 -07001220 BMCWEB_LOG_DEBUG << "Cannot match rules " << req.url;
1221 res.result(boost::beast::http::status::not_found);
1222 res.end();
1223 return;
1224 }
1225
1226 if (ruleIndex >= rules.size())
1227 throw std::runtime_error("Trie internal structure corrupted!");
1228
1229 if (ruleIndex == ruleSpecialRedirectSlash)
1230 {
1231 BMCWEB_LOG_INFO << "Redirecting to a url with trailing slash: "
1232 << req.url;
Ed Tanousde5c9f32019-03-26 09:17:55 -07001233 res.result(boost::beast::http::status::moved_permanently);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001234
1235 // TODO absolute url building
1236 if (req.getHeaderValue("Host").empty())
1237 {
1238 res.addHeader("Location", std::string(req.url) + "/");
1239 }
1240 else
1241 {
1242 res.addHeader("Location",
1243 (req.isSecure ? "https://" : "http://") +
1244 std::string(req.getHeaderValue("Host")) +
1245 std::string(req.url) + "/");
1246 }
1247 res.end();
1248 return;
1249 }
1250
Ed Tanous271584a2019-07-09 16:24:22 -07001251 if ((rules[ruleIndex]->getMethods() &
1252 (1U << static_cast<uint32_t>(req.method()))) == 0)
Ed Tanous1abe55e2018-09-05 08:30:59 -07001253 {
1254 BMCWEB_LOG_DEBUG << "Rule found but method mismatch: " << req.url
1255 << " with " << req.methodString() << "("
Ed Tanous271584a2019-07-09 16:24:22 -07001256 << static_cast<uint32_t>(req.method()) << ") / "
Ed Tanous1abe55e2018-09-05 08:30:59 -07001257 << rules[ruleIndex]->getMethods();
Ed Tanousde5c9f32019-03-26 09:17:55 -07001258 res.result(boost::beast::http::status::method_not_allowed);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001259 res.end();
1260 return;
1261 }
1262
1263 BMCWEB_LOG_DEBUG << "Matched rule '" << rules[ruleIndex]->rule << "' "
Ed Tanous271584a2019-07-09 16:24:22 -07001264 << static_cast<uint32_t>(req.method()) << " / "
Ed Tanous1abe55e2018-09-05 08:30:59 -07001265 << rules[ruleIndex]->getMethods();
1266
RAJESWARAN THILLAIGOVINDAN61dbeef2019-12-13 04:26:54 -06001267 if (req.session == nullptr)
James Feist7166bf02019-12-10 16:52:14 +00001268 {
1269 rules[ruleIndex]->handle(req, res, found.second);
James Feist7166bf02019-12-10 16:52:14 +00001270 return;
1271 }
RAJESWARAN THILLAIGOVINDAN61dbeef2019-12-13 04:26:54 -06001272
1273 crow::connections::systemBus->async_method_call(
1274 [&req, &res, &rules, ruleIndex, found](
1275 const boost::system::error_code ec,
1276 std::map<std::string, std::variant<bool, std::string,
1277 std::vector<std::string>>>
1278 userInfo) {
1279 if (ec)
1280 {
1281 BMCWEB_LOG_ERROR << "GetUserInfo failed...";
1282 res.result(
1283 boost::beast::http::status::internal_server_error);
1284 res.end();
1285 return;
1286 }
1287
1288 const std::string* userRolePtr = nullptr;
1289 auto userInfoIter = userInfo.find("UserPrivilege");
1290 if (userInfoIter != userInfo.end())
1291 {
1292 userRolePtr =
1293 std::get_if<std::string>(&userInfoIter->second);
1294 }
1295
1296 std::string userRole{};
1297 if (userRolePtr != nullptr)
1298 {
1299 userRole = *userRolePtr;
1300 BMCWEB_LOG_DEBUG << "userName = " << req.session->username
1301 << " userRole = " << *userRolePtr;
1302 }
1303
Joseph Reynolds3bf4e632020-02-06 14:44:32 -06001304 bool* remoteUserPtr = nullptr;
1305 auto remoteUserIter = userInfo.find("RemoteUser");
1306 if (remoteUserIter != userInfo.end())
1307 {
1308 remoteUserPtr = std::get_if<bool>(&remoteUserIter->second);
1309 }
1310 if (remoteUserPtr == nullptr)
1311 {
1312 BMCWEB_LOG_ERROR
1313 << "RemoteUser property missing or wrong type";
1314 res.result(
1315 boost::beast::http::status::internal_server_error);
1316 res.end();
1317 return;
1318 }
1319 bool remoteUser = *remoteUserPtr;
1320
1321 bool passwordExpired = false; // default for remote user
1322 if (!remoteUser)
1323 {
1324 bool* passwordExpiredPtr = nullptr;
1325 auto passwordExpiredIter =
1326 userInfo.find("UserPasswordExpired");
1327 if (passwordExpiredIter != userInfo.end())
1328 {
1329 passwordExpiredPtr =
1330 std::get_if<bool>(&passwordExpiredIter->second);
1331 }
1332 if (passwordExpiredPtr != nullptr)
1333 {
1334 passwordExpired = *passwordExpiredPtr;
1335 }
1336 else
1337 {
1338 BMCWEB_LOG_ERROR
1339 << "UserPasswordExpired property is expected for"
1340 " local user but is missing or wrong type";
1341 res.result(
1342 boost::beast::http::status::internal_server_error);
1343 res.end();
1344 return;
1345 }
1346 }
1347
Ed Tanous23a21a12020-07-25 04:45:05 +00001348 // Get the userprivileges from the role
RAJESWARAN THILLAIGOVINDAN61dbeef2019-12-13 04:26:54 -06001349 redfish::Privileges userPrivileges =
1350 redfish::getUserPrivileges(userRole);
1351
Joseph Reynolds3bf4e632020-02-06 14:44:32 -06001352 // Set isConfigureSelfOnly based on D-Bus results. This
1353 // ignores the results from both pamAuthenticateUser and the
1354 // value from any previous use of this session.
1355 req.session->isConfigureSelfOnly = passwordExpired;
1356
Ed Tanous23a21a12020-07-25 04:45:05 +00001357 // Modifyprivileges if isConfigureSelfOnly.
Joseph Reynolds3bf4e632020-02-06 14:44:32 -06001358 if (req.session->isConfigureSelfOnly)
1359 {
Ed Tanous23a21a12020-07-25 04:45:05 +00001360 // Remove allprivileges except ConfigureSelf
Joseph Reynolds3bf4e632020-02-06 14:44:32 -06001361 userPrivileges = userPrivileges.intersection(
1362 redfish::Privileges{"ConfigureSelf"});
1363 BMCWEB_LOG_DEBUG << "Operation limited to ConfigureSelf";
1364 }
1365
RAJESWARAN THILLAIGOVINDAN61dbeef2019-12-13 04:26:54 -06001366 if (!rules[ruleIndex]->checkPrivileges(userPrivileges))
1367 {
1368 res.result(boost::beast::http::status::forbidden);
Joseph Reynolds3bf4e632020-02-06 14:44:32 -06001369 if (req.session->isConfigureSelfOnly)
1370 {
1371 redfish::messages::passwordChangeRequired(
1372 res, "/redfish/v1/AccountService/Accounts/" +
1373 req.session->username);
1374 }
RAJESWARAN THILLAIGOVINDAN61dbeef2019-12-13 04:26:54 -06001375 res.end();
1376 return;
1377 }
1378
1379 req.userRole = userRole;
1380
1381 rules[ruleIndex]->handle(req, res, found.second);
1382 },
1383 "xyz.openbmc_project.User.Manager", "/xyz/openbmc_project/user",
1384 "xyz.openbmc_project.User.Manager", "GetUserInfo",
1385 req.session->username);
Ed Tanous7045c8d2017-04-03 10:04:37 -07001386 }
Ed Tanous7045c8d2017-04-03 10:04:37 -07001387
Ed Tanous1abe55e2018-09-05 08:30:59 -07001388 void debugPrint()
1389 {
Ed Tanous271584a2019-07-09 16:24:22 -07001390 for (size_t i = 0; i < perMethods.size(); i++)
Tanousf00032d2018-11-05 01:18:10 -03001391 {
Ed Tanous23a21a12020-07-25 04:45:05 +00001392 BMCWEB_LOG_DEBUG << boost::beast::http::to_string(
1393 static_cast<boost::beast::http::verb>(i));
Tanousf00032d2018-11-05 01:18:10 -03001394 perMethods[i].trie.debugPrint();
1395 }
Ed Tanous3dac7492017-08-02 13:46:20 -07001396 }
Ed Tanousb4a7bfa2017-04-04 17:23:00 -07001397
Ed Tanous1abe55e2018-09-05 08:30:59 -07001398 std::vector<const std::string*> getRoutes(const std::string& parent)
1399 {
Ed Tanous1abe55e2018-09-05 08:30:59 -07001400 std::vector<const std::string*> ret;
Tanousf00032d2018-11-05 01:18:10 -03001401
1402 for (const PerMethod& pm : perMethods)
Ed Tanous1abe55e2018-09-05 08:30:59 -07001403 {
Tanousf00032d2018-11-05 01:18:10 -03001404 std::vector<unsigned> x;
1405 pm.trie.findRouteIndexes(parent, x);
1406 for (unsigned index : x)
1407 {
1408 ret.push_back(&pm.rules[index]->rule);
1409 }
Ed Tanous1abe55e2018-09-05 08:30:59 -07001410 }
1411 return ret;
1412 }
1413
1414 private:
Tanousf00032d2018-11-05 01:18:10 -03001415 struct PerMethod
1416 {
1417 std::vector<BaseRule*> rules;
1418 Trie trie;
1419 // rule index 0, 1 has special meaning; preallocate it to avoid
1420 // duplication.
1421 PerMethod() : rules(2)
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001422 {}
Tanousf00032d2018-11-05 01:18:10 -03001423 };
Ed Tanous888880a2020-08-24 13:48:50 -07001424
1425 const static size_t maxHttpVerbCount =
1426 static_cast<size_t>(boost::beast::http::verb::trace);
1427
Tanousf00032d2018-11-05 01:18:10 -03001428 std::array<PerMethod, maxHttpVerbCount> perMethods;
1429 std::vector<std::unique_ptr<BaseRule>> allRules;
Ed Tanous7045c8d2017-04-03 10:04:37 -07001430};
Ed Tanous1abe55e2018-09-05 08:30:59 -07001431} // namespace crow