Bitcoin Core 28.99.0
P2P Digital Currency
net_permissions.cpp
Go to the documentation of this file.
1// Copyright (c) 2009-2021 The Bitcoin Core developers
2// Distributed under the MIT software license, see the accompanying
3// file COPYING or http://www.opensource.org/licenses/mit-license.php.
4
5#include <common/messages.h>
6#include <common/system.h>
7#include <net_permissions.h>
8#include <netbase.h>
9#include <util/translation.h>
10
12
13const std::vector<std::string> NET_PERMISSIONS_DOC{
14 "bloomfilter (allow requesting BIP37 filtered blocks and transactions)",
15 "noban (do not ban for misbehavior; implies download)",
16 "forcerelay (relay transactions that are already in the mempool; implies relay)",
17 "relay (relay even in -blocksonly mode, and unlimited transaction announcements)",
18 "mempool (allow requesting BIP35 mempool contents)",
19 "download (allow getheaders during IBD, no disconnect after maxuploadtarget limit)",
20 "addr (responses to GETADDR avoid hitting the cache and contain random records with the most up-to-date info)"
21};
22
23namespace {
24
25// Parse the following format: "perm1,perm2@xxxxxx"
26static bool TryParsePermissionFlags(const std::string& str, NetPermissionFlags& output, ConnectionDirection* output_connection_direction, size_t& readen, bilingual_str& error)
27{
30 const auto atSeparator = str.find('@');
31
32 // if '@' is not found (ie, "xxxxx"), the caller should apply implicit permissions
33 if (atSeparator == std::string::npos) {
35 readen = 0;
36 }
37 // else (ie, "perm1,perm2@xxxxx"), let's enumerate the permissions by splitting by ',' and calculate the flags
38 else {
39 readen = 0;
40 // permissions == perm1,perm2
41 const auto permissions = str.substr(0, atSeparator);
42 while (readen < permissions.length()) {
43 const auto commaSeparator = permissions.find(',', readen);
44 const auto len = commaSeparator == std::string::npos ? permissions.length() - readen : commaSeparator - readen;
45 // permission == perm1
46 const auto permission = permissions.substr(readen, len);
47 readen += len; // We read "perm1"
48 if (commaSeparator != std::string::npos) readen++; // We read ","
49
50 if (permission == "bloomfilter" || permission == "bloom") NetPermissions::AddFlag(flags, NetPermissionFlags::BloomFilter);
51 else if (permission == "noban") NetPermissions::AddFlag(flags, NetPermissionFlags::NoBan);
52 else if (permission == "forcerelay") NetPermissions::AddFlag(flags, NetPermissionFlags::ForceRelay);
53 else if (permission == "mempool") NetPermissions::AddFlag(flags, NetPermissionFlags::Mempool);
54 else if (permission == "download") NetPermissions::AddFlag(flags, NetPermissionFlags::Download);
55 else if (permission == "all") NetPermissions::AddFlag(flags, NetPermissionFlags::All);
56 else if (permission == "relay") NetPermissions::AddFlag(flags, NetPermissionFlags::Relay);
57 else if (permission == "addr") NetPermissions::AddFlag(flags, NetPermissionFlags::Addr);
58 else if (permission == "in") connection_direction |= ConnectionDirection::In;
59 else if (permission == "out") {
60 if (output_connection_direction == nullptr) {
61 // Only NetWhitebindPermissions() should pass a nullptr.
62 error = _("whitebind may only be used for incoming connections (\"out\" was passed)");
63 return false;
64 }
65 connection_direction |= ConnectionDirection::Out;
66 }
67 else if (permission.length() == 0); // Allow empty entries
68 else {
69 error = strprintf(_("Invalid P2P permission: '%s'"), permission);
70 return false;
71 }
72 }
73 readen++;
74 }
75
76 // By default, whitelist only applies to incoming connections
77 if (connection_direction == ConnectionDirection::None) {
78 connection_direction = ConnectionDirection::In;
79 } else if (flags == NetPermissionFlags::None) {
80 error = strprintf(_("Only direction was set, no permissions: '%s'"), str);
81 return false;
82 }
83
84 output = flags;
85 if (output_connection_direction) *output_connection_direction = connection_direction;
86 error = Untranslated("");
87 return true;
88}
89
90}
91
93{
94 std::vector<std::string> strings;
95 if (NetPermissions::HasFlag(flags, NetPermissionFlags::BloomFilter)) strings.emplace_back("bloomfilter");
96 if (NetPermissions::HasFlag(flags, NetPermissionFlags::NoBan)) strings.emplace_back("noban");
97 if (NetPermissions::HasFlag(flags, NetPermissionFlags::ForceRelay)) strings.emplace_back("forcerelay");
98 if (NetPermissions::HasFlag(flags, NetPermissionFlags::Relay)) strings.emplace_back("relay");
99 if (NetPermissions::HasFlag(flags, NetPermissionFlags::Mempool)) strings.emplace_back("mempool");
100 if (NetPermissions::HasFlag(flags, NetPermissionFlags::Download)) strings.emplace_back("download");
101 if (NetPermissions::HasFlag(flags, NetPermissionFlags::Addr)) strings.emplace_back("addr");
102 return strings;
103}
104
106{
108 size_t offset;
109 if (!TryParsePermissionFlags(str, flags, /*output_connection_direction=*/nullptr, offset, error)) return false;
110
111 const std::string strBind = str.substr(offset);
112 const std::optional<CService> addrBind{Lookup(strBind, 0, false)};
113 if (!addrBind.has_value()) {
114 error = ResolveErrMsg("whitebind", strBind);
115 return false;
116 }
117 if (addrBind.value().GetPort() == 0) {
118 error = strprintf(_("Need to specify a port with -whitebind: '%s'"), strBind);
119 return false;
120 }
121
122 output.m_flags = flags;
123 output.m_service = addrBind.value();
124 error = Untranslated("");
125 return true;
126}
127
128bool NetWhitelistPermissions::TryParse(const std::string& str, NetWhitelistPermissions& output, ConnectionDirection& output_connection_direction, bilingual_str& error)
129{
131 size_t offset;
132 // Only NetWhitebindPermissions should pass a nullptr for output_connection_direction.
133 if (!TryParsePermissionFlags(str, flags, &output_connection_direction, offset, error)) return false;
134
135 const std::string net = str.substr(offset);
136 const CSubNet subnet{LookupSubNet(net)};
137 if (!subnet.IsValid()) {
138 error = strprintf(_("Invalid netmask specified in -whitelist: '%s'"), net);
139 return false;
140 }
141
142 output.m_flags = flags;
143 output.m_subnet = subnet;
144 error = Untranslated("");
145 return true;
146}
int flags
Definition: bitcoin-tx.cpp:536
NetPermissionFlags m_flags
static void AddFlag(NetPermissionFlags &flags, NetPermissionFlags f)
static std::vector< std::string > ToStrings(NetPermissionFlags flags)
static bool HasFlag(NetPermissionFlags flags, NetPermissionFlags f)
static bool TryParse(const std::string &str, NetWhitebindPermissions &output, bilingual_str &error)
static bool TryParse(const std::string &str, NetWhitelistPermissions &output, ConnectionDirection &output_connection_direction, bilingual_str &error)
is a home for simple string functions returning descriptive messages that are used in RPC and GUI int...
bilingual_str ResolveErrMsg(const std::string &optname, const std::string &strBind)
Definition: messages.cpp:148
const std::vector< std::string > NET_PERMISSIONS_DOC
NetPermissionFlags
CSubNet LookupSubNet(const std::string &subnet_str)
Parse and resolve a specified subnet string into the appropriate internal representation.
Definition: netbase.cpp:756
std::vector< CService > Lookup(const std::string &name, uint16_t portDefault, bool fAllowLookup, unsigned int nMaxSolutions, DNSLookupFn dns_lookup_function)
Resolve a service string to its corresponding service.
Definition: netbase.cpp:195
ConnectionDirection
Definition: netbase.h:33
Bilingual messages:
Definition: translation.h:24
#define strprintf
Format arguments and return the string or write to given std::ostream (see tinyformat::format doc for...
Definition: tinyformat.h:1172
consteval auto _(util::TranslatedLiteral str)
Definition: translation.h:79
bilingual_str Untranslated(std::string original)
Mark a bilingual_str as untranslated.
Definition: translation.h:82