From 805a19435879767d37106cf3ee1ece0244388f57 Mon Sep 17 00:00:00 2001 From: Francesco Paolo Severino Date: Fri, 11 Oct 2024 14:10:06 +0200 Subject: [PATCH 1/7] Update to Swift Testing for PassesTests --- .github/workflows/test.yml | 11 +- .swift-format | 2 +- Package.swift | 6 +- Sources/Orders/OrdersService.swift | 3 +- .../PassKit/Testing}/SecretMiddleware.swift | 10 +- .../PassKit/Testing/isLoggingConfigured.swift | 10 + Sources/Passes/PassesService.swift | 3 +- Tests/PassesTests/EncryptedPassesTests.swift | 278 ++--- Tests/PassesTests/PassesTests.swift | 1011 ++++++++--------- Tests/PassesTests/SecretMiddleware.swift | 14 - Tests/PassesTests/withApp.swift | 38 + 11 files changed, 661 insertions(+), 725 deletions(-) rename {Tests/OrdersTests => Sources/PassKit/Testing}/SecretMiddleware.swift (52%) create mode 100644 Sources/PassKit/Testing/isLoggingConfigured.swift delete mode 100644 Tests/PassesTests/SecretMiddleware.swift create mode 100644 Tests/PassesTests/withApp.swift diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml index c69731e..2054aea 100644 --- a/.github/workflows/test.yml +++ b/.github/workflows/test.yml @@ -7,16 +7,9 @@ on: push: { branches: [ main ] } jobs: - lint: - runs-on: ubuntu-latest - container: swift:noble - steps: - - name: Check out PassKit - uses: actions/checkout@v4 - - name: Run format lint check - run: swift format lint --strict --recursive --parallel . - unit-tests: uses: vapor/ci/.github/workflows/run-unit-tests.yml@main + with: + with_linting: true secrets: CODECOV_TOKEN: ${{ secrets.CODECOV_TOKEN }} diff --git a/.swift-format b/.swift-format index 360ca2c..47901d1 100644 --- a/.swift-format +++ b/.swift-format @@ -11,7 +11,7 @@ "lineBreakBeforeControlFlowKeywords": false, "lineBreakBeforeEachArgument": false, "lineBreakBeforeEachGenericRequirement": false, - "lineLength": 100, + "lineLength": 140, "maximumBlankLines": 1, "multiElementCollectionTrailingCommas": true, "noAssignmentInExpressions": { diff --git a/Package.swift b/Package.swift index cdefd8f..4ce018b 100644 --- a/Package.swift +++ b/Package.swift @@ -11,13 +11,13 @@ let package = Package( .library(name: "Orders", targets: ["Orders"]), ], dependencies: [ - .package(url: "https://github.com/vapor/vapor.git", from: "4.105.2"), - .package(url: "https://github.com/vapor/fluent.git", from: "4.11.0"), + .package(url: "https://github.com/vapor/vapor.git", from: "4.106.0"), + .package(url: "https://github.com/vapor/fluent.git", from: "4.12.0"), .package(url: "https://github.com/vapor/apns.git", from: "4.2.0"), .package(url: "https://github.com/vapor-community/Zip.git", from: "2.2.3"), .package(url: "https://github.com/apple/swift-certificates.git", from: "1.5.0"), // used in tests - .package(url: "https://github.com/vapor/fluent-sqlite-driver.git", from: "4.7.4"), + .package(url: "https://github.com/vapor/fluent-sqlite-driver.git", from: "4.8.0"), ], targets: [ .target( diff --git a/Sources/Orders/OrdersService.swift b/Sources/Orders/OrdersService.swift index 3cc897f..2577513 100644 --- a/Sources/Orders/OrdersService.swift +++ b/Sources/Orders/OrdersService.swift @@ -10,8 +10,7 @@ import Vapor /// The main class that handles Wallet orders. public final class OrdersService: Sendable { - private let service: - OrdersServiceCustom + private let service: OrdersServiceCustom /// Initializes the service and registers all the routes required for Apple Wallet to work. /// diff --git a/Tests/OrdersTests/SecretMiddleware.swift b/Sources/PassKit/Testing/SecretMiddleware.swift similarity index 52% rename from Tests/OrdersTests/SecretMiddleware.swift rename to Sources/PassKit/Testing/SecretMiddleware.swift index fef1940..ec9d64f 100644 --- a/Tests/OrdersTests/SecretMiddleware.swift +++ b/Sources/PassKit/Testing/SecretMiddleware.swift @@ -1,11 +1,13 @@ import Vapor -struct SecretMiddleware: AsyncMiddleware { +package struct SecretMiddleware: AsyncMiddleware { let secret: String - func respond( - to request: Request, chainingTo next: any AsyncResponder - ) async throws -> Response { + package init(secret: String) { + self.secret = secret + } + + package func respond(to request: Request, chainingTo next: any AsyncResponder) async throws -> Response { guard request.headers.first(name: "X-Secret") == secret else { throw Abort(.unauthorized, reason: "Incorrect X-Secret header.") } diff --git a/Sources/PassKit/Testing/isLoggingConfigured.swift b/Sources/PassKit/Testing/isLoggingConfigured.swift new file mode 100644 index 0000000..ba27a1f --- /dev/null +++ b/Sources/PassKit/Testing/isLoggingConfigured.swift @@ -0,0 +1,10 @@ +import Vapor + +package let isLoggingConfigured: Bool = { + LoggingSystem.bootstrap { label in + var handler = StreamLogHandler.standardOutput(label: label) + handler.logLevel = .debug + return handler + } + return true +}() diff --git a/Sources/Passes/PassesService.swift b/Sources/Passes/PassesService.swift index 6151d48..4fa593a 100644 --- a/Sources/Passes/PassesService.swift +++ b/Sources/Passes/PassesService.swift @@ -72,8 +72,7 @@ public final class PassesService: Sendable { /// - passes: The passes to include in the bundle. /// - db: The `Database` to use. /// - Returns: The bundle of passes as `Data`. - public func generatePassesContent(for passes: [Pass], on db: any Database) async throws -> Data - { + public func generatePassesContent(for passes: [Pass], on db: any Database) async throws -> Data { try await service.generatePassesContent(for: passes, on: db) } diff --git a/Tests/PassesTests/EncryptedPassesTests.swift b/Tests/PassesTests/EncryptedPassesTests.swift index 971f850..af17d4f 100644 --- a/Tests/PassesTests/EncryptedPassesTests.swift +++ b/Tests/PassesTests/EncryptedPassesTests.swift @@ -1,177 +1,135 @@ -import Fluent -import FluentSQLiteDriver import PassKit +import Testing import XCTVapor import Zip @testable import Passes -final class EncryptedPassesTests: XCTestCase { +struct EncryptedPassesTests { let delegate = EncryptedPassesDelegate() let passesURI = "/api/passes/v1/" - var passesService: PassesService! - var app: Application! - - override func setUp() async throws { - self.app = try await Application.make(.testing) - app.databases.use(.sqlite(.memory), as: .sqlite) - - PassesService.register(migrations: app.migrations) - app.migrations.add(CreatePassData()) - passesService = try PassesService( - app: app, - delegate: delegate, - pushRoutesMiddleware: SecretMiddleware(secret: "foo"), - logger: app.logger - ) - app.databases.middleware.use(PassDataMiddleware(service: passesService), on: .sqlite) - - try await app.autoMigrate() - - Zip.addCustomFileExtension("pkpass") - } - override func tearDown() async throws { - try await app.autoRevert() - try await self.app.asyncShutdown() - self.app = nil + @Test func passGeneration() async throws { + try await withApp(delegate: delegate) { app, passesService in + let passData = PassData(title: "Test Pass") + try await passData.create(on: app.db) + let pass = try await passData.$pass.get(on: app.db) + let data = try await passesService.generatePassContent(for: pass, on: app.db) + let passURL = FileManager.default.temporaryDirectory.appendingPathComponent("test.pkpass") + try data.write(to: passURL) + let passFolder = try Zip.quickUnzipFile(passURL) + + #expect(FileManager.default.fileExists(atPath: passFolder.path.appending("/signature"))) + + let passJSONData = try String(contentsOfFile: passFolder.path.appending("/pass.json")).data(using: .utf8) + let passJSON = try JSONSerialization.jsonObject(with: passJSONData!) as! [String: Any] + #expect(passJSON["authenticationToken"] as? String == pass.authenticationToken) + let passID = try pass.requireID().uuidString + #expect(passJSON["serialNumber"] as? String == passID) + #expect(passJSON["description"] as? String == passData.title) + + let manifestJSONData = try String(contentsOfFile: passFolder.path.appending("/manifest.json")).data(using: .utf8) + let manifestJSON = try JSONSerialization.jsonObject(with: manifestJSONData!) as! [String: Any] + let iconData = try Data(contentsOf: passFolder.appendingPathComponent("/icon.png")) + let iconHash = Array(Insecure.SHA1.hash(data: iconData)).hex + #expect(manifestJSON["icon.png"] as? String == iconHash) + } } - func testPassGeneration() async throws { - let passData = PassData(title: "Test Pass") - try await passData.create(on: app.db) - let pass = try await passData.$pass.get(on: app.db) - let data = try await passesService.generatePassContent(for: pass, on: app.db) - let passURL = FileManager.default.temporaryDirectory.appendingPathComponent("test.pkpass") - try data.write(to: passURL) - let passFolder = try Zip.quickUnzipFile(passURL) - - XCTAssert(FileManager.default.fileExists(atPath: passFolder.path.appending("/signature"))) - - let passJSONData = try String(contentsOfFile: passFolder.path.appending("/pass.json")).data( - using: .utf8) - let passJSON = try JSONSerialization.jsonObject(with: passJSONData!) as! [String: Any] - XCTAssertEqual(passJSON["authenticationToken"] as? String, pass.authenticationToken) - try XCTAssertEqual(passJSON["serialNumber"] as? String, pass.requireID().uuidString) - XCTAssertEqual(passJSON["description"] as? String, passData.title) - - let manifestJSONData = try String( - contentsOfFile: passFolder.path.appending("/manifest.json") - ).data(using: .utf8) - let manifestJSON = - try JSONSerialization.jsonObject(with: manifestJSONData!) as! [String: Any] - let iconData = try Data(contentsOf: passFolder.appendingPathComponent("/icon.png")) - let iconHash = Array(Insecure.SHA1.hash(data: iconData)).hex - XCTAssertEqual(manifestJSON["icon.png"] as? String, iconHash) - } + @Test func personalizationAPI() async throws { + try await withApp(delegate: delegate) { app, passesService in + let passData = PassData(title: "Personalize") + try await passData.create(on: app.db) + let pass = try await passData.$pass.get(on: app.db) + let personalizationDict = PersonalizationDictionaryDTO( + personalizationToken: "1234567890", + requiredPersonalizationInfo: .init( + emailAddress: "test@example.com", + familyName: "Doe", + fullName: "John Doe", + givenName: "John", + isoCountryCode: "US", + phoneNumber: "1234567890", + postalCode: "12345" + ) + ) - func testPersonalizationAPI() async throws { - let passData = PassData(title: "Personalize") - try await passData.create(on: app.db) - let pass = try await passData.$pass.get(on: app.db) - let personalizationDict = PersonalizationDictionaryDTO( - personalizationToken: "1234567890", - requiredPersonalizationInfo: .init( - emailAddress: "test@example.com", - familyName: "Doe", - fullName: "John Doe", - givenName: "John", - isoCountryCode: "US", - phoneNumber: "1234567890", - postalCode: "12345" + try await app.test( + .POST, + "\(passesURI)passes/\(pass.passTypeIdentifier)/\(pass.requireID())/personalize", + headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], + beforeRequest: { req async throws in + try req.content.encode(personalizationDict) + }, + afterResponse: { res async throws in + #expect(res.status == .ok) + #expect(res.body != nil) + #expect(res.headers.contentType?.description == "application/octet-stream") + } ) - ) - - try await app.test( - .POST, - "\(passesURI)passes/\(pass.passTypeIdentifier)/\(pass.requireID())/personalize", - headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], - beforeRequest: { req async throws in - try req.content.encode(personalizationDict) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .ok) - XCTAssertNotNil(res.body) - XCTAssertEqual(res.headers.contentType?.description, "application/octet-stream") - } - ) - - let personalizationQuery = try await UserPersonalization.query(on: app.db).all() - XCTAssertEqual(personalizationQuery.count, 1) - let passPersonalizationID = try await Pass.query(on: app.db).first()? - ._$userPersonalization.get(on: app.db)? - .requireID() - XCTAssertEqual(personalizationQuery[0]._$id.value, passPersonalizationID) - XCTAssertEqual( - personalizationQuery[0]._$emailAddress.value, - personalizationDict.requiredPersonalizationInfo.emailAddress) - XCTAssertEqual( - personalizationQuery[0]._$familyName.value, - personalizationDict.requiredPersonalizationInfo.familyName) - XCTAssertEqual( - personalizationQuery[0]._$fullName.value, - personalizationDict.requiredPersonalizationInfo.fullName) - XCTAssertEqual( - personalizationQuery[0]._$givenName.value, - personalizationDict.requiredPersonalizationInfo.givenName) - XCTAssertEqual( - personalizationQuery[0]._$isoCountryCode.value, - personalizationDict.requiredPersonalizationInfo.isoCountryCode) - XCTAssertEqual( - personalizationQuery[0]._$phoneNumber.value, - personalizationDict.requiredPersonalizationInfo.phoneNumber) - XCTAssertEqual( - personalizationQuery[0]._$postalCode.value, - personalizationDict.requiredPersonalizationInfo.postalCode) + + let personalizationQuery = try await UserPersonalization.query(on: app.db).all() + #expect(personalizationQuery.count == 1) + let passPersonalizationID = try await Pass.query(on: app.db).first()?._$userPersonalization.get(on: app.db)?.requireID() + #expect(personalizationQuery[0]._$id.value == passPersonalizationID) + #expect(personalizationQuery[0]._$emailAddress.value == personalizationDict.requiredPersonalizationInfo.emailAddress) + #expect(personalizationQuery[0]._$familyName.value == personalizationDict.requiredPersonalizationInfo.familyName) + #expect(personalizationQuery[0]._$fullName.value == personalizationDict.requiredPersonalizationInfo.fullName) + #expect(personalizationQuery[0]._$givenName.value == personalizationDict.requiredPersonalizationInfo.givenName) + #expect(personalizationQuery[0]._$isoCountryCode.value == personalizationDict.requiredPersonalizationInfo.isoCountryCode) + #expect(personalizationQuery[0]._$phoneNumber.value == personalizationDict.requiredPersonalizationInfo.phoneNumber) + #expect(personalizationQuery[0]._$postalCode.value == personalizationDict.requiredPersonalizationInfo.postalCode) + } } - func testAPNSClient() async throws { - XCTAssertNotNil(app.apns.client(.init(string: "passes"))) - - let passData = PassData(title: "Test Pass") - try await passData.create(on: app.db) - let pass = try await passData._$pass.get(on: app.db) - - try await passesService.sendPushNotificationsForPass( - id: pass.requireID(), of: pass.passTypeIdentifier, on: app.db) - - let deviceLibraryIdentifier = "abcdefg" - let pushToken = "1234567890" - - try await app.test( - .POST, - "\(passesURI)push/\(pass.passTypeIdentifier)/\(pass.requireID())", - headers: ["X-Secret": "foo"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .noContent) - } - ) - - try await app.test( - .POST, - "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\(pass.requireID())", - headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], - beforeRequest: { req async throws in - try req.content.encode(RegistrationDTO(pushToken: pushToken)) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .created) - } - ) - - try await app.test( - .POST, - "\(passesURI)push/\(pass.passTypeIdentifier)/\(pass.requireID())", - headers: ["X-Secret": "foo"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .internalServerError) - } - ) - - // Test `PassDataMiddleware` update method - passData.title = "Test Pass 2" - do { - try await passData.update(on: app.db) - } catch {} + @Test func apnsClient() async throws { + try await withApp(delegate: delegate) { app, passesService in + #expect(app.apns.client(.init(string: "passes")) != nil) + + let passData = PassData(title: "Test Pass") + try await passData.create(on: app.db) + let pass = try await passData._$pass.get(on: app.db) + + try await passesService.sendPushNotificationsForPass(id: pass.requireID(), of: pass.passTypeIdentifier, on: app.db) + + let deviceLibraryIdentifier = "abcdefg" + let pushToken = "1234567890" + + try await app.test( + .POST, + "\(passesURI)push/\(pass.passTypeIdentifier)/\(pass.requireID())", + headers: ["X-Secret": "foo"], + afterResponse: { res async throws in + #expect(res.status == .noContent) + } + ) + + try await app.test( + .POST, + "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\(pass.requireID())", + headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], + beforeRequest: { req async throws in + try req.content.encode(RegistrationDTO(pushToken: pushToken)) + }, + afterResponse: { res async throws in + #expect(res.status == .created) + } + ) + + try await app.test( + .POST, + "\(passesURI)push/\(pass.passTypeIdentifier)/\(pass.requireID())", + headers: ["X-Secret": "foo"], + afterResponse: { res async throws in + #expect(res.status == .internalServerError) + } + ) + + // Test `PassDataMiddleware` update method + passData.title = "Test Pass 2" + do { + try await passData.update(on: app.db) + } catch {} + } } } diff --git a/Tests/PassesTests/PassesTests.swift b/Tests/PassesTests/PassesTests.swift index 6720847..55a8b20 100644 --- a/Tests/PassesTests/PassesTests.swift +++ b/Tests/PassesTests/PassesTests.swift @@ -1,570 +1,523 @@ -import Fluent -import FluentSQLiteDriver +import FluentKit import PassKit +import Testing import XCTVapor import Zip @testable import Passes -final class PassesTests: XCTestCase { +struct PassesTests { let delegate = TestPassesDelegate() let passesURI = "/api/passes/v1/" - var passesService: PassesService! - var app: Application! - - override func setUp() async throws { - self.app = try await Application.make(.testing) - app.databases.use(.sqlite(.memory), as: .sqlite) - - PassesService.register(migrations: app.migrations) - app.migrations.add(CreatePassData()) - passesService = try PassesService( - app: app, - delegate: delegate, - pushRoutesMiddleware: SecretMiddleware(secret: "foo"), - logger: app.logger - ) - app.databases.middleware.use(PassDataMiddleware(service: passesService), on: .sqlite) - - try await app.autoMigrate() - - Zip.addCustomFileExtension("pkpass") - } - - override func tearDown() async throws { - try await app.autoRevert() - try await self.app.asyncShutdown() - self.app = nil - } - func testPassGeneration() async throws { - let passData = PassData(title: "Test Pass") - try await passData.create(on: app.db) - let pass = try await passData.$pass.get(on: app.db) - let data = try await passesService.generatePassContent(for: pass, on: app.db) - let passURL = FileManager.default.temporaryDirectory.appendingPathComponent("test.pkpass") - try data.write(to: passURL) - let passFolder = try Zip.quickUnzipFile(passURL) - - XCTAssert(FileManager.default.fileExists(atPath: passFolder.path.appending("/signature"))) - - let passJSONData = try String(contentsOfFile: passFolder.path.appending("/pass.json")).data( - using: .utf8) - let passJSON = try JSONSerialization.jsonObject(with: passJSONData!) as! [String: Any] - XCTAssertEqual(passJSON["authenticationToken"] as? String, pass.authenticationToken) - try XCTAssertEqual(passJSON["serialNumber"] as? String, pass.requireID().uuidString) - XCTAssertEqual(passJSON["description"] as? String, passData.title) - - let manifestJSONData = try String( - contentsOfFile: passFolder.path.appending("/manifest.json") - ).data(using: .utf8) - let manifestJSON = - try JSONSerialization.jsonObject(with: manifestJSONData!) as! [String: Any] - let iconData = try Data(contentsOf: passFolder.appendingPathComponent("/icon.png")) - let iconHash = Array(Insecure.SHA1.hash(data: iconData)).hex - XCTAssertEqual(manifestJSON["icon.png"] as? String, iconHash) - XCTAssertNotNil(manifestJSON["logo.png"]) - XCTAssertNotNil(manifestJSON["personalizationLogo.png"]) + @Test func passGeneration() async throws { + try await withApp(delegate: delegate) { app, passesService in + let passData = PassData(title: "Test Pass") + try await passData.create(on: app.db) + let pass = try await passData.$pass.get(on: app.db) + let data = try await passesService.generatePassContent(for: pass, on: app.db) + let passURL = FileManager.default.temporaryDirectory.appendingPathComponent("test.pkpass") + try data.write(to: passURL) + let passFolder = try Zip.quickUnzipFile(passURL) + + #expect(FileManager.default.fileExists(atPath: passFolder.path.appending("/signature"))) + + let passJSONData = try String(contentsOfFile: passFolder.path.appending("/pass.json")).data(using: .utf8) + let passJSON = try JSONSerialization.jsonObject(with: passJSONData!) as! [String: Any] + #expect(passJSON["authenticationToken"] as? String == pass.authenticationToken) + let passID = try pass.requireID().uuidString + #expect(passJSON["serialNumber"] as? String == passID) + #expect(passJSON["description"] as? String == passData.title) + + let manifestJSONData = try String(contentsOfFile: passFolder.path.appending("/manifest.json")).data(using: .utf8) + let manifestJSON = try JSONSerialization.jsonObject(with: manifestJSONData!) as! [String: Any] + let iconData = try Data(contentsOf: passFolder.appendingPathComponent("/icon.png")) + let iconHash = Array(Insecure.SHA1.hash(data: iconData)).hex + #expect(manifestJSON["icon.png"] as? String == iconHash) + #expect(manifestJSON["logo.png"] != nil) + #expect(manifestJSON["personalizationLogo.png"] != nil) + } } - func testPassesGeneration() async throws { - let passData1 = PassData(title: "Test Pass 1") - try await passData1.create(on: app.db) - let pass1 = try await passData1.$pass.get(on: app.db) + @Test func passesGeneration() async throws { + try await withApp(delegate: delegate) { app, passesService in + let passData1 = PassData(title: "Test Pass 1") + try await passData1.create(on: app.db) + let pass1 = try await passData1.$pass.get(on: app.db) - let passData2 = PassData(title: "Test Pass 2") - try await passData2.create(on: app.db) - let pass2 = try await passData2._$pass.get(on: app.db) + let passData2 = PassData(title: "Test Pass 2") + try await passData2.create(on: app.db) + let pass2 = try await passData2._$pass.get(on: app.db) - let data = try await passesService.generatePassesContent(for: [pass1, pass2], on: app.db) - XCTAssertNotNil(data) + let data = try await passesService.generatePassesContent(for: [pass1, pass2], on: app.db) + #expect(data != nil) - do { - let data = try await passesService.generatePassesContent(for: [pass1], on: app.db) - XCTFail("Expected error, got \(data)") - } catch let error as PassesError { - XCTAssertEqual(error, .invalidNumberOfPasses) + do { + let data = try await passesService.generatePassesContent(for: [pass1], on: app.db) + Issue.record("Expected error, got \(data)") + } catch let error as PassesError { + #expect(error == .invalidNumberOfPasses) + } } } - func testPersonalization() async throws { - let passData = PassData(title: "Personalize") - try await passData.create(on: app.db) - let pass = try await passData.$pass.get(on: app.db) - let data = try await passesService.generatePassContent(for: pass, on: app.db) - let passURL = FileManager.default.temporaryDirectory.appendingPathComponent("test.pkpass") - try data.write(to: passURL) - let passFolder = try Zip.quickUnzipFile(passURL) - - let passJSONData = try String(contentsOfFile: passFolder.path.appending("/pass.json")).data( - using: .utf8) - let passJSON = try JSONSerialization.jsonObject(with: passJSONData!) as! [String: Any] - XCTAssertEqual(passJSON["authenticationToken"] as? String, pass.authenticationToken) - try XCTAssertEqual(passJSON["serialNumber"] as? String, pass.requireID().uuidString) - XCTAssertEqual(passJSON["description"] as? String, passData.title) - - let personalizationJSONData = try String( - contentsOfFile: passFolder.path.appending("/personalization.json") - ).data(using: .utf8) - let personalizationJSON = - try JSONSerialization.jsonObject(with: personalizationJSONData!) as! [String: Any] - XCTAssertEqual(personalizationJSON["description"] as? String, "Hello, World!") - - let manifestJSONData = try String( - contentsOfFile: passFolder.path.appending("/manifest.json") - ).data(using: .utf8) - let manifestJSON = - try JSONSerialization.jsonObject(with: manifestJSONData!) as! [String: Any] - let iconData = try Data( - contentsOf: passFolder.appendingPathComponent("/personalizationLogo.png")) - let iconHash = Array(Insecure.SHA1.hash(data: iconData)).hex - XCTAssertEqual(manifestJSON["personalizationLogo.png"] as? String, iconHash) + @Test func personalization() async throws { + try await withApp(delegate: delegate) { app, passesService in + let passData = PassData(title: "Personalize") + try await passData.create(on: app.db) + let pass = try await passData.$pass.get(on: app.db) + let data = try await passesService.generatePassContent(for: pass, on: app.db) + let passURL = FileManager.default.temporaryDirectory.appendingPathComponent("test.pkpass") + try data.write(to: passURL) + let passFolder = try Zip.quickUnzipFile(passURL) + + let passJSONData = try String(contentsOfFile: passFolder.path.appending("/pass.json")).data(using: .utf8) + let passJSON = try JSONSerialization.jsonObject(with: passJSONData!) as! [String: Any] + #expect(passJSON["authenticationToken"] as? String == pass.authenticationToken) + let passID = try pass.requireID().uuidString + #expect(passJSON["serialNumber"] as? String == passID) + #expect(passJSON["description"] as? String == passData.title) + + let personalizationJSONData = try String(contentsOfFile: passFolder.path.appending("/personalization.json")).data(using: .utf8) + let personalizationJSON = try JSONSerialization.jsonObject(with: personalizationJSONData!) as! [String: Any] + #expect(personalizationJSON["description"] as? String == "Hello, World!") + + let manifestJSONData = try String(contentsOfFile: passFolder.path.appending("/manifest.json")).data(using: .utf8) + let manifestJSON = try JSONSerialization.jsonObject(with: manifestJSONData!) as! [String: Any] + let iconData = try Data(contentsOf: passFolder.appendingPathComponent("/personalizationLogo.png")) + let iconHash = Array(Insecure.SHA1.hash(data: iconData)).hex + #expect(manifestJSON["personalizationLogo.png"] as? String == iconHash) + } } // Tests the API Apple Wallet calls to get passes - func testGetPassFromAPI() async throws { - let passData = PassData(title: "Test Pass") - try await passData.create(on: app.db) - let pass = try await passData.$pass.get(on: app.db) - - try await app.test( - .GET, - "\(passesURI)passes/\(pass.passTypeIdentifier)/\(pass.requireID())", - headers: [ - "Authorization": "ApplePass \(pass.authenticationToken)", - "If-Modified-Since": "0", - ], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .ok) - XCTAssertNotNil(res.body) - XCTAssertEqual(res.headers.contentType?.description, "application/vnd.apple.pkpass") - XCTAssertNotNil(res.headers.lastModified) - } - ) - - // Test call with invalid authentication token - try await app.test( - .GET, - "\(passesURI)passes/\(pass.passTypeIdentifier)/\(pass.requireID())", - headers: [ - "Authorization": "ApplePass invalid-token", - "If-Modified-Since": "0", - ], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .unauthorized) - } - ) - - // Test distant future `If-Modified-Since` date - try await app.test( - .GET, - "\(passesURI)passes/\(pass.passTypeIdentifier)/\(pass.requireID())", - headers: [ - "Authorization": "ApplePass \(pass.authenticationToken)", - "If-Modified-Since": "2147483647", - ], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .notModified) - } - ) - - // Test call with invalid pass ID - try await app.test( - .GET, - "\(passesURI)passes/\(pass.passTypeIdentifier)/invalid-uuid", - headers: [ - "Authorization": "ApplePass \(pass.authenticationToken)", - "If-Modified-Since": "0", - ], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .badRequest) - } - ) - - // Test call with invalid pass type identifier - try await app.test( - .GET, - "\(passesURI)passes/pass.com.example.InvalidType/\(pass.requireID())", - headers: [ - "Authorization": "ApplePass \(pass.authenticationToken)", - "If-Modified-Since": "0", - ], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .notFound) - } - ) + @Test func getPassFromAPI() async throws { + try await withApp(delegate: delegate) { app, passesService in + let passData = PassData(title: "Test Pass") + try await passData.create(on: app.db) + let pass = try await passData.$pass.get(on: app.db) + + try await app.test( + .GET, + "\(passesURI)passes/\(pass.passTypeIdentifier)/\(pass.requireID())", + headers: [ + "Authorization": "ApplePass \(pass.authenticationToken)", + "If-Modified-Since": "0", + ], + afterResponse: { res async throws in + #expect(res.status == .ok) + #expect(res.body != nil) + #expect(res.headers.contentType?.description == "application/vnd.apple.pkpass") + #expect(res.headers.lastModified != nil) + } + ) + + // Test call with invalid authentication token + try await app.test( + .GET, + "\(passesURI)passes/\(pass.passTypeIdentifier)/\(pass.requireID())", + headers: [ + "Authorization": "ApplePass invalid-token", + "If-Modified-Since": "0", + ], + afterResponse: { res async throws in + #expect(res.status == .unauthorized) + } + ) + + // Test distant future `If-Modified-Since` date + try await app.test( + .GET, + "\(passesURI)passes/\(pass.passTypeIdentifier)/\(pass.requireID())", + headers: [ + "Authorization": "ApplePass \(pass.authenticationToken)", + "If-Modified-Since": "2147483647", + ], + afterResponse: { res async throws in + #expect(res.status == .notModified) + } + ) + + // Test call with invalid pass ID + try await app.test( + .GET, + "\(passesURI)passes/\(pass.passTypeIdentifier)/invalid-uuid", + headers: [ + "Authorization": "ApplePass \(pass.authenticationToken)", + "If-Modified-Since": "0", + ], + afterResponse: { res async throws in + #expect(res.status == .badRequest) + } + ) + + // Test call with invalid pass type identifier + try await app.test( + .GET, + "\(passesURI)passes/pass.com.example.InvalidType/\(pass.requireID())", + headers: [ + "Authorization": "ApplePass \(pass.authenticationToken)", + "If-Modified-Since": "0", + ], + afterResponse: { res async throws in + #expect(res.status == .notFound) + } + ) + } } - func testPersonalizationAPI() async throws { - let passData = PassData(title: "Personalize") - try await passData.create(on: app.db) - let pass = try await passData.$pass.get(on: app.db) - let personalizationDict = PersonalizationDictionaryDTO( - personalizationToken: "1234567890", - requiredPersonalizationInfo: .init( - emailAddress: "test@example.com", - familyName: "Doe", - fullName: "John Doe", - givenName: "John", - isoCountryCode: "US", - phoneNumber: "1234567890", - postalCode: "12345" + @Test func personalizationAPI() async throws { + try await withApp(delegate: delegate) { app, passesService in + let passData = PassData(title: "Personalize") + try await passData.create(on: app.db) + let pass = try await passData.$pass.get(on: app.db) + let personalizationDict = PersonalizationDictionaryDTO( + personalizationToken: "1234567890", + requiredPersonalizationInfo: .init( + emailAddress: "test@example.com", + familyName: "Doe", + fullName: "John Doe", + givenName: "John", + isoCountryCode: "US", + phoneNumber: "1234567890", + postalCode: "12345" + ) ) - ) - - try await app.test( - .POST, - "\(passesURI)passes/\(pass.passTypeIdentifier)/\(pass.requireID())/personalize", - beforeRequest: { req async throws in - try req.content.encode(personalizationDict) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .ok) - XCTAssertNotNil(res.body) - XCTAssertEqual(res.headers.contentType?.description, "application/octet-stream") - } - ) - - let personalizationQuery = try await UserPersonalization.query(on: app.db).all() - XCTAssertEqual(personalizationQuery.count, 1) - let passPersonalizationID = try await Pass.query(on: app.db).first()? - ._$userPersonalization.get(on: app.db)? - .requireID() - XCTAssertEqual(personalizationQuery[0]._$id.value, passPersonalizationID) - XCTAssertEqual( - personalizationQuery[0]._$emailAddress.value, - personalizationDict.requiredPersonalizationInfo.emailAddress) - XCTAssertEqual( - personalizationQuery[0]._$familyName.value, - personalizationDict.requiredPersonalizationInfo.familyName) - XCTAssertEqual( - personalizationQuery[0]._$fullName.value, - personalizationDict.requiredPersonalizationInfo.fullName) - XCTAssertEqual( - personalizationQuery[0]._$givenName.value, - personalizationDict.requiredPersonalizationInfo.givenName) - XCTAssertEqual( - personalizationQuery[0]._$isoCountryCode.value, - personalizationDict.requiredPersonalizationInfo.isoCountryCode) - XCTAssertEqual( - personalizationQuery[0]._$phoneNumber.value, - personalizationDict.requiredPersonalizationInfo.phoneNumber) - XCTAssertEqual( - personalizationQuery[0]._$postalCode.value, - personalizationDict.requiredPersonalizationInfo.postalCode) - - // Test call with invalid pass ID - try await app.test( - .POST, - "\(passesURI)passes/\(pass.passTypeIdentifier)/invalid-uuid/personalize", - beforeRequest: { req async throws in - try req.content.encode(personalizationDict) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .badRequest) - } - ) - - // Test call with invalid pass type identifier - try await app.test( - .POST, - "\(passesURI)passes/pass.com.example.InvalidType/\(pass.requireID())/personalize", - beforeRequest: { req async throws in - try req.content.encode(personalizationDict) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .notFound) - } - ) + + try await app.test( + .POST, + "\(passesURI)passes/\(pass.passTypeIdentifier)/\(pass.requireID())/personalize", + beforeRequest: { req async throws in + try req.content.encode(personalizationDict) + }, + afterResponse: { res async throws in + #expect(res.status == .ok) + #expect(res.body != nil) + #expect(res.headers.contentType?.description == "application/octet-stream") + } + ) + + let personalizationQuery = try await UserPersonalization.query(on: app.db).all() + #expect(personalizationQuery.count == 1) + let passPersonalizationID = try await Pass.query(on: app.db).first()?._$userPersonalization.get(on: app.db)?.requireID() + #expect(personalizationQuery[0]._$id.value == passPersonalizationID) + #expect(personalizationQuery[0]._$emailAddress.value == personalizationDict.requiredPersonalizationInfo.emailAddress) + #expect(personalizationQuery[0]._$familyName.value == personalizationDict.requiredPersonalizationInfo.familyName) + #expect(personalizationQuery[0]._$fullName.value == personalizationDict.requiredPersonalizationInfo.fullName) + #expect(personalizationQuery[0]._$givenName.value == personalizationDict.requiredPersonalizationInfo.givenName) + #expect(personalizationQuery[0]._$isoCountryCode.value == personalizationDict.requiredPersonalizationInfo.isoCountryCode) + #expect(personalizationQuery[0]._$phoneNumber.value == personalizationDict.requiredPersonalizationInfo.phoneNumber) + #expect(personalizationQuery[0]._$postalCode.value == personalizationDict.requiredPersonalizationInfo.postalCode) + + // Test call with invalid pass ID + try await app.test( + .POST, + "\(passesURI)passes/\(pass.passTypeIdentifier)/invalid-uuid/personalize", + beforeRequest: { req async throws in + try req.content.encode(personalizationDict) + }, + afterResponse: { res async throws in + #expect(res.status == .badRequest) + } + ) + + // Test call with invalid pass type identifier + try await app.test( + .POST, + "\(passesURI)passes/pass.com.example.InvalidType/\(pass.requireID())/personalize", + beforeRequest: { req async throws in + try req.content.encode(personalizationDict) + }, + afterResponse: { res async throws in + #expect(res.status == .notFound) + } + ) + } } - func testAPIDeviceRegistration() async throws { - let passData = PassData(title: "Test Pass") - try await passData.create(on: app.db) - let pass = try await passData.$pass.get(on: app.db) - let deviceLibraryIdentifier = "abcdefg" - let pushToken = "1234567890" - - try await app.test( - .GET, - "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)?passesUpdatedSince=0", - afterResponse: { res async throws in - XCTAssertEqual(res.status, .noContent) - } - ) - - try await app.test( - .DELETE, - "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\(pass.requireID())", - headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .notFound) - } - ) - - // Test registration without authentication token - try await app.test( - .POST, - "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\(pass.requireID())", - beforeRequest: { req async throws in - try req.content.encode(RegistrationDTO(pushToken: pushToken)) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .unauthorized) - } - ) - - // Test registration of a non-existing pass - try await app.test( - .POST, - "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\("pass.com.example.NotFound")/\(UUID().uuidString)", - headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], - beforeRequest: { req async throws in - try req.content.encode(RegistrationDTO(pushToken: pushToken)) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .notFound) - } - ) - - // Test call without DTO - try await app.test( - .POST, - "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\(pass.requireID())", - headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .badRequest) - } - ) - - // Test call with invalid UUID - try await app.test( - .POST, - "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\("not-a-uuid")", - headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], - beforeRequest: { req async throws in - try req.content.encode(RegistrationDTO(pushToken: pushToken)) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .badRequest) - } - ) - - try await app.test( - .POST, - "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\(pass.requireID())", - headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], - beforeRequest: { req async throws in - try req.content.encode(RegistrationDTO(pushToken: pushToken)) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .created) - } - ) - - // Test registration of an already registered device - try await app.test( - .POST, - "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\(pass.requireID())", - headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], - beforeRequest: { req async throws in - try req.content.encode(RegistrationDTO(pushToken: pushToken)) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .ok) - } - ) - - try await app.test( - .GET, - "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)?passesUpdatedSince=0", - afterResponse: { res async throws in - let passes = try res.content.decode(PassesForDeviceDTO.self) - XCTAssertEqual(passes.serialNumbers.count, 1) - let passID = try pass.requireID() - XCTAssertEqual(passes.serialNumbers[0], passID.uuidString) - XCTAssertEqual(passes.lastUpdated, String(pass.updatedAt!.timeIntervalSince1970)) - } - ) - - try await app.test( - .GET, - "\(passesURI)push/\(pass.passTypeIdentifier)/\(pass.requireID())", - headers: ["X-Secret": "foo"], - afterResponse: { res async throws in - let pushTokens = try res.content.decode([String].self) - XCTAssertEqual(pushTokens.count, 1) - XCTAssertEqual(pushTokens[0], pushToken) - } - ) - - // Test call with invalid UUID - try await app.test( - .GET, - "\(passesURI)push/\(pass.passTypeIdentifier)/\("not-a-uuid")", - headers: ["X-Secret": "foo"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .badRequest) - } - ) - - // Test call with invalid UUID - try await app.test( - .DELETE, - "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\("not-a-uuid")", - headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .badRequest) - } - ) - - try await app.test( - .DELETE, - "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\(pass.requireID())", - headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .ok) - } - ) + @Test func apiDeviceRegistration() async throws { + try await withApp(delegate: delegate) { app, passesService in + let passData = PassData(title: "Test Pass") + try await passData.create(on: app.db) + let pass = try await passData.$pass.get(on: app.db) + let deviceLibraryIdentifier = "abcdefg" + let pushToken = "1234567890" + + try await app.test( + .GET, + "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)?passesUpdatedSince=0", + afterResponse: { res async throws in + #expect(res.status == .noContent) + } + ) + + try await app.test( + .DELETE, + "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\(pass.requireID())", + headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], + afterResponse: { res async throws in + #expect(res.status == .notFound) + } + ) + + // Test registration without authentication token + try await app.test( + .POST, + "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\(pass.requireID())", + beforeRequest: { req async throws in + try req.content.encode(RegistrationDTO(pushToken: pushToken)) + }, + afterResponse: { res async throws in + #expect(res.status == .unauthorized) + } + ) + + // Test registration of a non-existing pass + try await app.test( + .POST, + "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\("pass.com.example.NotFound")/\(UUID().uuidString)", + headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], + beforeRequest: { req async throws in + try req.content.encode(RegistrationDTO(pushToken: pushToken)) + }, + afterResponse: { res async throws in + #expect(res.status == .notFound) + } + ) + + // Test call without DTO + try await app.test( + .POST, + "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\(pass.requireID())", + headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], + afterResponse: { res async throws in + #expect(res.status == .badRequest) + } + ) + + // Test call with invalid UUID + try await app.test( + .POST, + "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\("not-a-uuid")", + headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], + beforeRequest: { req async throws in + try req.content.encode(RegistrationDTO(pushToken: pushToken)) + }, + afterResponse: { res async throws in + #expect(res.status == .badRequest) + } + ) + + try await app.test( + .POST, + "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\(pass.requireID())", + headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], + beforeRequest: { req async throws in + try req.content.encode(RegistrationDTO(pushToken: pushToken)) + }, + afterResponse: { res async throws in + #expect(res.status == .created) + } + ) + + // Test registration of an already registered device + try await app.test( + .POST, + "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\(pass.requireID())", + headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], + beforeRequest: { req async throws in + try req.content.encode(RegistrationDTO(pushToken: pushToken)) + }, + afterResponse: { res async throws in + #expect(res.status == .ok) + } + ) + + try await app.test( + .GET, + "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)?passesUpdatedSince=0", + afterResponse: { res async throws in + let passes = try res.content.decode(PassesForDeviceDTO.self) + #expect(passes.serialNumbers.count == 1) + let passID = try pass.requireID() + #expect(passes.serialNumbers[0] == passID.uuidString) + #expect(passes.lastUpdated == String(pass.updatedAt!.timeIntervalSince1970)) + } + ) + + try await app.test( + .GET, + "\(passesURI)push/\(pass.passTypeIdentifier)/\(pass.requireID())", + headers: ["X-Secret": "foo"], + afterResponse: { res async throws in + let pushTokens = try res.content.decode([String].self) + #expect(pushTokens.count == 1) + #expect(pushTokens[0] == pushToken) + } + ) + + // Test call with invalid UUID + try await app.test( + .GET, + "\(passesURI)push/\(pass.passTypeIdentifier)/\("not-a-uuid")", + headers: ["X-Secret": "foo"], + afterResponse: { res async throws in + #expect(res.status == .badRequest) + } + ) + + // Test call with invalid UUID + try await app.test( + .DELETE, + "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\("not-a-uuid")", + headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], + afterResponse: { res async throws in + #expect(res.status == .badRequest) + } + ) + + try await app.test( + .DELETE, + "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\(pass.requireID())", + headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], + afterResponse: { res async throws in + #expect(res.status == .ok) + } + ) + } } - func testErrorLog() async throws { - let log1 = "Error 1" - let log2 = "Error 2" - - try await app.test( - .POST, - "\(passesURI)log", - beforeRequest: { req async throws in - try req.content.encode(ErrorLogDTO(logs: [log1, log2])) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .ok) - } - ) - - let logs = try await PassesErrorLog.query(on: app.db).all() - XCTAssertEqual(logs.count, 2) - XCTAssertEqual(logs[0].message, log1) - XCTAssertEqual(logs[1]._$message.value, log2) - - // Test call with no DTO - try await app.test( - .POST, - "\(passesURI)log", - afterResponse: { res async throws in - XCTAssertEqual(res.status, .badRequest) - } - ) - - // Test call with empty logs - try await app.test( - .POST, - "\(passesURI)log", - beforeRequest: { req async throws in - try req.content.encode(ErrorLogDTO(logs: [])) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .badRequest) - } - ) + @Test func errorLog() async throws { + try await withApp(delegate: delegate) { app, passesService in + let log1 = "Error 1" + let log2 = "Error 2" + + try await app.test( + .POST, + "\(passesURI)log", + beforeRequest: { req async throws in + try req.content.encode(ErrorLogDTO(logs: [log1, log2])) + }, + afterResponse: { res async throws in + #expect(res.status == .ok) + } + ) + + let logs = try await PassesErrorLog.query(on: app.db).all() + #expect(logs.count == 2) + #expect(logs[0].message == log1) + #expect(logs[1]._$message.value == log2) + + // Test call with no DTO + try await app.test( + .POST, + "\(passesURI)log", + afterResponse: { res async throws in + #expect(res.status == .badRequest) + } + ) + + // Test call with empty logs + try await app.test( + .POST, + "\(passesURI)log", + beforeRequest: { req async throws in + try req.content.encode(ErrorLogDTO(logs: [])) + }, + afterResponse: { res async throws in + #expect(res.status == .badRequest) + } + ) + } } - func testAPNSClient() async throws { - XCTAssertNotNil(app.apns.client(.init(string: "passes"))) + @Test func apnsClient() async throws { + try await withApp(delegate: delegate) { app, passesService in + #expect(app.apns.client(.init(string: "passes")) != nil) - let passData = PassData(title: "Test Pass") - try await passData.create(on: app.db) - let pass = try await passData._$pass.get(on: app.db) + let passData = PassData(title: "Test Pass") + try await passData.create(on: app.db) + let pass = try await passData._$pass.get(on: app.db) - try await passesService.sendPushNotificationsForPass( - id: pass.requireID(), of: pass.passTypeIdentifier, on: app.db) + try await passesService.sendPushNotificationsForPass(id: pass.requireID(), of: pass.passTypeIdentifier, on: app.db) - let deviceLibraryIdentifier = "abcdefg" - let pushToken = "1234567890" + let deviceLibraryIdentifier = "abcdefg" + let pushToken = "1234567890" - try await app.test( - .POST, - "\(passesURI)push/\(pass.passTypeIdentifier)/\(pass.requireID())", - headers: ["X-Secret": "foo"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .noContent) - } - ) - - try await app.test( - .POST, - "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\(pass.requireID())", - headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], - beforeRequest: { req async throws in - try req.content.encode(RegistrationDTO(pushToken: pushToken)) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .created) - } - ) - - try await app.test( - .POST, - "\(passesURI)push/\(pass.passTypeIdentifier)/\(pass.requireID())", - headers: ["X-Secret": "foo"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .internalServerError) - } - ) - - // Test call with invalid UUID - try await app.test( - .POST, - "\(passesURI)push/\(pass.passTypeIdentifier)/\("not-a-uuid")", - headers: ["X-Secret": "foo"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .badRequest) + try await app.test( + .POST, + "\(passesURI)push/\(pass.passTypeIdentifier)/\(pass.requireID())", + headers: ["X-Secret": "foo"], + afterResponse: { res async throws in + #expect(res.status == .noContent) + } + ) + + try await app.test( + .POST, + "\(passesURI)devices/\(deviceLibraryIdentifier)/registrations/\(pass.passTypeIdentifier)/\(pass.requireID())", + headers: ["Authorization": "ApplePass \(pass.authenticationToken)"], + beforeRequest: { req async throws in + try req.content.encode(RegistrationDTO(pushToken: pushToken)) + }, + afterResponse: { res async throws in + #expect(res.status == .created) + } + ) + + try await app.test( + .POST, + "\(passesURI)push/\(pass.passTypeIdentifier)/\(pass.requireID())", + headers: ["X-Secret": "foo"], + afterResponse: { res async throws in + #expect(res.status == .internalServerError) + } + ) + + // Test call with invalid UUID + try await app.test( + .POST, + "\(passesURI)push/\(pass.passTypeIdentifier)/\("not-a-uuid")", + headers: ["X-Secret": "foo"], + afterResponse: { res async throws in + #expect(res.status == .badRequest) + } + ) + + // Test `PassDataMiddleware` update method + passData.title = "Test Pass 2" + do { + try await passData.update(on: app.db) + } catch let error as HTTPClientError { + #expect(error.self == .remoteConnectionClosed) } - ) - - // Test `PassDataMiddleware` update method - passData.title = "Test Pass 2" - do { - try await passData.update(on: app.db) - } catch let error as HTTPClientError { - XCTAssertEqual(error.self, .remoteConnectionClosed) } } - func testPassesError() { - XCTAssertEqual( - PassesError.templateNotDirectory.description, - "PassesError(errorType: templateNotDirectory)") - XCTAssertEqual( - PassesError.pemCertificateMissing.description, - "PassesError(errorType: pemCertificateMissing)") - XCTAssertEqual( - PassesError.pemPrivateKeyMissing.description, - "PassesError(errorType: pemPrivateKeyMissing)") - XCTAssertEqual( - PassesError.opensslBinaryMissing.description, - "PassesError(errorType: opensslBinaryMissing)") - XCTAssertEqual( - PassesError.invalidNumberOfPasses.description, - "PassesError(errorType: invalidNumberOfPasses)") + @Test func passesError() { + #expect(PassesError.templateNotDirectory.description == "PassesError(errorType: templateNotDirectory)") + #expect(PassesError.pemCertificateMissing.description == "PassesError(errorType: pemCertificateMissing)") + #expect(PassesError.pemPrivateKeyMissing.description == "PassesError(errorType: pemPrivateKeyMissing)") + #expect(PassesError.opensslBinaryMissing.description == "PassesError(errorType: opensslBinaryMissing)") + #expect(PassesError.invalidNumberOfPasses.description == "PassesError(errorType: invalidNumberOfPasses)") } - func testDefaultDelegate() async throws { + @Test func defaultDelegate() async throws { let delegate = DefaultPassesDelegate() - XCTAssertEqual(delegate.wwdrCertificate, "WWDR.pem") - XCTAssertEqual(delegate.pemCertificate, "passcertificate.pem") - XCTAssertEqual(delegate.pemPrivateKey, "passkey.pem") - XCTAssertNil(delegate.pemPrivateKeyPassword) - XCTAssertEqual(delegate.sslBinary, URL(fileURLWithPath: "/usr/bin/openssl")) - XCTAssertFalse(delegate.generateSignatureFile(in: URL(fileURLWithPath: ""))) - - let passData = PassData(title: "Test Pass") - try await passData.create(on: app.db) - let pass = try await passData.$pass.get(on: app.db) - let data = try await delegate.encodePersonalization( - for: pass, db: app.db, encoder: JSONEncoder()) - XCTAssertNil(data) + #expect(delegate.wwdrCertificate == "WWDR.pem") + #expect(delegate.pemCertificate == "passcertificate.pem") + #expect(delegate.pemPrivateKey == "passkey.pem") + #expect(delegate.pemPrivateKeyPassword == nil) + #expect(delegate.sslBinary == URL(fileURLWithPath: "/usr/bin/openssl")) + #expect(!delegate.generateSignatureFile(in: URL(fileURLWithPath: ""))) + + try await withApp(delegate: delegate) { app, passesService in + let passData = PassData(title: "Test Pass") + try await passData.create(on: app.db) + let pass = try await passData.$pass.get(on: app.db) + let data = try await delegate.encodePersonalization(for: pass, db: app.db, encoder: JSONEncoder()) + #expect(data == nil) + } } } @@ -573,9 +526,7 @@ final class DefaultPassesDelegate: PassesDelegate { func template(for pass: P, db: any Database) async throws -> URL { URL(fileURLWithPath: "") } - func encode( - pass: P, db: any Database, encoder: JSONEncoder - ) async throws -> Data { + func encode(pass: P, db: any Database, encoder: JSONEncoder) async throws -> Data { Data() } } diff --git a/Tests/PassesTests/SecretMiddleware.swift b/Tests/PassesTests/SecretMiddleware.swift deleted file mode 100644 index fef1940..0000000 --- a/Tests/PassesTests/SecretMiddleware.swift +++ /dev/null @@ -1,14 +0,0 @@ -import Vapor - -struct SecretMiddleware: AsyncMiddleware { - let secret: String - - func respond( - to request: Request, chainingTo next: any AsyncResponder - ) async throws -> Response { - guard request.headers.first(name: "X-Secret") == secret else { - throw Abort(.unauthorized, reason: "Incorrect X-Secret header.") - } - return try await next.respond(to: request) - } -} diff --git a/Tests/PassesTests/withApp.swift b/Tests/PassesTests/withApp.swift new file mode 100644 index 0000000..6ff4a1c --- /dev/null +++ b/Tests/PassesTests/withApp.swift @@ -0,0 +1,38 @@ +import FluentKit +import FluentSQLiteDriver +import PassKit +import Passes +import Testing +import Vapor +import Zip + +func withApp( + delegate: some PassesDelegate, + _ body: (Application, PassesService) async throws -> Void +) async throws { + let app = try await Application.make(.testing) + + try #require(isLoggingConfigured) + + app.databases.use(.sqlite(.memory), as: .sqlite) + + PassesService.register(migrations: app.migrations) + app.migrations.add(CreatePassData()) + let passesService = try PassesService( + app: app, + delegate: delegate, + pushRoutesMiddleware: SecretMiddleware(secret: "foo"), + logger: app.logger + ) + + app.databases.middleware.use(PassDataMiddleware(service: passesService), on: .sqlite) + + try await app.autoMigrate() + + Zip.addCustomFileExtension("pkpass") + + try await body(app, passesService) + + try await app.autoRevert() + try await app.asyncShutdown() +} From 2e130213abb5a404112c5871ed6dcc9841536402 Mon Sep 17 00:00:00 2001 From: Francesco Paolo Severino Date: Fri, 11 Oct 2024 17:13:16 +0200 Subject: [PATCH 2/7] Update OrdersTests to Swift Testing --- Tests/OrdersTests/EncryptedOrdersTests.swift | 179 ++--- Tests/OrdersTests/OrdersTests.swift | 761 +++++++++---------- Tests/OrdersTests/withApp.swift | 38 + Tests/PassesTests/PassesTests.swift | 16 +- 4 files changed, 487 insertions(+), 507 deletions(-) create mode 100644 Tests/OrdersTests/withApp.swift diff --git a/Tests/OrdersTests/EncryptedOrdersTests.swift b/Tests/OrdersTests/EncryptedOrdersTests.swift index 6e39f4c..5ddf2cc 100644 --- a/Tests/OrdersTests/EncryptedOrdersTests.swift +++ b/Tests/OrdersTests/EncryptedOrdersTests.swift @@ -1,116 +1,89 @@ -import Fluent -import FluentSQLiteDriver +import FluentKit import PassKit +import Testing import XCTVapor import Zip @testable import Orders -final class EncryptedOrdersTests: XCTestCase { +struct EncryptedOrdersTests { let delegate = EncryptedOrdersDelegate() let ordersURI = "/api/orders/v1/" - var ordersService: OrdersService! - var app: Application! - override func setUp() async throws { - self.app = try await Application.make(.testing) - app.databases.use(.sqlite(.memory), as: .sqlite) - - OrdersService.register(migrations: app.migrations) - app.migrations.add(CreateOrderData()) - ordersService = try OrdersService( - app: app, - delegate: delegate, - pushRoutesMiddleware: SecretMiddleware(secret: "foo"), - logger: app.logger - ) - app.databases.middleware.use(OrderDataMiddleware(service: ordersService), on: .sqlite) - - try await app.autoMigrate() - - Zip.addCustomFileExtension("order") - } - - override func tearDown() async throws { - try await app.autoRevert() - try await self.app.asyncShutdown() - self.app = nil + @Test func orderGeneration() async throws { + try await withApp(delegate: delegate) { app, ordersService in + let orderData = OrderData(title: "Test Order") + try await orderData.create(on: app.db) + let order = try await orderData.$order.get(on: app.db) + let data = try await ordersService.generateOrderContent(for: order, on: app.db) + let orderURL = FileManager.default.temporaryDirectory.appendingPathComponent("test.order") + try data.write(to: orderURL) + let orderFolder = try Zip.quickUnzipFile(orderURL) + + #expect(FileManager.default.fileExists(atPath: orderFolder.path.appending("/signature"))) + + let passJSONData = try String(contentsOfFile: orderFolder.path.appending("/order.json")).data(using: .utf8) + let passJSON = try JSONSerialization.jsonObject(with: passJSONData!) as! [String: Any] + #expect(passJSON["authenticationToken"] as? String == order.authenticationToken) + let orderID = try order.requireID().uuidString + #expect(passJSON["orderIdentifier"] as? String == orderID) + + let manifestJSONData = try String(contentsOfFile: orderFolder.path.appending("/manifest.json")).data(using: .utf8) + let manifestJSON = try JSONSerialization.jsonObject(with: manifestJSONData!) as! [String: Any] + let iconData = try Data(contentsOf: orderFolder.appendingPathComponent("/icon.png")) + let iconHash = Array(SHA256.hash(data: iconData)).hex + #expect(manifestJSON["icon.png"] as? String == iconHash) + } } - func testOrderGeneration() async throws { - let orderData = OrderData(title: "Test Order") - try await orderData.create(on: app.db) - let order = try await orderData.$order.get(on: app.db) - let data = try await ordersService.generateOrderContent(for: order, on: app.db) - let orderURL = FileManager.default.temporaryDirectory.appendingPathComponent("test.order") - try data.write(to: orderURL) - let orderFolder = try Zip.quickUnzipFile(orderURL) - - XCTAssert(FileManager.default.fileExists(atPath: orderFolder.path.appending("/signature"))) - - let passJSONData = try String(contentsOfFile: orderFolder.path.appending("/order.json")) - .data(using: .utf8) - let passJSON = try JSONSerialization.jsonObject(with: passJSONData!) as! [String: Any] - XCTAssertEqual(passJSON["authenticationToken"] as? String, order.authenticationToken) - try XCTAssertEqual(passJSON["orderIdentifier"] as? String, order.requireID().uuidString) - - let manifestJSONData = try String( - contentsOfFile: orderFolder.path.appending("/manifest.json") - ).data(using: .utf8) - let manifestJSON = - try JSONSerialization.jsonObject(with: manifestJSONData!) as! [String: Any] - let iconData = try Data(contentsOf: orderFolder.appendingPathComponent("/icon.png")) - let iconHash = Array(SHA256.hash(data: iconData)).hex - XCTAssertEqual(manifestJSON["icon.png"] as? String, iconHash) - } - - func testAPNSClient() async throws { - XCTAssertNotNil(app.apns.client(.init(string: "orders"))) - - let orderData = OrderData(title: "Test Order") - try await orderData.create(on: app.db) - let order = try await orderData._$order.get(on: app.db) - - try await ordersService.sendPushNotificationsForOrder( - id: order.requireID(), of: order.orderTypeIdentifier, on: app.db) - - let deviceLibraryIdentifier = "abcdefg" - let pushToken = "1234567890" - - try await app.test( - .POST, - "\(ordersURI)push/\(order.orderTypeIdentifier)/\(order.requireID())", - headers: ["X-Secret": "foo"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .noContent) - } - ) - - try await app.test( - .POST, - "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\(order.requireID())", - headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], - beforeRequest: { req async throws in - try req.content.encode(RegistrationDTO(pushToken: pushToken)) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .created) - } - ) - - try await app.test( - .POST, - "\(ordersURI)push/\(order.orderTypeIdentifier)/\(order.requireID())", - headers: ["X-Secret": "foo"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .internalServerError) - } - ) - - // Test `OrderDataMiddleware` update method - orderData.title = "Test Order 2" - do { - try await orderData.update(on: app.db) - } catch {} + @Test func apnsClient() async throws { + try await withApp(delegate: delegate) { app, ordersService in + #expect(app.apns.client(.init(string: "orders")) != nil) + + let orderData = OrderData(title: "Test Order") + try await orderData.create(on: app.db) + let order = try await orderData._$order.get(on: app.db) + + try await ordersService.sendPushNotificationsForOrder(id: order.requireID(), of: order.orderTypeIdentifier, on: app.db) + + let deviceLibraryIdentifier = "abcdefg" + let pushToken = "1234567890" + + try await app.test( + .POST, + "\(ordersURI)push/\(order.orderTypeIdentifier)/\(order.requireID())", + headers: ["X-Secret": "foo"], + afterResponse: { res async throws in + #expect(res.status == .noContent) + } + ) + + try await app.test( + .POST, + "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\(order.requireID())", + headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], + beforeRequest: { req async throws in + try req.content.encode(RegistrationDTO(pushToken: pushToken)) + }, + afterResponse: { res async throws in + #expect(res.status == .created) + } + ) + + try await app.test( + .POST, + "\(ordersURI)push/\(order.orderTypeIdentifier)/\(order.requireID())", + headers: ["X-Secret": "foo"], + afterResponse: { res async throws in + #expect(res.status == .internalServerError) + } + ) + + // Test `OrderDataMiddleware` update method + orderData.title = "Test Order 2" + do { + try await orderData.update(on: app.db) + } catch {} + } } } diff --git a/Tests/OrdersTests/OrdersTests.swift b/Tests/OrdersTests/OrdersTests.swift index 9833474..aaddc8e 100644 --- a/Tests/OrdersTests/OrdersTests.swift +++ b/Tests/OrdersTests/OrdersTests.swift @@ -1,421 +1,392 @@ -import Fluent -import FluentSQLiteDriver +import FluentKit import PassKit +import Testing import XCTVapor import Zip @testable import Orders -final class OrdersTests: XCTestCase { +struct OrdersTests { let delegate = TestOrdersDelegate() let ordersURI = "/api/orders/v1/" - var ordersService: OrdersService! - var app: Application! - - override func setUp() async throws { - self.app = try await Application.make(.testing) - app.databases.use(.sqlite(.memory), as: .sqlite) - - OrdersService.register(migrations: app.migrations) - app.migrations.add(CreateOrderData()) - ordersService = try OrdersService( - app: app, - delegate: delegate, - pushRoutesMiddleware: SecretMiddleware(secret: "foo"), - logger: app.logger - ) - app.databases.middleware.use(OrderDataMiddleware(service: ordersService), on: .sqlite) - - try await app.autoMigrate() - - Zip.addCustomFileExtension("order") - } - - override func tearDown() async throws { - try await app.autoRevert() - try await self.app.asyncShutdown() - self.app = nil - } - func testOrderGeneration() async throws { - let orderData = OrderData(title: "Test Order") - try await orderData.create(on: app.db) - let order = try await orderData.$order.get(on: app.db) - let data = try await ordersService.generateOrderContent(for: order, on: app.db) - let orderURL = FileManager.default.temporaryDirectory.appendingPathComponent("test.order") - try data.write(to: orderURL) - let orderFolder = try Zip.quickUnzipFile(orderURL) - - XCTAssert(FileManager.default.fileExists(atPath: orderFolder.path.appending("/signature"))) - - let passJSONData = try String(contentsOfFile: orderFolder.path.appending("/order.json")) - .data(using: .utf8) - let passJSON = try JSONSerialization.jsonObject(with: passJSONData!) as! [String: Any] - XCTAssertEqual(passJSON["authenticationToken"] as? String, order.authenticationToken) - try XCTAssertEqual(passJSON["orderIdentifier"] as? String, order.requireID().uuidString) - - let manifestJSONData = try String( - contentsOfFile: orderFolder.path.appending("/manifest.json") - ).data(using: .utf8) - let manifestJSON = - try JSONSerialization.jsonObject(with: manifestJSONData!) as! [String: Any] - let iconData = try Data(contentsOf: orderFolder.appendingPathComponent("/icon.png")) - let iconHash = Array(SHA256.hash(data: iconData)).hex - XCTAssertEqual(manifestJSON["icon.png"] as? String, iconHash) - XCTAssertNotNil(manifestJSON["pet_store_logo.png"]) + @Test func orderGeneration() async throws { + try await withApp(delegate: delegate) { app, ordersService in + let orderData = OrderData(title: "Test Order") + try await orderData.create(on: app.db) + let order = try await orderData.$order.get(on: app.db) + let data = try await ordersService.generateOrderContent(for: order, on: app.db) + let orderURL = FileManager.default.temporaryDirectory.appendingPathComponent("test.order") + try data.write(to: orderURL) + let orderFolder = try Zip.quickUnzipFile(orderURL) + + #expect(FileManager.default.fileExists(atPath: orderFolder.path.appending("/signature"))) + + let passJSONData = try String(contentsOfFile: orderFolder.path.appending("/order.json")).data(using: .utf8) + let passJSON = try JSONSerialization.jsonObject(with: passJSONData!) as! [String: Any] + #expect(passJSON["authenticationToken"] as? String == order.authenticationToken) + let orderID = try order.requireID().uuidString + #expect(passJSON["orderIdentifier"] as? String == orderID) + + let manifestJSONData = try String(contentsOfFile: orderFolder.path.appending("/manifest.json")).data(using: .utf8) + let manifestJSON = try JSONSerialization.jsonObject(with: manifestJSONData!) as! [String: Any] + let iconData = try Data(contentsOf: orderFolder.appendingPathComponent("/icon.png")) + let iconHash = Array(SHA256.hash(data: iconData)).hex + #expect(manifestJSON["icon.png"] as? String == iconHash) + #expect(manifestJSON["pet_store_logo.png"] != nil) + } } // Tests the API Apple Wallet calls to get orders - func testGetOrderFromAPI() async throws { - let orderData = OrderData(title: "Test Order") - try await orderData.create(on: app.db) - let order = try await orderData.$order.get(on: app.db) - - try await app.test( - .GET, - "\(ordersURI)orders/\(order.orderTypeIdentifier)/\(order.requireID())", - headers: [ - "Authorization": "AppleOrder \(order.authenticationToken)", - "If-Modified-Since": "0", - ], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .ok) - XCTAssertNotNil(res.body) - XCTAssertEqual(res.headers.contentType?.description, "application/vnd.apple.order") - XCTAssertNotNil(res.headers.lastModified) - } - ) - - // Test call with invalid authentication token - try await app.test( - .GET, - "\(ordersURI)orders/\(order.orderTypeIdentifier)/\(order.requireID())", - headers: [ - "Authorization": "AppleOrder invalidToken", - "If-Modified-Since": "0", - ], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .unauthorized) - } - ) - - // Test distant future `If-Modified-Since` date - try await app.test( - .GET, - "\(ordersURI)orders/\(order.orderTypeIdentifier)/\(order.requireID())", - headers: [ - "Authorization": "AppleOrder \(order.authenticationToken)", - "If-Modified-Since": "2147483647", - ], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .notModified) - } - ) - - // Test call with invalid order ID - try await app.test( - .GET, - "\(ordersURI)orders/\(order.orderTypeIdentifier)/invalidID", - headers: [ - "Authorization": "AppleOrder \(order.authenticationToken)", - "If-Modified-Since": "0", - ], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .badRequest) - } - ) - - // Test call with invalid order type identifier - try await app.test( - .GET, - "\(ordersURI)orders/order.com.example.InvalidType/\(order.requireID())", - headers: [ - "Authorization": "AppleOrder \(order.authenticationToken)", - "If-Modified-Since": "0", - ], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .notFound) - } - ) + @Test func getOrderFromAPI() async throws { + try await withApp(delegate: delegate) { app, ordersService in + let orderData = OrderData(title: "Test Order") + try await orderData.create(on: app.db) + let order = try await orderData.$order.get(on: app.db) + + try await app.test( + .GET, + "\(ordersURI)orders/\(order.orderTypeIdentifier)/\(order.requireID())", + headers: [ + "Authorization": "AppleOrder \(order.authenticationToken)", + "If-Modified-Since": "0", + ], + afterResponse: { res async throws in + #expect(res.status == .ok) + #expect(res.body != nil) + #expect(res.headers.contentType?.description == "application/vnd.apple.order") + #expect(res.headers.lastModified != nil) + } + ) + + // Test call with invalid authentication token + try await app.test( + .GET, + "\(ordersURI)orders/\(order.orderTypeIdentifier)/\(order.requireID())", + headers: [ + "Authorization": "AppleOrder invalidToken", + "If-Modified-Since": "0", + ], + afterResponse: { res async throws in + #expect(res.status == .unauthorized) + } + ) + + // Test distant future `If-Modified-Since` date + try await app.test( + .GET, + "\(ordersURI)orders/\(order.orderTypeIdentifier)/\(order.requireID())", + headers: [ + "Authorization": "AppleOrder \(order.authenticationToken)", + "If-Modified-Since": "2147483647", + ], + afterResponse: { res async throws in + #expect(res.status == .notModified) + } + ) + + // Test call with invalid order ID + try await app.test( + .GET, + "\(ordersURI)orders/\(order.orderTypeIdentifier)/invalidID", + headers: [ + "Authorization": "AppleOrder \(order.authenticationToken)", + "If-Modified-Since": "0", + ], + afterResponse: { res async throws in + #expect(res.status == .badRequest) + } + ) + + // Test call with invalid order type identifier + try await app.test( + .GET, + "\(ordersURI)orders/order.com.example.InvalidType/\(order.requireID())", + headers: [ + "Authorization": "AppleOrder \(order.authenticationToken)", + "If-Modified-Since": "0", + ], + afterResponse: { res async throws in + #expect(res.status == .notFound) + } + ) + } } - func testAPIDeviceRegistration() async throws { - let orderData = OrderData(title: "Test Order") - try await orderData.create(on: app.db) - let order = try await orderData.$order.get(on: app.db) - let deviceLibraryIdentifier = "abcdefg" - let pushToken = "1234567890" - - try await app.test( - .GET, - "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)?ordersModifiedSince=0", - afterResponse: { res async throws in - XCTAssertEqual(res.status, .noContent) - } - ) - - try await app.test( - .DELETE, - "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\(order.requireID())", - headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .notFound) - } - ) - - // Test registration without authentication token - try await app.test( - .POST, - "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\(order.requireID())", - beforeRequest: { req async throws in - try req.content.encode(RegistrationDTO(pushToken: pushToken)) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .unauthorized) - } - ) - - // Test registration of a non-existing order - try await app.test( - .POST, - "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\("order.com.example.NotFound")/\(UUID().uuidString)", - headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], - beforeRequest: { req async throws in - try req.content.encode(RegistrationDTO(pushToken: pushToken)) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .notFound) - } - ) - - // Test call without DTO - try await app.test( - .POST, - "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\(order.requireID())", - headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .badRequest) - } - ) - - // Test call with invalid UUID - try await app.test( - .POST, - "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\("not-a-uuid")", - headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], - beforeRequest: { req async throws in - try req.content.encode(RegistrationDTO(pushToken: pushToken)) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .badRequest) - } - ) - - try await app.test( - .POST, - "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\(order.requireID())", - headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], - beforeRequest: { req async throws in - try req.content.encode(RegistrationDTO(pushToken: pushToken)) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .created) - } - ) - - // Test registration of an already registered device - try await app.test( - .POST, - "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\(order.requireID())", - headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], - beforeRequest: { req async throws in - try req.content.encode(RegistrationDTO(pushToken: pushToken)) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .ok) - } - ) - - try await app.test( - .GET, - "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)?ordersModifiedSince=0", - afterResponse: { res async throws in - let orders = try res.content.decode(OrdersForDeviceDTO.self) - XCTAssertEqual(orders.orderIdentifiers.count, 1) - let orderID = try order.requireID() - XCTAssertEqual(orders.orderIdentifiers[0], orderID.uuidString) - XCTAssertEqual(orders.lastModified, String(order.updatedAt!.timeIntervalSince1970)) - } - ) - - try await app.test( - .GET, - "\(ordersURI)push/\(order.orderTypeIdentifier)/\(order.requireID())", - headers: ["X-Secret": "foo"], - afterResponse: { res async throws in - let pushTokens = try res.content.decode([String].self) - XCTAssertEqual(pushTokens.count, 1) - XCTAssertEqual(pushTokens[0], pushToken) - } - ) - - // Test call with invalid UUID - try await app.test( - .GET, - "\(ordersURI)push/\(order.orderTypeIdentifier)/\("not-a-uuid")", - headers: ["X-Secret": "foo"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .badRequest) - } - ) - - // Test call with invalid UUID - try await app.test( - .DELETE, - "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\("not-a-uuid")", - headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .badRequest) - } - ) - - try await app.test( - .DELETE, - "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\(order.requireID())", - headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .ok) - } - ) + @Test func apiDeviceRegistration() async throws { + try await withApp(delegate: delegate) { app, ordersService in + let orderData = OrderData(title: "Test Order") + try await orderData.create(on: app.db) + let order = try await orderData.$order.get(on: app.db) + let deviceLibraryIdentifier = "abcdefg" + let pushToken = "1234567890" + + try await app.test( + .GET, + "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)?ordersModifiedSince=0", + afterResponse: { res async throws in + #expect(res.status == .noContent) + } + ) + + try await app.test( + .DELETE, + "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\(order.requireID())", + headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], + afterResponse: { res async throws in + #expect(res.status == .notFound) + } + ) + + // Test registration without authentication token + try await app.test( + .POST, + "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\(order.requireID())", + beforeRequest: { req async throws in + try req.content.encode(RegistrationDTO(pushToken: pushToken)) + }, + afterResponse: { res async throws in + #expect(res.status == .unauthorized) + } + ) + + // Test registration of a non-existing order + try await app.test( + .POST, + "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\("order.com.example.NotFound")/\(UUID().uuidString)", + headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], + beforeRequest: { req async throws in + try req.content.encode(RegistrationDTO(pushToken: pushToken)) + }, + afterResponse: { res async throws in + #expect(res.status == .notFound) + } + ) + + // Test call without DTO + try await app.test( + .POST, + "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\(order.requireID())", + headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], + afterResponse: { res async throws in + #expect(res.status == .badRequest) + } + ) + + // Test call with invalid UUID + try await app.test( + .POST, + "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\("not-a-uuid")", + headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], + beforeRequest: { req async throws in + try req.content.encode(RegistrationDTO(pushToken: pushToken)) + }, + afterResponse: { res async throws in + #expect(res.status == .badRequest) + } + ) + + try await app.test( + .POST, + "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\(order.requireID())", + headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], + beforeRequest: { req async throws in + try req.content.encode(RegistrationDTO(pushToken: pushToken)) + }, + afterResponse: { res async throws in + #expect(res.status == .created) + } + ) + + // Test registration of an already registered device + try await app.test( + .POST, + "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\(order.requireID())", + headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], + beforeRequest: { req async throws in + try req.content.encode(RegistrationDTO(pushToken: pushToken)) + }, + afterResponse: { res async throws in + #expect(res.status == .ok) + } + ) + + try await app.test( + .GET, + "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)?ordersModifiedSince=0", + afterResponse: { res async throws in + let orders = try res.content.decode(OrdersForDeviceDTO.self) + #expect(orders.orderIdentifiers.count == 1) + let orderID = try order.requireID() + #expect(orders.orderIdentifiers[0] == orderID.uuidString) + #expect(orders.lastModified == String(order.updatedAt!.timeIntervalSince1970)) + } + ) + + try await app.test( + .GET, + "\(ordersURI)push/\(order.orderTypeIdentifier)/\(order.requireID())", + headers: ["X-Secret": "foo"], + afterResponse: { res async throws in + let pushTokens = try res.content.decode([String].self) + #expect(pushTokens.count == 1) + #expect(pushTokens[0] == pushToken) + } + ) + + // Test call with invalid UUID + try await app.test( + .GET, + "\(ordersURI)push/\(order.orderTypeIdentifier)/\("not-a-uuid")", + headers: ["X-Secret": "foo"], + afterResponse: { res async throws in + #expect(res.status == .badRequest) + } + ) + + // Test call with invalid UUID + try await app.test( + .DELETE, + "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\("not-a-uuid")", + headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], + afterResponse: { res async throws in + #expect(res.status == .badRequest) + } + ) + + try await app.test( + .DELETE, + "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\(order.requireID())", + headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], + afterResponse: { res async throws in + #expect(res.status == .ok) + } + ) + } } - func testErrorLog() async throws { - let log1 = "Error 1" - let log2 = "Error 2" - - try await app.test( - .POST, - "\(ordersURI)log", - beforeRequest: { req async throws in - try req.content.encode(ErrorLogDTO(logs: [log1, log2])) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .ok) - } - ) - - let logs = try await OrdersErrorLog.query(on: app.db).all() - XCTAssertEqual(logs.count, 2) - XCTAssertEqual(logs[0].message, log1) - XCTAssertEqual(logs[1]._$message.value, log2) - - // Test call with no DTO - try await app.test( - .POST, - "\(ordersURI)log", - afterResponse: { res async throws in - XCTAssertEqual(res.status, .badRequest) - } - ) - - // Test call with empty logs - try await app.test( - .POST, - "\(ordersURI)log", - beforeRequest: { req async throws in - try req.content.encode(ErrorLogDTO(logs: [])) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .badRequest) - } - ) + @Test func errorLog() async throws { + try await withApp(delegate: delegate) { app, ordersService in + let log1 = "Error 1" + let log2 = "Error 2" + + try await app.test( + .POST, + "\(ordersURI)log", + beforeRequest: { req async throws in + try req.content.encode(ErrorLogDTO(logs: [log1, log2])) + }, + afterResponse: { res async throws in + #expect(res.status == .ok) + } + ) + + let logs = try await OrdersErrorLog.query(on: app.db).all() + #expect(logs.count == 2) + #expect(logs[0].message == log1) + #expect(logs[1]._$message.value == log2) + + // Test call with no DTO + try await app.test( + .POST, + "\(ordersURI)log", + afterResponse: { res async throws in + #expect(res.status == .badRequest) + } + ) + + // Test call with empty logs + try await app.test( + .POST, + "\(ordersURI)log", + beforeRequest: { req async throws in + try req.content.encode(ErrorLogDTO(logs: [])) + }, + afterResponse: { res async throws in + #expect(res.status == .badRequest) + } + ) + } } - func testAPNSClient() async throws { - XCTAssertNotNil(app.apns.client(.init(string: "orders"))) - - let orderData = OrderData(title: "Test Order") - try await orderData.create(on: app.db) - let order = try await orderData._$order.get(on: app.db) - - try await ordersService.sendPushNotificationsForOrder( - id: order.requireID(), of: order.orderTypeIdentifier, on: app.db) - - let deviceLibraryIdentifier = "abcdefg" - let pushToken = "1234567890" - - try await app.test( - .POST, - "\(ordersURI)push/\(order.orderTypeIdentifier)/\(order.requireID())", - headers: ["X-Secret": "foo"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .noContent) - } - ) - - try await app.test( - .POST, - "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\(order.requireID())", - headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], - beforeRequest: { req async throws in - try req.content.encode(RegistrationDTO(pushToken: pushToken)) - }, - afterResponse: { res async throws in - XCTAssertEqual(res.status, .created) - } - ) - - try await app.test( - .POST, - "\(ordersURI)push/\(order.orderTypeIdentifier)/\(order.requireID())", - headers: ["X-Secret": "foo"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .internalServerError) - } - ) - - // Test call with invalid UUID - try await app.test( - .POST, - "\(ordersURI)push/\(order.orderTypeIdentifier)/\("not-a-uuid")", - headers: ["X-Secret": "foo"], - afterResponse: { res async throws in - XCTAssertEqual(res.status, .badRequest) + @Test func apnsClient() async throws { + try await withApp(delegate: delegate) { app, ordersService in + #expect(app.apns.client(.init(string: "orders")) != nil) + + let orderData = OrderData(title: "Test Order") + try await orderData.create(on: app.db) + let order = try await orderData._$order.get(on: app.db) + + try await ordersService.sendPushNotificationsForOrder(id: order.requireID(), of: order.orderTypeIdentifier, on: app.db) + + let deviceLibraryIdentifier = "abcdefg" + let pushToken = "1234567890" + + try await app.test( + .POST, + "\(ordersURI)push/\(order.orderTypeIdentifier)/\(order.requireID())", + headers: ["X-Secret": "foo"], + afterResponse: { res async throws in + #expect(res.status == .noContent) + } + ) + + try await app.test( + .POST, + "\(ordersURI)devices/\(deviceLibraryIdentifier)/registrations/\(order.orderTypeIdentifier)/\(order.requireID())", + headers: ["Authorization": "AppleOrder \(order.authenticationToken)"], + beforeRequest: { req async throws in + try req.content.encode(RegistrationDTO(pushToken: pushToken)) + }, + afterResponse: { res async throws in + #expect(res.status == .created) + } + ) + + try await app.test( + .POST, + "\(ordersURI)push/\(order.orderTypeIdentifier)/\(order.requireID())", + headers: ["X-Secret": "foo"], + afterResponse: { res async throws in + #expect(res.status == .internalServerError) + } + ) + + // Test call with invalid UUID + try await app.test( + .POST, + "\(ordersURI)push/\(order.orderTypeIdentifier)/\("not-a-uuid")", + headers: ["X-Secret": "foo"], + afterResponse: { res async throws in + #expect(res.status == .badRequest) + } + ) + + // Test `OrderDataMiddleware` update method + orderData.title = "Test Order 2" + do { + try await orderData.update(on: app.db) + } catch let error as HTTPClientError { + #expect(error.self == .remoteConnectionClosed) } - ) - - // Test `OrderDataMiddleware` update method - orderData.title = "Test Order 2" - do { - try await orderData.update(on: app.db) - } catch let error as HTTPClientError { - XCTAssertEqual(error.self, .remoteConnectionClosed) } } - func testOrdersError() { - XCTAssertEqual( - OrdersError.templateNotDirectory.description, - "OrdersError(errorType: templateNotDirectory)") - XCTAssertEqual( - OrdersError.pemCertificateMissing.description, - "OrdersError(errorType: pemCertificateMissing)") - XCTAssertEqual( - OrdersError.pemPrivateKeyMissing.description, - "OrdersError(errorType: pemPrivateKeyMissing)") - XCTAssertEqual( - OrdersError.opensslBinaryMissing.description, - "OrdersError(errorType: opensslBinaryMissing)") + @Test func ordersError() { + #expect(OrdersError.templateNotDirectory.description == "OrdersError(errorType: templateNotDirectory)") + #expect(OrdersError.pemCertificateMissing.description == "OrdersError(errorType: pemCertificateMissing)") + #expect(OrdersError.pemPrivateKeyMissing.description == "OrdersError(errorType: pemPrivateKeyMissing)") + #expect(OrdersError.opensslBinaryMissing.description == "OrdersError(errorType: opensslBinaryMissing)") } - func testDefaultDelegate() { + @Test func defaultDelegate() { let delegate = DefaultOrdersDelegate() - XCTAssertEqual(delegate.wwdrCertificate, "WWDR.pem") - XCTAssertEqual(delegate.pemCertificate, "ordercertificate.pem") - XCTAssertEqual(delegate.pemPrivateKey, "orderkey.pem") - XCTAssertNil(delegate.pemPrivateKeyPassword) - XCTAssertEqual(delegate.sslBinary, URL(fileURLWithPath: "/usr/bin/openssl")) - XCTAssertFalse(delegate.generateSignatureFile(in: URL(fileURLWithPath: ""))) + #expect(delegate.wwdrCertificate == "WWDR.pem") + #expect(delegate.pemCertificate == "ordercertificate.pem") + #expect(delegate.pemPrivateKey == "orderkey.pem") + #expect(delegate.pemPrivateKeyPassword == nil) + #expect(delegate.sslBinary == URL(fileURLWithPath: "/usr/bin/openssl")) + #expect(!delegate.generateSignatureFile(in: URL(fileURLWithPath: ""))) } } @@ -424,9 +395,7 @@ final class DefaultOrdersDelegate: OrdersDelegate { func template(for order: O, db: any Database) async throws -> URL { URL(fileURLWithPath: "") } - func encode( - order: O, db: any Database, encoder: JSONEncoder - ) async throws -> Data { + func encode(order: O, db: any Database, encoder: JSONEncoder) async throws -> Data { Data() } } diff --git a/Tests/OrdersTests/withApp.swift b/Tests/OrdersTests/withApp.swift new file mode 100644 index 0000000..ab5df1d --- /dev/null +++ b/Tests/OrdersTests/withApp.swift @@ -0,0 +1,38 @@ +import FluentKit +import FluentSQLiteDriver +import Orders +import PassKit +import Testing +import Vapor +import Zip + +func withApp( + delegate: some OrdersDelegate, + _ body: (Application, OrdersService) async throws -> Void +) async throws { + let app = try await Application.make(.testing) + + try #require(isLoggingConfigured) + + app.databases.use(.sqlite(.memory), as: .sqlite) + + OrdersService.register(migrations: app.migrations) + app.migrations.add(CreateOrderData()) + let passesService = try OrdersService( + app: app, + delegate: delegate, + pushRoutesMiddleware: SecretMiddleware(secret: "foo"), + logger: app.logger + ) + + app.databases.middleware.use(OrderDataMiddleware(service: passesService), on: .sqlite) + + try await app.autoMigrate() + + Zip.addCustomFileExtension("order") + + try await body(app, passesService) + + try await app.autoRevert() + try await app.asyncShutdown() +} diff --git a/Tests/PassesTests/PassesTests.swift b/Tests/PassesTests/PassesTests.swift index 55a8b20..7fd7032 100644 --- a/Tests/PassesTests/PassesTests.swift +++ b/Tests/PassesTests/PassesTests.swift @@ -503,19 +503,19 @@ struct PassesTests { } @Test func defaultDelegate() async throws { - let delegate = DefaultPassesDelegate() - #expect(delegate.wwdrCertificate == "WWDR.pem") - #expect(delegate.pemCertificate == "passcertificate.pem") - #expect(delegate.pemPrivateKey == "passkey.pem") - #expect(delegate.pemPrivateKeyPassword == nil) - #expect(delegate.sslBinary == URL(fileURLWithPath: "/usr/bin/openssl")) - #expect(!delegate.generateSignatureFile(in: URL(fileURLWithPath: ""))) + let defaultDelegate = DefaultPassesDelegate() + #expect(defaultDelegate.wwdrCertificate == "WWDR.pem") + #expect(defaultDelegate.pemCertificate == "passcertificate.pem") + #expect(defaultDelegate.pemPrivateKey == "passkey.pem") + #expect(defaultDelegate.pemPrivateKeyPassword == nil) + #expect(defaultDelegate.sslBinary == URL(fileURLWithPath: "/usr/bin/openssl")) + #expect(!defaultDelegate.generateSignatureFile(in: URL(fileURLWithPath: ""))) try await withApp(delegate: delegate) { app, passesService in let passData = PassData(title: "Test Pass") try await passData.create(on: app.db) let pass = try await passData.$pass.get(on: app.db) - let data = try await delegate.encodePersonalization(for: pass, db: app.db, encoder: JSONEncoder()) + let data = try await defaultDelegate.encodePersonalization(for: pass, db: app.db, encoder: JSONEncoder()) #expect(data == nil) } } From 190fb19d728f4ac67e3d6fb3846e6420f112e336 Mon Sep 17 00:00:00 2001 From: Francesco Paolo Severino Date: Fri, 11 Oct 2024 20:18:26 +0200 Subject: [PATCH 3/7] Add test coverage for `SecretMiddleware` --- Tests/OrdersTests/OrdersTests.swift | 10 ++++++++++ Tests/PassesTests/PassesTests.swift | 10 ++++++++++ 2 files changed, 20 insertions(+) diff --git a/Tests/OrdersTests/OrdersTests.swift b/Tests/OrdersTests/OrdersTests.swift index aaddc8e..c03ab97 100644 --- a/Tests/OrdersTests/OrdersTests.swift +++ b/Tests/OrdersTests/OrdersTests.swift @@ -322,6 +322,16 @@ struct OrdersTests { let deviceLibraryIdentifier = "abcdefg" let pushToken = "1234567890" + // Test call with incorrect secret + try await app.test( + .POST, + "\(ordersURI)push/\(order.orderTypeIdentifier)/\(order.requireID())", + headers: ["X-Secret": "bar"], + afterResponse: { res async throws in + #expect(res.status == .unauthorized) + } + ) + try await app.test( .POST, "\(ordersURI)push/\(order.orderTypeIdentifier)/\(order.requireID())", diff --git a/Tests/PassesTests/PassesTests.swift b/Tests/PassesTests/PassesTests.swift index 7fd7032..0a34811 100644 --- a/Tests/PassesTests/PassesTests.swift +++ b/Tests/PassesTests/PassesTests.swift @@ -444,6 +444,16 @@ struct PassesTests { let deviceLibraryIdentifier = "abcdefg" let pushToken = "1234567890" + // Test call with incorrect secret + try await app.test( + .POST, + "\(passesURI)push/\(pass.passTypeIdentifier)/\(pass.requireID())", + headers: ["X-Secret": "bar"], + afterResponse: { res async throws in + #expect(res.status == .unauthorized) + } + ) + try await app.test( .POST, "\(passesURI)push/\(pass.passTypeIdentifier)/\(pass.requireID())", From f33856c35594827da25c4aa1bafcef9ccbf1669a Mon Sep 17 00:00:00 2001 From: Francesco Paolo Severino <96546612+fpseverino@users.noreply.github.com> Date: Sat, 12 Oct 2024 11:20:20 +0200 Subject: [PATCH 4/7] Update Tests/PassesTests/PassesTests.swift Co-authored-by: Tim Condon <0xTim@users.noreply.github.com> --- Tests/PassesTests/PassesTests.swift | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/Tests/PassesTests/PassesTests.swift b/Tests/PassesTests/PassesTests.swift index 0a34811..466be02 100644 --- a/Tests/PassesTests/PassesTests.swift +++ b/Tests/PassesTests/PassesTests.swift @@ -39,7 +39,8 @@ struct PassesTests { } } - @Test func passesGeneration() async throws { + @Test("Test generating multiple passes") + func passesGeneration() async throws { try await withApp(delegate: delegate) { app, passesService in let passData1 = PassData(title: "Test Pass 1") try await passData1.create(on: app.db) From ee3000c9d8ce4e3cf1ac65ce6007c191f3165080 Mon Sep 17 00:00:00 2001 From: Francesco Paolo Severino Date: Sat, 12 Oct 2024 11:31:46 +0200 Subject: [PATCH 5/7] Add labels to tests and suites --- Tests/OrdersTests/EncryptedOrdersTests.swift | 7 +++-- Tests/OrdersTests/OrdersTests.swift | 23 ++++++++++------ Tests/PassesTests/EncryptedPassesTests.swift | 10 +++++-- Tests/PassesTests/PassesTests.swift | 29 +++++++++++++------- 4 files changed, 46 insertions(+), 23 deletions(-) diff --git a/Tests/OrdersTests/EncryptedOrdersTests.swift b/Tests/OrdersTests/EncryptedOrdersTests.swift index 5ddf2cc..78e6334 100644 --- a/Tests/OrdersTests/EncryptedOrdersTests.swift +++ b/Tests/OrdersTests/EncryptedOrdersTests.swift @@ -6,11 +6,13 @@ import Zip @testable import Orders +@Suite("Orders Tests with encrypted PEM key") struct EncryptedOrdersTests { let delegate = EncryptedOrdersDelegate() let ordersURI = "/api/orders/v1/" - @Test func orderGeneration() async throws { + @Test("Test order generation") + func orderGeneration() async throws { try await withApp(delegate: delegate) { app, ordersService in let orderData = OrderData(title: "Test Order") try await orderData.create(on: app.db) @@ -36,7 +38,8 @@ struct EncryptedOrdersTests { } } - @Test func apnsClient() async throws { + @Test("Test APNS client") + func apnsClient() async throws { try await withApp(delegate: delegate) { app, ordersService in #expect(app.apns.client(.init(string: "orders")) != nil) diff --git a/Tests/OrdersTests/OrdersTests.swift b/Tests/OrdersTests/OrdersTests.swift index c03ab97..84d5edf 100644 --- a/Tests/OrdersTests/OrdersTests.swift +++ b/Tests/OrdersTests/OrdersTests.swift @@ -6,11 +6,13 @@ import Zip @testable import Orders +@Suite("Orders Tests") struct OrdersTests { let delegate = TestOrdersDelegate() let ordersURI = "/api/orders/v1/" - @Test func orderGeneration() async throws { + @Test("Test order generation") + func orderGeneration() async throws { try await withApp(delegate: delegate) { app, ordersService in let orderData = OrderData(title: "Test Order") try await orderData.create(on: app.db) @@ -37,8 +39,8 @@ struct OrdersTests { } } - // Tests the API Apple Wallet calls to get orders - @Test func getOrderFromAPI() async throws { + @Test("Getting order from Apple Wallet API") + func getOrderFromAPI() async throws { try await withApp(delegate: delegate) { app, ordersService in let orderData = OrderData(title: "Test Order") try await orderData.create(on: app.db) @@ -113,7 +115,8 @@ struct OrdersTests { } } - @Test func apiDeviceRegistration() async throws { + @Test("Test device registration API") + func apiDeviceRegistration() async throws { try await withApp(delegate: delegate) { app, ordersService in let orderData = OrderData(title: "Test Order") try await orderData.create(on: app.db) @@ -265,7 +268,8 @@ struct OrdersTests { } } - @Test func errorLog() async throws { + @Test("Test error logging") + func errorLog() async throws { try await withApp(delegate: delegate) { app, ordersService in let log1 = "Error 1" let log2 = "Error 2" @@ -309,7 +313,8 @@ struct OrdersTests { } } - @Test func apnsClient() async throws { + @Test("Test APNS client") + func apnsClient() async throws { try await withApp(delegate: delegate) { app, ordersService in #expect(app.apns.client(.init(string: "orders")) != nil) @@ -382,14 +387,16 @@ struct OrdersTests { } } - @Test func ordersError() { + @Test("Test OrdersError") + func ordersError() { #expect(OrdersError.templateNotDirectory.description == "OrdersError(errorType: templateNotDirectory)") #expect(OrdersError.pemCertificateMissing.description == "OrdersError(errorType: pemCertificateMissing)") #expect(OrdersError.pemPrivateKeyMissing.description == "OrdersError(errorType: pemPrivateKeyMissing)") #expect(OrdersError.opensslBinaryMissing.description == "OrdersError(errorType: opensslBinaryMissing)") } - @Test func defaultDelegate() { + @Test("Test default OrdersDelegate properties") + func defaultDelegate() { let delegate = DefaultOrdersDelegate() #expect(delegate.wwdrCertificate == "WWDR.pem") #expect(delegate.pemCertificate == "ordercertificate.pem") diff --git a/Tests/PassesTests/EncryptedPassesTests.swift b/Tests/PassesTests/EncryptedPassesTests.swift index af17d4f..5596d03 100644 --- a/Tests/PassesTests/EncryptedPassesTests.swift +++ b/Tests/PassesTests/EncryptedPassesTests.swift @@ -5,11 +5,13 @@ import Zip @testable import Passes +@Suite("Passes Tests with encrypted PEM key") struct EncryptedPassesTests { let delegate = EncryptedPassesDelegate() let passesURI = "/api/passes/v1/" - @Test func passGeneration() async throws { + @Test("Test pass generation") + func passGeneration() async throws { try await withApp(delegate: delegate) { app, passesService in let passData = PassData(title: "Test Pass") try await passData.create(on: app.db) @@ -36,7 +38,8 @@ struct EncryptedPassesTests { } } - @Test func personalizationAPI() async throws { + @Test("Personalizable pass Apple Wallet API") + func personalizationAPI() async throws { try await withApp(delegate: delegate) { app, passesService in let passData = PassData(title: "Personalize") try await passData.create(on: app.db) @@ -82,7 +85,8 @@ struct EncryptedPassesTests { } } - @Test func apnsClient() async throws { + @Test("Test APNS client") + func apnsClient() async throws { try await withApp(delegate: delegate) { app, passesService in #expect(app.apns.client(.init(string: "passes")) != nil) diff --git a/Tests/PassesTests/PassesTests.swift b/Tests/PassesTests/PassesTests.swift index 466be02..a5c8451 100644 --- a/Tests/PassesTests/PassesTests.swift +++ b/Tests/PassesTests/PassesTests.swift @@ -6,11 +6,13 @@ import Zip @testable import Passes +@Suite("Passes Tests") struct PassesTests { let delegate = TestPassesDelegate() let passesURI = "/api/passes/v1/" - @Test func passGeneration() async throws { + @Test("Test pass generation") + func passGeneration() async throws { try await withApp(delegate: delegate) { app, passesService in let passData = PassData(title: "Test Pass") try await passData.create(on: app.db) @@ -62,7 +64,8 @@ struct PassesTests { } } - @Test func personalization() async throws { + @Test("Test personalizable pass") + func personalization() async throws { try await withApp(delegate: delegate) { app, passesService in let passData = PassData(title: "Personalize") try await passData.create(on: app.db) @@ -91,8 +94,8 @@ struct PassesTests { } } - // Tests the API Apple Wallet calls to get passes - @Test func getPassFromAPI() async throws { + @Test("Getting pass from Apple Wallet API") + func getPassFromAPI() async throws { try await withApp(delegate: delegate) { app, passesService in let passData = PassData(title: "Test Pass") try await passData.create(on: app.db) @@ -167,7 +170,8 @@ struct PassesTests { } } - @Test func personalizationAPI() async throws { + @Test("Personalizable pass Apple Wallet API") + func personalizationAPI() async throws { try await withApp(delegate: delegate) { app, passesService in let passData = PassData(title: "Personalize") try await passData.create(on: app.db) @@ -236,7 +240,8 @@ struct PassesTests { } } - @Test func apiDeviceRegistration() async throws { + @Test("Test device registration API") + func apiDeviceRegistration() async throws { try await withApp(delegate: delegate) { app, passesService in let passData = PassData(title: "Test Pass") try await passData.create(on: app.db) @@ -388,7 +393,8 @@ struct PassesTests { } } - @Test func errorLog() async throws { + @Test("Test error logging") + func errorLog() async throws { try await withApp(delegate: delegate) { app, passesService in let log1 = "Error 1" let log2 = "Error 2" @@ -432,7 +438,8 @@ struct PassesTests { } } - @Test func apnsClient() async throws { + @Test("Test APNS client") + func apnsClient() async throws { try await withApp(delegate: delegate) { app, passesService in #expect(app.apns.client(.init(string: "passes")) != nil) @@ -505,7 +512,8 @@ struct PassesTests { } } - @Test func passesError() { + @Test("Test PassesError") + func passesError() { #expect(PassesError.templateNotDirectory.description == "PassesError(errorType: templateNotDirectory)") #expect(PassesError.pemCertificateMissing.description == "PassesError(errorType: pemCertificateMissing)") #expect(PassesError.pemPrivateKeyMissing.description == "PassesError(errorType: pemPrivateKeyMissing)") @@ -513,7 +521,8 @@ struct PassesTests { #expect(PassesError.invalidNumberOfPasses.description == "PassesError(errorType: invalidNumberOfPasses)") } - @Test func defaultDelegate() async throws { + @Test("Test default PassesDelegate properties") + func defaultDelegate() async throws { let defaultDelegate = DefaultPassesDelegate() #expect(defaultDelegate.wwdrCertificate == "WWDR.pem") #expect(defaultDelegate.pemCertificate == "passcertificate.pem") From 47089984719e16df23995e6af18904893b9096a6 Mon Sep 17 00:00:00 2001 From: Francesco Paolo Severino Date: Sat, 12 Oct 2024 11:35:12 +0200 Subject: [PATCH 6/7] Update Swift compatibility badge --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 6f978d7..4ec7ae8 100644 --- a/README.md +++ b/README.md @@ -13,7 +13,7 @@ - Swift 5.10+ + Swift 6.0+
From 6e8d07c453b9b6815fbbd1458937ecc61f404dee Mon Sep 17 00:00:00 2001 From: Francesco Paolo Severino Date: Sat, 12 Oct 2024 15:02:57 +0200 Subject: [PATCH 7/7] Fix test names --- Tests/OrdersTests/EncryptedOrdersTests.swift | 6 +++--- Tests/OrdersTests/OrdersTests.swift | 14 +++++++------- Tests/PassesTests/EncryptedPassesTests.swift | 8 ++++---- Tests/PassesTests/PassesTests.swift | 20 ++++++++++---------- 4 files changed, 24 insertions(+), 24 deletions(-) diff --git a/Tests/OrdersTests/EncryptedOrdersTests.swift b/Tests/OrdersTests/EncryptedOrdersTests.swift index 78e6334..368b7dc 100644 --- a/Tests/OrdersTests/EncryptedOrdersTests.swift +++ b/Tests/OrdersTests/EncryptedOrdersTests.swift @@ -6,12 +6,12 @@ import Zip @testable import Orders -@Suite("Orders Tests with encrypted PEM key") +@Suite("Orders Tests with Encrypted PEM Key") struct EncryptedOrdersTests { let delegate = EncryptedOrdersDelegate() let ordersURI = "/api/orders/v1/" - @Test("Test order generation") + @Test("Order Generation") func orderGeneration() async throws { try await withApp(delegate: delegate) { app, ordersService in let orderData = OrderData(title: "Test Order") @@ -38,7 +38,7 @@ struct EncryptedOrdersTests { } } - @Test("Test APNS client") + @Test("APNS Client") func apnsClient() async throws { try await withApp(delegate: delegate) { app, ordersService in #expect(app.apns.client(.init(string: "orders")) != nil) diff --git a/Tests/OrdersTests/OrdersTests.swift b/Tests/OrdersTests/OrdersTests.swift index 84d5edf..c7d1557 100644 --- a/Tests/OrdersTests/OrdersTests.swift +++ b/Tests/OrdersTests/OrdersTests.swift @@ -11,7 +11,7 @@ struct OrdersTests { let delegate = TestOrdersDelegate() let ordersURI = "/api/orders/v1/" - @Test("Test order generation") + @Test("Order Generation") func orderGeneration() async throws { try await withApp(delegate: delegate) { app, ordersService in let orderData = OrderData(title: "Test Order") @@ -39,7 +39,7 @@ struct OrdersTests { } } - @Test("Getting order from Apple Wallet API") + @Test("Getting Order from Apple Wallet API") func getOrderFromAPI() async throws { try await withApp(delegate: delegate) { app, ordersService in let orderData = OrderData(title: "Test Order") @@ -115,7 +115,7 @@ struct OrdersTests { } } - @Test("Test device registration API") + @Test("Device Registration API") func apiDeviceRegistration() async throws { try await withApp(delegate: delegate) { app, ordersService in let orderData = OrderData(title: "Test Order") @@ -268,7 +268,7 @@ struct OrdersTests { } } - @Test("Test error logging") + @Test("Error Logging") func errorLog() async throws { try await withApp(delegate: delegate) { app, ordersService in let log1 = "Error 1" @@ -313,7 +313,7 @@ struct OrdersTests { } } - @Test("Test APNS client") + @Test("APNS Client") func apnsClient() async throws { try await withApp(delegate: delegate) { app, ordersService in #expect(app.apns.client(.init(string: "orders")) != nil) @@ -387,7 +387,7 @@ struct OrdersTests { } } - @Test("Test OrdersError") + @Test("OrdersError") func ordersError() { #expect(OrdersError.templateNotDirectory.description == "OrdersError(errorType: templateNotDirectory)") #expect(OrdersError.pemCertificateMissing.description == "OrdersError(errorType: pemCertificateMissing)") @@ -395,7 +395,7 @@ struct OrdersTests { #expect(OrdersError.opensslBinaryMissing.description == "OrdersError(errorType: opensslBinaryMissing)") } - @Test("Test default OrdersDelegate properties") + @Test("Default OrdersDelegate Properties") func defaultDelegate() { let delegate = DefaultOrdersDelegate() #expect(delegate.wwdrCertificate == "WWDR.pem") diff --git a/Tests/PassesTests/EncryptedPassesTests.swift b/Tests/PassesTests/EncryptedPassesTests.swift index 5596d03..f533c6e 100644 --- a/Tests/PassesTests/EncryptedPassesTests.swift +++ b/Tests/PassesTests/EncryptedPassesTests.swift @@ -5,12 +5,12 @@ import Zip @testable import Passes -@Suite("Passes Tests with encrypted PEM key") +@Suite("Passes Tests with Encrypted PEM Key") struct EncryptedPassesTests { let delegate = EncryptedPassesDelegate() let passesURI = "/api/passes/v1/" - @Test("Test pass generation") + @Test("Pass Generation") func passGeneration() async throws { try await withApp(delegate: delegate) { app, passesService in let passData = PassData(title: "Test Pass") @@ -38,7 +38,7 @@ struct EncryptedPassesTests { } } - @Test("Personalizable pass Apple Wallet API") + @Test("Personalizable Pass Apple Wallet API") func personalizationAPI() async throws { try await withApp(delegate: delegate) { app, passesService in let passData = PassData(title: "Personalize") @@ -85,7 +85,7 @@ struct EncryptedPassesTests { } } - @Test("Test APNS client") + @Test("APNS Client") func apnsClient() async throws { try await withApp(delegate: delegate) { app, passesService in #expect(app.apns.client(.init(string: "passes")) != nil) diff --git a/Tests/PassesTests/PassesTests.swift b/Tests/PassesTests/PassesTests.swift index a5c8451..7742d25 100644 --- a/Tests/PassesTests/PassesTests.swift +++ b/Tests/PassesTests/PassesTests.swift @@ -11,7 +11,7 @@ struct PassesTests { let delegate = TestPassesDelegate() let passesURI = "/api/passes/v1/" - @Test("Test pass generation") + @Test("Pass Generation") func passGeneration() async throws { try await withApp(delegate: delegate) { app, passesService in let passData = PassData(title: "Test Pass") @@ -41,7 +41,7 @@ struct PassesTests { } } - @Test("Test generating multiple passes") + @Test("Generating Multiple Passes") func passesGeneration() async throws { try await withApp(delegate: delegate) { app, passesService in let passData1 = PassData(title: "Test Pass 1") @@ -64,7 +64,7 @@ struct PassesTests { } } - @Test("Test personalizable pass") + @Test("Personalizable Passes") func personalization() async throws { try await withApp(delegate: delegate) { app, passesService in let passData = PassData(title: "Personalize") @@ -94,7 +94,7 @@ struct PassesTests { } } - @Test("Getting pass from Apple Wallet API") + @Test("Getting Pass from Apple Wallet API") func getPassFromAPI() async throws { try await withApp(delegate: delegate) { app, passesService in let passData = PassData(title: "Test Pass") @@ -170,7 +170,7 @@ struct PassesTests { } } - @Test("Personalizable pass Apple Wallet API") + @Test("Personalizable Pass Apple Wallet API") func personalizationAPI() async throws { try await withApp(delegate: delegate) { app, passesService in let passData = PassData(title: "Personalize") @@ -240,7 +240,7 @@ struct PassesTests { } } - @Test("Test device registration API") + @Test("Device Registration API") func apiDeviceRegistration() async throws { try await withApp(delegate: delegate) { app, passesService in let passData = PassData(title: "Test Pass") @@ -393,7 +393,7 @@ struct PassesTests { } } - @Test("Test error logging") + @Test("Error Logging") func errorLog() async throws { try await withApp(delegate: delegate) { app, passesService in let log1 = "Error 1" @@ -438,7 +438,7 @@ struct PassesTests { } } - @Test("Test APNS client") + @Test("APNS Client") func apnsClient() async throws { try await withApp(delegate: delegate) { app, passesService in #expect(app.apns.client(.init(string: "passes")) != nil) @@ -512,7 +512,7 @@ struct PassesTests { } } - @Test("Test PassesError") + @Test("PassesError") func passesError() { #expect(PassesError.templateNotDirectory.description == "PassesError(errorType: templateNotDirectory)") #expect(PassesError.pemCertificateMissing.description == "PassesError(errorType: pemCertificateMissing)") @@ -521,7 +521,7 @@ struct PassesTests { #expect(PassesError.invalidNumberOfPasses.description == "PassesError(errorType: invalidNumberOfPasses)") } - @Test("Test default PassesDelegate properties") + @Test("Default PassesDelegate Properties") func defaultDelegate() async throws { let defaultDelegate = DefaultPassesDelegate() #expect(defaultDelegate.wwdrCertificate == "WWDR.pem")