#include "control_interface.hpp" #include #include #include #include #include #include #include namespace wanda { // 'socket_deleter' implementation socket_deleter::~socket_deleter() { if (std::filesystem::exists(path)) { std::filesystem::remove(path); } } // 'control_interface' implementation control_interface::control_interface(control_interface::key key, boost::asio::io_service &service, control_interface::protocol::endpoint endpoint, std::shared_ptr logger) : keyed{key}, m_service{service}, m_endpoint{std::move(endpoint)}, m_socket{m_service}, m_acceptor{m_service}, m_logger{logger} { } boost::system::error_code control_interface::start() { if (auto error = boost::system::error_code{}; m_acceptor.open(m_endpoint.protocol(), error)) { return error; } if (auto error = boost::system::error_code{}; m_acceptor.bind(m_endpoint, error)) { return error; } if (auto error = boost::system::error_code{}; m_acceptor.listen(128, error)) { return error; } else { perform_accept(); return error; } } boost::system::error_code control_interface::shutdown() { for (auto &connection : m_connections) { connection->close(); } auto error = boost::system::error_code{}; return m_acceptor.close(error); } void control_interface::perform_accept() { m_acceptor.async_accept(m_socket, [that = shared_from_this(), this](auto const &error) { if (error && error != boost::asio::error::operation_aborted) { m_logger->error("failed to accept connection because '{}'", error); } else { m_logger->info("new incoming controller connection"); auto [connection, inserted] = m_connections.insert(make_control_connection(std::move(m_socket))); if (inserted) { (*connection)->add(this); (*connection)->start(); } perform_accept(); } }); } void control_interface::on_close(control_connection::pointer connection) { if (static_cast(connection->current_state()) >= static_cast(control_connection::state::established)) { m_logger->info("controller connection closed"); } else { m_logger->info("controller connection aborted before it could be established"); } m_connections.erase(connection); } void control_interface::on_received(control_connection::pointer connection, message message) { if (m_connections.find(connection) == m_connections.cend()) { m_logger->error("received message from an unknown connection"); return; } if (message.source != message_source_controller) { m_logger->error("received a deamon message"); return; } if (auto state = connection->current_state(); message.command == message_command_hello && state == control_connection::state::fresh) { m_logger->info("controller connection established"); if (message.argument.has_value()) { m_logger->info("remote controller version '{}'", *message.argument); } connection->send({message_source_daemon, message_command_hello, message_argument_hello}); connection->update(control_connection::state::established); } else { m_logger->warn("ignoring unknown message '{}'", message); } } control_interface::pointer make_interface(boost::asio::io_service &service, std::filesystem::path file, std::shared_ptr logger) { if (std::filesystem::exists(file)) { logger->error("file '{}' exists", file.native()); return {}; } control_interface::protocol::endpoint endpoint{file.string()}; return std::make_shared(control_interface::key{}, service, std::move(endpoint), logger); } } // namespace wanda