|  | #pragma once | 
|  | #include "baserule.hpp" | 
|  | #include "ruleparametertraits.hpp" | 
|  | #include "websocket.hpp" | 
|  |  | 
|  | #include <boost/beast/http/verb.hpp> | 
|  |  | 
|  | #include <functional> | 
|  | #include <limits> | 
|  | #include <string> | 
|  | #include <type_traits> | 
|  |  | 
|  | namespace crow | 
|  | { | 
|  | namespace detail | 
|  | { | 
|  | namespace routing_handler_call_helper | 
|  | { | 
|  |  | 
|  | template <typename Func, typename... ArgsWrapped> | 
|  | struct Wrapped | 
|  | { | 
|  | template <typename... Args> | 
|  | void set(Func f) | 
|  | { | 
|  | handler = std::move(f); | 
|  | } | 
|  |  | 
|  | std::function<void(ArgsWrapped...)> handler; | 
|  |  | 
|  | void operator()(const Request& req, | 
|  | const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, | 
|  | const std::vector<std::string>& params) | 
|  | { | 
|  | if constexpr (sizeof...(ArgsWrapped) == 2) | 
|  | { | 
|  | handler(req, asyncResp); | 
|  | } | 
|  | else if constexpr (sizeof...(ArgsWrapped) == 3) | 
|  | { | 
|  | handler(req, asyncResp, params[0]); | 
|  | } | 
|  | else if constexpr (sizeof...(ArgsWrapped) == 4) | 
|  | { | 
|  | handler(req, asyncResp, params[0], params[1]); | 
|  | } | 
|  | else if constexpr (sizeof...(ArgsWrapped) == 5) | 
|  | { | 
|  | handler(req, asyncResp, params[0], params[1], params[2]); | 
|  | } | 
|  | else if constexpr (sizeof...(ArgsWrapped) == 6) | 
|  | { | 
|  | handler(req, asyncResp, params[0], params[1], params[2], params[3]); | 
|  | } | 
|  | else if constexpr (sizeof...(ArgsWrapped) == 7) | 
|  | { | 
|  | handler(req, asyncResp, params[0], params[1], params[2], params[3], | 
|  | params[4]); | 
|  | } | 
|  | } | 
|  | }; | 
|  | } // namespace routing_handler_call_helper | 
|  | } // namespace detail | 
|  |  | 
|  | class DynamicRule : public BaseRule, public RuleParameterTraits<DynamicRule> | 
|  | { | 
|  | public: | 
|  | explicit DynamicRule(const std::string& ruleIn) : BaseRule(ruleIn) {} | 
|  |  | 
|  | void validate() override | 
|  | { | 
|  | if (!erasedHandler) | 
|  | { | 
|  | throw std::runtime_error("no handler for url " + rule); | 
|  | } | 
|  | } | 
|  |  | 
|  | void handle(const Request& req, | 
|  | const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, | 
|  | const std::vector<std::string>& params) override | 
|  | { | 
|  | erasedHandler(req, asyncResp, params); | 
|  | } | 
|  |  | 
|  | template <typename Func> | 
|  | void operator()(Func f) | 
|  | { | 
|  | using boost::callable_traits::args_t; | 
|  | constexpr size_t arity = std::tuple_size<args_t<Func>>::value; | 
|  | constexpr auto is = std::make_integer_sequence<unsigned, arity>{}; | 
|  | erasedHandler = wrap(std::move(f), is); | 
|  | } | 
|  |  | 
|  | // enable_if Arg1 == request && Arg2 == Response | 
|  | // enable_if Arg1 == request && Arg2 != response | 
|  | // enable_if Arg1 != request | 
|  |  | 
|  | template <typename Func, unsigned... Indices> | 
|  | std::function<void(const Request&, | 
|  | const std::shared_ptr<bmcweb::AsyncResp>&, | 
|  | const std::vector<std::string>&)> | 
|  | wrap(Func f, std::integer_sequence<unsigned, Indices...> /*is*/) | 
|  | { | 
|  | using function_t = crow::utility::FunctionTraits<Func>; | 
|  |  | 
|  | auto ret = detail::routing_handler_call_helper::Wrapped< | 
|  | Func, typename function_t::template arg<Indices>...>(); | 
|  | ret.template set<typename function_t::template arg<Indices>...>( | 
|  | std::move(f)); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | private: | 
|  | std::function<void(const Request&, | 
|  | const std::shared_ptr<bmcweb::AsyncResp>&, | 
|  | const std::vector<std::string>&)> | 
|  | erasedHandler; | 
|  | }; | 
|  |  | 
|  | } // namespace crow |