As versões 2 das APIs de teste do Compose (createComposeRule,
createAndroidComposeRule, runComposeUiTest,
runAndroidComposeUiTest etc.) já estão disponíveis para melhorar o controle sobre a
execução de corrotinas. Essa atualização não duplica toda a superfície da API. Apenas as APIs que estabelecem o ambiente de teste foram atualizadas.
As APIs v1 foram descontinuadas, e é altamente recomendável migrar para as novas APIs. A migração verifica se os testes estão alinhados com o comportamento padrão de corrotinas e evita problemas de compatibilidade futuros. Para conferir uma lista das APIs v1 descontinuadas, consulte Mapeamentos de API.
Essas mudanças estão incluídas em
androidx.compose.ui:ui-test-junit4:1.11.0-alpha03+ e
androidx.compose.ui:ui-test:1.11.0-alpha03+.
Enquanto as APIs v1 dependiam do UnconfinedTestDispatcher, as APIs v2 usam o StandardTestDispatcher por padrão para a composição em execução. Essa mudança
alinha o comportamento de teste do Compose com as APIs runTest padrão e oferece
controle explícito sobre a ordem de execução da corrotina.
Mapeamentos de API
Ao fazer upgrade para as APIs v2, você pode usar Localizar e substituir para atualizar as importações de pacotes e adotar as novas mudanças de dispatcher.
Como alternativa, peça ao Gemini para fazer uma migração para a v2 das APIs de teste do Compose com o seguinte comando:
Comando de IA
Migrar das APIs de teste v1 para as APIs de teste v2
Essa solicitação vai usar este guia para migrar para as APIs de teste da v2.
Migrate to Compose testing v2 APIs using the official
migration guide.Use a tabela a seguir para mapear as APIs v1 descontinuadas e as substituições na v2:
Suspenso (v1) |
Substituição (v2) |
|---|---|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Compatibilidade com versões anteriores e exceções
As APIs v1 atuais estão descontinuadas, mas continuam usando
UnconfinedTestDispatcher para manter o comportamento atual e evitar mudanças
importantes.
A única exceção em que o comportamento padrão mudou é a seguinte:
O agente de teste padrão usado para executar a composição na
classe AndroidComposeUiTestEnvironment mudou de
UnconfinedTestDispatcher para StandardTestDispatcher. Isso afeta casos em que
você cria uma instância usando o construtor ou uma subclasse
AndroidComposeUiTestEnvironment e chama esse construtor.
Mudança principal: impacto na execução de corrotinas
A principal diferença entre as versões v1 e v2 das APIs é como as corrotinas são despachadas:
- APIs v1 (
UnconfinedTestDispatcher): quando uma corrotina era iniciada, ela era executada imediatamente na linha de execução atual, muitas vezes terminando antes da próxima linha de código de teste ser executada. Ao contrário do comportamento de produção, essa execução imediata pode mascarar inadvertidamente problemas de tempo real ou condições de corrida que ocorreriam em um aplicativo ativo. - APIs v2 (
StandardTestDispatcher): quando uma corrotina é iniciada, ela é enfileirada e não é executada até que o teste avance explicitamente o relógio virtual. As APIs de teste padrão do Compose (comowaitForIdle()) já processam essa sincronização. Portanto, a maioria dos testes que dependem dessas APIs padrão continuará funcionando sem mudanças.
Falhas comuns e como corrigir
Se os testes falharem após o upgrade para a v2, provavelmente eles vão apresentar o seguinte padrão:
- Falha: você inicia uma tarefa (por exemplo, uma ViewModel carrega dados), mas sua asserção falha imediatamente porque os dados ainda estão em um estado "Carregando".
- Causa: com as APIs v2, as corrotinas são enfileiradas em vez de executadas imediatamente. A tarefa foi enfileirada, mas nunca executada antes da verificação do resultado.
- Correção: avance o tempo explicitamente. É necessário informar explicitamente ao dispatcher v2 quando executar o trabalho.
Abordagem anterior
Na v1, a tarefa era iniciada e concluída imediatamente. Na v2, o código a seguir
falha porque loadData() ainda não foi executado.
// In v1, this launched and finished immediately.
viewModel.loadData()
// In v2, this fails because loadData() hasn't actually run yet!
assertEquals(Success, viewModel.state.value)
Abordagem recomendada
Use waitForIdle ou runOnIdle para executar tarefas enfileiradas antes de
fazer asserções.
Opção 1: usar waitForIdle avança o relógio até que a interface fique inativa,
verificando se a corrotina foi executada.
viewModel.loadData()
// Explicitly run all queued tasks
composeTestRule.waitForIdle()
assertEquals(Success, viewModel.state.value)
Opção 2: usar runOnIdle executa o bloco de código na linha de execução da UI depois que
a UI fica inativa.
viewModel.loadData()
// Run the assertion after the UI is idle
composeTestRule.runOnIdle {
assertEquals(Success, viewModel.state.value)
}
Sincronização manual
Em cenários que envolvem sincronização manual, como quando o avanço automático é
desativado, o lançamento de uma corrotina não resulta em execução imediata porque
o relógio de teste está pausado. Para executar corrotinas na fila sem
avançar o relógio virtual, use a API runCurrent(). Isso executa tarefas
programadas para o tempo virtual atual.
composeTestRule.mainClock.scheduler.runCurrent()
Ao contrário de waitForIdle(), que avança o relógio de teste até que a interface
se estabilize, runCurrent() executa tarefas pendentes enquanto mantém o tempo
virtual atual. Esse comportamento permite a verificação de estados intermediários que
seriam ignorados se o relógio fosse avançado para um estado ocioso.
O programador de testes usado no ambiente de teste é exposto. Esse
agendador pode ser usado em conjunto com a API runTest do Kotlin para
sincronizar o relógio de teste.
Migrar para runComposeUiTest
Se você estiver usando APIs de teste do Compose com a API runTest do Kotlin,
é altamente recomendável mudar para runComposeUiTest.
Abordagem anterior
Usar createComposeRule em conjunto com runTest cria dois relógios separados: um para o Compose e outro para o escopo da corrotina de teste. Essa
configuração pode forçar você a sincronizar manualmente o programador de testes.
@get:Rule val composeTestRule = createComposeRule() @Test fun testWithCoroutines() { composeTestRule.setContent { var status by remember { mutableStateOf("Loading...") } LaunchedEffect(Unit) { delay(1000) status = "Done!" } Text(text = status) } // NOT RECOMMENDED // Fails: runTest creates a new, separate scheduler. // Advancing time here does NOT advance the compose clock. // To fix this without migrating, you would need to share the scheduler // by passing 'composeTestRule.mainClock.scheduler' to runTest. runTest { composeTestRule.onNodeWithText("Loading...").assertIsDisplayed() advanceTimeBy(1000) composeTestRule.onNodeWithText("Done!").assertIsDisplayed() } }
Abordagem recomendada
A API runComposeUiTest executa automaticamente o bloco de teste no próprio escopo runTest. O relógio de teste é sincronizado com o ambiente do Compose, então
não é mais necessário gerenciar o programador manualmente.
@Test fun testWithCoroutines() = runComposeUiTest { setContent { var status by remember { mutableStateOf("Loading...") } LaunchedEffect(Unit) { delay(1000) status = "Done!" } Text(text = status) } onNodeWithText("Loading...").assertIsDisplayed() mainClock.advanceTimeBy(1000 + 16 /* Frame buffer */) onNodeWithText("Done!").assertIsDisplayed() } }