mirror of
https://github.com/matrix-construct/construct
synced 2024-11-19 08:21:09 +01:00
ircd::client: Add timer logic to client socket/io.
Note: Not sure if I like two reference counters for each client, ideally, but we'll see where it goes after abstractions.
This commit is contained in:
parent
a9f967d16a
commit
fcd410d656
4 changed files with 96 additions and 19 deletions
|
@ -59,7 +59,9 @@ enum class dc
|
||||||
bool connected(const client &) noexcept;
|
bool connected(const client &) noexcept;
|
||||||
bool disconnect(std::nothrow_t, client &, const dc & = dc::FIN) noexcept;
|
bool disconnect(std::nothrow_t, client &, const dc & = dc::FIN) noexcept;
|
||||||
void disconnect(client &, const dc & = dc::FIN);
|
void disconnect(client &, const dc & = dc::FIN);
|
||||||
void set_recv(client &);
|
void recv_cancel(client &);
|
||||||
|
void recv_next(client &, const std::chrono::milliseconds &timeout);
|
||||||
|
void recv_next(client &);
|
||||||
|
|
||||||
// Destroys a client. This only removes the client from the clients list,
|
// Destroys a client. This only removes the client from the clients list,
|
||||||
// and may result in a destruction and disconnect, or it may not.
|
// and may result in a destruction and disconnect, or it may not.
|
||||||
|
@ -67,7 +69,7 @@ void finished(client &);
|
||||||
|
|
||||||
// Creates a client.
|
// Creates a client.
|
||||||
std::shared_ptr<client> add_client();
|
std::shared_ptr<client> add_client();
|
||||||
std::shared_ptr<client> add_client(std::unique_ptr<struct sock>);
|
std::shared_ptr<client> add_client(std::shared_ptr<struct sock>);
|
||||||
|
|
||||||
using clist = std::list<std::shared_ptr<client>>;
|
using clist = std::list<std::shared_ptr<client>>;
|
||||||
const clist &clients();
|
const clist &clients();
|
||||||
|
|
|
@ -35,17 +35,22 @@ using boost::system::error_code;
|
||||||
using boost::asio::steady_timer;
|
using boost::asio::steady_timer;
|
||||||
|
|
||||||
struct sock
|
struct sock
|
||||||
|
:std::enable_shared_from_this<sock>
|
||||||
{
|
{
|
||||||
using message_flags = boost::asio::socket_base::message_flags;
|
using message_flags = boost::asio::socket_base::message_flags;
|
||||||
|
|
||||||
ip::tcp::socket sd;
|
ip::tcp::socket sd;
|
||||||
steady_timer timer;
|
steady_timer timer;
|
||||||
|
bool timedout;
|
||||||
|
|
||||||
operator const ip::tcp::socket &() const { return sd; }
|
operator const ip::tcp::socket &() const { return sd; }
|
||||||
operator ip::tcp::socket &() { return sd; }
|
operator ip::tcp::socket &() { return sd; }
|
||||||
ip::tcp::endpoint remote() const { return sd.remote_endpoint(); }
|
ip::tcp::endpoint remote() const { return sd.remote_endpoint(); }
|
||||||
ip::tcp::endpoint local() const { return sd.local_endpoint(); }
|
ip::tcp::endpoint local() const { return sd.local_endpoint(); }
|
||||||
|
|
||||||
|
void handle_timeout(const std::weak_ptr<sock>, const error_code &);
|
||||||
|
template<class duration> void set_timeout(const duration &);
|
||||||
|
|
||||||
template<class mutable_buffers> auto recv_some(const mutable_buffers &, const message_flags & = 0);
|
template<class mutable_buffers> auto recv_some(const mutable_buffers &, const message_flags & = 0);
|
||||||
template<class mutable_buffers> auto recv(const mutable_buffers &);
|
template<class mutable_buffers> auto recv(const mutable_buffers &);
|
||||||
|
|
||||||
|
@ -74,7 +79,9 @@ template<class const_buffers>
|
||||||
auto
|
auto
|
||||||
sock::send(const const_buffers &bufs)
|
sock::send(const const_buffers &bufs)
|
||||||
{
|
{
|
||||||
return async_write(sd, bufs, yield(continuation()));
|
const auto ret(async_write(sd, bufs, yield(continuation())));
|
||||||
|
timer.cancel();
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Block until something transmitted, returns amount
|
// Block until something transmitted, returns amount
|
||||||
|
@ -91,7 +98,9 @@ template<class mutable_buffers>
|
||||||
auto
|
auto
|
||||||
sock::recv(const mutable_buffers &bufs)
|
sock::recv(const mutable_buffers &bufs)
|
||||||
{
|
{
|
||||||
return async_read(sd, bufs, yield(continuation()));
|
const auto ret(async_read(sd, bufs, yield(continuation())));
|
||||||
|
timer.cancel();
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Block until something in buffers, returns size
|
// Block until something in buffers, returns size
|
||||||
|
@ -103,6 +112,43 @@ sock::recv_some(const mutable_buffers &bufs,
|
||||||
return sd.async_receive(bufs, flags, yield(continuation()));
|
return sd.async_receive(bufs, flags, yield(continuation()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<class duration>
|
||||||
|
void
|
||||||
|
sock::set_timeout(const duration &t)
|
||||||
|
{
|
||||||
|
if(t < duration(0))
|
||||||
|
return;
|
||||||
|
|
||||||
|
timer.expires_from_now(t);
|
||||||
|
timer.async_wait(std::bind(&sock::handle_timeout, this, shared_from_this(), ph::_1));
|
||||||
|
}
|
||||||
|
|
||||||
|
inline void
|
||||||
|
sock::handle_timeout(const std::weak_ptr<sock> wp,
|
||||||
|
const error_code &ec)
|
||||||
|
{
|
||||||
|
using namespace boost::system::errc;
|
||||||
|
|
||||||
|
if(!wp.expired()) switch(ec.value())
|
||||||
|
{
|
||||||
|
case success:
|
||||||
|
{
|
||||||
|
timedout = true;
|
||||||
|
error_code sd_ec;
|
||||||
|
sd.cancel(sd_ec);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case operation_canceled:
|
||||||
|
timedout = false;
|
||||||
|
return;
|
||||||
|
|
||||||
|
default:
|
||||||
|
log::error("sock::handle_timeout(): unexpected: %s\n", ec.message().c_str());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
inline uint16_t
|
inline uint16_t
|
||||||
local_port(const sock &sock)
|
local_port(const sock &sock)
|
||||||
{
|
{
|
||||||
|
|
|
@ -114,7 +114,7 @@ namespace ircd
|
||||||
{
|
{
|
||||||
struct rbuf rbuf;
|
struct rbuf rbuf;
|
||||||
clist::const_iterator clit;
|
clist::const_iterator clit;
|
||||||
std::unique_ptr<struct sock> sock;
|
std::shared_ptr<struct sock> sock;
|
||||||
|
|
||||||
client();
|
client();
|
||||||
client(const client &) = delete;
|
client(const client &) = delete;
|
||||||
|
@ -149,7 +149,7 @@ ircd::clients()
|
||||||
}
|
}
|
||||||
|
|
||||||
std::shared_ptr<client>
|
std::shared_ptr<client>
|
||||||
ircd::add_client(std::unique_ptr<struct sock> sock)
|
ircd::add_client(std::shared_ptr<struct sock> sock)
|
||||||
{
|
{
|
||||||
auto client(add_client());
|
auto client(add_client());
|
||||||
client->sock = std::move(sock);
|
client->sock = std::move(sock);
|
||||||
|
@ -157,7 +157,7 @@ ircd::add_client(std::unique_ptr<struct sock> sock)
|
||||||
string(remote_address(*client)).c_str(),
|
string(remote_address(*client)).c_str(),
|
||||||
string(local_address(*client)).c_str());
|
string(local_address(*client)).c_str());
|
||||||
|
|
||||||
set_recv(*client);
|
recv_next(*client);
|
||||||
return client;
|
return client;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -221,7 +221,14 @@ catch(...)
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
ircd::set_recv(client &client)
|
ircd::recv_next(client &client)
|
||||||
|
{
|
||||||
|
recv_next(client, std::chrono::milliseconds(-1));
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
ircd::recv_next(client &client,
|
||||||
|
const std::chrono::milliseconds &timeout)
|
||||||
{
|
{
|
||||||
using boost::asio::async_read;
|
using boost::asio::async_read;
|
||||||
|
|
||||||
|
@ -229,36 +236,45 @@ ircd::set_recv(client &client)
|
||||||
rbuf.reset();
|
rbuf.reset();
|
||||||
|
|
||||||
auto &sock(*client.sock);
|
auto &sock(*client.sock);
|
||||||
|
sock.set_timeout(timeout);
|
||||||
async_read(sock.sd, mutable_buffers_1(rbuf.buf.data(), rbuf.buf.size()),
|
async_read(sock.sd, mutable_buffers_1(rbuf.buf.data(), rbuf.buf.size()),
|
||||||
std::bind(&rbuf::handle_pck, &rbuf, ph::_1, ph::_2),
|
std::bind(&rbuf::handle_pck, &rbuf, ph::_1, ph::_2),
|
||||||
std::bind(&ircd::handle_recv, std::ref(client), ph::_1, ph::_2));
|
std::bind(&ircd::handle_recv, std::ref(client), ph::_1, ph::_2));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
ircd::recv_cancel(client &client)
|
||||||
|
{
|
||||||
|
auto &sock(socket(client));
|
||||||
|
sock.sd.cancel();
|
||||||
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
ircd::handle_recv(client &client,
|
ircd::handle_recv(client &client,
|
||||||
const error_code &ec,
|
const error_code &ec,
|
||||||
const size_t bytes)
|
const size_t bytes)
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if(!handle_error(client, ec))
|
if(!handle_ec(client, ec))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
auto &rbuf(client.rbuf);
|
auto &rbuf(client.rbuf);
|
||||||
auto &reel(rbuf.reel);
|
auto &reel(rbuf.reel);
|
||||||
execute(client, reel);
|
execute(client, reel);
|
||||||
}
|
recv_next(client);
|
||||||
catch(const rfc1459::syntax_error &e)
|
|
||||||
{
|
|
||||||
std::cerr << e.what() << std::endl;
|
|
||||||
}
|
}
|
||||||
catch(const std::exception &e)
|
catch(const std::exception &e)
|
||||||
{
|
{
|
||||||
std::cerr << "errored: " << e.what() << std::endl;
|
log::error("client[%s]: error: %s",
|
||||||
|
string(remote_address(client)).c_str(),
|
||||||
|
e.what());
|
||||||
|
|
||||||
|
finished(client);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool
|
bool
|
||||||
ircd::handle_error(client &client,
|
ircd::handle_ec(client &client,
|
||||||
const error_code &ec)
|
const error_code &ec)
|
||||||
{
|
{
|
||||||
using namespace boost::system::errc;
|
using namespace boost::system::errc;
|
||||||
using boost::asio::error::eof;
|
using boost::asio::error::eof;
|
||||||
|
@ -268,8 +284,21 @@ ircd::handle_error(client &client,
|
||||||
|
|
||||||
switch(ec.value())
|
switch(ec.value())
|
||||||
{
|
{
|
||||||
case success: return true;
|
case success: return handle_ec_success(client);
|
||||||
default: throw boost::system::system_error(ec);
|
case operation_canceled: return handle_ec_cancel(client);
|
||||||
|
case eof: return handle_ec_eof(client);
|
||||||
|
default: throw boost::system::system_error(ec);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
ircd::handle_ec_success(client &client)
|
||||||
|
{
|
||||||
|
auto &sock(*client.sock);
|
||||||
|
{
|
||||||
|
error_code ec;
|
||||||
|
sock.timer.cancel(ec);
|
||||||
|
assert(ec == boost::system::errc::success);
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
|
@ -120,7 +120,7 @@ bool
|
||||||
listener::accept()
|
listener::accept()
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
auto sock(std::make_unique<sock>());
|
auto sock(std::make_shared<sock>());
|
||||||
acceptor.async_accept(sock->sd, yield(continuation()));
|
acceptor.async_accept(sock->sd, yield(continuation()));
|
||||||
add_client(std::move(sock));
|
add_client(std::move(sock));
|
||||||
return true;
|
return true;
|
||||||
|
|
Loading…
Reference in a new issue