* | Inspire Internet Relay Chat Daemon |
* +------------------------------------+
*
- * InspIRCd: (C) 2002-2008 InspIRCd Development Team
- * See: http://www.inspircd.org/wiki/index.php/Credits
+ * 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.
#include "inspircd.h"
#include "m_cap.h"
-/* $ModDesc: Provides a pointless /dalinfo command, demo module */
+/* $ModDesc: Provides the CAP negotiation mechanism seen in ratbox-derived ircds */
/*
- * std::string type;
- * std::vector<std::string> parameters;
- * User* user;
- * Module* creator;
- */
+CAP LS
+:alfred.staticbox.net CAP * LS :multi-prefix sasl
+CAP REQ :multi-prefix
+:alfred.staticbox.net CAP * ACK :multi-prefix
+CAP CLEAR
+:alfred.staticbox.net CAP * ACK :-multi-prefix
+CAP REQ :multi-prefix
+:alfred.staticbox.net CAP * ACK :multi-prefix
+CAP LIST
+:alfred.staticbox.net CAP * LIST :multi-prefix
+CAP END
+*/
/** Handle /CAP
*/
class CommandCAP : public Command
{
- Module* Creator;
public:
- /* Command 'dalinfo', takes no parameters and needs no special modes */
- CommandCAP (InspIRCd* Instance, Module* mod) : Command(Instance,"CAP", 0, 1, true), Creator(mod)
+ CommandCAP (InspIRCd* Instance, Module* mod) : Command(Instance, mod, "CAP", 0, 1, true)
{
- this->source = "m_cap.so";
}
- CmdResult Handle (const char* const* parameters, int pcnt, User *user)
+ CmdResult Handle (const std::vector<std::string> ¶meters, User *user)
{
- irc::string subcommand = parameters[0];
+ irc::string subcommand = parameters[0].c_str();
+
if (subcommand == "REQ")
{
CapData Data;
- Data.type = parameters[1];
+
+ Data.type = subcommand;
Data.user = user;
- Data.creator = this->Creator;
- Data.parameter = (pcnt > 1 ? parameters[1] : "");
+ Data.creator = this->creator;
+
+ if (parameters.size() < 2)
+ return CMD_FAILURE;
+
+ // tokenize the input into a nice list of requested caps
+ std::string param = parameters[1];
+ std::string cap_;
+ irc::spacesepstream cap_stream(param);
+
+ while (cap_stream.GetToken(cap_))
+ {
+ Data.wanted.push_back(cap_);
+ }
user->Extend("CAP_REGHOLD");
- Event event((char*) &Data, (Module*)this->Creator, "cap_req");
+ Event event((char*) &Data, this->creator, "cap_req");
event.Send(this->ServerInstance);
+
+ if (Data.ack.size() > 0)
+ {
+ std::string AckResult = irc::stringjoiner(" ", Data.ack, 0, Data.ack.size() - 1).GetJoined();
+ user->WriteServ("CAP * ACK :%s", AckResult.c_str());
+ }
+
+ if (Data.wanted.size() > 0)
+ {
+ std::string NakResult = irc::stringjoiner(" ", Data.wanted, 0, Data.wanted.size() - 1).GetJoined();
+ user->WriteServ("CAP * NAK :%s", NakResult.c_str());
+ }
}
else if (subcommand == "END")
{
user->Shrink("CAP_REGHOLD");
}
- else if (subcommand == "LS")
+ else if ((subcommand == "LS") || (subcommand == "LIST"))
{
CapData Data;
+
+ Data.type = subcommand;
+ Data.user = user;
+ Data.creator = this->creator;
+
user->Extend("CAP_REGHOLD");
- Data.type = "LS";
+ Event event((char*) &Data, this->creator, subcommand == "LS" ? "cap_ls" : "cap_list");
+ event.Send(this->ServerInstance);
+
+ std::string Result;
+ if (Data.wanted.size() > 0)
+ Result = irc::stringjoiner(" ", Data.wanted, 0, Data.wanted.size() - 1).GetJoined();
+ else
+ Result = "";
+
+ user->WriteServ("CAP * %s :%s", subcommand.c_str(), Result.c_str());
+ }
+ else if (subcommand == "CLEAR")
+ {
+ CapData Data;
+
+ Data.type = subcommand;
Data.user = user;
- Data.creator = this->Creator;
- Data.parameter.clear();
+ Data.creator = this->creator;
- Event event((char*) &Data, (Module*)this->Creator, "cap_ls");
+ user->Extend("CAP_REGHOLD");
+ Event event((char*) &Data, this->creator, "cap_clear");
event.Send(this->ServerInstance);
+
+ std::string Result = irc::stringjoiner(" ", Data.ack, 0, Data.ack.size() - 1).GetJoined();
+ user->WriteServ("CAP * ACK :%s", Result.c_str());
}
+ else
+ {
+ user->WriteNumeric(ERR_INVALIDCAPSUBCOMMAND, "* %s :Invalid CAP subcommand", subcommand.c_str());
+ }
+
return CMD_FAILURE;
}
};
class ModuleCAP : public Module
{
- CommandCAP* newcommand;
+ CommandCAP newcommand;
public:
ModuleCAP(InspIRCd* Me)
- : Module(Me)
+ : Module(Me), newcommand(Me, this)
{
- // Create a new command
- newcommand = new CommandCAP(ServerInstance, this);
- ServerInstance->AddCommand(newcommand);
+ ServerInstance->AddCommand(&newcommand);
- Implementation eventlist[] = { I_OnCheckReady, I_OnCleanup, I_OnUserDisconnect, I_OnRequest };
- ServerInstance->Modules->Attach(eventlist, this, 5);
+ Implementation eventlist[] = { I_OnCheckReady };
+ ServerInstance->Modules->Attach(eventlist, this, 1);
}
- virtual bool OnCheckReady(User* user)
+ virtual ModResult OnCheckReady(User* user)
{
/* Users in CAP state get held until CAP END */
if (user->GetExt("CAP_REGHOLD"))
- return true;
+ return MOD_RES_DENY;
- return false;
+ return MOD_RES_PASSTHRU;
}
virtual ~ModuleCAP()
virtual Version GetVersion()
{
- return Version(1, 1, 0, 0, VF_VENDOR, API_VERSION);
+ return Version("$Id$", VF_VENDOR, API_VERSION);
}
};