Guida al test dell'hilt

Uno dei vantaggi dell'utilizzo di framework di iniezione delle dipendenze come Hilt è che semplifica il test del codice.

Test delle unità

Hilt non è necessario per i test unitari, poiché quando testi una classe che utilizza l'inserimento del costruttore, non devi utilizzare Hilt per creare un'istanza di questa classe. In alternativa, puoi chiamare direttamente un costruttore di classe passando dipendenze false o simulate, proprio come faresti se il costruttore non fosse annotato:

@ActivityScoped
class AnalyticsAdapter @Inject constructor(
  private val service: AnalyticsService
) { ... }

class AnalyticsAdapterTest {

  @Test
  fun `Happy path`() {
    // You don't need Hilt to create an instance of AnalyticsAdapter.
    // You can pass a fake or mock AnalyticsService.
    val adapter = AnalyticsAdapter(fakeAnalyticsService)
    assertEquals(...)
  }
}

Lo stesso vale per le classi ViewModel ottenute chiamando hiltViewModel() nei tuoi composable. Nei test delle unità, crea ViewModel direttamente con i fake. Per informazioni su come lo stato passa da un ViewModel ai composable, vedi Stato e Jetpack Compose e Dove sollevare lo stato.

Test end-to-end

Per i test di integrazione, Hilt inserisce le dipendenze come farebbe nel codice di produzione. Il test con Hilt non richiede manutenzione perché Hilt genera automaticamente un nuovo insieme di componenti per ogni test.

Aggiunta di dipendenze di test

Per utilizzare Hilt nei test, includi la dipendenza hilt-android-testing nel tuo progetto:

dependencies {
    // For Robolectric tests.
    testImplementation("com.google.dagger:hilt-android-testing:2.57.1")
    kspTest("com.google.dagger:hilt-android-compiler:2.57.1")

    // For instrumented tests.
    androidTestImplementation("com.google.dagger:hilt-android-testing:2.57.1")
    kspAndroidTest("com.google.dagger:hilt-android-compiler:2.57.1")

    // Compose UI test rule.
    androidTestImplementation("androidx.compose.ui:ui-test-junit4")
    debugImplementation("androidx.compose.ui:ui-test-manifest")

}

Configurazione del test dell'interfaccia utente

Devi annotare qualsiasi test UI che utilizza Hilt con @HiltAndroidTest. Questa annotazione è responsabile della generazione dei componenti Hilt per ogni test.

Inoltre, devi aggiungere HiltAndroidRule alla classe di test. Gestisce lo stato dei componenti e viene utilizzato per eseguire l'inserimento nel test:

@HiltAndroidTest
class SettingsScreenTest {

    @get:Rule(order = 0)
    val hiltRule = HiltAndroidRule(this)

    @get:Rule(order = 1)
    val composeRule = createAndroidComposeRule<HiltTestActivity>()

    // Compose UI tests here.
}

Successivamente, il test deve conoscere la classe Application che Hilt genera automaticamente per te.

Per consentire a Hilt di inserire le dipendenze, devi creare un'attività vuota denominata HiltTestActivity nel set di risorse androidTest e annotarla con @AndroidEntryPoint. createAndroidComposeRule utilizza quindi questa attività come host per i tuoi contenuti componibili.

Test dell'applicazione

Devi eseguire test strumentati che utilizzano Hilt in un oggetto Application che supporta Hilt. La libreria fornisce HiltTestApplication da utilizzare nei test. Se i test richiedono un'applicazione di base diversa, consulta Applicazione personalizzata per i test.

Devi impostare l'applicazione di test in modo che venga eseguita nei test strumentati o nei test Robolectric. Le seguenti istruzioni non sono specifiche per Hilt, ma sono linee guida generali su come specificare un'applicazione personalizzata da eseguire nei test.

Impostare l'applicazione di test nei test strumentati

Per utilizzare l'applicazione di test Hilt nei test strumentati, devi configurare un nuovo test runner. In questo modo Hilt funziona per tutti i test strumentati nel tuo progetto. Segui questi passaggi:

  1. Crea una classe personalizzata che estenda AndroidJUnitRunner in androidTest.
  2. Esegui l'override della funzione newApplication e trasmetti il nome dell'applicazione di test Hilt generata.
// A custom runner to set up the instrumented application class for tests.
class CustomTestRunner : AndroidJUnitRunner() {

