]> git.netwichtig.de Git - user/henk/code/inspircd.git/blob - src/modules/m_redirect.cpp
Backported usermode +L from insp21. Stops forced redirection given by cmode +L
[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.c_str(), ServerInstance->Config->Limits.ChanMax))
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) && !IS_OPER(source))
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                         channel->SetModeParam('L', parameter);
74                         return MODEACTION_ALLOW;
75                 }
76                 else
77                 {
78                         if (channel->IsModeSet('L'))
79                         {
80                                 channel->SetModeParam('L', "");
81                                 return MODEACTION_ALLOW;
82                         }
83                 }
84
85                 return MODEACTION_DENY;
86
87         }
88 };
89
90 /** Handles usermode +L to stop forced redirection and print an error.
91 */
92 class AntiRedirect : public SimpleUserModeHandler
93 {
94         public:
95                 AntiRedirect(Module* Creator) : SimpleUserModeHandler(Creator, "antiredirect", 'L') {}
96 };
97
98 class ModuleRedirect : public Module
99 {
100
101         Redirect re;
102         AntiRedirect re_u;
103         bool UseUsermode;
104
105  public:
106
107         ModuleRedirect()
108                 : re(this), re_u(this)
109         {
110                 /* Setting this here so it isn't changable by rehasing the config later. */
111                 UseUsermode = ServerInstance->Config->ConfValue("redirect")->getBool("antiredirect");
112
113                 /* Channel mode */
114                 if (!ServerInstance->Modes->AddMode(&re))
115                         throw ModuleException("Could not add new modes!");
116
117                 /* Check to see if the usermode is enabled in the config */
118                 if (UseUsermode)
119                 {
120                         /* Log noting that this breaks compatability. */
121                         ServerInstance->Logs->Log("m_redirect", 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.");
122
123                         /* Try to add the usermode */
124                         if (!ServerInstance->Modes->AddMode(&re_u))
125                                 throw ModuleException("Could not add new modes!");
126                 }
127
128                 Implementation eventlist[] = { I_OnUserPreJoin };
129                 ServerInstance->Modules->Attach(eventlist, this, 1);
130         }
131
132
133         virtual ModResult OnUserPreJoin(User* user, Channel* chan, const char* cname, std::string &privs, const std::string &keygiven)
134         {
135                 if (chan)
136                 {
137                         if (chan->IsModeSet('L') && chan->IsModeSet('l'))
138                         {
139                                 if (chan->GetUserCounter() >= atoi(chan->GetModeParameter('l').c_str()))
140                                 {
141                                         std::string channel = chan->GetModeParameter('L');
142
143                                         /* sometimes broken ulines can make circular or chained +L, avoid this */
144                                         Channel* destchan = NULL;
145                                         destchan = ServerInstance->FindChan(channel);
146                                         if (destchan && destchan->IsModeSet('L'))
147                                         {
148                                                 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);
149                                                 return MOD_RES_DENY;
150                                         }
151                                         /* We check the bool value here to make sure we have it enabled, if we don't then
152                                                 usermode +L might be assigned to something else. */
153                                         if (UseUsermode && user->IsModeSet('L'))
154                                         {
155                                                 user->WriteNumeric(470, "%s %s %s :Force redirection stopped.",
156                                                 user->nick.c_str(), cname, channel.c_str());
157                                                 return MOD_RES_DENY;
158                                         }
159                                         else
160                                         {
161                                                 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, channel.c_str());
162                                                 Channel::JoinUser(user, channel.c_str(), false, "", false, ServerInstance->Time());
163                                                 return MOD_RES_DENY;
164                                         }
165                                 }
166                         }
167                 }
168                 return MOD_RES_PASSTHRU;
169         }
170
171         virtual ~ModuleRedirect()
172         {
173         }
174
175         virtual Version GetVersion()
176         {
177                 return Version("Provides channel mode +L (limit redirection) and user mode +L (no forced redirection)", VF_VENDOR);
178         }
179 };
180
181 MODULE_INIT(ModuleRedirect)