1

У меня есть сборка на TeamCity, которая запускает модульные тесты через Gradle. Промежуточно, тесты, которые связаны с фрагментами или действиями, терпят неудачу с ClassNotFoundException для таких классов, как android.support.v4.app.FragmentTransitionCompat21$ViewRetriever или android.support.v4.app.ActivityCompat21$SharedElementCallback21. Тесты не срабатывают при запуске фрагмента, и я попытался использовать все методы для начала фрагмента из этого вопроса - How can I test fragments with Robolectric?.Тест модуля Robolectric для фрагмента прерывается с исключением класса ClassNotFoundException

Вот пример для теста:

@Test 
public void ContactSupportFragment_CallBtnClicked_CallWasMade() throws Exception 
{ 
    ContactSupportFragment fragment = new ContactSupportFragment(); 
    startFragment(fragment); 

    LinearLayout btnCall = (LinearLayout) fragment.getView().findViewById(R.id.contact_support_call_btn); 
    btnCall.performClick(); 

    Mockito.verify(techSupportCall, Mockito.times(1)).call(Mockito.any(Context.class), 
     Mockito.eq(Robolectric.application.getString(R.string.tech_support_phone_number))); 
} 

Вот пример для трассировки стека:

java.lang.NoClassDefFoundError: android/support/v4/app/FragmentTransitionCompat21$ViewRetriever 
at android.support.v4.app.FragmentManagerImpl.beginTransaction(FragmentManager.java:481) 
at org.robolectric.util.FragmentTestUtil.startFragment(FragmentTestUtil.java:25) 
at com.asurion.solutohome.callsupport.ContactSupportFragmentTest.ContactSupportFragment_CallBtnClicked_CallWasMade(ContactSupportFragmentTest.java:70) 
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50) 
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12) 
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47) 
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17) 
at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:26) 
at org.robolectric.RobolectricTestRunner$2.evaluate(RobolectricTestRunner.java:236) 
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325) 
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78) 
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57) 
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290) 
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71) 
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288) 
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58) 
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268) 
at org.robolectric.RobolectricTestRunner$1.evaluate(RobolectricTestRunner.java:158) 
at org.junit.runners.ParentRunner.run(ParentRunner.java:363) 
at org.gradle.api.internal.tasks.testing.junit.JUnitTestClassExecuter.runTestClass(JUnitTestClassExecuter.java:86) 
at org.gradle.api.internal.tasks.testing.junit.JUnitTestClassExecuter.execute(JUnitTestClassExecuter.java:49) 
at org.gradle.api.internal.tasks.testing.junit.JUnitTestClassProcessor.processTestClass(JUnitTestClassProcessor.java:69) 
at org.gradle.api.internal.tasks.testing.SuiteTestClassProcessor.processTestClass(SuiteTestClassProcessor.java:48) 
at org.gradle.messaging.dispatch.ReflectionDispatch.dispatch(ReflectionDispatch.java:35) 
at org.gradle.messaging.dispatch.ReflectionDispatch.dispatch(ReflectionDispatch.java:24) 
at org.gradle.messaging.dispatch.ContextClassLoaderDispatch.dispatch(ContextClassLoaderDispatch.java:32) 
at org.gradle.messaging.dispatch.ProxyDispatchAdapter$DispatchingInvocationHandler.invoke(ProxyDispatchAdapter.java:93) 
at com.sun.proxy.$Proxy2.processTestClass(Unknown Source) 
at org.gradle.api.internal.tasks.testing.worker.TestWorker.processTestClass(TestWorker.java:105) 
at org.gradle.messaging.dispatch.ReflectionDispatch.dispatch(ReflectionDispatch.java:35) 
at org.gradle.messaging.dispatch.ReflectionDispatch.dispatch(ReflectionDispatch.java:24) 
at org.gradle.messaging.remote.internal.hub.MessageHub$Handler.run(MessageHub.java:360) 
at org.gradle.internal.concurrent.DefaultExecutorFactory$StoppableExecutorImpl$1.run(DefaultExecutorFactory.java:64) 
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1145) 
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:615) 
at java.lang.Thread.run(Thread.java:745) 
Caused by: java.lang.ClassNotFoundException: android.support.v4.app.FragmentTransitionCompat21$ViewRetriever 
at org.robolectric.bytecode.AsmInstrumentingClassLoader.loadClass(AsmInstrumentingClassLoader.java:88) 
at android.support.v4.app.FragmentManagerImpl.$$robo$$FragmentManagerImpl_917e_beginTransaction(FragmentManager.java:481) 
at android.support.v4.app.FragmentManagerImpl.beginTransaction(FragmentManager.java) 
at org.robolectric.util.FragmentTestUtil.startFragment(FragmentTestUtil.java:25) 
at com.asurion.solutohome.callsupport.ContactSupportFragmentTest.ContactSupportFragment_CallBtnClicked_CallWasMade(ContactSupportFragmentTest.java:70) 
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) 
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57) 
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) 
at java.lang.reflect.Method.invoke(Method.java:606) 
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50) 
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12) 
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47) 
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17) 
at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:26) 
at org.robolectric.RobolectricTestRunner$2.evaluate(RobolectricTestRunner.java:236) 
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325) 
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78) 
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57) 
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290) 
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71) 
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288) 
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58) 
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268) 
at org.robolectric.RobolectricTestRunner$1.evaluate(RobolectricTestRunner.java:158) 
at org.junit.runners.ParentRunner.run(ParentRunner.java:363) 
at org.gradle.api.internal.tasks.testing.junit.JUnitTestClassExecuter.runTestClass(JUnitTestClassExecuter.java:86) 
at org.gradle.api.internal.tasks.testing.junit.JUnitTestClassExecuter.execute(JUnitTestClassExecuter.java:49) 
at org.gradle.api.internal.tasks.testing.junit.JUnitTestClassProcessor.processTestClass(JUnitTestClassProcessor.java:69) 
at org.gradle.api.internal.tasks.testing.SuiteTestClassProcessor.processTestClass(SuiteTestClassProcessor.java:48) 
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) 
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57) 
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) 
at java.lang.reflect.Method.invoke(Method.java:606) 
at org.gradle.messaging.dispatch.ReflectionDispatch.dispatch(ReflectionDispatch.java:35) 
at org.gradle.messaging.dispatch.ReflectionDispatch.dispatch(ReflectionDispatch.java:24) 
at org.gradle.messaging.dispatch.ContextClassLoaderDispatch.dispatch(ContextClassLoaderDispatch.java:32) 
at org.gradle.messaging.dispatch.ProxyDispatchAdapter$DispatchingInvocationHandler.invoke(ProxyDispatchAdapter.java:93) 
at com.sun.proxy.$Proxy2.processTestClass(Unknown Source) 
at org.gradle.api.internal.tasks.testing.worker.TestWorker.processTestClass(TestWorker.java:105) 
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) 
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57) 
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) 
at java.lang.reflect.Method.invoke(Method.java:606) 

