非同期操作

この章では、Espresso IdlingResourcesを使用して非同期操作をテストする方法を学習します。

最新のアプリケーションの課題の1つは、スムーズなユーザーエクスペリエンスを提供することです。スムーズなユーザーエクスペリエンスを提供するには、アプリケーションプロセスに数ミリ秒以上かかることがないようにするために、バックグラウンドで多くの作業が必要になります。バックグラウンドタスクは、リモートAPI /データベースからデータをフェッチする単純なタスクからコストのかかる複雑なタスクまで多岐にわたります。過去の課題に対処するために、開発者はバックグラウンドスレッドでコストのかかる長時間実行タスクを記述し、バックグラウンドスレッドが完了するとメインのUIThreadと同期していました。

マルチスレッドアプリケーションの開発が複雑な場合、そのアプリケーションのテストケースの作成はさらに複雑になります。たとえば、必要なデータがデータベースからロードされる前に、AdapterViewをテストしないでください。データのフェッチが別のスレッドで行われる場合、テストはスレッドが完了するまで待機する必要があります。したがって、テスト環境はバックグラウンドスレッドとUIスレッドの間で同期する必要があります。Espressoは、マルチスレッドアプリケーションをテストするための優れたサポートを提供します。アプリケーションは次の方法でスレッドを使用し、espressoはすべてのシナリオをサポートします。

ユーザーインターフェイスのスレッド

複雑なUI要素でスムーズなユーザーエクスペリエンスを提供するために、AndroidSDKによって内部的に使用されます。Espressoはこのシナリオを透過的にサポートし、構成や特別なコーディングは必要ありません。

非同期タスク

最新のプログラミング言語は、スレッドプログラミングの複雑さを伴わずに軽量スレッドを実行するために、非同期プログラミングをサポートしています。非同期タスクは、espressoフレームワークによって透過的にサポートされます。

ユーザースレッド

開発者は、データベースから複雑なデータや大きなデータをフェッチするために新しいスレッドを開始する場合があります。このシナリオをサポートするために、espressoはアイドリングリソースの概念を提供します。

この章では、アイドリングリソースの概念とその方法を学びましょう。

概要概要

アイドリングリソースの概念は非常にシンプルで直感的です。基本的な考え方は、長時間実行されているプロセスが別のスレッドで開始されるたびに変数(ブール値)を作成して、プロセスが実行されているかどうかを識別し、テスト環境に登録することです。テスト中に、テストランナーは登録された変数が見つかった場合はそれをチェックし、その実行ステータスを見つけます。実行ステータスがtrueの場合、テストランナーはステータスがfalseになるまで待機します。

Espressoは、実行ステータスを維持するためのインターフェイスIdlingResourcesを提供します。実装する主なメソッドはisIdleNow()です。isIdleNow()がtrueを返す場合、espressoはテストプロセスを再開するか、isIdleNow()がfalseを返すまで待機します。IdlingResourcesを実装し、派生クラスを使用する必要があります。Espressoは、ワークロードを軽減するために、組み込みのIdlingResources実装の一部も提供します。それらは次のとおりです、

CountingIdlingResource

