7cdd88b11a
Document the two-target layout, per-test Kratos account isolation, the seed-before-login precondition rules (requiresResidence / seedAccountPreconditions), how to run the phased runner, and how to add a suite. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
2.2 KiB
2.2 KiB
UI Test Rules
These rules are non-negotiable. Every test, every suite, every helper must follow them.
Element Interaction
- All text fields use
fillTextField(identifier:)— never rawtap()+typeText()in test bodies - All buttons/elements found by accessibility identifier — never
label CONTAINSfor app elements - No coordinate taps anywhere —
app.coordinate(withNormalizedOffset:)is banned
Timeouts
defaultTimeout= 2 seconds — if an element on the current screen isn't there in 2s, the app is brokennavigationTimeout= 5 seconds — screen transitions, tab switchesloginTimeout= 15 seconds — initial auth flow only (cold start)- No retry loops in test helpers — tap once, check once, fail fast
Independence
- Every suite runs alone, in combination, or in parallel — no ordering dependencies
- Every test gets its OWN isolated account —
AuthenticatedUITestCasemints a fresh Kratos identity per test (uit_<domain>_<uuid>@test.honeydue.local), seeds under its own token, and deletes it in tearDown (cascading all data). Never share an account across tests. - No shared mutable state — no
static var, no class-level properties mutated across tests
Clarity
- One logical assertion per test — test name describes the exact behavior
XCTFailwith a message that tells you what went wrong without reading the code- No
guard ... else { return }that silently passes — if a precondition fails,XCTFailand stop
Speed
- No
sleep(),usleep(), orThread.sleepin tests — condition-based waits only - If
focusAndTypecan't get focus in one tap, the test fails — no 3-attempt retry loops - Target: each individual test completes in under 15 seconds (excluding setUp/tearDown)
Preconditions
- Seed UI-gated preconditions BEFORE login — a fresh account is empty at login, so data the UI must display has to be seeded before the app loads it. Use
requiresResidence(seeds a residence) or overrideseedAccountPreconditions(_:)to seed a full scenario via API. Preconditions are infrastructure, not what the test is testing — establish them via API, never UI.