diff --git a/CMakeLists.txt b/CMakeLists.txt index 2a3cf7b..5c1678a 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -19,7 +19,7 @@ set(EXEC_NAME "jay") add_executable(${EXEC_NAME} main.cpp jay.cpp) add_subdirectory(component) -add_subdirectory(handler) +add_subdirectory(connection) add_subdirectory(module) target_include_directories(${EXEC_NAME} PRIVATE ${GLOOX_INCLUDE_DIRS}) diff --git a/component/CMakeLists.txt b/component/CMakeLists.txt index b68f13c..7e77ed0 100644 --- a/component/CMakeLists.txt +++ b/component/CMakeLists.txt @@ -3,6 +3,7 @@ set(SOURCES logger.cpp actor.cpp router.cpp + core.cpp ) set(HEADERS @@ -10,6 +11,7 @@ set(HEADERS logger.h actor.h router.h + core.h ) target_sources(${EXEC_NAME} PRIVATE ${SOURCES}) diff --git a/component/actor.cpp b/component/actor.cpp index f1e97fa..dd98794 100644 --- a/component/actor.cpp +++ b/component/actor.cpp @@ -11,3 +11,7 @@ Actor::Actor(const std::string& jid, const std::string& group) : void Actor::setGroup(const std::string& newGroup) { group = newGroup; } + +std::string Actor::getGroup() const { + return group; +} diff --git a/component/actor.h b/component/actor.h index b8c6137..6594d3e 100644 --- a/component/actor.h +++ b/component/actor.h @@ -10,6 +10,7 @@ public: Actor(const std::string& jid, const std::string& group); void setGroup(const std::string& newGroup); + std::string getGroup() const; public: const std::string jid; diff --git a/component/config.cpp b/component/config.cpp index 9912d64..0dfb9af 100644 --- a/component/config.cpp +++ b/component/config.cpp @@ -46,7 +46,7 @@ gloox::TLSPolicy Config::getTLSPolicy() const { } -std::map Config::getOwners() const { +std::map Config::getActors() const { std::map result; YAML::Node owners = root["actors"]; if (!owners.IsMap()) diff --git a/component/config.h b/component/config.h index 4d2bdeb..f238a75 100644 --- a/component/config.h +++ b/component/config.h @@ -30,7 +30,7 @@ public: std::string getFullJID() const; std::string getPassword() const; std::string getResource() const; - std::map getOwners() const; + std::map getActors() const; gloox::LogLevel getLogLevel() const; gloox::TLSPolicy getTLSPolicy() const; Module getModuleConfig(const std::string& name) const; diff --git a/component/core.cpp b/component/core.cpp new file mode 100644 index 0000000..995a64c --- /dev/null +++ b/component/core.cpp @@ -0,0 +1,10 @@ +// SPDX-FileCopyrightText: 2024 Yury Gubich +// SPDX-License-Identifier: GPL-3.0-or-later + +#include "core.h" + +Core::Core(const std::string& configPath): + config(configPath), + router(), + logger() +{} diff --git a/component/core.h b/component/core.h new file mode 100644 index 0000000..f8c08f5 --- /dev/null +++ b/component/core.h @@ -0,0 +1,20 @@ +// SPDX-FileCopyrightText: 2024 Yury Gubich +// SPDX-License-Identifier: GPL-3.0-or-later + +#pragma once + +#include + +#include "config.h" +#include "router.h" +#include "logger.h" + +class Core { +public: + Core(const std::string& configPath); + + Config config; + Router router; + Logger logger; + +}; diff --git a/component/logger.cpp b/component/logger.cpp index 03ead4a..050d520 100644 --- a/component/logger.cpp +++ b/component/logger.cpp @@ -78,15 +78,10 @@ void writeTags(gloox::LogArea area) { std::cout << '\t'; } -Logger::Logger(gloox::LogSink& sink, gloox::LogLevel level): - sink(sink) -{ - sink.registerLogHandler(level, gloox::LogAreaAll, this); -} +Logger::Logger() +{} -Logger::~Logger() { - sink.removeLogHandler(this); -} +Logger::~Logger() {} void Logger::handleLog(gloox::LogLevel level, gloox::LogArea area, const std::string& message) { writeTimestamp(); diff --git a/component/logger.h b/component/logger.h index bf86085..ca470d9 100644 --- a/component/logger.h +++ b/component/logger.h @@ -6,15 +6,10 @@ #include #include -class Logger : public gloox::LogHandler { +class Logger: public gloox::LogHandler { public: - Logger(gloox::LogSink& sink, gloox::LogLevel level); + Logger(); ~Logger(); void handleLog(gloox::LogLevel level, gloox::LogArea area, const std::string& message) override; - - -private: - gloox::LogSink& sink; - gloox::LogLevel level; }; diff --git a/component/router.cpp b/component/router.cpp index 00576d1..8789e84 100644 --- a/component/router.cpp +++ b/component/router.cpp @@ -3,6 +3,8 @@ #include "router.h" +#include "module/module.h" + Router::Router(): actors(), defaultGroup("Stranger") @@ -43,4 +45,13 @@ void Router::routeMessage(const std::string& sender, const std::string& body) { module->message(aItr->second, args); } +std::map Router::getActors() const { + std::map result; + + for (const std::pair>& pair : actors) + result.emplace(pair.first, pair.second->getGroup()); + + return result; +} + diff --git a/component/router.h b/component/router.h index e8e3039..3655551 100644 --- a/component/router.h +++ b/component/router.h @@ -9,7 +9,10 @@ #include #include "actor.h" -#include "module/module.h" + +namespace Module { + class Module; +}; class Router { public: @@ -19,6 +22,8 @@ public: void registerActor(const std::string& key, const std::string& group); void routeMessage(const std::string& sender, const std::string& body); + std::map getActors() const; + private: typedef std::map> Modules; typedef std::map> Actors; diff --git a/handler/CMakeLists.txt b/connection/CMakeLists.txt similarity index 79% rename from handler/CMakeLists.txt rename to connection/CMakeLists.txt index 1c5525d..28a7488 100644 --- a/handler/CMakeLists.txt +++ b/connection/CMakeLists.txt @@ -1,10 +1,8 @@ set(SOURCES - message.cpp connection.cpp ) set(HEADERS - message.h connection.h ) diff --git a/connection/connection.cpp b/connection/connection.cpp new file mode 100644 index 0000000..b5ced9e --- /dev/null +++ b/connection/connection.cpp @@ -0,0 +1,71 @@ +// SPDX-FileCopyrightText: 2024 Yury Gubich +// SPDX-License-Identifier: GPL-3.0-or-later + +#include "connection.h" + +Connection::Connection(const std::shared_ptr& core): + state(initial), + core(core), + gloox() +{} + +Connection::~Connection() noexcept { + deinitialize(); +} + +void Connection::initiialize() { + if (state != initial) + return; + + gloox = std::make_unique(core->config.getFullJID(), core->config.getPassword()); + gloox->registerConnectionListener(this); + gloox->registerMessageHandler(this); + + gloox->logInstance().registerLogHandler(core->config.getLogLevel(), gloox::LogAreaAll, &core->logger); + + gloox::Disco* disco = gloox->disco(); + disco->setVersion("Jay", "0.0.1"); + disco->setIdentity("client", "bot"); + + gloox->setTls(core->config.getTLSPolicy()); + gloox->setSASLMechanisms(gloox::SaslMechAll); + gloox->setStreamManagement(true, true); + + state = disconnected; +} + +void Connection::deinitialize() { + if (state == initial) + return; + + gloox->logInstance().removeLogHandler(&core->logger); + + gloox->removeMessageHandler(this); + gloox->removeConnectionListener(this); + gloox = nullptr; + + state = initial; +} + +void Connection::connect() { + if (state != disconnected) + return; + + state = connected; + gloox->connect(true); + state = disconnected; +} + +void Connection::send(const std::string& jid, const std::string& body) { + gloox->send(gloox::Message(gloox::Message::Chat, jid, body)); +} + +void Connection::handleMessage(const gloox::Message& message, gloox::MessageSession* session) { + core->router.routeMessage(message.from().bare(), message.body()); +} + +void Connection::onConnect() {} +void Connection::onDisconnect(gloox::ConnectionError e) {} +bool Connection::onTLSConnect(const gloox::CertInfo&) { return true; } + + diff --git a/connection/connection.h b/connection/connection.h new file mode 100644 index 0000000..e933387 --- /dev/null +++ b/connection/connection.h @@ -0,0 +1,47 @@ +// SPDX-FileCopyrightText: 2024 Yury Gubich +// SPDX-License-Identifier: GPL-3.0-or-later + +#pragma once + +#include + +#include +#include +#include +#include +#include + +#include "component/core.h" + +class Connection: + public gloox::ConnectionListener, + public gloox::MessageHandler +{ +public: + enum State { + initial, + disconnected, + connected + }; + +public: + Connection(const std::shared_ptr& core); + ~Connection() noexcept; + + void initiialize(); + void deinitialize(); + void connect(); + void send(const std::string& jid, const std::string& body); + +public: + void onConnect() override; + void onDisconnect(gloox::ConnectionError e) override; + bool onTLSConnect(const gloox::CertInfo&) override; + void handleMessage(const gloox::Message& message, gloox::MessageSession* session = 0) override; + +private: + State state; + std::shared_ptr core; + std::unique_ptr gloox; +}; + diff --git a/handler/connection.cpp b/handler/connection.cpp deleted file mode 100644 index d4e43e7..0000000 --- a/handler/connection.cpp +++ /dev/null @@ -1,27 +0,0 @@ -// SPDX-FileCopyrightText: 2024 Yury Gubich -// SPDX-License-Identifier: GPL-3.0-or-later - -#include "connection.h" - -Connection::Connection(const std::shared_ptr& config, const std::shared_ptr& client): - config(config), - client(client) -{ - client->registerConnectionListener(this); -} - -Connection::~Connection() { - if (std::shared_ptr cl = client.lock()) - cl->removeConnectionListener(this); -} - - -void Connection::onConnect() {} - -void Connection::onDisconnect(gloox::ConnectionError e) {} - -bool Connection::onTLSConnect(const gloox::CertInfo& info) { - return true; -} - - diff --git a/handler/connection.h b/handler/connection.h deleted file mode 100644 index ef6ee38..0000000 --- a/handler/connection.h +++ /dev/null @@ -1,25 +0,0 @@ -// SPDX-FileCopyrightText: 2024 Yury Gubich -// SPDX-License-Identifier: GPL-3.0-or-later - -#pragma once - -#include - -#include "gloox/client.h" -#include "gloox/connectionlistener.h" - -#include "component/config.h" - -class Connection : public gloox::ConnectionListener { -public: - Connection(const std::shared_ptr& config, const std::shared_ptr& client); - ~Connection(); - - void onConnect() override; - void onDisconnect(gloox::ConnectionError e) override; - bool onTLSConnect(const gloox::CertInfo&) override; - -private: - std::weak_ptr config; - std::weak_ptr client; -}; diff --git a/handler/message.cpp b/handler/message.cpp deleted file mode 100644 index 86dab2e..0000000 --- a/handler/message.cpp +++ /dev/null @@ -1,23 +0,0 @@ -// SPDX-FileCopyrightText: 2024 Yury Gubich -// SPDX-License-Identifier: GPL-3.0-or-later - -#include "message.h" - -#include - -Message::Message(const std::shared_ptr& config, const std::shared_ptr& client, const Callback& callback): - callback(callback), - config(config), - client(client) -{ - client->registerMessageHandler(this); -} - -Message::~Message() { - if (std::shared_ptr cl = client.lock()) - cl->removeMessageHandler(this); -} - -void Message::handleMessage(const gloox::Message& message, gloox::MessageSession* session) { - callback(message.from().bare(), message.body()); -} diff --git a/handler/message.h b/handler/message.h deleted file mode 100644 index 60a64ed..0000000 --- a/handler/message.h +++ /dev/null @@ -1,30 +0,0 @@ -// SPDX-FileCopyrightText: 2024 Yury Gubich -// SPDX-License-Identifier: GPL-3.0-or-later - -#pragma once - -#include -#include -#include - -#include -#include -#include - -#include "component/config.h" - -class Message : public gloox::MessageHandler { -private: - typedef std::function Callback; - -public: - Message(const std::shared_ptr& config, const std::shared_ptr& client, const Callback& callback); - ~Message(); - - void handleMessage(const gloox::Message& message, gloox::MessageSession* session = 0) override; - -private: - Callback callback; - std::weak_ptr config; - std::weak_ptr client; -}; diff --git a/jay.cpp b/jay.cpp index b88928d..179c5c3 100644 --- a/jay.cpp +++ b/jay.cpp @@ -5,86 +5,55 @@ #include "module/actor.h" -static const std::map(const std::shared_ptr)>> moduleNames = { - {"actor", [](const std::shared_ptr client) { return std::make_shared(client); }} - +static const std::map< + std::string, + std::function< + std::shared_ptr( + const std::shared_ptr&, + const std::shared_ptr& + ) + > +> moduleNames = { + {"actor", [](const std::shared_ptr& core, const std::shared_ptr& connection) { return std::make_shared(core, connection); }} }; Jay::Jay(const std::string& configPath): - runMutex(), - config(std::make_shared(configPath)), - router(std::make_unique()), - client(), - messageHandler(), - connectionHandler(), - modules(), - loggers() + core(std::make_shared(configPath)), + connection(std::make_shared(core)), + modules() {} -Jay::~Jay() {} - -bool Jay::isConfigValid() const { - return config->isValid(); +Jay::~Jay() { + connection->deinitialize(); } +bool Jay::isConfigValid() const { + return core->config.isValid(); +} void Jay::run() { - std::lock_guard lock(runMutex); - initialize(); - client->connect(true); + connection->connect(); } void Jay::initialize() { - createClient(); - - if (!messageHandler) - messageHandler = std::make_unique( - config, - client, - std::bind(&Router::routeMessage, router.get(), std::placeholders::_1, std::placeholders::_2) - ); - - if (!connectionHandler) - connectionHandler = std::make_unique(config, client); - + connection->initiialize(); createModules(); createActors(); } -void Jay::createClient() { - if (client) - return; - - client = std::make_shared(config->getFullJID(), config->getPassword()); - addLogger(config->getLogLevel()); - - gloox::Disco* disco = client->disco(); - - disco->setVersion("Jay", "0.0.1"); - disco->setIdentity("client", "bot"); - - client->setTls(config->getTLSPolicy()); - client->setSASLMechanisms(gloox::SaslMechAll); - client->setStreamManagement(true, true); -} - void Jay::createActors() { - for (const std::pair& pair : config->getOwners()) - router->registerActor(pair.first, pair.second); + for (const std::pair& pair : core->config.getActors()) + core->router.registerActor(pair.first, pair.second); } void Jay::createModules() { for (const auto& pair : moduleNames) { - Config::Module conf = config->getModuleConfig(pair.first); + Config::Module conf = core->config.getModuleConfig(pair.first); if (!conf.enabled) continue; - modules.emplace_back(pair.second(client)); - router->registerModule(pair.first, modules.back()); + modules.emplace_back(pair.second(core, connection)); + core->router.registerModule(pair.first, modules.back()); } } - -void Jay::addLogger(gloox::LogLevel level) { - loggers.emplace_back(std::make_unique(client->logInstance(), level)); -} diff --git a/jay.h b/jay.h index 66a66c2..603f0d4 100644 --- a/jay.h +++ b/jay.h @@ -7,17 +7,14 @@ #include #include #include -#include +#include #include #include #include -#include "component/logger.h" -#include "component/config.h" -#include "component/router.h" -#include "handler/message.h" -#include "handler/connection.h" +#include "component/core.h" +#include "connection/connection.h" #include "module/module.h" class Jay { @@ -30,19 +27,12 @@ public: void run(); private: - void addLogger(gloox::LogLevel level); void initialize(); - void createClient(); void createActors(); void createModules(); private: - std::mutex runMutex; - std::shared_ptr config; - std::unique_ptr router; - std::shared_ptr client; - std::unique_ptr messageHandler; - std::unique_ptr connectionHandler; + std::shared_ptr core; + std::shared_ptr connection; std::vector> modules; - std::vector> loggers; }; diff --git a/module/actor.cpp b/module/actor.cpp index 7188ec0..4d66b1b 100644 --- a/module/actor.cpp +++ b/module/actor.cpp @@ -3,13 +3,33 @@ #include "actor.h" -Module::Actor::Actor(const std::shared_ptr& client): - Module(client) +Module::Actor::Actor(const std::shared_ptr& core, const std::shared_ptr& connection): + Module(core, connection) {} Module::Actor::~Actor() noexcept {} void Module::Actor::message(const std::shared_ptr<::Actor>& actor, const Module::Module::Tokens& args) { + std::string result; + if (args.front() == "list") - sendMessage(actor, "Me\nMyself\nI"); + result = list(); + + if (!result.empty()) + connection->send(actor->jid, result); +} + +std::string Module::Actor::list() { + std::string result; + for (const std::pair& pair : core->router.getActors()) { + if (!result.empty()) + result.append(1, '\n'); + + result += pair.first + ": " + pair.second; + } + + if (result.empty()) + result += "There are no actors currently"; + + return result; } diff --git a/module/actor.h b/module/actor.h index 66d31bf..d4e3b8d 100644 --- a/module/actor.h +++ b/module/actor.h @@ -9,10 +9,13 @@ namespace Module { class Actor : public Module { public: - Actor(const std::shared_ptr& client); + Actor(const std::shared_ptr& core, const std::shared_ptr& connection); ~Actor() noexcept; virtual void message(const std::shared_ptr<::Actor>& actor, const Tokens& args) override; + +private: + std::string list(); }; } diff --git a/module/module.cpp b/module/module.cpp index 557fb5d..07ed4ca 100644 --- a/module/module.cpp +++ b/module/module.cpp @@ -5,8 +5,9 @@ #include "gloox/message.h" -Module::Module::Module(const std::shared_ptr& client): - client(client) +Module::Module::Module(const std::shared_ptr& core, const std::shared_ptr& connection): + core(core), + connection(connection) {} Module::Module::~Module() noexcept {} @@ -25,11 +26,3 @@ std::vector Module::Module::split(const std::string& string, const return result; } - -void Module::Module::sendMessage(const std::shared_ptr& actor, const std::string& body) { - client->send(gloox::Message( - gloox::Message::Chat, - actor->jid, - body - )); -} diff --git a/module/module.h b/module/module.h index 1c97233..7cb038b 100644 --- a/module/module.h +++ b/module/module.h @@ -7,9 +7,9 @@ #include #include -#include "gloox/client.h" - +#include "component/core.h" #include "component/actor.h" +#include "connection/connection.h" namespace Module { @@ -18,8 +18,7 @@ public: typedef std::vector Tokens; protected: - Module(const std::shared_ptr& client); - void sendMessage(const std::shared_ptr& actor, const std::string& body); + Module(const std::shared_ptr& core, const std::shared_ptr& connection); public: virtual ~Module() noexcept; @@ -29,7 +28,8 @@ public: virtual void message(const std::shared_ptr& actor, const Tokens& args) = 0; protected: - std::shared_ptr client; + std::shared_ptr core; + std::shared_ptr connection; };