mirror of
https://github.com/XRPLF/rippled.git
synced 2025-11-22 12:05:53 +00:00
Core: * Test buffer_cat iterator move members HTTP: * Fixed yield / resume in writer * Fixed message serialization with chunked encoding * Test yield / resume in writer * Test all conditional branches during message serialization * Test chunked encoding * Increase coverage on parse_error * Add parse_error::general WebSocket: * Add error::general * Increase coverage in error
188 lines
4.5 KiB
C++
188 lines
4.5 KiB
C++
//
|
|
// 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_SYNC_ECHO_PEER_H_INCLUDED
|
|
#define BEAST_WEBSOCKET_SYNC_ECHO_PEER_H_INCLUDED
|
|
|
|
#include <beast/core/streambuf.hpp>
|
|
#include <beast/websocket.hpp>
|
|
#include <boost/optional.hpp>
|
|
#include <functional>
|
|
#include <iostream>
|
|
#include <memory>
|
|
#include <thread>
|
|
|
|
namespace beast {
|
|
namespace websocket {
|
|
|
|
// Synchronous WebSocket echo client/server
|
|
//
|
|
class sync_echo_peer
|
|
{
|
|
public:
|
|
using endpoint_type = boost::asio::ip::tcp::endpoint;
|
|
using address_type = boost::asio::ip::address;
|
|
using socket_type = boost::asio::ip::tcp::socket;
|
|
|
|
private:
|
|
bool log_ = false;
|
|
boost::asio::io_service ios_;
|
|
socket_type sock_;
|
|
boost::asio::ip::tcp::acceptor acceptor_;
|
|
std::thread thread_;
|
|
|
|
public:
|
|
sync_echo_peer(bool server, endpoint_type ep)
|
|
: sock_(ios_)
|
|
, acceptor_(ios_)
|
|
{
|
|
error_code ec;
|
|
acceptor_.open(ep.protocol(), ec);
|
|
maybe_throw(ec, "open");
|
|
acceptor_.set_option(
|
|
boost::asio::socket_base::reuse_address{true});
|
|
acceptor_.bind(ep, ec);
|
|
maybe_throw(ec, "bind");
|
|
acceptor_.listen(
|
|
boost::asio::socket_base::max_connections, ec);
|
|
maybe_throw(ec, "listen");
|
|
acceptor_.async_accept(sock_,
|
|
std::bind(&sync_echo_peer::on_accept, this,
|
|
beast::asio::placeholders::error));
|
|
thread_ = std::thread{[&]{ ios_.run(); }};
|
|
}
|
|
|
|
~sync_echo_peer()
|
|
{
|
|
error_code ec;
|
|
ios_.dispatch(
|
|
[&]{ acceptor_.close(ec); });
|
|
thread_.join();
|
|
}
|
|
|
|
endpoint_type
|
|
local_endpoint() const
|
|
{
|
|
return acceptor_.local_endpoint();
|
|
}
|
|
|
|
private:
|
|
void
|
|
fail(error_code ec, std::string what)
|
|
{
|
|
if(log_)
|
|
std::cerr <<
|
|
what << ": " << ec.message() << std::endl;
|
|
}
|
|
|
|
void
|
|
fail(int id, error_code ec, std::string what)
|
|
{
|
|
if(log_)
|
|
std::cerr << "#" << std::to_string(id) << " " <<
|
|
what << ": " << ec.message() << std::endl;
|
|
}
|
|
|
|
void
|
|
maybe_throw(error_code ec, std::string what)
|
|
{
|
|
if(ec)
|
|
{
|
|
fail(ec, what);
|
|
throw ec;
|
|
}
|
|
}
|
|
|
|
struct lambda
|
|
{
|
|
int id;
|
|
sync_echo_peer& self;
|
|
socket_type sock;
|
|
boost::asio::io_service::work work;
|
|
|
|
lambda(int id_, sync_echo_peer& self_,
|
|
socket_type&& sock_)
|
|
: id(id_)
|
|
, self(self_)
|
|
, sock(std::move(sock_))
|
|
, work(sock.get_io_service())
|
|
{
|
|
}
|
|
|
|
void operator()()
|
|
{
|
|
self.do_peer(id, std::move(sock));
|
|
}
|
|
};
|
|
|
|
void
|
|
on_accept(error_code ec)
|
|
{
|
|
if(ec == boost::asio::error::operation_aborted)
|
|
return;
|
|
maybe_throw(ec, "accept");
|
|
static int id_ = 0;
|
|
std::thread{lambda{++id_, *this, std::move(sock_)}}.detach();
|
|
acceptor_.async_accept(sock_,
|
|
std::bind(&sync_echo_peer::on_accept, this,
|
|
beast::asio::placeholders::error));
|
|
}
|
|
|
|
struct identity
|
|
{
|
|
template<class Body, class Headers>
|
|
void
|
|
operator()(http::message<true, Body, Headers>& req)
|
|
{
|
|
req.headers.replace("User-Agent", "sync_echo_client");
|
|
}
|
|
|
|
template<class Body, class Headers>
|
|
void
|
|
operator()(http::message<false, Body, Headers>& resp)
|
|
{
|
|
resp.headers.replace("Server", "sync_echo_server");
|
|
}
|
|
};
|
|
|
|
void
|
|
do_peer(int id, socket_type&& sock)
|
|
{
|
|
websocket::stream<socket_type> ws(std::move(sock));
|
|
ws.set_option(decorate(identity{}));
|
|
ws.set_option(read_message_max(64 * 1024 * 1024));
|
|
error_code ec;
|
|
ws.accept(ec);
|
|
if(ec)
|
|
{
|
|
fail(id, ec, "accept");
|
|
return;
|
|
}
|
|
for(;;)
|
|
{
|
|
websocket::opcode op;
|
|
beast::streambuf sb;
|
|
ws.read(op, sb, ec);
|
|
if(ec)
|
|
break;
|
|
ws.set_option(websocket::message_type(op));
|
|
ws.write(sb.data(), ec);
|
|
if(ec)
|
|
break;
|
|
}
|
|
if(ec && ec != websocket::error::closed)
|
|
{
|
|
fail(id, ec, "read");
|
|
}
|
|
}
|
|
};
|
|
|
|
} // websocket
|
|
} // beast
|
|
|
|
#endif
|