これにより、実行中のタスクの内部カウンターが維持されます。これは、increment(メソッドとdecrement()メソッドを公開しますインクリメント()はカウンターに1を追加し、デクリメント()はカウンターから1を削除します。isIdleNow()は、アクティブなタスクがない場合にのみtrueを返します。

UriIdlingResource

これはCounintIdlingResourceに似ていますが、ネットワーク遅延を取得するために、カウンターを長期間ゼロにする必要がある点が異なります。

IdlingThreadPoolExecutor

これは、現在のスレッドプールでアクティブな実行中のタスクの数を維持するためのThreadPoolExecutorのカスタム実装です。

IdlingScheduledThreadPoolExecutor

これはIdlingThreadPoolExecutorに似ていますが、タスクとScheduledThreadPoolExecutorのカスタム実装もスケジュールします。

上記のIdlingResourcesの実装またはカスタム実装のいずれかがアプリケーションで使用されている場合は、以下のようにIdlingRegistryクラスを使用してアプリケーションをテストする前に、それをテスト環境にも登録する必要があります。

IdlingRegistry.getInstance().register(MyIdlingResource.getIdlingResource());

また、以下のようにテストが完了したら削除できます。

IdlingRegistry.getInstance().unregister(MyIdlingResource.getIdlingResource());

Espressoはこの機能を別のパッケージで提供し、パッケージはapp.gradleで以下のように構成する必要があります。

dependencies {
   implementation 'androidx.test.espresso:espresso-idling-resource:3.1.1'
   androidTestImplementation "androidx.test.espresso.idling:idlingconcurrent:3.1.1"
}

サンプルアプリケーション

別のスレッドでWebサービスから取得して果物を一覧表示する簡単なアプリケーションを作成し、アイドリングリソースの概念を使用してテストしてみましょう。

  • AndroidStudioを起動します。

  • 前に説明したように新しいプロジェクトを作成し、MyIdlingFruitAppという名前を付けます

  • 移行使っAndroidXフレームワークにアプリケーションをリファクタリングする→移行をAndroidXのオプションメニュー。

  • 以下に指定するように、app / build.gradleにエスプレッソアイドリングリソースライブラリを追加します(そして同期します)。

dependencies {
   implementation 'androidx.test.espresso:espresso-idling-resource:3.1.1'
   androidTestImplementation "androidx.test.espresso.idling:idlingconcurrent:3.1.1"
}
  • メインアクティビティのデフォルトデザインを削除し、ListViewを追加します。activity_main.xmlの内容は次のとおりです。

<?xml version = "1.0" encoding = "utf-8"?>
<RelativeLayout xmlns:android = "http://schemas.android.com/apk/res/android"
   xmlns:app = "http://schemas.android.com/apk/res-auto"
   xmlns:tools = "http://schemas.android.com/tools"
   android:layout_width = "match_parent"
   android:layout_height = "match_parent"
   tools:context = ".MainActivity">
   <ListView
      android:id = "@+id/listView"
      android:layout_width = "wrap_content"
      android:layout_height = "wrap_content" />
</RelativeLayout>
  • 新しいレイアウトリソースitem.xmlを追加して、リストビューのアイテムテンプレートを指定します。item.xmlの内容は次のとおりです。

<?xml version = "1.0" encoding = "utf-8"?>
<TextView xmlns:android = "http://schemas.android.com/apk/res/android"
   android:id = "@+id/name"
   android:layout_width = "fill_parent"
   android:layout_height = "fill_parent"
   android:padding = "8dp"
/>
  • 新しいクラス–MyIdlingResourceを作成します。MyIdlingResourceは、IdlingResourceを1つの場所に保持し、必要に応じてフェッチするために使用されます。この例では、CountingIdlingResourceを使用します。

package com.tutorialspoint.espressosamples.myidlingfruitapp;
import androidx.test.espresso.IdlingResource;
import androidx.test.espresso.idling.CountingIdlingResource;

public class MyIdlingResource {
   private static CountingIdlingResource mCountingIdlingResource =
      new CountingIdlingResource("my_idling_resource");
   public static void increment() {
      mCountingIdlingResource.increment();
   }
   public static void decrement() {
      mCountingIdlingResource.decrement();
   }
   public static IdlingResource getIdlingResource() {
      return mCountingIdlingResource;
   }
}
  • グローバル変数を宣言し、mIdlingResource型のCountingIdlingResourceMainActivityの以下のようなクラス、

@Nullable
private CountingIdlingResource mIdlingResource = null;
  • 以下のように、Webからフルーツリストをフェッチするプライベートメソッドを記述します。

private ArrayList<String> getFruitList(String data) {
   ArrayList<String> fruits = new ArrayList<String>();
   try {
      // Get url from async task and set it into a local variable
      URL url = new URL(data);
      Log.e("URL", url.toString());
      
      // Create new HTTP connection
      HttpURLConnection conn = (HttpURLConnection) url.openConnection();
      
      // Set HTTP connection method as "Get"
      conn.setRequestMethod("GET");
      
      // Do a http request and get the response code
      int responseCode = conn.getResponseCode();
      
      // check the response code and if success, get response content
      if (responseCode == HttpURLConnection.HTTP_OK) {
         BufferedReader in = new BufferedReader(new InputStreamReader(conn.getInputStream()));
         String line;
         StringBuffer response = new StringBuffer();
         while ((line = in.readLine()) != null) {
            response.append(line);
         }
         in.close();
         JSONArray jsonArray = new JSONArray(response.toString());
         Log.e("HTTPResponse", response.toString());
         for(int i = 0; i < jsonArray.length(); i++) {
            JSONObject jsonObject = jsonArray.getJSONObject(i);
            String name = String.valueOf(jsonObject.getString("name"));
            fruits.add(name);
         }
      } else {
         throw new IOException("Unable to fetch data from url");
      }
      conn.disconnect();
   } catch (IOException | JSONException e) {
      e.printStackTrace();
   }
   return fruits;
}
  • onCreate()メソッドで新しいタスクを作成し、getFruitListメソッドを使用してWebからデータをフェッチした後、新しいアダプターを作成してリストビューに設定します。また、スレッドでの作業が完了したら、アイドリングリソースをデクリメントします。コードは次のとおりです。

// Get data
class FruitTask implements Runnable {
   ListView listView;
   CountingIdlingResource idlingResource;
   FruitTask(CountingIdlingResource idlingRes, ListView listView) {
      this.listView = listView;
      this.idlingResource = idlingRes;
   }
   public void run() {
      //code to do the HTTP request
      final ArrayList<String> fruitList = getFruitList("http://<your domain or IP>/fruits.json");
      try {
         synchronized (this){
            runOnUiThread(new Runnable() {
               @Override
               public void run() {
                  // Create adapter and set it to list view
                  final ArrayAdapter adapter = new
                     ArrayAdapter(MainActivity.this, R.layout.item, fruitList);
                  ListView listView = (ListView)findViewById(R.id.listView);
                  listView.setAdapter(adapter);
               }
            });
         }
      } catch (Exception e) {
         e.printStackTrace();
      }
      if (!MyIdlingResource.getIdlingResource().isIdleNow()) {
         MyIdlingResource.decrement(); // Set app as idle.
      }
   }
}

ここでは、フルーツのURLはhttp:// <yourdomainまたはIP / fruits.jsonと見なされ、JSONとしてフォーマットされます。内容は以下の通りです、

[ 
   {
      "name":"Apple"
   },
   {
      "name":"Banana"
   },
   {
      "name":"Cherry"
   },
   {
      "name":"Dates"
   },
   {
      "name":"Elderberry"
   },
   {
      "name":"Fig"
   },
   {
      "name":"Grapes"
   },
   {
      "name":"Grapefruit"
   },
   {
      "name":"Guava"
   },
   {
      "name":"Jack fruit"
   },
   {
      "name":"Lemon"
   },
   {
      "name":"Mango"
   },
   {
      "name":"Orange"
   },
   {
      "name":"Papaya"
   },
   {
      "name":"Pears"
   },
   {
      "name":"Peaches"
   },
   {
      "name":"Pineapple"
   },
   {
      "name":"Plums"
   },
   {
      "name":"Raspberry"
   },
   {
      "name":"Strawberry"
   },
   {
      "name":"Watermelon"
   }
]

Note −ファイルをローカルWebサーバーに配置して使用します。

  • ここで、ビューを見つけ、FruitTaskを渡して新しいスレッドを作成し、アイドリングリソースをインクリメントして、最後にタスクを開始します。

// Find list view
ListView listView = (ListView) findViewById(R.id.listView);
Thread fruitTask = new Thread(new FruitTask(this.mIdlingResource, listView));
MyIdlingResource.increment();
fruitTask.start();
  • MainActivityの完全なコードは次のとおりです。

package com.tutorialspoint.espressosamples.myidlingfruitapp;

import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.annotation.VisibleForTesting;
import androidx.appcompat.app.AppCompatActivity;
import androidx.test.espresso.idling.CountingIdlingResource;

import android.os.Bundle;
import android.util.Log;
import android.widget.ArrayAdapter;
import android.widget.ListView;

import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.HttpURLConnection;
import java.net.URL;
import java.util.ArrayList;

public class MainActivity extends AppCompatActivity {
   @Nullable
   private CountingIdlingResource mIdlingResource = null;
   @Override
   protected void onCreate(Bundle savedInstanceState) {
      super.onCreate(savedInstanceState);
      setContentView(R.layout.activity_main);
      
      // Get data
      class FruitTask implements Runnable {
         ListView listView;
         CountingIdlingResource idlingResource;
         FruitTask(CountingIdlingResource idlingRes, ListView listView) {
            this.listView = listView;
            this.idlingResource = idlingRes;
         }
         public void run() {
            //code to do the HTTP request
            final ArrayList<String> fruitList = getFruitList(
               "http://<yourdomain or IP>/fruits.json");
            try {
               synchronized (this){
                  runOnUiThread(new Runnable() {
                     @Override
                     public void run() {
                        // Create adapter and set it to list view
                        final ArrayAdapter adapter = new ArrayAdapter(
                           MainActivity.this, R.layout.item, fruitList);
                        ListView listView = (ListView) findViewById(R.id.listView);
                        listView.setAdapter(adapter);
                     }
                  });
               }
            } catch (Exception e) {
               e.printStackTrace();
            }
            if (!MyIdlingResource.getIdlingResource().isIdleNow()) {
               MyIdlingResource.decrement(); // Set app as idle.
            }
         }
      }
      // Find list view
      ListView listView = (ListView) findViewById(R.id.listView);
      Thread fruitTask = new Thread(new FruitTask(this.mIdlingResource, listView));
      MyIdlingResource.increment();
      fruitTask.start();
   }
   private ArrayList<String> getFruitList(String data) {
      ArrayList<String> fruits = new ArrayList<String>();
      try {
         // Get url from async task and set it into a local variable
         URL url = new URL(data);
         Log.e("URL", url.toString());
         
         // Create new HTTP connection
         HttpURLConnection conn = (HttpURLConnection) url.openConnection();
         
         // Set HTTP connection method as "Get"
         conn.setRequestMethod("GET");
         
         // Do a http request and get the response code
         int responseCode = conn.getResponseCode();
         
         // check the response code and if success, get response content
         if (responseCode == HttpURLConnection.HTTP_OK) {
            BufferedReader in = new BufferedReader(new InputStreamReader(conn.getInputStream()));
            String line;
            StringBuffer response = new StringBuffer();
            while ((line = in.readLine()) != null) {
               response.append(line);
            }
            in.close();
            JSONArray jsonArray = new JSONArray(response.toString());
            Log.e("HTTPResponse", response.toString());
            
            for(int i = 0; i < jsonArray.length(); i++) {
               JSONObject jsonObject = jsonArray.getJSONObject(i);
               String name = String.valueOf(jsonObject.getString("name"));
               fruits.add(name);
            }
         } else {
            throw new IOException("Unable to fetch data from url");
         }
         conn.disconnect();
      } catch (IOException | JSONException e) {
         e.printStackTrace();
      }
      return fruits;
   }
}
  • 次に、アプリケーションマニフェストファイルAndroidManifest.xmlに以下の構成を追加します

<uses-permission android:name = "android.permission.INTERNET" />
  • 次に、上記のコードをコンパイルして、アプリケーションを実行します。My IdlingFruitアプリのスクリーンショットは次のとおりです。

  • ここで、ExampleInstrumentedTest.javaファイルを開き、以下に指定されているようにActivityTestRuleを追加します。

@Rule
public ActivityTestRule<MainActivity> mActivityRule = 
   new ActivityTestRule<MainActivity>(MainActivity.class);
Also, make sure the test configuration is done in app/build.gradle
dependencies {
   testImplementation 'junit:junit:4.12'
   androidTestImplementation 'androidx.test:runner:1.1.1'
   androidTestImplementation 'androidx.test:rules:1.1.1'
   androidTestImplementation 'androidx.test.espresso:espresso-core:3.1.1'
   implementation 'androidx.test.espresso:espresso-idling-resource:3.1.1'
   androidTestImplementation "androidx.test.espresso.idling:idlingconcurrent:3.1.1"
}
  • 以下のようにリストビューをテストするための新しいテストケースを追加します。

@Before
public void registerIdlingResource() {
   IdlingRegistry.getInstance().register(MyIdlingResource.getIdlingResource());
}
@Test
public void contentTest() {
   // click a child item
   onData(allOf())
   .inAdapterView(withId(R.id.listView))
   .atPosition(10)
   .perform(click());
}
@After
public void unregisterIdlingResource() {
   IdlingRegistry.getInstance().unregister(MyIdlingResource.getIdlingResource());
}
  • 最後に、Android Studioのコンテキストメニューを使用してテストケースを実行し、すべてのテストケースが成功しているかどうかを確認します。