Testowanie mapowania

To krótkie wprowadzenie do mapowania testów i wyjaśnienie, jak rozpocząć konfigurowanie testów w projekcie Android Open Source Project (AOSP).

Informacje o mapowaniu testów

Mapowanie testów to podejście oparte na Gerrit, które umożliwia deweloperom tworzenie reguł testów przed i po przesłaniu bezpośrednio w drzewie źródłowym Androida oraz pozostawienie decyzji dotyczących gałęzi i urządzeń do przetestowania infrastrukturze testowej. Definicje mapowania testów to pliki JSON o nazwie TEST_MAPPING, które można umieścić w dowolnym katalogu źródłowym.

Atest może używać plików TEST_MAPPING do przeprowadzania testów przed przesłaniem w powiązanych katalogach. Dzięki mapowaniu testów możesz dodać ten sam zestaw testów do weryfikacji przed przesłaniem, wprowadzając minimalne zmiany w drzewie źródłowym Androida.

Oto kilka przykładów:

Mapowanie testów opiera się na platformie testowej Trade Federation (TF) do wykonywania testów i raportowania wyników.

Definiowanie grup testowych

Mapowanie testów grupuje testy w grupy testowe. Nazwa grupy testowej może być dowolnym ciągiem znaków. Na przykład presubmit może być nazwą grupy testów, które mają być uruchamiane podczas weryfikowania zmian. Z kolei postsubmit może być nazwą testów używanych do weryfikowania kompilacji po scaleniu zmian.

Reguły skryptu kompilacji pakietu

Aby platforma testowa Trade Federation mogła uruchamiać moduły testowe dla danej kompilacji, moduły te muszą mieć ustawioną wartość test_suites dla Soong lub LOCAL_COMPATIBILITY_SUITE ustawioną dla Make na jeden z tych 2 zestawów:

  • general-tests to testy, które nie zależą od funkcji specyficznych dla urządzenia (takich jak sprzęt specyficzny dla dostawcy, którego większość urządzeń nie ma). Większość testów powinna znajdować się w zestawie general-tests, nawet jeśli są one specyficzne dla jednego ABI, liczby bitów lub funkcji sprzętowych, takich jak HWASan (dla każdego ABI jest osobny cel test_suites), a nawet jeśli muszą być uruchamiane na urządzeniu.
  • device-tests to testy, które zależą od funkcji specyficznych dla urządzenia. Zazwyczaj te testy znajdują się w katalogu vendor/. Specyficzne dla urządzenia odnosi się tylko do funkcji, które są unikalne dla danego urządzenia, dlatego dotyczy to zarówno testów JUnit, jak i GTest (które zwykle powinny być oznaczane jako general-tests, nawet jeśli są specyficzne dla ABI).

Przykłady:

Android.bp: test_suites: ["general-tests"],
Android.mk: LOCAL_COMPATIBILITY_SUITE := general-tests

Konfigurowanie testów do uruchamiania w zestawie testów

Aby test mógł być uruchamiany w zestawie testów:

  • nie może mieć dostawcy kompilacji;
  • musi po zakończeniu działania zwalniać miejsce, na przykład usuwając wszystkie tymczasowe pliki wygenerowane podczas testu;
  • musi przywrócić domyślne lub pierwotne ustawienia systemu;
  • nie powinien zakładać, że urządzenie jest w określonym stanie, np. że jest gotowe do rootowania. Większość testów nie wymaga uprawnień roota. Jeśli test wymaga uprawnień roota, powinien to określić za pomocą elementu RootTargetPreparer w pliku AndroidTest.xml, jak w tym przykładzie:

    <target_preparer class="com.android.tradefed.targetprep.RootTargetPreparer"/>
    

Tworzenie plików mapowania testów

W katalogu, który wymaga pokrycia testami, dodaj plik JSON TEST_MAPPINGpodobny do tego przykładu. Te reguły zapewniają, że testy będą uruchamiane podczas weryfikacji przed przesłaniem, gdy w tym katalogu lub w którymkolwiek z jego podkatalogów zostaną zmienione pliki.

