#include #include #include #include #include #include #include #include #include #include namespace DB { namespace ErrorCodes { extern const int SYSTEM_ERROR; } namespace { struct NetworkInterfaces { ifaddrs * ifaddr; NetworkInterfaces() { if (getifaddrs(&ifaddr) == -1) { throwFromErrno("Cannot getifaddrs", ErrorCodes::SYSTEM_ERROR); } } bool hasAddress(const Poco::Net::IPAddress & address) const { ifaddrs * iface; for (iface = ifaddr; iface != nullptr; iface = iface->ifa_next) { /// Point-to-point (VPN) addresses may have NULL ifa_addr if (!iface->ifa_addr) continue; auto family = iface->ifa_addr->sa_family; std::optional interface_address; switch (family) { /// We interested only in IP-adresses case AF_INET: { interface_address.emplace(*(iface->ifa_addr)); break; } case AF_INET6: { interface_address.emplace(&reinterpret_cast(iface->ifa_addr)->sin6_addr, sizeof(struct in6_addr)); break; } default: continue; } /** Compare the addresses without taking into account `scope`. * Theoretically, this may not be correct - depends on `route` setting * - through which interface we will actually access the specified address. */ if (interface_address->length() == address.length() && 0 == memcmp(interface_address->addr(), address.addr(), address.length())) return true; } return false; } ~NetworkInterfaces() { freeifaddrs(ifaddr); } }; } bool isLocalAddress(const Poco::Net::IPAddress & address) { /** 127.0.0.1 is treat as local address unconditionally. * ::1 is also treat as local address unconditionally. * * 127.0.0.{2..255} are not treat as local addresses, because they are used in tests * to emulate distributed queries across localhost. * * But 127.{0,1}.{0,1}.{0,1} are treat as local addresses, * because they are used in Debian for localhost. */ if (address.isLoopback()) { if (address.family() == Poco::Net::AddressFamily::IPv4) { using Digits = std::array; Digits digits = unalignedLoad(address.addr()); /// The address is located in memory in big endian form. if (digits[0] == 127 && digits[1] <= 1 && digits[2] <= 1 && digits[3] <= 1) { return true; } } else if (address.family() == Poco::Net::AddressFamily::IPv6) { return true; } } NetworkInterfaces interfaces; return interfaces.hasAddress(address); } bool isLocalAddress(const Poco::Net::SocketAddress & address, UInt16 clickhouse_port) { return clickhouse_port == address.port() && isLocalAddress(address.host()); } size_t getHostNameDifference(const std::string & local_hostname, const std::string & host) { size_t hostname_difference = 0; for (size_t i = 0; i < std::min(local_hostname.length(), host.length()); ++i) if (local_hostname[i] != host[i]) ++hostname_difference; return hostname_difference; } }