UnitTest сопрограммы Kotlin использовать случай MVP - PullRequest
7 голосов
/ 04 июля 2019

Я пытаюсь смоделировать ответ от моих прецедентов, этот прецедент работает с сопрограммами.

fun getData() {
    view?.showLoading()
    getProductsUseCase.execute(this::onSuccessApi, this::onErrorApi)
}

Мой прецедент используется в презентаторе.

GetProductsUseCase имеет следующий код:

class GetProductsUseCase (private var productsRepository: ProductsRepository) : UseCase<MutableMap<String, Product>>() {

    override suspend fun executeUseCase(): MutableMap<String, Product> {
        val products =productsRepository.getProductsFromApi()
        return products
    }
}

My BaseUseCase

abstract class UseCase<T> {

    abstract suspend fun executeUseCase(): Any

    fun execute(
        onSuccess: (T) -> Unit,
        genericError: () -> Unit) {
        GlobalScope.launch {
            val result = async {
                try {
                    executeUseCase()
                } catch (e: Exception) {
                    GenericError()
                }
            }
            GlobalScope.launch(Dispatchers.Main) {
                when {
                    result.await() is GenericError -> genericError()
                    else -> onSuccess(result.await() as T)
                }
            }
        }
    }

}

Этот сценарий useCase вызывает мой репозиторий:

override suspend fun getProductsFromApi(): MutableMap<String, Product> {
    val productsResponse = safeApiCall(
        call = {apiService.getProductsList()},
        error = "Error fetching products"
    )
    productsResponse?.let {
                    return productsMapper.fromResponseToDomain(it)!!
    }
    return mutableMapOf()
}

Y попытка смоделировать мой ответ, но проверка всегда заканчивается неудачей.

@RunWith(MockitoJUnitRunner::class)
class HomePresenterTest {

    lateinit var presenter: HomePresenter

    @Mock
    lateinit var view: HomeView

    @Mock
    lateinit var getProductsUseCase: GetProductsUseCase

    @Mock
    lateinit var updateProductsUseCase: UpdateProductsUseCase

    private lateinit var products: MutableMap<String, Product>

    private val testDispatcher = TestCoroutineDispatcher()
    private val testScope = TestCoroutineScope(testDispatcher)

    @Mock
    lateinit var productsRepository:ProductsRepositoryImpl

    @Before
    fun setUp() {
        Dispatchers.setMain(testDispatcher)
        products = ProductsMotherObject.createEmptyModel()
        presenter = HomePresenter(view, getProductsUseCase, updateProductsUseCase, products)
    }

    @After
    fun after() {
        Dispatchers.resetMain()
        testScope.cleanupTestCoroutines()
    }

    //...

    @Test
    fun a() = testScope.runBlockingTest {
        setTasksNotAvailable(productsRepository)
        presenter.getDataFromApi()

        verify(view).setUpRecyclerView(products.values.toMutableList())
    }

    private suspend fun setTasksNotAvailable(dataSource: ProductsRepository) {
        `when`(dataSource.getProductsFromApi()).thenReturn((mutableMapOf()))
    }
}

Я не знаю, что происходит.В журнале написано:

"Wanted but not invoked:
view.setUpRecyclerView([]);
-> at com.myProject.HomePresenterTest$a$1.invokeSuspend(HomePresenterTest.kt:165)

However, there was exactly 1 interaction with this mock:
view.showLoading();"

1 Ответ

4 голосов
/ 06 июля 2019

Проблема в том, как вы создаете GetProductsUseCase.

. Вы не создаете его с помощью mocked версии своего ProductsRepository, но при этом высмеиваете вызовы ProductsRepository.

Попробуйте создать GetProductsUseCase вручную и без использования @Mock

// no @Mock
lateinit var getProductsUseCase: GetProductsUseCase


@Before
fun setUp() {
    // ...
    // after your mocks are initialized...
    getProductsUseCase = GetProductsUseCase(productsRepository) //<- this uses mocked ProductsRepository
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...