Przykład

Oto przykładowy plik TEST_MAPPING (w formacie JSON, ale z obsługą komentarzy):

{
  "presubmit": [
    // JUnit test with options and file patterns.
    {
      "name": "CtsWindowManagerDeviceTestCases",
      "options": [
        {
          "include-annotation": "android.platform.test.annotations.RequiresDevice"
        }
      ],
      "file_patterns": ["(/|^)Window[^/]*\\.java", "(/|^)Activity[^/]*\\.java"]
    },
    // Device-side GTest with options.
    {
      "name" : "hello_world_test",
      "options": [
        {
          "native-test-flag": "\"servicename1 servicename2\""
        },
        {
          "native-test-timeout": "6000"
        }
      ]
    }
    // Host-side GTest.
    {
      "name" : "net_test_avrcp",
      "host" : true
    }
  ],
  "postsubmit": [
    {
      "name": "CtsDeqpTestCases",
      "options": [
        {
          // Use regex in include-filter which is supported in AndroidJUnitTest
          "include-filter": "dEQP-EGL.functional.color_clears.*"
        }
      ]
    }
  ],
  "imports": [
    {
      "path": "frameworks/base/services/core/java/com/android/server/am"
    }
  ]
}

Ustaw atrybuty

W tym przykładzie, presubmit i postsubmit to nazwy poszczególnych grup testowych. Więcej informacji o grupach testowych znajdziesz w sekcji Definiowanie grup testowych.

Wartością atrybutu name możesz ustawić nazwę modułu testowego lub nazwę testu integracyjnego Trade Federation name (ścieżkę do pliku XML testu, np. uiautomator/uiautomator-demo) . Pamiętaj, że pole name nie może używać nazwy klasy ani nazwy metody testowej.namename Aby zawęzić zakres testów do uruchomienia, użyj opcji takich jak include-filter. Zobacz include-filter przykładowe użycie.

Ustawienie host testu wskazuje, czy test jest testem bez urządzenia uruchamianym na hoście. Wartością domyślną jest false, co oznacza, że test wymaga urządzenia. Obsługiwane typy testów to HostGTest w przypadku plików binarnych GTest i HostTest w przypadku testów JUnit.

Atrybut file_patterns umożliwia ustawienie listy ciągów wyrażeń regularnych do dopasowywania ścieżki względnej dowolnego pliku kodu źródłowego (względem katalogu zawierającego plik TEST_MAPPING). W przykładzie, test CtsWindowManagerDeviceTestCases jest uruchamiany przed przesłaniem tylko wtedy, gdy plik Java zaczyna się od Window lub Activity, i znajduje się w tym samym katalogu co TEST_MAPPING plik lub w którymkolwiek z jego podkatalogów. Ukośniki odwrotne (\) muszą być poprzedzone znakiem ucieczki, ponieważ znajdują się w pliku JSON.

Atrybut imports umożliwia uwzględnianie testów w innych plikach TEST_MAPPING bez kopiowania treści. Uwzględniane są też pliki TEST_MAPPING w katalogach nadrzędnych importowanej ścieżki. Mapowanie testów umożliwia importowanie zagnieżdżone. Oznacza to, że 2 pliki TEST_MAPPING mogą importować się nawzajem, a mapowanie testów może scalać uwzględnione testy.

Atrybut options zawiera dodatkowe opcje wiersza poleceń Tradefed.

Aby uzyskać pełną listę dostępnych opcji dla danego testu, uruchom:

tradefed.sh run commandAndExit [test_module] --help

Więcej informacji o tym, jak działają opcje, znajdziesz w artykule Obsługa opcji w Tradefed.

Uruchamianie testów za pomocą Atest

Aby lokalnie wykonać reguły testów przed przesłaniem:

  1. Przejdź do katalogu zawierającego plik TEST_MAPPING.
  2. Uruchom to polecenie:

    atest
    

