c52ce4d497
Migrate the XCUITest suite off the legacy shared-account model (and the prior Django-style auth assumptions) to a parallel-safe, domain-organized architecture, validated end-to-end against the live Kratos stack. Isolation (parallel-safe by construction): - Core/Fixtures/TestAccount.swift: each test mints its own pre-verified Kratos identity (uit_<domain>_<uuid>@test.honeydue.local), logs in, seeds under its own token, and deletes the identity in teardown (cascading all data + clearing Kratos). No shared testuser; parallel workers no longer race. - AuthenticatedUITestCase rewritten to that model (member surface preserved); adds requiresResidence / seedAccountPreconditions to seed UI-gated data BEFORE login (a fresh account is empty at login). Organization (255 tests preserved, none dropped): - 21 domain suites under Auth/ Onboarding/ Residence/ Task/ Contractor/ Document/ Sharing/ Navigation/ Smoke/ CrossCutting/ E2E/, consistent <Domain>UITests naming. Removes the Suite1..11 / AAA_ / ZZ_ / Tests/Rebuild naming chaos and the overlapping task/residence/auth suites. Runner + test plans: - run_ui_tests.sh: Smoke gate -> Seed -> Parallel(8 workers) -> Sweep. The parallel phase runs the whole target minus phase-managed suites via -skip-testing, so new suites auto-include (no hand-maintained list to drift). Drops the 2-worker cap and Suite6 isolation (isolation made them moot). - HoneyDueUITests.xctestplan skips the 4 phase-managed suites; adds Smoke.xctestplan. Kratos auth fixes folded in (login/verify/reset endpoints removed under Kratos): real Mailpit verification codes replace the obsolete fixed "123456"; teardown deletes Kratos identities; admin-panel login uses the correct seeded password. Build green; isolation, parallelism, and the precondition/sharing migrations validated against the live stack (0 leaked accounts). Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
84 lines
2.7 KiB
Swift
84 lines
2.7 KiB
Swift
import Foundation
|
|
import XCTest
|
|
|
|
/// High-level account lifecycle management for UI tests.
|
|
enum TestAccountManager {
|
|
|
|
// MARK: - Credential Generation
|
|
|
|
/// Generate unique credentials with a timestamp + random suffix to avoid collisions.
|
|
static func uniqueCredentials(prefix: String = "uit") -> (username: String, email: String, password: String) {
|
|
let stamp = Int(Date().timeIntervalSince1970)
|
|
let random = Int.random(in: 1000...9999)
|
|
let username = "\(prefix)_\(stamp)_\(random)"
|
|
let email = "\(username)@test.example.com"
|
|
let password = "Pass\(stamp)!"
|
|
return (username, email, password)
|
|
}
|
|
|
|
// MARK: - Account Creation
|
|
|
|
/// Create a verified account via the backend API. Returns a ready-to-use session.
|
|
/// Calls `XCTFail` and returns nil if any step fails.
|
|
static func createVerifiedAccount(
|
|
file: StaticString = #filePath,
|
|
line: UInt = #line
|
|
) -> TestSession? {
|
|
let creds = uniqueCredentials()
|
|
|
|
guard let session = TestAccountAPIClient.createVerifiedAccount(
|
|
username: creds.username,
|
|
email: creds.email,
|
|
password: creds.password
|
|
) else {
|
|
XCTFail("Failed to create verified account for \(creds.username)", file: file, line: line)
|
|
return nil
|
|
}
|
|
|
|
return session
|
|
}
|
|
|
|
/// Create an unverified account (Kratos identity with an unverified email).
|
|
/// Useful for testing the verification gate. Returns a ready-to-use session.
|
|
static func createUnverifiedAccount(
|
|
file: StaticString = #filePath,
|
|
line: UInt = #line
|
|
) -> TestSession? {
|
|
let creds = uniqueCredentials()
|
|
|
|
guard let session = TestAccountAPIClient.createUnverifiedAccount(
|
|
username: creds.username,
|
|
email: creds.email,
|
|
password: creds.password
|
|
) else {
|
|
XCTFail("Failed to create unverified account for \(creds.username)", file: file, line: line)
|
|
return nil
|
|
}
|
|
|
|
return session
|
|
}
|
|
|
|
// MARK: - Seeded Accounts
|
|
|
|
/// Login with a pre-seeded account that already exists in the database.
|
|
static func loginSeededAccount(
|
|
username: String = "admin",
|
|
password: String = "Test1234",
|
|
file: StaticString = #filePath,
|
|
line: UInt = #line
|
|
) -> TestSession? {
|
|
guard let response = TestAccountAPIClient.login(username: username, password: password) else {
|
|
XCTFail("Failed to login seeded account '\(username)'", file: file, line: line)
|
|
return nil
|
|
}
|
|
|
|
return TestSession(
|
|
token: response.token,
|
|
user: response.user,
|
|
username: username,
|
|
password: password
|
|
)
|
|
}
|
|
|
|
}
|