diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/modes/umode_s.cpp | 132 |
1 files changed, 56 insertions, 76 deletions
diff --git a/src/modes/umode_s.cpp b/src/modes/umode_s.cpp index 2650a31bc..d06ef64cb 100644 --- a/src/modes/umode_s.cpp +++ b/src/modes/umode_s.cpp @@ -32,15 +32,11 @@ ModeUserServerNoticeMask::ModeUserServerNoticeMask() : ModeHandler(NULL, "snomas ModeAction ModeUserServerNoticeMask::OnModeChange(User* source, User* dest, Channel*, std::string ¶meter, bool adding) { - /* Set the array fields */ if (adding) { - /* Fix for bug #310 reported by Smartys */ - if (!dest->IsModeSet(this)) - dest->snomasks.reset(); - dest->SetMode(this, true); - parameter = ProcessNoticeMasks(dest, parameter.c_str()); + // Process the parameter (remove chars we don't understand, remove redundant chars, etc.) + parameter = ProcessNoticeMasks(dest, parameter); return MODEACTION_ALLOW; } else @@ -48,20 +44,28 @@ ModeAction ModeUserServerNoticeMask::OnModeChange(User* source, User* dest, Chan if (dest->IsModeSet(this)) { dest->SetMode(this, false); + dest->snomasks.reset(); return MODEACTION_ALLOW; } } - /* Allow the change */ + // Mode not set and trying to unset, deny return MODEACTION_DENY; } std::string ModeUserServerNoticeMask::GetUserParameter(User* user) { - std::string masks = FormatNoticeMasks(user); - if (masks.length()) - masks = "+" + masks; - return masks; + std::string ret; + if (!user->IsModeSet(this)) + return ret; + + ret.push_back('+'); + for (unsigned char n = 0; n < 64; n++) + { + if (user->snomasks[n]) + ret.push_back(n + 'A'); + } + return ret; } void ModeUserServerNoticeMask::OnParameterMissing(User* user, User* dest, Channel* channel) @@ -69,15 +73,14 @@ void ModeUserServerNoticeMask::OnParameterMissing(User* user, User* dest, Channe user->WriteNotice("*** The user mode +s requires a parameter (server notice mask). Please provide a parameter, e.g. '+s +*'."); } -std::string ModeUserServerNoticeMask::ProcessNoticeMasks(User* user, const char *sm) +std::string ModeUserServerNoticeMask::ProcessNoticeMasks(User* user, const std::string& input) { - bool adding = true, oldadding = false; - const char *c = sm; - std::string output; + bool adding = true; + std::bitset<64> curr = user->snomasks; - while (c && *c) + for (std::string::const_iterator i = input.begin(); i != input.end(); ++i) { - switch (*c) + switch (*i) { case '+': adding = true; @@ -86,82 +89,59 @@ std::string ModeUserServerNoticeMask::ProcessNoticeMasks(User* user, const char adding = false; break; case '*': - for (unsigned char d = 'a'; d <= 'z'; d++) + for (size_t j = 0; j < 64; j++) { - if (!ServerInstance->SNO->masks[d - 'a'].Description.empty()) - { - if ((!user->IsNoticeMaskSet(d) && adding) || (user->IsNoticeMaskSet(d) && !adding)) - { - if ((oldadding != adding) || (!output.length())) - output += (adding ? '+' : '-'); - - SetNoticeMask(user, d, adding); - - output += d; - } - oldadding = adding; - char u = toupper(d); - if ((!user->IsNoticeMaskSet(u) && adding) || (user->IsNoticeMaskSet(u) && !adding)) - { - if ((oldadding != adding) || (!output.length())) - output += (adding ? '+' : '-'); - - SetNoticeMask(user, u, adding); - - output += u; - } - oldadding = adding; - } + if (ServerInstance->SNO->IsSnomaskUsable(j+'A')) + curr[j] = adding; } break; default: - if (isalpha(*c)) + // For local users check whether the given snomask is valid and enabled - IsSnomaskUsable() tests both. + // For remote users accept what we were told, unless the snomask char is not a letter. + if (IS_LOCAL(user)) { - if ((!user->IsNoticeMaskSet(*c) && adding) || (user->IsNoticeMaskSet(*c) && !adding)) + if (!ServerInstance->SNO->IsSnomaskUsable(*i)) { - if ((oldadding != adding) || (!output.length())) - output += (adding ? '+' : '-'); - - SetNoticeMask(user, *c, adding); - - output += *c; + user->WriteNumeric(ERR_UNKNOWNSNOMASK, "%s %c :is unknown snomask char to me", user->nick.c_str(), *i); + continue; } } - else - user->WriteNumeric(ERR_UNKNOWNSNOMASK, "%s %c :is unknown snomask char to me", user->nick.c_str(), *c); + else if (!(((*i >= 'a') && (*i <= 'z')) || ((*i >= 'A') && (*i <= 'Z')))) + continue; - oldadding = adding; + size_t index = ((*i) - 'A'); + curr[index] = adding; break; } - - c++; } - std::string s = this->FormatNoticeMasks(user); - if (s.length() == 0) + std::string plus = "+"; + std::string minus = "-"; + + // Apply changes and construct two strings consisting of the newly added and the removed snomask chars + for (size_t i = 0; i < 64; i++) { - user->SetMode(this, false); + bool isset = curr[i]; + if (user->snomasks[i] != isset) + { + user->snomasks[i] = isset; + std::string& appendhere = (isset ? plus : minus); + appendhere.push_back(i+'A'); + } } - return output; -} + // Create the final string that will be shown to the user and sent to servers + // Form: "+ABc-de" + std::string output; + if (plus.length() > 1) + output = plus; -std::string ModeUserServerNoticeMask::FormatNoticeMasks(User* user) -{ - std::string data; + if (minus.length() > 1) + output += minus; - for (unsigned char n = 0; n < 64; n++) - { - if (user->snomasks[n]) - data.push_back(n + 65); - } - - return data; -} + // Unset the snomask usermode itself if every snomask was unset + if (user->snomasks.none()) + user->SetMode(this, false); -void ModeUserServerNoticeMask::SetNoticeMask(User* user, unsigned char sm, bool value) -{ - if (!isalpha(sm)) - return; - user->snomasks[sm-65] = value; + return output; } |