Uruchamiane są wszystkie testy przed przesłaniem skonfigurowane w plikach TEST_MAPPING bieżącego katalogu i jego katalogów nadrzędnych. Atest znajduje i uruchamia 2 testy przed przesłaniem (A i B).

Jest to najprostszy sposób na uruchomienie testów przed przesłaniem w plikach TEST_MAPPING w bieżącym katalogu roboczym (CWD) i katalogach nadrzędnych. Atest znajduje i używa pliku TEST_MAPPING w CWD i wszystkich jego katalogach nadrzędnych.

Struktura kodu źródłowego

Ten przykład pokazuje, jak skonfigurować pliki TEST_MAPPING w drzewie źródłowym:

src
├── project_1
│   └── TEST_MAPPING
├── project_2
│   └── TEST_MAPPING
└── TEST_MAPPING

Zawartość pliku src/TEST_MAPPING:

{
  "presubmit": [
    {
      "name": "A"
    }
  ]
}

Zawartość pliku src/project_1/TEST_MAPPING:

{
  "presubmit": [
    {
      "name": "B"
    }
  ],
  "postsubmit": [
    {
      "name": "C"
    }
  ],
  "other_group": [
    {
      "name": "X"
    }
  ]}

Zawartość pliku src/project_2/TEST_MAPPING:

{
  "presubmit": [
    {
      "name": "D"
    }
  ],
  "import": [
    {
      "path": "src/project_1"
    }
  ]}

Określanie katalogów docelowych

Możesz określić katalog docelowy, aby uruchamiać testy w plikach TEST_MAPPING w tym katalogu. To polecenie uruchamia 2 testy (A i B):

atest --test-mapping src/project_1

Uruchamianie reguł testów po przesłaniu

Za pomocą tego polecenia możesz też uruchamiać reguły testów po przesłaniu zdefiniowane w pliku TEST_MAPPING w src_path (domyślnie CWD) i jego katalogach nadrzędnych:

atest [--test-mapping] [src_path]:postsubmit

Uruchamianie tylko testów, które nie wymagają urządzenia

Za pomocą opcji --host możesz uruchamiać w Atest tylko te testy skonfigurowane na hoście, które nie wymagają urządzenia. Bez tej opcji Atest uruchamia oba testy – te, które wymagają urządzenia, i te, które są uruchamiane na hoście bez urządzenia. Testy są uruchamiane w 2 osobnych zestawach:

atest [--test-mapping] --host

Identyfikowanie grup testowych

W poleceniu Atest możesz określić grupy testowe. To polecenie uruchamia wszystkie testy postsubmit powiązane z plikami w katalogu src/project_1, który zawiera tylko 1 test (C).

Możesz też użyć :all, aby uruchomić wszystkie testy niezależnie od grupy. To polecenie uruchamia 4 testy (A, B, C i X):

atest --test-mapping src/project_1:all

Uwzględnianie podkatalogów

Domyślnie uruchamianie testów w TEST_MAPPING za pomocą Atest powoduje uruchamianie tylko testów przed przesłaniem skonfigurowanych w pliku TEST_MAPPING w CWD (lub podanym katalogu) i jego katalogach nadrzędnych. Jeśli chcesz uruchamiać testy we wszystkich plikach TEST_MAPPING w podkatalogach, użyj opcji --include-subdir, aby wymusić uwzględnienie tych testów przez Atest.

atest --include-subdir

Bez opcji --include-subdir Atest uruchamia tylko test A. Z opcją --include-subdir Atest uruchamia 2 testy (A i B).

Obsługa komentarzy na poziomie wiersza

Możesz dodać komentarz w formacie // na poziomie wiersza, aby uzupełnić plik TEST_MAPPING o opis ustawienia, które następuje. ATest i Trade Federation przetwarzają wstępnie TEST_MAPPING do prawidłowego formatu JSON bez komentarzy. Aby plik JSON był przejrzysty, obsługiwany jest tylko komentarz w formacie // na poziomie wiersza.

Przykład:

{
  // For presubmit test group.
  "presubmit": [
    {
      // Run test on module A.
      "name": "A"
    }
  ]
}