]> git.netwichtig.de Git - user/henk/code/inspircd.git/blob - src/modules/m_redirect.cpp
e40339fa6218b4a63665a1068712ebf45ac1503d
[user/henk/code/inspircd.git] / src / modules / m_redirect.cpp
1 /*
2  * InspIRCd -- Internet Relay Chat Daemon
3  *
4  *   Copyright (C) 2012 Shawn Smith <ShawnSmith0828@gmail.com>
5  *   Copyright (C) 2009 Daniel De Graaf <danieldg@inspircd.org>
6  *   Copyright (C) 2007-2008 Robin Burchell <robin+git@viroteck.net>
7  *   Copyright (C) 2007 Dennis Friis <peavey@inspircd.org>
8  *   Copyright (C) 2004, 2006 Craig Edwards <craigedwards@brainbox.cc>
9  *   Copyright (C) 2006 Oliver Lupton <oliverlupton@gmail.com>
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 #include "inspircd.h"
26
27 /* $ModDesc: Provides channel mode +L (limit redirection) and usermode +L (no forced redirection) */
28
29 /** Handle channel mode +L
30  */
31 class Redirect : public ModeHandler
32 {
33  public:
34         Redirect(Module* Creator) : ModeHandler(Creator, "redirect", 'L', PARAM_SETONLY, MODETYPE_CHANNEL) { }
35
36         ModeAction OnModeChange(User* source, User* dest, Channel* channel, std::string &parameter, bool adding)
37         {
38                 if (adding)
39                 {
40                         if (IS_LOCAL(source))
41                         {
42                                 if (!ServerInstance->IsChannel(parameter))
43                                 {
44                                         source->WriteNumeric(403, "%s %s :Invalid channel name", source->nick.c_str(), parameter.c_str());
45                                         parameter.clear();
46                                         return MODEACTION_DENY;
47                                 }
48                         }
49
50                         if (IS_LOCAL(source) && !source->IsOper())
51                         {
52                                 Channel* c = ServerInstance->FindChan(parameter);
53                                 if (!c)
54                                 {
55                                         source->WriteNumeric(690, "%s :Target channel %s must exist to be set as a redirect.",source->nick.c_str(),parameter.c_str());
56                                         parameter.clear();
57                                         return MODEACTION_DENY;
58                                 }
59                                 else if (c->GetPrefixValue(source) < OP_VALUE)
60                                 {
61                                         source->WriteNumeric(690, "%s :You must be opped on %s to set it as a redirect.",source->nick.c_str(),parameter.c_str());
62                                         parameter.clear();
63                                         return MODEACTION_DENY;
64                                 }
65                         }
66
67                         if (channel->GetModeParameter('L') == parameter)
68                                 return MODEACTION_DENY;
69                         /*
70                          * We used to do some checking for circular +L here, but there is no real need for this any more especially as we
71                          * now catch +L looping in PreJoin. Remove it, since O(n) logic makes me sad, and we catch it anyway. :) -- w00t
72                          */
73                         return MODEACTION_ALLOW;
74                 }
75                 else
76                 {
77                         if (channel->IsModeSet('L'))
78                         {
79                                 return MODEACTION_ALLOW;
80                         }
81                 }
82
83                 return MODEACTION_DENY;
84
85         }
86 };
87
88 /** Handles usermode +L to stop forced redirection and print an error.
89 */
90 class AntiRedirect : public SimpleUserModeHandler
91 {
92         public:
93                 AntiRedirect(Module* Creator) : SimpleUserModeHandler(Creator, "antiredirect", 'L') {}
94 };
95
96 class ModuleRedirect : public Module
97 {
98
99         Redirect re;
100         AntiRedirect re_u;
101         bool UseUsermode;
102
103  public:
104
105         ModuleRedirect()
106                 : re(this), re_u(this)
107         {
108         }
109
110         void init() CXX11_OVERRIDE
111         {
112                 /* Setting this here so it isn't changable by rehasing the config later. */
113                 UseUsermode = ServerInstance->Config->ConfValue("redirect")->getBool("antiredirect");
114
115                 /* Channel mode */
116                 ServerInstance->Modules->AddService(re);
117
118                 /* Check to see if the usermode is enabled in the config */
119                 if (UseUsermode)
120                 {
121                         /* Log noting that this breaks compatability. */
122                         ServerInstance->Logs->Log("m_redirect", LOG_DEFAULT, "REDIRECT: Enabled usermode +L. This breaks linking with servers that do not have this enabled. This is disabled by default in the 2.0 branch but will be enabled in the next version.");
123
124                         /* Try to add the usermode */
125                         ServerInstance->Modules->AddService(re_u);
126                 }
127
128                 Implementation eventlist[] = { I_OnUserPreJoin };
129                 ServerInstance->Modules->Attach(eventlist, this, sizeof(eventlist)/sizeof(Implementation));
130         }
131
132         ModResult OnUserPreJoin(LocalUser* user, Channel* chan, const std::string& cname, std::string& privs, const std::string& keygiven) CXX11_OVERRIDE
133         {
134                 if (chan)
135                 {
136                         if (chan->IsModeSet('L') && chan->IsModeSet('l'))
137                         {
138                                 if (chan->GetUserCounter() >= ConvToInt(chan->GetModeParameter('l')))
139                                 {
140                                         std::string channel = chan->GetModeParameter('L');
141
142                                         /* sometimes broken ulines can make circular or chained +L, avoid this */
143                                         Channel* destchan = ServerInstance->FindChan(channel);
144                                         if (destchan && destchan->IsModeSet('L'))
145                                         {
146                                                 user->WriteNumeric(470, "%s %s * :You may not join this channel. A redirect is set, but you may not be redirected as it is a circular loop.", user->nick.c_str(), cname.c_str());
147                                                 return MOD_RES_DENY;
148                                         }
149                                         /* We check the bool value here to make sure we have it enabled, if we don't then
150                                                 usermode +L might be assigned to something else. */
151                                         if (UseUsermode && user->IsModeSet('L'))
152                                         {
153                                                 user->WriteNumeric(470, "%s %s %s :Force redirection stopped.", user->nick.c_str(), cname.c_str(), channel.c_str());
154                                                 return MOD_RES_DENY;
155                                         }
156                                         else
157                                         {
158                                                 user->WriteNumeric(470, "%s %s %s :You may not join this channel, so you are automatically being transferred to the redirect channel.", user->nick.c_str(), cname.c_str(), channel.c_str());
159                                                 Channel::JoinUser(user, channel);
160                                                 return MOD_RES_DENY;
161                                         }
162                                 }
163                         }
164                 }
165                 return MOD_RES_PASSTHRU;
166         }
167
168         Version GetVersion() CXX11_OVERRIDE
169         {
170                 return Version("Provides channel mode +L (limit redirection) and user mode +L (no forced redirection)", VF_VENDOR);
171         }
172 };
173
174 MODULE_INIT(ModuleRedirect)