From df2fe0bc4bf6240e47abf26ff88ffdfc9373f481 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?louiz=E2=80=99?= Date: Sun, 16 Aug 2020 15:51:47 +0200 Subject: Fix the parsing of IRC messages, especially with trailing spaces --- src/irc/irc_message.cpp | 39 ++++++++++++++++++--------------------- src/irc/irc_message.hpp | 4 +++- 2 files changed, 21 insertions(+), 22 deletions(-) (limited to 'src') diff --git a/src/irc/irc_message.cpp b/src/irc/irc_message.cpp index 14fdb0e..62fe9a7 100644 --- a/src/irc/irc_message.cpp +++ b/src/irc/irc_message.cpp @@ -1,33 +1,30 @@ #include #include -IrcMessage::IrcMessage(std::string&& line) +IrcMessage::IrcMessage(std::stringstream ss) { - std::string::size_type pos; - - // optional prefix - if (line[0] == ':') + if (ss.peek() == ':') { - pos = line.find(' '); - this->prefix = line.substr(1, pos - 1); - line = line.substr(pos + 1, std::string::npos); + ss.ignore(); + ss >> this->prefix; } - // command - pos = line.find(' '); - this->command = line.substr(0, pos); - line = line.substr(pos + 1, std::string::npos); - // arguments - do + ss >> this->command; + while (ss >> std::ws) { - if (line[0] == ':') + std::string arg; + if (ss.peek() == ':') + { + ss.ignore(); + std::getline(ss, arg); + } + else { - this->arguments.emplace_back(line.substr(1, std::string::npos)); - break ; + ss >> arg; + if (arg.empty()) + break; } - pos = line.find(' '); - this->arguments.emplace_back(line.substr(0, pos)); - line = line.substr(pos + 1, std::string::npos); - } while (pos != std::string::npos); + this->arguments.push_back(std::move(arg)); + } } IrcMessage::IrcMessage(std::string&& prefix, diff --git a/src/irc/irc_message.hpp b/src/irc/irc_message.hpp index 269a12a..5475fd2 100644 --- a/src/irc/irc_message.hpp +++ b/src/irc/irc_message.hpp @@ -4,11 +4,13 @@ #include #include #include +#include class IrcMessage { public: - IrcMessage(std::string&& line); + IrcMessage(std::stringstream ss); + IrcMessage(std::string str): IrcMessage{std::stringstream{str}} {} IrcMessage(std::string&& prefix, std::string&& command, std::vector&& args); IrcMessage(std::string&& command, std::vector&& args); ~IrcMessage() = default; -- cgit v1.2.3 From b98434b5d04d1ada9b24475e17ee8947d96ad1e3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?louiz=E2=80=99?= Date: Sun, 16 Aug 2020 16:05:15 +0200 Subject: In CAP messages, handle the last arg as a list of capabilities Instead of just one. This fixes the issue of the "trailing whitespace" since we now split it on ' ' Fix #3442 --- src/irc/irc_client.cpp | 25 ++++++++++++++----------- 1 file changed, 14 insertions(+), 11 deletions(-) (limited to 'src') diff --git a/src/irc/irc_client.cpp b/src/irc/irc_client.cpp index 3ae5ac6..5f0d9b9 100644 --- a/src/irc/irc_client.cpp +++ b/src/irc/irc_client.cpp @@ -1340,19 +1340,22 @@ long int IrcClient::get_throttle_limit() const void IrcClient::on_cap(const IrcMessage &message) { const auto& sub_command = message.arguments[1]; - const auto& cap = message.arguments[2]; - auto it = this->capabilities.find(cap); - if (it == this->capabilities.end()) + const auto& caps = utils::split(message.arguments[2], ' ', false); + for (const auto& cap: caps) { - log_warning("Received a CAP message for something we didn’t ask, or that we already handled."); - return; + auto it = this->capabilities.find(cap); + if (it == this->capabilities.end()) + { + log_warning("Received a CAP message for something we didn’t ask, or that we already handled: [", cap, "]"); + return; + } + Capability& capability = it->second; + if (sub_command == "ACK") + capability.on_ack(); + else if (sub_command == "NACK") + capability.on_nack(); + this->capabilities.erase(it); } - Capability& capability = it->second; - if (sub_command == "ACK") - capability.on_ack(); - else if (sub_command == "NACK") - capability.on_nack(); - this->capabilities.erase(it); if (this->capabilities.empty()) this->cap_end(); } -- cgit v1.2.3