我试图使用Robolectric进行单元测试;它们在Android Studio下运行正常,但在命令行中运行时完全相同的测试失败 - 这是一个大问题,我需要能够从我的持续集成平台运行它们,而不仅仅是从IDE运行它们。
我怀疑我错过了一些命令行参数(例如,类路径或类似的东西)或者调用错误的任务 - 否则测试将无法从Android Studio运行。一些相关细节;测试看起来像这样:
@RunWith(RobolectricTestRunner.class)
@Config(manifest = "app/src/main/AndroidManifest.xml", resourceDir = "res", emulateSdk = 19)
public class SplashActivityTest {
@Test
public void testActivity() {
SplashActivity splashActivity = new SplashActivity();
String appName = splashActivity.getString(R.string.app_name); // HERE, line 20
assertEquals(appName, "App");
}
}
如上所述,它在Android Studio中正常运行(通过右键单击测试文件并选择Run 'SplashActivityTest'
),但是当从命令行运行它时,它会在标有HERE
的行中失败,使用以下堆栈跟踪:
android.content.res.Resources$NotFoundException: unknown resource 2131492893
at org.robolectric.shadows.ShadowAssetManager.getAndResolve(ShadowAssetManager.java:309)
at org.robolectric.shadows.ShadowAssetManager.getResourceText(ShadowAssetManager.java:69)
at android.content.res.AssetManager.getResourceText(AssetManager.java)
at android.content.res.Resources.getText(Resources.java:240)
at org.robolectric.shadows.ShadowResources.getText(ShadowResources.java:361)
at android.content.res.Resources.getText(Resources.java)
at android.content.res.Resources.getString(Resources.java:330)
at org.robolectric.shadows.ShadowContext.getString(ShadowContext.java:39)
at org.robolectric.shadows.ShadowContextWrapper.getString(ShadowContextWrapper.java:69)
at android.content.Context.getString(Context.java)
at path.to.myApp.activities.SplashActivityTest.testActivity(SplashActivityTest.java:20)
// ... and so on ...
我使用它从命令行运行(请注意,在这里和在Android Studio中我使用Gradle包装器):
project-root$ ./gradlew test --continue
另外:我使用Android Studio 1.1.0
,Gradle版本为2.3
,Robolectric的版本为3.0-SNAPSHOT
,Robolectric的Gradle插件版本为1.0.1
答案 0 :(得分:8)
事实证明这是一个已知问题。从命令行运行测试时,在错误的目录中查找项目的资源,这里是问题的link;目前解决方法是编写一个自定义测试运行器,如演示here:
public class RobolectricGradleTestRunner extends RobolectricTestRunner {
public RobolectricGradleTestRunner(Class<?> testClass) throws InitializationError {
super(testClass);
String buildVariant = (BuildConfig.FLAVOR.isEmpty() ? "" : BuildConfig.FLAVOR+ "/") + BuildConfig.BUILD_TYPE;
String intermediatesPath = BuildConfig.class.getResource("").toString().replace("file:", "");
intermediatesPath = intermediatesPath.substring(0, intermediatesPath.indexOf("/classes"));
System.setProperty("android.package", BuildConfig.APPLICATION_ID);
System.setProperty("android.manifest", intermediatesPath + "/manifests/full/" + buildVariant + "/AndroidManifest.xml");
System.setProperty("android.resources", intermediatesPath + "/res/" + buildVariant);
System.setProperty("android.assets", intermediatesPath + "/assets/" + buildVariant);
}
}
必须相应更新测试用例以使用自定义测试运行器而不是 @Config
注释:
@RunWith(RobolectricGradleTestRunner.class)
public class SplashActivityTest {
// tests same as before
}
此外,建议在从命令行运行测试时执行clean
,因为解决方法取决于构建目录的内容,我们不希望有陈旧的数据:
project-root$ ./gradlew clean test --continue
答案 1 :(得分:0)
我以不同的方式解决了同样的错误。 我在测试中使用了自定义类阴影,因此我必须创建自己的TestRunner类,如下所示:
public class MyRobolectricTestRunner extends RobolectricTestRunner {
public MyRobolectricTestRunner(Class<?> klass) throws InitializationError {
super(klass);
}
public InstrumentationConfiguration createClassLoaderConfig() {
InstrumentationConfiguration.Builder builder = InstrumentationConfiguration.newBuilder();
builder.addInstrumentedClass(AppUtils.class.getName());
return builder.build();
}
}
但是上面的代码只有在我从Andorid工作室运行单元测试时才有效。请查看以两种方式为我工作的代码:在Studio中和从命令行:
public class MyRobolectricTestRunner extends RobolectricTestRunner {
public MyRobolectricTestRunner(Class<?> klass) throws InitializationError {
super(klass);
}
@Override
public InstrumentationConfiguration createClassLoaderConfig() {
InstrumentationConfiguration.Builder builder = InstrumentationConfiguration.newBuilder();
builder.addInstrumentedPackage(com.example.stackoverflow.AppUtils.class.getPackage().getName());
return builder.build();
}
}
请注意我有依赖性:
testCompile 'org.robolectric:robolectric:3.0'