]> git.netwichtig.de Git - user/henk/code/inspircd.git/blob - src/modules/m_messageflood.cpp
Remove current time parameter of the Timer constructor
[user/henk/code/inspircd.git] / src / modules / m_messageflood.cpp
1 /*
2  * InspIRCd -- Internet Relay Chat Daemon
3  *
4  *   Copyright (C) 2009 Daniel De Graaf <danieldg@inspircd.org>
5  *   Copyright (C) 2008 Pippijn van Steenhoven <pip88nl@gmail.com>
6  *   Copyright (C) 2007 Robin Burchell <robin+git@viroteck.net>
7  *   Copyright (C) 2007 John Brooks <john.brooks@dereferenced.net>
8  *   Copyright (C) 2007 Dennis Friis <peavey@inspircd.org>
9  *   Copyright (C) 2006 Craig Edwards <craigedwards@brainbox.cc>
10  *   Copyright (C) 2006 Oliver Lupton <oliverlupton@gmail.com>
11  *
12  * This file is part of InspIRCd.  InspIRCd is free software: you can
13  * redistribute it and/or modify it under the terms of the GNU General Public
14  * License as published by the Free Software Foundation, version 2.
15  *
16  * This program is distributed in the hope that it will be useful, but WITHOUT
17  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
18  * FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more
19  * details.
20  *
21  * You should have received a copy of the GNU General Public License
22  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
23  */
24
25
26 #include "inspircd.h"
27
28 /** Holds flood settings and state for mode +f
29  */
30 class floodsettings
31 {
32  public:
33         bool ban;
34         unsigned int secs;
35         unsigned int lines;
36         time_t reset;
37         std::map<User*, unsigned int> counters;
38
39         floodsettings(bool a, int b, int c) : ban(a), secs(b), lines(c)
40         {
41                 reset = ServerInstance->Time() + secs;
42         }
43
44         bool addmessage(User* who)
45         {
46                 if (ServerInstance->Time() > reset)
47                 {
48                         counters.clear();
49                         reset = ServerInstance->Time() + secs;
50                 }
51
52                 return (++counters[who] >= this->lines);
53         }
54
55         void clear(User* who)
56         {
57                 std::map<User*, unsigned int>::iterator iter = counters.find(who);
58                 if (iter != counters.end())
59                 {
60                         counters.erase(iter);
61                 }
62         }
63 };
64
65 /** Handles channel mode +f
66  */
67 class MsgFlood : public ParamMode<MsgFlood, SimpleExtItem<floodsettings> >
68 {
69  public:
70         MsgFlood(Module* Creator)
71                 : ParamMode<MsgFlood, SimpleExtItem<floodsettings> >(Creator, "flood", 'f')
72         {
73         }
74
75         ModeAction OnSet(User* source, Channel* channel, std::string& parameter)
76         {
77                 std::string::size_type colon = parameter.find(':');
78                 if ((colon == std::string::npos) || (parameter.find('-') != std::string::npos))
79                 {
80                         source->WriteNumeric(608, "%s :Invalid flood parameter", channel->name.c_str());
81                         return MODEACTION_DENY;
82                 }
83
84                 /* Set up the flood parameters for this channel */
85                 bool ban = (parameter[0] == '*');
86                 unsigned int nlines = ConvToInt(parameter.substr(ban ? 1 : 0, ban ? colon-1 : colon));
87                 unsigned int nsecs = ConvToInt(parameter.substr(colon+1));
88
89                 if ((nlines<2) || (nsecs<1))
90                 {
91                         source->WriteNumeric(608, "%s :Invalid flood parameter", channel->name.c_str());
92                         return MODEACTION_DENY;
93                 }
94
95                 ext.set(channel, new floodsettings(ban, nsecs, nlines));
96                 return MODEACTION_ALLOW;
97         }
98
99         void SerializeParam(Channel* chan, const floodsettings* fs, std::string& out)
100         {
101                 if (fs->ban)
102                         out.push_back('*');
103                 out.append(ConvToStr(fs->lines)).push_back(':');
104                 out.append(ConvToStr(fs->secs));
105         }
106 };
107
108 class ModuleMsgFlood : public Module
109 {
110         MsgFlood mf;
111
112  public:
113
114         ModuleMsgFlood()
115                 : mf(this)
116         {
117         }
118
119         ModResult OnUserPreMessage(User* user, void* voiddest, int target_type, std::string& text, char status, CUList& exempt_list, MessageType msgtype) CXX11_OVERRIDE
120         {
121                 if (target_type != TYPE_CHANNEL)
122                         return MOD_RES_PASSTHRU;
123
124                 Channel* dest = static_cast<Channel*>(voiddest);
125                 if ((!IS_LOCAL(user)) || !dest->IsModeSet(mf))
126                         return MOD_RES_PASSTHRU;
127
128                 if (ServerInstance->OnCheckExemption(user,dest,"flood") == MOD_RES_ALLOW)
129                         return MOD_RES_PASSTHRU;
130
131                 floodsettings *f = mf.ext.get(dest);
132                 if (f)
133                 {
134                         if (f->addmessage(user))
135                         {
136                                 /* Youre outttta here! */
137                                 f->clear(user);
138                                 if (f->ban)
139                                 {
140                                         std::vector<std::string> parameters;
141                                         parameters.push_back(dest->name);
142                                         parameters.push_back("+b");
143                                         parameters.push_back("*!*@" + user->dhost);
144                                         ServerInstance->Modes->Process(parameters, ServerInstance->FakeClient);
145                                 }
146
147                                 const std::string kickMessage = "Channel flood triggered (limit is " + ConvToStr(f->lines) +
148                                         " in " + ConvToStr(f->secs) + " secs)";
149
150                                 dest->KickUser(ServerInstance->FakeClient, user, kickMessage);
151
152                                 return MOD_RES_DENY;
153                         }
154                 }
155
156                 return MOD_RES_PASSTHRU;
157         }
158
159         void Prioritize()
160         {
161                 // we want to be after all modules that might deny the message (e.g. m_muteban, m_noctcp, m_blockcolor, etc.)
162                 ServerInstance->Modules->SetPriority(this, I_OnUserPreMessage, PRIORITY_LAST);
163         }
164
165         Version GetVersion() CXX11_OVERRIDE
166         {
167                 return Version("Provides channel mode +f (message flood protection)", VF_VENDOR);
168         }
169 };
170
171 MODULE_INIT(ModuleMsgFlood)