]> git.netwichtig.de Git - user/henk/code/inspircd.git/blob - src/modules/m_redirect.cpp
Merge insp20
[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                         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         }
111
112         void init() CXX11_OVERRIDE
113         {
114                 /* Setting this here so it isn't changable by rehasing the config later. */
115                 UseUsermode = ServerInstance->Config->ConfValue("redirect")->getBool("antiredirect");
116
117                 /* Channel mode */
118                 ServerInstance->Modules->AddService(re);
119
120                 /* Check to see if the usermode is enabled in the config */
121                 if (UseUsermode)
122                 {
123                         /* Log noting that this breaks compatability. */
124                         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.");
125
126                         /* Try to add the usermode */
127                         ServerInstance->Modules->AddService(re_u);
128                 }
129
130                 Implementation eventlist[] = { I_OnUserPreJoin };
131                 ServerInstance->Modules->Attach(eventlist, this, sizeof(eventlist)/sizeof(Implementation));
132         }
133
134         ModResult OnUserPreJoin(LocalUser* user, Channel* chan, const std::string& cname, std::string& privs, const std::string& keygiven) CXX11_OVERRIDE
135         {
136                 if (chan)
137                 {
138                         if (chan->IsModeSet('L') && chan->IsModeSet('l'))
139                         {
140                                 if (chan->GetUserCounter() >= ConvToInt(chan->GetModeParameter('l')))
141                                 {
142                                         std::string channel = chan->GetModeParameter('L');
143
144                                         /* sometimes broken ulines can make circular or chained +L, avoid this */
145                                         Channel* 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.c_str());
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.", user->nick.c_str(), cname.c_str(), channel.c_str());
156                                                 return MOD_RES_DENY;
157                                         }
158                                         else
159                                         {
160                                                 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());
161                                                 Channel::JoinUser(user, channel);
162                                                 return MOD_RES_DENY;
163                                         }
164                                 }
165                         }
166                 }
167                 return MOD_RES_PASSTHRU;
168         }
169
170         Version GetVersion() CXX11_OVERRIDE
171         {
172                 return Version("Provides channel mode +L (limit redirection) and user mode +L (no forced redirection)", VF_VENDOR);
173         }
174 };
175
176 MODULE_INIT(ModuleRedirect)