summaryrefslogtreecommitdiff
path: root/boost/beast/websocket/impl/stream.ipp
diff options
context:
space:
mode:
Diffstat (limited to 'boost/beast/websocket/impl/stream.ipp')
-rw-r--r--boost/beast/websocket/impl/stream.ipp717
1 files changed, 717 insertions, 0 deletions
diff --git a/boost/beast/websocket/impl/stream.ipp b/boost/beast/websocket/impl/stream.ipp
new file mode 100644
index 0000000000..f8e4a5fc01
--- /dev/null
+++ b/boost/beast/websocket/impl/stream.ipp
@@ -0,0 +1,717 @@
+//
+// 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_STREAM_IPP
+#define BOOST_BEAST_WEBSOCKET_IMPL_STREAM_IPP
+
+#include <boost/beast/websocket/rfc6455.hpp>
+#include <boost/beast/websocket/teardown.hpp>
+#include <boost/beast/websocket/detail/hybi13.hpp>
+#include <boost/beast/websocket/detail/pmd_extension.hpp>
+#include <boost/beast/version.hpp>
+#include <boost/beast/http/read.hpp>
+#include <boost/beast/http/write.hpp>
+#include <boost/beast/http/rfc7230.hpp>
+#include <boost/beast/core/buffers_cat.hpp>
+#include <boost/beast/core/buffers_prefix.hpp>
+#include <boost/beast/core/buffers_suffix.hpp>
+#include <boost/beast/core/flat_static_buffer.hpp>
+#include <boost/beast/core/type_traits.hpp>
+#include <boost/beast/core/detail/clamp.hpp>
+#include <boost/beast/core/detail/type_traits.hpp>
+#include <boost/assert.hpp>
+#include <boost/endian/buffers.hpp>
+#include <boost/make_unique.hpp>
+#include <boost/throw_exception.hpp>
+#include <algorithm>
+#include <memory>
+#include <stdexcept>
+#include <utility>
+
+#include <iostream>
+
+namespace boost {
+namespace beast {
+namespace websocket {
+
+template<class NextLayer>
+template<class... Args>
+stream<NextLayer>::
+stream(Args&&... args)
+ : stream_(std::forward<Args>(args)...)
+ , tok_(1)
+{
+ BOOST_ASSERT(rd_buf_.max_size() >=
+ max_control_frame_size);
+}
+
+template<class NextLayer>
+std::size_t
+stream<NextLayer>::
+read_size_hint(
+ std::size_t initial_size) const
+{
+ using beast::detail::clamp;
+ std::size_t result;
+ BOOST_ASSERT(initial_size > 0);
+ if(! pmd_ || (! rd_done_ && ! pmd_->rd_set))
+ {
+ // current message is uncompressed
+
+ if(rd_done_)
+ {
+ // first message frame
+ result = initial_size;
+ goto done;
+ }
+ else if(rd_fh_.fin)
+ {
+ // last message frame
+ BOOST_ASSERT(rd_remain_ > 0);
+ result = clamp(rd_remain_);
+ goto done;
+ }
+ }
+ result = (std::max)(
+ initial_size, clamp(rd_remain_));
+done:
+ BOOST_ASSERT(result != 0);
+ return result;
+}
+
+template<class NextLayer>
+template<class DynamicBuffer, class>
+std::size_t
+stream<NextLayer>::
+read_size_hint(DynamicBuffer& buffer) const
+{
+ static_assert(
+ boost::asio::is_dynamic_buffer<DynamicBuffer>::value,
+ "DynamicBuffer requirements not met");
+ auto const initial_size = (std::min)(
+ +tcp_frame_size,
+ buffer.max_size() - buffer.size());
+ if(initial_size == 0)
+ return 1; // buffer is full
+ return read_size_hint(initial_size);
+}
+
+template<class NextLayer>
+void
+stream<NextLayer>::
+set_option(permessage_deflate const& o)
+{
+ if( o.server_max_window_bits > 15 ||
+ o.server_max_window_bits < 9)
+ BOOST_THROW_EXCEPTION(std::invalid_argument{
+ "invalid server_max_window_bits"});
+ if( o.client_max_window_bits > 15 ||
+ o.client_max_window_bits < 9)
+ BOOST_THROW_EXCEPTION(std::invalid_argument{
+ "invalid client_max_window_bits"});
+ if( o.compLevel < 0 ||
+ o.compLevel > 9)
+ BOOST_THROW_EXCEPTION(std::invalid_argument{
+ "invalid compLevel"});
+ if( o.memLevel < 1 ||
+ o.memLevel > 9)
+ BOOST_THROW_EXCEPTION(std::invalid_argument{
+ "invalid memLevel"});
+ pmd_opts_ = o;
+}
+
+//------------------------------------------------------------------------------
+
+template<class NextLayer>
+void
+stream<NextLayer>::
+open(role_type role)
+{
+ // VFALCO TODO analyze and remove dupe code in reset()
+ role_ = role;
+ status_ = status::open;
+ rd_remain_ = 0;
+ rd_cont_ = false;
+ rd_done_ = true;
+ // Can't clear this because accept uses it
+ //rd_buf_.reset();
+ rd_fh_.fin = false;
+ rd_close_ = false;
+ wr_close_ = false;
+ wr_block_.reset();
+ rd_block_.reset();
+ cr_.code = close_code::none;
+
+ wr_cont_ = false;
+ wr_buf_size_ = 0;
+
+ if(((role_ == role_type::client && pmd_opts_.client_enable) ||
+ (role_ == role_type::server && pmd_opts_.server_enable)) &&
+ pmd_config_.accept)
+ {
+ pmd_normalize(pmd_config_);
+ pmd_.reset(new pmd_t);
+ if(role_ == role_type::client)
+ {
+ pmd_->zi.reset(
+ pmd_config_.server_max_window_bits);
+ pmd_->zo.reset(
+ pmd_opts_.compLevel,
+ pmd_config_.client_max_window_bits,
+ pmd_opts_.memLevel,
+ zlib::Strategy::normal);
+ }
+ else
+ {
+ pmd_->zi.reset(
+ pmd_config_.client_max_window_bits);
+ pmd_->zo.reset(
+ pmd_opts_.compLevel,
+ pmd_config_.server_max_window_bits,
+ pmd_opts_.memLevel,
+ zlib::Strategy::normal);
+ }
+ }
+}
+
+template<class NextLayer>
+void
+stream<NextLayer>::
+close()
+{
+ wr_buf_.reset();
+ pmd_.reset();
+}
+
+template<class NextLayer>
+void
+stream<NextLayer>::
+reset()
+{
+ BOOST_ASSERT(status_ != status::open);
+ rd_remain_ = 0;
+ rd_cont_ = false;
+ rd_done_ = true;
+ rd_buf_.consume(rd_buf_.size());
+ rd_fh_.fin = false;
+ rd_close_ = false;
+ wr_close_ = false;
+ wr_cont_ = false;
+ wr_block_.reset();
+ rd_block_.reset();
+ cr_.code = close_code::none;
+}
+
+// Called before each write frame
+template<class NextLayer>
+void
+stream<NextLayer>::
+begin_msg()
+{
+ wr_frag_ = wr_frag_opt_;
+ wr_compress_ = static_cast<bool>(pmd_);
+
+ // Maintain the write buffer
+ if( wr_compress_ ||
+ role_ == role_type::client)
+ {
+ if(! wr_buf_ || wr_buf_size_ != wr_buf_opt_)
+ {
+ wr_buf_size_ = wr_buf_opt_;
+ wr_buf_ = boost::make_unique_noinit<
+ std::uint8_t[]>(wr_buf_size_);
+ }
+ }
+ else
+ {
+ wr_buf_size_ = wr_buf_opt_;
+ wr_buf_.reset();
+ }
+}
+
+//------------------------------------------------------------------------------
+
+// Attempt to read a complete frame header.
+// Returns `false` if more bytes are needed
+template<class NextLayer>
+template<class DynamicBuffer>
+bool
+stream<NextLayer>::
+parse_fh(
+ detail::frame_header& fh,
+ DynamicBuffer& b,
+ close_code& code)
+{
+ using boost::asio::buffer;
+ using boost::asio::buffer_copy;
+ using boost::asio::buffer_size;
+ auto const err =
+ [&](close_code cv)
+ {
+ code = cv;
+ return false;
+ };
+ if(buffer_size(b.data()) < 2)
+ {
+ code = close_code::none;
+ return false;
+ }
+ buffers_suffix<typename
+ DynamicBuffer::const_buffers_type> cb{
+ b.data()};
+ std::size_t need;
+ {
+ std::uint8_t tmp[2];
+ cb.consume(buffer_copy(buffer(tmp), cb));
+ fh.len = tmp[1] & 0x7f;
+ switch(fh.len)
+ {
+ case 126: need = 2; break;
+ case 127: need = 8; break;
+ default:
+ need = 0;
+ }
+ fh.mask = (tmp[1] & 0x80) != 0;
+ if(fh.mask)
+ need += 4;
+ if(buffer_size(cb) < need)
+ {
+ code = close_code::none;
+ return false;
+ }
+ fh.op = static_cast<
+ detail::opcode>(tmp[0] & 0x0f);
+ fh.fin = (tmp[0] & 0x80) != 0;
+ fh.rsv1 = (tmp[0] & 0x40) != 0;
+ fh.rsv2 = (tmp[0] & 0x20) != 0;
+ fh.rsv3 = (tmp[0] & 0x10) != 0;
+ }
+ switch(fh.op)
+ {
+ case detail::opcode::binary:
+ case detail::opcode::text:
+ if(rd_cont_)
+ {
+ // new data frame when continuation expected
+ return err(close_code::protocol_error);
+ }
+ if((fh.rsv1 && ! pmd_) ||
+ fh.rsv2 || fh.rsv3)
+ {
+ // reserved bits not cleared
+ return err(close_code::protocol_error);
+ }
+ if(pmd_)
+ pmd_->rd_set = fh.rsv1;
+ break;
+
+ case detail::opcode::cont:
+ if(! rd_cont_)
+ {
+ // continuation without an active message
+ return err(close_code::protocol_error);
+ }
+ if(fh.rsv1 || fh.rsv2 || fh.rsv3)
+ {
+ // reserved bits not cleared
+ return err(close_code::protocol_error);
+ }
+ break;
+
+ default:
+ if(detail::is_reserved(fh.op))
+ {
+ // reserved opcode
+ return err(close_code::protocol_error);
+ }
+ if(! fh.fin)
+ {
+ // fragmented control message
+ return err(close_code::protocol_error);
+ }
+ if(fh.len > 125)
+ {
+ // invalid length for control message
+ return err(close_code::protocol_error);
+ }
+ if(fh.rsv1 || fh.rsv2 || fh.rsv3)
+ {
+ // reserved bits not cleared
+ return err(close_code::protocol_error);
+ }
+ break;
+ }
+ // unmasked frame from client
+ if(role_ == role_type::server && ! fh.mask)
+ return err(close_code::protocol_error);
+ // masked frame from server
+ if(role_ == role_type::client && fh.mask)
+ return err(close_code::protocol_error);
+ if(detail::is_control(fh.op) &&
+ buffer_size(cb) < need + fh.len)
+ {
+ // Make the entire control frame payload
+ // get read in before we return `true`
+ return false;
+ }
+ switch(fh.len)
+ {
+ case 126:
+ {
+ std::uint8_t tmp[2];
+ BOOST_ASSERT(buffer_size(cb) >= sizeof(tmp));
+ cb.consume(buffer_copy(buffer(tmp), cb));
+ fh.len = detail::big_uint16_to_native(&tmp[0]);
+ // length not canonical
+ if(fh.len < 126)
+ return err(close_code::protocol_error);
+ break;
+ }
+ case 127:
+ {
+ std::uint8_t tmp[8];
+ BOOST_ASSERT(buffer_size(cb) >= sizeof(tmp));
+ cb.consume(buffer_copy(buffer(tmp), cb));
+ fh.len = detail::big_uint64_to_native(&tmp[0]);
+ // length not canonical
+ if(fh.len < 65536)
+ return err(close_code::protocol_error);
+ break;
+ }
+ }
+ if(fh.mask)
+ {
+ std::uint8_t tmp[4];
+ BOOST_ASSERT(buffer_size(cb) >= sizeof(tmp));
+ cb.consume(buffer_copy(buffer(tmp), cb));
+ fh.key = detail::little_uint32_to_native(&tmp[0]);
+ detail::prepare_key(rd_key_, fh.key);
+ }
+ else
+ {
+ // initialize this otherwise operator== breaks
+ fh.key = 0;
+ }
+ if(! detail::is_control(fh.op))
+ {
+ if(fh.op != detail::opcode::cont)
+ {
+ rd_size_ = 0;
+ rd_op_ = fh.op;
+ }
+ else
+ {
+ if(rd_size_ > (std::numeric_limits<
+ std::uint64_t>::max)() - fh.len)
+ return err(close_code::too_big);
+ }
+ if(! pmd_ || ! pmd_->rd_set)
+ {
+ if(rd_msg_max_ && beast::detail::sum_exceeds(
+ rd_size_, fh.len, rd_msg_max_))
+ return err(close_code::too_big);
+ }
+ rd_cont_ = ! fh.fin;
+ rd_remain_ = fh.len;
+ }
+ b.consume(b.size() - buffer_size(cb));
+ code = close_code::none;
+ return true;
+}
+
+template<class NextLayer>
+template<class DynamicBuffer>
+void
+stream<NextLayer>::
+write_close(DynamicBuffer& db, close_reason const& cr)
+{
+ using namespace boost::endian;
+ detail::frame_header fh;
+ fh.op = detail::opcode::close;
+ fh.fin = true;
+ fh.rsv1 = false;
+ fh.rsv2 = false;
+ fh.rsv3 = false;
+ fh.len = cr.code == close_code::none ?
+ 0 : 2 + cr.reason.size();
+ if(role_ == role_type::client)
+ {
+ fh.mask = true;
+ fh.key = wr_gen_();
+ }
+ else
+ {
+ fh.mask = false;
+ }
+ detail::write(db, fh);
+ if(cr.code != close_code::none)
+ {
+ detail::prepared_key key;
+ if(fh.mask)
+ detail::prepare_key(key, fh.key);
+ {
+ std::uint8_t tmp[2];
+ ::new(&tmp[0]) big_uint16_buf_t{
+ (std::uint16_t)cr.code};
+ auto mb = db.prepare(2);
+ boost::asio::buffer_copy(mb,
+ boost::asio::buffer(tmp));
+ if(fh.mask)
+ detail::mask_inplace(mb, key);
+ db.commit(2);
+ }
+ if(! cr.reason.empty())
+ {
+ auto mb = db.prepare(cr.reason.size());
+ boost::asio::buffer_copy(mb,
+ boost::asio::const_buffer(
+ cr.reason.data(), cr.reason.size()));
+ if(fh.mask)
+ detail::mask_inplace(mb, key);
+ db.commit(cr.reason.size());
+ }
+ }
+}
+
+template<class NextLayer>
+template<class DynamicBuffer>
+void
+stream<NextLayer>::
+write_ping(DynamicBuffer& db,
+ detail::opcode code, ping_data const& data)
+{
+ detail::frame_header fh;
+ fh.op = code;
+ fh.fin = true;
+ fh.rsv1 = false;
+ fh.rsv2 = false;
+ fh.rsv3 = false;
+ fh.len = data.size();
+ fh.mask = role_ == role_type::client;
+ if(fh.mask)
+ fh.key = wr_gen_();
+ detail::write(db, fh);
+ if(data.empty())
+ return;
+ detail::prepared_key key;
+ if(fh.mask)
+ detail::prepare_key(key, fh.key);
+ auto mb = db.prepare(data.size());
+ boost::asio::buffer_copy(mb,
+ boost::asio::const_buffer(
+ data.data(), data.size()));
+ if(fh.mask)
+ detail::mask_inplace(mb, key);
+ db.commit(data.size());
+}
+
+//------------------------------------------------------------------------------
+
+template<class NextLayer>
+template<class Decorator>
+request_type
+stream<NextLayer>::
+build_request(detail::sec_ws_key_type& key,
+ string_view host,
+ string_view target,
+ Decorator const& decorator)
+{
+ request_type req;
+ req.target(target);
+ req.version(11);
+ req.method(http::verb::get);
+ req.set(http::field::host, host);
+ req.set(http::field::upgrade, "websocket");
+ req.set(http::field::connection, "upgrade");
+ detail::make_sec_ws_key(key, wr_gen_);
+ req.set(http::field::sec_websocket_key, key);
+ req.set(http::field::sec_websocket_version, "13");
+ if(pmd_opts_.client_enable)
+ {
+ detail::pmd_offer config;
+ config.accept = true;
+ config.server_max_window_bits =
+ pmd_opts_.server_max_window_bits;
+ config.client_max_window_bits =
+ pmd_opts_.client_max_window_bits;
+ config.server_no_context_takeover =
+ pmd_opts_.server_no_context_takeover;
+ config.client_no_context_takeover =
+ pmd_opts_.client_no_context_takeover;
+ detail::pmd_write(req, config);
+ }
+ decorator(req);
+ if(! req.count(http::field::user_agent))
+ req.set(http::field::user_agent,
+ BOOST_BEAST_VERSION_STRING);
+ return req;
+}
+
+template<class NextLayer>
+template<class Body, class Allocator, class Decorator>
+response_type
+stream<NextLayer>::
+build_response(http::request<Body,
+ http::basic_fields<Allocator>> const& req,
+ Decorator const& decorator)
+{
+ auto const decorate =
+ [&decorator](response_type& res)
+ {
+ decorator(res);
+ if(! res.count(http::field::server))
+ {
+ BOOST_STATIC_ASSERT(sizeof(BOOST_BEAST_VERSION_STRING) < 20);
+ static_string<20> s(BOOST_BEAST_VERSION_STRING);
+ res.set(http::field::server, s);
+ }
+ };
+ auto err =
+ [&](std::string const& text)
+ {
+ response_type res;
+ res.version(req.version());
+ res.result(http::status::bad_request);
+ res.body() = text;
+ res.prepare_payload();
+ decorate(res);
+ return res;
+ };
+ if(req.version() < 11)
+ return err("HTTP version 1.1 required");
+ if(req.method() != http::verb::get)
+ return err("Wrong method");
+ if(! is_upgrade(req))
+ return err("Expected Upgrade request");
+ if(! req.count(http::field::host))
+ return err("Missing Host");
+ if(! req.count(http::field::sec_websocket_key))
+ return err("Missing Sec-WebSocket-Key");
+ auto const key = req[http::field::sec_websocket_key];
+ if(key.size() > detail::sec_ws_key_type::max_size_n)
+ return err("Invalid Sec-WebSocket-Key");
+ {
+ auto const version =
+ req[http::field::sec_websocket_version];
+ if(version.empty())
+ return err("Missing Sec-WebSocket-Version");
+ if(version != "13")
+ {
+ response_type res;
+ res.result(http::status::upgrade_required);
+ res.version(req.version());
+ res.set(http::field::sec_websocket_version, "13");
+ res.prepare_payload();
+ decorate(res);
+ return res;
+ }
+ }
+
+ response_type res;
+ {
+ detail::pmd_offer offer;
+ detail::pmd_offer unused;
+ pmd_read(offer, req);
+ pmd_negotiate(res, unused, offer, pmd_opts_);
+ }
+ res.result(http::status::switching_protocols);
+ res.version(req.version());
+ res.set(http::field::upgrade, "websocket");
+ res.set(http::field::connection, "upgrade");
+ {
+ detail::sec_ws_accept_type acc;
+ detail::make_sec_ws_accept(acc, key);
+ res.set(http::field::sec_websocket_accept, acc);
+ }
+ decorate(res);
+ return res;
+}
+
+template<class NextLayer>
+void
+stream<NextLayer>::
+on_response(response_type const& res,
+ detail::sec_ws_key_type const& key, error_code& ec)
+{
+ bool const success = [&]()
+ {
+ if(res.version() < 11)
+ return false;
+ if(res.result() != http::status::switching_protocols)
+ return false;
+ if(! http::token_list{res[http::field::connection]}.exists("upgrade"))
+ return false;
+ if(! http::token_list{res[http::field::upgrade]}.exists("websocket"))
+ return false;
+ if(res.count(http::field::sec_websocket_accept) != 1)
+ return false;
+ detail::sec_ws_accept_type acc;
+ detail::make_sec_ws_accept(acc, key);
+ if(acc.compare(
+ res[http::field::sec_websocket_accept]) != 0)
+ return false;
+ return true;
+ }();
+ if(! success)
+ {
+ ec = error::handshake_failed;
+ return;
+ }
+ ec.assign(0, ec.category());
+ detail::pmd_offer offer;
+ pmd_read(offer, res);
+ // VFALCO see if offer satisfies pmd_config_,
+ // return an error if not.
+ pmd_config_ = offer; // overwrite for now
+ open(role_type::client);
+}
+
+// _Fail the WebSocket Connection_
+template<class NextLayer>
+void
+stream<NextLayer>::
+do_fail(
+ std::uint16_t code, // if set, send a close frame first
+ error_code ev, // error code to use upon success
+ error_code& ec) // set to the error, else set to ev
+{
+ BOOST_ASSERT(ev);
+ status_ = status::closing;
+ if(code != close_code::none && ! wr_close_)
+ {
+ wr_close_ = true;
+ detail::frame_buffer fb;
+ write_close<
+ flat_static_buffer_base>(fb, code);
+ boost::asio::write(stream_, fb.data(), ec);
+ if(! check_ok(ec))
+ return;
+ }
+ using beast::websocket::teardown;
+ teardown(role_, stream_, ec);
+ if(ec == boost::asio::error::eof)
+ {
+ // Rationale:
+ // http://stackoverflow.com/questions/25587403/boost-asio-ssl-async-shutdown-always-finishes-with-an-error
+ ec.assign(0, ec.category());
+ }
+ if(! ec)
+ ec = ev;
+ if(ec && ec != error::closed)
+ status_ = status::failed;
+ else
+ status_ = status::closed;
+ close();
+}
+
+} // websocket
+} // beast
+} // boost
+
+#endif