blob: 6d28405df62a82d27306adb392f14aaefadd5ec7 [file] [log] [blame]
Ed Tanous7045c8d2017-04-03 10:04:37 -07001#pragma once
Adriana Kobylak0e1cf262019-12-05 13:57:57 -06002#include "config.h"
3
Ed Tanous1abe55e2018-09-05 08:30:59 -07004#include "http_utility.hpp"
5
Ed Tanous7045c8d2017-04-03 10:04:37 -07006#include <atomic>
Ed Tanouse0d918b2018-03-27 17:41:04 -07007#include <boost/algorithm/string.hpp>
Ed Tanous257f5792018-03-17 14:40:09 -07008#include <boost/algorithm/string/predicate.hpp>
Ed Tanous8f626352018-12-19 14:51:54 -08009#include <boost/asio/io_context.hpp>
Ed Tanous3112a142018-11-29 15:45:10 -080010#include <boost/asio/ip/tcp.hpp>
Ed Tanous2f1ebcd2019-02-13 19:39:07 -080011#include <boost/asio/ssl.hpp>
Ed Tanous3112a142018-11-29 15:45:10 -080012#include <boost/beast/core/flat_static_buffer.hpp>
Ed Tanouse278c182019-03-13 16:23:37 -070013#if BOOST_VERSION >= 107000
14#include <boost/beast/ssl/ssl_stream.hpp>
15#else
Ed Tanous2f1ebcd2019-02-13 19:39:07 -080016#include <boost/beast/experimental/core/ssl_stream.hpp>
Ed Tanouse278c182019-03-13 16:23:37 -070017#endif
Ed Tanouse0d918b2018-03-27 17:41:04 -070018#include <boost/beast/http.hpp>
19#include <boost/beast/websocket.hpp>
Ed Tanous1abe55e2018-09-05 08:30:59 -070020#include <chrono>
Ed Tanous1abe55e2018-09-05 08:30:59 -070021#include <vector>
22
Ed Tanousc94ad492019-10-10 15:39:33 -070023#include "http_response.h"
24#include "logging.h"
25#include "middleware_context.h"
26#include "timer_queue.h"
27#include "utility.h"
Ed Tanous7045c8d2017-04-03 10:04:37 -070028
Ed Tanous1abe55e2018-09-05 08:30:59 -070029namespace crow
30{
Ed Tanous257f5792018-03-17 14:40:09 -070031
Ed Tanous1abe55e2018-09-05 08:30:59 -070032inline void prettyPrintJson(crow::Response& res)
33{
Jason M. Bills193ad2f2018-09-26 15:08:52 -070034 std::string value = res.jsonValue.dump(4, ' ', true);
Ed Tanousa29c9972018-11-29 15:54:32 -080035 utility::escapeHtml(value);
36 utility::convertToLinks(value);
Ed Tanous1abe55e2018-09-05 08:30:59 -070037 res.body() = "<html>\n"
38 "<head>\n"
39 "<title>Redfish API</title>\n"
40 "<link rel=\"stylesheet\" type=\"text/css\" "
41 "href=\"/styles/default.css\">\n"
42 "<script src=\"/highlight.pack.js\"></script>"
43 "<script>hljs.initHighlightingOnLoad();</script>"
44 "</head>\n"
45 "<body>\n"
46 "<div style=\"max-width: 576px;margin:0 auto;\">\n"
47 "<img src=\"/DMTF_Redfish_logo_2017.svg\" alt=\"redfish\" "
48 "height=\"406px\" "
49 "width=\"576px\">\n"
50 "<br>\n"
51 "<pre>\n"
52 "<code class=\"json\">" +
53 value +
54 "</code>\n"
55 "</pre>\n"
56 "</div>\n"
57 "</body>\n"
58 "</html>\n";
Ed Tanous93ef5802019-01-03 10:15:41 -080059 res.addHeader("Content-Type", "text/html;charset=UTF-8");
Ed Tanous257f5792018-03-17 14:40:09 -070060}
61
Ed Tanous7045c8d2017-04-03 10:04:37 -070062using namespace boost;
63using tcp = asio::ip::tcp;
64
Ed Tanous1abe55e2018-09-05 08:30:59 -070065namespace detail
66{
67template <typename MW> struct CheckBeforeHandleArity3Const
68{
69 template <typename T,
70 void (T::*)(Request&, Response&, typename MW::Context&) const =
71 &T::beforeHandle>
72 struct Get
73 {
74 };
Ed Tanous7045c8d2017-04-03 10:04:37 -070075};
76
Ed Tanous1abe55e2018-09-05 08:30:59 -070077template <typename MW> struct CheckBeforeHandleArity3
78{
79 template <typename T, void (T::*)(Request&, Response&,
80 typename MW::Context&) = &T::beforeHandle>
81 struct Get
82 {
83 };
Ed Tanous7045c8d2017-04-03 10:04:37 -070084};
85
Ed Tanous1abe55e2018-09-05 08:30:59 -070086template <typename MW> struct CheckAfterHandleArity3Const
87{
88 template <typename T,
89 void (T::*)(Request&, Response&, typename MW::Context&) const =
90 &T::afterHandle>
91 struct Get
92 {
93 };
Ed Tanous7045c8d2017-04-03 10:04:37 -070094};
95
Ed Tanous1abe55e2018-09-05 08:30:59 -070096template <typename MW> struct CheckAfterHandleArity3
97{
98 template <typename T, void (T::*)(Request&, Response&,
99 typename MW::Context&) = &T::afterHandle>
100 struct Get
101 {
102 };
Ed Tanous7045c8d2017-04-03 10:04:37 -0700103};
104
Ed Tanous1abe55e2018-09-05 08:30:59 -0700105template <typename T> struct IsBeforeHandleArity3Impl
106{
107 template <typename C>
108 static std::true_type
109 f(typename CheckBeforeHandleArity3Const<T>::template Get<C>*);
Ed Tanous7045c8d2017-04-03 10:04:37 -0700110
Ed Tanous1abe55e2018-09-05 08:30:59 -0700111 template <typename C>
112 static std::true_type
113 f(typename CheckBeforeHandleArity3<T>::template Get<C>*);
Ed Tanous7045c8d2017-04-03 10:04:37 -0700114
Ed Tanous1abe55e2018-09-05 08:30:59 -0700115 template <typename C> static std::false_type f(...);
Ed Tanous7045c8d2017-04-03 10:04:37 -0700116
Ed Tanous1abe55e2018-09-05 08:30:59 -0700117 public:
Ed Tanous0c838cf2019-10-24 10:01:46 -0700118 static constexpr bool value = decltype(f<T>(nullptr))::value;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700119};
120
Ed Tanous1abe55e2018-09-05 08:30:59 -0700121template <typename T> struct IsAfterHandleArity3Impl
122{
123 template <typename C>
124 static std::true_type
125 f(typename CheckAfterHandleArity3Const<T>::template Get<C>*);
Ed Tanous7045c8d2017-04-03 10:04:37 -0700126
Ed Tanous1abe55e2018-09-05 08:30:59 -0700127 template <typename C>
128 static std::true_type
129 f(typename CheckAfterHandleArity3<T>::template Get<C>*);
Ed Tanous7045c8d2017-04-03 10:04:37 -0700130
Ed Tanous1abe55e2018-09-05 08:30:59 -0700131 template <typename C> static std::false_type f(...);
Ed Tanous7045c8d2017-04-03 10:04:37 -0700132
Ed Tanous1abe55e2018-09-05 08:30:59 -0700133 public:
Ed Tanous0c838cf2019-10-24 10:01:46 -0700134 static constexpr bool value = decltype(f<T>(nullptr))::value;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700135};
136
137template <typename MW, typename Context, typename ParentContext>
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700138typename std::enable_if<!IsBeforeHandleArity3Impl<MW>::value>::type
Ed Tanous1abe55e2018-09-05 08:30:59 -0700139 beforeHandlerCall(MW& mw, Request& req, Response& res, Context& ctx,
140 ParentContext& /*parent_ctx*/)
141{
142 mw.beforeHandle(req, res, ctx.template get<MW>(), ctx);
Ed Tanous7045c8d2017-04-03 10:04:37 -0700143}
144
145template <typename MW, typename Context, typename ParentContext>
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700146typename std::enable_if<IsBeforeHandleArity3Impl<MW>::value>::type
Ed Tanous1abe55e2018-09-05 08:30:59 -0700147 beforeHandlerCall(MW& mw, Request& req, Response& res, Context& ctx,
148 ParentContext& /*parent_ctx*/)
149{
150 mw.beforeHandle(req, res, ctx.template get<MW>());
Ed Tanous7045c8d2017-04-03 10:04:37 -0700151}
152
153template <typename MW, typename Context, typename ParentContext>
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700154typename std::enable_if<!IsAfterHandleArity3Impl<MW>::value>::type
Ed Tanous1abe55e2018-09-05 08:30:59 -0700155 afterHandlerCall(MW& mw, Request& req, Response& res, Context& ctx,
156 ParentContext& /*parent_ctx*/)
157{
158 mw.afterHandle(req, res, ctx.template get<MW>(), ctx);
Ed Tanous7045c8d2017-04-03 10:04:37 -0700159}
160
161template <typename MW, typename Context, typename ParentContext>
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700162typename std::enable_if<IsAfterHandleArity3Impl<MW>::value>::type
Ed Tanous1abe55e2018-09-05 08:30:59 -0700163 afterHandlerCall(MW& mw, Request& req, Response& res, Context& ctx,
164 ParentContext& /*parent_ctx*/)
165{
166 mw.afterHandle(req, res, ctx.template get<MW>());
Ed Tanous7045c8d2017-04-03 10:04:37 -0700167}
168
Ed Tanous271584a2019-07-09 16:24:22 -0700169template <size_t N, typename Context, typename Container, typename CurrentMW,
Ed Tanous7045c8d2017-04-03 10:04:37 -0700170 typename... Middlewares>
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700171bool middlewareCallHelper(Container& middlewares, Request& req, Response& res,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700172 Context& ctx)
173{
174 using parent_context_t = typename Context::template partial<N - 1>;
175 beforeHandlerCall<CurrentMW, Context, parent_context_t>(
Ed Tanous7045c8d2017-04-03 10:04:37 -0700176 std::get<N>(middlewares), req, res, ctx,
177 static_cast<parent_context_t&>(ctx));
Ed Tanous7045c8d2017-04-03 10:04:37 -0700178
Ed Tanous1abe55e2018-09-05 08:30:59 -0700179 if (res.isCompleted())
180 {
181 afterHandlerCall<CurrentMW, Context, parent_context_t>(
182 std::get<N>(middlewares), req, res, ctx,
183 static_cast<parent_context_t&>(ctx));
184 return true;
185 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700186
Ed Tanous1abe55e2018-09-05 08:30:59 -0700187 if (middlewareCallHelper<N + 1, Context, Container, Middlewares...>(
188 middlewares, req, res, ctx))
189 {
190 afterHandlerCall<CurrentMW, Context, parent_context_t>(
191 std::get<N>(middlewares), req, res, ctx,
192 static_cast<parent_context_t&>(ctx));
193 return true;
194 }
195
196 return false;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700197}
198
Ed Tanous271584a2019-07-09 16:24:22 -0700199template <size_t N, typename Context, typename Container>
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700200bool middlewareCallHelper(Container& /*middlewares*/, Request& /*req*/,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700201 Response& /*res*/, Context& /*ctx*/)
202{
203 return false;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700204}
205
Ed Tanous271584a2019-07-09 16:24:22 -0700206template <size_t N, typename Context, typename Container>
Ed Tanous1abe55e2018-09-05 08:30:59 -0700207typename std::enable_if<(N < 0)>::type
208 afterHandlersCallHelper(Container& /*middlewares*/, Context& /*Context*/,
209 Request& /*req*/, Response& /*res*/)
210{
Ed Tanous7045c8d2017-04-03 10:04:37 -0700211}
212
Ed Tanous271584a2019-07-09 16:24:22 -0700213template <size_t N, typename Context, typename Container>
Ed Tanous1abe55e2018-09-05 08:30:59 -0700214typename std::enable_if<(N == 0)>::type
215 afterHandlersCallHelper(Container& middlewares, Context& ctx, Request& req,
216 Response& res)
217{
218 using parent_context_t = typename Context::template partial<N - 1>;
219 using CurrentMW = typename std::tuple_element<
220 N, typename std::remove_reference<Container>::type>::type;
221 afterHandlerCall<CurrentMW, Context, parent_context_t>(
222 std::get<N>(middlewares), req, res, ctx,
223 static_cast<parent_context_t&>(ctx));
Ed Tanous7045c8d2017-04-03 10:04:37 -0700224}
Ed Tanous1abe55e2018-09-05 08:30:59 -0700225
Ed Tanous271584a2019-07-09 16:24:22 -0700226template <size_t N, typename Context, typename Container>
Ed Tanous1abe55e2018-09-05 08:30:59 -0700227typename std::enable_if<(N > 0)>::type
228 afterHandlersCallHelper(Container& middlewares, Context& ctx, Request& req,
229 Response& res)
230{
231 using parent_context_t = typename Context::template partial<N - 1>;
232 using CurrentMW = typename std::tuple_element<
233 N, typename std::remove_reference<Container>::type>::type;
234 afterHandlerCall<CurrentMW, Context, parent_context_t>(
235 std::get<N>(middlewares), req, res, ctx,
236 static_cast<parent_context_t&>(ctx));
237 afterHandlersCallHelper<N - 1, Context, Container>(middlewares, ctx, req,
238 res);
239}
240} // namespace detail
Ed Tanous7045c8d2017-04-03 10:04:37 -0700241
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700242#ifdef BMCWEB_ENABLE_DEBUG
Ed Tanouse0d918b2018-03-27 17:41:04 -0700243static std::atomic<int> connectionCount;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700244#endif
Jennifer Leeacb7cfb2018-06-07 16:08:15 -0700245
Adriana Kobylak0e1cf262019-12-05 13:57:57 -0600246// request body limit size set by the BMCWEB_HTTP_REQ_BODY_LIMIT_MB option
247constexpr unsigned int httpReqBodyLimit =
248 1024 * 1024 * BMCWEB_HTTP_REQ_BODY_LIMIT_MB;
Jennifer Leeacb7cfb2018-06-07 16:08:15 -0700249
Ed Tanous7045c8d2017-04-03 10:04:37 -0700250template <typename Adaptor, typename Handler, typename... Middlewares>
Jan Sowinskiee52ae12020-01-09 16:28:32 +0000251class Connection : public std::enable_shared_from_this<
252 Connection<Adaptor, Handler, Middlewares...>>
Ed Tanous1abe55e2018-09-05 08:30:59 -0700253{
254 public:
Ed Tanous271584a2019-07-09 16:24:22 -0700255 Connection(boost::asio::io_context& ioService, Handler* handlerIn,
256 const std::string& ServerNameIn,
257 std::tuple<Middlewares...>* middlewaresIn,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700258 std::function<std::string()>& get_cached_date_str_f,
Ed Tanous271584a2019-07-09 16:24:22 -0700259 detail::TimerQueue& timerQueueIn, Adaptor adaptorIn) :
Ed Tanousceac6f72018-12-02 11:58:47 -0800260 adaptor(std::move(adaptorIn)),
Ed Tanous271584a2019-07-09 16:24:22 -0700261 handler(handlerIn), serverName(ServerNameIn),
262 middlewares(middlewaresIn), getCachedDateStr(get_cached_date_str_f),
263 timerQueue(timerQueueIn)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700264 {
265 parser.emplace(std::piecewise_construct, std::make_tuple());
Adriana Kobylak0e1cf262019-12-05 13:57:57 -0600266 // Temporarily set by the BMCWEB_HTTP_REQ_BODY_LIMIT_MB variable; Need
267 // to modify uploading/authentication mechanism to a better method that
268 // disallows a DOS attack based on a large file size.
Ed Tanous1abe55e2018-09-05 08:30:59 -0700269 parser->body_limit(httpReqBodyLimit);
270 req.emplace(parser->get());
Kowalski, Kamil55e43f62019-07-10 13:12:57 +0200271
272#ifdef BMCWEB_ENABLE_MUTUAL_TLS_AUTHENTICATION
Zbigniew Kurzynski009c2a42019-11-14 13:37:15 +0100273 auto ca_available = !std::filesystem::is_empty(
274 std::filesystem::path(ensuressl::trustStorePath));
275 if (ca_available && crow::persistent_data::SessionStore::getInstance()
276 .getAuthMethodsConfig()
277 .tls)
278 {
279 adaptor.set_verify_mode(boost::asio::ssl::verify_peer);
280 SSL_set_session_id_context(
281 adaptor.native_handle(),
282 reinterpret_cast<const unsigned char*>(serverName.c_str()),
Zbigniew Kurzynskicac94c52019-11-07 12:55:04 +0100283 static_cast<unsigned int>(serverName.length()));
Zbigniew Kurzynski009c2a42019-11-14 13:37:15 +0100284 BMCWEB_LOG_DEBUG << this << " TLS is enabled on this connection.";
285 }
286
Zbigniew Kurzynski2658d982019-11-19 18:01:08 +0100287 adaptor.set_verify_callback([this](
288 bool preverified,
289 boost::asio::ssl::verify_context& ctx) {
290 // do nothing if TLS is disabled
291 if (!crow::persistent_data::SessionStore::getInstance()
292 .getAuthMethodsConfig()
293 .tls)
294 {
295 BMCWEB_LOG_DEBUG << this << " TLS auth_config is disabled";
Kowalski, Kamil55e43f62019-07-10 13:12:57 +0200296 return true;
Zbigniew Kurzynski2658d982019-11-19 18:01:08 +0100297 }
298
299 // We always return true to allow full auth flow
300 if (!preverified)
301 {
Zbigniew Kurzynski009c2a42019-11-14 13:37:15 +0100302 BMCWEB_LOG_DEBUG << this << " TLS preverification failed.";
Zbigniew Kurzynski2658d982019-11-19 18:01:08 +0100303 return true;
304 }
305
306 X509_STORE_CTX* cts = ctx.native_handle();
307 if (cts == nullptr)
308 {
Zbigniew Kurzynski009c2a42019-11-14 13:37:15 +0100309 BMCWEB_LOG_DEBUG << this << " Cannot get native TLS handle.";
Zbigniew Kurzynski2658d982019-11-19 18:01:08 +0100310 return true;
311 }
312
313 // Get certificate
314 X509* peerCert =
315 X509_STORE_CTX_get_current_cert(ctx.native_handle());
316 if (peerCert == nullptr)
317 {
Zbigniew Kurzynski009c2a42019-11-14 13:37:15 +0100318 BMCWEB_LOG_DEBUG << this
319 << " Cannot get current TLS certificate.";
Zbigniew Kurzynski2658d982019-11-19 18:01:08 +0100320 return true;
321 }
322
323 // Check if certificate is OK
324 int error = X509_STORE_CTX_get_error(cts);
325 if (error != X509_V_OK)
326 {
Zbigniew Kurzynski009c2a42019-11-14 13:37:15 +0100327 BMCWEB_LOG_INFO << this << " Last TLS error is: " << error;
Zbigniew Kurzynski2658d982019-11-19 18:01:08 +0100328 return true;
329 }
330 // Check that we have reached final certificate in chain
331 int32_t depth = X509_STORE_CTX_get_error_depth(cts);
332 if (depth != 0)
333
334 {
335 BMCWEB_LOG_DEBUG
336 << this << " Certificate verification in progress (depth "
337 << depth << "), waiting to reach final depth";
338 return true;
339 }
340
341 BMCWEB_LOG_DEBUG << this
342 << " Certificate verification of final depth";
343
344 // Verify KeyUsage
345 bool isKeyUsageDigitalSignature = false;
346 bool isKeyUsageKeyAgreement = false;
347
348 ASN1_BIT_STRING* usage = static_cast<ASN1_BIT_STRING*>(
349 X509_get_ext_d2i(peerCert, NID_key_usage, NULL, NULL));
350
351 if (usage == nullptr)
352 {
Zbigniew Kurzynski009c2a42019-11-14 13:37:15 +0100353 BMCWEB_LOG_DEBUG << this << " TLS usage is null";
Zbigniew Kurzynski2658d982019-11-19 18:01:08 +0100354 return true;
355 }
356
357 for (int i = 0; i < usage->length; i++)
358 {
359 if (KU_DIGITAL_SIGNATURE & usage->data[i])
360 {
361 isKeyUsageDigitalSignature = true;
362 }
363 if (KU_KEY_AGREEMENT & usage->data[i])
364 {
365 isKeyUsageKeyAgreement = true;
366 }
367 }
368
369 if (!isKeyUsageDigitalSignature || !isKeyUsageKeyAgreement)
370 {
371 BMCWEB_LOG_DEBUG << this
372 << " Certificate ExtendedKeyUsage does "
373 "not allow provided certificate to "
374 "be used for user authentication";
375 return true;
376 }
Zbigniew Kurzynski09d02f82020-03-30 13:41:42 +0200377 ASN1_BIT_STRING_free(usage);
Zbigniew Kurzynski2658d982019-11-19 18:01:08 +0100378
379 // Determine that ExtendedKeyUsage includes Client Auth
380
381 stack_st_ASN1_OBJECT* extUsage = static_cast<stack_st_ASN1_OBJECT*>(
382 X509_get_ext_d2i(peerCert, NID_ext_key_usage, NULL, NULL));
383
384 if (extUsage == nullptr)
385 {
Zbigniew Kurzynski009c2a42019-11-14 13:37:15 +0100386 BMCWEB_LOG_DEBUG << this << " TLS extUsage is null";
Zbigniew Kurzynski2658d982019-11-19 18:01:08 +0100387 return true;
388 }
389
390 bool isExKeyUsageClientAuth = false;
391 for (int i = 0; i < sk_ASN1_OBJECT_num(extUsage); i++)
392 {
393 if (NID_client_auth ==
394 OBJ_obj2nid(sk_ASN1_OBJECT_value(extUsage, i)))
395 {
396 isExKeyUsageClientAuth = true;
397 break;
398 }
399 }
Zbigniew Kurzynski09d02f82020-03-30 13:41:42 +0200400 sk_ASN1_OBJECT_free(extUsage);
Zbigniew Kurzynski2658d982019-11-19 18:01:08 +0100401
402 // Certificate has to have proper key usages set
403 if (!isExKeyUsageClientAuth)
404 {
405 BMCWEB_LOG_DEBUG << this
406 << " Certificate ExtendedKeyUsage does "
407 "not allow provided certificate to "
408 "be used for user authentication";
409 return true;
410 }
411 std::string sslUser;
412 // Extract username contained in CommonName
413 sslUser.resize(256, '\0');
414
415 int status = X509_NAME_get_text_by_NID(
416 X509_get_subject_name(peerCert), NID_commonName, sslUser.data(),
417 static_cast<int>(sslUser.size()));
418
419 if (status == -1)
420 {
Zbigniew Kurzynski009c2a42019-11-14 13:37:15 +0100421 BMCWEB_LOG_DEBUG
422 << this << " TLS cannot get username to create session";
Zbigniew Kurzynski2658d982019-11-19 18:01:08 +0100423 return true;
424 }
425
426 size_t lastChar = sslUser.find('\0');
427 if (lastChar == std::string::npos || lastChar == 0)
428 {
Zbigniew Kurzynski009c2a42019-11-14 13:37:15 +0100429 BMCWEB_LOG_DEBUG << this << " Invalid TLS user name";
Zbigniew Kurzynski2658d982019-11-19 18:01:08 +0100430 return true;
431 }
432 sslUser.resize(lastChar);
433
434 session = persistent_data::SessionStore::getInstance()
435 .generateUserSession(
436 sslUser,
437 crow::persistent_data::PersistenceType::TIMEOUT);
Zbigniew Kurzynski009c2a42019-11-14 13:37:15 +0100438 if (auto sp = session.lock())
439 {
440 BMCWEB_LOG_DEBUG << this
441 << " Generating TLS session: " << sp->uniqueId;
442 }
Zbigniew Kurzynski2658d982019-11-19 18:01:08 +0100443 return true;
444 });
Kowalski, Kamil55e43f62019-07-10 13:12:57 +0200445#endif // BMCWEB_ENABLE_MUTUAL_TLS_AUTHENTICATION
446
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700447#ifdef BMCWEB_ENABLE_DEBUG
Ed Tanous1abe55e2018-09-05 08:30:59 -0700448 connectionCount++;
449 BMCWEB_LOG_DEBUG << this << " Connection open, total "
450 << connectionCount;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700451#endif
Ed Tanous1abe55e2018-09-05 08:30:59 -0700452 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700453
Ed Tanous1abe55e2018-09-05 08:30:59 -0700454 ~Connection()
455 {
456 res.completeRequestHandler = nullptr;
457 cancelDeadlineTimer();
Ed Tanous55c7b7a2018-05-22 15:27:24 -0700458#ifdef BMCWEB_ENABLE_DEBUG
Ed Tanous1abe55e2018-09-05 08:30:59 -0700459 connectionCount--;
460 BMCWEB_LOG_DEBUG << this << " Connection closed, total "
461 << connectionCount;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700462#endif
Ed Tanous7045c8d2017-04-03 10:04:37 -0700463 }
464
Ed Tanousceac6f72018-12-02 11:58:47 -0800465 Adaptor& socket()
Ed Tanous1abe55e2018-09-05 08:30:59 -0700466 {
Ed Tanousceac6f72018-12-02 11:58:47 -0800467 return adaptor;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700468 }
469
Ed Tanous1abe55e2018-09-05 08:30:59 -0700470 void start()
471 {
Ed Tanous7045c8d2017-04-03 10:04:37 -0700472
Ed Tanousceac6f72018-12-02 11:58:47 -0800473 startDeadline();
474 // TODO(ed) Abstract this to a more clever class with the idea of an
475 // asynchronous "start"
476 if constexpr (std::is_same_v<Adaptor,
477 boost::beast::ssl_stream<
478 boost::asio::ip::tcp::socket>>)
479 {
Jan Sowinskiee52ae12020-01-09 16:28:32 +0000480 adaptor.async_handshake(boost::asio::ssl::stream_base::server,
481 [this, self(shared_from_this())](
482 const boost::system::error_code& ec) {
483 if (ec)
484 {
485 return;
486 }
487 doReadHeaders();
488 });
Ed Tanousceac6f72018-12-02 11:58:47 -0800489 }
490 else
491 {
492 doReadHeaders();
493 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700494 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700495
Ed Tanous1abe55e2018-09-05 08:30:59 -0700496 void handle()
497 {
498 cancelDeadlineTimer();
499 bool isInvalidRequest = false;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700500
Ed Tanous1abe55e2018-09-05 08:30:59 -0700501 // Check for HTTP version 1.1.
502 if (req->version() == 11)
503 {
504 if (req->getHeaderValue(boost::beast::http::field::host).empty())
505 {
506 isInvalidRequest = true;
Ed Tanousde5c9f32019-03-26 09:17:55 -0700507 res.result(boost::beast::http::status::bad_request);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700508 }
509 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700510
Ed Tanouse278c182019-03-13 16:23:37 -0700511 BMCWEB_LOG_INFO << "Request: "
512 << " " << this << " HTTP/" << req->version() / 10 << "."
513 << req->version() % 10 << ' ' << req->methodString()
514 << " " << req->target();
Ed Tanous7045c8d2017-04-03 10:04:37 -0700515
Ed Tanous1abe55e2018-09-05 08:30:59 -0700516 needToCallAfterHandlers = false;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700517
Ed Tanous1abe55e2018-09-05 08:30:59 -0700518 if (!isInvalidRequest)
519 {
raviteja-b4722efe2020-02-03 12:23:18 -0600520 req->socket = [this, self = shared_from_this()]() -> Adaptor& {
521 return self->socket();
522 };
523
Ed Tanous1abe55e2018-09-05 08:30:59 -0700524 res.completeRequestHandler = [] {};
Ed Tanouse278c182019-03-13 16:23:37 -0700525 res.isAliveHelper = [this]() -> bool { return isAlive(); };
Ed Tanous7045c8d2017-04-03 10:04:37 -0700526
Ed Tanous1abe55e2018-09-05 08:30:59 -0700527 ctx = detail::Context<Middlewares...>();
Ed Tanouse278c182019-03-13 16:23:37 -0700528 req->middlewareContext = static_cast<void*>(&ctx);
529 req->ioService = static_cast<decltype(req->ioService)>(
530 &adaptor.get_executor().context());
Kowalski, Kamil55e43f62019-07-10 13:12:57 +0200531
532#ifdef BMCWEB_ENABLE_MUTUAL_TLS_AUTHENTICATION
533 if (auto sp = session.lock())
534 {
Zbigniew Kurzynski009c2a42019-11-14 13:37:15 +0100535 // set cookie only if this is req from the browser.
536 if (req->getHeaderValue("User-Agent").empty())
537 {
538 BMCWEB_LOG_DEBUG << this << " TLS session: " << sp->uniqueId
539 << " will be used for this request.";
540 req->session = sp;
541 }
542 else
543 {
544 std::string_view cookieValue =
545 req->getHeaderValue("Cookie");
546 if (cookieValue.empty() ||
547 cookieValue.find("SESSION=") == std::string::npos)
548 {
549 res.addHeader("Set-Cookie",
550 "XSRF-TOKEN=" + sp->csrfToken +
551 "; Secure\r\nSet-Cookie: SESSION=" +
552 sp->sessionToken +
Zbigniew Kurzynski26139a52019-12-11 19:11:18 +0100553 "; Secure; HttpOnly\r\nSet-Cookie: "
554 "IsAuthenticated=true; Secure");
Zbigniew Kurzynski009c2a42019-11-14 13:37:15 +0100555 BMCWEB_LOG_DEBUG
556 << this << " TLS session: " << sp->uniqueId
557 << " with cookie will be used for this request.";
558 req->session = sp;
559 }
560 }
Kowalski, Kamil55e43f62019-07-10 13:12:57 +0200561 }
562#endif // BMCWEB_ENABLE_MUTUAL_TLS_AUTHENTICATION
563
Ed Tanous1abe55e2018-09-05 08:30:59 -0700564 detail::middlewareCallHelper<
Ed Tanous271584a2019-07-09 16:24:22 -0700565 0U, decltype(ctx), decltype(*middlewares), Middlewares...>(
Ed Tanous1abe55e2018-09-05 08:30:59 -0700566 *middlewares, *req, res, ctx);
Ed Tanous7045c8d2017-04-03 10:04:37 -0700567
Ed Tanous1abe55e2018-09-05 08:30:59 -0700568 if (!res.completed)
569 {
Jan Sowinskiee52ae12020-01-09 16:28:32 +0000570 needToCallAfterHandlers = true;
571 res.completeRequestHandler = [self(shared_from_this())] {
572 self->completeRequest();
573 };
Ed Tanous1abe55e2018-09-05 08:30:59 -0700574 if (req->isUpgrade() &&
575 boost::iequals(
576 req->getHeaderValue(boost::beast::http::field::upgrade),
577 "websocket"))
578 {
579 handler->handleUpgrade(*req, res, std::move(adaptor));
Jan Sowinskiee52ae12020-01-09 16:28:32 +0000580 // delete lambda with self shared_ptr
581 // to enable connection destruction
582 res.completeRequestHandler = nullptr;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700583 return;
584 }
Ed Tanous1abe55e2018-09-05 08:30:59 -0700585 handler->handle(*req, res);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700586 }
587 else
588 {
589 completeRequest();
590 }
591 }
592 else
593 {
594 completeRequest();
595 }
596 }
Ed Tanouse0d918b2018-03-27 17:41:04 -0700597
Ed Tanouse278c182019-03-13 16:23:37 -0700598 bool isAlive()
599 {
600
601 if constexpr (std::is_same_v<Adaptor,
602 boost::beast::ssl_stream<
603 boost::asio::ip::tcp::socket>>)
604 {
605 return adaptor.next_layer().is_open();
606 }
607 else
608 {
609 return adaptor.is_open();
610 }
611 }
612 void close()
613 {
Ed Tanouse278c182019-03-13 16:23:37 -0700614 if constexpr (std::is_same_v<Adaptor,
615 boost::beast::ssl_stream<
616 boost::asio::ip::tcp::socket>>)
617 {
618 adaptor.next_layer().close();
Kowalski, Kamil55e43f62019-07-10 13:12:57 +0200619#ifdef BMCWEB_ENABLE_MUTUAL_TLS_AUTHENTICATION
620 if (auto sp = session.lock())
621 {
Zbigniew Kurzynski009c2a42019-11-14 13:37:15 +0100622 BMCWEB_LOG_DEBUG << this
623 << " Removing TLS session: " << sp->uniqueId;
Kowalski, Kamil55e43f62019-07-10 13:12:57 +0200624 persistent_data::SessionStore::getInstance().removeSession(sp);
625 }
626#endif // BMCWEB_ENABLE_MUTUAL_TLS_AUTHENTICATION
Ed Tanouse278c182019-03-13 16:23:37 -0700627 }
628 else
629 {
630 adaptor.close();
631 }
632 }
633
Ed Tanous1abe55e2018-09-05 08:30:59 -0700634 void completeRequest()
635 {
636 BMCWEB_LOG_INFO << "Response: " << this << ' ' << req->url << ' '
637 << res.resultInt() << " keepalive=" << req->keepAlive();
Ed Tanous7045c8d2017-04-03 10:04:37 -0700638
Ed Tanous1abe55e2018-09-05 08:30:59 -0700639 if (needToCallAfterHandlers)
640 {
641 needToCallAfterHandlers = false;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700642
Ed Tanous1abe55e2018-09-05 08:30:59 -0700643 // call all afterHandler of middlewares
Ed Tanous271584a2019-07-09 16:24:22 -0700644 detail::afterHandlersCallHelper<sizeof...(Middlewares) - 1,
Ed Tanousb01bf292019-03-25 19:25:26 +0000645 decltype(ctx),
646 decltype(*middlewares)>(
647 *middlewares, ctx, *req, res);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700648 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700649
Ed Tanouse278c182019-03-13 16:23:37 -0700650 if (!isAlive())
Ed Tanous1abe55e2018-09-05 08:30:59 -0700651 {
652 // BMCWEB_LOG_DEBUG << this << " delete (socket is closed) " <<
653 // isReading
654 // << ' ' << isWriting;
655 // delete this;
Jan Sowinskiee52ae12020-01-09 16:28:32 +0000656
657 // delete lambda with self shared_ptr
658 // to enable connection destruction
659 res.completeRequestHandler = nullptr;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700660 return;
661 }
662 if (res.body().empty() && !res.jsonValue.empty())
663 {
664 if (http_helpers::requestPrefersHtml(*req))
665 {
666 prettyPrintJson(res);
667 }
668 else
669 {
670 res.jsonMode();
Jason M. Bills193ad2f2018-09-26 15:08:52 -0700671 res.body() = res.jsonValue.dump(2, ' ', true);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700672 }
673 }
Ed Tanous7045c8d2017-04-03 10:04:37 -0700674
Ed Tanous1abe55e2018-09-05 08:30:59 -0700675 if (res.resultInt() >= 400 && res.body().empty())
676 {
677 res.body() = std::string(res.reason());
678 }
Ed Tanous6295bec2019-09-03 10:11:01 -0700679
680 if (res.result() == boost::beast::http::status::no_content)
681 {
682 // Boost beast throws if content is provided on a no-content
683 // response. Ideally, this would never happen, but in the case that
684 // it does, we don't want to throw.
685 BMCWEB_LOG_CRITICAL
Zbigniew Kurzynski2658d982019-11-19 18:01:08 +0100686 << this << " Response content provided but code was no-content";
Ed Tanous6295bec2019-09-03 10:11:01 -0700687 res.body().clear();
688 }
689
Ed Tanous1abe55e2018-09-05 08:30:59 -0700690 res.addHeader(boost::beast::http::field::server, serverName);
691 res.addHeader(boost::beast::http::field::date, getCachedDateStr());
692
693 res.keepAlive(req->keepAlive());
694
695 doWrite();
Jan Sowinskiee52ae12020-01-09 16:28:32 +0000696
697 // delete lambda with self shared_ptr
698 // to enable connection destruction
699 res.completeRequestHandler = nullptr;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700700 }
701
702 private:
703 void doReadHeaders()
704 {
Ed Tanous1abe55e2018-09-05 08:30:59 -0700705 BMCWEB_LOG_DEBUG << this << " doReadHeaders";
706
707 // Clean up any previous Connection.
708 boost::beast::http::async_read_header(
Ed Tanousceac6f72018-12-02 11:58:47 -0800709 adaptor, buffer, *parser,
Jan Sowinskiee52ae12020-01-09 16:28:32 +0000710 [this,
711 self(shared_from_this())](const boost::system::error_code& ec,
712 std::size_t bytes_transferred) {
Ed Tanous1abe55e2018-09-05 08:30:59 -0700713 BMCWEB_LOG_ERROR << this << " async_read_header "
714 << bytes_transferred << " Bytes";
715 bool errorWhileReading = false;
716 if (ec)
717 {
718 errorWhileReading = true;
719 BMCWEB_LOG_ERROR
720 << this << " Error while reading: " << ec.message();
721 }
722 else
723 {
724 // if the adaptor isn't open anymore, and wasn't handed to a
725 // websocket, treat as an error
Ed Tanouse278c182019-03-13 16:23:37 -0700726 if (!isAlive() && !req->isUpgrade())
Ed Tanous1abe55e2018-09-05 08:30:59 -0700727 {
728 errorWhileReading = true;
729 }
730 }
731
732 if (errorWhileReading)
733 {
734 cancelDeadlineTimer();
Ed Tanouse278c182019-03-13 16:23:37 -0700735 close();
Ed Tanous1abe55e2018-09-05 08:30:59 -0700736 BMCWEB_LOG_DEBUG << this << " from read(1)";
Ed Tanous1abe55e2018-09-05 08:30:59 -0700737 return;
738 }
739
740 // Compute the url parameters for the request
741 req->url = req->target();
742 std::size_t index = req->url.find("?");
Ed Tanous39e77502019-03-04 17:35:53 -0800743 if (index != std::string_view::npos)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700744 {
Jason M. Bills43fcbe52018-10-16 15:19:20 -0700745 req->url = req->url.substr(0, index);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700746 }
747 req->urlParams = QueryString(std::string(req->target()));
748 doRead();
749 });
750 }
751
752 void doRead()
753 {
Ed Tanous1abe55e2018-09-05 08:30:59 -0700754 BMCWEB_LOG_DEBUG << this << " doRead";
755
756 boost::beast::http::async_read(
Ed Tanousceac6f72018-12-02 11:58:47 -0800757 adaptor, buffer, *parser,
Jan Sowinskiee52ae12020-01-09 16:28:32 +0000758 [this,
759 self(shared_from_this())](const boost::system::error_code& ec,
760 std::size_t bytes_transferred) {
Zbigniew Kurzynski2658d982019-11-19 18:01:08 +0100761 BMCWEB_LOG_DEBUG << this << " async_read " << bytes_transferred
Ed Tanous1abe55e2018-09-05 08:30:59 -0700762 << " Bytes";
Ed Tanous1abe55e2018-09-05 08:30:59 -0700763
764 bool errorWhileReading = false;
765 if (ec)
766 {
Zbigniew Kurzynski2658d982019-11-19 18:01:08 +0100767 BMCWEB_LOG_ERROR
768 << this << " Error while reading: " << ec.message();
Ed Tanous1abe55e2018-09-05 08:30:59 -0700769 errorWhileReading = true;
770 }
771 else
772 {
Ed Tanouse278c182019-03-13 16:23:37 -0700773 if (!isAlive())
Ed Tanous1abe55e2018-09-05 08:30:59 -0700774 {
775 errorWhileReading = true;
776 }
777 }
778 if (errorWhileReading)
779 {
780 cancelDeadlineTimer();
Ed Tanouse278c182019-03-13 16:23:37 -0700781 close();
Ed Tanous1abe55e2018-09-05 08:30:59 -0700782 BMCWEB_LOG_DEBUG << this << " from read(1)";
Ed Tanous1abe55e2018-09-05 08:30:59 -0700783 return;
784 }
785 handle();
786 });
787 }
788
789 void doWrite()
790 {
Zbigniew Kurzynski2658d982019-11-19 18:01:08 +0100791 BMCWEB_LOG_DEBUG << this << " doWrite";
Ed Tanous1abe55e2018-09-05 08:30:59 -0700792 res.preparePayload();
793 serializer.emplace(*res.stringResponse);
794 boost::beast::http::async_write(
Ed Tanousceac6f72018-12-02 11:58:47 -0800795 adaptor, *serializer,
Jan Sowinskiee52ae12020-01-09 16:28:32 +0000796 [this,
797 self(shared_from_this())](const boost::system::error_code& ec,
798 std::size_t bytes_transferred) {
Zbigniew Kurzynski2658d982019-11-19 18:01:08 +0100799 BMCWEB_LOG_DEBUG << this << " async_write " << bytes_transferred
Ed Tanous1abe55e2018-09-05 08:30:59 -0700800 << " bytes";
801
802 if (ec)
803 {
804 BMCWEB_LOG_DEBUG << this << " from write(2)";
Ed Tanous1abe55e2018-09-05 08:30:59 -0700805 return;
806 }
Ed Tanousceac6f72018-12-02 11:58:47 -0800807 if (!res.keepAlive())
Ed Tanous1abe55e2018-09-05 08:30:59 -0700808 {
Ed Tanouse278c182019-03-13 16:23:37 -0700809 close();
Ed Tanous1abe55e2018-09-05 08:30:59 -0700810 BMCWEB_LOG_DEBUG << this << " from write(1)";
Ed Tanous1abe55e2018-09-05 08:30:59 -0700811 return;
812 }
813
814 serializer.reset();
815 BMCWEB_LOG_DEBUG << this << " Clearing response";
816 res.clear();
817 parser.emplace(std::piecewise_construct, std::make_tuple());
818 parser->body_limit(httpReqBodyLimit); // reset body limit for
819 // newly created parser
820 buffer.consume(buffer.size());
821
822 req.emplace(parser->get());
823 doReadHeaders();
824 });
825 }
826
Ed Tanous1abe55e2018-09-05 08:30:59 -0700827 void cancelDeadlineTimer()
828 {
Jan Sowinskiee52ae12020-01-09 16:28:32 +0000829 if (timerCancelKey)
830 {
831 BMCWEB_LOG_DEBUG << this << " timer cancelled: " << &timerQueue
832 << ' ' << *timerCancelKey;
833 timerQueue.cancel(*timerCancelKey);
834 timerCancelKey.reset();
835 }
Ed Tanous1abe55e2018-09-05 08:30:59 -0700836 }
837
James Feistf0af8592020-03-27 16:28:59 -0700838 void startDeadline(size_t timerIterations = 0)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700839 {
James Feistf0af8592020-03-27 16:28:59 -0700840 // drop all connections after 1 minute, this time limit was chosen
841 // arbitrarily and can be adjusted later if needed
842 constexpr const size_t maxReadAttempts =
843 (60 / detail::timerQueueTimeoutSeconds);
844
Ed Tanous1abe55e2018-09-05 08:30:59 -0700845 cancelDeadlineTimer();
846
James Feistf0af8592020-03-27 16:28:59 -0700847 timerCancelKey = timerQueue.add([this, self(shared_from_this()),
848 readCount{parser->get().body().size()},
849 timerIterations{timerIterations + 1}] {
850 // Mark timer as not active to avoid canceling it during
851 // Connection destructor which leads to double free issue
852 timerCancelKey.reset();
853 if (!isAlive())
854 {
855 return;
856 }
Jan Sowinski2b5e08e2020-01-09 17:16:02 +0100857
James Feistf0af8592020-03-27 16:28:59 -0700858 // Restart timer if read is in progress.
859 // With threshold can be used to drop slow connections
860 // to protect against slow-rate DoS attack
861 if ((parser->get().body().size() > readCount) &&
862 (timerIterations < maxReadAttempts))
863 {
864 BMCWEB_LOG_DEBUG << this << " restart timer - read in progress";
865 startDeadline(timerIterations);
866 return;
867 }
Jan Sowinski2b5e08e2020-01-09 17:16:02 +0100868
James Feistf0af8592020-03-27 16:28:59 -0700869 close();
870 });
James Feistcb6cb492020-04-03 13:36:17 -0700871
872 if (!timerCancelKey)
873 {
874 close();
875 return;
876 }
Ed Tanous1abe55e2018-09-05 08:30:59 -0700877 BMCWEB_LOG_DEBUG << this << " timer added: " << &timerQueue << ' '
Jan Sowinskiee52ae12020-01-09 16:28:32 +0000878 << *timerCancelKey;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700879 }
880
881 private:
882 Adaptor adaptor;
883 Handler* handler;
884
Ed Tanousa24526d2018-12-10 15:17:59 -0800885 // Making this a std::optional allows it to be efficiently destroyed and
Ed Tanous1abe55e2018-09-05 08:30:59 -0700886 // re-created on Connection reset
Ed Tanousa24526d2018-12-10 15:17:59 -0800887 std::optional<
Ed Tanous1abe55e2018-09-05 08:30:59 -0700888 boost::beast::http::request_parser<boost::beast::http::string_body>>
889 parser;
890
Ed Tanous3112a142018-11-29 15:45:10 -0800891 boost::beast::flat_static_buffer<8192> buffer;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700892
Ed Tanousa24526d2018-12-10 15:17:59 -0800893 std::optional<boost::beast::http::response_serializer<
Ed Tanous1abe55e2018-09-05 08:30:59 -0700894 boost::beast::http::string_body>>
895 serializer;
896
Ed Tanousa24526d2018-12-10 15:17:59 -0800897 std::optional<crow::Request> req;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700898 crow::Response res;
Kowalski, Kamil55e43f62019-07-10 13:12:57 +0200899#ifdef BMCWEB_ENABLE_MUTUAL_TLS_AUTHENTICATION
900 std::weak_ptr<crow::persistent_data::UserSession> session;
901#endif // BMCWEB_ENABLE_MUTUAL_TLS_AUTHENTICATION
Ed Tanous1abe55e2018-09-05 08:30:59 -0700902
903 const std::string& serverName;
904
Jan Sowinskiee52ae12020-01-09 16:28:32 +0000905 std::optional<size_t> timerCancelKey;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700906
Ed Tanous1abe55e2018-09-05 08:30:59 -0700907 bool needToCallAfterHandlers{};
908 bool needToStartReadAfterComplete{};
Ed Tanous1abe55e2018-09-05 08:30:59 -0700909
910 std::tuple<Middlewares...>* middlewares;
911 detail::Context<Middlewares...> ctx;
912
913 std::function<std::string()>& getCachedDateStr;
914 detail::TimerQueue& timerQueue;
Jan Sowinskiee52ae12020-01-09 16:28:32 +0000915
916 using std::enable_shared_from_this<
917 Connection<Adaptor, Handler, Middlewares...>>::shared_from_this;
Ed Tanous3112a142018-11-29 15:45:10 -0800918};
Ed Tanous1abe55e2018-09-05 08:30:59 -0700919} // namespace crow