    override fun newApplication(cl: ClassLoader?, name: String?, context: Context?): Application {
        return super.newApplication(cl, HiltTestApplication::class.java.name, context)
    }
}

Successivamente, configura questo test runner nel file Gradle come descritto nella guida ai test delle unità strumentati. Assicurati di utilizzare il classpath completo:

android {
    defaultConfig {
        // Replace com.example.android.dagger with your class path.
        testInstrumentationRunner = "com.example.android.dagger.CustomTestRunner"
    }
}
Impostare l'applicazione di test nei test Robolectric

Se utilizzi Robolectric per testare il livello UI, puoi specificare l'applicazione da utilizzare nel file robolectric.properties:

application = dagger.hilt.android.testing.HiltTestApplication

In alternativa, puoi configurare l'applicazione su ogni test singolarmente utilizzando l'annotazione @Config di Robolectric:

@HiltAndroidTest
@Config(application = HiltTestApplication::class)
class SettingsScreenTest {

  @get:Rule
  var hiltRule = HiltAndroidRule(this)

  // Robolectric tests here.
}

Funzionalità di test

Una volta che Hilt è pronto per l'uso nei test, puoi utilizzare diverse funzionalità per personalizzare la procedura di test.

Inserire tipi nei test

Per inserire tipi in un test, utilizza @Inject per la field injection. Per indicare a Hilt di compilare i campi @Inject, chiama hiltRule.inject().

Vedi il seguente esempio di test strumentato:

@HiltAndroidTest
class SettingsScreenTest {

    @get:Rule(order = 0)
    val hiltRule = HiltAndroidRule(this)

    @get:Rule(order = 1)
    val composeRule = createAndroidComposeRule<HiltTestActivity>()

    @Inject
    lateinit var analyticsAdapter: AnalyticsAdapter

    @Before
    fun init() {
        hiltRule.inject()
    }

    @Test
    fun settingsScreen_showsTitle() {
        composeRule.setContent {
            SettingsScreen()
        }
        composeRule.onNodeWithText("Settings").assertIsDisplayed()
        // analyticsRepository is available here.
    }
}

Sostituire un'associazione

Se devi inserire un'istanza fittizia o simulata di una dipendenza, devi indicare a Hilt di non utilizzare l'associazione utilizzata nel codice di produzione e di utilizzarne un'altra. Per sostituire un binding, devi sostituire il modulo che contiene il binding con un modulo di test che contiene i binding che vuoi utilizzare nel test.

Ad esempio, supponiamo che il codice di produzione dichiari un binding per AnalyticsService nel seguente modo:

@Module
@InstallIn(SingletonComponent::class)
abstract class AnalyticsModule {

  @Singleton
  @Binds
  abstract fun bindAnalyticsService(
    analyticsServiceImpl: AnalyticsServiceImpl
  ): AnalyticsService
}

Per sostituire l'associazione AnalyticsService nei test, crea un nuovo modulo Hilt nella cartella test o androidTest con la dipendenza fittizia e annotala con @TestInstallIn. Tutti i test in quella cartella vengono inseriti con la dipendenza fittizia.

@Module
@TestInstallIn(
    components = [SingletonComponent::class],
    replaces = [AnalyticsModule::class]
)
abstract class FakeAnalyticsModule {

  @Singleton
  @Binds
  abstract fun bindAnalyticsService(
    fakeAnalyticsService: FakeAnalyticsService
  ): AnalyticsService
}

Poiché i composable in genere utilizzano queste dipendenze indirettamente tramite una ViewModel ottenuta con hiltViewModel(), è sufficiente sostituire l'associazione in Hilt. Il componente componibile in fase di test rileva automaticamente il fake.

Sostituire un binding in un singolo test

Per sostituire un binding in un singolo test anziché in tutti i test, disinstalla un modulo Hilt da un test utilizzando l'annotazione @UninstallModules e crea un nuovo modulo di test all'interno del test.

Seguendo l'esempio AnalyticsService della versione precedente, inizia dicendo a Hilt di ignorare il modulo di produzione utilizzando l'annotazione @UninstallModules nella classe di test:

@UninstallModules(AnalyticsModule::class)
@HiltAndroidTest
class SettingsScreenTest { ... }

Successivamente, devi sostituire l'associazione. Crea un nuovo modulo all'interno della classe di test che definisce l'associazione del test:

@UninstallModules(AnalyticsModule::class)
@HiltAndroidTest
class SettingsScreenTest {

  @Module
  @InstallIn(SingletonComponent::class)
  abstract class TestModule {

