8 #include <chainparams.h>
13 #include <test/data/asmap.raw.h>
18 #include <boost/test/unit_test.hpp>
23 using namespace std::literals;
31 return std::clamp<int32_t>(node_ctx.
args->
GetIntArg(
"-checkaddrman", 100), 0, 1000000);
36 const std::optional<CNetAddr> addr{
LookupHost(
ip,
false)};
37 BOOST_CHECK_MESSAGE(addr.has_value(),
strprintf(
"failed to resolve: %s",
ip));
43 const std::optional<CService> serv{
Lookup(
ip, port,
false)};
44 BOOST_CHECK_MESSAGE(serv.has_value(),
strprintf(
"failed to resolve: %s:%i",
ip, port));
51 std::vector<bool> result(vector_size);
52 for (
int byte_i = 0; byte_i < vector_size / 8; ++byte_i) {
53 unsigned char cur_byte =
source[byte_i];
54 for (
int bit_i = 0; bit_i < 8; ++bit_i) {
55 result[byte_i * 8 + bit_i] = (cur_byte >> bit_i) & 1;
71 auto addr_null = addrman->Select().first;
78 auto addr_ret1 = addrman->Select().first;
100 std::vector<CAddress> vAddr;
123 auto addr_ret2 = addrman->Select().first;
124 BOOST_CHECK(addr_ret2.ToStringAddrPort() ==
"250.1.1.1:8333" || addr_ret2.ToStringAddrPort() ==
"250.1.1.1:8334");
130 bool new_only =
true;
131 auto addr_ret3 = addrman->Select(new_only).first;
138 BOOST_CHECK(!addrman->Select(
false).first.IsValid());
139 BOOST_CHECK(!addrman->Select(
true).first.IsValid());
155 BOOST_CHECK(!addrman->Select(
true).first.IsValid());
176 BOOST_CHECK(addrman->Add({CAddress(addr5, NODE_NONE)}, addr3));
178 BOOST_CHECK(addrman->Add({CAddress(addr6, NODE_NONE)}, addr3));
187 std::set<uint16_t> ports;
188 for (
int i = 0; i < 20; ++i) {
189 ports.insert(addrman->Select().first.GetPort());
216 i2p_addr.
SetSpecial(
"udhdrtrcetjm5sxzskjyr5ztpeszydbh4dpl3pl4utgqqw2v4jna.b32.i2p");
234 i2p_addr2.
SetSpecial(
"c4gfnttsuwqomiygupdqqqyy5y5emnk5c73hrfvatri67prd7vyq.b32.i2p");
240 bool new_selected{
false};
241 bool tried_selected{
false};
243 while (!new_selected || !tried_selected) {
245 BOOST_REQUIRE(selected == i2p_addr || selected == i2p_addr2);
246 if (selected == i2p_addr) {
247 tried_selected =
true;
263 i2p_addr.
SetSpecial(
"udhdrtrcetjm5sxzskjyr5ztpeszydbh4dpl3pl4utgqqw2v4jna.b32.i2p");
283 uint32_t num_addrs{0};
287 while (num_addrs < 22) {
297 uint32_t collisions{1};
310 const auto start_time{Now<NodeSeconds>()};
311 addr.
nTime = start_time;
314 for (
unsigned int i = 1; i < 20; ++i) {
315 std::string addr_ip{
ToString(i % 256) +
"." +
ToString(i >> 8 % 256) +
".1.1"};
317 addrman->Add({addr},
source);
326 for (
unsigned int i = 1; i < 400; ++i) {
327 std::string addr_ip{
ToString(i % 256) +
"." +
ToString(i >> 8 % 256) +
".1.1"};
329 addr.nTime = start_time + std::chrono::seconds{i};
330 addrman->Add({addr},
source);
332 AddressPosition addr_pos_multi = addrman->FindAddressEntry(addr).value();
344 uint32_t num_addrs{0};
348 while (num_addrs < 35) {
376 std::vector<CAddress> vAddr1 = addrman->GetAddr(0, 0, std::nullopt);
380 addr1.nTime = Now<NodeSeconds>();
382 addr2.nTime = Now<NodeSeconds>();
384 addr3.
nTime = Now<NodeSeconds>();
386 addr4.
nTime = Now<NodeSeconds>();
388 addr5.
nTime = Now<NodeSeconds>();
393 BOOST_CHECK(addrman->Add({addr1, addr3, addr5}, source1));
394 BOOST_CHECK(addrman->Add({addr2, addr4}, source2));
407 for (
unsigned int i = 1; i < (8 * 256); i++) {
408 int octet1 = i % 256;
409 int octet2 = i >> 8 % 256;
414 addr.
nTime = Now<NodeSeconds>();
415 addrman->Add({addr},
ResolveIP(strAddr));
419 std::vector<CAddress> vAddr = addrman->GetAddr(2500, 23, std::nullopt);
421 size_t percent23 = (addrman->Size() * 23) / 100;
455 std::set<int> buckets;
456 for (
int i = 0; i < 255; i++) {
461 buckets.insert(bucket);
468 for (
int j = 0; j < 255; j++) {
473 buckets.insert(bucket);
505 std::set<int> buckets;
506 for (
int i = 0; i < 255; i++) {
511 buckets.insert(bucket);
518 for (
int j = 0; j < 4 * 255; j++) {
524 buckets.insert(bucket);
531 for (
int p = 0; p < 255; p++) {
536 buckets.insert(bucket);
556 std::vector<bool> asmap =
FromBytes(asmap_raw,
sizeof(asmap_raw) * 8);
583 std::set<int> buckets;
584 for (
int j = 0; j < 255; j++) {
589 buckets.insert(bucket);
596 for (
int j = 0; j < 255; j++) {
601 buckets.insert(bucket);
610 std::vector<bool> asmap =
FromBytes(asmap_raw,
sizeof(asmap_raw) * 8);
636 std::set<int> buckets;
637 for (
int i = 0; i < 255; i++) {
642 buckets.insert(bucket);
649 for (
int j = 0; j < 4 * 255; j++) {
655 buckets.insert(bucket);
662 for (
int p = 0; p < 255; p++) {
667 buckets.insert(bucket);
674 for (
int p = 0; p < 255; p++) {
679 buckets.insert(bucket);
688 std::vector<bool> asmap1 =
FromBytes(asmap_raw,
sizeof(asmap_raw) * 8);
692 auto addrman_asmap1 = std::make_unique<AddrMan>(netgroupman,
DETERMINISTIC, ratio);
693 auto addrman_asmap1_dup = std::make_unique<AddrMan>(netgroupman,
DETERMINISTIC, ratio);
701 addrman_asmap1->Add({addr}, default_source);
703 stream << *addrman_asmap1;
705 stream >> *addrman_asmap1_dup;
707 AddressPosition addr_pos1 = addrman_asmap1->FindAddressEntry(addr).value();
708 AddressPosition addr_pos2 = addrman_asmap1_dup->FindAddressEntry(addr).value();
715 stream << *addrman_asmap1;
716 stream >> *addrman_noasmap;
717 AddressPosition addr_pos3 = addrman_noasmap->FindAddressEntry(addr).value();
723 addrman_asmap1 = std::make_unique<AddrMan>(netgroupman,
DETERMINISTIC, ratio);
725 addrman_noasmap->Add({addr}, default_source);
726 stream << *addrman_noasmap;
727 stream >> *addrman_asmap1;
729 AddressPosition addr_pos4 = addrman_asmap1->FindAddressEntry(addr).value();
735 addrman_asmap1 = std::make_unique<AddrMan>(netgroupman,
DETERMINISTIC, ratio);
739 addrman_noasmap->Add({addr,
addr2}, default_source);
743 stream << *addrman_noasmap;
744 stream >> *addrman_asmap1;
763 addrman->Add({new1, tried1, new2, tried2},
CNetAddr{});
764 addrman->Good(tried1);
765 addrman->Good(tried2);
766 BOOST_REQUIRE_EQUAL(addrman->Size(), 4);
770 const std::string str{stream.
str()};
773 const char new2_raw[]{6, 6, 6, 6};
774 const uint8_t new2_raw_replacement[]{0, 0, 0, 0};
775 pos = str.find(new2_raw, 0,
sizeof(new2_raw));
776 BOOST_REQUIRE(pos != std::string::npos);
777 BOOST_REQUIRE(pos +
sizeof(new2_raw_replacement) <= stream.
size());
778 memcpy(stream.
data() + pos, new2_raw_replacement,
sizeof(new2_raw_replacement));
780 const char tried2_raw[]{8, 8, 8, 8};
781 const uint8_t tried2_raw_replacement[]{255, 255, 255, 255};
782 pos = str.find(tried2_raw, 0,
sizeof(tried2_raw));
783 BOOST_REQUIRE(pos != std::string::npos);
784 BOOST_REQUIRE(pos +
sizeof(tried2_raw_replacement) <= stream.
size());
785 memcpy(stream.
data() + pos, tried2_raw_replacement,
sizeof(tried2_raw_replacement));
799 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
803 for (
unsigned int i = 1; i < 23; i++) {
809 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
814 for (
unsigned int i = 1; i < 23; i++) {
821 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
831 for (
unsigned int i = 1; i < 36; i++) {
843 BOOST_CHECK_EQUAL(addrman->SelectTriedCollision().first.ToStringAddrPort(),
"250.1.1.19:0");
848 addrman->ResolveCollisions();
849 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
852 for (
unsigned int i = 37; i < 59; i++) {
863 BOOST_CHECK_EQUAL(addrman->SelectTriedCollision().first.ToStringAddrPort(),
"250.1.1.10:0");
870 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() !=
"[::]:0");
873 addrman->ResolveCollisions();
874 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
884 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
888 for (
unsigned int i = 1; i < 36; i++) {
901 auto info = addrman->SelectTriedCollision().first;
907 addrman->Attempt(info,
false, Now<NodeSeconds>() - 61s);
910 addrman->ResolveCollisions();
911 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
920 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
926 BOOST_CHECK_EQUAL(addrman->SelectTriedCollision().first.ToStringAddrPort(),
"250.1.1.36:0");
930 addrman->ResolveCollisions();
931 BOOST_CHECK(addrman->SelectTriedCollision().first.ToStringAddrPort() ==
"[::]:0");
943 ssPeersIn << addrman;
951 std::optional<CService>
addr1,
addr2, addr3, addr4;
956 addr3 =
Lookup(
"250.7.3.3", 9999,
false);
958 addr3 =
Lookup(
"250.7.3.3"s, 9999,
false);
960 addr4 =
Lookup(
"250.7.3.3\0example.com"s, 9999,
false);
964 const std::optional<CService>
source{
Lookup(
"252.5.1.1", 8333,
false)};
972 bool exceptionThrown =
false;
977 unsigned char pchMsgTmp[4];
978 ssPeers1 >> pchMsgTmp;
979 ssPeers1 >> addrman1;
980 }
catch (
const std::exception&) {
981 exceptionThrown =
true;
1002 unsigned char nVersion = 1;
1004 s << ((
unsigned char)32);
1012 const std::optional<CService> serv{
Lookup(
"252.1.1.1", 7777,
false)};
1013 BOOST_REQUIRE(serv.has_value());
1015 std::optional<CNetAddr> resolved{
LookupHost(
"252.2.2.2",
false)};
1016 BOOST_REQUIRE(resolved.has_value());
1027 bool exceptionThrown =
false;
1031 unsigned char pchMsgTmp[4];
1032 ssPeers1 >> pchMsgTmp;
1033 ssPeers1 >> addrman1;
1034 }
catch (
const std::exception&) {
1035 exceptionThrown =
true;
1054 const auto start_time{Now<NodeSeconds>() - 10000s};
1055 addr.
nTime = start_time;
1061 addr_diff_port.
nTime = start_time;
1062 addrman->Connected(addr_diff_port);
1064 std::vector<CAddress> vAddr1{addrman->GetAddr(0, 0, std::nullopt)};
1070 addrman->Connected(addr);
1072 std::vector<CAddress> vAddr2 = addrman->GetAddr(0, 0, std::nullopt);
1074 BOOST_CHECK(vAddr2.at(0).nTime >= start_time + 10000s);
1105 i2p_addr.
SetSpecial(
"UDHDrtrcetjm5sxzskjyr5ztpeszydbh4dpl3pl4utgqqw2v4jna.b32.I2P");
void ReadFromStream(AddrMan &addr, CDataStream &ssPeers)
Only used by tests.
static constexpr int ADDRMAN_NEW_BUCKET_COUNT
static NetGroupManager EMPTY_NETGROUPMAN
static CService ResolveService(const std::string &ip, uint16_t port=0)
static int32_t GetCheckRatio(const NodeContext &node_ctx)
static CDataStream AddrmanToStream(const AddrMan &addrman)
static const bool DETERMINISTIC
BOOST_AUTO_TEST_CASE(addrman_simple)
static std::vector< bool > FromBytes(const unsigned char *source, int vector_size)
static CDataStream MakeCorruptPeersDat()
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
Return integer argument or default value.
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...
const CMessageHeader::MessageStartChars & MessageStart() const
bool SetSpecial(const std::string &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
static const int CLIENT_VERSION
bitcoind-res.rc includes this file, but it cannot cope with real c++ code.
BOOST_AUTO_TEST_SUITE_END()
static CService ip(uint32_t i)
static const std::string addr1
static const std::string addr2
@ NET_ONION
TOR (v2 or v3)
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.
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.
#define BOOST_CHECK_THROW(stmt, excMatch)
#define BOOST_CHECK_EQUAL(v1, v2)
#define BOOST_CHECK(expr)
std::string ToString(const T &t)
Locale-independent version of std::to_string.
Test-only struct, capturing info about 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
static const int PROTOCOL_VERSION
network protocol versioning