mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-11-10 22:18:54 +01:00
net: Make DNS lookup code testable
This commit is contained in:
116
src/netbase.cpp
116
src/netbase.cpp
@@ -42,6 +42,50 @@ bool fNameLookup = DEFAULT_NAME_LOOKUP;
|
|||||||
int g_socks5_recv_timeout = 20 * 1000;
|
int g_socks5_recv_timeout = 20 * 1000;
|
||||||
static std::atomic<bool> interruptSocks5Recv(false);
|
static std::atomic<bool> interruptSocks5Recv(false);
|
||||||
|
|
||||||
|
std::vector<CNetAddr> WrappedGetAddrInfo(const std::string& name, bool allow_lookup)
|
||||||
|
{
|
||||||
|
addrinfo ai_hint{};
|
||||||
|
// We want a TCP port, which is a streaming socket type
|
||||||
|
ai_hint.ai_socktype = SOCK_STREAM;
|
||||||
|
ai_hint.ai_protocol = IPPROTO_TCP;
|
||||||
|
// We don't care which address family (IPv4 or IPv6) is returned
|
||||||
|
ai_hint.ai_family = AF_UNSPEC;
|
||||||
|
// If we allow lookups of hostnames, use the AI_ADDRCONFIG flag to only
|
||||||
|
// return addresses whose family we have an address configured for.
|
||||||
|
//
|
||||||
|
// If we don't allow lookups, then use the AI_NUMERICHOST flag for
|
||||||
|
// getaddrinfo to only decode numerical network addresses and suppress
|
||||||
|
// hostname lookups.
|
||||||
|
ai_hint.ai_flags = allow_lookup ? AI_ADDRCONFIG : AI_NUMERICHOST;
|
||||||
|
|
||||||
|
addrinfo* ai_res{nullptr};
|
||||||
|
const int n_err{getaddrinfo(name.c_str(), nullptr, &ai_hint, &ai_res)};
|
||||||
|
if (n_err != 0) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Traverse the linked list starting with ai_trav.
|
||||||
|
addrinfo* ai_trav{ai_res};
|
||||||
|
std::vector<CNetAddr> resolved_addresses;
|
||||||
|
while (ai_trav != nullptr) {
|
||||||
|
if (ai_trav->ai_family == AF_INET) {
|
||||||
|
assert(ai_trav->ai_addrlen >= sizeof(sockaddr_in));
|
||||||
|
resolved_addresses.emplace_back(reinterpret_cast<sockaddr_in*>(ai_trav->ai_addr)->sin_addr);
|
||||||
|
}
|
||||||
|
if (ai_trav->ai_family == AF_INET6) {
|
||||||
|
assert(ai_trav->ai_addrlen >= sizeof(sockaddr_in6));
|
||||||
|
const sockaddr_in6* s6{reinterpret_cast<sockaddr_in6*>(ai_trav->ai_addr)};
|
||||||
|
resolved_addresses.emplace_back(s6->sin6_addr, s6->sin6_scope_id);
|
||||||
|
}
|
||||||
|
ai_trav = ai_trav->ai_next;
|
||||||
|
}
|
||||||
|
freeaddrinfo(ai_res);
|
||||||
|
|
||||||
|
return resolved_addresses;
|
||||||
|
}
|
||||||
|
|
||||||
|
DNSLookupFn g_dns_lookup{WrappedGetAddrInfo};
|
||||||
|
|
||||||
enum Network ParseNetwork(const std::string& net_in) {
|
enum Network ParseNetwork(const std::string& net_in) {
|
||||||
std::string net = ToLower(net_in);
|
std::string net = ToLower(net_in);
|
||||||
if (net == "ipv4") return NET_IPV4;
|
if (net == "ipv4") return NET_IPV4;
|
||||||
@@ -87,7 +131,7 @@ std::vector<std::string> GetNetworkNames(bool append_unroutable)
|
|||||||
return names;
|
return names;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool static LookupIntern(const std::string& name, std::vector<CNetAddr>& vIP, unsigned int nMaxSolutions, bool fAllowLookup)
|
static bool LookupIntern(const std::string& name, std::vector<CNetAddr>& vIP, unsigned int nMaxSolutions, bool fAllowLookup, DNSLookupFn dns_lookup_function)
|
||||||
{
|
{
|
||||||
vIP.clear();
|
vIP.clear();
|
||||||
|
|
||||||
@@ -109,54 +153,16 @@ bool static LookupIntern(const std::string& name, std::vector<CNetAddr>& vIP, un
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct addrinfo aiHint;
|
for (const CNetAddr& resolved : dns_lookup_function(name, fAllowLookup)) {
|
||||||
memset(&aiHint, 0, sizeof(struct addrinfo));
|
if (nMaxSolutions > 0 && vIP.size() >= nMaxSolutions) {
|
||||||
|
break;
|
||||||
// We want a TCP port, which is a streaming socket type
|
|
||||||
aiHint.ai_socktype = SOCK_STREAM;
|
|
||||||
aiHint.ai_protocol = IPPROTO_TCP;
|
|
||||||
// We don't care which address family (IPv4 or IPv6) is returned
|
|
||||||
aiHint.ai_family = AF_UNSPEC;
|
|
||||||
// If we allow lookups of hostnames, use the AI_ADDRCONFIG flag to only
|
|
||||||
// return addresses whose family we have an address configured for.
|
|
||||||
//
|
|
||||||
// If we don't allow lookups, then use the AI_NUMERICHOST flag for
|
|
||||||
// getaddrinfo to only decode numerical network addresses and suppress
|
|
||||||
// hostname lookups.
|
|
||||||
aiHint.ai_flags = fAllowLookup ? AI_ADDRCONFIG : AI_NUMERICHOST;
|
|
||||||
struct addrinfo *aiRes = nullptr;
|
|
||||||
int nErr = getaddrinfo(name.c_str(), nullptr, &aiHint, &aiRes);
|
|
||||||
if (nErr)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
// Traverse the linked list starting with aiTrav, add all non-internal
|
|
||||||
// IPv4,v6 addresses to vIP while respecting nMaxSolutions.
|
|
||||||
struct addrinfo *aiTrav = aiRes;
|
|
||||||
while (aiTrav != nullptr && (nMaxSolutions == 0 || vIP.size() < nMaxSolutions))
|
|
||||||
{
|
|
||||||
CNetAddr resolved;
|
|
||||||
if (aiTrav->ai_family == AF_INET)
|
|
||||||
{
|
|
||||||
assert(aiTrav->ai_addrlen >= sizeof(sockaddr_in));
|
|
||||||
resolved = CNetAddr(((struct sockaddr_in*)(aiTrav->ai_addr))->sin_addr);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (aiTrav->ai_family == AF_INET6)
|
|
||||||
{
|
|
||||||
assert(aiTrav->ai_addrlen >= sizeof(sockaddr_in6));
|
|
||||||
struct sockaddr_in6* s6 = (struct sockaddr_in6*) aiTrav->ai_addr;
|
|
||||||
resolved = CNetAddr(s6->sin6_addr, s6->sin6_scope_id);
|
|
||||||
}
|
}
|
||||||
/* Never allow resolving to an internal address. Consider any such result invalid */
|
/* Never allow resolving to an internal address. Consider any such result invalid */
|
||||||
if (!resolved.IsInternal()) {
|
if (!resolved.IsInternal()) {
|
||||||
vIP.push_back(resolved);
|
vIP.push_back(resolved);
|
||||||
}
|
}
|
||||||
|
|
||||||
aiTrav = aiTrav->ai_next;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
freeaddrinfo(aiRes);
|
|
||||||
|
|
||||||
return (vIP.size() > 0);
|
return (vIP.size() > 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -175,7 +181,7 @@ bool static LookupIntern(const std::string& name, std::vector<CNetAddr>& vIP, un
|
|||||||
* @see Lookup(const char *, std::vector<CService>&, int, bool, unsigned int)
|
* @see Lookup(const char *, std::vector<CService>&, int, bool, unsigned int)
|
||||||
* for additional parameter descriptions.
|
* for additional parameter descriptions.
|
||||||
*/
|
*/
|
||||||
bool LookupHost(const std::string& name, std::vector<CNetAddr>& vIP, unsigned int nMaxSolutions, bool fAllowLookup)
|
bool LookupHost(const std::string& name, std::vector<CNetAddr>& vIP, unsigned int nMaxSolutions, bool fAllowLookup, DNSLookupFn dns_lookup_function)
|
||||||
{
|
{
|
||||||
if (!ValidAsCString(name)) {
|
if (!ValidAsCString(name)) {
|
||||||
return false;
|
return false;
|
||||||
@@ -187,7 +193,7 @@ bool LookupHost(const std::string& name, std::vector<CNetAddr>& vIP, unsigned in
|
|||||||
strHost = strHost.substr(1, strHost.size() - 2);
|
strHost = strHost.substr(1, strHost.size() - 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
return LookupIntern(strHost, vIP, nMaxSolutions, fAllowLookup);
|
return LookupIntern(strHost, vIP, nMaxSolutions, fAllowLookup, dns_lookup_function);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -196,13 +202,13 @@ bool LookupHost(const std::string& name, std::vector<CNetAddr>& vIP, unsigned in
|
|||||||
* @see LookupHost(const std::string&, std::vector<CNetAddr>&, unsigned int, bool) for
|
* @see LookupHost(const std::string&, std::vector<CNetAddr>&, unsigned int, bool) for
|
||||||
* additional parameter descriptions.
|
* additional parameter descriptions.
|
||||||
*/
|
*/
|
||||||
bool LookupHost(const std::string& name, CNetAddr& addr, bool fAllowLookup)
|
bool LookupHost(const std::string& name, CNetAddr& addr, bool fAllowLookup, DNSLookupFn dns_lookup_function)
|
||||||
{
|
{
|
||||||
if (!ValidAsCString(name)) {
|
if (!ValidAsCString(name)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
std::vector<CNetAddr> vIP;
|
std::vector<CNetAddr> vIP;
|
||||||
LookupHost(name, vIP, 1, fAllowLookup);
|
LookupHost(name, vIP, 1, fAllowLookup, dns_lookup_function);
|
||||||
if(vIP.empty())
|
if(vIP.empty())
|
||||||
return false;
|
return false;
|
||||||
addr = vIP.front();
|
addr = vIP.front();
|
||||||
@@ -229,7 +235,7 @@ bool LookupHost(const std::string& name, CNetAddr& addr, bool fAllowLookup)
|
|||||||
* @returns Whether or not the service string successfully resolved to any
|
* @returns Whether or not the service string successfully resolved to any
|
||||||
* resulting services.
|
* resulting services.
|
||||||
*/
|
*/
|
||||||
bool Lookup(const std::string& name, std::vector<CService>& vAddr, int portDefault, bool fAllowLookup, unsigned int nMaxSolutions)
|
bool Lookup(const std::string& name, std::vector<CService>& vAddr, int portDefault, bool fAllowLookup, unsigned int nMaxSolutions, DNSLookupFn dns_lookup_function)
|
||||||
{
|
{
|
||||||
if (name.empty() || !ValidAsCString(name)) {
|
if (name.empty() || !ValidAsCString(name)) {
|
||||||
return false;
|
return false;
|
||||||
@@ -239,7 +245,7 @@ bool Lookup(const std::string& name, std::vector<CService>& vAddr, int portDefau
|
|||||||
SplitHostPort(name, port, hostname);
|
SplitHostPort(name, port, hostname);
|
||||||
|
|
||||||
std::vector<CNetAddr> vIP;
|
std::vector<CNetAddr> vIP;
|
||||||
bool fRet = LookupIntern(hostname, vIP, nMaxSolutions, fAllowLookup);
|
bool fRet = LookupIntern(hostname, vIP, nMaxSolutions, fAllowLookup, dns_lookup_function);
|
||||||
if (!fRet)
|
if (!fRet)
|
||||||
return false;
|
return false;
|
||||||
vAddr.resize(vIP.size());
|
vAddr.resize(vIP.size());
|
||||||
@@ -254,13 +260,13 @@ bool Lookup(const std::string& name, std::vector<CService>& vAddr, int portDefau
|
|||||||
* @see Lookup(const char *, std::vector<CService>&, int, bool, unsigned int)
|
* @see Lookup(const char *, std::vector<CService>&, int, bool, unsigned int)
|
||||||
* for additional parameter descriptions.
|
* for additional parameter descriptions.
|
||||||
*/
|
*/
|
||||||
bool Lookup(const std::string& name, CService& addr, int portDefault, bool fAllowLookup)
|
bool Lookup(const std::string& name, CService& addr, int portDefault, bool fAllowLookup, DNSLookupFn dns_lookup_function)
|
||||||
{
|
{
|
||||||
if (!ValidAsCString(name)) {
|
if (!ValidAsCString(name)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
std::vector<CService> vService;
|
std::vector<CService> vService;
|
||||||
bool fRet = Lookup(name, vService, portDefault, fAllowLookup, 1);
|
bool fRet = Lookup(name, vService, portDefault, fAllowLookup, 1, dns_lookup_function);
|
||||||
if (!fRet)
|
if (!fRet)
|
||||||
return false;
|
return false;
|
||||||
addr = vService[0];
|
addr = vService[0];
|
||||||
@@ -277,7 +283,7 @@ bool Lookup(const std::string& name, CService& addr, int portDefault, bool fAllo
|
|||||||
* @see Lookup(const char *, CService&, int, bool) for additional parameter
|
* @see Lookup(const char *, CService&, int, bool) for additional parameter
|
||||||
* descriptions.
|
* descriptions.
|
||||||
*/
|
*/
|
||||||
CService LookupNumeric(const std::string& name, int portDefault)
|
CService LookupNumeric(const std::string& name, int portDefault, DNSLookupFn dns_lookup_function)
|
||||||
{
|
{
|
||||||
if (!ValidAsCString(name)) {
|
if (!ValidAsCString(name)) {
|
||||||
return {};
|
return {};
|
||||||
@@ -285,7 +291,7 @@ CService LookupNumeric(const std::string& name, int portDefault)
|
|||||||
CService addr;
|
CService addr;
|
||||||
// "1.2:345" will fail to resolve the ip, but will still set the port.
|
// "1.2:345" will fail to resolve the ip, but will still set the port.
|
||||||
// If the ip fails to resolve, re-init the result.
|
// If the ip fails to resolve, re-init the result.
|
||||||
if(!Lookup(name, addr, portDefault, false))
|
if(!Lookup(name, addr, portDefault, false, dns_lookup_function))
|
||||||
addr = CService();
|
addr = CService();
|
||||||
return addr;
|
return addr;
|
||||||
}
|
}
|
||||||
@@ -811,7 +817,7 @@ bool ConnectThroughProxy(const proxyType& proxy, const std::string& strDest, int
|
|||||||
*
|
*
|
||||||
* @returns Whether the operation succeeded or not.
|
* @returns Whether the operation succeeded or not.
|
||||||
*/
|
*/
|
||||||
bool LookupSubNet(const std::string& strSubnet, CSubNet& ret)
|
bool LookupSubNet(const std::string& strSubnet, CSubNet& ret, DNSLookupFn dns_lookup_function)
|
||||||
{
|
{
|
||||||
if (!ValidAsCString(strSubnet)) {
|
if (!ValidAsCString(strSubnet)) {
|
||||||
return false;
|
return false;
|
||||||
@@ -822,7 +828,7 @@ bool LookupSubNet(const std::string& strSubnet, CSubNet& ret)
|
|||||||
std::string strAddress = strSubnet.substr(0, slash);
|
std::string strAddress = strSubnet.substr(0, slash);
|
||||||
// TODO: Use LookupHost(const std::string&, CNetAddr&, bool) instead to just get
|
// TODO: Use LookupHost(const std::string&, CNetAddr&, bool) instead to just get
|
||||||
// one CNetAddr.
|
// one CNetAddr.
|
||||||
if (LookupHost(strAddress, vIP, 1, false))
|
if (LookupHost(strAddress, vIP, 1, false, dns_lookup_function))
|
||||||
{
|
{
|
||||||
CNetAddr network = vIP[0];
|
CNetAddr network = vIP[0];
|
||||||
if (slash != strSubnet.npos)
|
if (slash != strSubnet.npos)
|
||||||
@@ -837,7 +843,7 @@ bool LookupSubNet(const std::string& strSubnet, CSubNet& ret)
|
|||||||
else // If not a valid number, try full netmask syntax
|
else // If not a valid number, try full netmask syntax
|
||||||
{
|
{
|
||||||
// Never allow lookup for netmask
|
// Never allow lookup for netmask
|
||||||
if (LookupHost(strNetmask, vIP, 1, false)) {
|
if (LookupHost(strNetmask, vIP, 1, false, dns_lookup_function)) {
|
||||||
ret = CSubNet(network, vIP[0]);
|
ret = CSubNet(network, vIP[0]);
|
||||||
return ret.IsValid();
|
return ret.IsValid();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -64,6 +64,11 @@ struct ProxyCredentials
|
|||||||
std::string password;
|
std::string password;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Wrapper for getaddrinfo(3). Do not use directly: call Lookup/LookupHost/LookupNumeric/LookupSubNet.
|
||||||
|
*/
|
||||||
|
std::vector<CNetAddr> WrappedGetAddrInfo(const std::string& name, bool allow_lookup);
|
||||||
|
|
||||||
enum Network ParseNetwork(const std::string& net);
|
enum Network ParseNetwork(const std::string& net);
|
||||||
std::string GetNetworkName(enum Network net);
|
std::string GetNetworkName(enum Network net);
|
||||||
/** Return a vector of publicly routable Network names; optionally append NET_UNROUTABLE. */
|
/** Return a vector of publicly routable Network names; optionally append NET_UNROUTABLE. */
|
||||||
@@ -74,12 +79,16 @@ bool IsProxy(const CNetAddr &addr);
|
|||||||
bool SetNameProxy(const proxyType &addrProxy);
|
bool SetNameProxy(const proxyType &addrProxy);
|
||||||
bool HaveNameProxy();
|
bool HaveNameProxy();
|
||||||
bool GetNameProxy(proxyType &nameProxyOut);
|
bool GetNameProxy(proxyType &nameProxyOut);
|
||||||
bool LookupHost(const std::string& name, std::vector<CNetAddr>& vIP, unsigned int nMaxSolutions, bool fAllowLookup);
|
|
||||||
bool LookupHost(const std::string& name, CNetAddr& addr, bool fAllowLookup);
|
using DNSLookupFn = std::function<std::vector<CNetAddr>(const std::string&, bool)>;
|
||||||
bool Lookup(const std::string& name, CService& addr, int portDefault, bool fAllowLookup);
|
extern DNSLookupFn g_dns_lookup;
|
||||||
bool Lookup(const std::string& name, std::vector<CService>& vAddr, int portDefault, bool fAllowLookup, unsigned int nMaxSolutions);
|
|
||||||
CService LookupNumeric(const std::string& name, int portDefault = 0);
|
bool LookupHost(const std::string& name, std::vector<CNetAddr>& vIP, unsigned int nMaxSolutions, bool fAllowLookup, DNSLookupFn dns_lookup_function = g_dns_lookup);
|
||||||
bool LookupSubNet(const std::string& strSubnet, CSubNet& subnet);
|
bool LookupHost(const std::string& name, CNetAddr& addr, bool fAllowLookup, DNSLookupFn dns_lookup_function = g_dns_lookup);
|
||||||
|
bool Lookup(const std::string& name, CService& addr, int portDefault, bool fAllowLookup, DNSLookupFn dns_lookup_function = g_dns_lookup);
|
||||||
|
bool Lookup(const std::string& name, std::vector<CService>& vAddr, int portDefault, bool fAllowLookup, unsigned int nMaxSolutions, DNSLookupFn dns_lookup_function = g_dns_lookup);
|
||||||
|
CService LookupNumeric(const std::string& name, int portDefault = 0, DNSLookupFn dns_lookup_function = g_dns_lookup);
|
||||||
|
bool LookupSubNet(const std::string& strSubnet, CSubNet& subnet, DNSLookupFn dns_lookup_function = g_dns_lookup);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a TCP socket in the given address family.
|
* Create a TCP socket in the given address family.
|
||||||
|
|||||||
Reference in New Issue
Block a user