Как я могу утверждать вывод наблюдаемой, которая использует последние из текстовых входов

У меня есть свойство на моей модели:

    let isValid: Driver<Bool>
    let credentials: Driver<(String, String)>
......
        credentials = .combineLatest(bindings.username, bindings.password, resultSelector: { (username, password) -> (String, String) in (username, password) })

        isValid = credentials.map { username, password in username.count > 0 && password.count > 7 }

Я хотел бы заявить, что установлено правильное состояние isValid когда действительные входы установлены.

Мой тест проходит ниже, однако это не похоже на правильный способ проверить этот сценарий.

В идеале я хотел бы начать с моих строк, как "" а затем передать значения, как если бы они были напечатаны, чтобы я мог утверждать, что состояние по умолчанию установлено, а затем изменяется.

Я также нахожу эти строки:

   .do(onNext: { state in
            if state {
                exp.fulfill()
            }
        })

немного "хакер".

   func test_is_valid_state_changes_when_inputs_correct_length() {

        let username: Driver<String> = .of("some_user_name")
        let password: Driver<String> = .of("some_user_password")

        let bindings = LoginViewModel.Bindings(username: username, password: password, loginTap: .empty(), doneTap: .empty())
        let sut = LoginViewModel(dependency: "", bindings: bindings)

        let scheduler = TestScheduler(initialClock: 0)
        let observer = scheduler.createObserver(Bool.self)

        let exp = expectation(description: "isValid Event")

        sut.isValid
            .asObservable()
            .do(onNext: { state in
                if state {
                    exp.fulfill()
                }
            })
            .subscribe(observer)
            .disposed(by: disposeBag)

        scheduler.start()

        waitForExpectations(timeout: 0.5) { error in
            XCTAssertNil(error)
            XCTAssertEqual(observer.events.count, 1)
            XCTAssertTrue(observer.events[0].value.element!) // swiftlint:disable:this force_unwrapping
        }
    }

1 ответ

Решение

Вам нужно использовать TestObservables, чтобы выполнить необходимое тестирование, и вам не нужен объект ожидания, потому что этот тест будет выполнен без проблем с многопоточностью.

    func test_is_valid_state_changes_when_inputs_correct_length() {
        let scheduler = TestScheduler(initialClock: 0)

        let username = scheduler.createHotObservable([.next(0, ""), .next(10, "h")])
        let password = scheduler.createHotObservable([.next(0, ""), .next(30, "p"), .next(40, "passwor"), .next(50, "password")])

        let bindings = LoginViewModel.Bindings(
            username: username.asDriver(onErrorRecover: { _ in XCTFail(); return .empty() }),
            password: password.asDriver(onErrorRecover: { _ in XCTFail(); return .empty() }),
            loginTap: .empty(),
            doneTap: .empty()
        )
        let disposeBag = DisposeBag()
        let sut = LoginViewModel(dependency: "", bindings: bindings)

        let observer = scheduler.createObserver(Bool.self)

        sut.isValid
            .drive(observer)
            .disposed(by: disposeBag)

        scheduler.start()

        XCTAssertEqual(observer.events, [
            .next(0, false),
            .next(10, false),
            .next(30, false),
            .next(40, false),
            .next(50, true)
        ])

    }
Другие вопросы по тегам