//
//  AppError+L10n.swift
//  Passepartout
//
//  Created by Davide De Rosa on 8/27/24.
//  Copyright (c) 2024 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 CommonLibrary
import CommonUtils
import Foundation
import PassepartoutKit

extension AppError: LocalizedError {
    public var errorDescription: String? {
        let V = Strings.Errors.App.self
        switch self {
        case .couldNotLaunch(let reason):
            return reason.localizedDescription

        case .emptyProducts:
            return V.emptyProducts

        case .emptyProfileName:
            return V.emptyProfileName

        case .ineligibleProfile:
            return nil

        case .interactiveLogin:
            return nil

        case .malformedModule(let module, let error):
            return V.malformedModule(module.moduleType.localizedDescription, error.localizedDescription)

        case .permissionDenied:
            return V.default

        case .generic(let error):
            return error.localizedDescription
        }
    }
}

// MARK: - App side

extension PassepartoutError: @retroactive LocalizedError {
    public var errorDescription: String? {
        switch code {
        case .connectionModuleRequired:
            return Strings.Errors.App.Passepartout.connectionModuleRequired

        case .corruptProviderModule:
            return Strings.Errors.App.Passepartout.corruptProviderModule(reason?.localizedDescription ?? "")

        case .incompatibleModules:
            return Strings.Errors.App.Passepartout.incompatibleModules

        case .invalidFields:
            let fields = (userInfo as? [String: String?])
                .map {
                    $0.map {
                        "\($0)=\($1?.description ?? "")"
                    }
                    .joined(separator: ",")
                }

            return [Strings.Errors.App.Passepartout.invalidFields, fields]
                .compactMap { $0 }
                .joined(separator: " ")

        case .missingProviderEntity:
            return Strings.Errors.App.Passepartout.missingProviderEntity

        case .noActiveModules:
            return Strings.Errors.App.Passepartout.noActiveModules

        case .parsing:
            let message = userInfo as? String ?? (reason as? LocalizedError)?.localizedDescription

            return [Strings.Errors.App.Passepartout.parsing, message]
                .compactMap { $0 }
                .joined(separator: " ")

        case .providerRequired:
            return Strings.Errors.App.Passepartout.providerRequired

        case .unhandled:
            return reason?.localizedDescription

        default:
            return Strings.Errors.App.Passepartout.default(code.rawValue)
        }
    }
}

// MARK: - Tunnel side

extension PassepartoutError.Code: StyledLocalizableEntity {
    public enum Style {
        case tunnel
    }

    public func localizedDescription(style: Style) -> String {
        switch style {
        case .tunnel:
            let V = Strings.Errors.Tunnel.self
            switch self {
            case .App.ineligibleProfile:
                return V.ineligible

            case .authentication:
                return V.auth

            case .crypto:
                return V.encryption

            case .dnsFailure:
                return V.dns

            case .timeout:
                return V.timeout

            case .OpenVPN.compressionMismatch:
                return V.compression

            case .OpenVPN.noRouting:
                return V.routing

            case .OpenVPN.serverShutdown:
                return V.shutdown

            case .OpenVPN.tlsFailure:
                return V.tls

            default:
                return V.generic
            }
        }
    }
}