mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-02-25 12:51:55 -05:00

that are otherwise private: - CConnman::m_nodes - CConnman::ConnectNodes() - CConnman::AlreadyConnectedToAddress() and update the #include headers per iwyu.
234 lines
6.6 KiB
C++
234 lines
6.6 KiB
C++
// Copyright (c) 2020-2022 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_TEST_UTIL_NET_H
|
|
#define BITCOIN_TEST_UTIL_NET_H
|
|
|
|
#include <compat/compat.h>
|
|
#include <net.h>
|
|
#include <net_permissions.h>
|
|
#include <net_processing.h>
|
|
#include <netaddress.h>
|
|
#include <node/connection_types.h>
|
|
#include <node/eviction.h>
|
|
#include <sync.h>
|
|
#include <util/sock.h>
|
|
|
|
#include <algorithm>
|
|
#include <array>
|
|
#include <cassert>
|
|
#include <chrono>
|
|
#include <cstdint>
|
|
#include <cstring>
|
|
#include <memory>
|
|
#include <string>
|
|
#include <unordered_map>
|
|
#include <vector>
|
|
|
|
class FastRandomContext;
|
|
|
|
template <typename C>
|
|
class Span;
|
|
|
|
struct ConnmanTestMsg : public CConnman {
|
|
using CConnman::CConnman;
|
|
|
|
void SetPeerConnectTimeout(std::chrono::seconds timeout)
|
|
{
|
|
m_peer_connect_timeout = timeout;
|
|
}
|
|
|
|
std::vector<CNode*> TestNodes()
|
|
{
|
|
LOCK(m_nodes_mutex);
|
|
return m_nodes;
|
|
}
|
|
|
|
void AddTestNode(CNode& node)
|
|
{
|
|
LOCK(m_nodes_mutex);
|
|
m_nodes.push_back(&node);
|
|
|
|
if (node.IsManualOrFullOutboundConn()) ++m_network_conn_counts[node.addr.GetNetwork()];
|
|
}
|
|
|
|
void ClearTestNodes()
|
|
{
|
|
LOCK(m_nodes_mutex);
|
|
for (CNode* node : m_nodes) {
|
|
delete node;
|
|
}
|
|
m_nodes.clear();
|
|
}
|
|
|
|
void Handshake(CNode& node,
|
|
bool successfully_connected,
|
|
ServiceFlags remote_services,
|
|
ServiceFlags local_services,
|
|
int32_t version,
|
|
bool relay_txs)
|
|
EXCLUSIVE_LOCKS_REQUIRED(NetEventsInterface::g_msgproc_mutex);
|
|
|
|
void ProcessMessagesOnce(CNode& node) EXCLUSIVE_LOCKS_REQUIRED(NetEventsInterface::g_msgproc_mutex) { m_msgproc->ProcessMessages(&node, flagInterruptMsgProc); }
|
|
|
|
void NodeReceiveMsgBytes(CNode& node, Span<const uint8_t> msg_bytes, bool& complete) const;
|
|
|
|
bool ReceiveMsgFrom(CNode& node, CSerializedNetMsg&& ser_msg) const;
|
|
void FlushSendBuffer(CNode& node) const;
|
|
|
|
bool AlreadyConnectedPublic(const CAddress& addr) { return AlreadyConnectedToAddress(addr); };
|
|
|
|
CNode* ConnectNodePublic(PeerManager& peerman, const char* pszDest, ConnectionType conn_type)
|
|
EXCLUSIVE_LOCKS_REQUIRED(!m_unused_i2p_sessions_mutex);
|
|
};
|
|
|
|
constexpr ServiceFlags ALL_SERVICE_FLAGS[]{
|
|
NODE_NONE,
|
|
NODE_NETWORK,
|
|
NODE_BLOOM,
|
|
NODE_WITNESS,
|
|
NODE_COMPACT_FILTERS,
|
|
NODE_NETWORK_LIMITED,
|
|
NODE_P2P_V2,
|
|
};
|
|
|
|
constexpr NetPermissionFlags ALL_NET_PERMISSION_FLAGS[]{
|
|
NetPermissionFlags::None,
|
|
NetPermissionFlags::BloomFilter,
|
|
NetPermissionFlags::Relay,
|
|
NetPermissionFlags::ForceRelay,
|
|
NetPermissionFlags::NoBan,
|
|
NetPermissionFlags::Mempool,
|
|
NetPermissionFlags::Addr,
|
|
NetPermissionFlags::Download,
|
|
NetPermissionFlags::Implicit,
|
|
NetPermissionFlags::All,
|
|
};
|
|
|
|
constexpr ConnectionType ALL_CONNECTION_TYPES[]{
|
|
ConnectionType::INBOUND,
|
|
ConnectionType::OUTBOUND_FULL_RELAY,
|
|
ConnectionType::MANUAL,
|
|
ConnectionType::FEELER,
|
|
ConnectionType::BLOCK_RELAY,
|
|
ConnectionType::ADDR_FETCH,
|
|
};
|
|
|
|
constexpr auto ALL_NETWORKS = std::array{
|
|
Network::NET_UNROUTABLE,
|
|
Network::NET_IPV4,
|
|
Network::NET_IPV6,
|
|
Network::NET_ONION,
|
|
Network::NET_I2P,
|
|
Network::NET_CJDNS,
|
|
Network::NET_INTERNAL,
|
|
};
|
|
|
|
/**
|
|
* A mocked Sock alternative that returns a statically contained data upon read and succeeds
|
|
* and ignores all writes. The data to be returned is given to the constructor and when it is
|
|
* exhausted an EOF is returned by further reads.
|
|
*/
|
|
class StaticContentsSock : public Sock
|
|
{
|
|
public:
|
|
explicit StaticContentsSock(const std::string& contents)
|
|
: Sock{INVALID_SOCKET},
|
|
m_contents{contents}
|
|
{
|
|
}
|
|
|
|
~StaticContentsSock() override { m_socket = INVALID_SOCKET; }
|
|
|
|
StaticContentsSock& operator=(Sock&& other) override
|
|
{
|
|
assert(false && "Move of Sock into MockSock not allowed.");
|
|
return *this;
|
|
}
|
|
|
|
ssize_t Send(const void*, size_t len, int) const override { return len; }
|
|
|
|
ssize_t Recv(void* buf, size_t len, int flags) const override
|
|
{
|
|
const size_t consume_bytes{std::min(len, m_contents.size() - m_consumed)};
|
|
std::memcpy(buf, m_contents.data() + m_consumed, consume_bytes);
|
|
if ((flags & MSG_PEEK) == 0) {
|
|
m_consumed += consume_bytes;
|
|
}
|
|
return consume_bytes;
|
|
}
|
|
|
|
int Connect(const sockaddr*, socklen_t) const override { return 0; }
|
|
|
|
int Bind(const sockaddr*, socklen_t) const override { return 0; }
|
|
|
|
int Listen(int) const override { return 0; }
|
|
|
|
std::unique_ptr<Sock> Accept(sockaddr* addr, socklen_t* addr_len) const override
|
|
{
|
|
if (addr != nullptr) {
|
|
// Pretend all connections come from 5.5.5.5:6789
|
|
memset(addr, 0x00, *addr_len);
|
|
const socklen_t write_len = static_cast<socklen_t>(sizeof(sockaddr_in));
|
|
if (*addr_len >= write_len) {
|
|
*addr_len = write_len;
|
|
sockaddr_in* addr_in = reinterpret_cast<sockaddr_in*>(addr);
|
|
addr_in->sin_family = AF_INET;
|
|
memset(&addr_in->sin_addr, 0x05, sizeof(addr_in->sin_addr));
|
|
addr_in->sin_port = htons(6789);
|
|
}
|
|
}
|
|
return std::make_unique<StaticContentsSock>("");
|
|
};
|
|
|
|
int GetSockOpt(int level, int opt_name, void* opt_val, socklen_t* opt_len) const override
|
|
{
|
|
std::memset(opt_val, 0x0, *opt_len);
|
|
return 0;
|
|
}
|
|
|
|
int SetSockOpt(int, int, const void*, socklen_t) const override { return 0; }
|
|
|
|
int GetSockName(sockaddr* name, socklen_t* name_len) const override
|
|
{
|
|
std::memset(name, 0x0, *name_len);
|
|
return 0;
|
|
}
|
|
|
|
bool SetNonBlocking() const override { return true; }
|
|
|
|
bool IsSelectable() const override { return true; }
|
|
|
|
bool Wait(std::chrono::milliseconds timeout,
|
|
Event requested,
|
|
Event* occurred = nullptr) const override
|
|
{
|
|
if (occurred != nullptr) {
|
|
*occurred = requested;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool WaitMany(std::chrono::milliseconds timeout, EventsPerSock& events_per_sock) const override
|
|
{
|
|
for (auto& [sock, events] : events_per_sock) {
|
|
(void)sock;
|
|
events.occurred = events.requested;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool IsConnected(std::string&) const override
|
|
{
|
|
return true;
|
|
}
|
|
|
|
private:
|
|
const std::string m_contents;
|
|
mutable size_t m_consumed{0};
|
|
};
|
|
|
|
std::vector<NodeEvictionCandidate> GetRandomNodeEvictionCandidates(int n_candidates, FastRandomContext& random_context);
|
|
|
|
#endif // BITCOIN_TEST_UTIL_NET_H
|