#include "bot.hpp" #include "challenge.hpp" #include "client.hpp" #include "connection.hpp" #include "openssl_utils.hpp" #include "registration.hpp" #include "settings.hpp" #include "irc_coroutine.hpp" #include #include #include #include #include #include #include using namespace std::literals; static auto start(boost::asio::io_context &io, const Settings &settings) -> void { Ref cert; if (settings.use_tls && not settings.tls_certfile.empty()) { cert = cert_from_file(settings.tls_certfile); } Ref key; if (settings.use_tls && not settings.tls_keyfile.empty()) { key = key_from_file(settings.tls_keyfile, ""); } const auto connection = std::make_shared(io); const auto client = Client::start(*connection); Registration::start({ .nickname = settings.nickname, .realname = settings.realname, .username = settings.username, .password = settings.password, .sasl_mechanism = settings.sasl_mechanism, .sasl_authcid = settings.sasl_authcid, .sasl_authzid = settings.sasl_authzid, .sasl_password = settings.sasl_password, }, client); const auto bot = Bot::start(client); if (not settings.challenge_username.empty() && not settings.challenge_key_file.empty()) { if (auto key = key_from_file(settings.challenge_key_file, settings.challenge_key_password)) { client->sig_registered.connect([&settings, connection, key = std::move(key)]() { Challenge::start(*connection, settings.challenge_username, key); }); } } connection->sig_disconnect.connect( [&io, &settings, client, bot]() { client->shutdown(); bot->shutdown(); auto timer = std::make_shared(io); timer->expires_after(5s); timer->async_wait([&io, &settings, timer](auto) { start(io, settings); }); } ); bot->sig_command.connect([connection](auto &cmd) { std::cout << "COMMAND " << cmd.command << " from " << cmd.account << std::endl; if (cmd.oper == "glguy" && cmd.command == "ping") { connection->send_notice("glguy", cmd.arguments); } }); connection->start({ .tls = settings.use_tls, .host = settings.host, .port = settings.service, .verify = settings.tls_hostname, .client_cert = std::move(cert), .client_key = std::move(key), }); } static auto get_settings(const char *filename) -> Settings { if (auto config_stream = std::ifstream{filename}) { return Settings::from_stream(config_stream); } else { BOOST_LOG_TRIVIAL(error) << "Unable to open configuration"; std::exit(1); } } auto main(int argc, char *argv[]) -> int { //boost::log::core::get()->set_filter(boost::log::trivial::severity >= boost::log::trivial::warning); if (argc != 2) { BOOST_LOG_TRIVIAL(error) << "Bad arguments"; return 1; } const auto settings = get_settings(argv[1]); auto io = boost::asio::io_context{}; start(io, settings); io.run(); }