// // Copyright 2022 New Vector Ltd // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // import XCTest @testable import ElementX @MainActor class LoginViewModelTests: XCTestCase { let defaultHomeserver = LoginHomeserver.mockMatrixDotOrg var viewModel: LoginScreenViewModelProtocol! var context: LoginScreenViewModelType.Context! @MainActor override func setUp() async throws { viewModel = LoginScreenViewModel(homeserver: defaultHomeserver) context = viewModel.context } func testMatrixDotOrg() { // Given the initial view model configured for matrix.org. let homeserver = defaultHomeserver // Then the view state should contain a homeserver that matches matrix.org and show the login form. XCTAssertEqual(context.viewState.homeserver, homeserver, "The homeserver data should match the original.") XCTAssertEqual(context.viewState.loginMode, .password, "The login form should be shown.") } func testBasicServer() { // Given a basic server example.com that only supports password registration. let homeserver = LoginHomeserver.mockBasicServer // When updating the view model with the server. viewModel.update(homeserver: homeserver) // Then the view state should be updated with the homeserver and show the login form. XCTAssertEqual(context.viewState.homeserver, homeserver, "The homeserver data should should match the new homeserver.") XCTAssertEqual(context.viewState.loginMode, .password, "The login form should be shown.") } func testUsernameWithEmptyPassword() { // Given a form with an empty username and password. XCTAssertTrue(context.password.isEmpty, "The initial value for the password should be empty.") XCTAssertTrue(context.username.isEmpty, "The initial value for the username should be empty.") XCTAssertFalse(context.viewState.hasValidCredentials, "The credentials should be invalid.") XCTAssertFalse(context.viewState.canSubmit, "The form should be blocked for submission.") // When entering a username without a password. context.username = "bob" context.password = "" // Then the credentials should be considered invalid. XCTAssertFalse(context.viewState.hasValidCredentials, "The credentials should be invalid.") XCTAssertFalse(context.viewState.canSubmit, "The form should be blocked for submission.") } func testEmptyUsernameWithPassword() { // Given a form with an empty username and password. XCTAssertTrue(context.password.isEmpty, "The initial value for the password should be empty.") XCTAssertTrue(context.username.isEmpty, "The initial value for the username should be empty.") XCTAssertFalse(context.viewState.hasValidCredentials, "The credentials should be invalid.") XCTAssertFalse(context.viewState.canSubmit, "The form should be blocked for submission.") // When entering a password without a username. context.username = "" context.password = "12345678" // Then the credentials should be considered invalid. XCTAssertFalse(context.viewState.hasValidCredentials, "The credentials should be invalid.") XCTAssertFalse(context.viewState.canSubmit, "The form should be blocked for submission.") } func testValidCredentials() { // Given a form with an empty username and password. XCTAssertTrue(context.password.isEmpty, "The initial value for the password should be empty.") XCTAssertTrue(context.username.isEmpty, "The initial value for the username should be empty.") XCTAssertFalse(context.viewState.hasValidCredentials, "The credentials should be invalid.") XCTAssertFalse(context.viewState.canSubmit, "The form should be blocked for submission.") // When entering a username and an 8-character password. context.username = "bob" context.password = "12345678" // Then the credentials should be considered valid. XCTAssertTrue(context.viewState.hasValidCredentials, "The credentials should be valid when the username and password are valid.") XCTAssertTrue(context.viewState.canSubmit, "The form should be ready to submit.") } func testLoadingServer() { // Given a form with valid credentials. context.username = "bob" context.password = "12345678" XCTAssertTrue(context.viewState.hasValidCredentials, "The credentials should be valid.") XCTAssertFalse(context.viewState.isLoading, "The view shouldn't start in a loading state.") XCTAssertTrue(context.viewState.canSubmit, "The form should be ready to submit.") // When updating the view model whilst loading a homeserver. viewModel.update(isLoading: true) // Then the view state should reflect that the homeserver is loading. XCTAssertTrue(context.viewState.isLoading, "The view should now be in a loading state.") XCTAssertFalse(context.viewState.canSubmit, "The form should be blocked for submission.") // When updating the view model after loading a homeserver. viewModel.update(isLoading: false) // Then the view state should reflect that the homeserver is now loaded. XCTAssertFalse(context.viewState.isLoading, "The view should be back in a loaded state.") XCTAssertTrue(context.viewState.canSubmit, "The form should be ready to submit.") } func testOIDCServer() { // Given a basic server example.com that supports OIDC registration. let homeserver = LoginHomeserver.mockOIDC // When updating the view model with the server. viewModel.update(homeserver: homeserver) // Then the view state should be updated with the homeserver and show the OIDC button. XCTAssertTrue(context.viewState.loginMode.supportsOIDCFlow, "The OIDC button should be shown.") } func testLogsForPassword() { // Given the coordinator and view model results that contain passwords. let password = "supersecretpassword" let viewModelAction: LoginScreenViewModelAction = .login(username: "Alice", password: password) // When creating a string representation of those results (e.g. for logging). let viewModelActionString = "\(viewModelAction)" // Then the password should not be included in that string. XCTAssertFalse("\(viewModelActionString)".contains(password), "The password must not be included in any strings.") } }