2#include <boost/asio.hpp>
3#include <boost/algorithm/string/predicate.hpp>
4#include <boost/lexical_cast.hpp>
5#include <boost/array.hpp>
10#include "crow/http_parser_merged.h"
12#include "crow/parser.h"
13#include "crow/http_response.h"
14#include "crow/logging.h"
15#include "crow/settings.h"
16#include "crow/dumb_timer_queue.h"
17#include "crow/middleware_context.h"
18#include "crow/socket_adaptors.h"
19#include "crow/compression.h"
23 using namespace boost;
24 using tcp = asio::ip::tcp;
28 template <
typename MW>
32 void (T::*)(
request&,
response&,
typename MW::context&)
const = &T::before_handle
38 template <
typename MW>
42 void (T::*)(
request&,
response&,
typename MW::context&) = &T::before_handle
48 template <
typename MW>
52 void (T::*)(
request&,
response&,
typename MW::context&)
const = &T::after_handle
58 template <
typename MW>
62 void (T::*)(
request&,
response&,
typename MW::context&) = &T::after_handle
78 static std::false_type f(...);
81 static const bool value =
decltype(f<T>(
nullptr))::value;
94 static std::false_type f(...);
97 static const bool value =
decltype(f<T>(
nullptr))::value;
100 template <
typename MW,
typename Context,
typename ParentContext>
101 typename std::enable_if<!is_before_handle_arity_3_impl<MW>::value>::type
102 before_handler_call(MW& mw,
request& req,
response& res, Context& ctx, ParentContext& )
104 mw.before_handle(req, res, ctx.template get<MW>(), ctx);
107 template <
typename MW,
typename Context,
typename ParentContext>
108 typename std::enable_if<is_before_handle_arity_3_impl<MW>::value>::type
109 before_handler_call(MW& mw,
request& req,
response& res, Context& ctx, ParentContext& )
111 mw.before_handle(req, res, ctx.template get<MW>());
114 template <
typename MW,
typename Context,
typename ParentContext>
115 typename std::enable_if<!is_after_handle_arity_3_impl<MW>::value>::type
116 after_handler_call(MW& mw, request& req, response& res, Context& ctx, ParentContext& )
118 mw.after_handle(req, res, ctx.template get<MW>(), ctx);
121 template <
typename MW,
typename Context,
typename ParentContext>
122 typename std::enable_if<is_after_handle_arity_3_impl<MW>::value>::type
123 after_handler_call(MW& mw, request& req, response& res, Context& ctx, ParentContext& )
125 mw.after_handle(req, res, ctx.template get<MW>());
128 template <
int N,
typename Context,
typename Container,
typename CurrentMW,
typename ... Middlewares>
129 bool middleware_call_helper(Container& middlewares, request& req, response& res, Context& ctx)
131 using parent_context_t =
typename Context::template partial<N-1>;
132 before_handler_call<CurrentMW, Context, parent_context_t>(std::get<N>(middlewares), req, res, ctx,
static_cast<parent_context_t&
>(ctx));
134 if (res.is_completed())
136 after_handler_call<CurrentMW, Context, parent_context_t>(std::get<N>(middlewares), req, res, ctx,
static_cast<parent_context_t&
>(ctx));
140 if (middleware_call_helper<N+1, Context, Container, Middlewares...>(middlewares, req, res, ctx))
142 after_handler_call<CurrentMW, Context, parent_context_t>(std::get<N>(middlewares), req, res, ctx,
static_cast<parent_context_t&
>(ctx));
149 template <
int N,
typename Context,
typename Container>
150 bool middleware_call_helper(Container& , request& , response& , Context& )
155 template <
int N,
typename Context,
typename Container>
156 typename std::enable_if<(N<0)>::type
157 after_handlers_call_helper(Container& , Context& , request& , response& )
161 template <
int N,
typename Context,
typename Container>
162 typename std::enable_if<(N==0)>::type after_handlers_call_helper(Container& middlewares, Context& ctx, request& req, response& res)
164 using parent_context_t =
typename Context::template partial<N-1>;
165 using CurrentMW =
typename std::tuple_element<N, typename std::remove_reference<Container>::type>::type;
166 after_handler_call<CurrentMW, Context, parent_context_t>(std::get<N>(middlewares), req, res, ctx,
static_cast<parent_context_t&
>(ctx));
169 template <
int N,
typename Context,
typename Container>
170 typename std::enable_if<(N>0)>::type after_handlers_call_helper(Container& middlewares, Context& ctx, request& req, response& res)
172 using parent_context_t =
typename Context::template partial<N-1>;
173 using CurrentMW =
typename std::tuple_element<N, typename std::remove_reference<Container>::type>::type;
174 after_handler_call<CurrentMW, Context, parent_context_t>(std::get<N>(middlewares), req, res, ctx,
static_cast<parent_context_t&
>(ctx));
175 after_handlers_call_helper<N-1, Context, Container>(middlewares, ctx, req, res);
179#ifdef CROW_ENABLE_DEBUG
180 static std::atomic<int> connectionCount;
184 template <
typename Adaptor,
typename Handler,
typename ... Middlewares>
190 boost::asio::io_service& io_service,
192 const std::string& server_name,
193 std::tuple<Middlewares...>* middlewares,
194 std::function<std::string()>& get_cached_date_str_f,
196 typename Adaptor::context* adaptor_ctx_
198 : adaptor_(io_service, adaptor_ctx_),
201 server_name_(server_name),
202 middlewares_(middlewares),
203 get_cached_date_str(get_cached_date_str_f),
204 timer_queue(timer_queue)
206#ifdef CROW_ENABLE_DEBUG
208 CROW_LOG_DEBUG <<
"Connection open, total " << connectionCount <<
", " <<
this;
214 res.complete_request_handler_ =
nullptr;
215 cancel_deadline_timer();
216#ifdef CROW_ENABLE_DEBUG
218 CROW_LOG_DEBUG <<
"Connection closed, total " << connectionCount <<
", " <<
this;
223 decltype(std::declval<Adaptor>().raw_socket())&
socket()
225 return adaptor_.raw_socket();
230 adaptor_.start([
this](
const boost::system::error_code& ec) {
247 if (parser_.check_version(1, 1) && parser_.headers.count(
"expect") && get_header_value(parser_.headers,
"expect") ==
"100-continue")
250 static std::string expect_100_continue =
"HTTP/1.1 100 Continue\r\n\r\n";
251 buffers_.emplace_back(expect_100_continue.data(), expect_100_continue.size());
258 cancel_deadline_timer();
259 bool is_invalid_request =
false;
260 add_keep_alive_ =
false;
262 req_ = std::move(parser_.to_request());
265 req.
remoteIpAddress = adaptor_.remote_endpoint().address().to_string();
267 if (parser_.check_version(1, 0))
270 if (req.headers.count(
"connection"))
272 if (boost::iequals(req.get_header_value(
"connection"),
"Keep-Alive"))
273 add_keep_alive_ =
true;
276 close_connection_ =
true;
278 else if (parser_.check_version(1, 1))
281 if (req.headers.count(
"connection"))
283 if (req.get_header_value(
"connection") ==
"close")
284 close_connection_ =
true;
285 else if (boost::iequals(req.get_header_value(
"connection"),
"Keep-Alive"))
286 add_keep_alive_ =
true;
288 if (!req.headers.count(
"host"))
290 is_invalid_request =
true;
293 if (parser_.is_upgrade())
295 if (req.get_header_value(
"upgrade") ==
"h2c")
302 close_connection_ =
true;
303 handler_->handle_upgrade(req, res, std::move(adaptor_));
309 CROW_LOG_INFO <<
"Request: " << boost::lexical_cast<std::string>(adaptor_.remote_endpoint()) <<
" " <<
this <<
" HTTP/" << parser_.http_major <<
"." << parser_.http_minor <<
' '
310 << method_name(req.method) <<
" " << req.url;
313 need_to_call_after_handlers_ =
false;
314 if (!is_invalid_request)
316 res.complete_request_handler_ = []{};
317 res.is_alive_helper_ = [
this]()->
bool{
return adaptor_.is_open(); };
319 ctx_ = detail::context<Middlewares...>();
320 req.middleware_context =
static_cast<void*
>(&ctx_);
321 req.io_service = &adaptor_.get_io_service();
322 detail::middleware_call_helper<0,
decltype(ctx_),
decltype(*middlewares_), Middlewares...>(*middlewares_, req, res, ctx_);
327 need_to_call_after_handlers_ =
true;
328 handler_->handle(req, res);
330 res.set_header(
"connection",
"Keep-Alive");
346 CROW_LOG_INFO <<
"Response: " <<
this <<
' ' << req_.
raw_url <<
' ' << res.code <<
' ' << close_connection_;
348 if (need_to_call_after_handlers_)
350 need_to_call_after_handlers_ =
false;
353 detail::after_handlers_call_helper<
354 (
static_cast<int>(
sizeof...(Middlewares))-1),
356 decltype(*middlewares_)>
357 (*middlewares_, ctx_, req_, res);
359#ifdef CROW_ENABLE_COMPRESSION
360 std::string accept_encoding = req_.get_header_value(
"Accept-Encoding");
361 if (!accept_encoding.empty() && res.compressed)
363 switch (handler_->compression_algorithm())
365 case compression::DEFLATE:
366 if (accept_encoding.find(
"deflate") != std::string::npos)
368 res.body = compression::compress_string(res.body, compression::algorithm::DEFLATE);
369 res.set_header(
"Content-Encoding",
"deflate");
372 case compression::GZIP:
373 if (accept_encoding.find(
"gzip") != std::string::npos)
375 res.body = compression::compress_string(res.body, compression::algorithm::GZIP);
376 res.set_header(
"Content-Encoding",
"gzip");
385 std::string location = res.get_header_value(
"Location");
386 if (!location.empty() && location.find(
"://", 0) == std::string::npos)
388 #ifdef CROW_ENABLE_SSL
389 location.insert(0,
"https://" + req_.get_header_value(
"Host"));
391 location.insert(0,
"http://" + req_.get_header_value(
"Host"));
393 res.set_header(
"location", location);
398 if (res.is_static_type())
409 void prepare_buffers()
412 res.complete_request_handler_ =
nullptr;
414 if (!adaptor_.is_open())
421 static std::unordered_map<int, std::string> statusCodes = {
422 {200,
"HTTP/1.1 200 OK\r\n"},
423 {201,
"HTTP/1.1 201 Created\r\n"},
424 {202,
"HTTP/1.1 202 Accepted\r\n"},
425 {204,
"HTTP/1.1 204 No Content\r\n"},
427 {300,
"HTTP/1.1 300 Multiple Choices\r\n"},
428 {301,
"HTTP/1.1 301 Moved Permanently\r\n"},
429 {302,
"HTTP/1.1 302 Found\r\n"},
430 {303,
"HTTP/1.1 303 See Other\r\n"},
431 {304,
"HTTP/1.1 304 Not Modified\r\n"},
432 {307,
"HTTP/1.1 307 Temporary Redirect\r\n"},
433 {308,
"HTTP/1.1 308 Permanent Redirect\r\n"},
435 {400,
"HTTP/1.1 400 Bad Request\r\n"},
436 {401,
"HTTP/1.1 401 Unauthorized\r\n"},
437 {403,
"HTTP/1.1 403 Forbidden\r\n"},
438 {404,
"HTTP/1.1 404 Not Found\r\n"},
439 {405,
"HTTP/1.1 405 Method Not Allowed\r\n"},
440 {413,
"HTTP/1.1 413 Payload Too Large\r\n"},
441 {422,
"HTTP/1.1 422 Unprocessable Entity\r\n"},
442 {429,
"HTTP/1.1 429 Too Many Requests\r\n"},
444 {500,
"HTTP/1.1 500 Internal Server Error\r\n"},
445 {501,
"HTTP/1.1 501 Not Implemented\r\n"},
446 {502,
"HTTP/1.1 502 Bad Gateway\r\n"},
447 {503,
"HTTP/1.1 503 Service Unavailable\r\n"},
450 static std::string seperator =
": ";
451 static std::string crlf =
"\r\n";
454 buffers_.reserve(4*(res.headers.size()+5)+3);
456 if (!statusCodes.count(res.code))
459 auto& status = statusCodes.find(res.code)->second;
460 buffers_.emplace_back(status.data(), status.size());
463 if (res.code >= 400 && res.body.empty())
464 res.body = statusCodes[res.code].substr(9);
466 for(
auto& kv : res.headers)
468 buffers_.emplace_back(kv.first.data(), kv.first.size());
469 buffers_.emplace_back(seperator.data(), seperator.size());
470 buffers_.emplace_back(kv.second.data(), kv.second.size());
471 buffers_.emplace_back(crlf.data(), crlf.size());
475 if (!res.manual_length_header && !res.headers.count(
"content-length"))
477 content_length_ = std::to_string(res.body.size());
478 static std::string content_length_tag =
"Content-Length: ";
479 buffers_.emplace_back(content_length_tag.data(), content_length_tag.size());
480 buffers_.emplace_back(content_length_.data(), content_length_.size());
481 buffers_.emplace_back(crlf.data(), crlf.size());
483 if (!res.headers.count(
"server"))
485 static std::string server_tag =
"Server: ";
486 buffers_.emplace_back(server_tag.data(), server_tag.size());
487 buffers_.emplace_back(server_name_.data(), server_name_.size());
488 buffers_.emplace_back(crlf.data(), crlf.size());
490 if (!res.headers.count(
"date"))
492 static std::string date_tag =
"Date: ";
493 date_str_ = get_cached_date_str();
494 buffers_.emplace_back(date_tag.data(), date_tag.size());
495 buffers_.emplace_back(date_str_.data(), date_str_.size());
496 buffers_.emplace_back(crlf.data(), crlf.size());
500 static std::string keep_alive_tag =
"Connection: Keep-Alive";
501 buffers_.emplace_back(keep_alive_tag.data(), keep_alive_tag.size());
502 buffers_.emplace_back(crlf.data(), crlf.size());
505 buffers_.emplace_back(crlf.data(), crlf.size());
509 void do_write_static()
512 boost::asio::write(adaptor_.socket(), buffers_);
514 if (res.file_info.statResult == 0)
516 std::ifstream is(res.file_info.path.c_str(), std::ios::in | std::ios::binary);
518 while (is.read(buf,
sizeof(buf)).gcount() > 0)
520 std::vector<asio::const_buffer> buffers;
521 buffers.push_back(boost::asio::buffer(buf));
522 do_write_sync(buffers);
531 void do_write_general()
533 if (res.body.length() < res_stream_threshold_)
535 res_body_copy_.swap(res.body);
536 buffers_.emplace_back(res_body_copy_.data(), res_body_copy_.size());
540 if (need_to_start_read_after_complete_)
542 need_to_start_read_after_complete_ =
false;
550 boost::asio::write(adaptor_.socket(), buffers_);
551 if (res.body.length() > 0)
554 std::vector<asio::const_buffer> buffers;
556 while (res.body.length() > 16384)
559 buf = res.body.substr(0, 16384);
560 res.body = res.body.substr(16384);
562 buffers.push_back(boost::asio::buffer(buf));
563 do_write_sync(buffers);
571 buffers.push_back(boost::asio::buffer(buf));
572 do_write_sync(buffers);
585 adaptor_.socket().async_read_some(boost::asio::buffer(buffer_),
586 [
this](
const boost::system::error_code& ec, std::size_t bytes_transferred)
588 bool error_while_reading =
true;
591 bool ret = parser_.feed(buffer_.data(), bytes_transferred);
592 if (ret && adaptor_.is_open())
594 error_while_reading = false;
598 if (error_while_reading)
600 cancel_deadline_timer();
602 adaptor_.shutdown_read();
605 CROW_LOG_DEBUG <<
this <<
" from read(1)";
608 else if (close_connection_)
610 cancel_deadline_timer();
616 else if (!need_to_call_after_handlers_)
624 need_to_start_read_after_complete_ =
true;
633 boost::asio::async_write(adaptor_.socket(), buffers_,
634 [&](
const boost::system::error_code& ec, std::size_t )
638 res_body_copy_.clear();
641 if (close_connection_)
643 adaptor_.shutdown_write();
645 CROW_LOG_DEBUG << this <<
" from write(1)";
651 CROW_LOG_DEBUG << this <<
" from write(2)";
657 inline void do_write_sync(std::vector<asio::const_buffer>& buffers)
660 boost::asio::write(adaptor_.socket(), buffers, [&](std::error_code ec, std::size_t) {
663 if (close_connection_)
665 adaptor_.shutdown_write();
667 CROW_LOG_DEBUG << this <<
" from write (sync)(1)";
674 CROW_LOG_ERROR << ec <<
" - happened while sending buffers";
675 CROW_LOG_DEBUG << this <<
" from write (sync)(2)";
684 CROW_LOG_DEBUG <<
this <<
" is_reading " << is_reading <<
" is_writing " << is_writing;
685 if (!is_reading && !is_writing)
687 CROW_LOG_DEBUG <<
this <<
" delete (idle) ";
692 void cancel_deadline_timer()
694 CROW_LOG_DEBUG <<
this <<
" timer cancelled: " << timer_cancel_key_.first <<
' ' << timer_cancel_key_.second;
695 timer_queue.cancel(timer_cancel_key_);
698 void start_deadline()
700 cancel_deadline_timer();
702 timer_cancel_key_ = timer_queue.
add([
this]
704 if (!adaptor_.is_open())
708 adaptor_.shutdown_readwrite();
711 CROW_LOG_DEBUG <<
this <<
" timer added: " << timer_cancel_key_.first <<
' ' << timer_cancel_key_.second;
718 boost::array<char, 4096> buffer_;
720 const unsigned res_stream_threshold_ = 1048576;
722 HTTPParser<Connection> parser_;
726 bool close_connection_ =
false;
728 const std::string& server_name_;
729 std::vector<boost::asio::const_buffer> buffers_;
731 std::string content_length_;
732 std::string date_str_;
733 std::string res_body_copy_;
736 detail::dumb_timer_queue::key timer_cancel_key_;
740 bool need_to_call_after_handlers_{};
741 bool need_to_start_read_after_complete_{};
742 bool add_keep_alive_{};
744 std::tuple<Middlewares...>* middlewares_;
745 detail::context<Middlewares...> ctx_;
747 std::function<std::string()>& get_cached_date_str;
748 detail::dumb_timer_queue& timer_queue;
An HTTP connection.
Definition: http_connection.h:186
decltype(std::declval< Adaptor >().raw_socket()) & socket()
The TCP socket on top of which the connection is established.
Definition: http_connection.h:223
void complete_request()
Call the after handle middleware and send the write the response to the connection.
Definition: http_connection.h:344
Fast timer queue for fixed tick value.
Definition: dumb_timer_queue.h:18
key add(std::function< void()> f)
Add a function to the queue.
Definition: dumb_timer_queue.h:36
Definition: http_connection.h:65
Definition: http_connection.h:55
Definition: http_connection.h:50
Definition: http_connection.h:60
Definition: http_connection.h:45
Definition: http_connection.h:35
Definition: http_connection.h:30
Definition: http_connection.h:40
Definition: http_connection.h:86
Definition: http_connection.h:70
An HTTP request.
Definition: http_request.h:27
std::string raw_url
The full URL containing the ? and URL parameters.
Definition: http_request.h:29
std::string remoteIpAddress
The IP address from which the request was sent.
Definition: http_request.h:34
HTTP response.
Definition: http_response.h:24