0
0
Fork 0
mirror of https://github.com/matrix-construct/construct synced 2024-12-27 07:54:05 +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:
Jason Volk 2016-09-12 17:10:04 -07:00
parent a9f967d16a
commit fcd410d656
4 changed files with 96 additions and 19 deletions

View file

@ -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();

View file

@ -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)
{ {

View file

@ -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;

View file

@ -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;