From a5c00b1548d7c8fc89358c26901fa534f7836cc1 Mon Sep 17 00:00:00 2001 From: Attila Molnar Date: Fri, 26 Feb 2016 15:40:22 +0100 Subject: Return ModResult from the OnSendWhoLine hook --- include/modules.h | 5 +++-- src/coremods/core_who.cpp | 6 +++--- src/modules.cpp | 2 +- src/modules/m_auditorium.cpp | 10 +++++----- src/modules/m_hideoper.cpp | 7 ++++--- src/modules/m_namesx.cpp | 11 ++++++----- 6 files changed, 22 insertions(+), 19 deletions(-) diff --git a/include/modules.h b/include/modules.h index d97f02046..a349a5d2a 100644 --- a/include/modules.h +++ b/include/modules.h @@ -990,9 +990,10 @@ class CoreExport Module : public classbase, public usecountbase * @param params The parameters to the /WHO query * @param user The user that this line of the query is about * @param memb The member shown in this line, NULL if no channel is in this line - * @param line The raw line to send; modifiable, if empty no line will be returned. + * @param line The raw line to send; modifiable. + * @param Return MOD_RES_PASSTHRU to allow the line to be displayed, MOD_RES_DENY to hide it */ - virtual void OnSendWhoLine(User* source, const std::vector& params, User* user, Membership* memb, std::string& line); + virtual ModResult OnSendWhoLine(User* source, const std::vector& params, User* user, Membership* memb, std::string& line); /** Called whenever a local user's IP is set for the first time, or when a local user's IP changes due to * a module like m_cgiirc changing it. diff --git a/src/coremods/core_who.cpp b/src/coremods/core_who.cpp index 1bc03c78e..6cccb3634 100644 --- a/src/coremods/core_who.cpp +++ b/src/coremods/core_who.cpp @@ -225,9 +225,9 @@ void CommandWho::SendWhoLine(User* user, const std::vector& parms, wholine.append(" :0 " + u->fullname); - FOREACH_MOD(OnSendWhoLine, (user, parms, u, memb, wholine)); - - if (!wholine.empty()) + ModResult res; + FIRST_MOD_RESULT(OnSendWhoLine, res, (user, parms, u, memb, wholine)); + if (res != MOD_RES_DENY) whoresults.push_back(wholine); } diff --git a/src/modules.cpp b/src/modules.cpp index b107a4b43..b9151b590 100644 --- a/src/modules.cpp +++ b/src/modules.cpp @@ -137,7 +137,7 @@ void Module::OnText(User*, void*, int, const std::string&, char, CUList&) { De ModResult Module::OnNamesListItem(User*, Membership*, std::string&, std::string&) { DetachEvent(I_OnNamesListItem); return MOD_RES_PASSTHRU; } ModResult Module::OnNumeric(User*, const Numeric::Numeric&) { DetachEvent(I_OnNumeric); return MOD_RES_PASSTHRU; } ModResult Module::OnAcceptConnection(int, ListenSocket*, irc::sockets::sockaddrs*, irc::sockets::sockaddrs*) { DetachEvent(I_OnAcceptConnection); return MOD_RES_PASSTHRU; } -void Module::OnSendWhoLine(User*, const std::vector&, User*, Membership*, std::string&) { DetachEvent(I_OnSendWhoLine); } +ModResult Module::OnSendWhoLine(User*, const std::vector&, User*, Membership*, std::string&) { DetachEvent(I_OnSendWhoLine); return MOD_RES_PASSTHRU; } void Module::OnSetUserIP(LocalUser*) { DetachEvent(I_OnSetUserIP); } #ifdef INSPIRCD_ENABLE_TESTSUITE diff --git a/src/modules/m_auditorium.cpp b/src/modules/m_auditorium.cpp index 7ad7ba1a3..6185d6d95 100644 --- a/src/modules/m_auditorium.cpp +++ b/src/modules/m_auditorium.cpp @@ -149,15 +149,15 @@ class ModuleAuditorium : public Module } } - void OnSendWhoLine(User* source, const std::vector& params, User* user, Membership* memb, std::string& line) CXX11_OVERRIDE + ModResult OnSendWhoLine(User* source, const std::vector& params, User* user, Membership* memb, std::string& line) CXX11_OVERRIDE { if (!memb) - return; + return MOD_RES_PASSTHRU; if (IsVisible(memb)) - return; + return MOD_RES_PASSTHRU; if (CanSee(source, memb)) - return; - line.clear(); + return MOD_RES_PASSTHRU; + return MOD_RES_DENY; } }; diff --git a/src/modules/m_hideoper.cpp b/src/modules/m_hideoper.cpp index 0fa5206ea..51b00de92 100644 --- a/src/modules/m_hideoper.cpp +++ b/src/modules/m_hideoper.cpp @@ -104,22 +104,23 @@ class ModuleHideOper : public Module, public Whois::LineEventListener return MOD_RES_PASSTHRU; } - void OnSendWhoLine(User* source, const std::vector& params, User* user, Membership* memb, std::string& line) CXX11_OVERRIDE + ModResult OnSendWhoLine(User* source, const std::vector& params, User* user, Membership* memb, std::string& line) CXX11_OVERRIDE { if (user->IsModeSet(hm) && !source->HasPrivPermission("users/auspex")) { // hide the "*" that marks the user as an oper from the /WHO line std::string::size_type spcolon = line.find(" :"); if (spcolon == std::string::npos) - return; // Another module hid the user completely + return MOD_RES_PASSTHRU; std::string::size_type sp = line.rfind(' ', spcolon-1); std::string::size_type pos = line.find('*', sp); if (pos != std::string::npos) line.erase(pos, 1); // hide the line completely if doing a "/who * o" query if (params.size() > 1 && params[1].find('o') != std::string::npos) - line.clear(); + return MOD_RES_DENY; } + return MOD_RES_PASSTHRU; } ModResult OnStats(Stats::Context& stats) CXX11_OVERRIDE diff --git a/src/modules/m_namesx.cpp b/src/modules/m_namesx.cpp index c906322bf..14f89807f 100644 --- a/src/modules/m_namesx.cpp +++ b/src/modules/m_namesx.cpp @@ -67,21 +67,21 @@ class ModuleNamesX : public Module return MOD_RES_PASSTHRU; } - void OnSendWhoLine(User* source, const std::vector& params, User* user, Membership* memb, std::string& line) CXX11_OVERRIDE + ModResult OnSendWhoLine(User* source, const std::vector& params, User* user, Membership* memb, std::string& line) CXX11_OVERRIDE { if ((!memb) || (!cap.get(source))) - return; + return MOD_RES_PASSTHRU; // Channel names can contain ":", and ":" as a 'start-of-token' delimiter is // only ever valid after whitespace, so... find the actual delimiter first! // Thanks to FxChiP for pointing this out. std::string::size_type pos = line.find(" :"); if (pos == std::string::npos || pos == 0) - return; + return MOD_RES_PASSTHRU; pos--; // Don't do anything if the user has no prefixes if ((line[pos] == 'H') || (line[pos] == 'G') || (line[pos] == '*')) - return; + return MOD_RES_PASSTHRU; // 352 21DAAAAAB #chan ident localhost insp21.test 21DAAAAAB H@ :0 a // pos @@ -89,10 +89,11 @@ class ModuleNamesX : public Module // Don't do anything if the user has only one prefix std::string prefixes = memb->GetAllPrefixChars(); if (prefixes.length() <= 1) - return; + return MOD_RES_PASSTHRU; line.erase(pos, 1); line.insert(pos, prefixes); + return MOD_RES_PASSTHRU; } }; -- cgit v1.2.3 From 2a16373ca12e0b4bbe72a487f1620a0d027414be Mon Sep 17 00:00:00 2001 From: Attila Molnar Date: Fri, 26 Feb 2016 15:44:39 +0100 Subject: m_hideoper Reorder checks in OnSendWhoLine() Check if the entire line should be hidden first --- src/modules/m_hideoper.cpp | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/src/modules/m_hideoper.cpp b/src/modules/m_hideoper.cpp index 51b00de92..5389ed18d 100644 --- a/src/modules/m_hideoper.cpp +++ b/src/modules/m_hideoper.cpp @@ -108,6 +108,10 @@ class ModuleHideOper : public Module, public Whois::LineEventListener { if (user->IsModeSet(hm) && !source->HasPrivPermission("users/auspex")) { + // Hide the line completely if doing a "/who * o" query + if ((params.size() > 1) && (params[1].find('o') != std::string::npos)) + return MOD_RES_DENY; + // hide the "*" that marks the user as an oper from the /WHO line std::string::size_type spcolon = line.find(" :"); if (spcolon == std::string::npos) @@ -116,9 +120,6 @@ class ModuleHideOper : public Module, public Whois::LineEventListener std::string::size_type pos = line.find('*', sp); if (pos != std::string::npos) line.erase(pos, 1); - // hide the line completely if doing a "/who * o" query - if (params.size() > 1 && params[1].find('o') != std::string::npos) - return MOD_RES_DENY; } return MOD_RES_PASSTHRU; } -- cgit v1.2.3 From 725c954efbdaca99701412e640a1a762b36a5f48 Mon Sep 17 00:00:00 2001 From: Attila Molnar Date: Fri, 26 Feb 2016 15:59:20 +0100 Subject: Send WHO reply numerics with User::WriteNumeric(), pass Numeric::Numeric objects to the OnSendWhoLine hook --- include/modules.h | 4 ++-- src/coremods/core_who.cpp | 41 ++++++++++++++++++++++------------------- src/modules.cpp | 2 +- src/modules/m_auditorium.cpp | 2 +- src/modules/m_hideoper.cpp | 13 +++++++------ src/modules/m_namesx.cpp | 23 ++++++----------------- 6 files changed, 39 insertions(+), 46 deletions(-) diff --git a/include/modules.h b/include/modules.h index a349a5d2a..46aa13e40 100644 --- a/include/modules.h +++ b/include/modules.h @@ -990,10 +990,10 @@ class CoreExport Module : public classbase, public usecountbase * @param params The parameters to the /WHO query * @param user The user that this line of the query is about * @param memb The member shown in this line, NULL if no channel is in this line - * @param line The raw line to send; modifiable. + * @param numeric Numeric to send; modifiable. * @param Return MOD_RES_PASSTHRU to allow the line to be displayed, MOD_RES_DENY to hide it */ - virtual ModResult OnSendWhoLine(User* source, const std::vector& params, User* user, Membership* memb, std::string& line); + virtual ModResult OnSendWhoLine(User* source, const std::vector& params, User* user, Membership* memb, Numeric::Numeric& numeric); /** Called whenever a local user's IP is set for the first time, or when a local user's IP changes due to * a module like m_cgiirc changing it. diff --git a/src/coremods/core_who.cpp b/src/coremods/core_who.cpp index 6cccb3634..d79006e84 100644 --- a/src/coremods/core_who.cpp +++ b/src/coremods/core_who.cpp @@ -63,7 +63,7 @@ class CommandWho : public Command syntax = "|||||0 [ohurmMiaplf]"; } - void SendWhoLine(User* user, const std::vector& parms, const std::string& initial, Membership* memb, User* u, std::vector& whoresults); + void SendWhoLine(User* user, const std::vector& parms, Membership* memb, User* u, std::vector& whoresults); /** Handle command. * @param parameters The parameters to the command * @param user The user issuing the command @@ -186,44 +186,48 @@ bool CommandWho::CanView(Channel* chan, User* user) return false; } -void CommandWho::SendWhoLine(User* user, const std::vector& parms, const std::string& initial, Membership* memb, User* u, std::vector& whoresults) +void CommandWho::SendWhoLine(User* user, const std::vector& parms, Membership* memb, User* u, std::vector& whoresults) { if (!memb) memb = get_first_visible_channel(u); - std::string wholine = initial + (memb ? memb->chan->name : "*") + " " + u->ident + " " + - (opt_showrealhost ? u->host : u->dhost) + " "; + Numeric::Numeric wholine(352); + wholine.push(memb ? memb->chan->name : "*").push(u->ident); + wholine.push(opt_showrealhost ? u->host : u->dhost); if (!ServerInstance->Config->HideWhoisServer.empty() && !user->HasPrivPermission("servers/auspex")) - wholine.append(ServerInstance->Config->HideWhoisServer); + wholine.push(ServerInstance->Config->HideWhoisServer); else - wholine.append(u->server->GetName()); + wholine.push(u->server->GetName()); - wholine.append(" " + u->nick + " "); + wholine.push(u->nick); + std::string param; /* away? */ if (u->IsAway()) { - wholine.append("G"); + param.push_back('G'); } else { - wholine.append("H"); + param.push_back('H'); } /* oper? */ if (u->IsOper()) { - wholine.push_back('*'); + param.push_back('*'); } if (memb) { char prefix = memb->GetPrefixChar(); if (prefix) - wholine.push_back(prefix); + param.push_back(prefix); } - wholine.append(" :0 " + u->fullname); + wholine.push(param); + wholine.push("0 "); + wholine.GetParams().back().append(u->fullname); ModResult res; FIRST_MOD_RESULT(OnSendWhoLine, res, (user, parms, u, memb, wholine)); @@ -253,8 +257,7 @@ CmdResult CommandWho::Handle (const std::vector& parameters, User * opt_far = false; opt_time = false; - std::vector whoresults; - std::string initial = "352 " + user->nick + " "; + std::vector whoresults; /* Change '0' into '*' so the wildcard matcher can grok it */ std::string matchtext = ((parameters[0] == "0") ? "*" : parameters[0]); @@ -337,7 +340,7 @@ CmdResult CommandWho::Handle (const std::vector& parameters, User * continue; } - SendWhoLine(user, parameters, initial, i->second, i->first, whoresults); + SendWhoLine(user, parameters, i->second, i->first, whoresults); } } } @@ -360,7 +363,7 @@ CmdResult CommandWho::Handle (const std::vector& parameters, User * continue; } - SendWhoLine(user, parameters, initial, NULL, oper, whoresults); + SendWhoLine(user, parameters, NULL, oper, whoresults); } } } @@ -377,14 +380,14 @@ CmdResult CommandWho::Handle (const std::vector& parameters, User * continue; } - SendWhoLine(user, parameters, initial, NULL, i->second, whoresults); + SendWhoLine(user, parameters, NULL, i->second, whoresults); } } } } /* Send the results out */ - for (std::vector::const_iterator n = whoresults.begin(); n != whoresults.end(); n++) - user->WriteServ(*n); + for (std::vector::const_iterator n = whoresults.begin(); n != whoresults.end(); ++n) + user->WriteNumeric(*n); user->WriteNumeric(RPL_ENDOFWHO, (*parameters[0].c_str() ? parameters[0] : "*"), "End of /WHO list."); // Penalize the user a bit for large queries diff --git a/src/modules.cpp b/src/modules.cpp index b9151b590..18f2dc4d1 100644 --- a/src/modules.cpp +++ b/src/modules.cpp @@ -137,7 +137,7 @@ void Module::OnText(User*, void*, int, const std::string&, char, CUList&) { De ModResult Module::OnNamesListItem(User*, Membership*, std::string&, std::string&) { DetachEvent(I_OnNamesListItem); return MOD_RES_PASSTHRU; } ModResult Module::OnNumeric(User*, const Numeric::Numeric&) { DetachEvent(I_OnNumeric); return MOD_RES_PASSTHRU; } ModResult Module::OnAcceptConnection(int, ListenSocket*, irc::sockets::sockaddrs*, irc::sockets::sockaddrs*) { DetachEvent(I_OnAcceptConnection); return MOD_RES_PASSTHRU; } -ModResult Module::OnSendWhoLine(User*, const std::vector&, User*, Membership*, std::string&) { DetachEvent(I_OnSendWhoLine); return MOD_RES_PASSTHRU; } +ModResult Module::OnSendWhoLine(User*, const std::vector&, User*, Membership*, Numeric::Numeric&) { DetachEvent(I_OnSendWhoLine); return MOD_RES_PASSTHRU; } void Module::OnSetUserIP(LocalUser*) { DetachEvent(I_OnSetUserIP); } #ifdef INSPIRCD_ENABLE_TESTSUITE diff --git a/src/modules/m_auditorium.cpp b/src/modules/m_auditorium.cpp index 6185d6d95..6f9eeb252 100644 --- a/src/modules/m_auditorium.cpp +++ b/src/modules/m_auditorium.cpp @@ -149,7 +149,7 @@ class ModuleAuditorium : public Module } } - ModResult OnSendWhoLine(User* source, const std::vector& params, User* user, Membership* memb, std::string& line) CXX11_OVERRIDE + ModResult OnSendWhoLine(User* source, const std::vector& params, User* user, Membership* memb, Numeric::Numeric& numeric) CXX11_OVERRIDE { if (!memb) return MOD_RES_PASSTHRU; diff --git a/src/modules/m_hideoper.cpp b/src/modules/m_hideoper.cpp index 5389ed18d..3d8f8910c 100644 --- a/src/modules/m_hideoper.cpp +++ b/src/modules/m_hideoper.cpp @@ -104,7 +104,7 @@ class ModuleHideOper : public Module, public Whois::LineEventListener return MOD_RES_PASSTHRU; } - ModResult OnSendWhoLine(User* source, const std::vector& params, User* user, Membership* memb, std::string& line) CXX11_OVERRIDE + ModResult OnSendWhoLine(User* source, const std::vector& params, User* user, Membership* memb, Numeric::Numeric& numeric) CXX11_OVERRIDE { if (user->IsModeSet(hm) && !source->HasPrivPermission("users/auspex")) { @@ -113,13 +113,14 @@ class ModuleHideOper : public Module, public Whois::LineEventListener return MOD_RES_DENY; // hide the "*" that marks the user as an oper from the /WHO line - std::string::size_type spcolon = line.find(" :"); - if (spcolon == std::string::npos) + // #chan ident localhost insp22.test nick H@ :0 Attila + if (numeric.GetParams().size() < 6) return MOD_RES_PASSTHRU; - std::string::size_type sp = line.rfind(' ', spcolon-1); - std::string::size_type pos = line.find('*', sp); + + std::string& param = numeric.GetParams()[5]; + const std::string::size_type pos = param.find('*'); if (pos != std::string::npos) - line.erase(pos, 1); + param.erase(pos, 1); } return MOD_RES_PASSTHRU; } diff --git a/src/modules/m_namesx.cpp b/src/modules/m_namesx.cpp index 14f89807f..beac968ef 100644 --- a/src/modules/m_namesx.cpp +++ b/src/modules/m_namesx.cpp @@ -67,32 +67,21 @@ class ModuleNamesX : public Module return MOD_RES_PASSTHRU; } - ModResult OnSendWhoLine(User* source, const std::vector& params, User* user, Membership* memb, std::string& line) CXX11_OVERRIDE + ModResult OnSendWhoLine(User* source, const std::vector& params, User* user, Membership* memb, Numeric::Numeric& numeric) CXX11_OVERRIDE { if ((!memb) || (!cap.get(source))) return MOD_RES_PASSTHRU; - // Channel names can contain ":", and ":" as a 'start-of-token' delimiter is - // only ever valid after whitespace, so... find the actual delimiter first! - // Thanks to FxChiP for pointing this out. - std::string::size_type pos = line.find(" :"); - if (pos == std::string::npos || pos == 0) - return MOD_RES_PASSTHRU; - pos--; - // Don't do anything if the user has no prefixes - if ((line[pos] == 'H') || (line[pos] == 'G') || (line[pos] == '*')) - return MOD_RES_PASSTHRU; - - // 352 21DAAAAAB #chan ident localhost insp21.test 21DAAAAAB H@ :0 a - // pos - // Don't do anything if the user has only one prefix std::string prefixes = memb->GetAllPrefixChars(); if (prefixes.length() <= 1) return MOD_RES_PASSTHRU; - line.erase(pos, 1); - line.insert(pos, prefixes); + // #chan ident localhost insp22.test nick H@ :0 Attila + if (numeric.GetParams().size() < 6) + return MOD_RES_PASSTHRU; + + numeric.GetParams()[5].append(prefixes, 1, std::string::npos); return MOD_RES_PASSTHRU; } }; -- cgit v1.2.3 From 1fd6f04e6df789ffd4d57e1535a0918d901c2271 Mon Sep 17 00:00:00 2001 From: Attila Molnar Date: Fri, 26 Feb 2016 16:03:00 +0100 Subject: Add RPL_WHOREPLY to the list of numerics Use it instead of the raw number --- include/numerics.h | 1 + src/coremods/core_who.cpp | 2 +- 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/include/numerics.h b/include/numerics.h index fcc5f5942..2f25ac0cb 100644 --- a/include/numerics.h +++ b/include/numerics.h @@ -104,6 +104,7 @@ enum RPL_INVITELIST = 346, // insp-specific (stolen from ircu) RPL_ENDOFINVITELIST = 347, // insp-specific (stolen from ircu) RPL_VERSION = 351, + RPL_WHOREPLY = 352, RPL_NAMREPLY = 353, RPL_LINKS = 364, RPL_ENDOFLINKS = 365, diff --git a/src/coremods/core_who.cpp b/src/coremods/core_who.cpp index d79006e84..739c675a9 100644 --- a/src/coremods/core_who.cpp +++ b/src/coremods/core_who.cpp @@ -191,7 +191,7 @@ void CommandWho::SendWhoLine(User* user, const std::vector& parms, if (!memb) memb = get_first_visible_channel(u); - Numeric::Numeric wholine(352); + Numeric::Numeric wholine(RPL_WHOREPLY); wholine.push(memb ? memb->chan->name : "*").push(u->ident); wholine.push(opt_showrealhost ? u->host : u->dhost); if (!ServerInstance->Config->HideWhoisServer.empty() && !user->HasPrivPermission("servers/auspex")) -- cgit v1.2.3