    @Singleton
    @Binds
    abstract fun bindAnalyticsService(
      fakeAnalyticsService: FakeAnalyticsService
    ): AnalyticsService
  }

  // ...
}

Questa operazione sostituisce solo l'associazione per una singola classe di test. Se vuoi sostituire l'associazione per tutte le classi di test, utilizza l'annotazione @TestInstallIn della sezione precedente. In alternativa, puoi inserire il binding di test nel modulo test per i test Robolectric o nel modulo androidTest per i test strumentati. Il consiglio è di utilizzare @TestInstallIn quando possibile.

Associazione di nuovi valori

Utilizza l'annotazione @BindValue per associare facilmente i campi nel test al grafico delle dipendenze di Hilt. Annota un campo con @BindValue e verrà associato al tipo di campo dichiarato con eventuali qualificatori presenti per quel campo.

Nell'esempio AnalyticsService, puoi sostituire AnalyticsService con un falso utilizzando @BindValue:

@UninstallModules(AnalyticsModule::class)
@HiltAndroidTest
class SettingsScreenTest {

  @BindValue @JvmField
  val analyticsService: AnalyticsService = FakeAnalyticsService()

  ...
}

In questo modo, è più semplice sostituire un binding e farvi riferimento nel test, in quanto puoi eseguire entrambe le operazioni contemporaneamente.

@BindValue funziona con i qualificatori e altre annotazioni di test. Ad esempio, se utilizzi librerie di test come Mockito, puoi utilizzarle in un test Robolectric nel seguente modo:

...
class SettingsScreenTest {
  ...

  @BindValue @ExampleQualifier @Mock
  lateinit var qualifiedVariable: ExampleCustomType

  // Robolectric tests here
}

Se devi aggiungere un multibinding, puoi utilizzare le annotazioni @BindValueIntoSet e @BindValueIntoMap al posto di @BindValue. @BindValueIntoMap richiede anche di annotare il campo con un'annotazione della chiave della mappa.

Casi speciali

Hilt fornisce anche funzionalità per supportare casi d'uso non standard.

Applicazione personalizzata per i test

Se non puoi utilizzare HiltTestApplication perché la tua applicazione di test deve estendere un'altra applicazione, annota una nuova classe o interfaccia con @CustomTestApplication, passando il valore della classe base che vuoi che l'applicazione Hilt generata estenda.

@CustomTestApplication genererà una classe Application pronta per il test con Hilt che estende l'applicazione che hai passato come parametro.

@CustomTestApplication(BaseApplication::class)
interface HiltTestApplication

Nell'esempio, Hilt genera un Application denominato HiltTestApplication_Application che estende la classe BaseApplication. In generale, il nome dell'applicazione generata è il nome della classe annotata a cui è stato aggiunto _Application. Devi impostare l'applicazione di test Hilt generata per l'esecuzione nei test strumentati o nei test Robolectric, come descritto in Applicazione di test.

Più oggetti TestRule nel test strumentato

I test dell'interfaccia utente di Compose combinano già HiltAndroidRule con una regola di test di Compose come createAndroidComposeRule. Se hai altri oggetti TestRule, assicurati che HiltAndroidRule venga eseguito per primo. Dichiara l'ordine di esecuzione con l'attributo order su @Rule:

@HiltAndroidTest
class SettingsScreenTest {

  @get:Rule(order = 0)
  var hiltRule = HiltAndroidRule(this)

  @get:Rule(order = 1)
  val composeRule = createAndroidComposeRule<HiltTestActivity>()

  @get:Rule(order = 2)
  val otherRule = SomeOtherRule()

  // UI tests here.
}

In alternativa, puoi racchiudere le regole con RuleChain, inserendo HiltAndroidRule come regola esterna.

@HiltAndroidTest
class SettingsScreenTest {

  @get:Rule
  var rule = RuleChain.outerRule(HiltAndroidRule(this)).
        around(SettingsScreenTestRule(...))

  // UI tests here.
}

Utilizzare un punto di ingresso prima che il componente singleton sia disponibile

L'annotazione @EarlyEntryPoint fornisce una via di fuga quando è necessario creare un punto di ingresso Hilt prima che il componente singleton sia disponibile in un test Hilt.

Per saperne di più su @EarlyEntryPoint, consulta la documentazione di Hilt.

Risorse aggiuntive

Per saperne di più sui test, consulta le seguenti risorse aggiuntive:

Documentazione

Visualizza contenuti