mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-01-28 03:57:31 -03:00
4ee78450ff
ea4c9fd4ab
test: Cover eviction by timeout in addrman_evictionworks (Martin Zumsande)4f1bb467b5
test: Add test for multiplicity in addrman new tables (Martin Zumsande)e880bb7836
test: Add test for updating addrman entries (Martin Zumsande)f02eee8c87
test: introduce utility function to retrieve an addrman (Martin Zumsande)f0e5efb824
test: Remove unused AddrManTest class (Martin Zumsande)b696d7870b
test: Remove tests for internal helper functions (Martin Zumsande)0538520091
test: use AddrMan instead of AddrManTest where possible (Martin Zumsande)1c65d427bb
test: Inline SimConnFail function (Martin Zumsande)5b7aac34f2
test: delete unused GetBucketAndEntry function (Amiti Uttarwar)2ba1e74e59
test: Update addrman_serialization unit test to use AddrMan's interface (Amiti Uttarwar)dad5f76021
addrman: Introduce a test-only function to lookup addresses (Amiti Uttarwar) Pull request description: This PR (joint work with Amiti Uttarwar) changes the addrman unit tests such that they only use the public `AddrMan` interface: This has the advantage that the tests are less implementation-dependent, i.e. it would be possible to rewrite the internal addrman implementation (as drafted [here](https://github.com/sipa/bitcoin/tree/202106_multiindex_addrman) for using a multiindex) without having to adjust the tests. This includes the following steps: * Adding a test-only function `FindAddressEntry()` to the public addrman interface which returns info about an address in addrman (e.g. bucket, position, whethe the address is in new or tried). Obviously we want to do this sparingly, but I think a single test-only function is ok (which could also be useful elsewhere, e.g. in fuzz tests). * Removal of the `AddrManTest` subclass which would reach into AddrMan's internals, using `AddrMan` instead * Removal of tests for internal helper functions that are not publicly exposed (these are still tested indirectly via the public functions calling them). * Additional tests for previously untested features such as multiplicity in the new tables, that can be tested with the help of `FindAddressEntry()`. All in all, this PR increases the unit test coverage of AddrMan by a bit. ACKs for top commit: jnewbery: ACKea4c9fd4ab
josibake: reACKea4c9fd4ab
Tree-SHA512: c2d4ec8bdc62ffd6055ddcd37dea85ec08c76889e9e417e8d7c62a96cf68a8bcbe8c67bec3344d91fa7d3c499f6d9f810962da1dddd38e70966186b10b8ab447
181 lines
8 KiB
C++
181 lines
8 KiB
C++
// Copyright (c) 2012 Pieter Wuille
|
|
// Copyright (c) 2012-2021 The Bitcoin Core developers
|
|
// Distributed under the MIT software license, see the accompanying
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
#ifndef BITCOIN_ADDRMAN_H
|
|
#define BITCOIN_ADDRMAN_H
|
|
|
|
#include <netaddress.h>
|
|
#include <protocol.h>
|
|
#include <streams.h>
|
|
#include <timedata.h>
|
|
|
|
#include <cstdint>
|
|
#include <memory>
|
|
#include <optional>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
class AddrManImpl;
|
|
|
|
/** Default for -checkaddrman */
|
|
static constexpr int32_t DEFAULT_ADDRMAN_CONSISTENCY_CHECKS{0};
|
|
|
|
/** Test-only struct, capturing info about an address in AddrMan */
|
|
struct AddressPosition {
|
|
// Whether the address is in the new or tried table
|
|
const bool tried;
|
|
|
|
// Addresses in the tried table should always have a multiplicity of 1.
|
|
// Addresses in the new table can have multiplicity between 1 and
|
|
// ADDRMAN_NEW_BUCKETS_PER_ADDRESS
|
|
const int multiplicity;
|
|
|
|
// If the address is in the new table, the bucket and position are
|
|
// populated based on the first source who sent the address.
|
|
// In certain edge cases, this may not be where the address is currently
|
|
// located.
|
|
const int bucket;
|
|
const int position;
|
|
|
|
bool operator==(AddressPosition other) {
|
|
return std::tie(tried, multiplicity, bucket, position) ==
|
|
std::tie(other.tried, other.multiplicity, other.bucket, other.position);
|
|
}
|
|
explicit AddressPosition(bool tried_in, int multiplicity_in, int bucket_in, int position_in)
|
|
: tried{tried_in}, multiplicity{multiplicity_in}, bucket{bucket_in}, position{position_in} {}
|
|
};
|
|
|
|
/** Stochastic address manager
|
|
*
|
|
* Design goals:
|
|
* * Keep the address tables in-memory, and asynchronously dump the entire table to peers.dat.
|
|
* * Make sure no (localized) attacker can fill the entire table with his nodes/addresses.
|
|
*
|
|
* To that end:
|
|
* * Addresses are organized into buckets that can each store up to 64 entries.
|
|
* * Addresses to which our node has not successfully connected go into 1024 "new" buckets.
|
|
* * Based on the address range (/16 for IPv4) of the source of information, or if an asmap is provided,
|
|
* the AS it belongs to (for IPv4/IPv6), 64 buckets are selected at random.
|
|
* * The actual bucket is chosen from one of these, based on the range in which the address itself is located.
|
|
* * The position in the bucket is chosen based on the full address.
|
|
* * One single address can occur in up to 8 different buckets to increase selection chances for addresses that
|
|
* are seen frequently. The chance for increasing this multiplicity decreases exponentially.
|
|
* * When adding a new address to an occupied position of a bucket, it will not replace the existing entry
|
|
* unless that address is also stored in another bucket or it doesn't meet one of several quality criteria
|
|
* (see IsTerrible for exact criteria).
|
|
* * Addresses of nodes that are known to be accessible go into 256 "tried" buckets.
|
|
* * Each address range selects at random 8 of these buckets.
|
|
* * The actual bucket is chosen from one of these, based on the full address.
|
|
* * When adding a new good address to an occupied position of a bucket, a FEELER connection to the
|
|
* old address is attempted. The old entry is only replaced and moved back to the "new" buckets if this
|
|
* attempt was unsuccessful.
|
|
* * Bucket selection is based on cryptographic hashing, using a randomly-generated 256-bit key, which should not
|
|
* be observable by adversaries.
|
|
* * Several indexes are kept for high performance. Setting m_consistency_check_ratio with the -checkaddrman
|
|
* configuration option will introduce (expensive) consistency checks for the entire data structure.
|
|
*/
|
|
class AddrMan
|
|
{
|
|
protected:
|
|
const std::unique_ptr<AddrManImpl> m_impl;
|
|
|
|
public:
|
|
explicit AddrMan(std::vector<bool> asmap, bool deterministic, int32_t consistency_check_ratio);
|
|
|
|
~AddrMan();
|
|
|
|
template <typename Stream>
|
|
void Serialize(Stream& s_) const;
|
|
|
|
template <typename Stream>
|
|
void Unserialize(Stream& s_);
|
|
|
|
//! Return the number of (unique) addresses in all tables.
|
|
size_t size() const;
|
|
|
|
/**
|
|
* Attempt to add one or more addresses to addrman's new table.
|
|
*
|
|
* @param[in] vAddr Address records to attempt to add.
|
|
* @param[in] source The address of the node that sent us these addr records.
|
|
* @param[in] nTimePenalty A "time penalty" to apply to the address record's nTime. If a peer
|
|
* sends us an address record with nTime=n, then we'll add it to our
|
|
* addrman with nTime=(n - nTimePenalty).
|
|
* @return true if at least one address is successfully added. */
|
|
bool Add(const std::vector<CAddress>& vAddr, const CNetAddr& source, int64_t nTimePenalty = 0);
|
|
|
|
/**
|
|
* Mark an address record as accessible and attempt to move it to addrman's tried table.
|
|
*
|
|
* @param[in] addr Address record to attempt to move to tried table.
|
|
* @param[in] nTime The time that we were last connected to this peer.
|
|
* @return true if the address is successfully moved from the new table to the tried table.
|
|
*/
|
|
bool Good(const CService& addr, int64_t nTime = GetAdjustedTime());
|
|
|
|
//! Mark an entry as connection attempted to.
|
|
void Attempt(const CService& addr, bool fCountFailure, int64_t nTime = GetAdjustedTime());
|
|
|
|
//! See if any to-be-evicted tried table entries have been tested and if so resolve the collisions.
|
|
void ResolveCollisions();
|
|
|
|
/**
|
|
* Randomly select an address in the tried table that another address is
|
|
* attempting to evict.
|
|
*
|
|
* @return CAddress The record for the selected tried peer.
|
|
* int64_t The last time we attempted to connect to that peer.
|
|
*/
|
|
std::pair<CAddress, int64_t> SelectTriedCollision();
|
|
|
|
/**
|
|
* Choose an address to connect to.
|
|
*
|
|
* @param[in] newOnly Whether to only select addresses from the new table.
|
|
* @return CAddress The record for the selected peer.
|
|
* int64_t The last time we attempted to connect to that peer.
|
|
*/
|
|
std::pair<CAddress, int64_t> Select(bool newOnly = false) const;
|
|
|
|
/**
|
|
* Return all or many randomly selected addresses, optionally by network.
|
|
*
|
|
* @param[in] max_addresses Maximum number of addresses to return (0 = all).
|
|
* @param[in] max_pct Maximum percentage of addresses to return (0 = all).
|
|
* @param[in] network Select only addresses of this network (nullopt = all).
|
|
*
|
|
* @return A vector of randomly selected addresses from vRandom.
|
|
*/
|
|
std::vector<CAddress> GetAddr(size_t max_addresses, size_t max_pct, std::optional<Network> network) const;
|
|
|
|
/** We have successfully connected to this peer. Calling this function
|
|
* updates the CAddress's nTime, which is used in our IsTerrible()
|
|
* decisions and gossiped to peers. Callers should be careful that updating
|
|
* this information doesn't leak topology information to network spies.
|
|
*
|
|
* net_processing calls this function when it *disconnects* from a peer to
|
|
* not leak information about currently connected peers.
|
|
*
|
|
* @param[in] addr The address of the peer we were connected to
|
|
* @param[in] nTime The time that we were last connected to this peer
|
|
*/
|
|
void Connected(const CService& addr, int64_t nTime = GetAdjustedTime());
|
|
|
|
//! Update an entry's service bits.
|
|
void SetServices(const CService& addr, ServiceFlags nServices);
|
|
|
|
const std::vector<bool>& GetAsmap() const;
|
|
|
|
/** Test-only function
|
|
* Find the address record in AddrMan and return information about its
|
|
* position.
|
|
* @param[in] addr The address record to look up.
|
|
* @return Information about the address record in AddrMan
|
|
* or nullopt if address is not found.
|
|
*/
|
|
std::optional<AddressPosition> FindAddressEntry(const CAddress& addr);
|
|
};
|
|
|
|
#endif // BITCOIN_ADDRMAN_H
|