boost/beast/experimental/core/impl/timeout_socket.hpp
//
// Copyright (c) 2018 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_CORE_IMPL_TIMOUT_SOCKET_HPP
#define BOOST_BEAST_CORE_IMPL_TIMOUT_SOCKET_HPP
#include <boost/beast/core/bind_handler.hpp>
#include <boost/beast/core/type_traits.hpp>
#include <boost/beast/experimental/core/detail/timeout_work_guard.hpp>
#include <boost/asio/executor_work_guard.hpp>
#include <memory>
#include <utility>
namespace boost {
namespace beast {
template<class Protocol, class Executor>
template<class Handler>
class basic_timeout_socket<Protocol, Executor>::async_op
{
Handler h_;
basic_timeout_socket& s_;
detail::timeout_work_guard work_;
public:
async_op(async_op&&) = default;
async_op(async_op const&) = delete;
template<class Buffers, class DeducedHandler>
async_op(
Buffers const& b,
DeducedHandler&& h,
basic_timeout_socket& s,
std::true_type)
: h_(std::forward<DeducedHandler>(h))
, s_(s)
, work_(s.rd_timer_)
{
s_.sock_.async_read_some(b, std::move(*this));
}
template<class Buffers, class DeducedHandler>
async_op(
Buffers const& b,
DeducedHandler&& h,
basic_timeout_socket& s,
std::false_type)
: h_(std::forward<DeducedHandler>(h))
, s_(s)
, work_(s.wr_timer_)
{
s_.sock_.async_write_some(b, std::move(*this));
}
using allocator_type =
boost::asio::associated_allocator_t<Handler>;
allocator_type
get_allocator() const noexcept
{
return (boost::asio::get_associated_allocator)(h_);
}
using executor_type =
boost::asio::associated_executor_t<Handler, decltype(
std::declval<basic_timeout_socket<Protocol>&>().get_executor())>;
executor_type
get_executor() const noexcept
{
return (boost::asio::get_associated_executor)(
h_, s_.get_executor());
}
friend
bool asio_handler_is_continuation(async_op* op)
{
using boost::asio::asio_handler_is_continuation;
return asio_handler_is_continuation(
std::addressof(op->h_));
}
template<class Function>
friend
void asio_handler_invoke(Function&& f, async_op* op)
{
using boost::asio::asio_handler_invoke;
asio_handler_invoke(f, std::addressof(op->h_));
}
void
operator()(error_code ec, std::size_t bytes_transferred)
{
if(s_.expired_)
{
BOOST_ASSERT(ec == boost::asio::error::operation_aborted);
ec = boost::asio::error::timed_out;
}
else
{
work_.complete();
}
h_(ec, bytes_transferred);
}
};
//------------------------------------------------------------------------------
template<class Protocol, class Executor>
basic_timeout_socket<Protocol, Executor>::
timer::
timer(basic_timeout_socket& s)
: detail::timeout_object(s.ex_.context())
, s_(s)
{
}
template<class Protocol, class Executor>
auto
basic_timeout_socket<Protocol, Executor>::
timer::
operator=(timer&&)
-> timer&
{
return *this;
}
template<class Protocol, class Executor>
void
basic_timeout_socket<Protocol, Executor>::
timer::
on_timeout()
{
boost::asio::post(
s_.ex_,
[this]()
{
s_.expired_ = true;
s_.sock_.cancel();
});
}
//------------------------------------------------------------------------------
template<class Protocol, class Executor>
template<class ExecutionContext, class>
basic_timeout_socket<Protocol, Executor>::
basic_timeout_socket(ExecutionContext& ctx)
: ex_(ctx.get_executor())
, rd_timer_(*this)
, wr_timer_(*this)
, sock_(ctx)
{
}
template<class Protocol, class Executor>
template<class MutableBufferSequence, class ReadHandler>
BOOST_ASIO_INITFN_RESULT_TYPE(ReadHandler,
void(boost::system::error_code, std::size_t))
basic_timeout_socket<Protocol, Executor>::
async_read_some(
MutableBufferSequence const& buffers,
ReadHandler&& handler)
{
static_assert(boost::asio::is_mutable_buffer_sequence<
MutableBufferSequence>::value,
"MutableBufferSequence requirements not met");
BOOST_BEAST_HANDLER_INIT(
ReadHandler, void(error_code, std::size_t));
async_op<BOOST_ASIO_HANDLER_TYPE(ReadHandler,
void(error_code, std::size_t))>(buffers,
std::forward<ReadHandler>(handler), *this,
std::true_type{});
return init.result.get();
}
template<class Protocol, class Executor>
template<class ConstBufferSequence, class WriteHandler>
BOOST_ASIO_INITFN_RESULT_TYPE(WriteHandler,
void(boost::system::error_code, std::size_t))
basic_timeout_socket<Protocol, Executor>::
async_write_some(
ConstBufferSequence const& buffers,
WriteHandler&& handler)
{
static_assert(boost::asio::is_const_buffer_sequence<
ConstBufferSequence>::value,
"ConstBufferSequence requirements not met");
BOOST_BEAST_HANDLER_INIT(
WriteHandler, void(error_code, std::size_t));
async_op<BOOST_ASIO_HANDLER_TYPE(WriteHandler,
void(error_code, std::size_t))>(buffers,
std::forward<WriteHandler>(handler), *this,
std::false_type{});
return init.result.get();
}
} // beast
} // boost
#endif