1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
|
/* +------------------------------------+
* | Inspire Internet Relay Chat Daemon |
* +------------------------------------+
*
* InspIRCd: (C) 2002-2008 InspIRCd Development Team
* See: http://www.inspircd.org/wiki/index.php/Credits
*
* This program is free but copyrighted software; see
* the file COPYING for details.
*
* ---------------------------------------------------
*/
#include "inspircd.h"
#include "m_cap.h"
/* $ModDesc: Provides the CAP negotiation mechanism seen in ratbox-derived ircds */
/*
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)
{
this->source = "m_cap.so";
}
CmdResult Handle (const char* const* parameters, int pcnt, User *user)
{
irc::string subcommand = parameters[0];
if (subcommand == "REQ")
{
CapData Data;
Data.type = parameters[1];
Data.user = user;
Data.creator = this->Creator;
Data.parameter = (pcnt > 1 ? parameters[1] : "");
user->Extend("CAP_REGHOLD");
Event event((char*) &Data, (Module*)this->Creator, "cap_req");
event.Send(this->ServerInstance);
}
else if (subcommand == "END")
{
user->Shrink("CAP_REGHOLD");
}
else if ((subcommand == "LS") || (subcommand == "LIST"))
{
CapData Data;
user->Extend("CAP_REGHOLD");
Data.type = subcommand;
Data.user = user;
Data.creator = this->Creator;
Data.parameter.clear();
Event event((char*) &Data, (Module*)this->Creator, subcommand == "LS" ? "cap_ls" : "cap_list");
event.Send(this->ServerInstance);
user->WriteServ("CAP * LS :%s", Data.parameter.c_str());
}
else
{
user->WriteServ("410 %s %s :Invalid CAP subcommand", user->nick, subcommand.c_str());
}
return CMD_FAILURE;
}
};
class ModuleCAP : public Module
{
CommandCAP* newcommand;
public:
ModuleCAP(InspIRCd* Me)
: Module(Me)
{
// Create a new command
newcommand = new CommandCAP(ServerInstance, this);
ServerInstance->AddCommand(newcommand);
Implementation eventlist[] = { I_OnCheckReady, I_OnCleanup, I_OnUserDisconnect, I_OnRequest };
ServerInstance->Modules->Attach(eventlist, this, 5);
}
virtual bool OnCheckReady(User* user)
{
/* Users in CAP state get held until CAP END */
if (user->GetExt("CAP_REGHOLD"))
return true;
return false;
}
virtual ~ModuleCAP()
{
}
virtual Version GetVersion()
{
return Version(1, 1, 0, 0, VF_VENDOR, API_VERSION);
}
};
MODULE_INIT(ModuleCAP)
|