/* * network.cpp * * This source file is part of the FoundationDB open source project * * Copyright 2013-2018 Apple Inc. and the FoundationDB project authors * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "Arena.h" #include "boost/asio.hpp" #include "flow/network.h" #include "flow/flow.h" #include "flow/UnitTest.h" bool IPAddress::operator==(const IPAddress& rhs) const { return addr == rhs.addr; } bool IPAddress::operator!=(const IPAddress& addr) const { return !(*this == addr); } bool IPAddress::operator<(const IPAddress& rhs) const { return addr < rhs.addr; } std::string IPAddress::toString() const { if (isV6()) { return boost::asio::ip::address_v6(std::get(addr)).to_string(); } else { auto ip = std::get(addr); return format("%d.%d.%d.%d", (ip >> 24) & 0xff, (ip >> 16) & 0xff, (ip >> 8) & 0xff, ip & 0xff); } } Optional IPAddress::parse(std::string const& str) { try { auto addr = boost::asio::ip::address::from_string(str); return addr.is_v6() ? IPAddress(addr.to_v6().to_bytes()) : IPAddress(addr.to_v4().to_ulong()); } catch (...) { return Optional(); } } bool IPAddress::isValid() const { if (isV6()) { const auto& ip = std::get(addr); return std::any_of(ip.begin(), ip.end(), [](uint8_t part) { return part != 0; }); } return std::get(addr) != 0; } NetworkAddress NetworkAddress::parse(std::string const& s) { if (s.empty()) { throw connection_string_invalid(); } bool isTLS = false; std::string f; if (s.size() > 4 && strcmp(s.c_str() + s.size() - 4, ":tls") == 0) { isTLS = true; f = s.substr(0, s.size() - 4); } else { f = s; } if (f[0] == '[') { // IPv6 address/port pair is represented as "[ip]:port" auto addrEnd = f.find_first_of(']'); if (addrEnd == std::string::npos || f[addrEnd + 1] != ':') { throw connection_string_invalid(); } auto port = std::stoi(f.substr(addrEnd + 2)); auto addr = IPAddress::parse(f.substr(1, addrEnd - 1)); if (!addr.present()) { throw connection_string_invalid(); } return NetworkAddress(addr.get(), port, true, isTLS); } else { // TODO: Use IPAddress::parse int a, b, c, d, port, count = -1; if (sscanf(f.c_str(), "%d.%d.%d.%d:%d%n", &a, &b, &c, &d, &port, &count) < 5 || count != f.size()) throw connection_string_invalid(); return NetworkAddress((a << 24) + (b << 16) + (c << 8) + d, port, true, isTLS); } } Optional NetworkAddress::parseOptional(std::string const& s) { try { return NetworkAddress::parse(s); } catch (Error& e) { ASSERT(e.code() == error_code_connection_string_invalid); return Optional(); } } std::vector NetworkAddress::parseList(std::string const& addrs) { // Split addrs on ',' and parse them individually std::vector coord; for (int p = 0; p <= addrs.size();) { int pComma = addrs.find_first_of(',', p); if (pComma == addrs.npos) pComma = addrs.size(); NetworkAddress parsedAddress = NetworkAddress::parse(addrs.substr(p, pComma - p)); coord.push_back(parsedAddress); p = pComma + 1; } return coord; } std::string NetworkAddress::toString() const { return formatIpPort(ip, port) + (isTLS() ? ":tls" : ""); } std::string toIPVectorString(const std::vector& ips) { std::string output; const char* space = ""; for (const auto& ip : ips) { output += format("%s%d.%d.%d.%d", space, (ip >> 24) & 0xff, (ip >> 16) & 0xff, (ip >> 8) & 0xff, ip & 0xff); space = " "; } return output; } std::string toIPVectorString(const std::vector& ips) { std::string output; const char* space = ""; for (auto ip : ips) { output += format("%s%s", space, ip.toString().c_str()); space = " "; } return output; } std::string formatIpPort(const IPAddress& ip, uint16_t port) { const char* patt = ip.isV6() ? "[%s]:%d" : "%s:%d"; return format(patt, ip.toString().c_str(), port); } Future> INetworkConnections::connect(const std::string& host, const std::string& service, bool useTLS) { // Use map to create an actor that returns an endpoint or throws Future pickEndpoint = map(resolveTCPEndpoint(host, service), [=](std::vector const& addresses) -> NetworkAddress { NetworkAddress addr = addresses[deterministicRandom()->randomInt(0, addresses.size())]; if (useTLS) addr.flags = NetworkAddress::FLAG_TLS; return addr; }); // Wait for the endpoint to return, then wait for connect(endpoint) and return it. // Template types are being provided explicitly because they can't be automatically deduced for some reason. return mapAsync>(NetworkAddress const&)>, Reference>( pickEndpoint, [=](NetworkAddress const& addr) -> Future> { return connectExternal(addr, host); }); } IUDPSocket::~IUDPSocket() {} const std::vector NetworkMetrics::starvationBins = { 1, 3500, 7000, 7500, 8500, 8900, 10500 }; TEST_CASE("/flow/network/ipaddress") { ASSERT(NetworkAddress::parse("[::1]:4800").toString() == "[::1]:4800"); { auto addr = "[2001:0db8:85a3:0000:0000:8a2e:0370:7334]:4800"; auto addrParsed = NetworkAddress::parse(addr); auto addrCompressed = "[2001:db8:85a3::8a2e:370:7334]:4800"; ASSERT(addrParsed.isV6()); ASSERT(!addrParsed.isTLS()); ASSERT(addrParsed.toString() == addrCompressed); } { auto addr = "[2001:0db8:85a3:0000:0000:8a2e:0370:7334]:4800:tls"; auto addrParsed = NetworkAddress::parse(addr); auto addrCompressed = "[2001:db8:85a3::8a2e:370:7334]:4800:tls"; ASSERT(addrParsed.isV6()); ASSERT(addrParsed.isTLS()); ASSERT(addrParsed.toString() == addrCompressed); } { auto addr = "2001:0db8:85a3:0000:0000:8a2e:0370:7334"; auto addrCompressed = "2001:db8:85a3::8a2e:370:7334"; auto addrParsed = IPAddress::parse(addr); ASSERT(addrParsed.present()); ASSERT(addrParsed.get().toString() == addrCompressed); } { auto addr = "2001"; auto addrParsed = IPAddress::parse(addr); ASSERT(!addrParsed.present()); } { auto addr = "8.8.8.8:12"; auto addrParsed = IPAddress::parse(addr); ASSERT(!addrParsed.present()); } return Void(); } NetworkInfo::NetworkInfo() : handshakeLock(new FlowLock(FLOW_KNOBS->TLS_HANDSHAKE_LIMIT)) {}