Паттерн MVP в разработке мобильных приложений — это довольно простой способ разгрузить ViewController и вынести часть логики в презентер. Презентер начинает обрастать логикой, которая легко поддается тестированию.


Пусть есть экран MelodyListViewController показывающий список мелодий. У него есть презентер MelodyListPresenter, который говорит ViewController что показывать. Данные презентер будет брать из сервиса MelodyService. MelodyService это обертка над базой данных и api клиентом, загружающая мелодии. Если сеть доступна, сервис берет данные с api, иначе с базы данных. Типы ошибок загрузки представлены в enum ServiceRequestError.


protocol MelodyListViewController: class {
    func showMelodies(melodies: [Melody])
    func showLoadError(error: ServiceRequestError)
}

protocol MelodyListPresenter {
    var view: MelodyListViewController? { get }
    var melodyService: MelodyService { get }

    func fetchMelodies() -> Promise<Void>
}

extension MelodyListPresenter {
    func fetchMelodies() -> Promise<Void> {
        return melodyService.getMelodies().done { melodies in
            self.view?.showMelodies(melodies: melodies)
        }.catch { error in
            self.view?.showLoadError(error: error)
        }
    }
}

protocol MelodyService {
    func getMelodies() -> Promise<[Melody]>
}

public enum ServiceRequestError: Error {
    case unknownError
    case noNetwork
    case noData
}

Построив такую структуру экрана, можно заняться тестированием. А именно тестированием получения данных презентером. Презентер имеет в зависимости MelodyService, поэтому необходимо мокировать этот протокол. Условимся, что Melody имеет статический метод mocks, который возвращает список произвольных мелодий.


class MelodyServiceMock: MelodyService, ServiceRequestMock {

    var emulatedResult: ServiceRequestResult = .error(.unknownError)

    func getMelodies() -> Promise<[Melody]> {
        let melodies = Melody.mocks()
        return mock(result: emulatedResult, model: melodies)
    }
}

enum ServiceRequestResult {
    case success
    case error(ServiceRequestError)
}

Также мокируем ViewController.


class MelodyListViewControllerMock: MelodyListViewController {

    var shownMelodies: [Melody]?
    var shownError: ServiceRequestError?

    func showMelodies(melodies: [Melody]) {
        shownMelodies = melodies
    }

    func showLoadError(error: ServiceRequestError) {
        shownError = error
    }
}

ServiceRequestMock это протокол, имеющий единственный метод func mock<T>(result: ServiceRequestResult, model: T) -> Promise<T>, который возвращает Promise. В этом Promise, зашиты либо мелодии, либо ошибка загрузки — то что передается в качестве симулируемого результата.


protocol ServiceRequestMock {
    func mock<T>(result: ServiceRequestResult, model: T) -> Promise<T>
}

extension ServiceRequestMock {
    func mock<T>(result: ServiceRequestResult, model: T) -> Promise<T> {
        return Promise { seal in
            switch result {
            case .success:
                return seal.fulfill(model)
            case .error(let requestError):
                return seal.reject(requestError)
            }
        }
    }
}

Таким образом мы предоставили все необходимое для тестирования презентера.


import XCTest
import PromiseKit

class MelodyListPresenterTests: XCTestCase {

    let view = MelodyListViewControllerMock()
    let melodyService = MelodyServiceMock()
    var presenter: MelodyListPresenterImp!

    override func setUp() {
        super.setUp()
        presenter = MelodyListPresenterImp(
            melodyService: melodyService, 
            view: view)
        view.presenter = presenter
    }

    func test_getMelodies_success() {
        // given
        let melodiesMock = Melody.mocks()
        melodyService.emulatedResult = .success

        // when
        let fetchMelodies = presenter.fetchMelodies()

        // then
        fetchMelodies.done { melodies in
            XCTAssertNotNil(self.view.shownMelodies)
            XCTAssert(self.view.shownMelodies == melodiesMock)
        }.catch { _ in
            XCTFail("Failed melodies upload")
        }
    }

    func test_getMelodies_fail() {
        // given
        melodyService.emulatedResult = .error(.noNetwork)

        // when
        let fetchMelodies = presenter.fetchMelodies()

        // then
        fetchMelodies.done { melodies in
            XCTFail("Mistakenly uploaded melodies")
        }.catch { _ in
            XCTAssertNotNil(self.view.shownError)
            XCTAssert(self.view.shownError is ServiceRequestError)
            XCTAssert(self.view.shownError as! ServiceRequestError == .noNetwork)
        }
    }
}

В итоге, у нас получился удобный инструмент для написания тестов.

Комментарии (0)