Ошибка XCTAssertEqual при использовании createColdObservable для RxTest

У меня странная проблема в моем тестовом коде. Я хочу протестировать BehaviourRelay на мой взгляд, модель меняет статус .fetching к .saving. У меня есть следующий тестовый код:

class BankViewModelTests: XCTestCase {
    
    private var scheduler: TestScheduler!
    private var bag: DisposeBag!
    private var user: UserModel!
    

    override func setUpWithError() throws {
        try! super.setUpWithError()
        
        let url = Bundle(for: type(of: self)).url(forResource: "User", withExtension: "json")!
        let jsonData = try! Data(contentsOf: url)
        let jsonDict = try! JSONSerialization.jsonObject(with: jsonData, options: .allowFragments) as! JSONDictionary
        user = UserModel(jsonDict)!
        
        scheduler = TestScheduler(initialClock: 0)
        bag = DisposeBag()
    }

    override func tearDownWithError() throws {
        user = nil
        scheduler = nil
        bag = nil
        try! super.tearDownWithError()
    }
    
}

extension BankViewModelTests {
    func testSavingStatus() {
        // Arrange
        
        let sut = BankViewModel(user: user)
        
        let status = scheduler.createObserver(BankViewModel.Status.self)
        sut.status.bind(to: status).disposed(by: bag)
        
        // Action
        
        scheduler.createColdObservable([.next(10, ())]).bind(to: sut.tappedSubmit).disposed(by: bag)
        
        scheduler.start()
        
        // Assert
        
        XCTAssertEqual(status.events, [
            .next(0, .fetching),
            .next(10, .saving)
        ])
    }
    
}

Мой Status enum выглядит так:

enum Status: Equatable {
    case fetching, fetchSuccess, saving, saveSuccess, failure(Error)

    public static func == (lhs: Status, rhs: Status) -> Bool {
        switch (lhs, rhs) {
        case (.fetching, .fetching),
             (.fetchSuccess, .fetchSuccess),
             (.saving, .saveSuccess),
             (.failure, .failure):
            return true
        default: return false
        }
    }
}

Когда я запускаю тест, я получаю следующее сообщение: XCTAssertEqual failed: ("[next(fetching) @ 0, next(saving) @ 10]") is not equal to ("[next(fetching) @ 0, next(saving) @ 10]")

Ясно, что эти события эквивалентны, так почему же это не удается?

1 ответ

Решение

У меня уже была похожая проблема с вами.

В моем случае переопределение Equatable из BankViewModel.Status make test не удалось.

Пожалуйста, проверьте. Есть ли в иерархии BankViewModel.Status переопределяющий протокол реализации Equatable и правильно ли отредактируйте.


Обновлено из комментариев

       enum Status: Equatable {
    case fetching, fetchSuccess, saving, saveSuccess, failure(Error)

    public static func == (lhs: Status, rhs: Status) -> Bool {
        switch (lhs, rhs) {
        case (.fetching, .fetching),
             (.fetchSuccess, .fetchSuccess),
             (.saving, .saveSuccess), // <- (.saving == .saving) always false so test make fail
             (.failure, .failure):
            return true
        default: return false
        }
    }
}

XCTAssertEqual(Status.saving, Status.saving) // Now, It should be failed because overriding Equatable implement

Сопоставьте и сравните переключатели слева и справа от статуса. Сохранение правильно

       enum Status: Equatable {
    case fetching, fetchSuccess, saving, saveSuccess, failure(Error)

    public static func == (lhs: Status, rhs: Status) -> Bool {
        switch (lhs, rhs) {
        case (.fetching, .fetching),
             (.fetchSuccess, .fetchSuccess),
             (.saving, .saving),
             (.saveSuccess, .saveSuccess),
             (.failure, .failure):
            return true
        default:
            return false
        }
    }
}
Другие вопросы по тегам