]> git.netwichtig.de Git - user/henk/code/inspircd.git/blob - src/modules/m_redirect.cpp
89097ade4cc07b318cea5722f90e4dc076dbb1d4
[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 /** Handle channel mode +L
28  */
29 class Redirect : public ModeHandler
30 {
31  public:
32         Redirect(Module* Creator) : ModeHandler(Creator, "redirect", 'L', PARAM_SETONLY, MODETYPE_CHANNEL) { }
33
34         ModeAction OnModeChange(User* source, User* dest, Channel* channel, std::string &parameter, bool adding)
35         {
36                 if (adding)
37                 {
38                         if (IS_LOCAL(source))
39                         {
40                                 if (!ServerInstance->IsChannel(parameter))
41                                 {
42                                         source->WriteNumeric(403, "%s %s :Invalid channel name", source->nick.c_str(), parameter.c_str());
43                                         parameter.clear();
44                                         return MODEACTION_DENY;
45                                 }
46                         }
47
48                         if (IS_LOCAL(source) && !source->IsOper())
49                         {
50                                 Channel* c = ServerInstance->FindChan(parameter);
51                                 if (!c)
52                                 {
53                                         source->WriteNumeric(690, "%s :Target channel %s must exist to be set as a redirect.",source->nick.c_str(),parameter.c_str());
54                                         parameter.clear();
55                                         return MODEACTION_DENY;
56                                 }
57                                 else if (c->GetPrefixValue(source) < OP_VALUE)
58                                 {
59                                         source->WriteNumeric(690, "%s :You must be opped on %s to set it as a redirect.",source->nick.c_str(),parameter.c_str());
60                                         parameter.clear();
61                                         return MODEACTION_DENY;
62                                 }
63                         }
64
65                         if (channel->GetModeParameter(this) == parameter)
66                                 return MODEACTION_DENY;
67                         /*
68                          * We used to do some checking for circular +L here, but there is no real need for this any more especially as we
69                          * now catch +L looping in PreJoin. Remove it, since O(n) logic makes me sad, and we catch it anyway. :) -- w00t
70                          */
71                         return MODEACTION_ALLOW;
72                 }
73                 else
74                 {
75                         if (channel->IsModeSet(this))
76                         {
77                                 return MODEACTION_ALLOW;
78                         }
79                 }
80
81                 return MODEACTION_DENY;
82
83         }
84 };
85
86 /** Handles usermode +L to stop forced redirection and print an error.
87 */
88 class AntiRedirect : public SimpleUserModeHandler
89 {
90         public:
91                 AntiRedirect(Module* Creator) : SimpleUserModeHandler(Creator, "antiredirect", 'L') {}
92 };
93
94 class ModuleRedirect : public Module
95 {
96         Redirect re;
97         AntiRedirect re_u;
98         ChanModeReference limitmode;
99         bool UseUsermode;
100
101  public:
102         ModuleRedirect()
103                 : re(this)
104                 , re_u(this)
105                 , limitmode(this, "limit")
106         {
107         }
108
109         void init() CXX11_OVERRIDE
110         {
111                 /* Setting this here so it isn't changable by rehasing the config later. */
112                 UseUsermode = ServerInstance->Config->ConfValue("redirect")->getBool("antiredirect");
113
114                 /* Channel mode */
115                 ServerInstance->Modules->AddService(re);
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(MODNAME, 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.");
122
123                         /* Try to add the usermode */
124                         ServerInstance->Modules->AddService(re_u);
125                 }
126         }
127
128         ModResult OnUserPreJoin(LocalUser* user, Channel* chan, const std::string& cname, std::string& privs, const std::string& keygiven) CXX11_OVERRIDE
129         {
130                 if (chan)
131                 {
132                         if (chan->IsModeSet(re) && chan->IsModeSet(limitmode))
133                         {
134                                 if (chan->GetUserCounter() >= ConvToInt(chan->GetModeParameter(limitmode)))
135                                 {
136                                         std::string channel = chan->GetModeParameter(&re);
137
138                                         /* sometimes broken ulines can make circular or chained +L, avoid this */
139                                         Channel* destchan = ServerInstance->FindChan(channel);
140                                         if (destchan && destchan->IsModeSet(re))
141                                         {
142                                                 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());
143                                                 return MOD_RES_DENY;
144                                         }
145                                         /* We check the bool value here to make sure we have it enabled, if we don't then
146                                                 usermode +L might be assigned to something else. */
147                                         if (UseUsermode && user->IsModeSet(re_u))
148                                         {
149                                                 user->WriteNumeric(470, "%s %s %s :Force redirection stopped.", user->nick.c_str(), cname.c_str(), channel.c_str());
150                                                 return MOD_RES_DENY;
151                                         }
152                                         else
153                                         {
154                                                 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());
155                                                 Channel::JoinUser(user, channel);
156                                                 return MOD_RES_DENY;
157                                         }
158                                 }
159                         }
160                 }
161                 return MOD_RES_PASSTHRU;
162         }
163
164         Version GetVersion() CXX11_OVERRIDE
165         {
166                 return Version("Provides channel mode +L (limit redirection) and user mode +L (no forced redirection)", VF_VENDOR);
167         }
168 };
169
170 MODULE_INIT(ModuleRedirect)