mirror of
https://github.com/passepartoutvpn/passepartout-apple.git
synced 2024-12-24 02:21:00 +00:00
e761979134
Especially for flaky tests: - Do objectWillChange.send() _before_ performing the change - Send more events to .didChange to be more deterministic about test expectations
464 lines
16 KiB
Swift
464 lines
16 KiB
Swift
//
|
|
// IAPManagerTests.swift
|
|
// Passepartout
|
|
//
|
|
// Created by Davide De Rosa on 9/12/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 Combine
|
|
@testable import CommonIAP
|
|
@testable import CommonLibrary
|
|
import CommonUtils
|
|
import Foundation
|
|
import XCTest
|
|
|
|
@MainActor
|
|
final class IAPManagerTests: XCTestCase {
|
|
private let olderBuildNumber = 500
|
|
|
|
private let defaultBuildNumber = 1000
|
|
|
|
private let newerBuildNumber = 1500
|
|
|
|
private var subscriptions: Set<AnyCancellable> = []
|
|
}
|
|
|
|
// MARK: - Actions
|
|
|
|
extension IAPManagerTests {
|
|
func test_givenProducts_whenFetchAppProducts_thenReturnsCorrespondingInAppProducts() async throws {
|
|
let reader = FakeAppReceiptReader()
|
|
let sut = IAPManager(receiptReader: reader)
|
|
|
|
let appProducts: [AppProduct] = [
|
|
.Full.iOS,
|
|
.Donations.huge
|
|
]
|
|
let inAppProducts = try await sut.purchasableProducts(for: appProducts)
|
|
inAppProducts.enumerated().forEach {
|
|
XCTAssertEqual($0.element.productIdentifier, appProducts[$0.offset].rawValue)
|
|
}
|
|
}
|
|
|
|
func test_givenProducts_whenPurchase_thenIsAddedToPurchasedProducts() async throws {
|
|
let reader = FakeAppReceiptReader()
|
|
await reader.setReceipt(withBuild: .max, products: [])
|
|
let sut = IAPManager(receiptReader: reader)
|
|
|
|
let appleTV: AppProduct = .Features.appleTV
|
|
XCTAssertFalse(sut.purchasedProducts.contains(appleTV))
|
|
do {
|
|
let purchasable = try await sut.purchasableProducts(for: [appleTV])
|
|
let purchasableAppleTV = try XCTUnwrap(purchasable.first)
|
|
let result = try await sut.purchase(purchasableAppleTV)
|
|
if result == .done {
|
|
XCTAssertTrue(sut.purchasedProducts.contains(appleTV))
|
|
} else {
|
|
XCTFail("Unexpected purchase() result: \(result)")
|
|
}
|
|
} catch {
|
|
XCTFail("Unexpected purchase() failure: \(error)")
|
|
}
|
|
}
|
|
}
|
|
|
|
// MARK: - Build products
|
|
|
|
extension IAPManagerTests {
|
|
func test_givenBuildProducts_whenOlder_thenFullVersion() async {
|
|
let reader = FakeAppReceiptReader()
|
|
await reader.setReceipt(withBuild: olderBuildNumber, identifiers: [])
|
|
let sut = IAPManager(receiptReader: reader) { build in
|
|
if build <= self.defaultBuildNumber {
|
|
return [.Full.allPlatforms]
|
|
}
|
|
return []
|
|
}
|
|
await sut.reloadReceipt()
|
|
XCTAssertTrue(sut.isEligible(for: AppFeature.fullV2Features))
|
|
}
|
|
|
|
func test_givenBuildProducts_whenNewer_thenFreeVersion() async {
|
|
let reader = FakeAppReceiptReader()
|
|
await reader.setReceipt(withBuild: newerBuildNumber, products: [])
|
|
let sut = IAPManager(receiptReader: reader) { build in
|
|
if build <= self.defaultBuildNumber {
|
|
return [.Full.allPlatforms]
|
|
}
|
|
return []
|
|
}
|
|
await sut.reloadReceipt()
|
|
XCTAssertFalse(sut.isEligible(for: AppFeature.fullV2Features))
|
|
}
|
|
}
|
|
|
|
// MARK: - Eligibility
|
|
|
|
extension IAPManagerTests {
|
|
func test_givenPurchasedFeature_whenReloadReceipt_thenIsEligible() async {
|
|
let reader = FakeAppReceiptReader()
|
|
let sut = IAPManager(receiptReader: reader)
|
|
|
|
XCTAssertFalse(sut.isEligible(for: AppFeature.fullV2Features))
|
|
|
|
await reader.setReceipt(withBuild: defaultBuildNumber, products: [.Full.allPlatforms])
|
|
XCTAssertFalse(sut.isEligible(for: AppFeature.fullV2Features))
|
|
|
|
await sut.reloadReceipt()
|
|
XCTAssertTrue(sut.isEligible(for: AppFeature.fullV2Features))
|
|
}
|
|
|
|
func test_givenPurchasedFeatures_thenIsOnlyEligibleForFeatures() async {
|
|
let reader = FakeAppReceiptReader()
|
|
await reader.setReceipt(withBuild: defaultBuildNumber, products: [
|
|
.Features.networkSettings
|
|
])
|
|
let sut = IAPManager(receiptReader: reader)
|
|
|
|
await sut.reloadReceipt()
|
|
XCTAssertTrue(sut.isEligible(for: .dns))
|
|
XCTAssertTrue(sut.isEligible(for: .httpProxy))
|
|
XCTAssertFalse(sut.isEligible(for: .onDemand))
|
|
XCTAssertTrue(sut.isEligible(for: .routing))
|
|
XCTAssertFalse(sut.isEligible(for: .sharing))
|
|
XCTAssertFalse(sut.isEligible(for: AppFeature.fullV2Features))
|
|
}
|
|
|
|
func test_givenPurchasedAndCancelledFeature_thenIsNotEligible() async {
|
|
let reader = FakeAppReceiptReader()
|
|
await reader.setReceipt(
|
|
withBuild: defaultBuildNumber,
|
|
products: [.Full.allPlatforms],
|
|
cancelledProducts: [.Full.allPlatforms]
|
|
)
|
|
let sut = IAPManager(receiptReader: reader)
|
|
|
|
await sut.reloadReceipt()
|
|
XCTAssertFalse(sut.isEligible(for: AppFeature.fullV2Features))
|
|
}
|
|
|
|
func test_givenFreeVersion_thenIsNotEligibleForAnyFeature() async {
|
|
let reader = FakeAppReceiptReader()
|
|
await reader.setReceipt(withBuild: defaultBuildNumber, products: [])
|
|
let sut = IAPManager(receiptReader: reader)
|
|
|
|
await sut.reloadReceipt()
|
|
XCTAssertFalse(sut.userLevel.isFullVersion)
|
|
AppFeature.fullV2Features.forEach {
|
|
XCTAssertFalse(sut.isEligible(for: $0))
|
|
}
|
|
}
|
|
|
|
func test_givenFreeVersion_thenIsNotEligibleForAppleTV() async {
|
|
let reader = FakeAppReceiptReader()
|
|
await reader.setReceipt(withBuild: defaultBuildNumber, products: [])
|
|
let sut = IAPManager(receiptReader: reader)
|
|
|
|
await sut.reloadReceipt()
|
|
XCTAssertFalse(sut.isEligible(for: .appleTV))
|
|
}
|
|
|
|
func test_givenFullV2Version_thenIsEligibleForAnyFeatureExceptExcluded() async {
|
|
let reader = FakeAppReceiptReader()
|
|
await reader.setReceipt(withBuild: defaultBuildNumber, products: [.Full.allPlatforms])
|
|
let sut = IAPManager(receiptReader: reader)
|
|
|
|
await sut.reloadReceipt()
|
|
let excluded: Set<AppFeature> = [
|
|
.appleTV,
|
|
.interactiveLogin
|
|
]
|
|
AppFeature.allCases.forEach {
|
|
if AppFeature.fullV2Features.contains($0) {
|
|
XCTAssertTrue(sut.isEligible(for: $0))
|
|
} else {
|
|
XCTAssertTrue(excluded.contains($0))
|
|
XCTAssertFalse(sut.isEligible(for: $0))
|
|
}
|
|
}
|
|
}
|
|
|
|
func test_givenAppleTV_thenIsEligibleForAppleTVAndSharing() async {
|
|
let reader = FakeAppReceiptReader()
|
|
await reader.setReceipt(withBuild: defaultBuildNumber, products: [.Features.appleTV])
|
|
let sut = IAPManager(receiptReader: reader)
|
|
|
|
await sut.reloadReceipt()
|
|
XCTAssertTrue(sut.isEligible(for: .appleTV))
|
|
XCTAssertTrue(sut.isEligible(for: .sharing))
|
|
}
|
|
|
|
func test_givenPlatformVersion_thenIsFullVersionForPlatform() async {
|
|
let reader = FakeAppReceiptReader()
|
|
let sut = IAPManager(receiptReader: reader)
|
|
|
|
#if os(macOS)
|
|
await reader.setReceipt(withBuild: defaultBuildNumber, products: [.Full.macOS, .Features.networkSettings])
|
|
await sut.reloadReceipt()
|
|
XCTAssertTrue(sut.isEligible(for: AppFeature.fullV2Features))
|
|
#else
|
|
await reader.setReceipt(withBuild: defaultBuildNumber, products: [.Full.iOS, .Features.networkSettings])
|
|
await sut.reloadReceipt()
|
|
XCTAssertTrue(sut.isEligible(for: AppFeature.fullV2Features))
|
|
#endif
|
|
}
|
|
|
|
func test_givenPlatformVersion_thenIsNotFullVersionForOtherPlatform() async {
|
|
let reader = FakeAppReceiptReader()
|
|
let sut = IAPManager(receiptReader: reader)
|
|
|
|
#if os(macOS)
|
|
await reader.setReceipt(withBuild: defaultBuildNumber, products: [.Full.iOS, .Features.networkSettings])
|
|
await sut.reloadReceipt()
|
|
XCTAssertFalse(sut.isEligible(for: AppFeature.fullV2Features))
|
|
#else
|
|
await reader.setReceipt(withBuild: defaultBuildNumber, products: [.Full.macOS, .Features.networkSettings])
|
|
await sut.reloadReceipt()
|
|
XCTAssertFalse(sut.isEligible(for: AppFeature.fullV2Features))
|
|
#endif
|
|
}
|
|
|
|
func test_givenUser_thenIsNotEligibleForFeedback() async {
|
|
let reader = FakeAppReceiptReader()
|
|
let sut = IAPManager(receiptReader: reader)
|
|
XCTAssertFalse(sut.isEligibleForFeedback())
|
|
}
|
|
|
|
func test_givenBeta_thenIsEligibleForFeedback() async {
|
|
let reader = FakeAppReceiptReader()
|
|
await reader.setReceipt(withBuild: .max, identifiers: [])
|
|
let sut = IAPManager(customUserLevel: .beta, receiptReader: reader)
|
|
await sut.reloadReceipt()
|
|
XCTAssertTrue(sut.isEligibleForFeedback())
|
|
}
|
|
|
|
func test_givenPayingUser_thenIsEligibleForFeedback() async {
|
|
let reader = FakeAppReceiptReader()
|
|
await reader.setReceipt(withBuild: .max, products: [.Full.iOS])
|
|
let sut = IAPManager(receiptReader: reader)
|
|
await sut.reloadReceipt()
|
|
XCTAssertTrue(sut.isEligibleForFeedback())
|
|
}
|
|
}
|
|
|
|
// MARK: - App level
|
|
|
|
extension IAPManagerTests {
|
|
func test_givenBetaApp_thenIsRestricted() async {
|
|
let reader = FakeAppReceiptReader()
|
|
let sut = IAPManager(customUserLevel: .beta, receiptReader: reader)
|
|
|
|
await sut.reloadReceipt()
|
|
XCTAssertTrue(sut.isRestricted)
|
|
XCTAssertTrue(sut.userLevel.isRestricted)
|
|
}
|
|
|
|
func test_givenBetaApp_thenIsNotEligibleForAllFeatures() async {
|
|
let reader = FakeAppReceiptReader()
|
|
let sut = IAPManager(customUserLevel: .beta, receiptReader: reader)
|
|
|
|
await sut.reloadReceipt()
|
|
XCTAssertFalse(sut.isEligible(for: AppFeature.allCases))
|
|
}
|
|
|
|
func test_givenBetaApp_thenIsEligibleForUserLevelFeatures() async {
|
|
let reader = FakeAppReceiptReader()
|
|
let sut = IAPManager(customUserLevel: .beta, receiptReader: reader)
|
|
|
|
let eligible = AppUserLevel.beta.features
|
|
|
|
await sut.reloadReceipt()
|
|
XCTAssertTrue(sut.isEligible(for: eligible))
|
|
}
|
|
|
|
func test_givenBetaApp_thenIsEligibleForUnrestrictedFeature() async {
|
|
let reader = FakeAppReceiptReader()
|
|
let sut = IAPManager(customUserLevel: .beta, receiptReader: reader, unrestrictedFeatures: [.onDemand])
|
|
|
|
var eligible = AppUserLevel.beta.features
|
|
eligible.append(.onDemand)
|
|
|
|
await sut.reloadReceipt()
|
|
XCTAssertTrue(sut.isEligible(for: eligible))
|
|
}
|
|
|
|
func test_givenFullV2App_thenIsFullVersion() async {
|
|
let reader = FakeAppReceiptReader()
|
|
let sut = IAPManager(customUserLevel: .fullV2, receiptReader: reader)
|
|
|
|
await sut.reloadReceipt()
|
|
XCTAssertTrue(sut.userLevel.isFullVersion)
|
|
}
|
|
|
|
func test_givenSubscriberApp_thenIsFullVersion() async {
|
|
let reader = FakeAppReceiptReader()
|
|
let sut = IAPManager(customUserLevel: .subscriber, receiptReader: reader)
|
|
|
|
await sut.reloadReceipt()
|
|
XCTAssertTrue(sut.userLevel.isFullVersion)
|
|
}
|
|
|
|
func test_givenFullV2App_thenIsEligibleForAnyFeatureExceptExcluded() async {
|
|
let reader = FakeAppReceiptReader()
|
|
let sut = IAPManager(customUserLevel: .fullV2, receiptReader: reader)
|
|
|
|
await sut.reloadReceipt()
|
|
let excluded: Set<AppFeature> = [
|
|
.appleTV,
|
|
.interactiveLogin
|
|
]
|
|
AppFeature.allCases.forEach {
|
|
if AppFeature.fullV2Features.contains($0) {
|
|
XCTAssertTrue(sut.isEligible(for: $0))
|
|
} else {
|
|
XCTAssertTrue(excluded.contains($0))
|
|
XCTAssertFalse(sut.isEligible(for: $0))
|
|
}
|
|
}
|
|
}
|
|
|
|
func test_givenSubscriberApp_thenIsEligibleForAnyFeature() async {
|
|
let reader = FakeAppReceiptReader()
|
|
let sut = IAPManager(customUserLevel: .subscriber, receiptReader: reader)
|
|
|
|
await sut.reloadReceipt()
|
|
AppFeature.fullV2Features.forEach {
|
|
XCTAssertTrue(sut.isEligible(for: $0))
|
|
}
|
|
XCTAssertTrue(sut.isEligible(for: .appleTV))
|
|
}
|
|
}
|
|
|
|
// MARK: - Beta
|
|
|
|
extension IAPManagerTests {
|
|
func test_givenChecker_whenReloadReceipt_thenIsBeta() async {
|
|
let betaChecker = MockBetaChecker()
|
|
betaChecker.isBeta = true
|
|
let sut = IAPManager(receiptReader: FakeAppReceiptReader(), betaChecker: betaChecker)
|
|
XCTAssertEqual(sut.userLevel, .undefined)
|
|
await sut.reloadReceipt()
|
|
XCTAssertEqual(sut.userLevel, .beta)
|
|
}
|
|
}
|
|
|
|
// MARK: - Receipts
|
|
|
|
extension IAPManagerTests {
|
|
func test_givenReceipts_whenReloadReceipt_thenPublishesEligibleFeatures() async {
|
|
let reader = FakeAppReceiptReader()
|
|
await reader.setReceipt(withBuild: .max, products: [
|
|
.Features.appleTV,
|
|
.Features.trustedNetworks
|
|
])
|
|
let sut = IAPManager(receiptReader: reader)
|
|
|
|
let exp = expectation(description: "Eligible features")
|
|
sut
|
|
.$eligibleFeatures
|
|
.dropFirst()
|
|
.sink { _ in
|
|
exp.fulfill()
|
|
}
|
|
.store(in: &subscriptions)
|
|
|
|
await sut.reloadReceipt()
|
|
await fulfillment(of: [exp], timeout: CommonLibraryTests.timeout)
|
|
|
|
XCTAssertEqual(sut.eligibleFeatures, [
|
|
.appleTV,
|
|
.onDemand,
|
|
.sharing // implied by Apple TV purchase
|
|
])
|
|
}
|
|
|
|
func test_givenInvalidReceipts_whenReloadReceipt_thenSkipsInvalid() async {
|
|
let reader = FakeAppReceiptReader()
|
|
await reader.setReceipt(withBuild: .max, products: [])
|
|
await reader.addPurchase(with: "foobar")
|
|
await reader.addPurchase(with: .Features.allProviders, expirationDate: Date().addingTimeInterval(-10))
|
|
await reader.addPurchase(with: .Features.appleTV)
|
|
await reader.addPurchase(with: .Features.networkSettings, expirationDate: Date().addingTimeInterval(10))
|
|
await reader.addPurchase(with: .Full.iOS, cancellationDate: Date().addingTimeInterval(-60))
|
|
|
|
let sut = IAPManager(receiptReader: reader)
|
|
await sut.reloadReceipt()
|
|
|
|
XCTAssertEqual(sut.eligibleFeatures, [
|
|
.appleTV,
|
|
.dns,
|
|
.httpProxy,
|
|
.routing,
|
|
.sharing
|
|
])
|
|
}
|
|
}
|
|
|
|
// MARK: - Observation
|
|
|
|
extension IAPManagerTests {
|
|
func test_givenManager_whenObserveObjects_thenReloadsReceipt() async {
|
|
let reader = FakeAppReceiptReader()
|
|
await reader.setReceipt(withBuild: .max, products: [.Full.allPlatforms])
|
|
let sut = IAPManager(receiptReader: reader)
|
|
|
|
XCTAssertEqual(sut.userLevel, .undefined)
|
|
XCTAssertTrue(sut.eligibleFeatures.isEmpty)
|
|
|
|
let exp = expectation(description: "Reload receipt")
|
|
sut
|
|
.$eligibleFeatures
|
|
.dropFirst()
|
|
.sink { _ in
|
|
exp.fulfill()
|
|
}
|
|
.store(in: &subscriptions)
|
|
|
|
sut.observeObjects()
|
|
await fulfillment(of: [exp], timeout: CommonLibraryTests.timeout)
|
|
|
|
XCTAssertNotEqual(sut.userLevel, .undefined)
|
|
XCTAssertFalse(sut.eligibleFeatures.isEmpty)
|
|
}
|
|
}
|
|
|
|
// MARK: -
|
|
|
|
private extension IAPManager {
|
|
convenience init(
|
|
customUserLevel: AppUserLevel? = nil,
|
|
inAppHelper: (any AppProductHelper)? = nil,
|
|
receiptReader: AppReceiptReader,
|
|
betaChecker: BetaChecker? = nil,
|
|
unrestrictedFeatures: Set<AppFeature> = [],
|
|
productsAtBuild: BuildProducts<AppProduct>? = nil
|
|
) {
|
|
self.init(
|
|
customUserLevel: customUserLevel,
|
|
inAppHelper: inAppHelper ?? FakeAppProductHelper(),
|
|
receiptReader: receiptReader,
|
|
betaChecker: betaChecker ?? TestFlightChecker(),
|
|
unrestrictedFeatures: unrestrictedFeatures,
|
|
productsAtBuild: productsAtBuild
|
|
)
|
|
}
|
|
}
|