1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
|
/*
* InspIRCd -- Internet Relay Chat Daemon
*
* Copyright (C) 2015 Attila Molnar <attilamolnar@hush.com>
*
* This file is part of InspIRCd. InspIRCd is free software: you can
* redistribute it and/or modify it under the terms of the GNU General Public
* License as published by the Free Software Foundation, version 2.
*
* This program is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
* FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
* details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#pragma once
#include "modules/cap.h"
namespace IRCv3
{
class WriteNeighborsWithCap;
template <typename T>
class CapTag;
}
class IRCv3::WriteNeighborsWithCap : public User::ForEachNeighborHandler
{
const Cap::Capability& cap;
ClientProtocol::Event& protoev;
void Execute(LocalUser* user) CXX11_OVERRIDE
{
if (cap.get(user))
user->Send(protoev);
}
public:
WriteNeighborsWithCap(User* user, ClientProtocol::Event& ev, const Cap::Capability& capability)
: cap(capability)
, protoev(ev)
{
user->ForEachNeighbor(*this, false);
}
};
/** Base class for simple message tags.
* Message tags provided by classes derived from this class will be sent to clients that have negotiated
* a client capability, also managed by this class.
*
* Derived classes specify the name of the capability and the message tag and provide a public GetValue()
* method with the following signature: const std::string* GetValue(ClientProtocol::Message& msg).
* The returned value determines whether to attach the tag to the message. If it is NULL, the tag won't
* be attached. If it is non-NULL the tag will be attached with the value in the string. If the string is
* empty the tag is attached without a value.
*
* Providers inheriting from this class don't accept incoming tags by default.
*
* For more control, inherit from ClientProtocol::MessageTagProvider directly.
*
* Template parameter T is the derived class.
*/
template <typename T>
class IRCv3::CapTag : public ClientProtocol::MessageTagProvider
{
Cap::Capability cap;
const std::string tagname;
bool ShouldSendTag(LocalUser* user, const ClientProtocol::MessageTagData& tagdata) CXX11_OVERRIDE
{
return cap.get(user);
}
void OnClientProtocolPopulateTags(ClientProtocol::Message& msg) CXX11_OVERRIDE
{
T& tag = static_cast<T&>(*this);
const std::string* const val = tag.GetValue(msg);
if (val)
msg.AddTag(tagname, this, *val);
}
public:
/** Constructor.
* @param mod Module that owns the tag.
* @param capname Name of the client capability.
* A client capability with this name will be created. It will be available to all clients and it won't
* have a value.
* See Cap::Capability for more info on client capabilities.
* @param Tagname Name of the message tag, to use in the protocol.
*/
CapTag(Module* mod, const std::string& capname, const std::string& Tagname)
: ClientProtocol::MessageTagProvider(mod)
, cap(mod, capname)
, tagname(Tagname)
{
}
};
|