]> git.netwichtig.de Git - user/henk/code/inspircd.git/blob - src/modules/m_services_account.cpp
Handle SASL failures during SASL_INIT (wrong mechanism, etc.)
[user/henk/code/inspircd.git] / src / modules / m_services_account.cpp
1 /*
2  * InspIRCd -- Internet Relay Chat Daemon
3  *
4  *   Copyright (C) 2012 Shawn Smith <shawn@inspircd.org>
5  *   Copyright (C) 2009 Daniel De Graaf <danieldg@inspircd.org>
6  *   Copyright (C) 2006-2008 Robin Burchell <robin+git@viroteck.net>
7  *   Copyright (C) 2008 Pippijn van Steenhoven <pip88nl@gmail.com>
8  *   Copyright (C) 2006, 2008 Craig Edwards <craigedwards@brainbox.cc>
9  *   Copyright (C) 2007 Dennis Friis <peavey@inspircd.org>
10  *
11  * This file is part of InspIRCd.  InspIRCd is free software: you can
12  * redistribute it and/or modify it under the terms of the GNU General Public
13  * License as published by the Free Software Foundation, version 2.
14  *
15  * This program is distributed in the hope that it will be useful, but WITHOUT
16  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
17  * FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more
18  * details.
19  *
20  * You should have received a copy of the GNU General Public License
21  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
22  */
23
24
25 /* $ModDesc: Provides support for ircu-style services accounts, including chmode +R, etc. */
26
27 #include "inspircd.h"
28 #include "account.h"
29
30 /** Channel mode +r - mark a channel as identified
31  */
32 class Channel_r : public ModeHandler
33 {
34  public:
35         Channel_r(Module* Creator) : ModeHandler(Creator, "c_registered", 'r', PARAM_NONE, MODETYPE_CHANNEL) { }
36
37         ModeAction OnModeChange(User* source, User* dest, Channel* channel, std::string &parameter, bool adding)
38         {
39                 // only a u-lined server may add or remove the +r mode.
40                 if (!IS_LOCAL(source))
41                 {
42                         // Only change the mode if it's not redundant
43                         if ((adding != channel->IsModeSet('r')))
44                         {
45                                 channel->SetMode('r',adding);
46                                 return MODEACTION_ALLOW;
47                         }
48                 }
49                 else
50                 {
51                         source->WriteNumeric(500, "%s :Only a server may modify the +r channel mode", source->nick.c_str());
52                 }
53                 return MODEACTION_DENY;
54         }
55 };
56
57 /** User mode +r - mark a user as identified
58  */
59 class User_r : public ModeHandler
60 {
61
62  public:
63         User_r(Module* Creator) : ModeHandler(Creator, "u_registered", 'r', PARAM_NONE, MODETYPE_USER) { }
64
65         ModeAction OnModeChange(User* source, User* dest, Channel* channel, std::string &parameter, bool adding)
66         {
67                 if (!IS_LOCAL(source))
68                 {
69                         if ((adding != dest->IsModeSet('r')))
70                         {
71                                 dest->SetMode('r',adding);
72                                 return MODEACTION_ALLOW;
73                         }
74                 }
75                 else
76                 {
77                         source->WriteNumeric(500, "%s :Only a server may modify the +r user mode", source->nick.c_str());
78                 }
79                 return MODEACTION_DENY;
80         }
81 };
82
83 /** Channel mode +R - unidentified users cannot join
84  */
85 class AChannel_R : public SimpleChannelModeHandler
86 {
87  public:
88         AChannel_R(Module* Creator) : SimpleChannelModeHandler(Creator, "reginvite", 'R') { }
89 };
90
91 /** User mode +R - unidentified users cannot message
92  */
93 class AUser_R : public SimpleUserModeHandler
94 {
95  public:
96         AUser_R(Module* Creator) : SimpleUserModeHandler(Creator, "regdeaf", 'R') { }
97 };
98
99 /** Channel mode +M - unidentified users cannot message channel
100  */
101 class AChannel_M : public SimpleChannelModeHandler
102 {
103  public:
104         AChannel_M(Module* Creator) : SimpleChannelModeHandler(Creator, "regmoderated", 'M') { }
105 };
106
107 class ModuleServicesAccount : public Module
108 {
109         AChannel_R m1;
110         AChannel_M m2;
111         AUser_R m3;
112         Channel_r m4;
113         User_r m5;
114         AccountExtItem accountname;
115         bool checking_ban;
116
117  public:
118         ModuleServicesAccount() : m1(this), m2(this), m3(this), m4(this), m5(this),
119                 accountname("accountname", this), checking_ban(false)
120         {
121         }
122
123         void init()
124         {
125                 ServiceProvider* providerlist[] = { &m1, &m2, &m3, &m4, &m5, &accountname };
126                 ServerInstance->Modules->AddServices(providerlist, sizeof(providerlist)/sizeof(ServiceProvider*));
127                 Implementation eventlist[] = { I_OnWhois, I_OnUserPreMessage, I_OnUserPreNotice, I_OnUserPreJoin, I_OnCheckBan,
128                         I_OnDecodeMetaData, I_On005Numeric, I_OnUserPostNick, I_OnSetConnectClass };
129
130                 ServerInstance->Modules->Attach(eventlist, this, sizeof(eventlist)/sizeof(Implementation));
131         }
132
133         void On005Numeric(std::string &t)
134         {
135                 ServerInstance->AddExtBanChar('R');
136                 ServerInstance->AddExtBanChar('U');
137         }
138
139         /* <- :twisted.oscnet.org 330 w00t2 w00t2 w00t :is logged in as */
140         void OnWhois(User* source, User* dest)
141         {
142                 std::string *account = accountname.get(dest);
143
144                 if (account)
145                 {
146                         ServerInstance->SendWhoisLine(source, dest, 330, "%s %s %s :is logged in as", source->nick.c_str(), dest->nick.c_str(), account->c_str());
147                 }
148
149                 if (dest->IsModeSet('r'))
150                 {
151                         /* user is registered */
152                         ServerInstance->SendWhoisLine(source, dest, 307, "%s %s :is a registered nick", source->nick.c_str(), dest->nick.c_str());
153                 }
154         }
155
156         void OnUserPostNick(User* user, const std::string &oldnick)
157         {
158                 /* On nickchange, if they have +r, remove it */
159                 if (user->IsModeSet('r') && assign(user->nick) != oldnick)
160                 {
161                         std::vector<std::string> modechange;
162                         modechange.push_back(user->nick);
163                         modechange.push_back("-r");
164                         ServerInstance->SendMode(modechange, ServerInstance->FakeClient);
165                 }
166         }
167
168         ModResult OnUserPreMessage(User* user,void* dest,int target_type, std::string &text, char status, CUList &exempt_list)
169         {
170                 if (!IS_LOCAL(user))
171                         return MOD_RES_PASSTHRU;
172
173                 std::string *account = accountname.get(user);
174                 bool is_registered = account && !account->empty();
175
176                 if (target_type == TYPE_CHANNEL)
177                 {
178                         Channel* c = (Channel*)dest;
179                         ModResult res = ServerInstance->OnCheckExemption(user,c,"regmoderated");
180
181                         if (c->IsModeSet('M') && !is_registered && res != MOD_RES_ALLOW)
182                         {
183                                 // user messaging a +M channel and is not registered
184                                 user->WriteNumeric(477, user->nick+" "+c->name+" :You need to be identified to a registered account to message this channel");
185                                 return MOD_RES_DENY;
186                         }
187                 }
188                 else if (target_type == TYPE_USER)
189                 {
190                         User* u = (User*)dest;
191
192                         if (u->IsModeSet('R') && !is_registered)
193                         {
194                                 // user messaging a +R user and is not registered
195                                 user->WriteNumeric(477, ""+ user->nick +" "+ u->nick +" :You need to be identified to a registered account to message this user");
196                                 return MOD_RES_DENY;
197                         }
198                 }
199                 return MOD_RES_PASSTHRU;
200         }
201
202         ModResult OnCheckBan(User* user, Channel* chan, const std::string& mask)
203         {
204                 if (checking_ban)
205                         return MOD_RES_PASSTHRU;
206
207                 if ((mask.length() > 2) && (mask[1] == ':'))
208                 {
209                         if (mask[0] == 'R')
210                         {
211                                 std::string *account = accountname.get(user);
212                                 if (account && InspIRCd::Match(*account, mask.substr(2)))
213                                         return MOD_RES_DENY;
214                         }
215                         else if (mask[0] == 'U')
216                         {
217                                 std::string *account = accountname.get(user);
218                                 /* If the user is registered we don't care. */
219                                 if (account)
220                                         return MOD_RES_PASSTHRU;
221
222                                 /* If we made it this far we know the user isn't registered
223                                         so just deny if it matches */
224                                 checking_ban = true;
225                                 bool result = chan->CheckBan(user, mask.substr(2));
226                                 checking_ban = false;
227
228                                 if (result)
229                                         return MOD_RES_DENY;
230                         }
231                 }
232
233                 /* If we made it this far then the ban wasn't an ExtBan
234                         or the user we were checking for didn't match either ExtBan */
235                 return MOD_RES_PASSTHRU;
236         }
237
238         ModResult OnUserPreNotice(User* user,void* dest,int target_type, std::string &text, char status, CUList &exempt_list)
239         {
240                 return OnUserPreMessage(user, dest, target_type, text, status, exempt_list);
241         }
242
243         ModResult OnUserPreJoin(User* user, Channel* chan, const char* cname, std::string &privs, const std::string &keygiven)
244         {
245                 if (!IS_LOCAL(user))
246                         return MOD_RES_PASSTHRU;
247
248                 std::string *account = accountname.get(user);
249                 bool is_registered = account && !account->empty();
250
251                 if (chan)
252                 {
253                         if (chan->IsModeSet('R'))
254                         {
255                                 if (!is_registered)
256                                 {
257                                         // joining a +R channel and not identified
258                                         user->WriteNumeric(477, user->nick + " " + chan->name + " :You need to be identified to a registered account to join this channel");
259                                         return MOD_RES_DENY;
260                                 }
261                         }
262                 }
263                 return MOD_RES_PASSTHRU;
264         }
265
266         // Whenever the linking module receives metadata from another server and doesnt know what
267         // to do with it (of course, hence the 'meta') it calls this method, and it is up to each
268         // module in turn to figure out if this metadata key belongs to them, and what they want
269         // to do with it.
270         // In our case we're only sending a single string around, so we just construct a std::string.
271         // Some modules will probably get much more complex and format more detailed structs and classes
272         // in a textual way for sending over the link.
273         void OnDecodeMetaData(Extensible* target, const std::string &extname, const std::string &extdata)
274         {
275                 User* dest = dynamic_cast<User*>(target);
276                 // check if its our metadata key, and its associated with a user
277                 if (dest && (extname == "accountname"))
278                 {
279                         std::string *account = accountname.get(dest);
280                         if (account && !account->empty())
281                         {
282                                 trim(*account);
283
284                                 if (IS_LOCAL(dest))
285                                         dest->WriteNumeric(900, "%s %s %s :You are now logged in as %s",
286                                                 dest->nick.c_str(), dest->GetFullHost().c_str(), account->c_str(), account->c_str());
287
288                                 AccountEvent(this, dest, *account).Send();
289                         }
290                         else
291                         {
292                                 AccountEvent(this, dest, "").Send();
293                         }
294                 }
295         }
296
297         ModResult OnSetConnectClass(LocalUser* user, ConnectClass* myclass)
298         {
299                 if (myclass->config->getBool("requireaccount") && !accountname.get(user))
300                         return MOD_RES_DENY;
301                 return MOD_RES_PASSTHRU;
302         }
303
304         Version GetVersion()
305         {
306                 return Version("Provides support for ircu-style services accounts, including chmode +R, etc.",VF_OPTCOMMON|VF_VENDOR);
307         }
308 };
309
310 MODULE_INIT(ModuleServicesAccount)