1 /* +------------------------------------+
2 * | Inspire Internet Relay Chat Daemon |
3 * +------------------------------------+
5 * InspIRCd: (C) 2002-2009 InspIRCd Development Team
6 * See: http://wiki.inspircd.org/Credits
8 * This program is free but copyrighted software; see
9 * the file COPYING for details.
11 * ---------------------------------------------------
17 /* $ModDesc: Provides the /shun command, which stops a user executing all commands except PING and PONG. */
19 class Shun : public XLine
22 std::string matchtext;
24 Shun(InspIRCd* Instance, time_t s_time, long d, std::string src, std::string re, std::string shunmask)
25 : XLine(Instance, s_time, d, src, re, "SHUN")
27 this->matchtext = shunmask;
40 if (InspIRCd::Match(u->GetFullHost(), matchtext) || InspIRCd::Match(u->GetFullRealHost(), matchtext) || InspIRCd::Match(u->nick+"!"+u->ident+"@"+u->GetIPString(), matchtext))
46 bool Matches(const std::string &s)
60 ServerInstance->SNO->WriteToSnoMask('x',"Removing expired shun %s (set by %s %ld seconds ago)",
61 this->matchtext.c_str(), this->source.c_str(), (long int)(ServerInstance->Time() - this->set_time));
64 const char* Displayable()
66 return matchtext.c_str();
70 /** An XLineFactory specialized to generate shun pointers
72 class ShunFactory : public XLineFactory
75 ShunFactory(InspIRCd* Instance) : XLineFactory(Instance, "SHUN") { }
79 XLine* Generate(time_t set_time, long duration, std::string source, std::string reason, std::string xline_specific_mask)
81 return new Shun(ServerInstance, set_time, duration, source, reason, xline_specific_mask);
85 //typedef std::vector<Shun> shunlist;
87 class CommandShun : public Command
90 CommandShun(InspIRCd* Me) : Command(Me, "SHUN", "o", 1, 3)
92 this->source = "m_shun.so";
93 this->syntax = "<nick!user@hostmask> [<shun-duration>] :<reason>";
96 CmdResult Handle(const std::vector<std::string>& parameters, User *user)
98 /* syntax: SHUN nick!user@host time :reason goes here */
99 /* 'time' is a human-readable timestring, like 2d3h2s. */
101 std::string target = parameters[0];
103 User *find = ServerInstance->FindNick(target.c_str());
105 target = std::string("*!*@") + find->GetIPString();
107 if (parameters.size() == 1)
109 if (ServerInstance->XLines->DelLine(target.c_str(), "SHUN", user))
111 ServerInstance->SNO->WriteToSnoMask('x',"%s Removed shun on %s.",user->nick.c_str(),target.c_str());
115 // XXX todo implement stats
116 user->WriteServ("NOTICE %s :*** Shun %s not found in list, try /stats S.",user->nick.c_str(),target.c_str());
121 else if (parameters.size() >= 2)
123 // Adding - XXX todo make this respect <insane> tag perhaps..
126 if (parameters.size() > 2)
128 duration = ServerInstance->Duration(parameters[1]);
129 expr = parameters[2];
134 expr = parameters[1];
140 r = new Shun(ServerInstance, ServerInstance->Time(), duration, user->nick.c_str(), expr.c_str(), target.c_str());
144 ; // Do nothing. If we get here, the regex was fucked up, and they already got told it fucked up.
149 if (ServerInstance->XLines->AddLine(r, user))
153 ServerInstance->SNO->WriteToSnoMask('x',"%s added permanent shun for %s: %s",
154 user->nick.c_str(), target.c_str(), expr.c_str());
158 time_t c_requires_crap = duration + ServerInstance->Time();
159 ServerInstance->SNO->WriteToSnoMask('x', "%s added timed shun for %s, expires on %s: %s",
160 user->nick.c_str(), target.c_str(), ServerInstance->TimeString(c_requires_crap).c_str(), expr.c_str());
163 ServerInstance->XLines->ApplyLines();
168 user->WriteServ("NOTICE %s :*** Shun for %s already exists", user->nick.c_str(), expr.c_str());
177 class ModuleShun : public Module
181 std::set<std::string> ShunEnabledCommands;
186 ModuleShun(InspIRCd* Me) : Module(Me), cmd(Me), f(Me)
188 ServerInstance->XLines->RegisterFactory(&f);
189 ServerInstance->AddCommand(&cmd);
191 Implementation eventlist[] = { I_OnStats, I_OnPreCommand, I_OnUserConnect, I_OnRehash };
192 ServerInstance->Modules->Attach(eventlist, this, 4);
196 virtual ~ModuleShun()
198 ServerInstance->XLines->DelAll("SHUN");
199 ServerInstance->XLines->UnregisterFactory(&f);
202 virtual int OnStats(char symbol, User* user, string_list& out)
207 ServerInstance->XLines->InvokeStats("SHUN", 223, user, out);
211 virtual void OnRehash(User* user)
213 ConfigReader MyConf(ServerInstance);
214 std::string cmds = MyConf.ReadValue("shun", "enabledcommands", 0);
217 cmds = "PING PONG QUIT";
219 ShunEnabledCommands.clear();
223 std::stringstream dcmds(cmds);
226 while (dcmds >> thiscmd)
228 ShunEnabledCommands.insert(thiscmd);
231 NotifyOfShun = MyConf.ReadFlag("shun", "notifyuser", "yes", 0);
232 affectopers = MyConf.ReadFlag("shun", "affectopers", "no", 0);
235 virtual void OnUserConnect(User* user)
240 // Apply lines on user connect
241 XLine *rl = ServerInstance->XLines->MatchesLine("SHUN", user);
250 virtual int OnPreCommand(std::string &command, std::vector<std::string>& parameters, User* user, bool validated, const std::string &original_line)
255 if (!ServerInstance->XLines->MatchesLine("SHUN", user))
257 /* Not shunned, don't touch. */
261 if (!affectopers && IS_OPER(user))
263 /* Don't do anything if the user is an operator and affectopers isn't set */
267 std::set<std::string>::iterator i = ShunEnabledCommands.find(command);
269 if (i == ShunEnabledCommands.end())
272 user->WriteServ("NOTICE %s :*** Command %s not processed, as you have been blocked from issuing commands (SHUN)", user->nick.c_str(), command.c_str());
276 if (command == "QUIT")
278 /* Allow QUIT but dont show any quit message */
281 else if (command == "PART")
287 /* if we're here, allow the command. */
291 virtual Version GetVersion()
293 return Version("$Id$",VF_VENDOR|VF_COMMON,API_VERSION);
297 MODULE_INIT(ModuleShun)