313 lines
8.3 KiB
Swift
313 lines
8.3 KiB
Swift
//
|
|
// OpenVPN+L10n.swift
|
|
// Passepartout
|
|
//
|
|
// Created by Davide De Rosa on 2/26/22.
|
|
// Copyright (c) 2023 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 PassepartoutLibrary
|
|
import TunnelKitOpenVPN
|
|
|
|
extension OpenVPN.Cipher: LocalizableEntity {
|
|
public var localizedDescription: String {
|
|
description
|
|
}
|
|
}
|
|
|
|
extension OpenVPN.Digest: LocalizableEntity {
|
|
public var localizedDescription: String {
|
|
description
|
|
}
|
|
}
|
|
|
|
extension OpenVPN.CompressionFraming: LocalizableEntity {
|
|
public var localizedDescription: String {
|
|
switch self {
|
|
case .disabled:
|
|
return L10n.Global.Strings.disabled
|
|
|
|
case .compLZO:
|
|
return Unlocalized.OpenVPN.compLZO
|
|
|
|
case .compress, .compressV2:
|
|
return Unlocalized.OpenVPN.compress
|
|
}
|
|
}
|
|
}
|
|
|
|
extension OpenVPN.CompressionAlgorithm: LocalizableEntity {
|
|
public var localizedDescription: String {
|
|
let V = L10n.Endpoint.Advanced.Openvpn.Items.self
|
|
switch self {
|
|
case .disabled:
|
|
return L10n.Global.Strings.disabled
|
|
|
|
case .LZO:
|
|
return Unlocalized.OpenVPN.lzo
|
|
|
|
case .other:
|
|
return V.CompressionAlgorithm.Value.other
|
|
}
|
|
}
|
|
}
|
|
|
|
extension OpenVPN.XORMethod: StyledLocalizableEntity {
|
|
public enum Style {
|
|
case short
|
|
|
|
case long
|
|
}
|
|
|
|
public func localizedDescription(style: Style) -> String {
|
|
switch style {
|
|
case .short:
|
|
return shortDescription
|
|
|
|
case .long:
|
|
return longDescription
|
|
}
|
|
}
|
|
|
|
private var shortDescription: String {
|
|
switch self {
|
|
case .xormask:
|
|
return Unlocalized.OpenVPN.XOR.xormask.rawValue
|
|
|
|
case .xorptrpos:
|
|
return Unlocalized.OpenVPN.XOR.xorptrpos.rawValue
|
|
|
|
case .reverse:
|
|
return Unlocalized.OpenVPN.XOR.reverse.rawValue
|
|
|
|
case .obfuscate:
|
|
return Unlocalized.OpenVPN.XOR.obfuscate.rawValue
|
|
}
|
|
}
|
|
|
|
private var longDescription: String {
|
|
switch self {
|
|
case .xormask(let mask):
|
|
return "\(shortDescription) \(mask.toHex())"
|
|
|
|
case .obfuscate(let mask):
|
|
return "\(shortDescription) \(mask.toHex())"
|
|
|
|
default:
|
|
return shortDescription
|
|
}
|
|
}
|
|
}
|
|
|
|
extension OpenVPN.PullMask: LocalizableEntity {
|
|
public var localizedDescription: String {
|
|
switch self {
|
|
case .routes:
|
|
return L10n.Endpoint.Advanced.Openvpn.Items.Route.caption
|
|
|
|
case .dns:
|
|
return Unlocalized.Network.dns
|
|
|
|
case .proxy:
|
|
return L10n.Global.Strings.proxy
|
|
}
|
|
}
|
|
}
|
|
|
|
extension OpenVPN.ConfigurationBuilder: StyledLocalizableEntity {
|
|
public enum Style {
|
|
case tlsWrap
|
|
|
|
case eku
|
|
}
|
|
|
|
public func localizedDescription(style: Style) -> String {
|
|
switch style {
|
|
case .tlsWrap:
|
|
return tlsWrap.tlsWrapDescription
|
|
|
|
case .eku:
|
|
return checksEKU.ekuDescription
|
|
}
|
|
}
|
|
}
|
|
|
|
extension OpenVPN.Configuration: StyledOptionalLocalizableEntity {
|
|
public enum OptionalStyle {
|
|
case keepAlive
|
|
|
|
case renegotiatesAfter
|
|
|
|
case randomizeEndpoint
|
|
|
|
case randomizeHostnames
|
|
}
|
|
|
|
public func localizedDescription(optionalStyle: OptionalStyle) -> String? {
|
|
switch optionalStyle {
|
|
case .keepAlive:
|
|
return keepAliveInterval?.keepAliveDescription
|
|
|
|
case .renegotiatesAfter:
|
|
return renegotiatesAfter?.renegotiatesAfterDescription
|
|
|
|
case .randomizeEndpoint:
|
|
return randomizeEndpoint?.randomizeEndpointDescription
|
|
|
|
case .randomizeHostnames:
|
|
return randomizeHostnames?.randomizeHostnamesDescription
|
|
}
|
|
}
|
|
}
|
|
|
|
private extension Optional where Wrapped == OpenVPN.TLSWrap {
|
|
var tlsWrapDescription: String {
|
|
guard let strategy = self?.strategy else {
|
|
return L10n.Global.Strings.disabled
|
|
}
|
|
let V = L10n.Endpoint.Advanced.Openvpn.Items.self
|
|
switch strategy {
|
|
case .auth:
|
|
return V.TlsWrapping.Value.auth
|
|
|
|
case .crypt:
|
|
return V.TlsWrapping.Value.crypt
|
|
}
|
|
}
|
|
}
|
|
|
|
private extension TimeInterval {
|
|
var keepAliveDescription: String {
|
|
let V = L10n.Endpoint.Advanced.Openvpn.Items.self
|
|
if self > 0 {
|
|
return V.KeepAlive.Value.seconds(Int(self))
|
|
} else {
|
|
return L10n.Global.Strings.disabled
|
|
}
|
|
}
|
|
}
|
|
|
|
private extension Optional where Wrapped == Bool {
|
|
var ekuDescription: String {
|
|
let V = L10n.Global.Strings.self
|
|
return (self ?? false) ? V.enabled : V.disabled
|
|
}
|
|
}
|
|
|
|
private extension TimeInterval {
|
|
var renegotiatesAfterDescription: String {
|
|
let V = L10n.Endpoint.Advanced.Openvpn.Items.self
|
|
if self > 0 {
|
|
return V.RenegotiationSeconds.Value.after(TimeInterval(self).localizedDescription)
|
|
} else {
|
|
return L10n.Global.Strings.disabled
|
|
}
|
|
}
|
|
}
|
|
|
|
private extension Bool {
|
|
var randomizeEndpointDescription: String {
|
|
let V = L10n.Global.Strings.self
|
|
return self ? V.enabled : V.disabled
|
|
}
|
|
|
|
var randomizeHostnamesDescription: String {
|
|
let V = L10n.Global.Strings.self
|
|
return self ? V.enabled : V.disabled
|
|
}
|
|
}
|
|
|
|
// MARK: - Errors
|
|
|
|
extension TunnelKitOpenVPNError: LocalizedError {
|
|
public var errorDescription: String? {
|
|
let V = L10n.Tunnelkit.Errors.Vpn.self
|
|
switch self {
|
|
case .socketActivity, .timeout:
|
|
return V.timeout
|
|
|
|
case .dnsFailure:
|
|
return V.dns
|
|
|
|
case .tlsInitialization, .tlsServerVerification, .tlsHandshake:
|
|
return V.tls
|
|
|
|
case .authentication:
|
|
return V.auth
|
|
|
|
case .encryptionInitialization, .encryptionData:
|
|
return V.encryption
|
|
|
|
case .serverCompression, .lzo:
|
|
return V.compression
|
|
|
|
case .networkChanged, .linkError:
|
|
return V.network
|
|
|
|
case .routing:
|
|
return V.routing
|
|
|
|
case .gatewayUnattainable:
|
|
return V.gateway
|
|
|
|
case .serverShutdown:
|
|
return V.shutdown
|
|
|
|
default:
|
|
return L10n.Global.Strings.unknown
|
|
}
|
|
}
|
|
}
|
|
|
|
extension OpenVPN.ConfigurationError: LocalizedError {
|
|
public var errorDescription: String? {
|
|
let V = L10n.Tunnelkit.Errors.Openvpn.self
|
|
switch self {
|
|
case .encryptionPassphrase:
|
|
pp_log.error("Could not parse configuration URL: unable to decrypt, passphrase required")
|
|
return V.passphraseRequired
|
|
|
|
case .unableToDecrypt(let error):
|
|
pp_log.error("Could not parse configuration URL: unable to decrypt, \(error.localizedDescription)")
|
|
return V.decryption
|
|
|
|
case .malformed(let option):
|
|
pp_log.error("Could not parse configuration URL: malformed option, \(option)")
|
|
return V.malformed(option)
|
|
|
|
case .missingConfiguration(let option):
|
|
pp_log.error("Could not parse configuration URL: missing configuration, \(option)")
|
|
return V.requiredOption(option)
|
|
|
|
case .unsupportedConfiguration(var option):
|
|
if option.contains("external") {
|
|
option.append(" (see FAQ)")
|
|
}
|
|
pp_log.error("Could not parse configuration URL: unsupported configuration, \(option)")
|
|
return V.unsupportedOption(option)
|
|
|
|
case .continuationPushReply:
|
|
assertionFailure("This is a server-side configuration parsing error")
|
|
return L10n.Global.Strings.unknown
|
|
}
|
|
}
|
|
}
|