| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797 |
- //
- // Copyright (c) 2016-2017 Vinnie Falco (vinnie dot falco at gmail dot com)
- //
- // Distributed under the Boost Software License, Version 1.0. (See accompanying
- // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
- //
- // Official repository: https://github.com/boostorg/beast
- //
- #ifndef BOOST_BEAST_WEBSOCKET_IMPL_ACCEPT_IPP
- #define BOOST_BEAST_WEBSOCKET_IMPL_ACCEPT_IPP
- #include <boost/beast/websocket/detail/type_traits.hpp>
- #include <boost/beast/http/empty_body.hpp>
- #include <boost/beast/http/parser.hpp>
- #include <boost/beast/http/read.hpp>
- #include <boost/beast/http/string_body.hpp>
- #include <boost/beast/http/write.hpp>
- #include <boost/beast/core/buffers_prefix.hpp>
- #include <boost/beast/core/handler_ptr.hpp>
- #include <boost/beast/core/detail/type_traits.hpp>
- #include <boost/asio/coroutine.hpp>
- #include <boost/asio/associated_allocator.hpp>
- #include <boost/asio/associated_executor.hpp>
- #include <boost/asio/executor_work_guard.hpp>
- #include <boost/asio/handler_continuation_hook.hpp>
- #include <boost/asio/handler_invoke_hook.hpp>
- #include <boost/asio/post.hpp>
- #include <boost/assert.hpp>
- #include <boost/throw_exception.hpp>
- #include <memory>
- #include <type_traits>
- namespace boost {
- namespace beast {
- namespace websocket {
- // Respond to an upgrade HTTP request
- template<class NextLayer, bool deflateSupported>
- template<class Handler>
- class stream<NextLayer, deflateSupported>::response_op
- : public boost::asio::coroutine
- {
- struct data
- {
- stream<NextLayer, deflateSupported>& ws;
- boost::asio::executor_work_guard<decltype(std::declval<
- stream<NextLayer, deflateSupported>&>().get_executor())> wg;
- error_code result;
- response_type res;
- template<class Body, class Allocator, class Decorator>
- data(
- Handler const&,
- stream<NextLayer, deflateSupported>& ws_,
- http::request<Body,
- http::basic_fields<Allocator>> const& req,
- Decorator const& decorator)
- : ws(ws_)
- , wg(ws.get_executor())
- , res(ws_.build_response(req, decorator, result))
- {
- }
- };
- handler_ptr<data, Handler> d_;
- public:
- response_op(response_op&&) = default;
- response_op(response_op const&) = delete;
- template<class DeducedHandler, class... Args>
- response_op(DeducedHandler&& h,
- stream<NextLayer, deflateSupported>& ws, Args&&... args)
- : d_(std::forward<DeducedHandler>(h),
- ws, std::forward<Args>(args)...)
- {
- }
- using allocator_type =
- boost::asio::associated_allocator_t<Handler>;
- allocator_type
- get_allocator() const noexcept
- {
- return (boost::asio::get_associated_allocator)(d_.handler());
- }
- using executor_type = boost::asio::associated_executor_t<
- Handler, decltype(std::declval<
- stream<NextLayer, deflateSupported>&>().get_executor())>;
- executor_type
- get_executor() const noexcept
- {
- return (boost::asio::get_associated_executor)(
- d_.handler(), d_->ws.get_executor());
- }
- void operator()(
- error_code ec = {},
- std::size_t bytes_transferred = 0);
- friend
- bool asio_handler_is_continuation(response_op* op)
- {
- using boost::asio::asio_handler_is_continuation;
- return asio_handler_is_continuation(
- std::addressof(op->d_.handler()));
- }
- template<class Function>
- friend
- void asio_handler_invoke(Function&& f, response_op* op)
- {
- using boost::asio::asio_handler_invoke;
- asio_handler_invoke(f, std::addressof(op->d_.handler()));
- }
- };
- template<class NextLayer, bool deflateSupported>
- template<class Handler>
- void
- stream<NextLayer, deflateSupported>::
- response_op<Handler>::
- operator()(
- error_code ec,
- std::size_t)
- {
- auto& d = *d_;
- BOOST_ASIO_CORO_REENTER(*this)
- {
- // Send response
- BOOST_ASIO_CORO_YIELD
- http::async_write(d.ws.next_layer(),
- d.res, std::move(*this));
- if(! ec)
- ec = d.result;
- if(! ec)
- {
- d.ws.do_pmd_config(d.res, is_deflate_supported{});
- d.ws.open(role_type::server);
- }
- {
- auto wg = std::move(d.wg);
- d_.invoke(ec);
- }
- }
- }
- //------------------------------------------------------------------------------
- // read and respond to an upgrade request
- //
- template<class NextLayer, bool deflateSupported>
- template<class Decorator, class Handler>
- class stream<NextLayer, deflateSupported>::accept_op
- : public boost::asio::coroutine
- {
- struct data
- {
- stream<NextLayer, deflateSupported>& ws;
- boost::asio::executor_work_guard<decltype(std::declval<
- stream<NextLayer, deflateSupported>&>().get_executor())> wg;
- Decorator decorator;
- http::request_parser<http::empty_body> p;
- data(
- Handler const&,
- stream<NextLayer, deflateSupported>& ws_,
- Decorator const& decorator_)
- : ws(ws_)
- , wg(ws.get_executor())
- , decorator(decorator_)
- {
- }
- };
- handler_ptr<data, Handler> d_;
- public:
- accept_op(accept_op&&) = default;
- accept_op(accept_op const&) = delete;
- template<class DeducedHandler, class... Args>
- accept_op(DeducedHandler&& h,
- stream<NextLayer, deflateSupported>& ws, Args&&... args)
- : d_(std::forward<DeducedHandler>(h),
- ws, std::forward<Args>(args)...)
- {
- }
- using allocator_type =
- boost::asio::associated_allocator_t<Handler>;
- allocator_type
- get_allocator() const noexcept
- {
- return (boost::asio::get_associated_allocator)(d_.handler());
- }
- using executor_type = boost::asio::associated_executor_t<
- Handler, decltype(std::declval<stream<NextLayer, deflateSupported>&>().get_executor())>;
- executor_type
- get_executor() const noexcept
- {
- return (boost::asio::get_associated_executor)(
- d_.handler(), d_->ws.get_executor());
- }
- template<class Buffers>
- void run(Buffers const& buffers);
- void operator()(
- error_code ec = {},
- std::size_t bytes_used = 0);
- friend
- bool asio_handler_is_continuation(accept_op* op)
- {
- using boost::asio::asio_handler_is_continuation;
- return asio_handler_is_continuation(
- std::addressof(op->d_.handler()));
- }
- template<class Function>
- friend
- void asio_handler_invoke(Function&& f, accept_op* op)
- {
- using boost::asio::asio_handler_invoke;
- asio_handler_invoke(f, std::addressof(op->d_.handler()));
- }
- };
- template<class NextLayer, bool deflateSupported>
- template<class Decorator, class Handler>
- template<class Buffers>
- void
- stream<NextLayer, deflateSupported>::
- accept_op<Decorator, Handler>::
- run(Buffers const& buffers)
- {
- using boost::asio::buffer_copy;
- using boost::asio::buffer_size;
- auto& d = *d_;
- error_code ec;
- boost::optional<typename
- static_buffer_base::mutable_buffers_type> mb;
- auto const len = buffer_size(buffers);
- try
- {
- mb.emplace(d.ws.rd_buf_.prepare(len));
- }
- catch(std::length_error const&)
- {
- ec = error::buffer_overflow;
- return (*this)(ec);
- }
- d.ws.rd_buf_.commit(
- buffer_copy(*mb, buffers));
- (*this)(ec);
- }
- template<class NextLayer, bool deflateSupported>
- template<class Decorator, class Handler>
- void
- stream<NextLayer, deflateSupported>::
- accept_op<Decorator, Handler>::
- operator()(error_code ec, std::size_t)
- {
- auto& d = *d_;
- BOOST_ASIO_CORO_REENTER(*this)
- {
- if(ec)
- {
- BOOST_ASIO_CORO_YIELD
- boost::asio::post(
- d.ws.get_executor(),
- bind_handler(std::move(*this), ec));
- }
- else
- {
- BOOST_ASIO_CORO_YIELD
- http::async_read(
- d.ws.next_layer(), d.ws.rd_buf_,
- d.p, std::move(*this));
- if(ec == http::error::end_of_stream)
- ec = error::closed;
- if(! ec)
- {
- // Arguments from our state must be
- // moved to the stack before releasing
- // the handler.
- auto& ws = d.ws;
- auto const req = d.p.release();
- auto const decorator = d.decorator;
- auto wg = std::move(d.wg);
- #if 1
- return response_op<Handler>{
- d_.release_handler(),
- ws, req, decorator}(ec);
- #else
- // VFALCO This *should* work but breaks
- // coroutine invariants in the unit test.
- // Also it calls reset() when it shouldn't.
- return ws.async_accept_ex(
- req, decorator, d_.release_handler());
- #endif
- }
- }
- {
- auto wg = std::move(d.wg);
- d_.invoke(ec);
- }
- }
- }
- //------------------------------------------------------------------------------
- template<class NextLayer, bool deflateSupported>
- void
- stream<NextLayer, deflateSupported>::
- accept()
- {
- static_assert(is_sync_stream<next_layer_type>::value,
- "SyncStream requirements not met");
- error_code ec;
- accept(ec);
- if(ec)
- BOOST_THROW_EXCEPTION(system_error{ec});
- }
- template<class NextLayer, bool deflateSupported>
- template<class ResponseDecorator>
- void
- stream<NextLayer, deflateSupported>::
- accept_ex(ResponseDecorator const& decorator)
- {
- static_assert(is_sync_stream<next_layer_type>::value,
- "SyncStream requirements not met");
- static_assert(detail::is_response_decorator<
- ResponseDecorator>::value,
- "ResponseDecorator requirements not met");
- error_code ec;
- accept_ex(decorator, ec);
- if(ec)
- BOOST_THROW_EXCEPTION(system_error{ec});
- }
- template<class NextLayer, bool deflateSupported>
- void
- stream<NextLayer, deflateSupported>::
- accept(error_code& ec)
- {
- static_assert(is_sync_stream<next_layer_type>::value,
- "SyncStream requirements not met");
- reset();
- do_accept(&default_decorate_res, ec);
- }
- template<class NextLayer, bool deflateSupported>
- template<class ResponseDecorator>
- void
- stream<NextLayer, deflateSupported>::
- accept_ex(ResponseDecorator const& decorator, error_code& ec)
- {
- static_assert(is_sync_stream<next_layer_type>::value,
- "SyncStream requirements not met");
- static_assert(detail::is_response_decorator<
- ResponseDecorator>::value,
- "ResponseDecorator requirements not met");
- reset();
- do_accept(decorator, ec);
- }
- template<class NextLayer, bool deflateSupported>
- template<class ConstBufferSequence>
- typename std::enable_if<! http::detail::is_header<
- ConstBufferSequence>::value>::type
- stream<NextLayer, deflateSupported>::
- accept(ConstBufferSequence const& buffers)
- {
- static_assert(is_sync_stream<next_layer_type>::value,
- "SyncStream requirements not met");
- static_assert(boost::asio::is_const_buffer_sequence<
- ConstBufferSequence>::value,
- "ConstBufferSequence requirements not met");
- error_code ec;
- accept(buffers, ec);
- if(ec)
- BOOST_THROW_EXCEPTION(system_error{ec});
- }
- template<class NextLayer, bool deflateSupported>
- template<
- class ConstBufferSequence,
- class ResponseDecorator>
- typename std::enable_if<! http::detail::is_header<
- ConstBufferSequence>::value>::type
- stream<NextLayer, deflateSupported>::
- accept_ex(
- ConstBufferSequence const& buffers,
- ResponseDecorator const &decorator)
- {
- static_assert(is_sync_stream<next_layer_type>::value,
- "SyncStream requirements not met");
- static_assert(boost::asio::is_const_buffer_sequence<
- ConstBufferSequence>::value,
- "ConstBufferSequence requirements not met");
- static_assert(detail::is_response_decorator<
- ResponseDecorator>::value,
- "ResponseDecorator requirements not met");
- error_code ec;
- accept_ex(buffers, decorator, ec);
- if(ec)
- BOOST_THROW_EXCEPTION(system_error{ec});
- }
- template<class NextLayer, bool deflateSupported>
- template<class ConstBufferSequence>
- typename std::enable_if<! http::detail::is_header<
- ConstBufferSequence>::value>::type
- stream<NextLayer, deflateSupported>::
- accept(
- ConstBufferSequence const& buffers, error_code& ec)
- {
- static_assert(is_sync_stream<next_layer_type>::value,
- "SyncStream requirements not met");
- static_assert(boost::asio::is_const_buffer_sequence<
- ConstBufferSequence>::value,
- "ConstBufferSequence requirements not met");
- using boost::asio::buffer_copy;
- using boost::asio::buffer_size;
- reset();
- boost::optional<typename
- static_buffer_base::mutable_buffers_type> mb;
- try
- {
- mb.emplace(rd_buf_.prepare(
- buffer_size(buffers)));
- }
- catch(std::length_error const&)
- {
- ec = error::buffer_overflow;
- return;
- }
- rd_buf_.commit(
- buffer_copy(*mb, buffers));
- do_accept(&default_decorate_res, ec);
- }
- template<class NextLayer, bool deflateSupported>
- template<
- class ConstBufferSequence,
- class ResponseDecorator>
- typename std::enable_if<! http::detail::is_header<
- ConstBufferSequence>::value>::type
- stream<NextLayer, deflateSupported>::
- accept_ex(
- ConstBufferSequence const& buffers,
- ResponseDecorator const& decorator,
- error_code& ec)
- {
- static_assert(is_sync_stream<next_layer_type>::value,
- "SyncStream requirements not met");
- static_assert(boost::asio::is_const_buffer_sequence<
- ConstBufferSequence>::value,
- "ConstBufferSequence requirements not met");
- static_assert(boost::asio::is_const_buffer_sequence<
- ConstBufferSequence>::value,
- "ConstBufferSequence requirements not met");
- using boost::asio::buffer_copy;
- using boost::asio::buffer_size;
- reset();
- boost::optional<typename
- static_buffer_base::mutable_buffers_type> mb;
- try
- {
- mb.emplace(rd_buf_.prepare(
- buffer_size(buffers)));
- }
- catch(std::length_error const&)
- {
- ec = error::buffer_overflow;
- return;
- }
- rd_buf_.commit(buffer_copy(*mb, buffers));
- do_accept(decorator, ec);
- }
- template<class NextLayer, bool deflateSupported>
- template<class Body, class Allocator>
- void
- stream<NextLayer, deflateSupported>::
- accept(
- http::request<Body,
- http::basic_fields<Allocator>> const& req)
- {
- static_assert(is_sync_stream<next_layer_type>::value,
- "SyncStream requirements not met");
- error_code ec;
- accept(req, ec);
- if(ec)
- BOOST_THROW_EXCEPTION(system_error{ec});
- }
- template<class NextLayer, bool deflateSupported>
- template<
- class Body, class Allocator,
- class ResponseDecorator>
- void
- stream<NextLayer, deflateSupported>::
- accept_ex(
- http::request<Body,
- http::basic_fields<Allocator>> const& req,
- ResponseDecorator const& decorator)
- {
- static_assert(is_sync_stream<next_layer_type>::value,
- "SyncStream requirements not met");
- static_assert(detail::is_response_decorator<
- ResponseDecorator>::value,
- "ResponseDecorator requirements not met");
- error_code ec;
- accept_ex(req, decorator, ec);
- if(ec)
- BOOST_THROW_EXCEPTION(system_error{ec});
- }
- template<class NextLayer, bool deflateSupported>
- template<class Body, class Allocator>
- void
- stream<NextLayer, deflateSupported>::
- accept(
- http::request<Body,
- http::basic_fields<Allocator>> const& req,
- error_code& ec)
- {
- static_assert(is_sync_stream<next_layer_type>::value,
- "SyncStream requirements not met");
- reset();
- do_accept(req, &default_decorate_res, ec);
- }
- template<class NextLayer, bool deflateSupported>
- template<
- class Body, class Allocator,
- class ResponseDecorator>
- void
- stream<NextLayer, deflateSupported>::
- accept_ex(
- http::request<Body,
- http::basic_fields<Allocator>> const& req,
- ResponseDecorator const& decorator,
- error_code& ec)
- {
- static_assert(is_sync_stream<next_layer_type>::value,
- "SyncStream requirements not met");
- static_assert(detail::is_response_decorator<
- ResponseDecorator>::value,
- "ResponseDecorator requirements not met");
- reset();
- do_accept(req, decorator, ec);
- }
- //------------------------------------------------------------------------------
- template<class NextLayer, bool deflateSupported>
- template<
- class AcceptHandler>
- BOOST_ASIO_INITFN_RESULT_TYPE(
- AcceptHandler, void(error_code))
- stream<NextLayer, deflateSupported>::
- async_accept(
- AcceptHandler&& handler)
- {
- static_assert(is_async_stream<next_layer_type>::value,
- "AsyncStream requirements not met");
- BOOST_BEAST_HANDLER_INIT(
- AcceptHandler, void(error_code));
- reset();
- accept_op<
- decltype(&default_decorate_res),
- BOOST_ASIO_HANDLER_TYPE(
- AcceptHandler, void(error_code))>{
- std::move(init.completion_handler),
- *this,
- &default_decorate_res}({});
- return init.result.get();
- }
- template<class NextLayer, bool deflateSupported>
- template<
- class ResponseDecorator,
- class AcceptHandler>
- BOOST_ASIO_INITFN_RESULT_TYPE(
- AcceptHandler, void(error_code))
- stream<NextLayer, deflateSupported>::
- async_accept_ex(
- ResponseDecorator const& decorator,
- AcceptHandler&& handler)
- {
- static_assert(is_async_stream<next_layer_type>::value,
- "AsyncStream requirements not met");
- static_assert(detail::is_response_decorator<
- ResponseDecorator>::value,
- "ResponseDecorator requirements not met");
- BOOST_BEAST_HANDLER_INIT(
- AcceptHandler, void(error_code));
- reset();
- accept_op<
- ResponseDecorator,
- BOOST_ASIO_HANDLER_TYPE(
- AcceptHandler, void(error_code))>{
- std::move(init.completion_handler),
- *this,
- decorator}({});
- return init.result.get();
- }
- template<class NextLayer, bool deflateSupported>
- template<
- class ConstBufferSequence,
- class AcceptHandler>
- typename std::enable_if<
- ! http::detail::is_header<ConstBufferSequence>::value,
- BOOST_ASIO_INITFN_RESULT_TYPE(
- AcceptHandler, void(error_code))>::type
- stream<NextLayer, deflateSupported>::
- async_accept(
- ConstBufferSequence const& buffers,
- AcceptHandler&& handler)
- {
- static_assert(is_async_stream<next_layer_type>::value,
- "AsyncStream requirements not met");
- static_assert(boost::asio::is_const_buffer_sequence<
- ConstBufferSequence>::value,
- "ConstBufferSequence requirements not met");
- BOOST_BEAST_HANDLER_INIT(
- AcceptHandler, void(error_code));
- reset();
- accept_op<
- decltype(&default_decorate_res),
- BOOST_ASIO_HANDLER_TYPE(
- AcceptHandler, void(error_code))>{
- std::move(init.completion_handler),
- *this,
- &default_decorate_res}.run(buffers);
- return init.result.get();
- }
- template<class NextLayer, bool deflateSupported>
- template<
- class ConstBufferSequence,
- class ResponseDecorator,
- class AcceptHandler>
- typename std::enable_if<
- ! http::detail::is_header<ConstBufferSequence>::value,
- BOOST_ASIO_INITFN_RESULT_TYPE(
- AcceptHandler, void(error_code))>::type
- stream<NextLayer, deflateSupported>::
- async_accept_ex(
- ConstBufferSequence const& buffers,
- ResponseDecorator const& decorator,
- AcceptHandler&& handler)
- {
- static_assert(is_async_stream<next_layer_type>::value,
- "AsyncStream requirements not met");
- static_assert(boost::asio::is_const_buffer_sequence<
- ConstBufferSequence>::value,
- "ConstBufferSequence requirements not met");
- static_assert(detail::is_response_decorator<
- ResponseDecorator>::value,
- "ResponseDecorator requirements not met");
- BOOST_BEAST_HANDLER_INIT(
- AcceptHandler, void(error_code));
- reset();
- accept_op<
- ResponseDecorator,
- BOOST_ASIO_HANDLER_TYPE(
- AcceptHandler, void(error_code))>{
- std::move(init.completion_handler),
- *this,
- decorator}.run(buffers);
- return init.result.get();
- }
- template<class NextLayer, bool deflateSupported>
- template<
- class Body, class Allocator,
- class AcceptHandler>
- BOOST_ASIO_INITFN_RESULT_TYPE(
- AcceptHandler, void(error_code))
- stream<NextLayer, deflateSupported>::
- async_accept(
- http::request<Body, http::basic_fields<Allocator>> const& req,
- AcceptHandler&& handler)
- {
- static_assert(is_async_stream<next_layer_type>::value,
- "AsyncStream requirements not met");
- BOOST_BEAST_HANDLER_INIT(
- AcceptHandler, void(error_code));
- reset();
- using boost::asio::asio_handler_is_continuation;
- response_op<
- BOOST_ASIO_HANDLER_TYPE(
- AcceptHandler, void(error_code))>{
- std::move(init.completion_handler),
- *this,
- req,
- &default_decorate_res}();
- return init.result.get();
- }
- template<class NextLayer, bool deflateSupported>
- template<
- class Body, class Allocator,
- class ResponseDecorator,
- class AcceptHandler>
- BOOST_ASIO_INITFN_RESULT_TYPE(
- AcceptHandler, void(error_code))
- stream<NextLayer, deflateSupported>::
- async_accept_ex(
- http::request<Body, http::basic_fields<Allocator>> const& req,
- ResponseDecorator const& decorator,
- AcceptHandler&& handler)
- {
- static_assert(is_async_stream<next_layer_type>::value,
- "AsyncStream requirements not met");
- static_assert(detail::is_response_decorator<
- ResponseDecorator>::value,
- "ResponseDecorator requirements not met");
- BOOST_BEAST_HANDLER_INIT(
- AcceptHandler, void(error_code));
- reset();
- using boost::asio::asio_handler_is_continuation;
- response_op<
- BOOST_ASIO_HANDLER_TYPE(
- AcceptHandler, void(error_code))>{
- std::move(init.completion_handler),
- *this,
- req,
- decorator}();
- return init.result.get();
- }
- //------------------------------------------------------------------------------
- template<class NextLayer, bool deflateSupported>
- template<class Decorator>
- void
- stream<NextLayer, deflateSupported>::
- do_accept(
- Decorator const& decorator,
- error_code& ec)
- {
- http::request_parser<http::empty_body> p;
- http::read(next_layer(), rd_buf_, p, ec);
- if(ec == http::error::end_of_stream)
- ec = error::closed;
- if(ec)
- return;
- do_accept(p.get(), decorator, ec);
- }
- template<class NextLayer, bool deflateSupported>
- template<class Body, class Allocator,
- class Decorator>
- void
- stream<NextLayer, deflateSupported>::
- do_accept(
- http::request<Body,
- http::basic_fields<Allocator>> const& req,
- Decorator const& decorator,
- error_code& ec)
- {
- error_code result;
- auto const res = build_response(req, decorator, result);
- http::write(stream_, res, ec);
- if(ec)
- return;
- ec = result;
- if(ec)
- {
- // VFALCO TODO Respect keep alive setting, perform
- // teardown if Connection: close.
- return;
- }
- do_pmd_config(res, is_deflate_supported{});
- open(role_type::server);
- }
- } // websocket
- } // beast
- } // boost
- #endif
|