]> git.netwichtig.de Git - user/henk/code/inspircd.git/blob - src/clientprotocol.cpp
Convert the ISO 8859-2 nationalchars files to codepage configs.
[user/henk/code/inspircd.git] / src / clientprotocol.cpp
1 /*
2  * InspIRCd -- Internet Relay Chat Daemon
3  *
4  *   Copyright (C) 2018-2020 Sadie Powell <sadie@witchery.services>
5  *   Copyright (C) 2018 Attila Molnar <attilamolnar@hush.com>
6  *
7  * This file is part of InspIRCd.  InspIRCd is free software: you can
8  * redistribute it and/or modify it under the terms of the GNU General Public
9  * License as published by the Free Software Foundation, version 2.
10  *
11  * This program is distributed in the hope that it will be useful, but WITHOUT
12  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
13  * FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more
14  * details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
18  */
19
20
21 #include "inspircd.h"
22
23 ClientProtocol::Serializer::Serializer(Module* mod, const char* Name)
24         : DataProvider(mod, std::string("serializer/") + Name)
25         , evprov(mod)
26 {
27 }
28
29 bool ClientProtocol::Serializer::HandleTag(LocalUser* user, const std::string& tagname, std::string& tagvalue, TagMap& tags) const
30 {
31         // Catch and block empty tags
32         if (tagname.empty())
33                 return false;
34
35         const ::Events::ModuleEventProvider::SubscriberList& list = evprov.GetSubscribers();
36         for (::Events::ModuleEventProvider::SubscriberList::const_iterator i = list.begin(); i != list.end(); ++i)
37         {
38                 MessageTagProvider* const tagprov = static_cast<MessageTagProvider*>(*i);
39                 const ModResult res = tagprov->OnProcessTag(user, tagname, tagvalue);
40                 if (res == MOD_RES_ALLOW)
41                         return tags.insert(std::make_pair(tagname, MessageTagData(tagprov, tagvalue))).second;
42                 else if (res == MOD_RES_DENY)
43                         break;
44         }
45
46         // No module handles the tag but that's not an error
47         return true;
48 }
49
50 ClientProtocol::TagSelection ClientProtocol::Serializer::MakeTagWhitelist(LocalUser* user, const TagMap& tagmap) const
51 {
52         TagSelection tagwl;
53         for (TagMap::const_iterator i = tagmap.begin(); i != tagmap.end(); ++i)
54         {
55                 const MessageTagData& tagdata = i->second;
56                 if (tagdata.tagprov->ShouldSendTag(user, tagdata))
57                         tagwl.Select(tagmap, i);
58         }
59         return tagwl;
60 }
61
62 const ClientProtocol::SerializedMessage& ClientProtocol::Serializer::SerializeForUser(LocalUser* user, Message& msg)
63 {
64         if (!msg.msginit_done)
65         {
66                 msg.msginit_done = true;
67                 FOREACH_MOD_CUSTOM(evprov, MessageTagProvider, OnPopulateTags, (msg));
68         }
69         return msg.GetSerialized(Message::SerializedInfo(this, MakeTagWhitelist(user, msg.GetTags())));
70 }
71
72 const ClientProtocol::SerializedMessage& ClientProtocol::Message::GetSerialized(const SerializedInfo& serializeinfo) const
73 {
74         // First check if the serialized line they're asking for is in the cache
75         for (SerializedList::const_iterator i = serlist.begin(); i != serlist.end(); ++i)
76         {
77                 const SerializedInfo& curr = i->first;
78                 if (curr == serializeinfo)
79                         return i->second;
80         }
81
82         // Not cached, generate it and put it in the cache for later use
83         serlist.push_back(std::make_pair(serializeinfo, serializeinfo.serializer->Serialize(*this, serializeinfo.tagwl)));
84         return serlist.back().second;
85 }
86
87 void ClientProtocol::Event::GetMessagesForUser(LocalUser* user, MessageList& messagelist)
88 {
89         if (!eventinit_done)
90         {
91                 eventinit_done = true;
92                 FOREACH_MOD_CUSTOM(*event, EventHook, OnEventInit, (*this));
93         }
94
95         // Most of the time there's only a single message but in rare cases there are more
96         if (initialmsg)
97                 messagelist.assign(1, initialmsg);
98         else
99                 messagelist = *initialmsglist;
100
101         // Let modules modify the message list
102         ModResult res;
103         FIRST_MOD_RESULT_CUSTOM(*event, EventHook, OnPreEventSend, res, (user, *this, messagelist));
104         if (res == MOD_RES_DENY)
105                 messagelist.clear();
106 }