2017-07-20 08:01:46 -07:00
|
|
|
//
|
|
|
|
|
// Copyright (c) 2013-2016 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)
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
#ifndef BEAST_WEBSOCKET_IMPL_STREAM_IPP
|
|
|
|
|
#define BEAST_WEBSOCKET_IMPL_STREAM_IPP
|
|
|
|
|
|
|
|
|
|
#include <beast/websocket/teardown.hpp>
|
|
|
|
|
#include <beast/websocket/detail/hybi13.hpp>
|
2016-10-24 18:41:25 -04:00
|
|
|
#include <beast/websocket/detail/pmd_extension.hpp>
|
2016-05-01 12:33:35 -04:00
|
|
|
#include <beast/http/read.hpp>
|
|
|
|
|
#include <beast/http/write.hpp>
|
|
|
|
|
#include <beast/http/reason.hpp>
|
2016-05-24 06:17:04 -04:00
|
|
|
#include <beast/http/rfc7230.hpp>
|
2016-05-07 14:57:15 -04:00
|
|
|
#include <beast/core/buffer_cat.hpp>
|
|
|
|
|
#include <beast/core/buffer_concepts.hpp>
|
|
|
|
|
#include <beast/core/consuming_buffers.hpp>
|
|
|
|
|
#include <beast/core/prepare_buffers.hpp>
|
|
|
|
|
#include <beast/core/static_streambuf.hpp>
|
|
|
|
|
#include <beast/core/stream_concepts.hpp>
|
2016-11-02 16:02:43 -04:00
|
|
|
#include <beast/core/detail/type_traits.hpp>
|
2016-09-25 12:17:32 -04:00
|
|
|
#include <boost/assert.hpp>
|
2017-07-20 08:01:46 -07:00
|
|
|
#include <boost/endian/buffers.hpp>
|
|
|
|
|
#include <algorithm>
|
|
|
|
|
#include <memory>
|
2016-10-24 18:41:25 -04:00
|
|
|
#include <stdexcept>
|
2017-07-20 08:01:46 -07:00
|
|
|
#include <utility>
|
|
|
|
|
|
|
|
|
|
namespace beast {
|
|
|
|
|
namespace websocket {
|
|
|
|
|
|
|
|
|
|
template<class NextLayer>
|
|
|
|
|
template<class... Args>
|
|
|
|
|
stream<NextLayer>::
|
|
|
|
|
stream(Args&&... args)
|
2016-04-30 13:00:33 -04:00
|
|
|
: stream_(std::forward<Args>(args)...)
|
2017-07-20 08:01:46 -07:00
|
|
|
{
|
2016-04-30 13:00:33 -04:00
|
|
|
}
|
|
|
|
|
|
2016-10-24 18:41:25 -04:00
|
|
|
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)
|
|
|
|
|
throw std::invalid_argument{
|
|
|
|
|
"invalid server_max_window_bits"};
|
|
|
|
|
if( o.client_max_window_bits > 15 ||
|
|
|
|
|
o.client_max_window_bits < 9)
|
|
|
|
|
throw std::invalid_argument{
|
|
|
|
|
"invalid client_max_window_bits"};
|
|
|
|
|
if( o.compLevel < 0 ||
|
|
|
|
|
o.compLevel > 9)
|
|
|
|
|
throw std::invalid_argument{
|
|
|
|
|
"invalid compLevel"};
|
|
|
|
|
if( o.memLevel < 1 ||
|
|
|
|
|
o.memLevel > 9)
|
|
|
|
|
throw std::invalid_argument{
|
|
|
|
|
"invalid memLevel"};
|
|
|
|
|
pmd_opts_ = o;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-20 08:01:46 -07:00
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
|
|
2016-05-15 16:22:25 -04:00
|
|
|
template<class NextLayer>
|
|
|
|
|
void
|
|
|
|
|
stream<NextLayer>::
|
|
|
|
|
reset()
|
|
|
|
|
{
|
|
|
|
|
failed_ = false;
|
2016-10-24 18:41:25 -04:00
|
|
|
rd_.cont = false;
|
2016-05-15 16:22:25 -04:00
|
|
|
wr_close_ = false;
|
2016-06-10 15:48:39 -04:00
|
|
|
wr_.cont = false;
|
2016-05-15 16:22:25 -04:00
|
|
|
wr_block_ = nullptr; // should be nullptr on close anyway
|
|
|
|
|
pong_data_ = nullptr; // should be nullptr on close anyway
|
|
|
|
|
|
|
|
|
|
stream_.buffer().consume(
|
|
|
|
|
stream_.buffer().size());
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-20 08:01:46 -07:00
|
|
|
template<class NextLayer>
|
2016-10-09 06:34:35 -04:00
|
|
|
http::request<http::empty_body>
|
2017-07-20 08:01:46 -07:00
|
|
|
stream<NextLayer>::
|
|
|
|
|
build_request(boost::string_ref const& host,
|
|
|
|
|
boost::string_ref const& resource, std::string& key)
|
|
|
|
|
{
|
2016-10-09 06:34:35 -04:00
|
|
|
http::request<http::empty_body> req;
|
2016-08-26 07:57:07 -04:00
|
|
|
req.url = { resource.data(), resource.size() };
|
2017-07-20 08:01:46 -07:00
|
|
|
req.version = 11;
|
2016-04-29 06:04:40 -04:00
|
|
|
req.method = "GET";
|
2016-11-10 05:34:49 -05:00
|
|
|
req.fields.insert("Host", host);
|
|
|
|
|
req.fields.insert("Upgrade", "websocket");
|
2017-07-20 08:01:46 -07:00
|
|
|
key = detail::make_sec_ws_key(maskgen_);
|
2016-11-10 05:34:49 -05:00
|
|
|
req.fields.insert("Sec-WebSocket-Key", key);
|
|
|
|
|
req.fields.insert("Sec-WebSocket-Version", "13");
|
2016-10-24 18:41:25 -04:00
|
|
|
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.fields, config);
|
|
|
|
|
}
|
2017-01-25 09:38:29 -05:00
|
|
|
d_(req);
|
2016-04-29 06:04:40 -04:00
|
|
|
http::prepare(req, http::connection::upgrade);
|
2017-07-20 08:01:46 -07:00
|
|
|
return req;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template<class NextLayer>
|
2016-11-10 05:34:49 -05:00
|
|
|
template<class Body, class Fields>
|
2016-10-09 06:34:35 -04:00
|
|
|
http::response<http::string_body>
|
2017-07-20 08:01:46 -07:00
|
|
|
stream<NextLayer>::
|
2016-11-10 05:34:49 -05:00
|
|
|
build_response(http::request<Body, Fields> const& req)
|
2017-07-20 08:01:46 -07:00
|
|
|
{
|
|
|
|
|
auto err =
|
|
|
|
|
[&](std::string const& text)
|
|
|
|
|
{
|
2016-10-09 06:34:35 -04:00
|
|
|
http::response<http::string_body> res;
|
2016-05-07 15:18:22 -04:00
|
|
|
res.status = 400;
|
|
|
|
|
res.reason = http::reason_string(res.status);
|
|
|
|
|
res.version = req.version;
|
|
|
|
|
res.body = text;
|
2017-01-25 09:38:29 -05:00
|
|
|
d_(res);
|
2016-05-15 16:22:25 -04:00
|
|
|
prepare(res,
|
|
|
|
|
(is_keep_alive(req) && keep_alive_) ?
|
|
|
|
|
http::connection::keep_alive :
|
|
|
|
|
http::connection::close);
|
2016-05-07 15:18:22 -04:00
|
|
|
return res;
|
2017-07-20 08:01:46 -07:00
|
|
|
};
|
|
|
|
|
if(req.version < 11)
|
|
|
|
|
return err("HTTP version 1.1 required");
|
2016-04-29 06:04:40 -04:00
|
|
|
if(req.method != "GET")
|
2017-07-20 08:01:46 -07:00
|
|
|
return err("Wrong method");
|
|
|
|
|
if(! is_upgrade(req))
|
|
|
|
|
return err("Expected Upgrade request");
|
2016-11-10 05:34:49 -05:00
|
|
|
if(! req.fields.exists("Host"))
|
2017-07-20 08:01:46 -07:00
|
|
|
return err("Missing Host");
|
2016-11-10 05:34:49 -05:00
|
|
|
if(! req.fields.exists("Sec-WebSocket-Key"))
|
2017-07-20 08:01:46 -07:00
|
|
|
return err("Missing Sec-WebSocket-Key");
|
2016-11-10 05:34:49 -05:00
|
|
|
if(! http::token_list{req.fields["Upgrade"]}.exists("websocket"))
|
2016-05-15 16:22:25 -04:00
|
|
|
return err("Missing websocket Upgrade token");
|
2017-07-20 08:01:46 -07:00
|
|
|
{
|
|
|
|
|
auto const version =
|
2016-11-10 05:34:49 -05:00
|
|
|
req.fields["Sec-WebSocket-Version"];
|
2017-07-20 08:01:46 -07:00
|
|
|
if(version.empty())
|
|
|
|
|
return err("Missing Sec-WebSocket-Version");
|
|
|
|
|
if(version != "13")
|
2016-05-15 16:22:25 -04:00
|
|
|
{
|
2016-10-09 06:34:35 -04:00
|
|
|
http::response<http::string_body> res;
|
2016-05-15 16:22:25 -04:00
|
|
|
res.status = 426;
|
|
|
|
|
res.reason = http::reason_string(res.status);
|
|
|
|
|
res.version = req.version;
|
2016-11-10 05:34:49 -05:00
|
|
|
res.fields.insert("Sec-WebSocket-Version", "13");
|
2016-10-24 18:41:25 -04:00
|
|
|
d_(res);
|
2016-05-15 16:22:25 -04:00
|
|
|
prepare(res,
|
|
|
|
|
(is_keep_alive(req) && keep_alive_) ?
|
|
|
|
|
http::connection::keep_alive :
|
|
|
|
|
http::connection::close);
|
|
|
|
|
return res;
|
|
|
|
|
}
|
2017-07-20 08:01:46 -07:00
|
|
|
}
|
2016-10-09 06:34:35 -04:00
|
|
|
http::response<http::string_body> res;
|
2016-10-24 18:41:25 -04:00
|
|
|
{
|
|
|
|
|
detail::pmd_offer offer;
|
|
|
|
|
detail::pmd_offer unused;
|
|
|
|
|
pmd_read(offer, req.fields);
|
|
|
|
|
pmd_negotiate(
|
|
|
|
|
res.fields, unused, offer, pmd_opts_);
|
|
|
|
|
}
|
2016-05-07 15:18:22 -04:00
|
|
|
res.status = 101;
|
|
|
|
|
res.reason = http::reason_string(res.status);
|
|
|
|
|
res.version = req.version;
|
2016-11-10 05:34:49 -05:00
|
|
|
res.fields.insert("Upgrade", "websocket");
|
2017-07-20 08:01:46 -07:00
|
|
|
{
|
|
|
|
|
auto const key =
|
2016-11-10 05:34:49 -05:00
|
|
|
req.fields["Sec-WebSocket-Key"];
|
|
|
|
|
res.fields.insert("Sec-WebSocket-Accept",
|
2017-07-20 08:01:46 -07:00
|
|
|
detail::make_sec_ws_accept(key));
|
|
|
|
|
}
|
2016-11-10 05:34:49 -05:00
|
|
|
res.fields.replace("Server", "Beast.WSProto");
|
2017-01-25 09:38:29 -05:00
|
|
|
d_(res);
|
2016-05-07 15:18:22 -04:00
|
|
|
http::prepare(res, http::connection::upgrade);
|
|
|
|
|
return res;
|
2017-07-20 08:01:46 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template<class NextLayer>
|
2016-11-10 05:34:49 -05:00
|
|
|
template<class Body, class Fields>
|
2017-07-20 08:01:46 -07:00
|
|
|
void
|
|
|
|
|
stream<NextLayer>::
|
2016-11-10 05:34:49 -05:00
|
|
|
do_response(http::response<Body, Fields> const& res,
|
2017-07-20 08:01:46 -07:00
|
|
|
boost::string_ref const& key, error_code& ec)
|
|
|
|
|
{
|
|
|
|
|
// VFALCO Review these error codes
|
|
|
|
|
auto fail = [&]{ ec = error::response_failed; };
|
2016-05-15 16:22:25 -04:00
|
|
|
if(res.version < 11)
|
|
|
|
|
return fail();
|
2016-05-07 15:18:22 -04:00
|
|
|
if(res.status != 101)
|
2017-07-20 08:01:46 -07:00
|
|
|
return fail();
|
2016-05-07 15:18:22 -04:00
|
|
|
if(! is_upgrade(res))
|
2017-07-20 08:01:46 -07:00
|
|
|
return fail();
|
2016-11-10 05:34:49 -05:00
|
|
|
if(! http::token_list{res.fields["Upgrade"]}.exists("websocket"))
|
2017-07-20 08:01:46 -07:00
|
|
|
return fail();
|
2016-11-10 05:34:49 -05:00
|
|
|
if(! res.fields.exists("Sec-WebSocket-Accept"))
|
2017-07-20 08:01:46 -07:00
|
|
|
return fail();
|
2016-11-10 05:34:49 -05:00
|
|
|
if(res.fields["Sec-WebSocket-Accept"] !=
|
2017-07-20 08:01:46 -07:00
|
|
|
detail::make_sec_ws_accept(key))
|
|
|
|
|
return fail();
|
2016-10-24 18:41:25 -04:00
|
|
|
detail::pmd_offer offer;
|
|
|
|
|
pmd_read(offer, res.fields);
|
|
|
|
|
// VFALCO see if offer satisfies pmd_config_,
|
|
|
|
|
// return an error if not.
|
|
|
|
|
pmd_config_ = offer; // overwrite for now
|
2016-05-15 16:22:25 -04:00
|
|
|
open(detail::role_type::client);
|
2017-07-20 08:01:46 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // websocket
|
|
|
|
|
} // beast
|
|
|
|
|
|
|
|
|
|
#endif
|