2 * InspIRCd -- Internet Relay Chat Daemon
4 * Copyright (C) 2009 Daniel De Graaf <danieldg@inspircd.org>
6 * This file is part of InspIRCd. InspIRCd is free software: you can
7 * redistribute it and/or modify it under the terms of the GNU General Public
8 * License as published by the Free Software Foundation, version 2.
10 * This program is distributed in the hope that it will be useful, but WITHOUT
11 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
12 * FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
15 * You should have received a copy of the GNU General Public License
16 * along with this program. If not, see <http://www.gnu.org/licenses/>.
22 /* $ModDesc: Provides channelmode +d <int>, to deny messages to a channel until <int> seconds. */
24 class DelayMsgMode : public ModeHandler
28 DelayMsgMode(Module* Parent) : ModeHandler(Parent, "delaymsg", 'd', PARAM_SETONLY, MODETYPE_CHANNEL)
29 , jointime("delaymsg", Parent)
31 levelrequired = OP_VALUE;
34 bool ResolveModeConflict(std::string &their_param, const std::string &our_param, Channel*)
36 return (atoi(their_param.c_str()) < atoi(our_param.c_str()));
39 ModeAction OnModeChange(User* source, User* dest, Channel* channel, std::string ¶meter, bool adding);
42 class ModuleDelayMsg : public Module
46 ModuleDelayMsg() : djm(this)
50 void init() CXX11_OVERRIDE
52 ServerInstance->Modules->AddService(djm);
53 ServerInstance->Modules->AddService(djm.jointime);
54 Implementation eventlist[] = { I_OnUserJoin, I_OnUserPreMessage};
55 ServerInstance->Modules->Attach(eventlist, this, sizeof(eventlist)/sizeof(Implementation));
57 Version GetVersion() CXX11_OVERRIDE;
58 void OnUserJoin(Membership* memb, bool sync, bool created, CUList&) CXX11_OVERRIDE;
59 ModResult OnUserPreMessage(User* user, void* dest, int target_type, std::string& text, char status, CUList& exempt_list, MessageType msgtype) CXX11_OVERRIDE;
62 ModeAction DelayMsgMode::OnModeChange(User* source, User* dest, Channel* channel, std::string ¶meter, bool adding)
66 if ((channel->IsModeSet('d')) && (channel->GetModeParameter('d') == parameter))
67 return MODEACTION_DENY;
69 /* Setting a new limit, sanity check */
70 long limit = atoi(parameter.c_str());
72 /* Wrap low values at 32768 */
76 parameter = ConvToStr(limit);
80 if (!channel->IsModeSet('d'))
81 return MODEACTION_DENY;
86 const UserMembList* names = channel->GetUsers();
87 for (UserMembCIter n = names->begin(); n != names->end(); ++n)
88 jointime.set(n->second, 0);
90 channel->SetModeParam('d', adding ? parameter : "");
91 return MODEACTION_ALLOW;
94 Version ModuleDelayMsg::GetVersion()
96 return Version("Provides channelmode +d <int>, to deny messages to a channel until <int> seconds.", VF_VENDOR);
99 void ModuleDelayMsg::OnUserJoin(Membership* memb, bool sync, bool created, CUList&)
101 if ((IS_LOCAL(memb->user)) && (memb->chan->IsModeSet('d')))
103 djm.jointime.set(memb, ServerInstance->Time());
107 ModResult ModuleDelayMsg::OnUserPreMessage(User* user, void* dest, int target_type, std::string& text, char status, CUList& exempt_list, MessageType msgtype)
110 if ((!user) || (!IS_LOCAL(user)))
111 return MOD_RES_PASSTHRU;
113 if ((target_type != TYPE_CHANNEL) || (msgtype != MSG_PRIVMSG))
114 return MOD_RES_PASSTHRU;
116 Channel* channel = (Channel*) dest;
117 Membership* memb = channel->GetUser(user);
120 return MOD_RES_PASSTHRU;
122 time_t ts = djm.jointime.get(memb);
125 return MOD_RES_PASSTHRU;
127 std::string len = channel->GetModeParameter('d');
129 if (ts + atoi(len.c_str()) > ServerInstance->Time())
131 if (channel->GetPrefixValue(user) < VOICE_VALUE)
133 user->WriteNumeric(404, "%s %s :You must wait %s seconds after joining to send to channel (+d)",
134 user->nick.c_str(), channel->name.c_str(), len.c_str());
140 /* Timer has expired, we can stop checking now */
141 djm.jointime.set(memb, 0);
143 return MOD_RES_PASSTHRU;
146 MODULE_INIT(ModuleDelayMsg)