12

Я пытаюсь интегрировать Espresso 2.0's AndoridJUnitRunner с ActivityUnitTestCase. Тем не менее, мои тесты сбой при startActivity() tries to initialize mMockParent = new MockParent().ActivityUnitTestCase throws RuntimeException при запуске с AndroidJUnitRunner

Вот что я сделал:

Создать новый проект с Intellij 14 CE и внести некоторые изменения в build.gradle.

android{ 
    defaultConfig { 
     applicationId "com.noob.testing" 
     minSdkVersion 9 
     targetSdkVersion 21 
     versionCode 1 
     versionName "1.0" 

     testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner" 

    } 
    compileOptions { 
     sourceCompatibility JavaVersion.VERSION_1_7 
     targetCompatibility JavaVersion.VERSION_1_7 
    } 
} 
dependencies { 
    compile fileTree(dir: 'libs', include: ['*.jar']) 
    compile 'com.android.support:appcompat-v7:21.0.+' 

    androidTestCompile('org.mockito:mockito-core:1.9.5') 
    androidTestCompile('com.google.dexmaker:dexmaker:1.2') 
    androidTestCompile('com.google.dexmaker:dexmaker-mockito:1.2') 

    androidTestCompile('com.android.support.test.espresso:espresso-core:2.0') 
    androidTestCompile('com.android.support.test:testing-support-lib:0.1') 
} 

Запишите единичный тест в стиле JUnit4.

@RunWith(AndroidJUnit4.class) 
public class MainActivityJUnit4Test extends ActivityUnitTestCase<MainActivity> { 
    public MainActivityJUnit4Test() { 
     super(MainActivity.class); 
    } 

    MainActivity activity; 

    @Before 
    public void setup() throws Exception { 
     injectInstrumentation(InstrumentationRegistry.getInstrumentation()); 
     super.setUp(); 
     ContextThemeWrapper context = new ContextThemeWrapper(getInstrumentation().getTargetContext(), R.style.AppTheme); 
     setActivityContext(context); 
     activity = startActivity(new Intent(Intent.ACTION_MAIN), null, null); 
    } 

    @Test 
    public void baseCase() { 
     TextView tv = (TextView) activity.findViewById(R.id.tv); 
     Assert.assertEquals("Hello World", tv.getText()); 

    } 
} 

Запустите тест, получив трассировку стека.

junit.framework.AssertionFailedError 
at junit.framework.Assert.fail(Assert.java:48) 
at junit.framework.Assert.assertTrue(Assert.java:20) 
at junit.framework.Assert.assertNotNull(Assert.java:218) 
at junit.framework.Assert.assertNotNull(Assert.java:211) 
at android.test.ActivityUnitTestCase.startActivity(ActivityUnitTestCase.java:147) 
at com.sdchang.testing.MainActivityJUnit4Test.setup(MainActivityJUnit4Test.java:32) 
at java.lang.reflect.Method.invokeNative(Native Method) 
at java.lang.reflect.Method.invoke(Method.java:525) 
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:45) 
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15) 
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:42) 
at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:27) 
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:263) 
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:68) 
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:47) 
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:231) 
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:60) 
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:229) 
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:50) 
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:222) 
at org.junit.runners.ParentRunner.run(ParentRunner.java:300) 
at org.junit.runners.Suite.runChild(Suite.java:128) 
at org.junit.runners.Suite.runChild(Suite.java:24) 
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:231) 
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:60) 
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:229) 
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:50) 
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:222) 
at org.junit.runners.ParentRunner.run(ParentRunner.java:300) 
at org.junit.runner.JUnitCore.run(JUnitCore.java:157) 
at org.junit.runner.JUnitCore.run(JUnitCore.java:136) 
at android.support.test.runner.AndroidJUnitRunner.onStart(AndroidJUnitRunner.java:270) 
at android.app.Instrumentation$InstrumentationThread.run(Instrumentation.java:1701) 

Этот стек трассировки фактически Rethrow из

java.lang.RuntimeException: Can't create handler inside thread that has not called Looper.prepare() 

, что происходит, когда startActivity() tries to initialize mMockParent = new MockParent():

  ComponentName cn = new ComponentName(mActivityClass.getPackage().getName(), 
        mActivityClass.getName()); 
      intent.setComponent(cn); 
      ActivityInfo info = new ActivityInfo(); 
      CharSequence title = mActivityClass.getName(); 
      mMockParent = new MockParent(); 
      String id = null; 

Я пропускаю что-нибудь еще, чтобы получить AndroidJUnitRunner работать с ActivityUnitTestCase? Любая помощь будет принята с благодарностью.

ответ

20

После того, как я ударил головой о стену, окна и различную мебель, я, наконец, получил свои тесты!

От a response on its issue tracker я узнал, что из инструментальной нити не может быть создано никаких новых обработчиков. Это означает, что startActivity() должен быть вызван в потоке пользовательского интерфейса. Это приводит к 3-мя решениям. runOnMainSync()

getInstrumentation().runOnMainSync(new Runnable() { 
    @Override 
    public void run() { 
     activity = startActivity(new Intent(Intent.ACTION_MAIN), null, null); 
    } 
}); 

2.Create собственный базовый класс

1.Use КИП, которая расширяет ActivityUnitTestCase и переопределить startActivity

@Override 
protected T startActivity(final Intent intent, final Bundle savedInstanceState, 
     final Object lastNonConfigurationInstance) { 
    return startActivityOnMainThread(intent, savedInstanceState, lastNonConfigurationInstance); 
} 

private T startActivityOnMainThread(final Intent intent, final Bundle savedInstanceState, 
     final Object lastNonConfigurationInstance) { 
    final AtomicReference<T> activityRef = new AtomicReference<>(); 
    final Runnable activityRunnable = new Runnable() { 
     @Override 
     public void run() { 
      activityRef.set(YourBaseActivityUnitTestCase.super.startActivity(
        intent, savedInstanceState, lastNonConfigurationInstance)); 
     } 
    }; 

    if (Looper.myLooper() != Looper.getMainLooper()) { 
     getInstrumentation().runOnMainSync(activityRunnable); 
    } else { 
     activityRunnable.run(); 
    } 

    return activityRef.get(); 
} 

3.Если вы звоните startActivity() в методе испытаний и ваш весь тест может выполняться в основном потоке, вы можете просто аннотировать свой тестовый метод с помощью @UiThreadTest.

В ходе тестирования я пробовал все 3 решения, , но только 1 & 2 будет работать.

+0

Я не тестировал это, но, возможно, 3 не работает, потому что @UiThreadTest, похоже, применим только к испытаниям, расширяющим «InstrumentationTestCase», а не «ActivityUnitTestCase»? [Android Dev Reference] (http://developer.android.com/reference/android/test/UiThreadTest.html) – twelve17

+0

Кстати, спасибо за косвенный ответ на вопрос, который у меня был, поэтому мне нужно было работать с runOnMainSync на все в моей ActivityUnitTestCase, и ответ заключается в том, что в итоге я добавил другие тесты, которые подклассифицировали «ActivityInstrumentationTestCase2», и я установил Espresso, и это потребовало переключения 'testInstrumentationRunner' в мой файл' build.gradle' с 'android.test.InstrumentationTestRunner' на 'android.support.test.runner.AndroidJUnitRunner', который изменяет предыдущее поведение, позволяющее напрямую обращаться к связанным с UI методам. – twelve17