]> git.netwichtig.de Git - user/henk/code/inspircd.git/blob - src/modules/m_shun.cpp
Make shun block client-only tags by default.
[user/henk/code/inspircd.git] / src / modules / m_shun.cpp
1 /*
2  * InspIRCd -- Internet Relay Chat Daemon
3  *
4  *   Copyright (C) 2019 Matt Schatz <genius3000@g3k.solutions>
5  *   Copyright (C) 2018 linuxdaemon <linuxdaemon.irc@gmail.com>
6  *   Copyright (C) 2017-2018 B00mX0r <b00mx0r@aureus.pw>
7  *   Copyright (C) 2013, 2017-2018, 2020 Sadie Powell <sadie@witchery.services>
8  *   Copyright (C) 2012-2016 Attila Molnar <attilamolnar@hush.com>
9  *   Copyright (C) 2012, 2018-2019 Robby <robby@chatbelgie.be>
10  *   Copyright (C) 2012 Jens Voss <DukePyrolator@anope.org>
11  *   Copyright (C) 2009 Uli Schlachter <psychon@inspircd.org>
12  *   Copyright (C) 2009 Matt Smith <dz@inspircd.org>
13  *   Copyright (C) 2009 John Brooks <special@inspircd.org>
14  *   Copyright (C) 2009 Dennis Friis <peavey@inspircd.org>
15  *   Copyright (C) 2009 Daniel De Graaf <danieldg@inspircd.org>
16  *   Copyright (C) 2008-2010 Craig Edwards <brain@inspircd.org>
17  *   Copyright (C) 2008-2009 Robin Burchell <robin+git@viroteck.net>
18  *   Copyright (C) 2008 Thomas Stagner <aquanight@inspircd.org>
19  *
20  * This file is part of InspIRCd.  InspIRCd is free software: you can
21  * redistribute it and/or modify it under the terms of the GNU General Public
22  * License as published by the Free Software Foundation, version 2.
23  *
24  * This program is distributed in the hope that it will be useful, but WITHOUT
25  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
26  * FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more
27  * details.
28  *
29  * You should have received a copy of the GNU General Public License
30  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
31  */
32
33
34 #include "inspircd.h"
35 #include "xline.h"
36 #include "modules/shun.h"
37 #include "modules/stats.h"
38
39
40 /** An XLineFactory specialized to generate shun pointers
41  */
42 class ShunFactory : public XLineFactory
43 {
44  public:
45         ShunFactory() : XLineFactory("SHUN") { }
46
47         /** Generate a shun
48         */
49         XLine* Generate(time_t set_time, unsigned long duration, const std::string& source, const std::string& reason, const std::string& xline_specific_mask) CXX11_OVERRIDE
50         {
51                 return new Shun(set_time, duration, source, reason, xline_specific_mask);
52         }
53
54         bool AutoApplyToUserList(XLine* x) CXX11_OVERRIDE
55         {
56                 return false;
57         }
58 };
59
60 class CommandShun : public Command
61 {
62  public:
63         CommandShun(Module* Creator) : Command(Creator, "SHUN", 1, 3)
64         {
65                 flags_needed = 'o';
66                 syntax = "<nick!user@host> [<duration> :<reason>]";
67         }
68
69         CmdResult Handle(User* user, const Params& parameters) CXX11_OVERRIDE
70         {
71                 /* syntax: SHUN nick!user@host time :reason goes here */
72                 /* 'time' is a human-readable timestring, like 2d3h2s. */
73
74                 std::string target = parameters[0];
75
76                 User *find = ServerInstance->FindNick(target);
77                 if ((find) && (find->registered == REG_ALL))
78                         target = std::string("*!*@") + find->GetIPString();
79
80                 if (parameters.size() == 1)
81                 {
82                         std::string reason;
83
84                         if (ServerInstance->XLines->DelLine(parameters[0].c_str(), "SHUN", reason, user))
85                         {
86                                 ServerInstance->SNO->WriteToSnoMask('x', "%s removed SHUN on %s: %s", user->nick.c_str(), parameters[0].c_str(), reason.c_str());
87                         }
88                         else if (ServerInstance->XLines->DelLine(target.c_str(), "SHUN", reason, user))
89                         {
90                                 ServerInstance->SNO->WriteToSnoMask('x', "%s removed SHUN on %s: %s", user->nick.c_str(), target.c_str(), reason.c_str());
91                         }
92                         else
93                         {
94                                 user->WriteNotice("*** Shun " + parameters[0] + " not found on the list.");
95                                 return CMD_FAILURE;
96                         }
97                 }
98                 else
99                 {
100                         // Adding - XXX todo make this respect <insane> tag perhaps..
101                         unsigned long duration;
102                         std::string expr;
103                         if (parameters.size() > 2)
104                         {
105                                 if (!InspIRCd::Duration(parameters[1], duration))
106                                 {
107                                         user->WriteNotice("*** Invalid duration for SHUN.");
108                                         return CMD_FAILURE;
109                                 }
110                                 expr = parameters[2];
111                         }
112                         else
113                         {
114                                 duration = 0;
115                                 expr = parameters[1];
116                         }
117
118                         Shun* r = new Shun(ServerInstance->Time(), duration, user->nick.c_str(), expr.c_str(), target.c_str());
119                         if (ServerInstance->XLines->AddLine(r, user))
120                         {
121                                 if (!duration)
122                                 {
123                                         ServerInstance->SNO->WriteToSnoMask('x', "%s added permanent SHUN for %s: %s",
124                                                 user->nick.c_str(), target.c_str(), expr.c_str());
125                                 }
126                                 else
127                                 {
128                                         ServerInstance->SNO->WriteToSnoMask('x', "%s added timed SHUN for %s, expires in %s (on %s): %s",
129                                                 user->nick.c_str(), target.c_str(), InspIRCd::DurationString(duration).c_str(),
130                                                 InspIRCd::TimeString(ServerInstance->Time() + duration).c_str(), expr.c_str());
131                                 }
132                         }
133                         else
134                         {
135                                 delete r;
136                                 user->WriteNotice("*** Shun for " + target + " already exists.");
137                                 return CMD_FAILURE;
138                         }
139                 }
140                 return CMD_SUCCESS;
141         }
142
143         RouteDescriptor GetRouting(User* user, const Params& parameters) CXX11_OVERRIDE
144         {
145                 if (IS_LOCAL(user))
146                         return ROUTE_LOCALONLY; // spanningtree will send ADDLINE
147
148                 return ROUTE_BROADCAST;
149         }
150 };
151
152 class ModuleShun : public Module, public Stats::EventListener
153 {
154  private:
155         CommandShun cmd;
156         ShunFactory shun;
157         insp::flat_set<std::string, irc::insensitive_swo> enabledcommands;
158         bool affectopers;
159         bool allowtags;
160         bool notifyuser;
161
162         bool IsShunned(LocalUser* user)
163         {
164                 // Exempt the user from shuns if they are an oper and affectopers is disabled.
165                 if (!affectopers && user->IsOper())
166                         return false;
167
168                 // Exempt the user from shuns if they are an oper with the servers/ignore-shun privilege.
169                 if (user->HasPrivPermission("servers/ignore-shun"))
170                         return false;
171
172                 // Check whether the user is actually shunned.
173                 return ServerInstance->XLines->MatchesLine("SHUN", user);
174         }
175
176  public:
177         ModuleShun()
178                 : Stats::EventListener(this)
179                 , cmd(this)
180         {
181         }
182
183         void init() CXX11_OVERRIDE
184         {
185                 ServerInstance->XLines->RegisterFactory(&shun);
186         }
187
188         ~ModuleShun()
189         {
190                 ServerInstance->XLines->DelAll("SHUN");
191                 ServerInstance->XLines->UnregisterFactory(&shun);
192         }
193
194         void Prioritize() CXX11_OVERRIDE
195         {
196                 Module* alias = ServerInstance->Modules->Find("m_alias.so");
197                 ServerInstance->Modules->SetPriority(this, I_OnPreCommand, PRIORITY_BEFORE, alias);
198         }
199
200         ModResult OnStats(Stats::Context& stats) CXX11_OVERRIDE
201         {
202                 if (stats.GetSymbol() != 'H')
203                         return MOD_RES_PASSTHRU;
204
205                 ServerInstance->XLines->InvokeStats("SHUN", stats);
206                 return MOD_RES_DENY;
207         }
208
209         void ReadConfig(ConfigStatus& status) CXX11_OVERRIDE
210         {
211                 ConfigTag* tag = ServerInstance->Config->ConfValue("shun");
212
213                 enabledcommands.clear();
214                 irc::spacesepstream enabledcmds(tag->getString("enabledcommands", "ADMIN OPER PING PONG QUIT", 1));
215                 for (std::string enabledcmd; enabledcmds.GetToken(enabledcmd); )
216                         enabledcommands.insert(enabledcmd);
217
218                 affectopers = tag->getBool("affectopers", false);
219                 allowtags = tag->getBool("allowtags");
220                 notifyuser = tag->getBool("notifyuser", true);
221         }
222
223         ModResult OnPreCommand(std::string& command, CommandBase::Params& parameters, LocalUser* user, bool validated) CXX11_OVERRIDE
224         {
225                 if (validated || !IsShunned(user))
226                         return MOD_RES_PASSTHRU;
227
228                 if (!enabledcommands.count(command))
229                 {
230                         if (notifyuser)
231                                 user->WriteNotice("*** " + command + " command not processed as you have been blocked from issuing commands.");
232                         return MOD_RES_DENY;
233                 }
234
235                 if (!allowtags)
236                 {
237                         // Remove all client tags.
238                         ClientProtocol::TagMap& tags = parameters.GetTags();
239                         for (ClientProtocol::TagMap::iterator tag = tags.begin(); tag != tags.end(); )
240                         {
241                                 if (tag->first[0] == '+')
242                                         tag = tags.erase(tag);
243                                 else
244                                         tag++;
245                         }
246                 }
247                 if (command == "QUIT")
248                 {
249                         /* Allow QUIT but dont show any quit message */
250                         parameters.clear();
251                 }
252                 else if ((command == "PART") && (parameters.size() > 1))
253                 {
254                         /* same for PART */
255                         parameters.pop_back();
256                 }
257
258                 /* if we're here, allow the command. */
259                 return MOD_RES_PASSTHRU;
260         }
261
262         Version GetVersion() CXX11_OVERRIDE
263         {
264                 return Version("Adds the /SHUN command which allows server operators to prevent users from executing commands.", VF_VENDOR|VF_COMMON);
265         }
266 };
267
268 MODULE_INIT(ModuleShun)