mirror of
https://github.com/passepartoutvpn/passepartout-apple.git
synced 2024-12-26 19:32:41 +00:00
e8c91fe00c
- Use "Default" for no-area pools (count > 1) - Otherwise, omit area selector (count == 1)
186 lines
4.9 KiB
Swift
186 lines
4.9 KiB
Swift
//
|
|
// Pool.swift
|
|
// Passepartout
|
|
//
|
|
// Created by Davide De Rosa on 6/11/18.
|
|
// Copyright (c) 2021 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 <http://www.gnu.org/licenses/>.
|
|
//
|
|
|
|
import Foundation
|
|
import TunnelKit
|
|
|
|
public class 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]?
|
|
|
|
// XXX: inefficient but convenient field (not serialized)
|
|
public func category(in infrastructure: Infrastructure) -> PoolCategory? {
|
|
for category in infrastructure.categories {
|
|
for group in category.groups {
|
|
for pool in group.pools {
|
|
if pool.id == id {
|
|
return category
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
public func supportedPresetIds(in infrastructure: Infrastructure) -> [String] {
|
|
let poolCategory = category(in: infrastructure)
|
|
return poolCategory?.presets ?? infrastructure.presets.map { $0.id }
|
|
}
|
|
|
|
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: Equatable
|
|
|
|
public static func == (lhs: Pool, rhs: Pool) -> Bool {
|
|
return lhs.id == rhs.id
|
|
}
|
|
|
|
// 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())
|
|
comps.append(area.capitalized)
|
|
}
|
|
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
|
|
}
|
|
}
|
|
|
|
public extension Array where Element: Pool {
|
|
func sortedPools() -> [Element] {
|
|
return sorted {
|
|
guard let larea = $0.area else {
|
|
guard let lnum = $0.num else {
|
|
return true
|
|
}
|
|
guard let rnum = $1.num else {
|
|
return false
|
|
}
|
|
guard lnum != rnum else {
|
|
return $0.secondaryId < $1.secondaryId
|
|
}
|
|
return lnum < rnum
|
|
}
|
|
guard let rarea = $1.area else {
|
|
return false
|
|
}
|
|
return larea < rarea
|
|
}
|
|
}
|
|
}
|