mirror of
https://github.com/passepartoutvpn/passepartout-apple.git
synced 2025-01-12 19:49:06 +00:00
d8c4e87239
Refactoring: - Get receipts from StoreKit Transaction.currentEntitlements - Search for the originally purchased build in the local receipt anyway (Kvitto) - Fall back to release receipt (Kvitto), if any, for feature eligibility in TestFlight builds - Parse and verify expiration date in subscriptions - Decouple in-app identifier composition from BundleConfiguration - Fix user level features only applied when a receipt was not found Testing: - Add StoreKit configuration - Fake purchases with PP_FAKE_IAP - Fake user level with PP_USER_LEVEL Then for reactive receipt reload, detect app activation differently: - iOS/tvOS on .scenePhase - macOS on launch and NSWorkspace.didActivateApplicationNotification As to features: - Credit former "Full version" purchasers with all current AND future features, except the Apple TV
108 lines
2.8 KiB
Swift
108 lines
2.8 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
|
|
}
|
|
|
|
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"
|
|
}
|
|
}
|