passepartout-apple/Passepartout/Library/Sources/AppDataProfiles/CDProfileRepositoryV3.swift
Davide 68df6066ba
Improve configuration on app launch/active (#821)
- Centralize context initialization/refresh in platform-specific app
delegates
- Prevent multiple calls to .onApplicationActive()
- Simplify local/remote profile fingerprint comparison
- Revert to always replacing Core Data entities
- The remote store somehow ended up having duplicates, which caused
repeated imports of remote profiles due to randomly different
fingerprints
- Optimize reload of in-app receipt
2024-11-06 18:42:42 +01:00

124 lines
3.7 KiB
Swift

//
// CDProfileRepositoryV3.swift
// Passepartout
//
// Created by Davide De Rosa on 8/11/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 AppData
import Combine
import CommonLibrary
import CommonUtils
import CoreData
import Foundation
import PassepartoutKit
extension AppData {
public static func cdProfileRepositoryV3(
registry: Registry,
coder: ProfileCoder,
context: NSManagedObjectContext,
observingResults: Bool,
onResultError: ((Error) -> CoreDataResultAction)?
) -> ProfileRepository {
let repository = CoreDataRepository<CDProfileV3, Profile>(
context: context,
observingResults: observingResults
) {
$0.sortDescriptors = [
.init(key: "name", ascending: true, selector: #selector(NSString.caseInsensitiveCompare)),
.init(key: "lastUpdate", ascending: true)
]
} fromMapper: {
try fromMapper($0, registry: registry, coder: coder)
} toMapper: {
try toMapper($0, $1, registry: registry, coder: coder)
} onResultError: {
onResultError?($0) ?? .ignore
}
return repository
}
}
private extension AppData {
static func fromMapper(
_ cdEntity: CDProfileV3,
registry: Registry,
coder: ProfileCoder
) throws -> Profile? {
guard let encoded = cdEntity.encoded else {
return nil
}
let profile = try registry.decodedProfile(from: encoded, with: coder)
return profile
}
static func toMapper(
_ profile: Profile,
_ context: NSManagedObjectContext,
registry: Registry,
coder: ProfileCoder
) throws -> CDProfileV3 {
let encoded = try registry.encodedProfile(profile, with: coder)
let cdProfile = CDProfileV3(context: context)
cdProfile.uuid = profile.id
cdProfile.name = profile.name
cdProfile.encoded = encoded
// redundant but convenient
let attributes = profile.attributes
cdProfile.isAvailableForTV = attributes.isAvailableForTV.map(NSNumber.init(value:))
cdProfile.expirationDate = attributes.expirationDate
cdProfile.lastUpdate = attributes.lastUpdate
cdProfile.fingerprint = attributes.fingerprint
return cdProfile
}
}
// MARK: - Specialization
extension CDProfileV3: CoreDataUniqueEntity {
}
extension Profile: UniqueEntity {
public var uuid: UUID? {
id
}
}
extension CoreDataRepository: ProfileRepository where T == Profile {
public nonisolated var profilesPublisher: AnyPublisher<[Profile], Never> {
entitiesPublisher
.map(\.entities)
.eraseToAnyPublisher()
}
public func saveProfile(_ profile: Profile) async throws {
try await saveEntities([profile])
}
public func removeProfiles(withIds profileIds: [Profile.ID]) async throws {
try await removeEntities(withIds: profileIds)
}
}