]> git.netwichtig.de Git - user/henk/code/inspircd.git/blob - src/modules/m_exemptchanops.cpp
Extend +X to take named modes in addition to mode letters
[user/henk/code/inspircd.git] / src / modules / m_exemptchanops.cpp
1 /*       +------------------------------------+
2  *       | Inspire Internet Relay Chat Daemon |
3  *       +------------------------------------+
4  *
5  *  InspIRCd: (C) 2002-2010 InspIRCd Development Team
6  * See: http://wiki.inspircd.org/Credits
7  *
8  * This program is free but copyrighted software; see
9  *            the file COPYING for details.
10  *
11  * ---------------------------------------------------
12  */
13
14 #include "inspircd.h"
15 #include "u_listmode.h"
16
17 /* $ModDesc: Provides the ability to allow channel operators to be exempt from certain modes. */
18
19 /** Handles channel mode +X
20  */
21 class ExemptChanOps : public ListModeBase
22 {
23  public:
24         ExemptChanOps(Module* Creator) : ListModeBase(Creator, "exemptchanops", 'X', "End of channel exemptchanops list", 954, 953, false, "exemptchanops") { }
25
26         bool ValidateParam(User* user, Channel* chan, std::string &word)
27         {
28                 // TODO actually make sure there's a prop for this
29                 if ((word.length() > 35) || (word.empty()))
30                 {
31                         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"));
32                         return false;
33                 }
34
35                 return true;
36         }
37
38         bool TellListTooLong(User* user, Channel* chan, std::string &word)
39         {
40                 user->WriteNumeric(959, "%s %s %s :Channel exemptchanops list is full", user->nick.c_str(), chan->name.c_str(), word.c_str());
41                 return true;
42         }
43
44         void TellAlreadyOnList(User* user, Channel* chan, std::string &word)
45         {
46                 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());
47         }
48
49         void TellNotSet(User* user, Channel* chan, std::string &word)
50         {
51                 user->WriteNumeric(958, "%s %s :No such exemptchanops word is set",user->nick.c_str(), chan->name.c_str());
52         }
53 };
54
55 class ModuleExemptChanOps : public Module
56 {
57         ExemptChanOps ec;
58         std::string defaults;
59
60  public:
61
62         ModuleExemptChanOps() : ec(this)
63         {
64         }
65
66         void init()
67         {
68                 ServerInstance->Modules->AddService(ec);
69                 Implementation eventlist[] = { I_OnChannelDelete, I_OnChannelRestrictionApply, I_OnRehash, I_OnSyncChannel };
70                 ServerInstance->Modules->Attach(eventlist, this, 4);
71
72                 OnRehash(NULL);
73         }
74
75         Version GetVersion()
76         {
77                 return Version("Provides the ability to allow channel operators to be exempt from certain modes.",VF_VENDOR);
78         }
79
80         void OnRehash(User* user)
81         {
82                 defaults = ServerInstance->Config->ConfValue("exemptchanops")->getString("defaults");
83                 ec.DoRehash();
84         }
85
86         void OnSyncChannel(Channel* chan, Module* proto, void* opaque)
87         {
88                 ec.DoSyncChannel(chan, proto, opaque);
89         }
90
91         ModeHandler* FindMode(const std::string& mid)
92         {
93                 if (mid.length() == 1)
94                         return ServerInstance->Modes->FindMode(mid[0], MODETYPE_CHANNEL);
95                 for(char c='A'; c < 'z'; c++)
96                 {
97                         ModeHandler* mh = ServerInstance->Modes->FindMode(c, MODETYPE_CHANNEL);
98                         if (mh && mh->name == mid)
99                                 return mh;
100                 }
101                 return NULL;
102         }
103
104         ModResult OnChannelRestrictionApply(User* user, Channel* chan, const char* restriction)
105         {
106                 unsigned int mypfx = chan->GetPrefixValue(user);
107                 irc::spacesepstream defaultstream(defaults);
108                 std::string minmode;
109                 std::string current;
110
111                 while (defaultstream.GetToken(current))
112                 {
113                         std::string::size_type pos = current.find(':');
114                         if (pos == std::string::npos)
115                                 continue;
116                         if (current.substr(0,pos) == restriction)
117                                 minmode = current[pos+1];
118                 }
119                 modelist* list = ec.extItem.get(chan);
120
121                 if (list)
122                 {
123                         for (modelist::iterator i = list->begin(); i != list->end(); ++i)
124                         {
125                                 std::string::size_type pos = (*i).mask.find(':');
126                                 if (pos == std::string::npos)
127                                         continue;
128                                 if ((*i).mask.substr(0,pos) == restriction)
129                                         minmode = (*i).mask.substr(pos + 1);
130                         }
131                 }
132
133                 ModeHandler* mh = FindMode(minmode);
134                 if (mh && mypfx >= mh->GetPrefixRank())
135                         return MOD_RES_ALLOW;
136                 if (mh || minmode == "*")
137                         return MOD_RES_DENY;
138                 return MOD_RES_PASSTHRU;
139         }
140 };
141
142 MODULE_INIT(ModuleExemptChanOps)