passepartout-apple/Library/Sources/CommonLibrary/Domain/BundleConfiguration+Main.swift
Davide ffb8829f4f
Reorganize Core Data containers (#1017)
Before anything, remove any code related to App Group containers from
tvOS target because they are not available. Include the beta receipt
override, it's broken for that reason.

In short:

- Store all Core Data containers locally. Do not use the App Group for
Core Data for consistency across platforms.
- Store logs in the App Group on iOS/macOS, but locally on tvOS (see
`urlForCaches`).

Then, rather than one container per model, merge models into:

- Local: Providers
- Remote: Profiles + Preferences (now in the same CloudKit container)

Reuse the remote model for backups too.

This change is safe because:

- Local profiles are stored via Network Extension in the keychain, not
Core Data
- Remote profiles are re-imported via CloudKit sync
- Providers are re-downloaded on first use
- Preferences are lost, but they are "cheap" data
- Profile backups are lost, but they were hidden anyway
2024-12-15 20:20:33 +01:00

110 lines
2.9 KiB
Swift

//
// BundleConfiguration+Main.swift
// Passepartout
//
// Created by Davide De Rosa on 7/1/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 Foundation
import PassepartoutKit
// WARNING: beware of Constants.shared dependency
extension BundleConfiguration {
public enum BundleKey: String {
case appStoreId
case cloudKitId
case userLevel
case groupId
case iapBundlePrefix
case keychainGroupId
case loginItemId
case tunnelId
// legacy v2
case legacyV2CloudKitId
case legacyV2TVCloudKitId
}
public static var mainDisplayName: String {
if isPreview {
return "preview-display-name"
}
return main.displayName
}
public static var mainVersionString: String {
if isPreview {
return "preview-1.2.3"
}
return main.versionString
}
public static func mainString(for key: BundleKey) -> String {
if isPreview {
return "preview-key(\(key.rawValue))"
}
guard let value: String = main.value(forKey: key.rawValue) else {
fatalError("Missing main bundle key: \(key.rawValue)")
}
return value
}
public static func mainIntegerIfPresent(for key: BundleKey) -> Int? {
if isPreview {
return nil
}
return main.value(forKey: key.rawValue)
}
public static var urlForReview: URL {
let appStoreId = mainString(for: .appStoreId)
guard let url = URL(string: "https://apps.apple.com/app/id\(appStoreId)?action=write-review") else {
fatalError("Unable to build urlForReview")
}
return url
}
}
private extension BundleConfiguration {
// WARNING: fails from package itself, e.g. in previews
static var main: BundleConfiguration {
guard let bundle = BundleConfiguration(.main, key: Constants.shared.bundleKey) else {
fatalError("Missing main bundle")
}
return bundle
}
static var isPreview: Bool {
ProcessInfo.processInfo.environment["XCODE_RUNNING_FOR_PREVIEWS"] == "1"
}
}