//
// NEProfileRepository.swift
// Passepartout
//
// Created by Davide De Rosa on 10/10/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 .
//
import Combine
import Foundation
import NetworkExtension
import PassepartoutKit
public final class NEProfileRepository: ProfileRepository {
private let repository: NETunnelManagerRepository
private let title: (Profile) -> String
private let profilesSubject: CurrentValueSubject<[Profile], Never>
private var subscriptions: Set
public init(repository: NETunnelManagerRepository, title: @escaping (Profile) -> String) {
self.repository = repository
self.title = title
profilesSubject = CurrentValueSubject([])
subscriptions = []
repository
.managersPublisher
.dropFirst()
.sink { [weak self] in
self?.onUpdatedManagers($0)
}
.store(in: &subscriptions)
}
public var profilesPublisher: AnyPublisher<[Profile], Never> {
profilesSubject.eraseToAnyPublisher()
}
public func fetchProfiles() async throws -> [Profile] {
let managers = try await repository.fetch()
let profiles = managers.compactMap {
do {
return try repository.profile(from: $0)
} catch {
pp_log(.App.profiles, .error, "Unable to decode profile from NE manager '\($0.localizedDescription ?? "")': \(error)")
return nil
}
}
profilesSubject.send(profiles)
return profiles
}
public func saveProfile(_ profile: Profile) async throws {
try await repository.save(profile, forConnecting: false, title: title)
if let index = profilesSubject.value.firstIndex(where: { $0.id == profile.id }) {
profilesSubject.value[index] = profile
} else {
profilesSubject.value.append(profile)
}
}
public func removeProfiles(withIds profileIds: [Profile.ID]) async throws {
guard !profileIds.isEmpty else {
return
}
var removedIds: Set = []
defer {
profilesSubject.value.removeAll {
removedIds.contains($0.id)
}
}
for id in profileIds {
try await repository.remove(profileId: id)
removedIds.insert(id)
}
}
public func removeAllProfiles() async throws {
try await removeProfiles(withIds: profilesSubject.value.map(\.id))
}
}
private extension NEProfileRepository {
func onUpdatedManagers(_ managers: [Profile.ID: NETunnelProviderManager]) {
let profiles = profilesSubject
.value
.filter {
managers.keys.contains($0.id)
}
let removedProfilesDescription = profilesSubject
.value
.filter {
!managers.keys.contains($0.id)
}
.map {
"\($0.name)(\($0.id)"
}
if !removedProfilesDescription.isEmpty {
pp_log(.App.profiles, .info, "Sync profiles removed externally: \(removedProfilesDescription)")
}
profilesSubject.send(profiles)
}
}