/* * charybdis: an advanced ircd. * client.c: Controls clients. * * Copyright (C) 1990 Jarkko Oikarinen and University of Oulu, Co Center * Copyright (C) 1996-2002 Hybrid Development Team * Copyright (C) 2002-2005 ircd-ratbox development team * Copyright (C) 2007 William Pitcock * Copyright (C) 2016 Charybdis Development Team * Copyright (C) 2016 Jason Volk * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 * USA */ #include #include #include "rbuf.h" namespace ircd { struct client :std::enable_shared_from_this { struct rbuf rbuf; clist::const_iterator clit; std::shared_ptr sock; client(); client(const client &) = delete; client &operator=(const client &) = delete; ~client() noexcept; }; client *me; std::unique_ptr client_db; clist client_list; hook::sequence h_client_added; bool handle_ec_eof(client &); bool handle_ec_cancel(client &); bool handle_ec_success(client &); bool handle_ec(client &, const error_code &); void handle_recv(client &, const error_code &, const size_t); } // namespace ircd using namespace ircd; client_init::client_init() { client_db.reset(new db::handle("clients")); } client_init::~client_init() noexcept { disconnect_all(); client_db.reset(nullptr); } client::client() { } client::~client() noexcept { } const clist & ircd::clients() { return client_list; } std::shared_ptr ircd::add_client(std::shared_ptr sock) { auto client(add_client()); client->sock = std::move(sock); log::info("New client[%s] on local[%s]", string(remote_address(*client)).c_str(), string(local_address(*client)).c_str()); h_client_added(*client); async_recv_next(*client); return client; } std::shared_ptr ircd::add_client() { auto client(std::make_shared()); client->clit = client_list.emplace(end(client_list), client); return client; } void ircd::finished(client &client) { const auto p(shared_from(client)); client_list.erase(client.clit); log::debug("client[%p] finished. (refs: %zu)", (const void *)p.get(), p.use_count()); } size_t ircd::recv(client &client, char *const &buf, const size_t &max, milliseconds &timeout) try { using std::chrono::duration_cast; auto &sock(socket(client)); sock.set_timeout(timeout); const auto ret(sock.recv_some(mutable_buffers_1(buf, max))); if(timeout < milliseconds(0)) return ret; sock.timer.cancel(); timeout = duration_cast(sock.timer.expires_from_now()); return ret; } catch(const boost::system::system_error &e) { using namespace boost::system::errc; using boost::asio::error::eof; switch(e.code().value()) { case success: return 0; case eof: throw disconnected(); case operation_canceled: if(socket(client).timedout) throw ctx::timeout(); else throw ctx::interrupted(); default: throw error("%s", e.what()); } } void ircd::send(client &client, text_terminate_t, const std::string &text) { send(client, text_terminate, text.c_str(), text.size()); } void ircd::send(client &client, text_terminate_t, const char *const &text) { send(client, text_terminate, text, strlen(text)); } void ircd::send(client &client, text_terminate_t, const char *const &text, const size_t &len) { assert(len <= 510); const std::array buffers {{ { text, len }, { "\r\n", 2 } }}; auto &sock(socket(client)); sock.send(buffers); } void ircd::send(client &client, text_raw_t, const std::string &text) { send(client, text_raw, text.c_str(), text.size()); } void ircd::send(client &client, text_raw_t, const char *const &text) { send(client, text_raw, text, strlen(text)); } void ircd::send(client &client, text_raw_t, const char *const &text, const size_t &len) { assert(len <= 512); auto &sock(socket(client)); sock.send(const_buffers_1(text, len)); } void ircd::disconnect_all() { for(auto &client : client_list) disconnect(std::nothrow, *client, dc::RST); } bool ircd::disconnect(std::nothrow_t, client &client, const dc &type) noexcept try { if(!client.sock) return true; disconnect(client, type); return true; } catch(...) { return false; } void ircd::disconnect(client &client, const dc &type) { auto &sock(socket(client)); sock.disconnect(type); } bool ircd::connected(const client &client) noexcept try { return socket(client).sd.is_open(); } catch(...) { return false; } void ircd::async_recv_next(client &client) { async_recv_next(client, std::chrono::milliseconds(-1)); } void ircd::async_recv_next(client &client, const std::chrono::milliseconds &timeout) { using boost::asio::async_read; auto &rbuf(client.rbuf); rbuf.reset(); auto &sock(*client.sock); sock.set_timeout(timeout); async_read(sock.sd, mutable_buffers_1(data(rbuf.buf), size(rbuf.buf)), std::bind(&rbuf::handle_pck, &rbuf, ph::_1, ph::_2), std::bind(&ircd::handle_recv, std::ref(client), ph::_1, ph::_2)); } void ircd::async_recv_cancel(client &client) { auto &sock(socket(client)); sock.sd.cancel(); } void ircd::handle_recv(client &client, const error_code &ec, const size_t bytes) try { if(!handle_ec(client, ec)) return; auto &rbuf(client.rbuf); execute(client, rbuf.reel); } catch(const std::exception &e) { log::error("client[%s]: error: %s", string(remote_address(client)).c_str(), e.what()); finished(client); } bool ircd::handle_ec(client &client, const error_code &ec) { using namespace boost::system::errc; using boost::asio::error::eof; if(client.rbuf.eptr) std::rethrow_exception(client.rbuf.eptr); switch(ec.value()) { case success: return handle_ec_success(client); 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; } bool ircd::handle_ec_cancel(client &client) { auto &sock(*client.sock); // The cancel can come from a timeout or directly. // If directly, the timer may still needs to be canceled if(!sock.timedout) { error_code ec; sock.timer.cancel(ec); assert(ec == boost::system::errc::success); log::debug("client[%s]: recv canceled", string(remote_address(client)).c_str()); return false; } log::debug("client[%s]: recv timeout", string(remote_address(client)).c_str()); finished(client); return false; } bool ircd::handle_ec_eof(client &client) { log::debug("client[%s]: eof", string(remote_address(client)).c_str()); finished(client); return false; } void ircd::execute(client &client, const uint8_t *const &ptr, const size_t &len) { execute(client, line(ptr, len)); } void ircd::execute(client &client, const std::string &l) { execute(client, line(l)); } void ircd::execute(client &client, tape &reel) { context([wp(weak_from(client)), &client, &reel] { // Hold the client for the lifetime of this context const life_guard lg(wp); while(!reel.empty()) try { auto &line(reel.front()); const scope pop([&reel] { reel.pop_front(); }); if(line.empty()) continue; auto &handle(cmds::find(command(line))); handle(client, std::move(line)); } catch(const std::exception &e) { log::error("vm: %s", e.what()); disconnect(client); finished(client); return; } async_recv_next(client); }, ctx::DEFER_POST | ctx::SELF_DESTRUCT); } void ircd::execute(client &c, line line) { if(line.empty()) return; auto &handle(cmds::find(command(line))); handle(c, std::move(line)); } std::string ircd::string(const ip_port &pair) { std::string ret(64, '\0'); ret.resize(snprintf(&ret.front(), ret.size(), "%s:%u", pair.first.c_str(), pair.second)); return ret; } ircd::ip_port ircd::local_address(const client &client) try { const auto &sock(socket(client)); return { local_ip(sock), local_port(sock) }; } catch(const std::bad_weak_ptr &) { return { "0.0.0.0"s, 0 }; } ircd::ip_port ircd::remote_address(const client &client) try { const auto &sock(socket(client)); return { remote_ip(sock), remote_port(sock) }; } catch(const std::bad_weak_ptr &) { return { "0.0.0.0"s, 0 }; } sock & ircd::socket(client &client) { if(unlikely(!has_socket(client))) throw std::bad_weak_ptr(); return *client.sock; } const sock & ircd::socket(const client &client) { if(unlikely(!has_socket(client))) throw std::bad_weak_ptr(); return *client.sock; } bool ircd::has_socket(const client &client) { return bool(client.sock); } std::weak_ptr ircd::weak_from(client &client) { return shared_from(client); } std::weak_ptr ircd::weak_from(const client &client) { return shared_from(client); } std::shared_ptr ircd::shared_from(client &client) { return client.shared_from_this(); } std::shared_ptr ircd::shared_from(const client &client) { return client.shared_from_this(); }