diff --git a/component/router.cpp b/component/router.cpp index 8069ef9..4154fcb 100644 --- a/component/router.cpp +++ b/component/router.cpp @@ -3,6 +3,8 @@ #include "router.h" +#include + #include "module/module.h" #include "connection/connection.h" @@ -42,12 +44,23 @@ void Router::routeMessage(const std::string& sender, const std::string& body) { if (aItr == actors.end()) aItr = actors.emplace(sender, std::make_shared(sender, defaultGroup)).first; - Shared::Strings args = Shared::split(body); - std::string moduleAlias = Module::Module::lower(args[0]); + std::string_view view(body); + std::string_view aliasView = Module::Module::pop(view); + + if (aliasView.size() > 1000) { + warn("Received a message with the first word larger than 1000 symbols, " + sender + "is clearly looking for a vulnerability"); + return onMessageResult(Shared::unhandled, sender); + } - Modules::iterator mItr = modules.find(moduleAlias); + if (view.empty()) { + debug("Incoming message from " + sender + " consists of only one word, modules have nothing to process"); + return onMessageResult(Shared::unhandled, sender); + } + + std::string alias(aliasView); + Modules::iterator mItr = modules.find(alias); if (mItr == modules.end()) { - debug("could not find module \"" + moduleAlias + "\" to handle message from " + sender); + debug("could not find module \"" + alias + "\" to handle message from " + sender); return onMessageResult(Shared::unhandled, sender); } @@ -57,10 +70,9 @@ void Router::routeMessage(const std::string& sender, const std::string& body) { return onMessageResult(Shared::error, sender); } - args.erase(args.begin()); Shared::Result result; try { - result = module->message(aItr->second, args); + result = module->message(aItr->second, view); if (result == Shared::success) debug("module \"" + mItr->first + "\" successfully handled message from " + sender); } catch (...) { diff --git a/connection/connection.cpp b/connection/connection.cpp index d04469b..7390563 100644 --- a/connection/connection.cpp +++ b/connection/connection.cpp @@ -76,7 +76,7 @@ void Connection::publish(const std::string& service, const std::string& node, co entry->addChild(new gloox::Tag("id", "urn:uuid:" + Shared::getUUID())); entry->addChild(new gloox::Tag("title", title)); - entry->addChild(new gloox::Tag("summary", body)); + entry->addChild(new gloox::Tag("content", body)); entry->addChild(new gloox::Tag("updated", Shared::getISOTimestamp())); entry->addChild(new gloox::Tag("published", Shared::getISOTimestamp())); diff --git a/module/actor.cpp b/module/actor.cpp index 85f3ebd..2ee64ba 100644 --- a/module/actor.cpp +++ b/module/actor.cpp @@ -9,26 +9,27 @@ Module::Actor::Actor(const std::shared_ptr& core, const Config::Module& co Module::Actor::~Actor() noexcept {} -Shared::Result Module::Actor::message(const std::shared_ptr<::Actor>& actor, const Shared::Strings& args) { - std::string result; +Shared::Result Module::Actor::message(const std::shared_ptr<::Actor>& actor, std::string_view view) { + std::string_view command = pop(view); - if (args.front() == "list") { + if (command == "list") { if (!hasPermission("read", actor)) return Shared::forbidden; - result = list(); - } else if (args.front() == "set") { + core->send(actor->jid, list()); + return Shared::success; + } + + if (command == "set") { if (!hasPermission("write", actor)) return Shared::forbidden; - if (args.size() < 3) + std::string_view jid = pop(view); + std::string_view group = pop(view); + if (jid.empty() || group.empty()) return Shared::error; - result = set(args[1], args[2]); - } - - if (!result.empty()) { - core->send(actor->jid, result); + core->send(actor->jid, set(jid, group)); return Shared::success; } @@ -50,8 +51,11 @@ std::string Module::Actor::list() { return result; } -std::string Module::Actor::set(const std::string& jid, const std::string& group) { - core->setGroup(lower(jid), group); +std::string Module::Actor::set(const std::string_view& jid, const std::string_view& group) { + std::string j(lower(std::string(jid))); + std::string g(group); + + core->setGroup(j, g); - return jid + " is now " + core->router.getGroup(jid); + return j + " is now " + core->router.getGroup(j); } diff --git a/module/actor.h b/module/actor.h index 5de3724..170fc88 100644 --- a/module/actor.h +++ b/module/actor.h @@ -14,11 +14,11 @@ public: Actor(const std::shared_ptr& core, const Config::Module& conf); ~Actor() noexcept; - virtual Shared::Result message(const std::shared_ptr<::Actor>& actor, const Shared::Strings& args) override; + virtual Shared::Result message(const std::shared_ptr<::Actor>& actor, std::string_view view) override; private: std::string list(); - std::string set(const std::string& jid, const std::string& group); + std::string set(const std::string_view& jid, const std::string_view& group); }; } diff --git a/module/module.cpp b/module/module.cpp index 0503485..784ae14 100644 --- a/module/module.cpp +++ b/module/module.cpp @@ -28,3 +28,22 @@ Module::Module::~Module() noexcept {} std::string Module::Module::lower(const std::string& text) { return std::ranges::to(text | std::views::transform(::tolower)); } + +std::string_view Module::Module::pop(std::string_view& input, char delimiter) { + std::size_t pos = input.find(delimiter); + std::string_view token; + + if (pos == std::string_view::npos) { + token = input; + input = {}; + } else { + token = input.substr(0, pos); + pos = input.find_first_not_of(' ', pos + 1); + if (pos == std::string_view::npos) + input = {}; + else + input.remove_prefix(pos); + } + + return token; +} diff --git a/module/module.h b/module/module.h index c05eb11..5640f92 100644 --- a/module/module.h +++ b/module/module.h @@ -8,6 +8,7 @@ #include #include #include +#include #include "shared/definitions.h" #include "shared/result.h" @@ -28,8 +29,9 @@ public: virtual ~Module() noexcept; static std::string lower(const std::string& text); + static std::string_view pop(std::string_view& input, char delimiter = ' '); - virtual Shared::Result message(const std::shared_ptr<::Actor>& actor, const Shared::Strings& args) = 0; + virtual Shared::Result message(const std::shared_ptr<::Actor>& actor, std::string_view view) = 0; public: const std::string name; diff --git a/module/publish.cpp b/module/publish.cpp index 75fcdd9..4fcf47d 100644 --- a/module/publish.cpp +++ b/module/publish.cpp @@ -11,29 +11,33 @@ Module::Publish::Publish(const std::shared_ptr& core, const Config::Module Module::Publish::~Publish() noexcept {} -Shared::Result Module::Publish::message(const std::shared_ptr<::Actor>& actor, const Shared::Strings& args) { - if (args.front() == "to") { +Shared::Result Module::Publish::message(const std::shared_ptr<::Actor>& actor, std::string_view view) { + std::string_view command = pop(view); + + if (command == "to") { if (!hasPermission("publish", actor)) return Shared::forbidden; - - if (args.size() < 3) + + std::string_view head = pop(view, '\n'); + std::string_view address = pop(head); + if (address.empty()) return Shared::error; - return to(args[1], args[2]); + return to(address, head, view); } return Shared::unhandled; } -Shared::Result Module::Publish::to(const std::string& address, const std::string& body) { - Shared::Strings parts = Shared::split(address, "@"); - if (parts.size() != 2) { +Shared::Result Module::Publish::to(std::string_view address, std::string_view title, std::string_view body) { + std::string_view node = pop(address, '@'); + if (node.empty() || address.empty()) { warn("Malformed address in \"to\" method"); return Shared::error; } try { - core->publish(parts[1], parts[0], "Completely testing stuff, early stages, ignore please", body); + core->publish(std::string(address), std::string(node), std::string(title), std::string(body)); return Shared::success; } catch (const std::exception& e) { error("Exception in \"to\" method: " + std::string(e.what())); diff --git a/module/publish.h b/module/publish.h index ca75d27..617a56f 100644 --- a/module/publish.h +++ b/module/publish.h @@ -16,10 +16,10 @@ public: Publish(const std::shared_ptr& core, const Config::Module& conf); ~Publish() noexcept; - virtual Shared::Result message(const std::shared_ptr<::Actor>& actor, const Shared::Strings& args) override; + virtual Shared::Result message(const std::shared_ptr<::Actor>& actor, std::string_view view) override; private: - Shared::Result to(const std::string& address, const std::string& body); + Shared::Result to(std::string_view address, std::string_view title, std::string_view body); }; } \ No newline at end of file