// // Pool.swift // Passepartout // // Created by Davide De Rosa on 6/11/18. // Copyright (c) 2019 Davide De Rosa. All rights reserved. // // https://github.com/passepartoutvpn // // This file is part of Passepartout. // // Passepartout is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // Passepartout is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with Passepartout. If not, see . // import Foundation import TunnelKit public struct Pool: Codable, Hashable { public enum CodingKeys: String, CodingKey { case id case country case extraCountries = "extra_countries" case area case num case tags // case location case hostname case isResolved = "resolved" case numericAddresses = "addrs" } public let id: String public let country: String public let extraCountries: [String]? public let area: String? public let num: Int? public let tags: [String]? // public let location: (Double, Double) public let hostname: String? public let isResolved: Bool? public let numericAddresses: [UInt32]? public func hasAddress(_ address: String) -> Bool { guard let numericAddresses = numericAddresses else { return false } guard let ipv4 = DNSResolver.ipv4(fromString: address) else { return false } return numericAddresses.contains(ipv4) } // XXX: inefficient, can't easily use lazy on struct public func addresses() -> [String] { var addrs = numericAddresses?.map { DNSResolver.string(fromIPv4: $0) } ?? [] if let hostname = hostname { addrs.insert(hostname, at: 0) } return addrs } // MARK: Hashable public func hash(into hasher: inout Hasher) { id.hash(into: &hasher) } } extension Pool { public var localizedCountry: String { return Utils.localizedCountry(country) } public var localizedId: String { var comps: [String] = [localizedCountry] if let secondaryId = optionalSecondaryId { comps.append(secondaryId) } return comps.joined(separator: " - ") } public var secondaryId: String { return optionalSecondaryId ?? "" } private var optionalSecondaryId: String? { var comps: [String] = [] if let extraCountries = extraCountries { comps.append(contentsOf: extraCountries.map { Utils.localizedCountry($0) }) } if let area = area { comps.append(area.uppercased()) } if let num = num { comps.append("#\(num)") } guard !comps.isEmpty else { return nil } var str = comps.joined(separator: " ") if let tags = tags { let suffix = tags.map { $0.uppercased() }.joined(separator: ",") str = "\(str) (\(suffix))" } return str } }