passepartout-apple/Passepartout/Library/Sources/CommonLibrary/Mock/MockAppProductHelper.swift
Davide 008b78cc7c
Upgrade to Xcode 16 (#858)
- Address further restrictions on actor-isolation by using `nonisolated`
on:
  - Combine subjects
  - Core Data context/controller
  - Blocks
- In previews using inline `@State`, create a custom view instead
- Use `@retroactive` in l10n extensions
- Fix compile error in WireGuardKit
2024-11-13 12:07:30 +01:00

79 lines
2.4 KiB
Swift

//
// MockAppProductHelper.swift
// Passepartout
//
// Created by Davide De Rosa on 12/19/23.
// 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 Combine
import CommonUtils
import Foundation
public actor MockAppProductHelper: AppProductHelper {
private let build: Int
public private(set) var products: [AppProduct: InAppProduct]
public nonisolated let receiptReader: MockAppReceiptReader
private nonisolated let didUpdateSubject: PassthroughSubject<Void, Never>
// set .max to skip entitled products
public init(build: Int = .max) {
self.build = build
products = [:]
receiptReader = MockAppReceiptReader()
didUpdateSubject = PassthroughSubject()
}
public nonisolated var canMakePurchases: Bool {
true
}
public nonisolated var didUpdate: AnyPublisher<Void, Never> {
didUpdateSubject.eraseToAnyPublisher()
}
public func fetchProducts() async throws -> [AppProduct: InAppProduct] {
products = AppProduct.all.reduce(into: [:]) {
$0[$1] = InAppProduct(
productIdentifier: $1.rawValue,
localizedTitle: $1.rawValue,
localizedPrice: "€10.0",
native: $1
)
}
await receiptReader.setReceipt(withBuild: build, identifiers: [])
didUpdateSubject.send()
return products
}
public func purchase(_ inAppProduct: InAppProduct) async throws -> InAppPurchaseResult {
await receiptReader.addPurchase(with: inAppProduct.productIdentifier)
didUpdateSubject.send()
return .done
}
public func restorePurchases() async throws {
didUpdateSubject.send()
}
}