Membuat pengujian unit lokal

Pengujian lokal berjalan langsung di workstation Anda sendiri, bukan di Android perangkat atau emulator. Dengan demikian, pengujian ini menggunakan Java Virtual Machine (JVM) lokal, bukan perangkat Android untuk menjalankan pengujian. Dengan pengujian lokal, Anda dapat mengevaluasi logika aplikasi Anda dengan lebih cepat. Namun, tidak dapat berinteraksi dengan Framework Android membuat batasan pada jenis pengujian yang dapat Anda jalankan.

Pengujian unit memverifikasi perilaku bagian kecil kode, unit yang sedang diuji. Hal ini dilakukan dengan mengeksekusi kode tersebut dan memeriksa hasilnya.

Pengujian unit biasanya sederhana, tetapi penyiapannya dapat bermasalah jika unit yang sedang diuji tidak dirancang dengan mempertimbangkan kemampuan pengujian:

  • Kode yang ingin Anda verifikasi harus dapat diakses dari pengujian. Sebagai misalnya, Anda tidak dapat menguji metode pribadi secara langsung. Sebagai gantinya, Anda menguji class menggunakan API publiknya.
  • Untuk menjalankan pengujian unit secara isolasi, dependensi unit dalam pengujian harus diganti dengan komponen yang Anda kontrol, seperti perangkat palsu atau pengujian ganda lainnya. Hal ini terutama menjadi masalah jika kode Anda bergantung pada framework Android.

Untuk mempelajari strategi pengujian unit umum di Android, baca Yang perlu diuji.

Lokasi pengujian lokal

Secara default, file sumber untuk pengujian unit lokal ditempatkan di module-name/src/test/. Direktori ini sudah ada saat Anda membuat project baru menggunakan Android Studio.

Menambahkan dependensi pengujian

Anda juga perlu mengonfigurasi dependensi pengujian untuk project Anda agar dapat menggunakan API standar yang disediakan oleh framework pengujian JUnit.

Untuk melakukannya, buka file build.gradle modul aplikasi Anda dan tentukan library berikut sebagai dependensi. Gunakan fungsi testImplementation untuk menunjukkan bahwa fungsi tersebut berlaku untuk set sumber pengujian lokal, bukan aplikasi:

dependencies {
  // Required -- JUnit 4 framework
  testImplementation "junit:junit:$jUnitVersion"
  // Optional -- Robolectric environment
  testImplementation "androidx.test:core:$androidXTestVersion"
  // Optional -- Mockito framework
  testImplementation "org.mockito:mockito-core:$mockitoVersion"
  // Optional -- mockito-kotlin
  testImplementation "org.mockito.kotlin:mockito-kotlin:$mockitoKotlinVersion"
  // Optional -- Mockk framework
  testImplementation "io.mockk:mockk:$mockkVersion"
}

Membuat class pengujian unit lokal

Anda menulis class pengujian unit lokal sebagai class pengujian JUnit 4.

Untuk melakukannya, buat kelas yang berisi satu atau beberapa metode pengujian, biasanya dalam module-name/src/test/. Metode pengujian dimulai dengan anotasi @Test dan berisi kode untuk menjalankan dan memverifikasi satu aspek komponen yang ingin Anda uji.

Contoh berikut menunjukkan cara menerapkan class pengujian unit lokal. Tujuan metode pengujian emailValidator_correctEmailSimple_returnsTrue()mencoba memverifikasi isValidEmail(),yang merupakan metode dalam aplikasi. Fungsi pengujian akan menampilkan benar (true) jika isValidEmail() juga menampilkan benar (true).

Kotlin

import org.junit.Assert.assertFalse
import org.junit.Assert.assertTrue
import org.junit.Test

class EmailValidatorTest {
  @Test fun emailValidator_CorrectEmailSimple_ReturnsTrue() {
    assertTrue(EmailValidator.isValidEmail("name@email.com"))
  }

}

Java

import org.junit.Test;

import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertTrue;

class EmailValidatorTest {
  @Test
  public void emailValidator_CorrectEmailSimple_ReturnsTrue() {
    assertTrue(EmailValidator.isValidEmail("name@email.com"));
  }
}

Anda harus membuat pengujian yang dapat dibaca yang mengevaluasi apakah komponen dalam aplikasi Anda menampilkan hasil yang diharapkan. Sebaiknya gunakan library pernyataan seperti sebagai junit.Assert, Hamcrest, atau Kebenaran. Cuplikan di atas adalah contoh cara menggunakan junit.Assert.

