Move integration test flows that don't require a backend to the UI tests

In an attempt to make them faster and less flakey.
This commit is contained in:
Stefan Ceriu
2026-04-28 13:03:43 +03:00
parent aa36bebd84
commit 5ec47c2580
3 changed files with 148 additions and 153 deletions

View File

@@ -22,5 +22,7 @@ extension UserSessionMock {
voiceMessageMediaManager = VoiceMessageMediaManagerMock()
sessionSecurityStatePublisher = CurrentValueSubject<SessionSecurityState, Never>(.init(verificationState: .verified, recoveryState: .enabled)).asCurrentValuePublisher()
liveLocationManager = LiveLocationManagerMock(.init())
}
}

View File

@@ -23,10 +23,6 @@ class UserFlowTests: XCTestCase {
func testUserFlow() {
checkRoomFlows()
checkSettings()
checkRoomCreation()
app.logout()
}
@@ -52,16 +48,8 @@ class UserFlowTests: XCTestCase {
sendMessages()
checkPhotoSharing()
checkDocumentSharing()
checkLocationSharing()
checkTimelineItemActionMenu()
checkRoomDetails()
// Go back to the room list
tapOnBackButton("Chats")
@@ -100,75 +88,6 @@ class UserFlowTests: XCTestCase {
app.buttons[A11yIdentifiers.roomScreen.composerToolbar.closeFormattingOptions].tap(.center)
}
private func checkPhotoSharing() {
tapOnButton(A11yIdentifiers.roomScreen.composerToolbar.openComposeOptions)
tapOnButton(A11yIdentifiers.roomScreen.attachmentPickerPhotoLibrary)
sleep(10) // Wait for the picker to load
// Tap on the second image. First one is always broken on simulators.
let secondImage = app.scrollViews.images.element(boundBy: 1)
XCTAssertTrue(secondImage.waitForExistence(timeout: 20.0)) // Photo library takes a bit to load
secondImage.tap(.center)
// Wait for the image to be processed and the new screen to appear
sleep(10)
// Cancel the upload flow
tapOnButton("Cancel", waitForDisappearance: true)
}
private func checkDocumentSharing() {
tapOnButton(A11yIdentifiers.roomScreen.composerToolbar.openComposeOptions)
tapOnButton(A11yIdentifiers.roomScreen.attachmentPickerDocuments)
sleep(10) // Wait for the picker to load
tapOnButton("Cancel", waitForDisappearance: true)
}
private func checkLocationSharing() {
tapOnButton(A11yIdentifiers.roomScreen.composerToolbar.openComposeOptions)
tapOnButton(A11yIdentifiers.roomScreen.attachmentPickerLocation)
sleep(10) // Wait for the picker to load
// The order of the alerts is a bit of a mistery so try twice
allowLocationPermissionOnce()
// Handle map loading errors (missing credentials)
let alertOkButton = app.alerts.firstMatch.buttons["OK"].firstMatch
if alertOkButton.waitForExistence(timeout: 10.0) {
alertOkButton.tap(.center)
}
allowLocationPermissionOnce()
tapOnButton("Close", waitForDisappearance: true)
}
private func allowLocationPermissionOnce() {
let springboard = XCUIApplication(bundleIdentifier: "com.apple.springboard")
let notificationAlertAllowButton = springboard.buttons["Allow Once"].firstMatch
if notificationAlertAllowButton.waitForExistence(timeout: 10.0) {
notificationAlertAllowButton.tap(.center)
}
}
private func checkRoomCreation() {
tapOnButton(A11yIdentifiers.homeScreen.startChat)
tapOnButton(A11yIdentifiers.startChatScreen.createRoom)
// Don't create the room, it will make the test account very noisy.
// The UI tests already test this flow with mocked data.
tapOnBackButton("Start chat")
tapOnButton("Cancel", waitForDisappearance: true)
}
private func checkTimelineItemActionMenu() {
// Long press on the last message
let lastMessage = app.cells.firstMatch
@@ -184,77 +103,6 @@ class UserFlowTests: XCTestCase {
}
}
private func checkRoomDetails() {
// Open the room details
let roomHeader = app.buttons[A11yIdentifiers.roomScreen.name]
XCTAssertTrue(roomHeader.waitForExistence(timeout: 10.0))
roomHeader.tap(.center)
// Swipe until the People button is hittable
let peopleButton = app.buttons[A11yIdentifiers.roomDetailsScreen.people]
if !peopleButton.isHittable {
var attempts = 0
while !peopleButton.isHittable, attempts < 5 {
app.swipeUp()
attempts += 1
}
}
// Open the room members list.
tapOnButton(A11yIdentifiers.roomDetailsScreen.people)
// Open the first member's details. Loading members for big rooms can take a while.
let firstRoomMember = app.scrollViews.buttons.firstMatch
XCTAssertTrue(firstRoomMember.waitForExistence(timeout: 1000.0))
firstRoomMember.tap(.center)
// Open the profile from the bottom sheet
let viewProfileButton = app.buttons[A11yIdentifiers.manageRoomMemberSheet.viewProfile]
XCTAssertTrue(viewProfileButton.waitForExistence(timeout: 10.0))
tapOnButton(A11yIdentifiers.manageRoomMemberSheet.viewProfile, waitForDisappearance: true)
// Go back to the room member details
tapOnBackButton("People")
// Go back to the room details
tapOnBackButton("Room info")
// Go back to the room
tapOnBackButton("Chat")
}
private func checkSettings() {
// On first login when multiple sheets get presented the profile button is not hittable
// Moving the scroll fixed it for some obscure reason
app.swipeDown()
let profileButton = app.buttons[A11yIdentifiers.homeScreen.userAvatar]
// `Failed to scroll to visible (by AX action) Button` https://stackoverflow.com/a/33534187/730924
profileButton.tap(.center)
// Open analytics
tapOnButton(A11yIdentifiers.settingsScreen.analytics)
// Go back to settings
tapOnBackButton("Settings")
// Open report a bug
tapOnButton(A11yIdentifiers.settingsScreen.reportBug)
// Go back to settings
tapOnBackButton("Settings")
// Open about
tapOnButton(A11yIdentifiers.settingsScreen.about)
// Go back to settings
tapOnBackButton("Settings")
// Close the settings
tapOnButton(A11yIdentifiers.settingsScreen.done)
}
private func tapOnButton(_ identifier: String, waitForDisappearance: Bool = false) {
let button = app.buttons[identifier]
XCTAssertTrue(button.waitForExistence(timeout: 10.0))

View File

@@ -76,6 +76,93 @@ class UserSessionScreenTests: XCTestCase {
XCTAssert(joinButton.waitForExistence(timeout: 10))
}
func testRoomDetails() {
let app = Application.launch(.userSessionScreen)
app.buttons[A11yIdentifiers.homeScreen.roomName(firstRoomName)].tap()
XCTAssert(app.buttons[firstRoomName].waitForExistence(timeout: 5.0))
// Open the room details
let roomHeader = app.buttons[A11yIdentifiers.roomScreen.name]
XCTAssertTrue(roomHeader.waitForExistence(timeout: 10.0))
roomHeader.tap(.center)
// Swipe until the People button is hittable
let peopleButton = app.buttons[A11yIdentifiers.roomDetailsScreen.people]
if !peopleButton.isHittable {
var attempts = 0
while !peopleButton.isHittable, attempts < 5 {
app.swipeUp()
attempts += 1
}
}
// Open the room members list.
app.buttons[A11yIdentifiers.roomDetailsScreen.people].tap()
// Open the first member's details. Loading members for big rooms can take a while.
let firstRoomMember = app.scrollViews.buttons.firstMatch
XCTAssertTrue(firstRoomMember.waitForExistence(timeout: 1000.0))
firstRoomMember.tap(.center)
// Open the profile from the bottom sheet
let viewProfileButton = app.buttons[A11yIdentifiers.manageRoomMemberSheet.viewProfile]
XCTAssertTrue(viewProfileButton.waitForExistence(timeout: 10.0))
app.buttons[A11yIdentifiers.manageRoomMemberSheet.viewProfile].tap()
// Go back to the room member details
tapOnBackButton("People", app)
// Go back to the room details
tapOnBackButton("Room info", app)
// Go back to the room
tapOnBackButton("Chat", app)
}
func testPhotoSharing() {
let app = Application.launch(.userSessionScreen)
app.buttons[A11yIdentifiers.homeScreen.roomName(firstRoomName)].tap()
XCTAssert(app.buttons[firstRoomName].waitForExistence(timeout: 5.0))
app.buttons[A11yIdentifiers.roomScreen.composerToolbar.openComposeOptions].tap()
app.buttons[A11yIdentifiers.roomScreen.attachmentPickerPhotoLibrary].tap()
// Tap on the second image. First one is always broken on simulators.
let secondImage = app.scrollViews.images.element(boundBy: 1)
XCTAssertTrue(secondImage.waitForExistence(timeout: 20.0)) // Photo library takes a bit to load
secondImage.tap(.center)
}
func testDocumentSharing() {
let app = Application.launch(.userSessionScreen)
app.buttons[A11yIdentifiers.homeScreen.roomName(firstRoomName)].tap()
XCTAssert(app.buttons[firstRoomName].waitForExistence(timeout: 5.0))
app.buttons[A11yIdentifiers.roomScreen.composerToolbar.openComposeOptions].tap()
app.buttons[A11yIdentifiers.roomScreen.attachmentPickerDocuments].tap()
}
func testLocationSharing() {
let app = Application.launch(.userSessionScreen)
app.buttons[A11yIdentifiers.homeScreen.roomName(firstRoomName)].tap()
XCTAssert(app.buttons[firstRoomName].waitForExistence(timeout: 5.0))
app.buttons[A11yIdentifiers.roomScreen.composerToolbar.openComposeOptions].tap()
app.buttons[A11yIdentifiers.roomScreen.attachmentPickerLocation].tap()
allowLocationPermissionOnce()
// Handle map loading errors (missing credentials)
let alertOkButton = app.alerts.firstMatch.buttons["OK"].firstMatch
if alertOkButton.waitForExistence(timeout: 10.0) {
alertOkButton.tap(.center)
}
}
func testSpaceExploration() async throws {
let app = Application.launch(.userSessionSpacesFlow)
@@ -188,4 +275,62 @@ class UserSessionScreenTests: XCTestCase {
try await Task.sleep(for: .seconds(1))
try await app.assertScreenshot(step: Step.spaceScreen)
}
func testSettings() {
let app = Application.launch(.userSessionScreen)
let profileButton = app.buttons[A11yIdentifiers.homeScreen.userAvatar]
// `Failed to scroll to visible (by AX action) Button` https://stackoverflow.com/a/33534187/730924
profileButton.tap(.center)
// Open analytics
app.buttons[A11yIdentifiers.settingsScreen.analytics].tap()
// Go back to settings
tapOnBackButton("Settings", app)
// Open report a bug
app.buttons[A11yIdentifiers.settingsScreen.reportBug].tap()
// Go back to settings
tapOnBackButton("Settings", app)
// Open about
app.buttons[A11yIdentifiers.settingsScreen.about].tap()
// Go back to settings
tapOnBackButton("Settings", app)
// Close the settings
app.buttons[A11yIdentifiers.settingsScreen.done].tap()
}
func testRoomCreation() {
let app = Application.launch(.userSessionScreen)
app.buttons[A11yIdentifiers.homeScreen.startChat].tap()
app.buttons[A11yIdentifiers.startChatScreen.createRoom].tap()
tapOnBackButton("Start chat", app)
}
private func allowLocationPermissionOnce() {
let springboard = XCUIApplication(bundleIdentifier: "com.apple.springboard")
let notificationAlertAllowButton = springboard.buttons["Allow Once"].firstMatch
if notificationAlertAllowButton.waitForExistence(timeout: 10.0) {
notificationAlertAllowButton.tap(.center)
}
}
/// Taps on a back button that the system configured with a label but no identifier.
///
/// When there are multiple buttons with the same label in the hierarchy, all the buttons we created
/// should have an identifier set, and so this method will ignore those picking the one with only a label.
private func tapOnBackButton(_ label: String = "Back", _ app: XCUIApplication) {
let button = app.buttons.matching(NSPredicate(format: "label == %@ && identifier == 'BackButton'", label)).firstMatch
XCTAssertTrue(button.waitForExistence(timeout: 10.0))
button.tap(.center)
}
}