使用 Robolectric 和 Android 生成代码覆盖率(测试)报告
我写过许多测试驱动开发与陷阱方面的文章。我认为,其中对测试驱动开发中遇到的陷阱的描述让整个介绍更加完整。测试驱动开发或者通常的测试中,最重要的是你清楚代码中哪些部分经过了测试,而哪些部分需要继续测试。
你可以使用 JaCoCo 搞定上述的问题,它对 Grandle 和 Robolectric 有较好的集成。
配置 build.gradle
第一步,配置 build.gradle。主要代码如下所示。完整代码见 GitHub 。
build.gradle
...
android {
...
buildTypes {
debug {
runProguard false
proguardFile 'proguard-rules.txt'
debuggable true
testCoverageEnabled = true
}
}
...
}
...
apply plugin: 'jacoco'
jacoco {
toolVersion = "0.7.1.201405082137"
}
def coverageSourceDirs = [
'../app/src/main/java'
]
task jacocoTestReport(type:JacocoReport, dependsOn: "testDebug") {
group = "Reporting"
description = "Generate Jacoco coverage reports"
classDirectories = fileTree(
dir: '../app/build/intermediates/classes/debug',
excludes: ['**/R.class',
'**/R$*.class',
'**/*$ViewInjector*.*',
'**/BuildConfig.*',
'**/Manifest*.*']
)
additionalSourceDirs = files(coverageSourceDirs)
sourceDirectories = files(coverageSourceDirs)
executionData = files('../app/build/jacoco/testDebug.exec')
reports {
xml.enabled = true
html.enabled = true
}
}
(现在)我们来看看其中最重要的配置。
- buildType 声明,开启代码覆盖测试。
android {
...
buildTypes {
debug {
runProguard false
proguardFile 'proguard-rules.txt'
debuggable true
testCoverageEnabled = true
}
}
...
}
- 加入一个 JaCoCo 插件,同时,指定使用最新版 :
apply plugin: 'jacoco'
jacoco {
toolVersion = "0.7.1.201405082137"
}
def coverageSourceDirs = [
'../app/src/main/java'
]
- 配置 converageSourceDirs,指定一个文件夹,JaCoCo 将对文件夹中的目标进行反射。
- 配置 JaCoCo 插件,指定你需要测试的类(它们已经经过编译)和不需要测试的类(比如 ButterKnife 注入的 ViewInjector)。
task jacocoTestReport(type:JacocoReport, dependsOn: "testDebug") {
group = "Reporting"
description = "Generate Jacoco coverage reports"
classDirectories = fileTree(
dir: '../app/build/intermediates/classes/debug',
excludes: ['**/R.class',
'**/R$*.class',
'**/*$ViewInjector*.*',
'**/BuildConfig.*',
'**/Manifest*.*']
)
additionalSourceDirs = files(coverageSourceDirs)
sourceDirectories = files(coverageSourceDirs)
executionData = files('../app/build/jacoco/testDebug.exec')
reports {
xml.enabled = true
html.enabled = true
}
}
执行 gradle 任务
修改 gradle.build 文件后,你必须执行与开发环境同步,以检查加入新插件后 gradle 也工作正常。
在使用 JaCoCo 生成测试报告前,还需要提供 testDebug.exec 文件。(提供文件)最简单的方法是打开命令行,对你的项目上执行如下命令 : $ ./gradlew clean assemble
这条命令会清空之前编译生成的 class 文件,并重新构建。
现在,你就可以使用 JaCoCo 生成测试报告啦,只要执行这条命令: $ ./gradlew jacocoTestReport
(你会看到)终端将开始执行 grable 构建脚本,其中,最后一项任务是使用 JaCoCo 生成测试报告:
> `$ ./gradlew jacocoTestReport :app:preBuild
:app:preDebugBuild
:app:checkDebugManifest
:app:preReleaseBuild
:app:prepareComAndroidSupportSupportV42000Library UP-TO-DATE
:app:prepareDeKeyboardsurferAndroidWidgetCrouton184Library UP-TO-DATE
:app:prepareDebugDependencies
:app:compileDebugAidl UP-TO-DATE
:app:compileDebugRenderscript UP-TO-DATE
:app:generateDebugBuildConfig UP-TO-DATE
:app:generateDebugAssets UP-TO-DATE
:app:mergeDebugAssets UP-TO-DATE
:app:generateDebugResValues UP-TO-DATE
:app:generateDebugResources UP-TO-DATE
:app:mergeDebugResources UP-TO-DATE
:app:processDebugManifest UP-TO-DATE
:app:processDebugResources UP-TO-DATE
:app:generateDebugSources UP-TO-DATE
:app:compileDebugJava UP-TO-DATE
:app:compileTestDebugJava
:app:processTestDebugResources UP-TO-DATE
:app:testDebugClasses
:app:testDebug
:app:jacocoTestReport
BUILD SUCCESSFUL
Total time: 29.482 secs
生成的代码覆盖率测试报告保存在 ./build/reports/jacoco/jacocoTestReport
中。
注意事项
- 你的 应用名
在我的例子中,Android module 名是"app"。因此包含 '../app/src/main/java'
中的代码。如果你的 Android module 名和例子中的不同,就请修改 gradle 文件中路径(所有涉及到 Android module 相关的路径)。比如,如果你的 Android module 名是 FooBar,配置文件中就应修改为 "..app/src/main/java"
。
- 产品类別
例子中没用指明产品类别,所以构建任务使用 "testDebug"
和 "../app/build/intermediates/classes/debug"
中的 class 文件。但是,如果你在应用中指定产品类别(比如. Local),Gradle 就找不到"testDebug"任务。所以,需要正确的命名,比如,这里你可以用 testLocalDebug 并包含正确的 class 文件: '../app/build/intermediated/classes/debug'
。
如果你有任何问题,别犹豫,直接来问我。代码在 Github 上已经更新,请 Check Out。
术语:code coverage 代码覆盖率:软件测试中用来表示被测软件中被测试代码占整个软件的比例或程度。
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论