Library Android yang dapat di-mock

Saat menjalankan pengujian unit lokal, Plugin Android Gradle akan menyertakan yang berisi semua API framework Android, sesuai dengan yang digunakan dalam project Anda. Library ini menyimpan semua metode dan class publik API tersebut, tetapi kode di dalam metode telah dihapus. Jika ada dari metode diakses, pengujian akan menampilkan pengecualian.

Hal ini memungkinkan pengujian lokal dibuat saat mereferensikan class dalam framework Android seperti Context. Lebih penting lagi, ini memungkinkan Anda menggunakan {i>mockup<i} Google Cloud dengan kelas Android.

Tiruan dependensi Android

Masalah umum adalah menemukan bahwa class menggunakan resource string. Anda dapat mendapatkan resource string dengan memanggil metode getString() di class Context. Namun, pengujian lokal tidak dapat menggunakan Context atau metodenya karena termasuk dalam framework Android. Idealnya, panggilan ke getString() adalah dikeluarkan dari kelas, tetapi hal ini tidak selalu praktis. Solusinya adalah membuat tiruan atau stub Context yang selalu menampilkan nilai yang sama saat metode getString()-nya dipanggil.

Dengan library Android Mockable dan framework tiruan seperti Mockito atau MockK, Anda dapat memprogram perilaku tiruan class Android dalam pengujian unit.

Untuk menambahkan objek tiruan ke pengujian unit lokal Anda menggunakan Mockito, ikuti tradisional:

  1. Sertakan dependensi library Mockito dalam file build.gradle, seperti yang dijelaskan dalam Menyiapkan lingkungan pengujian Anda.
  2. Di awal definisi class pengujian unit, tambahkan anotasi @RunWith(MockitoJUnitRunner.class). Anotasi ini memberi tahu runner pengujian Mockito untuk memvalidasi bahwa penggunaan framework sudah benar dan menyederhanakan inisialisasi objek tiruan Anda.
  3. Untuk membuat objek tiruan untuk dependensi Android, tambahkan anotasi @Mock sebelum deklarasi {i>field<i}.
  4. Untuk menghentikan perilaku dependensi, Anda dapat menentukan kondisi dan mengembalikan nilai saat kondisi terpenuhi dengan menggunakan metode when() dan thenReturn().

Contoh berikut menunjukkan cara membuat pengujian unit yang menggunakan objek Context tiruan di Kotlin yang dibuat dengan Mockito-Kotlin.

import android.content.Context
import org.junit.Assert.assertEquals
import org.junit.Test
import org.junit.runner.RunWith
import org.mockito.Mock
import org.mockito.junit.MockitoJUnitRunner
import org.mockito.kotlin.doReturn
import org.mockito.kotlin.mock

private const val FAKE_STRING = "HELLO WORLD"

@RunWith(MockitoJUnitRunner::class)
class MockedContextTest {

  @Mock
  private lateinit var mockContext: Context

  @Test
  fun readStringFromContext_LocalizedString() {
    // Given a mocked Context injected into the object under test...
    val mockContext = mock<Context> {
        on { getString(R.string.name_label) } doReturn FAKE_STRING
    }

    val myObjectUnderTest = ClassUnderTest(mockContext)

    // ...when the string is returned from the object under test...
    val result: String = myObjectUnderTest.getName()

    // ...then the result should be the expected one.
    assertEquals(result, FAKE_STRING)
  }
}

Untuk mempelajari lebih lanjut cara menggunakan framework Mockito, lihat referensi Mockito API dan class SharedPreferencesHelperTest dalam kode contoh. Coba juga Android Testing Codelab.

Error: "Metode ... tidak ditiru"

Library Android Mockable akan menampilkan pengecualian jika Anda mencoba mengakses salah satu metodenya dengan pesan Error: "Method ... not mocked.

Jika pengecualian yang dimunculkan bermasalah bagi pengujian, Anda dapat mengubah perilaku sehingga metode akan menampilkan null atau nol, bergantung pada jenis nilai yang ditampilkan. Untuk melakukannya, tambahkan konfigurasi berikut ke file file build.gradle level teratas di Groovy:

android {
  ...
  testOptions {
    unitTests.returnDefaultValues = true
  }