2 * InspIRCd -- Internet Relay Chat Daemon
4 * Copyright (C) 2009 Daniel De Graaf <danieldg@inspircd.org>
5 * Copyright (C) 2008 Craig Edwards <craigedwards@brainbox.cc>
7 * This file is part of InspIRCd. InspIRCd is free software: you can
8 * redistribute it and/or modify it under the terms of the GNU General Public
9 * License as published by the Free Software Foundation, version 2.
11 * This program is distributed in the hope that it will be useful, but WITHOUT
12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
13 * FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
16 * You should have received a copy of the GNU General Public License
17 * along with this program. If not, see <http://www.gnu.org/licenses/>.
22 #include "modules/cap.h"
24 /* $ModDesc: Provides the CAP negotiation mechanism seen in ratbox-derived ircds */
28 :alfred.staticbox.net CAP * LS :multi-prefix sasl
30 :alfred.staticbox.net CAP * ACK :multi-prefix
32 :alfred.staticbox.net CAP * ACK :-multi-prefix
34 :alfred.staticbox.net CAP * ACK :multi-prefix
36 :alfred.staticbox.net CAP * LIST :multi-prefix
42 class CommandCAP : public Command
46 CommandCAP (Module* mod) : Command(mod, "CAP", 1),
47 reghold("CAP_REGHOLD", mod)
49 works_before_reg = true;
52 CmdResult Handle (const std::vector<std::string> ¶meters, User *user)
54 irc::string subcommand = parameters[0].c_str();
56 if (subcommand == "REQ")
58 if (parameters.size() < 2)
61 CapEvent Data(creator, user, CapEvent::CAPEVENT_REQ);
63 // tokenize the input into a nice list of requested caps
65 irc::spacesepstream cap_stream(parameters[1]);
67 while (cap_stream.GetToken(cap_))
69 Data.wanted.push_back(cap_);
75 if (Data.ack.size() > 0)
77 std::string AckResult = irc::stringjoiner(" ", Data.ack, 0, Data.ack.size() - 1).GetJoined();
78 user->WriteServ("CAP %s ACK :%s", user->nick.c_str(), AckResult.c_str());
81 if (Data.wanted.size() > 0)
83 std::string NakResult = irc::stringjoiner(" ", Data.wanted, 0, Data.wanted.size() - 1).GetJoined();
84 user->WriteServ("CAP %s NAK :%s", user->nick.c_str(), NakResult.c_str());
87 else if (subcommand == "END")
91 else if ((subcommand == "LS") || (subcommand == "LIST"))
93 CapEvent Data(creator, user, subcommand == "LS" ? CapEvent::CAPEVENT_LS : CapEvent::CAPEVENT_LIST);
99 if (Data.wanted.size() > 0)
100 Result = irc::stringjoiner(" ", Data.wanted, 0, Data.wanted.size() - 1).GetJoined();
102 user->WriteServ("CAP %s %s :%s", user->nick.c_str(), subcommand.c_str(), Result.c_str());
104 else if (subcommand == "CLEAR")
106 CapEvent Data(creator, user, CapEvent::CAPEVENT_CLEAR);
108 reghold.set(user, 1);
112 if (!Data.ack.empty())
113 Result = irc::stringjoiner(" ", Data.ack, 0, Data.ack.size() - 1).GetJoined();
114 user->WriteServ("CAP %s ACK :%s", user->nick.c_str(), Result.c_str());
118 user->WriteNumeric(ERR_INVALIDCAPSUBCOMMAND, "%s %s :Invalid CAP subcommand", user->nick.c_str(), subcommand.c_str());
126 class ModuleCAP : public Module
137 ServerInstance->Modules->AddService(cmd);
138 ServerInstance->Modules->AddService(cmd.reghold);
140 Implementation eventlist[] = { I_OnCheckReady };
141 ServerInstance->Modules->Attach(eventlist, this, sizeof(eventlist)/sizeof(Implementation));
144 ModResult OnCheckReady(LocalUser* user)
146 /* Users in CAP state get held until CAP END */
147 if (cmd.reghold.get(user))
150 return MOD_RES_PASSTHRU;
155 return Version("Client CAP extension support", VF_VENDOR);
159 MODULE_INIT(ModuleCAP)