8#include <chainparams.h>
13#include <test/data/asmap.raw.h>
19#include <boost/test/unit_test.hpp>
25using namespace std::literals;
34 return std::clamp<int32_t>(node_ctx.
args->
GetIntArg(
"-checkaddrman", 100), 0, 1000000);
39 const std::optional<CNetAddr> addr{
LookupHost(
ip,
false)};
40 BOOST_CHECK_MESSAGE(addr.has_value(),
strprintf(
"failed to resolve: %s",
ip));
46 const std::optional<CService> serv{
Lookup(
ip, port,
false)};
47 BOOST_CHECK_MESSAGE(serv.has_value(),
strprintf(
"failed to resolve: %s:%i",
ip, port));
61 auto addr_null = addrman->Select().first;
68 auto addr_ret1 = addrman->Select().first;
90 std::vector<CAddress> vAddr;
100 auto now = Now<NodeSeconds>();
107 addr.
nTime = Now<NodeSeconds>();
115 addr_helper.
nTime = Now<NodeSeconds>();
121 addrman->Attempt(addr,
true, Now<NodeSeconds>() - 61
s);
124 std::vector<CAddress> filtered{addrman->GetAddr(0, 0, std::nullopt)};
128 std::vector<CAddress> unfiltered{addrman->GetAddr(0, 0, std::nullopt,
false)};
138 const auto base_time{Now<NodeSeconds>() - 10000
s};
143 caddr1.
nTime = base_time;
145 const auto time_penalty{3600
s};
147 BOOST_CHECK(addrman->Add({caddr1}, source1, time_penalty));
149 auto addr_pos1{addrman->FindAddressEntry(caddr1)};
150 BOOST_REQUIRE(addr_pos1.has_value());
152 std::vector<CAddress> addresses{addrman->GetAddr(0, 0, std::nullopt)};
153 BOOST_REQUIRE_EQUAL(addresses.size(), 1U);
161 caddr2.
nTime = base_time;
163 BOOST_CHECK(addrman->Add({caddr2}, source2, time_penalty));
165 addresses = addrman->GetAddr(0, 0, std::nullopt);
166 BOOST_REQUIRE_EQUAL(addresses.size(), 2U);
168 CAddress retrieved_addr2{addresses[0]};
169 BOOST_CHECK(retrieved_addr2.nTime == base_time - time_penalty);
188 auto addr_ret2 = addrman->Select().first;
189 BOOST_CHECK(addr_ret2.ToStringAddrPort() ==
"250.1.1.1:8333" || addr_ret2.ToStringAddrPort() ==
"250.1.1.1:8334");
195 bool new_only =
true;
196 auto addr_ret3 = addrman->Select(new_only).first;
203 BOOST_CHECK(!addrman->Select(
false).first.IsValid());
204 BOOST_CHECK(!addrman->Select(
true).first.IsValid());
220 BOOST_CHECK(!addrman->Select(
true).first.IsValid());
241 BOOST_CHECK(addrman->Add({CAddress(addr5, NODE_NONE)}, addr3));
243 BOOST_CHECK(addrman->Add({CAddress(addr6, NODE_NONE)}, addr3));
252 std::set<uint16_t> ports;
253 for (
int i = 0; i < 20; ++i) {
254 ports.insert(addrman->Select().first.GetPort());
262 BOOST_CHECK(!addrman->Select(
true, {NET_IPV4}).first.IsValid());
263 BOOST_CHECK(!addrman->Select(
false, {NET_IPV4}).first.IsValid());
272 BOOST_CHECK(!addrman->Select(
false, {NET_IPV6}).first.IsValid());
273 BOOST_CHECK(!addrman->Select(
false, {NET_ONION}).first.IsValid());
274 BOOST_CHECK(!addrman->Select(
false, {NET_I2P}).first.IsValid());
275 BOOST_CHECK(!addrman->Select(
false, {NET_CJDNS}).first.IsValid());
276 BOOST_CHECK(!addrman->Select(
true, {NET_CJDNS}).first.IsValid());
281 i2p_addr.
SetSpecial(
"udhdrtrcetjm5sxzskjyr5ztpeszydbh4dpl3pl4utgqqw2v4jna.b32.i2p");
284 BOOST_CHECK(addrman->Select(
true, {NET_I2P}).first == i2p_addr);
285 BOOST_CHECK(addrman->Select(
false, {NET_I2P}).first == i2p_addr);
288 BOOST_CHECK(addrman->Select(
false, nets_with_entries).first.IsValid());
289 BOOST_CHECK(!addrman->Select(
false, {NET_IPV6}).first.IsValid());
290 BOOST_CHECK(!addrman->Select(
false, {NET_ONION}).first.IsValid());
291 BOOST_CHECK(!addrman->Select(
false, {NET_CJDNS}).first.IsValid());
293 BOOST_CHECK(!addrman->Select(
false, nets_without_entries).first.IsValid());
298 BOOST_CHECK(!addrman->Select(
true, {NET_I2P}).first.IsValid());
299 BOOST_CHECK(addrman->Select(
false, {NET_I2P}).first == i2p_addr);
303 i2p_addr2.
SetSpecial(
"c4gfnttsuwqomiygupdqqqyy5y5emnk5c73hrfvatri67prd7vyq.b32.i2p");
306 BOOST_CHECK(addrman->Select(
true, {NET_I2P}).first == i2p_addr2);
309 bool new_selected{
false};
310 bool tried_selected{
false};
313 while (--counter > 0 && (!new_selected || !tried_selected)) {
315 BOOST_REQUIRE(selected == i2p_addr || selected == i2p_addr2);
316 if (selected == i2p_addr) {
317 tried_selected =
true;
336 i2p_addr.
SetSpecial(
"udhdrtrcetjm5sxzskjyr5ztpeszydbh4dpl3pl4utgqqw2v4jna.b32.i2p");
356 uint32_t num_addrs{0};
360 while (num_addrs < 22) {
370 uint32_t collisions{1};
383 const auto start_time{Now<NodeSeconds>()};
384 addr.
nTime = start_time;
387 for (
unsigned int i = 1; i < 20; ++i) {
388 std::string addr_ip{
ToString(i % 256) +
"." +
ToString(i >> 8 % 256) +
".1.1"};
390 addrman->Add({addr},
source);
399 for (
unsigned int i = 1; i < 400; ++i) {
400 std::string addr_ip{
ToString(i % 256) +
"." +
ToString(i >> 8 % 256) +
".1.1"};
402 addr.nTime = start_time + std::chrono::seconds{i};
403 addrman->Add({addr},
source);
405 AddressPosition addr_pos_multi = addrman->FindAddressEntry(addr).value();
417 uint32_t num_addrs{0};
421 while (num_addrs < 35) {
449 std::vector<CAddress> vAddr1 = addrman->GetAddr(0, 0, std::nullopt);
453 addr1.nTime = Now<NodeSeconds>();
455 addr2.nTime = Now<NodeSeconds>();
457 addr3.
nTime = Now<NodeSeconds>();
459 addr4.
nTime = Now<NodeSeconds>();
461 addr5.
nTime = Now<NodeSeconds>();
466 BOOST_CHECK(addrman->Add({addr1, addr3, addr5}, source1));
467 BOOST_CHECK(addrman->Add({addr2, addr4}, source2));
480 for (
unsigned int i = 1; i < (8 * 256); i++) {
481 int octet1 = i % 256;
482 int octet2 = i >> 8 % 256;
487 addr.
nTime = Now<NodeSeconds>();
488 addrman->Add({addr},
ResolveIP(strAddr));
492 std::vector<CAddress> vAddr = addrman->GetAddr(2500, 23, std::nullopt);
494 size_t percent23 = (addrman->Size() * 23) / 100;
507 addr1.nTime = Now<NodeSeconds>();
516 addr3.
nTime = Now<NodeSeconds>();
517 addrman->Good(addr3, Now<NodeSeconds>());
521 for (
size_t i = 0; i < 3; ++i) {
522 addrman->Attempt(addr3,
true, Now<NodeSeconds>() - 61
s);
528 addr4.
nTime = Now<NodeSeconds>() + 11min;
563 std::set<int> buckets;
564 for (
int i = 0; i < 255; i++) {
569 buckets.insert(bucket);
576 for (
int j = 0; j < 255; j++) {
581 buckets.insert(bucket);
613 std::set<int> buckets;
614 for (
int i = 0; i < 255; i++) {
619 buckets.insert(bucket);
626 for (
int j = 0; j < 4 * 255; j++) {
632 buckets.insert(bucket);
639 for (
int p = 0; p < 255; p++) {
644 buckets.insert(bucket);
690 std::set<int> buckets;
691 for (
int j = 0; j < 255; j++) {
696 buckets.insert(bucket);
703 for (
int j = 0; j < 255; j++) {
708 buckets.insert(bucket);
742 std::set<int> buckets;
743 for (
int i = 0; i < 255; i++) {
748 buckets.insert(bucket);
755 for (
int j = 0; j < 4 * 255; j++) {
761 buckets.insert(bucket);
768 for (
int p = 0; p < 255; p++) {
773 buckets.insert(bucket);
780 for (
int p = 0; p < 255; p++) {
785 buckets.insert(bucket);
797 auto addrman_asmap1 = std::make_unique<AddrMan>(netgroupman,
DETERMINISTIC, ratio);
798 auto addrman_asmap1_dup = std::make_unique<AddrMan>(netgroupman,
DETERMINISTIC, ratio);
806 addrman_asmap1->Add({addr}, default_source);
808 stream << *addrman_asmap1;
810 stream >> *addrman_asmap1_dup;
812 AddressPosition addr_pos1 = addrman_asmap1->FindAddressEntry(addr).value();
813 AddressPosition addr_pos2 = addrman_asmap1_dup->FindAddressEntry(addr).value();
820 stream << *addrman_asmap1;
821 stream >> *addrman_noasmap;
822 AddressPosition addr_pos3 = addrman_noasmap->FindAddressEntry(addr).value();
828 addrman_asmap1 = std::make_unique<AddrMan>(netgroupman,
DETERMINISTIC, ratio);
830 addrman_noasmap->Add({addr}, default_source);
831 stream << *addrman_noasmap;
832 stream >> *addrman_asmap1;
834 AddressPosition addr_pos4 = addrman_asmap1->FindAddressEntry(addr).value();
840 addrman_asmap1 = std::make_unique<AddrMan>(netgroupman,
DETERMINISTIC, ratio);
844 addrman_noasmap->Add({addr,
addr2}, default_source);
848 stream << *addrman_noasmap;
849 stream >> *addrman_asmap1;
868 addrman->Add({new1, tried1, new2, tried2},
CNetAddr{});
869 addrman->Good(tried1);
870 addrman->Good(tried2);
871 BOOST_REQUIRE_EQUAL(addrman->Size(), 4);
875 const std::string str{stream.str()};
878 const char new2_raw[]{6, 6, 6, 6};
879 const uint8_t new2_raw_replacement[]{0, 0, 0, 0};
880 pos = str.find(new2_raw, 0,
sizeof(new2_raw));
881 BOOST_REQUIRE(pos != std::string::npos);
882 BOOST_REQUIRE(pos +
sizeof(new2_raw_replacement) <= stream.size());
883 memcpy(stream.data() + pos, new2_raw_replacement,
sizeof(new2_raw_replacement));
885 const char tried2_raw[]{8, 8, 8, 8};
886 const uint8_t tried2_raw_replacement[]{255, 255, 255, 255};
887 pos = str.find(tried2_raw, 0,
sizeof(tried2_raw));
888 BOOST_REQUIRE(pos != std::string::npos);
889 BOOST_REQUIRE(pos +
sizeof(tried2_raw_replacement) <= stream.size());
890 memcpy(stream.data() + pos, tried2_raw_replacement,
sizeof(tried2_raw_replacement));
904 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
908 for (
unsigned int i = 1; i < 23; i++) {
914 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
919 for (
unsigned int i = 1; i < 23; i++) {
926 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
936 for (
unsigned int i = 1; i < 36; i++) {
948 BOOST_CHECK_EQUAL(addrman->SelectTriedCollision().first.ToStringAddrPort(),
"250.1.1.19:0");
953 addrman->ResolveCollisions();
954 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
957 for (
unsigned int i = 37; i < 59; i++) {
968 BOOST_CHECK_EQUAL(addrman->SelectTriedCollision().first.ToStringAddrPort(),
"250.1.1.10:0");
975 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() !=
"[::]:0");
978 addrman->ResolveCollisions();
979 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
989 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
993 for (
unsigned int i = 1; i < 36; i++) {
1006 auto info = addrman->SelectTriedCollision().first;
1012 addrman->Attempt(info,
false, Now<NodeSeconds>() - 61
s);
1015 addrman->ResolveCollisions();
1016 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
1025 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
1031 BOOST_CHECK_EQUAL(addrman->SelectTriedCollision().first.ToStringAddrPort(),
"250.1.1.36:0");
1036 addrman->ResolveCollisions();
1037 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
1049 ssPeersIn << addrman;
1057 std::optional<CService>
addr1,
addr2, addr3, addr4;
1062 addr3 =
Lookup(
"250.7.3.3", 9999,
false);
1064 addr3 =
Lookup(
"250.7.3.3"s, 9999,
false);
1066 addr4 =
Lookup(
"250.7.3.3\0example.com"s, 9999,
false);
1070 const std::optional<CService>
source{
Lookup(
"252.5.1.1", 8333,
false)};
1078 bool exceptionThrown =
false;
1083 unsigned char pchMsgTmp[4];
1084 ssPeers1 >> pchMsgTmp;
1085 ssPeers1 >> addrman1;
1086 }
catch (
const std::exception&) {
1087 exceptionThrown =
true;
1108 unsigned char nVersion = 1;
1110 s << ((
unsigned char)32);
1118 const std::optional<CService> serv{
Lookup(
"252.1.1.1", 7777,
false)};
1119 BOOST_REQUIRE(serv.has_value());
1121 std::optional<CNetAddr> resolved{
LookupHost(
"252.2.2.2",
false)};
1122 BOOST_REQUIRE(resolved.has_value());
1133 bool exceptionThrown =
false;
1137 unsigned char pchMsgTmp[4];
1138 ssPeers1 >> pchMsgTmp;
1139 ssPeers1 >> addrman1;
1140 }
catch (
const std::exception&) {
1141 exceptionThrown =
true;
1160 const auto start_time{Now<NodeSeconds>() - 10000
s};
1161 addr.nTime = start_time;
1167 addr_diff_port.
nTime = start_time;
1168 addrman->Connected(addr_diff_port);
1170 std::vector<CAddress> vAddr1{addrman->GetAddr(0, 0, std::nullopt)};
1176 addrman->Connected(addr);
1178 std::vector<CAddress> vAddr2 = addrman->GetAddr(0, 0, std::nullopt);
1180 BOOST_CHECK(vAddr2.at(0).nTime >= start_time + 10000
s);
1185 addr_v2.
nTime = start_time;
1187 std::vector<CAddress> vAddr3{addrman->GetAddr(0, 0, std::nullopt)};
1193 std::vector<CAddress> vAddr4{addrman->GetAddr(0, 0, std::nullopt)};
1199 std::vector<CAddress> vAddr5{addrman->GetAddr(0, 0, std::nullopt)};
1205 std::vector<CAddress> vAddr6{addrman->GetAddr(0, 0, std::nullopt)};
1237 i2p_addr.
SetSpecial(
"UDHDrtrcetjm5sxzskjyr5ztpeszydbh4dpl3pl4utgqqw2v4jna.b32.I2P");
void ReadFromStream(AddrMan &addr, DataStream &ssPeers)
Only used by tests.
static constexpr int32_t ADDRMAN_MAX_FAILURES
How many successive failures are allowed ...
static constexpr auto ADDRMAN_MIN_FAIL
... in at least this duration
static constexpr int ADDRMAN_NEW_BUCKET_COUNT
static CService ResolveService(const std::string &ip, uint16_t port=0)
static int32_t GetCheckRatio(const NodeContext &node_ctx)
static auto AddrmanToStream(const AddrMan &addrman)
static const bool DETERMINISTIC
BOOST_AUTO_TEST_CASE(addrman_simple)
static auto MakeCorruptPeersDat()
static auto EMPTY_NETGROUPMAN
static CNetAddr ResolveIP(const std::string &ip)
const CChainParams & Params()
Return the currently selected parameters.
Extended statistics about a CAddress.
int GetNewBucket(const uint256 &nKey, const CNetAddr &src, const NetGroupManager &netgroupman) const
Calculate in which "new" bucket this entry belongs, given a certain source.
int GetTriedBucket(const uint256 &nKey, const NetGroupManager &netgroupman) const
Calculate in which "tried" bucket this entry belongs.
Stochastic address manager.
int64_t GetIntArg(const std::string &strArg, int64_t nDefault) const EXCLUSIVE_LOCKS_REQUIRED(!cs_args)
A CService with information about it as peer.
NodeSeconds nTime
Always included in serialization. The behavior is unspecified if the value is not representable as ui...
static constexpr SerParams V1_DISK
const MessageStartChars & MessageStart() const
bool SetSpecial(std::string_view addr)
Parse a Tor or I2P address and set this object to it.
A combination of a network address (CNetAddr) and a (TCP) port.
std::vector< unsigned char > GetKey() const
Double ended buffer combining vector and stream-like interfaces.
A writer stream (for serialization) that computes a 256-bit hash.
static NetGroupManager NoAsmap()
static NetGroupManager WithEmbeddedAsmap(std::span< const std::byte > asmap)
Helper to initialize the global NodeClock, let a duration elapse, and reset it after use in a test.
BOOST_FIXTURE_TEST_SUITE(cuckoocache_tests, BasicTestingSetup)
Test Suite for CuckooCache.
BOOST_AUTO_TEST_SUITE_END()
static CService ip(uint32_t i)
static const std::string addr1
static const std::string addr2
std::string ToString(const T &t)
Locale-independent version of std::to_string.
@ NET_ONION
TOR (v2 or v3)
std::vector< CNetAddr > LookupHost(const std::string &name, unsigned int nMaxSolutions, bool fAllowLookup, DNSLookupFn dns_lookup_function)
Resolve a host string to its corresponding network addresses.
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.
#define BOOST_CHECK_THROW(stmt, excMatch)
#define BOOST_CHECK_EQUAL(v1, v2)
#define BOOST_CHECK(expr)
Location information for an address in AddrMan.
NodeContext struct containing references to chain state and connection state.
void SetMockTime(int64_t nMockTimeIn)
DEPRECATED Use SetMockTime with chrono type.
std::chrono::time_point< NodeClock, std::chrono::seconds > NodeSeconds