mirror of
https://github.com/XRPLF/clio.git
synced 2025-11-05 04:15:51 +00:00
Fixes #1704. Please review and commit clang-tidy fixes. Co-authored-by: kuznetsss <15742918+kuznetsss@users.noreply.github.com>
215 lines
7.2 KiB
C++
215 lines
7.2 KiB
C++
//------------------------------------------------------------------------------
|
|
/*
|
|
This file is part of clio: https://github.com/XRPLF/clio
|
|
Copyright (c) 2024, the clio developers.
|
|
|
|
Permission to use, copy, modify, and distribute this software for any
|
|
purpose with or without fee is hereby granted, provided that the above
|
|
copyright notice and this permission notice appear in all copies.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
*/
|
|
//==============================================================================
|
|
|
|
#include "util/TestWebSocketClient.hpp"
|
|
|
|
#include "util/Assert.hpp"
|
|
#include "util/TestHttpClient.hpp"
|
|
#include "util/WithTimeout.hpp"
|
|
|
|
#include <boost/asio/buffer.hpp>
|
|
#include <boost/asio/io_context.hpp>
|
|
#include <boost/asio/ip/tcp.hpp>
|
|
#include <boost/asio/spawn.hpp>
|
|
#include <boost/asio/ssl/context.hpp>
|
|
#include <boost/asio/ssl/stream_base.hpp>
|
|
#include <boost/asio/ssl/verify_mode.hpp>
|
|
#include <boost/beast/core/buffers_to_string.hpp>
|
|
#include <boost/beast/core/flat_buffer.hpp>
|
|
#include <boost/beast/core/stream_traits.hpp>
|
|
#include <boost/beast/http/field.hpp>
|
|
#include <boost/beast/version.hpp>
|
|
#include <boost/beast/websocket/rfc6455.hpp>
|
|
#include <boost/beast/websocket/stream.hpp>
|
|
#include <boost/beast/websocket/stream_base.hpp>
|
|
#include <fmt/core.h>
|
|
|
|
#include <chrono>
|
|
#include <optional>
|
|
#include <string>
|
|
#include <string_view>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
namespace http = boost::beast::http;
|
|
namespace net = boost::asio;
|
|
namespace ssl = boost::asio::ssl;
|
|
using tcp = boost::asio::ip::tcp;
|
|
|
|
void
|
|
WebSocketSyncClient::connect(std::string const& host, std::string const& port, std::vector<WebHeader> additionalHeaders)
|
|
{
|
|
auto const results = resolver_.resolve(host, port);
|
|
auto const ep = net::connect(ws_.next_layer(), results);
|
|
|
|
// Update the host_ string. This will provide the value of the
|
|
// Host HTTP header during the WebSocket handshake.
|
|
// See https://tools.ietf.org/html/rfc7230#section-5.4
|
|
auto const hostPort = host + ':' + std::to_string(ep.port());
|
|
|
|
ws_.set_option(boost::beast::websocket::stream_base::decorator([additionalHeaders = std::move(additionalHeaders
|
|
)](boost::beast::websocket::request_type& req) {
|
|
req.set(http::field::user_agent, std::string(BOOST_BEAST_VERSION_STRING) + " websocket-client-coro");
|
|
for (auto const& header : additionalHeaders) {
|
|
req.set(header.name, header.value);
|
|
}
|
|
}));
|
|
|
|
ws_.handshake(hostPort, "/");
|
|
}
|
|
|
|
void
|
|
WebSocketSyncClient::disconnect()
|
|
{
|
|
ws_.close(boost::beast::websocket::close_code::normal);
|
|
}
|
|
|
|
std::string
|
|
WebSocketSyncClient::syncPost(std::string const& body)
|
|
{
|
|
boost::beast::flat_buffer buffer;
|
|
|
|
ws_.write(net::buffer(std::string(body)));
|
|
ws_.read(buffer);
|
|
|
|
return boost::beast::buffers_to_string(buffer.data());
|
|
}
|
|
|
|
void
|
|
WebServerSslSyncClient::connect(std::string const& host, std::string const& port)
|
|
{
|
|
boost::asio::ssl::context ctx(boost::asio::ssl::context::sslv23);
|
|
ctx.set_default_verify_paths();
|
|
ctx.set_verify_mode(ssl::verify_none);
|
|
|
|
tcp::resolver resolver{ioc_};
|
|
ws_.emplace(ioc_, ctx);
|
|
|
|
auto const results = resolver.resolve(host, port);
|
|
net::connect(ws_->next_layer().next_layer(), results.begin(), results.end());
|
|
ws_->next_layer().handshake(ssl::stream_base::client);
|
|
|
|
ws_->set_option(boost::beast::websocket::stream_base::decorator([](boost::beast::websocket::request_type& req) {
|
|
req.set(http::field::user_agent, std::string(BOOST_BEAST_VERSION_STRING) + " websocket-client-coro");
|
|
}));
|
|
|
|
ws_->handshake(host, "/");
|
|
}
|
|
|
|
void
|
|
WebServerSslSyncClient::disconnect()
|
|
{
|
|
ws_->close(boost::beast::websocket::close_code::normal);
|
|
}
|
|
|
|
std::string
|
|
WebServerSslSyncClient::syncPost(std::string const& body)
|
|
{
|
|
boost::beast::flat_buffer buffer;
|
|
ws_->write(net::buffer(std::string(body)));
|
|
ws_->read(buffer);
|
|
|
|
return boost::beast::buffers_to_string(buffer.data());
|
|
}
|
|
|
|
WebSocketAsyncClient::WebSocketAsyncClient(boost::asio::io_context& ioContext) : stream_{ioContext}
|
|
{
|
|
}
|
|
|
|
std::optional<boost::system::error_code>
|
|
WebSocketAsyncClient::connect(
|
|
std::string const& host,
|
|
std::string const& port,
|
|
boost::asio::yield_context yield,
|
|
std::chrono::steady_clock::duration timeout,
|
|
std::vector<WebHeader> additionalHeaders
|
|
)
|
|
{
|
|
auto const results = boost::asio::ip::tcp::resolver{yield.get_executor()}.resolve(host, port);
|
|
ASSERT(not results.empty(), "Could not resolve {}:{}", host, port);
|
|
|
|
boost::system::error_code error;
|
|
boost::beast::get_lowest_layer(stream_).expires_after(timeout);
|
|
stream_.next_layer().async_connect(results, yield[error]);
|
|
if (error)
|
|
return error;
|
|
|
|
boost::beast::websocket::stream_base::timeout wsTimeout{};
|
|
stream_.get_option(wsTimeout);
|
|
wsTimeout.handshake_timeout = timeout;
|
|
stream_.set_option(wsTimeout);
|
|
boost::beast::get_lowest_layer(stream_).expires_never();
|
|
|
|
stream_.set_option(boost::beast::websocket::stream_base::decorator([additionalHeaders = std::move(additionalHeaders
|
|
)](boost::beast::websocket::request_type& req) {
|
|
for (auto const& header : additionalHeaders) {
|
|
req.set(header.name, header.value);
|
|
}
|
|
}));
|
|
stream_.async_handshake(fmt::format("{}:{}", host, port), "/", yield[error]);
|
|
|
|
if (error)
|
|
return error;
|
|
|
|
return std::nullopt;
|
|
}
|
|
|
|
std::optional<boost::system::error_code>
|
|
WebSocketAsyncClient::send(
|
|
boost::asio::yield_context yield,
|
|
std::string_view message,
|
|
std::chrono::steady_clock::duration timeout
|
|
)
|
|
{
|
|
auto const error = util::withTimeout(
|
|
[this, &message](auto&& cyield) { stream_.async_write(net::buffer(message), cyield); }, yield, timeout
|
|
);
|
|
|
|
if (error)
|
|
return error;
|
|
return std::nullopt;
|
|
}
|
|
|
|
std::expected<std::string, boost::system::error_code>
|
|
WebSocketAsyncClient::receive(boost::asio::yield_context yield, std::chrono::steady_clock::duration timeout)
|
|
{
|
|
boost::beast::flat_buffer buffer{};
|
|
auto error =
|
|
util::withTimeout([this, &buffer](auto&& cyield) { stream_.async_read(buffer, cyield); }, yield, timeout);
|
|
if (error)
|
|
return std::unexpected{error};
|
|
return boost::beast::buffers_to_string(buffer.data());
|
|
}
|
|
|
|
void
|
|
WebSocketAsyncClient::gracefulClose(boost::asio::yield_context yield, std::chrono::steady_clock::duration timeout)
|
|
{
|
|
boost::beast::websocket::stream_base::timeout wsTimeout{};
|
|
stream_.get_option(wsTimeout);
|
|
wsTimeout.handshake_timeout = timeout;
|
|
stream_.set_option(wsTimeout);
|
|
stream_.async_close(boost::beast::websocket::close_code::normal, yield);
|
|
}
|
|
|
|
void
|
|
WebSocketAsyncClient::close()
|
|
{
|
|
boost::beast::get_lowest_layer(stream_).close();
|
|
}
|