-/* +------------------------------------+
- * | Inspire Internet Relay Chat Daemon |
- * +------------------------------------+
+/*
+ * InspIRCd -- Internet Relay Chat Daemon
*
- * Inspire is copyright (C) 2002-2004 ChatSpike-Dev.
- * E-mail:
- * <brain@chatspike.net>
- * <Craig@chatspike.net>
- *
- * Written by Craig Edwards, Craig McLure, and others.
- * This program is free but copyrighted software; see
- * the file COPYING for details.
+ * Copyright (C) 2004-2006, 2008 Craig Edwards <craigedwards@brainbox.cc>
+ * Copyright (C) 2007 Dennis Friis <peavey@inspircd.org>
+ * Copyright (C) 2007 Robin Burchell <robin+git@viroteck.net>
*
- * ---------------------------------------------------
+ * This file is part of InspIRCd. InspIRCd is free software: you can
+ * redistribute it and/or modify it under the terms of the GNU General Public
+ * License as published by the Free Software Foundation, version 2.
+ *
+ * This program is distributed in the hope that it will be useful, but WITHOUT
+ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
+ * FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
+ * details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
-#include <stdio.h>
-#include <string>
-#include "users.h"
-#include "channels.h"
-#include "modules.h"
-/* $ModDesc: Provides support for /KNOCK and mode +K */
+#include "inspircd.h"
-Server *Srv;
-
-void handle_knock(char **parameters, int pcnt, userrec *user)
+enum
{
- chanrec* c = Srv->FindChannel(parameters[0]);
- std::string line = "";
+ // From UnrealIRCd.
+ ERR_CANNOTKNOCK = 480
+};
- for (int i = 1; i < pcnt - 1; i++)
- {
- line = line + std::string(parameters[i]) + " ";
- }
- line = line + std::string(parameters[pcnt-1]);
+/** Handles the /KNOCK command
+ */
+class CommandKnock : public Command
+{
+ SimpleChannelModeHandler& noknockmode;
+ ChanModeReference inviteonlymode;
- if (c->IsCustomModeSet('K'))
+ public:
+ bool sendnotice;
+ bool sendnumeric;
+ CommandKnock(Module* Creator, SimpleChannelModeHandler& Noknockmode)
+ : Command(Creator,"KNOCK", 2, 2)
+ , noknockmode(Noknockmode)
+ , inviteonlymode(Creator, "inviteonly")
{
- WriteServ(user->fd,"480 %s :Can't KNOCK on %s, +K is set.",user->nick, c->name);
- return;
+ syntax = "<channel> <reason>";
+ Penalty = 5;
}
- if (c->inviteonly)
+
+ CmdResult Handle(const std::vector<std::string>& parameters, User* user) CXX11_OVERRIDE
{
- WriteChannelWithServ((char*)Srv->GetServerName().c_str(),c,user,"NOTICE %s :User %s is KNOCKing on %s (%s)",c->name,user->nick,c->name,line.c_str());
- WriteServ(user->fd,"NOTICE %s :KNOCKing on %s",user->nick,c->name);
- return;
+ Channel* c = ServerInstance->FindChan(parameters[0]);
+ if (!c)
+ {
+ user->WriteNumeric(Numerics::NoSuchChannel(parameters[0]));
+ return CMD_FAILURE;
+ }
+
+ if (c->HasUser(user))
+ {
+ user->WriteNumeric(ERR_KNOCKONCHAN, c->name, InspIRCd::Format("Can't KNOCK on %s, you are already on that channel.", c->name.c_str()));
+ return CMD_FAILURE;
+ }
+
+ if (c->IsModeSet(noknockmode))
+ {
+ user->WriteNumeric(ERR_CANNOTKNOCK, InspIRCd::Format("Can't KNOCK on %s, +K is set.", c->name.c_str()));
+ return CMD_FAILURE;
+ }
+
+ if (!c->IsModeSet(inviteonlymode))
+ {
+ user->WriteNumeric(ERR_CHANOPEN, c->name, InspIRCd::Format("Can't KNOCK on %s, channel is not invite only so knocking is pointless!", c->name.c_str()));
+ return CMD_FAILURE;
+ }
+
+ if (sendnotice)
+ c->WriteNotice(InspIRCd::Format("User %s is KNOCKing on %s (%s)", user->nick.c_str(), c->name.c_str(), parameters[1].c_str()));
+
+ if (sendnumeric)
+ c->WriteChannelWithServ(ServerInstance->Config->ServerName, "710 %s %s %s :is KNOCKing: %s", c->name.c_str(), c->name.c_str(), user->GetFullHost().c_str(), parameters[1].c_str());
+
+ user->WriteNotice("KNOCKing on " + c->name);
+ return CMD_SUCCESS;
}
- else
+
+ RouteDescriptor GetRouting(User* user, const std::vector<std::string>& parameters) CXX11_OVERRIDE
{
- WriteServ(user->fd,"480 %s :Can't KNOCK on %s, channel is not invite only so knocking is pointless!",user->nick, c->name);
- return;
+ return ROUTE_OPT_BCAST;
}
-}
-
+};
class ModuleKnock : public Module
{
+ SimpleChannelModeHandler kn;
+ CommandKnock cmd;
+
public:
ModuleKnock()
+ : kn(this, "noknock", 'K')
+ , cmd(this, kn)
{
- Srv = new Server;
-
- Srv->AddExtendedMode('K',MT_CHANNEL,false,0,0);
- Srv->AddCommand("KNOCK",handle_knock,0,2);
}
- virtual void On005Numeric(std::string &output)
- {
- std::stringstream line(output);
- std::string temp1, temp2;
- while (!line.eof())
- {
- line >> temp1;
- if (temp1.substr(0,10) == "CHANMODES=")
- {
- // append the chanmode to the end
- temp1 = temp1.substr(10,temp1.length());
- temp1 = "CHANMODES=" + temp1 + "K";
- }
- temp2 = temp2 + temp1 + " ";
- }
- if (temp2.length())
+ void ReadConfig(ConfigStatus& status) CXX11_OVERRIDE
+ {
+ std::string knocknotify = ServerInstance->Config->ConfValue("knock")->getString("notify");
+ if (stdalgo::string::equalsci(knocknotify, "numeric"))
{
- output = temp2.substr(0,temp2.length()-1) + std::string(" KNOCK")
+ cmd.sendnotice = false;
+ cmd.sendnumeric = true;
}
- else output = output + std::string(" KNOCK");
- }
-
- virtual ~ModuleKnock()
- {
- delete Srv;
- }
-
- virtual Version GetVersion()
- {
- return Version(1,0,0,1);
- }
-
- virtual int OnExtendedMode(userrec* user, void* target, char modechar, int type, bool mode_on, string_list ¶ms)
- {
- // check if this is our mode character...
- if ((modechar == 'K') && (type == MT_CHANNEL))
- {
- return 1;
+ else if (stdalgo::string::equalsci(knocknotify, "both"))
+ {
+ cmd.sendnotice = true;
+ cmd.sendnumeric = true;
}
else
{
- return 0;
+ cmd.sendnotice = true;
+ cmd.sendnumeric = false;
}
}
-};
-
-// stuff down here is the module-factory stuff. For basic modules you can ignore this.
-
-class ModuleKnockFactory : public ModuleFactory
-{
- public:
- ModuleKnockFactory()
- {
- }
-
- ~ModuleKnockFactory()
+ Version GetVersion() CXX11_OVERRIDE
{
+ return Version("Provides support for /KNOCK and channel mode +K", VF_OPTCOMMON | VF_VENDOR);
}
-
- virtual Module * CreateModule()
- {
- return new ModuleKnock;
- }
-
};
-
-extern "C" void * init_module( void )
-{
- return new ModuleKnockFactory;
-}
-
+MODULE_INIT(ModuleKnock)