accept.hpp 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863
  1. //
  2. // Copyright (c) 2016-2019 Vinnie Falco (vinnie dot falco at gmail dot com)
  3. //
  4. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  5. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  6. //
  7. // Official repository: https://github.com/boostorg/beast
  8. //
  9. #ifndef BOOST_BEAST_WEBSOCKET_IMPL_ACCEPT_IPP
  10. #define BOOST_BEAST_WEBSOCKET_IMPL_ACCEPT_IPP
  11. #include <boost/beast/websocket/impl/stream_impl.hpp>
  12. #include <boost/beast/websocket/detail/type_traits.hpp>
  13. #include <boost/beast/http/empty_body.hpp>
  14. #include <boost/beast/http/parser.hpp>
  15. #include <boost/beast/http/read.hpp>
  16. #include <boost/beast/http/string_body.hpp>
  17. #include <boost/beast/http/write.hpp>
  18. #include <boost/beast/core/async_base.hpp>
  19. #include <boost/beast/core/buffer_traits.hpp>
  20. #include <boost/beast/core/stream_traits.hpp>
  21. #include <boost/beast/core/detail/buffer.hpp>
  22. #include <boost/beast/core/detail/type_traits.hpp>
  23. #include <boost/beast/version.hpp>
  24. #include <boost/asio/coroutine.hpp>
  25. #include <boost/asio/post.hpp>
  26. #include <boost/assert.hpp>
  27. #include <boost/throw_exception.hpp>
  28. #include <memory>
  29. #include <type_traits>
  30. namespace boost {
  31. namespace beast {
  32. namespace websocket {
  33. //------------------------------------------------------------------------------
  34. namespace detail {
  35. template<class Body, class Allocator>
  36. void
  37. impl_base<true>::
  38. build_response_pmd(
  39. http::response<http::string_body>& res,
  40. http::request<Body,
  41. http::basic_fields<Allocator>> const& req)
  42. {
  43. pmd_offer offer;
  44. pmd_offer unused;
  45. pmd_read(offer, req);
  46. pmd_negotiate(res, unused, offer, pmd_opts_);
  47. }
  48. template<class Body, class Allocator>
  49. void
  50. impl_base<false>::
  51. build_response_pmd(
  52. http::response<http::string_body>&,
  53. http::request<Body,
  54. http::basic_fields<Allocator>> const&)
  55. {
  56. }
  57. } // detail
  58. template<class NextLayer, bool deflateSupported>
  59. template<class Body, class Allocator, class Decorator>
  60. response_type
  61. stream<NextLayer, deflateSupported>::impl_type::
  62. build_response(
  63. http::request<Body,
  64. http::basic_fields<Allocator>> const& req,
  65. Decorator const& decorator,
  66. error_code& result)
  67. {
  68. auto const decorate =
  69. [this, &decorator](response_type& res)
  70. {
  71. decorator_opt(res);
  72. decorator(res);
  73. if(! res.count(http::field::server))
  74. {
  75. // VFALCO this is weird..
  76. BOOST_STATIC_ASSERT(sizeof(
  77. BOOST_BEAST_VERSION_STRING) < 20);
  78. static_string<20> s(BOOST_BEAST_VERSION_STRING);
  79. res.set(http::field::server, s);
  80. }
  81. };
  82. auto err =
  83. [&](error e)
  84. {
  85. result = e;
  86. response_type res;
  87. res.version(req.version());
  88. res.result(http::status::bad_request);
  89. res.body() = result.message();
  90. res.prepare_payload();
  91. decorate(res);
  92. return res;
  93. };
  94. if(req.version() != 11)
  95. return err(error::bad_http_version);
  96. if(req.method() != http::verb::get)
  97. return err(error::bad_method);
  98. if(! req.count(http::field::host))
  99. return err(error::no_host);
  100. {
  101. auto const it = req.find(http::field::connection);
  102. if(it == req.end())
  103. return err(error::no_connection);
  104. if(! http::token_list{it->value()}.exists("upgrade"))
  105. return err(error::no_connection_upgrade);
  106. }
  107. {
  108. auto const it = req.find(http::field::upgrade);
  109. if(it == req.end())
  110. return err(error::no_upgrade);
  111. if(! http::token_list{it->value()}.exists("websocket"))
  112. return err(error::no_upgrade_websocket);
  113. }
  114. string_view key;
  115. {
  116. auto const it = req.find(http::field::sec_websocket_key);
  117. if(it == req.end())
  118. return err(error::no_sec_key);
  119. key = it->value();
  120. if(key.size() > detail::sec_ws_key_type::max_size_n)
  121. return err(error::bad_sec_key);
  122. }
  123. {
  124. auto const it = req.find(http::field::sec_websocket_version);
  125. if(it == req.end())
  126. return err(error::no_sec_version);
  127. if(it->value() != "13")
  128. {
  129. response_type res;
  130. res.result(http::status::upgrade_required);
  131. res.version(req.version());
  132. res.set(http::field::sec_websocket_version, "13");
  133. result = error::bad_sec_version;
  134. res.body() = result.message();
  135. res.prepare_payload();
  136. decorate(res);
  137. return res;
  138. }
  139. }
  140. response_type res;
  141. res.result(http::status::switching_protocols);
  142. res.version(req.version());
  143. res.set(http::field::upgrade, "websocket");
  144. res.set(http::field::connection, "upgrade");
  145. {
  146. detail::sec_ws_accept_type acc;
  147. detail::make_sec_ws_accept(acc, key);
  148. res.set(http::field::sec_websocket_accept, acc);
  149. }
  150. this->build_response_pmd(res, req);
  151. decorate(res);
  152. result = {};
  153. return res;
  154. }
  155. //------------------------------------------------------------------------------
  156. /** Respond to an HTTP request
  157. */
  158. template<class NextLayer, bool deflateSupported>
  159. template<class Handler>
  160. class stream<NextLayer, deflateSupported>::response_op
  161. : public beast::stable_async_base<
  162. Handler, beast::executor_type<stream>>
  163. , public net::coroutine
  164. {
  165. boost::weak_ptr<impl_type> wp_;
  166. error_code result_; // must come before res_
  167. response_type& res_;
  168. public:
  169. template<
  170. class Handler_,
  171. class Body, class Allocator,
  172. class Decorator>
  173. response_op(
  174. Handler_&& h,
  175. boost::shared_ptr<impl_type> const& sp,
  176. http::request<Body,
  177. http::basic_fields<Allocator>> const& req,
  178. Decorator const& decorator,
  179. bool cont = false)
  180. : stable_async_base<Handler,
  181. beast::executor_type<stream>>(
  182. std::forward<Handler_>(h),
  183. sp->stream().get_executor())
  184. , wp_(sp)
  185. , res_(beast::allocate_stable<response_type>(*this,
  186. sp->build_response(req, decorator, result_)))
  187. {
  188. (*this)({}, 0, cont);
  189. }
  190. void operator()(
  191. error_code ec = {},
  192. std::size_t bytes_transferred = 0,
  193. bool cont = true)
  194. {
  195. boost::ignore_unused(bytes_transferred);
  196. auto sp = wp_.lock();
  197. if(! sp)
  198. {
  199. ec = net::error::operation_aborted;
  200. return this->complete(cont, ec);
  201. }
  202. auto& impl = *sp;
  203. BOOST_ASIO_CORO_REENTER(*this)
  204. {
  205. impl.change_status(status::handshake);
  206. impl.update_timer(this->get_executor());
  207. // Send response
  208. BOOST_ASIO_CORO_YIELD
  209. http::async_write(
  210. impl.stream(), res_, std::move(*this));
  211. if(impl.check_stop_now(ec))
  212. goto upcall;
  213. if(! ec)
  214. ec = result_;
  215. if(! ec)
  216. {
  217. impl.do_pmd_config(res_);
  218. impl.open(role_type::server);
  219. }
  220. upcall:
  221. this->complete(cont, ec);
  222. }
  223. }
  224. };
  225. //------------------------------------------------------------------------------
  226. // read and respond to an upgrade request
  227. //
  228. template<class NextLayer, bool deflateSupported>
  229. template<class Handler, class Decorator>
  230. class stream<NextLayer, deflateSupported>::accept_op
  231. : public beast::stable_async_base<
  232. Handler, beast::executor_type<stream>>
  233. , public net::coroutine
  234. {
  235. boost::weak_ptr<impl_type> wp_;
  236. http::request_parser<http::empty_body>& p_;
  237. Decorator d_;
  238. public:
  239. template<class Handler_, class Buffers>
  240. accept_op(
  241. Handler_&& h,
  242. boost::shared_ptr<impl_type> const& sp,
  243. Decorator const& decorator,
  244. Buffers const& buffers)
  245. : stable_async_base<Handler,
  246. beast::executor_type<stream>>(
  247. std::forward<Handler_>(h),
  248. sp->stream().get_executor())
  249. , wp_(sp)
  250. , p_(beast::allocate_stable<
  251. http::request_parser<http::empty_body>>(*this))
  252. , d_(decorator)
  253. {
  254. auto& impl = *sp;
  255. error_code ec;
  256. auto const mb =
  257. beast::detail::dynamic_buffer_prepare(
  258. impl.rd_buf, buffer_bytes(buffers),
  259. ec, error::buffer_overflow);
  260. if(! ec)
  261. impl.rd_buf.commit(
  262. net::buffer_copy(*mb, buffers));
  263. (*this)(ec);
  264. }
  265. void operator()(
  266. error_code ec = {},
  267. std::size_t bytes_transferred = 0,
  268. bool cont = true)
  269. {
  270. boost::ignore_unused(bytes_transferred);
  271. auto sp = wp_.lock();
  272. if(! sp)
  273. {
  274. ec = net::error::operation_aborted;
  275. return this->complete(cont, ec);
  276. }
  277. auto& impl = *sp;
  278. BOOST_ASIO_CORO_REENTER(*this)
  279. {
  280. impl.change_status(status::handshake);
  281. impl.update_timer(this->get_executor());
  282. // The constructor could have set ec
  283. if(ec)
  284. goto upcall;
  285. BOOST_ASIO_CORO_YIELD
  286. http::async_read(impl.stream(),
  287. impl.rd_buf, p_, std::move(*this));
  288. if(ec == http::error::end_of_stream)
  289. ec = error::closed;
  290. if(impl.check_stop_now(ec))
  291. goto upcall;
  292. {
  293. // Arguments from our state must be
  294. // moved to the stack before releasing
  295. // the handler.
  296. auto const req = p_.release();
  297. auto const decorator = d_;
  298. response_op<Handler>(
  299. this->release_handler(),
  300. sp, req, decorator, true);
  301. return;
  302. }
  303. upcall:
  304. this->complete(cont, ec);
  305. }
  306. }
  307. };
  308. template<class NextLayer, bool deflateSupported>
  309. struct stream<NextLayer, deflateSupported>::
  310. run_response_op
  311. {
  312. template<
  313. class AcceptHandler,
  314. class Body, class Allocator,
  315. class Decorator>
  316. void
  317. operator()(
  318. AcceptHandler&& h,
  319. boost::shared_ptr<impl_type> const& sp,
  320. http::request<Body,
  321. http::basic_fields<Allocator>> const* m,
  322. Decorator const& d)
  323. {
  324. // If you get an error on the following line it means
  325. // that your handler does not meet the documented type
  326. // requirements for the handler.
  327. static_assert(
  328. beast::detail::is_invocable<AcceptHandler,
  329. void(error_code)>::value,
  330. "AcceptHandler type requirements not met");
  331. response_op<
  332. typename std::decay<AcceptHandler>::type>(
  333. std::forward<AcceptHandler>(h), sp, *m, d);
  334. }
  335. };
  336. template<class NextLayer, bool deflateSupported>
  337. struct stream<NextLayer, deflateSupported>::
  338. run_accept_op
  339. {
  340. template<
  341. class AcceptHandler,
  342. class Decorator,
  343. class Buffers>
  344. void
  345. operator()(
  346. AcceptHandler&& h,
  347. boost::shared_ptr<impl_type> const& sp,
  348. Decorator const& d,
  349. Buffers const& b)
  350. {
  351. // If you get an error on the following line it means
  352. // that your handler does not meet the documented type
  353. // requirements for the handler.
  354. static_assert(
  355. beast::detail::is_invocable<AcceptHandler,
  356. void(error_code)>::value,
  357. "AcceptHandler type requirements not met");
  358. accept_op<
  359. typename std::decay<AcceptHandler>::type,
  360. Decorator>(
  361. std::forward<AcceptHandler>(h),
  362. sp,
  363. d,
  364. b);
  365. }
  366. };
  367. //------------------------------------------------------------------------------
  368. template<class NextLayer, bool deflateSupported>
  369. template<class Body, class Allocator,
  370. class Decorator>
  371. void
  372. stream<NextLayer, deflateSupported>::
  373. do_accept(
  374. http::request<Body,
  375. http::basic_fields<Allocator>> const& req,
  376. Decorator const& decorator,
  377. error_code& ec)
  378. {
  379. impl_->change_status(status::handshake);
  380. error_code result;
  381. auto const res = impl_->build_response(req, decorator, result);
  382. http::write(impl_->stream(), res, ec);
  383. if(ec)
  384. return;
  385. ec = result;
  386. if(ec)
  387. {
  388. // VFALCO TODO Respect keep alive setting, perform
  389. // teardown if Connection: close.
  390. return;
  391. }
  392. impl_->do_pmd_config(res);
  393. impl_->open(role_type::server);
  394. }
  395. template<class NextLayer, bool deflateSupported>
  396. template<class Buffers, class Decorator>
  397. void
  398. stream<NextLayer, deflateSupported>::
  399. do_accept(
  400. Buffers const& buffers,
  401. Decorator const& decorator,
  402. error_code& ec)
  403. {
  404. impl_->reset();
  405. auto const mb =
  406. beast::detail::dynamic_buffer_prepare(
  407. impl_->rd_buf, buffer_bytes(buffers), ec,
  408. error::buffer_overflow);
  409. if(ec)
  410. return;
  411. impl_->rd_buf.commit(net::buffer_copy(*mb, buffers));
  412. http::request_parser<http::empty_body> p;
  413. http::read(next_layer(), impl_->rd_buf, p, ec);
  414. if(ec == http::error::end_of_stream)
  415. ec = error::closed;
  416. if(ec)
  417. return;
  418. do_accept(p.get(), decorator, ec);
  419. }
  420. //------------------------------------------------------------------------------
  421. template<class NextLayer, bool deflateSupported>
  422. void
  423. stream<NextLayer, deflateSupported>::
  424. accept()
  425. {
  426. static_assert(is_sync_stream<next_layer_type>::value,
  427. "SyncStream type requirements not met");
  428. error_code ec;
  429. accept(ec);
  430. if(ec)
  431. BOOST_THROW_EXCEPTION(system_error{ec});
  432. }
  433. template<class NextLayer, bool deflateSupported>
  434. void
  435. stream<NextLayer, deflateSupported>::
  436. accept(error_code& ec)
  437. {
  438. static_assert(is_sync_stream<next_layer_type>::value,
  439. "SyncStream type requirements not met");
  440. do_accept(
  441. net::const_buffer{},
  442. &default_decorate_res, ec);
  443. }
  444. template<class NextLayer, bool deflateSupported>
  445. template<class ConstBufferSequence>
  446. typename std::enable_if<! http::detail::is_header<
  447. ConstBufferSequence>::value>::type
  448. stream<NextLayer, deflateSupported>::
  449. accept(ConstBufferSequence const& buffers)
  450. {
  451. static_assert(is_sync_stream<next_layer_type>::value,
  452. "SyncStream type requirements not met");
  453. static_assert(net::is_const_buffer_sequence<
  454. ConstBufferSequence>::value,
  455. "ConstBufferSequence type requirements not met");
  456. error_code ec;
  457. accept(buffers, ec);
  458. if(ec)
  459. BOOST_THROW_EXCEPTION(system_error{ec});
  460. }
  461. template<class NextLayer, bool deflateSupported>
  462. template<class ConstBufferSequence>
  463. typename std::enable_if<! http::detail::is_header<
  464. ConstBufferSequence>::value>::type
  465. stream<NextLayer, deflateSupported>::
  466. accept(
  467. ConstBufferSequence const& buffers, error_code& ec)
  468. {
  469. static_assert(is_sync_stream<next_layer_type>::value,
  470. "SyncStream type requirements not met");
  471. static_assert(net::is_const_buffer_sequence<
  472. ConstBufferSequence>::value,
  473. "ConstBufferSequence type requirements not met");
  474. do_accept(buffers, &default_decorate_res, ec);
  475. }
  476. template<class NextLayer, bool deflateSupported>
  477. template<class Body, class Allocator>
  478. void
  479. stream<NextLayer, deflateSupported>::
  480. accept(
  481. http::request<Body,
  482. http::basic_fields<Allocator>> const& req)
  483. {
  484. static_assert(is_sync_stream<next_layer_type>::value,
  485. "SyncStream type requirements not met");
  486. error_code ec;
  487. accept(req, ec);
  488. if(ec)
  489. BOOST_THROW_EXCEPTION(system_error{ec});
  490. }
  491. template<class NextLayer, bool deflateSupported>
  492. template<class Body, class Allocator>
  493. void
  494. stream<NextLayer, deflateSupported>::
  495. accept(
  496. http::request<Body,
  497. http::basic_fields<Allocator>> const& req,
  498. error_code& ec)
  499. {
  500. static_assert(is_sync_stream<next_layer_type>::value,
  501. "SyncStream type requirements not met");
  502. impl_->reset();
  503. do_accept(req, &default_decorate_res, ec);
  504. }
  505. //------------------------------------------------------------------------------
  506. template<class NextLayer, bool deflateSupported>
  507. template<
  508. class AcceptHandler>
  509. BOOST_BEAST_ASYNC_RESULT1(AcceptHandler)
  510. stream<NextLayer, deflateSupported>::
  511. async_accept(
  512. AcceptHandler&& handler)
  513. {
  514. static_assert(is_async_stream<next_layer_type>::value,
  515. "AsyncStream type requirements not met");
  516. impl_->reset();
  517. return net::async_initiate<
  518. AcceptHandler,
  519. void(error_code)>(
  520. run_accept_op{},
  521. handler,
  522. impl_,
  523. &default_decorate_res,
  524. net::const_buffer{});
  525. }
  526. template<class NextLayer, bool deflateSupported>
  527. template<
  528. class ResponseDecorator,
  529. class AcceptHandler>
  530. BOOST_BEAST_ASYNC_RESULT1(AcceptHandler)
  531. stream<NextLayer, deflateSupported>::
  532. async_accept_ex(
  533. ResponseDecorator const& decorator,
  534. AcceptHandler&& handler)
  535. {
  536. static_assert(is_async_stream<next_layer_type>::value,
  537. "AsyncStream type requirements not met");
  538. static_assert(detail::is_response_decorator<
  539. ResponseDecorator>::value,
  540. "ResponseDecorator requirements not met");
  541. impl_->reset();
  542. return net::async_initiate<
  543. AcceptHandler,
  544. void(error_code)>(
  545. run_accept_op{},
  546. handler,
  547. impl_,
  548. decorator,
  549. net::const_buffer{});
  550. }
  551. template<class NextLayer, bool deflateSupported>
  552. template<
  553. class ConstBufferSequence,
  554. class AcceptHandler>
  555. typename std::enable_if<
  556. ! http::detail::is_header<ConstBufferSequence>::value,
  557. BOOST_BEAST_ASYNC_RESULT1(AcceptHandler)>::type
  558. stream<NextLayer, deflateSupported>::
  559. async_accept(
  560. ConstBufferSequence const& buffers,
  561. AcceptHandler&& handler)
  562. {
  563. static_assert(is_async_stream<next_layer_type>::value,
  564. "AsyncStream type requirements not met");
  565. static_assert(net::is_const_buffer_sequence<
  566. ConstBufferSequence>::value,
  567. "ConstBufferSequence type requirements not met");
  568. impl_->reset();
  569. return net::async_initiate<
  570. AcceptHandler,
  571. void(error_code)>(
  572. run_accept_op{},
  573. handler,
  574. impl_,
  575. &default_decorate_res,
  576. buffers);
  577. }
  578. template<class NextLayer, bool deflateSupported>
  579. template<
  580. class ConstBufferSequence,
  581. class ResponseDecorator,
  582. class AcceptHandler>
  583. typename std::enable_if<
  584. ! http::detail::is_header<ConstBufferSequence>::value,
  585. BOOST_BEAST_ASYNC_RESULT1(AcceptHandler)>::type
  586. stream<NextLayer, deflateSupported>::
  587. async_accept_ex(
  588. ConstBufferSequence const& buffers,
  589. ResponseDecorator const& decorator,
  590. AcceptHandler&& handler)
  591. {
  592. static_assert(is_async_stream<next_layer_type>::value,
  593. "AsyncStream type requirements not met");
  594. static_assert(net::is_const_buffer_sequence<
  595. ConstBufferSequence>::value,
  596. "ConstBufferSequence type requirements not met");
  597. static_assert(detail::is_response_decorator<
  598. ResponseDecorator>::value,
  599. "ResponseDecorator requirements not met");
  600. impl_->reset();
  601. return net::async_initiate<
  602. AcceptHandler,
  603. void(error_code)>(
  604. run_accept_op{},
  605. handler,
  606. impl_,
  607. decorator,
  608. buffers);
  609. }
  610. template<class NextLayer, bool deflateSupported>
  611. template<
  612. class Body, class Allocator,
  613. class AcceptHandler>
  614. BOOST_BEAST_ASYNC_RESULT1(AcceptHandler)
  615. stream<NextLayer, deflateSupported>::
  616. async_accept(
  617. http::request<Body, http::basic_fields<Allocator>> const& req,
  618. AcceptHandler&& handler)
  619. {
  620. static_assert(is_async_stream<next_layer_type>::value,
  621. "AsyncStream type requirements not met");
  622. impl_->reset();
  623. return net::async_initiate<
  624. AcceptHandler,
  625. void(error_code)>(
  626. run_response_op{},
  627. handler,
  628. impl_,
  629. &req,
  630. &default_decorate_res);
  631. }
  632. template<class NextLayer, bool deflateSupported>
  633. template<
  634. class Body, class Allocator,
  635. class ResponseDecorator,
  636. class AcceptHandler>
  637. BOOST_BEAST_ASYNC_RESULT1(AcceptHandler)
  638. stream<NextLayer, deflateSupported>::
  639. async_accept_ex(
  640. http::request<Body, http::basic_fields<Allocator>> const& req,
  641. ResponseDecorator const& decorator,
  642. AcceptHandler&& handler)
  643. {
  644. static_assert(is_async_stream<next_layer_type>::value,
  645. "AsyncStream type requirements not met");
  646. static_assert(detail::is_response_decorator<
  647. ResponseDecorator>::value,
  648. "ResponseDecorator requirements not met");
  649. impl_->reset();
  650. return net::async_initiate<
  651. AcceptHandler,
  652. void(error_code)>(
  653. run_response_op{},
  654. handler,
  655. impl_,
  656. &req,
  657. decorator);
  658. }
  659. //------------------------------------------------------------------------------
  660. template<class NextLayer, bool deflateSupported>
  661. template<class ResponseDecorator>
  662. void
  663. stream<NextLayer, deflateSupported>::
  664. accept_ex(ResponseDecorator const& decorator)
  665. {
  666. #ifndef BOOST_BEAST_ALLOW_DEPRECATED
  667. static_assert(sizeof(ResponseDecorator) == 0,
  668. BOOST_BEAST_DEPRECATION_STRING);
  669. #endif
  670. static_assert(is_sync_stream<next_layer_type>::value,
  671. "SyncStream type requirements not met");
  672. static_assert(detail::is_response_decorator<
  673. ResponseDecorator>::value,
  674. "ResponseDecorator requirements not met");
  675. error_code ec;
  676. accept_ex(decorator, ec);
  677. if(ec)
  678. BOOST_THROW_EXCEPTION(system_error{ec});
  679. }
  680. template<class NextLayer, bool deflateSupported>
  681. template<class ResponseDecorator>
  682. void
  683. stream<NextLayer, deflateSupported>::
  684. accept_ex(ResponseDecorator const& decorator, error_code& ec)
  685. {
  686. #ifndef BOOST_BEAST_ALLOW_DEPRECATED
  687. static_assert(sizeof(ResponseDecorator) == 0,
  688. BOOST_BEAST_DEPRECATION_STRING);
  689. #endif
  690. static_assert(is_sync_stream<next_layer_type>::value,
  691. "SyncStream type requirements not met");
  692. static_assert(detail::is_response_decorator<
  693. ResponseDecorator>::value,
  694. "ResponseDecorator requirements not met");
  695. do_accept(
  696. net::const_buffer{},
  697. decorator, ec);
  698. }
  699. template<class NextLayer, bool deflateSupported>
  700. template<
  701. class ConstBufferSequence,
  702. class ResponseDecorator>
  703. typename std::enable_if<! http::detail::is_header<
  704. ConstBufferSequence>::value>::type
  705. stream<NextLayer, deflateSupported>::
  706. accept_ex(
  707. ConstBufferSequence const& buffers,
  708. ResponseDecorator const &decorator)
  709. {
  710. #ifndef BOOST_BEAST_ALLOW_DEPRECATED
  711. static_assert(sizeof(ResponseDecorator) == 0,
  712. BOOST_BEAST_DEPRECATION_STRING);
  713. #endif
  714. static_assert(is_sync_stream<next_layer_type>::value,
  715. "SyncStream type requirements not met");
  716. static_assert(net::is_const_buffer_sequence<
  717. ConstBufferSequence>::value,
  718. "ConstBufferSequence type requirements not met");
  719. static_assert(detail::is_response_decorator<
  720. ResponseDecorator>::value,
  721. "ResponseDecorator requirements not met");
  722. error_code ec;
  723. accept_ex(buffers, decorator, ec);
  724. if(ec)
  725. BOOST_THROW_EXCEPTION(system_error{ec});
  726. }
  727. template<class NextLayer, bool deflateSupported>
  728. template<
  729. class ConstBufferSequence,
  730. class ResponseDecorator>
  731. typename std::enable_if<! http::detail::is_header<
  732. ConstBufferSequence>::value>::type
  733. stream<NextLayer, deflateSupported>::
  734. accept_ex(
  735. ConstBufferSequence const& buffers,
  736. ResponseDecorator const& decorator,
  737. error_code& ec)
  738. {
  739. #ifndef BOOST_BEAST_ALLOW_DEPRECATED
  740. static_assert(sizeof(ResponseDecorator) == 0,
  741. BOOST_BEAST_DEPRECATION_STRING);
  742. #endif
  743. static_assert(is_sync_stream<next_layer_type>::value,
  744. "SyncStream type requirements not met");
  745. static_assert(net::is_const_buffer_sequence<
  746. ConstBufferSequence>::value,
  747. "ConstBufferSequence type requirements not met");
  748. static_assert(net::is_const_buffer_sequence<
  749. ConstBufferSequence>::value,
  750. "ConstBufferSequence type requirements not met");
  751. do_accept(buffers, decorator, ec);
  752. }
  753. template<class NextLayer, bool deflateSupported>
  754. template<
  755. class Body, class Allocator,
  756. class ResponseDecorator>
  757. void
  758. stream<NextLayer, deflateSupported>::
  759. accept_ex(
  760. http::request<Body,
  761. http::basic_fields<Allocator>> const& req,
  762. ResponseDecorator const& decorator)
  763. {
  764. #ifndef BOOST_BEAST_ALLOW_DEPRECATED
  765. static_assert(sizeof(ResponseDecorator) == 0,
  766. BOOST_BEAST_DEPRECATION_STRING);
  767. #endif
  768. static_assert(is_sync_stream<next_layer_type>::value,
  769. "SyncStream type requirements not met");
  770. static_assert(detail::is_response_decorator<
  771. ResponseDecorator>::value,
  772. "ResponseDecorator requirements not met");
  773. error_code ec;
  774. accept_ex(req, decorator, ec);
  775. if(ec)
  776. BOOST_THROW_EXCEPTION(system_error{ec});
  777. }
  778. template<class NextLayer, bool deflateSupported>
  779. template<
  780. class Body, class Allocator,
  781. class ResponseDecorator>
  782. void
  783. stream<NextLayer, deflateSupported>::
  784. accept_ex(
  785. http::request<Body,
  786. http::basic_fields<Allocator>> const& req,
  787. ResponseDecorator const& decorator,
  788. error_code& ec)
  789. {
  790. #ifndef BOOST_BEAST_ALLOW_DEPRECATED
  791. static_assert(sizeof(ResponseDecorator) == 0,
  792. BOOST_BEAST_DEPRECATION_STRING);
  793. #endif
  794. static_assert(is_sync_stream<next_layer_type>::value,
  795. "SyncStream type requirements not met");
  796. static_assert(detail::is_response_decorator<
  797. ResponseDecorator>::value,
  798. "ResponseDecorator requirements not met");
  799. impl_->reset();
  800. do_accept(req, decorator, ec);
  801. }
  802. } // websocket
  803. } // beast
  804. } // boost
  805. #endif