8#include <chainparams.h>
13#include <test/data/asmap.raw.h>
18#include <boost/test/unit_test.hpp>
24using namespace std::literals;
33 return std::clamp<int32_t>(node_ctx.
args->
GetIntArg(
"-checkaddrman", 100), 0, 1000000);
38 const std::optional<CNetAddr> addr{
LookupHost(
ip,
false)};
39 BOOST_CHECK_MESSAGE(addr.has_value(),
strprintf(
"failed to resolve: %s",
ip));
45 const std::optional<CService> serv{
Lookup(
ip, port,
false)};
46 BOOST_CHECK_MESSAGE(serv.has_value(),
strprintf(
"failed to resolve: %s:%i",
ip, port));
60 auto addr_null = addrman->Select().first;
67 auto addr_ret1 = addrman->Select().first;
89 std::vector<CAddress> vAddr;
101 const auto base_time{Now<NodeSeconds>() - 10000
s};
106 caddr1.
nTime = base_time;
108 const auto time_penalty{3600
s};
110 BOOST_CHECK(addrman->Add({caddr1}, source1, time_penalty));
112 auto addr_pos1{addrman->FindAddressEntry(caddr1)};
113 BOOST_REQUIRE(addr_pos1.has_value());
115 std::vector<CAddress> addresses{addrman->GetAddr(0, 0, std::nullopt)};
116 BOOST_REQUIRE_EQUAL(addresses.size(), 1U);
124 caddr2.
nTime = base_time;
126 BOOST_CHECK(addrman->Add({caddr2}, source2, time_penalty));
128 addresses = addrman->GetAddr(0, 0, std::nullopt);
129 BOOST_REQUIRE_EQUAL(addresses.size(), 2U);
131 CAddress retrieved_addr2{addresses[0]};
132 BOOST_CHECK(retrieved_addr2.nTime == base_time - time_penalty);
151 auto addr_ret2 = addrman->Select().first;
152 BOOST_CHECK(addr_ret2.ToStringAddrPort() ==
"250.1.1.1:8333" || addr_ret2.ToStringAddrPort() ==
"250.1.1.1:8334");
158 bool new_only =
true;
159 auto addr_ret3 = addrman->Select(new_only).first;
166 BOOST_CHECK(!addrman->Select(
false).first.IsValid());
167 BOOST_CHECK(!addrman->Select(
true).first.IsValid());
183 BOOST_CHECK(!addrman->Select(
true).first.IsValid());
204 BOOST_CHECK(addrman->Add({CAddress(addr5, NODE_NONE)}, addr3));
206 BOOST_CHECK(addrman->Add({CAddress(addr6, NODE_NONE)}, addr3));
215 std::set<uint16_t> ports;
216 for (
int i = 0; i < 20; ++i) {
217 ports.insert(addrman->Select().first.GetPort());
225 BOOST_CHECK(!addrman->Select(
true, {NET_IPV4}).first.IsValid());
226 BOOST_CHECK(!addrman->Select(
false, {NET_IPV4}).first.IsValid());
235 BOOST_CHECK(!addrman->Select(
false, {NET_IPV6}).first.IsValid());
236 BOOST_CHECK(!addrman->Select(
false, {NET_ONION}).first.IsValid());
237 BOOST_CHECK(!addrman->Select(
false, {NET_I2P}).first.IsValid());
238 BOOST_CHECK(!addrman->Select(
false, {NET_CJDNS}).first.IsValid());
239 BOOST_CHECK(!addrman->Select(
true, {NET_CJDNS}).first.IsValid());
244 i2p_addr.
SetSpecial(
"udhdrtrcetjm5sxzskjyr5ztpeszydbh4dpl3pl4utgqqw2v4jna.b32.i2p");
247 BOOST_CHECK(addrman->Select(
true, {NET_I2P}).first == i2p_addr);
248 BOOST_CHECK(addrman->Select(
false, {NET_I2P}).first == i2p_addr);
251 BOOST_CHECK(addrman->Select(
false, nets_with_entries).first.IsValid());
252 BOOST_CHECK(!addrman->Select(
false, {NET_IPV6}).first.IsValid());
253 BOOST_CHECK(!addrman->Select(
false, {NET_ONION}).first.IsValid());
254 BOOST_CHECK(!addrman->Select(
false, {NET_CJDNS}).first.IsValid());
256 BOOST_CHECK(!addrman->Select(
false, nets_without_entries).first.IsValid());
261 BOOST_CHECK(!addrman->Select(
true, {NET_I2P}).first.IsValid());
262 BOOST_CHECK(addrman->Select(
false, {NET_I2P}).first == i2p_addr);
266 i2p_addr2.
SetSpecial(
"c4gfnttsuwqomiygupdqqqyy5y5emnk5c73hrfvatri67prd7vyq.b32.i2p");
269 BOOST_CHECK(addrman->Select(
true, {NET_I2P}).first == i2p_addr2);
272 bool new_selected{
false};
273 bool tried_selected{
false};
276 while (--counter > 0 && (!new_selected || !tried_selected)) {
278 BOOST_REQUIRE(selected == i2p_addr || selected == i2p_addr2);
279 if (selected == i2p_addr) {
280 tried_selected =
true;
299 i2p_addr.
SetSpecial(
"udhdrtrcetjm5sxzskjyr5ztpeszydbh4dpl3pl4utgqqw2v4jna.b32.i2p");
319 uint32_t num_addrs{0};
323 while (num_addrs < 22) {
333 uint32_t collisions{1};
346 const auto start_time{Now<NodeSeconds>()};
347 addr.
nTime = start_time;
350 for (
unsigned int i = 1; i < 20; ++i) {
351 std::string addr_ip{
ToString(i % 256) +
"." +
ToString(i >> 8 % 256) +
".1.1"};
353 addrman->Add({addr},
source);
362 for (
unsigned int i = 1; i < 400; ++i) {
363 std::string addr_ip{
ToString(i % 256) +
"." +
ToString(i >> 8 % 256) +
".1.1"};
365 addr.nTime = start_time + std::chrono::seconds{i};
366 addrman->Add({addr},
source);
368 AddressPosition addr_pos_multi = addrman->FindAddressEntry(addr).value();
380 uint32_t num_addrs{0};
384 while (num_addrs < 35) {
412 std::vector<CAddress> vAddr1 = addrman->GetAddr(0, 0, std::nullopt);
416 addr1.nTime = Now<NodeSeconds>();
418 addr2.nTime = Now<NodeSeconds>();
420 addr3.
nTime = Now<NodeSeconds>();
422 addr4.
nTime = Now<NodeSeconds>();
424 addr5.
nTime = Now<NodeSeconds>();
429 BOOST_CHECK(addrman->Add({addr1, addr3, addr5}, source1));
430 BOOST_CHECK(addrman->Add({addr2, addr4}, source2));
443 for (
unsigned int i = 1; i < (8 * 256); i++) {
444 int octet1 = i % 256;
445 int octet2 = i >> 8 % 256;
450 addr.
nTime = Now<NodeSeconds>();
451 addrman->Add({addr},
ResolveIP(strAddr));
455 std::vector<CAddress> vAddr = addrman->GetAddr(2500, 23, std::nullopt);
457 size_t percent23 = (addrman->Size() * 23) / 100;
470 addr1.nTime = Now<NodeSeconds>();
479 addr3.
nTime = Now<NodeSeconds>();
480 addrman->Good(addr3, Now<NodeSeconds>());
484 for (
size_t i = 0; i < 3; ++i) {
485 addrman->Attempt(addr3,
true, Now<NodeSeconds>() - 61
s);
491 addr4.
nTime = Now<NodeSeconds>() + 11min;
526 std::set<int> buckets;
527 for (
int i = 0; i < 255; i++) {
532 buckets.insert(bucket);
539 for (
int j = 0; j < 255; j++) {
544 buckets.insert(bucket);
576 std::set<int> buckets;
577 for (
int i = 0; i < 255; i++) {
582 buckets.insert(bucket);
589 for (
int j = 0; j < 4 * 255; j++) {
595 buckets.insert(bucket);
602 for (
int p = 0; p < 255; p++) {
607 buckets.insert(bucket);
653 std::set<int> buckets;
654 for (
int j = 0; j < 255; j++) {
659 buckets.insert(bucket);
666 for (
int j = 0; j < 255; j++) {
671 buckets.insert(bucket);
705 std::set<int> buckets;
706 for (
int i = 0; i < 255; i++) {
711 buckets.insert(bucket);
718 for (
int j = 0; j < 4 * 255; j++) {
724 buckets.insert(bucket);
731 for (
int p = 0; p < 255; p++) {
736 buckets.insert(bucket);
743 for (
int p = 0; p < 255; p++) {
748 buckets.insert(bucket);
760 auto addrman_asmap1 = std::make_unique<AddrMan>(netgroupman,
DETERMINISTIC, ratio);
761 auto addrman_asmap1_dup = std::make_unique<AddrMan>(netgroupman,
DETERMINISTIC, ratio);
769 addrman_asmap1->Add({addr}, default_source);
771 stream << *addrman_asmap1;
773 stream >> *addrman_asmap1_dup;
775 AddressPosition addr_pos1 = addrman_asmap1->FindAddressEntry(addr).value();
776 AddressPosition addr_pos2 = addrman_asmap1_dup->FindAddressEntry(addr).value();
783 stream << *addrman_asmap1;
784 stream >> *addrman_noasmap;
785 AddressPosition addr_pos3 = addrman_noasmap->FindAddressEntry(addr).value();
791 addrman_asmap1 = std::make_unique<AddrMan>(netgroupman,
DETERMINISTIC, ratio);
793 addrman_noasmap->Add({addr}, default_source);
794 stream << *addrman_noasmap;
795 stream >> *addrman_asmap1;
797 AddressPosition addr_pos4 = addrman_asmap1->FindAddressEntry(addr).value();
803 addrman_asmap1 = std::make_unique<AddrMan>(netgroupman,
DETERMINISTIC, ratio);
807 addrman_noasmap->Add({addr,
addr2}, default_source);
811 stream << *addrman_noasmap;
812 stream >> *addrman_asmap1;
831 addrman->Add({new1, tried1, new2, tried2},
CNetAddr{});
832 addrman->Good(tried1);
833 addrman->Good(tried2);
834 BOOST_REQUIRE_EQUAL(addrman->Size(), 4);
838 const std::string str{stream.str()};
841 const char new2_raw[]{6, 6, 6, 6};
842 const uint8_t new2_raw_replacement[]{0, 0, 0, 0};
843 pos = str.find(new2_raw, 0,
sizeof(new2_raw));
844 BOOST_REQUIRE(pos != std::string::npos);
845 BOOST_REQUIRE(pos +
sizeof(new2_raw_replacement) <= stream.size());
846 memcpy(stream.data() + pos, new2_raw_replacement,
sizeof(new2_raw_replacement));
848 const char tried2_raw[]{8, 8, 8, 8};
849 const uint8_t tried2_raw_replacement[]{255, 255, 255, 255};
850 pos = str.find(tried2_raw, 0,
sizeof(tried2_raw));
851 BOOST_REQUIRE(pos != std::string::npos);
852 BOOST_REQUIRE(pos +
sizeof(tried2_raw_replacement) <= stream.size());
853 memcpy(stream.data() + pos, tried2_raw_replacement,
sizeof(tried2_raw_replacement));
867 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
871 for (
unsigned int i = 1; i < 23; i++) {
877 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
882 for (
unsigned int i = 1; i < 23; i++) {
889 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
899 for (
unsigned int i = 1; i < 36; i++) {
911 BOOST_CHECK_EQUAL(addrman->SelectTriedCollision().first.ToStringAddrPort(),
"250.1.1.19:0");
916 addrman->ResolveCollisions();
917 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
920 for (
unsigned int i = 37; i < 59; i++) {
931 BOOST_CHECK_EQUAL(addrman->SelectTriedCollision().first.ToStringAddrPort(),
"250.1.1.10:0");
938 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() !=
"[::]:0");
941 addrman->ResolveCollisions();
942 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
952 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
956 for (
unsigned int i = 1; i < 36; i++) {
969 auto info = addrman->SelectTriedCollision().first;
975 addrman->Attempt(info,
false, Now<NodeSeconds>() - 61
s);
978 addrman->ResolveCollisions();
979 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
988 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
994 BOOST_CHECK_EQUAL(addrman->SelectTriedCollision().first.ToStringAddrPort(),
"250.1.1.36:0");
998 addrman->ResolveCollisions();
999 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
1011 ssPeersIn << addrman;
1019 std::optional<CService>
addr1,
addr2, addr3, addr4;
1024 addr3 =
Lookup(
"250.7.3.3", 9999,
false);
1026 addr3 =
Lookup(
"250.7.3.3"s, 9999,
false);
1028 addr4 =
Lookup(
"250.7.3.3\0example.com"s, 9999,
false);
1032 const std::optional<CService>
source{
Lookup(
"252.5.1.1", 8333,
false)};
1040 bool exceptionThrown =
false;
1045 unsigned char pchMsgTmp[4];
1046 ssPeers1 >> pchMsgTmp;
1047 ssPeers1 >> addrman1;
1048 }
catch (
const std::exception&) {
1049 exceptionThrown =
true;
1070 unsigned char nVersion = 1;
1072 s << ((
unsigned char)32);
1080 const std::optional<CService> serv{
Lookup(
"252.1.1.1", 7777,
false)};
1081 BOOST_REQUIRE(serv.has_value());
1083 std::optional<CNetAddr> resolved{
LookupHost(
"252.2.2.2",
false)};
1084 BOOST_REQUIRE(resolved.has_value());
1095 bool exceptionThrown =
false;
1099 unsigned char pchMsgTmp[4];
1100 ssPeers1 >> pchMsgTmp;
1101 ssPeers1 >> addrman1;
1102 }
catch (
const std::exception&) {
1103 exceptionThrown =
true;
1122 const auto start_time{Now<NodeSeconds>() - 10000
s};
1123 addr.nTime = start_time;
1129 addr_diff_port.
nTime = start_time;
1130 addrman->Connected(addr_diff_port);
1132 std::vector<CAddress> vAddr1{addrman->GetAddr(0, 0, std::nullopt)};
1138 addrman->Connected(addr);
1140 std::vector<CAddress> vAddr2 = addrman->GetAddr(0, 0, std::nullopt);
1142 BOOST_CHECK(vAddr2.at(0).nTime >= start_time + 10000
s);
1147 addr_v2.
nTime = start_time;
1149 std::vector<CAddress> vAddr3{addrman->GetAddr(0, 0, std::nullopt)};
1155 std::vector<CAddress> vAddr4{addrman->GetAddr(0, 0, std::nullopt)};
1161 std::vector<CAddress> vAddr5{addrman->GetAddr(0, 0, std::nullopt)};
1167 std::vector<CAddress> vAddr6{addrman->GetAddr(0, 0, std::nullopt)};
1199 i2p_addr.
SetSpecial(
"UDHDrtrcetjm5sxzskjyr5ztpeszydbh4dpl3pl4utgqqw2v4jna.b32.I2P");
void ReadFromStream(AddrMan &addr, DataStream &ssPeers)
Only used by tests.
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
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)
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.
int64_t GetTime()
DEPRECATED Use either ClockType::now() or Now<TimePointType>() if a cast is needed.
void SetMockTime(int64_t nMockTimeIn)
DEPRECATED Use SetMockTime with chrono type.
std::chrono::time_point< NodeClock, std::chrono::seconds > NodeSeconds