]> git.netwichtig.de Git - user/henk/code/inspircd.git/blob - src/modules/m_exemptchanops.cpp
Migrate u_listmode.h into the core, change +b to use it
[user/henk/code/inspircd.git] / src / modules / m_exemptchanops.cpp
1 /*
2  * InspIRCd -- Internet Relay Chat Daemon
3  *
4  *   Copyright (C) 2009-2010 Daniel De Graaf <danieldg@inspircd.org>
5  *
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.
9  *
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
13  * details.
14  *
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/>.
17  */
18
19
20 #include "inspircd.h"
21 #include "listmode.h"
22
23 /* $ModDesc: Provides the ability to allow channel operators to be exempt from certain modes. */
24
25 /** Handles channel mode +X
26  */
27 class ExemptChanOps : public ListModeBase
28 {
29  public:
30         ExemptChanOps(Module* Creator) : ListModeBase(Creator, "exemptchanops", 'X', "End of channel exemptchanops list", 954, 953, false, "exemptchanops") { }
31
32         bool ValidateParam(User* user, Channel* chan, std::string &word)
33         {
34                 // TODO actually make sure there's a prop for this
35                 if ((word.length() > 35) || (word.empty()))
36                 {
37                         user->WriteNumeric(955, "%s %s %s :word is too %s for exemptchanops list",user->nick.c_str(), chan->name.c_str(), word.c_str(), (word.empty() ? "short" : "long"));
38                         return false;
39                 }
40
41                 return true;
42         }
43
44         void TellListTooLong(User* user, Channel* chan, std::string &word)
45         {
46                 user->WriteNumeric(959, "%s %s %s :Channel exemptchanops list is full", user->nick.c_str(), chan->name.c_str(), word.c_str());
47         }
48
49         void TellAlreadyOnList(User* user, Channel* chan, std::string &word)
50         {
51                 user->WriteNumeric(957, "%s %s :The word %s is already on the exemptchanops list",user->nick.c_str(), chan->name.c_str(), word.c_str());
52         }
53
54         void TellNotSet(User* user, Channel* chan, std::string &word)
55         {
56                 user->WriteNumeric(958, "%s %s :No such exemptchanops word is set",user->nick.c_str(), chan->name.c_str());
57         }
58 };
59
60 class ExemptHandler : public HandlerBase3<ModResult, User*, Channel*, const std::string&>
61 {
62  public:
63         ExemptChanOps ec;
64         ExemptHandler(Module* me) : ec(me) {}
65
66         ModeHandler* FindMode(const std::string& mid)
67         {
68                 if (mid.length() == 1)
69                         return ServerInstance->Modes->FindMode(mid[0], MODETYPE_CHANNEL);
70                 for(char c='A'; c < 'z'; c++)
71                 {
72                         ModeHandler* mh = ServerInstance->Modes->FindMode(c, MODETYPE_CHANNEL);
73                         if (mh && mh->name == mid)
74                                 return mh;
75                 }
76                 return NULL;
77         }
78
79         ModResult Call(User* user, Channel* chan, const std::string& restriction)
80         {
81                 unsigned int mypfx = chan->GetPrefixValue(user);
82                 std::string minmode;
83
84                 ListModeBase::ModeList* list = ec.GetList(chan);
85
86                 if (list)
87                 {
88                         for (ListModeBase::ModeList::iterator i = list->begin(); i != list->end(); ++i)
89                         {
90                                 std::string::size_type pos = (*i).mask.find(':');
91                                 if (pos == std::string::npos)
92                                         continue;
93                                 if ((*i).mask.substr(0,pos) == restriction)
94                                         minmode = (*i).mask.substr(pos + 1);
95                         }
96                 }
97
98                 ModeHandler* mh = FindMode(minmode);
99                 if (mh && mypfx >= mh->GetPrefixRank())
100                         return MOD_RES_ALLOW;
101                 if (mh || minmode == "*")
102                         return MOD_RES_DENY;
103
104                 return ServerInstance->HandleOnCheckExemption.Call(user, chan, restriction);
105         }
106 };
107
108 class ModuleExemptChanOps : public Module
109 {
110         std::string defaults;
111         ExemptHandler eh;
112
113  public:
114         ModuleExemptChanOps() : eh(this)
115         {
116         }
117
118         void init()
119         {
120                 ServerInstance->Modules->AddService(eh.ec);
121                 Implementation eventlist[] = { I_OnRehash, I_OnSyncChannel };
122                 ServerInstance->Modules->Attach(eventlist, this, sizeof(eventlist)/sizeof(Implementation));
123                 ServerInstance->OnCheckExemption = &eh;
124
125                 OnRehash(NULL);
126         }
127
128         ~ModuleExemptChanOps()
129         {
130                 ServerInstance->OnCheckExemption = &ServerInstance->HandleOnCheckExemption;
131         }
132
133         Version GetVersion()
134         {
135                 return Version("Provides the ability to allow channel operators to be exempt from certain modes.",VF_VENDOR);
136         }
137
138         void OnRehash(User* user)
139         {
140                 eh.ec.DoRehash();
141         }
142
143         void OnSyncChannel(Channel* chan, Module* proto, void* opaque)
144         {
145                 eh.ec.DoSyncChannel(chan, proto, opaque);
146         }
147 };
148
149 MODULE_INIT(ModuleExemptChanOps)