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 CAP negotiation mechanism seen in ratbox-derived ircds */
21 :alfred.staticbox.net CAP * LS :multi-prefix sasl
23 :alfred.staticbox.net CAP * ACK :multi-prefix
25 :alfred.staticbox.net CAP * ACK :-multi-prefix
27 :alfred.staticbox.net CAP * ACK :multi-prefix
29 :alfred.staticbox.net CAP * LIST :multi-prefix
35 class CommandCAP : public Command
38 CommandCAP (InspIRCd* Instance, Module* mod) : Command(Instance, mod, "CAP", 0, 1, true)
42 CmdResult Handle (const std::vector<std::string> ¶meters, User *user)
44 irc::string subcommand = parameters[0].c_str();
46 if (subcommand == "REQ")
50 Data.type = subcommand;
52 Data.creator = this->creator;
54 if (parameters.size() < 2)
57 // tokenize the input into a nice list of requested caps
58 std::string param = parameters[1];
60 irc::spacesepstream cap_stream(param);
62 while (cap_stream.GetToken(cap_))
64 Data.wanted.push_back(cap_);
67 user->Extend("CAP_REGHOLD");
68 Event event((char*) &Data, this->creator, "cap_req");
69 event.Send(this->ServerInstance);
71 if (Data.ack.size() > 0)
73 std::string AckResult = irc::stringjoiner(" ", Data.ack, 0, Data.ack.size() - 1).GetJoined();
74 user->WriteServ("CAP * ACK :%s", AckResult.c_str());
77 if (Data.wanted.size() > 0)
79 std::string NakResult = irc::stringjoiner(" ", Data.wanted, 0, Data.wanted.size() - 1).GetJoined();
80 user->WriteServ("CAP * NAK :%s", NakResult.c_str());
83 else if (subcommand == "END")
85 user->Shrink("CAP_REGHOLD");
87 else if ((subcommand == "LS") || (subcommand == "LIST"))
91 Data.type = subcommand;
93 Data.creator = this->creator;
95 user->Extend("CAP_REGHOLD");
96 Event event((char*) &Data, this->creator, subcommand == "LS" ? "cap_ls" : "cap_list");
97 event.Send(this->ServerInstance);
100 if (Data.wanted.size() > 0)
101 Result = irc::stringjoiner(" ", Data.wanted, 0, Data.wanted.size() - 1).GetJoined();
105 user->WriteServ("CAP * %s :%s", subcommand.c_str(), Result.c_str());
107 else if (subcommand == "CLEAR")
111 Data.type = subcommand;
113 Data.creator = this->creator;
115 user->Extend("CAP_REGHOLD");
116 Event event((char*) &Data, this->creator, "cap_clear");
117 event.Send(this->ServerInstance);
119 std::string Result = irc::stringjoiner(" ", Data.ack, 0, Data.ack.size() - 1).GetJoined();
120 user->WriteServ("CAP * ACK :%s", Result.c_str());
124 user->WriteNumeric(ERR_INVALIDCAPSUBCOMMAND, "* %s :Invalid CAP subcommand", subcommand.c_str());
131 class ModuleCAP : public Module
133 CommandCAP newcommand;
135 ModuleCAP(InspIRCd* Me)
136 : Module(Me), newcommand(Me, this)
138 ServerInstance->AddCommand(&newcommand);
140 Implementation eventlist[] = { I_OnCheckReady };
141 ServerInstance->Modules->Attach(eventlist, this, 1);
144 virtual bool OnCheckReady(User* user)
146 /* Users in CAP state get held until CAP END */
147 if (user->GetExt("CAP_REGHOLD"))
157 virtual Version GetVersion()
159 return Version("$Id$", VF_VENDOR, API_VERSION);
163 MODULE_INIT(ModuleCAP)