Это не произошло еще локально, только на агента TeamCity, но я убежищем 't нашел что-то другое между моей машиной и агентом (тот же SDK, тот же Gradle build и т. д.). Что может вызвать эти исключения?

+0

Если вы попытаетесь запустить его во второй раз без очистки. Проходит ли это? –

+0

@EugenMartynov он проходит, но поскольку он терпит неудачу, я не мог быть уверен, что это потому, что я не работаю чистым ... Если это связано, что это значит, проблема? – Fastas

+0

На Robolectic Github есть несколько вопросов (https://github.com/robolectric/robolectric/issues/1321). Проблема всплывает на CI строит и проходит, когда выполняется второй раз без очистки. Это похоже на выпуск плагина robolectric gradle, который беспорядок с именем задач и соответствующими классами. –

ответ

2

Следующее предложение @ EugenMartynov я добавил дополнительный сборный шаг сборки к моей сборке Teamcity, прежде чем строить тесты, и до сих пор все сборки проходят. Похоже, что у Robolectric действительно есть некоторые проблемы с сборками CI.

Итак, подведем итог, вместо строительства раз и ходовых испытаний, моя сборка работает:

  • чистые и строить
  • строить заново и запустить тесты

Все тесты проходят каждую сборку ,

+0

Мне удалось перезапустить тест сборки через град как задачу перед тестом перед запуском тестов, и это не решило проблему для меня. – markshiz

+0

Мне пришлось вручную запускать 'gradle gather; gradle test', чтобы это успешно работало. кажется, что что-то не полностью разрешилось в системе сборки. – markshiz

0

Я не очень в теме Robolectric, но StackTrace отправленного выглядит, как вы забыли добавить двоичные файлы или банку, который включает в себя класс android.support.v4.app.FragmentTransitionCompat21 (я думаю, что это класс включен в SDK android), в ваш путь к классам.

+0

Я еще не очень опытен с Gradle, но насколько я могу судить, понятие classpaths немного отличается от него. Модуль уже имеет зависимость от библиотеки поддержки, и я понимаю, что добавление его в путь к классам влияет только на сам процесс сборки. – Fastas

0

Согласно @Fastas ответ. Вы можете запустить

gradle test 

Это решит проблему, но по этой причине она пока неизвестна.

С моей стороны, я думаю, что это связано с зависимостями от моего проекта app, который не полностью разрешает первую сборку внешнего проекта app-test. Я использую следующий код в app-test/build.gradle.

dependencies { 
    // other dependencies ... 

    compile project(':app') 
    testCompile androidModule.android.applicationVariants.toList().first().javaCompile.classpath 
    testCompile androidModule.android.applicationVariants.toList().first().javaCompile.outputs.files 
    testCompile files(androidModule.plugins.findPlugin("com.android.application").getBootClasspath()) 
} 

выше, возможно, classpath или outputs.files не будет полностью решен до assemble завершения задания.

Update: 23 марта 2015

Перемещение моего тестового набора для экспериментальной проверкой поддержки в модуле приложения решен этот вопрос для меня.

Смежные вопросы