-/* +------------------------------------+\r * | Inspire Internet Relay Chat Daemon |\r * +------------------------------------+\r *\r * InspIRCd: (C) 2002-2007 InspIRCd Development Team\r * See: http://www.inspircd.org/wiki/index.php/Credits\r *\r * This program is free but copyrighted software; see\r * the file COPYING for details.\r *\r * ---------------------------------------------------\r */\r\r/* $ModDesc: Adds timed bans */\r\r#include "inspircd.h"\r#include "users.h"\r#include "channels.h"\r#include "modules.h"\r#include "hashcomp.h"\r#include "configreader.h"\r\r/** Holds a timed ban\r */\rclass TimedBan : public classbase\r{\r public:\r std::string channel;\r std::string mask;\r time_t expire;\r};\r\rtypedef std::vector<TimedBan> timedbans;\rtimedbans TimedBanList;\r\r/** Handle /TBAN\r */\rclass cmd_tban : public command_t\r{\r public:\r cmd_tban (InspIRCd* Instance) : command_t(Instance,"TBAN", 0, 3)\r {\r this->source = "m_timedbans.so";\r syntax = "<channel> <duration> <banmask>";\r }\r\r CmdResult Handle (const char** parameters, int pcnt, userrec *user)\r {\r chanrec* channel = ServerInstance->FindChan(parameters[0]);\r if (channel)\r {\r int cm = channel->GetStatus(user);\r if ((cm == STATUS_HOP) || (cm == STATUS_OP))\r {\r if (!ServerInstance->IsValidMask(parameters[2]))\r {\r user->WriteServ("NOTICE "+std::string(user->nick)+" :Invalid ban mask");\r return CMD_FAILURE;\r }\r for (BanList::iterator i = channel->bans.begin(); i != channel->bans.end(); i++)\r {\r if (!strcasecmp(i->data,parameters[2]))\r {\r user->WriteServ("NOTICE "+std::string(user->nick)+" :The ban "+std::string(parameters[2])+" is already on the banlist of "+std::string(parameters[0]));\r return CMD_FAILURE;\r }\r }\r TimedBan T;\r std::string channelname = parameters[0];\r long duration = ServerInstance->Duration(parameters[1]);\r unsigned long expire = duration + time(NULL);\r if (duration < 1)\r {\r user->WriteServ("NOTICE "+std::string(user->nick)+" :Invalid ban time");\r return CMD_FAILURE;\r }\r std::string mask = parameters[2];\r const char *setban[32];\r setban[0] = parameters[0];\r setban[1] = "+b";\r setban[2] = parameters[2];\r // use CallCommandHandler to make it so that the user sets the mode\r // themselves\r ServerInstance->CallCommandHandler("MODE",setban,3,user);\r /* Check if the ban was actually added (e.g. banlist was NOT full) */\r bool was_added = false;\r for (BanList::iterator i = channel->bans.begin(); i != channel->bans.end(); i++)\r if (!strcasecmp(i->data,mask.c_str()))\r was_added = true;\r if (was_added)\r {\r T.channel = channelname;\r T.mask = mask;\r T.expire = expire;\r TimedBanList.push_back(T);\r channel->WriteChannelWithServ(ServerInstance->Config->ServerName, "NOTICE %s :%s added a timed ban on %s lasting for %ld seconds.", channel->name, user->nick, mask.c_str(), duration);\r return CMD_SUCCESS;\r }\r return CMD_FAILURE;\r }\r else user->WriteServ("482 %s %s :You must be at least a half-operator to change modes on this channel",user->nick, channel->name);\r return CMD_FAILURE;\r }\r user->WriteServ("401 %s %s :No such channel",user->nick, parameters[0]);\r return CMD_FAILURE;\r }\r};\r\rclass ModuleTimedBans : public Module\r{\r cmd_tban* mycommand;\r public:\r ModuleTimedBans(InspIRCd* Me)\r : Module(Me)\r {\r \r mycommand = new cmd_tban(ServerInstance);\r ServerInstance->AddCommand(mycommand);\r TimedBanList.clear();\r }\r \r virtual ~ModuleTimedBans()\r {\r TimedBanList.clear();\r }\r\r void Implements(char* List)\r {\r List[I_OnDelBan] = List[I_OnBackgroundTimer] = 1;\r }\r\r virtual int OnDelBan(userrec* source, chanrec* chan, const std::string &banmask)\r {\r irc::string listitem = banmask.c_str();\r irc::string thischan = chan->name;\r for (timedbans::iterator i = TimedBanList.begin(); i < TimedBanList.end(); i++)\r {\r irc::string target = i->mask.c_str();\r irc::string tchan = i->channel.c_str();\r if ((listitem == target) && (tchan == thischan))\r {\r TimedBanList.erase(i);\r break;\r }\r }\r return 0;\r }\r\r virtual void OnBackgroundTimer(time_t curtime)\r {\r bool again = true;\r while (again)\r {\r again = false;\r for (timedbans::iterator i = TimedBanList.begin(); i < TimedBanList.end(); i++)\r {\r if (curtime > i->expire)\r {\r chanrec* cr = ServerInstance->FindChan(i->channel);\r again = true;\r if (cr)\r {\r cr->WriteChannelWithServ(ServerInstance->Config->ServerName, "NOTICE %s :Timed ban on %s expired.", cr->name, i->mask.c_str());\r const char *setban[3];\r setban[0] = i->channel.c_str();\r setban[1] = "-b";\r setban[2] = i->mask.c_str();\r // kludge alert!\r // ::SendMode expects a userrec* to send the numeric replies\r // back to, so we create it a fake user that isnt in the user\r // hash and set its descriptor to FD_MAGIC_NUMBER so the data\r // falls into the abyss :p\r userrec* temp = new userrec(ServerInstance);\r temp->SetFd(FD_MAGIC_NUMBER);\r /* FIX: Send mode remotely*/\r std::deque<std::string> n;\r n.push_back(setban[0]);\r n.push_back("-b");\r n.push_back(setban[2]);\r ServerInstance->SendMode(setban,3,temp);\r Event rmode((char *)&n, NULL, "send_mode");\r rmode.Send(ServerInstance);\r DELETE(temp);\r }\r else\r {\r /* Where the hell did our channel go?! */\r TimedBanList.erase(i);\r }\r // we used to delete the item here, but we dont need to as the servermode above does it for us,\r break;\r }\r }\r }\r }\r \r virtual Version GetVersion()\r {\r return Version(1,1,0,0,VF_VENDOR,API_VERSION);\r }\r};\r\rMODULE_INIT(ModuleTimedBans)\r\r
\ No newline at end of file
+/* +------------------------------------+
+ * | Inspire Internet Relay Chat Daemon |
+ * +------------------------------------+
+ *
+ * InspIRCd: (C) 2002-2009 InspIRCd Development Team
+ * See: http://wiki.inspircd.org/Credits
+ *
+ * This program is free but copyrighted software; see
+ * the file COPYING for details.
+ *
+ * ---------------------------------------------------
+ */
+
+/* $ModDesc: Adds timed bans */
+
+#include "inspircd.h"
+
+/** Holds a timed ban
+ */
+class TimedBan : public classbase
+{
+ public:
+ std::string channel;
+ std::string mask;
+ time_t expire;
+};
+
+typedef std::vector<TimedBan> timedbans;
+timedbans TimedBanList;
+
+/** Handle /TBAN
+ */
+class CommandTban : public Command
+{
+ public:
+ CommandTban (InspIRCd* Instance) : Command(Instance,"TBAN", 0, 3)
+ {
+ this->source = "m_timedbans.so";
+ syntax = "<channel> <duration> <banmask>";
+ TRANSLATE4(TR_TEXT, TR_TEXT, TR_TEXT, TR_END);
+ }
+
+ CmdResult Handle (const std::vector<std::string> ¶meters, User *user)
+ {
+ Channel* channel = ServerInstance->FindChan(parameters[0]);
+ if (channel)
+ {
+ int cm = channel->GetStatus(user);
+ if ((cm == STATUS_HOP) || (cm == STATUS_OP))
+ {
+ if (!ServerInstance->IsValidMask(parameters[2]))
+ {
+ user->WriteServ("NOTICE "+std::string(user->nick)+" :Invalid ban mask");
+ return CMD_FAILURE;
+ }
+ for (BanList::iterator i = channel->bans.begin(); i != channel->bans.end(); i++)
+ {
+ if (!strcasecmp(i->data.c_str(), parameters[2].c_str()))
+ {
+ user->WriteServ("NOTICE "+std::string(user->nick)+" :The ban "+parameters[2]+" is already on the banlist of "+parameters[0]);
+ return CMD_FAILURE;
+ }
+ }
+ TimedBan T;
+ std::string channelname = parameters[0];
+ long duration = ServerInstance->Duration(parameters[1]);
+ unsigned long expire = duration + ServerInstance->Time();
+ if (duration < 1)
+ {
+ user->WriteServ("NOTICE "+std::string(user->nick)+" :Invalid ban time");
+ return CMD_FAILURE;
+ }
+ std::string mask = parameters[2];
+ std::vector<std::string> setban;
+ setban.push_back(parameters[0]);
+ setban.push_back("+b");
+ setban.push_back(parameters[2]);
+ // use CallCommandHandler to make it so that the user sets the mode
+ // themselves
+ ServerInstance->CallCommandHandler("MODE",setban,user);
+ /* Check if the ban was actually added (e.g. banlist was NOT full) */
+ bool was_added = false;
+ for (BanList::iterator i = channel->bans.begin(); i != channel->bans.end(); i++)
+ if (!strcasecmp(i->data.c_str(), mask.c_str()))
+ was_added = true;
+ if (was_added)
+ {
+ CUList tmp;
+ T.channel = channelname;
+ T.mask = mask;
+ T.expire = expire;
+ TimedBanList.push_back(T);
+ channel->WriteAllExcept(user, true, '@', tmp, "NOTICE %s :%s added a timed ban on %s lasting for %ld seconds.", channel->name.c_str(), user->nick.c_str(), mask.c_str(), duration);
+ ServerInstance->PI->SendChannelNotice(channel, '@', user->nick + " added a timed ban on " + mask + " lasting for " + ConvToStr(duration) + " seconds.");
+ if (ServerInstance->Config->AllowHalfop)
+ {
+ channel->WriteAllExcept(user, true, '%', tmp, "NOTICE %s :%s added a timed ban on %s lasting for %ld seconds.", channel->name.c_str(), user->nick.c_str(), mask.c_str(), duration);
+ ServerInstance->PI->SendChannelNotice(channel, '%', user->nick + " added a timed ban on " + mask + " lasting for " + ConvToStr(duration) + " seconds.");
+ }
+ return CMD_SUCCESS;
+ }
+ return CMD_FAILURE;
+ }
+ else user->WriteNumeric(482, "%s %s :You must be at least a%soperator to change modes on this channel",user->nick.c_str(), channel->name.c_str(),
+ ServerInstance->Config->AllowHalfop ? " half-" : " channel ");
+ return CMD_FAILURE;
+ }
+ user->WriteNumeric(401, "%s %s :No such channel",user->nick.c_str(), parameters[0].c_str());
+ return CMD_FAILURE;
+ }
+};
+
+class ModuleTimedBans : public Module
+{
+ CommandTban* mycommand;
+ public:
+ ModuleTimedBans(InspIRCd* Me)
+ : Module(Me)
+ {
+
+ mycommand = new CommandTban(ServerInstance);
+ ServerInstance->AddCommand(mycommand);
+ TimedBanList.clear();
+ Implementation eventlist[] = { I_OnDelBan, I_OnBackgroundTimer };
+ ServerInstance->Modules->Attach(eventlist, this, 2);
+ }
+
+ virtual ~ModuleTimedBans()
+ {
+ TimedBanList.clear();
+ }
+
+ virtual int OnDelBan(User* source, Channel* chan, const std::string &banmask)
+ {
+ irc::string listitem = banmask.c_str();
+ irc::string thischan = chan->name.c_str();
+ for (timedbans::iterator i = TimedBanList.begin(); i != TimedBanList.end(); i++)
+ {
+ irc::string target = i->mask.c_str();
+ irc::string tchan = i->channel.c_str();
+ if ((listitem == target) && (tchan == thischan))
+ {
+ TimedBanList.erase(i);
+ break;
+ }
+ }
+ return 0;
+ }
+
+ virtual void OnBackgroundTimer(time_t curtime)
+ {
+ for (timedbans::iterator i = TimedBanList.begin(); i != TimedBanList.end();)
+ {
+ if (curtime > i->expire)
+ {
+ std::string chan = i->channel;
+ std::string mask = i->mask;
+ Channel* cr = ServerInstance->FindChan(chan);
+ i = TimedBanList.erase(i);
+ if (cr)
+ {
+ std::vector<std::string> setban;
+ setban.push_back(chan);
+ setban.push_back("-b");
+ setban.push_back(mask);
+
+ CUList empty;
+ std::string expiry = "*** Timed ban on " + chan + " expired.";
+ cr->WriteAllExcept(ServerInstance->FakeClient, true, '@', empty, "NOTICE %s :%s", cr->name.c_str(), expiry.c_str());
+ ServerInstance->PI->SendChannelNotice(cr, '@', expiry);
+ if (ServerInstance->Config->AllowHalfop)
+ {
+ cr->WriteAllExcept(ServerInstance->FakeClient, true, '%', empty, "NOTICE %s :%s", cr->name.c_str(), expiry.c_str());
+ ServerInstance->PI->SendChannelNotice(cr, '%', expiry);
+ }
+
+ ServerInstance->SendMode(setban, ServerInstance->FakeClient);
+ ServerInstance->PI->SendMode(chan, ServerInstance->Modes->GetLastParseParams(), ServerInstance->Modes->GetLastParseTranslate());
+ }
+ }
+ else
+ ++i;
+ }
+ }
+
+ virtual Version GetVersion()
+ {
+ return Version("$Id$", VF_COMMON | VF_VENDOR, API_VERSION);
+ }
+};
+
+MODULE_INIT(ModuleTimedBans)
+