2022-04-12 13:09:14 +00:00
|
|
|
//
|
2023-05-24 16:19:47 +00:00
|
|
|
// CDWebServicesRepository.swift
|
2022-04-12 13:09:14 +00:00
|
|
|
// Passepartout
|
|
|
|
//
|
2023-05-24 16:19:47 +00:00
|
|
|
// Created by Davide De Rosa on 5/22/23.
|
2024-01-14 13:34:21 +00:00
|
|
|
// Copyright (c) 2024 Davide De Rosa. All rights reserved.
|
2022-04-12 13:09:14 +00:00
|
|
|
//
|
|
|
|
// 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 CoreData
|
2023-04-04 07:50:45 +00:00
|
|
|
import Foundation
|
2022-06-23 21:31:01 +00:00
|
|
|
import PassepartoutCore
|
2023-05-24 16:19:47 +00:00
|
|
|
import PassepartoutProviders
|
2022-04-12 13:09:14 +00:00
|
|
|
import PassepartoutServices
|
|
|
|
|
2023-05-24 16:19:47 +00:00
|
|
|
final class CDWebServicesRepository: WebServicesRepository {
|
2022-04-12 13:09:14 +00:00
|
|
|
private let context: NSManagedObjectContext
|
2023-03-17 20:55:47 +00:00
|
|
|
|
2023-05-24 16:19:47 +00:00
|
|
|
init(_ context: NSManagedObjectContext) {
|
2022-04-12 13:09:14 +00:00
|
|
|
self.context = context
|
|
|
|
}
|
2023-03-17 20:55:47 +00:00
|
|
|
|
2023-05-24 16:19:47 +00:00
|
|
|
func mergeIndex(_ index: WSProvidersIndex) throws {
|
|
|
|
let request = CDProvider.fetchRequest()
|
|
|
|
request.propertiesToFetch = [
|
|
|
|
"name",
|
|
|
|
"fullName"
|
|
|
|
]
|
|
|
|
do {
|
|
|
|
let providers = try context.fetch(request)
|
|
|
|
|
|
|
|
let indexNames = index.metadata.map(\.name)
|
|
|
|
let existingNames = providers.compactMap(\.name)
|
|
|
|
pp_log.debug("Fetched providers: \(indexNames)")
|
|
|
|
pp_log.debug("Existing providers: \(existingNames)")
|
|
|
|
|
|
|
|
let newNames = Set(indexNames).subtracting(existingNames)
|
|
|
|
pp_log.info("New providers: \(newNames)")
|
|
|
|
|
|
|
|
// add new
|
|
|
|
index.metadata.filter {
|
|
|
|
newNames.contains($0.name)
|
|
|
|
}.forEach {
|
|
|
|
_ = ProviderMapper(context).toDTO($0)
|
|
|
|
pp_log.info("Creating new provider metadata: \($0)")
|
|
|
|
}
|
|
|
|
|
|
|
|
// update existing
|
|
|
|
providers.forEach { dto in
|
|
|
|
guard let name = dto.name else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
guard let ws = index.metadata.first(where: {
|
|
|
|
$0.name == name
|
|
|
|
}) else {
|
|
|
|
// delete if not in new index
|
|
|
|
pp_log.info("Deleting provider: \(name)")
|
|
|
|
context.delete(dto)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
pp_log.info("Updating provider: \(name)")
|
|
|
|
dto.fullName = ws.fullName
|
|
|
|
dto.lastUpdate = Date()
|
|
|
|
}
|
|
|
|
|
|
|
|
try context.save()
|
|
|
|
} catch {
|
|
|
|
context.rollback()
|
|
|
|
throw error
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-12 13:09:14 +00:00
|
|
|
func saveInfrastructure(
|
|
|
|
_ infrastructure: WSProviderInfrastructure,
|
|
|
|
vpnProtocol: VPNProtocolType,
|
|
|
|
lastUpdate: Date
|
|
|
|
) throws {
|
|
|
|
do {
|
|
|
|
let provider = try providerDTO(forName: infrastructure.name) ?? {
|
|
|
|
let provider = CDProvider(context: context)
|
|
|
|
provider.name = infrastructure.name
|
|
|
|
provider.fullName = infrastructure.fullName
|
|
|
|
provider.lastUpdate = Date()
|
|
|
|
return provider
|
|
|
|
}()
|
|
|
|
|
2022-05-05 08:55:40 +00:00
|
|
|
let request = fetchRequest(infrastructure.name, vpnProtocol)
|
2022-05-05 08:45:38 +00:00
|
|
|
let existing = try context.fetch(request)
|
|
|
|
existing.forEach(context.delete)
|
2023-03-17 20:55:47 +00:00
|
|
|
|
2022-04-12 13:09:14 +00:00
|
|
|
let dto = InfrastructureMapper(
|
|
|
|
context,
|
|
|
|
infrastructure.name,
|
|
|
|
vpnProtocol
|
|
|
|
).toDTO(infrastructure)
|
|
|
|
dto.provider = provider
|
|
|
|
dto.lastUpdate = lastUpdate
|
|
|
|
provider.addToInfrastructures(dto)
|
|
|
|
|
|
|
|
try context.save()
|
|
|
|
} catch {
|
|
|
|
context.rollback()
|
|
|
|
throw error
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-05 08:55:40 +00:00
|
|
|
private func fetchRequest(_ name: ProviderName, _ vpnProtocol: VPNProtocolType) -> NSFetchRequest<CDInfrastructure> {
|
|
|
|
let request = CDInfrastructure.fetchRequest()
|
|
|
|
request.predicate = NSPredicate(
|
|
|
|
format: "provider.name == %@ AND vpnProtocol == %@",
|
|
|
|
name,
|
|
|
|
vpnProtocol.rawValue
|
|
|
|
)
|
|
|
|
return request
|
|
|
|
}
|
2023-03-17 20:55:47 +00:00
|
|
|
|
2022-04-12 13:09:14 +00:00
|
|
|
private func providerDTO(forName name: ProviderName) throws -> CDProvider? {
|
|
|
|
let request = CDProvider.fetchRequest()
|
|
|
|
request.sortDescriptors = [
|
|
|
|
.init(keyPath: \CDProvider.lastUpdate, ascending: false)
|
|
|
|
]
|
|
|
|
request.predicate = NSPredicate(
|
|
|
|
format: "name == %@",
|
|
|
|
name
|
|
|
|
)
|
|
|
|
return try context.fetch(request).first
|
|
|
|
}
|
|
|
|
}
|