Android 对项目库进行检测单元测试

Android Instrumented unit test on project library

我继承了一个 Android 由主应用程序和一个库组成的项目。我的目标是测试整个项目,所以我从库开始:首先我使用 JUnit 测试一些 classes/method,编写我的库项目的测试 undeer app/src/test/java(单元测试),同时生成一个Android Studio 的代码覆盖率报告。当我找到一个使用 Parcel(来自 Android Framework)的 Class 时,问题就来了,所以这次我不得不设置一个仪器测试(app/src/androidTest/java),也在库项目中。

当我尝试 运行 插桩测试时,插入一个 Android 设备,我收到此消息:

Started running tests
Test running failed: Instrumentation run failed due to 'java.lang.ClassNotFoundException'
Empty test suite.

我不明白为什么 "Empty test suite" 我的测试中有几个 @Test 方法 class。我是否需要设置 Gradle 如何在测试期间 link 主应用程序和库 o 也许我必须将所有测试移动到主应用程序中?!

这是我的 gradle 库模块:

apply plugin: 'com.android.library'


    android {

        compileSdkVersion rootProject.ext.compileSdkVersion
        buildToolsVersion rootProject.ext.buildToolsVersion

        defaultConfig {
            minSdkVersion rootProject.ext.minSdkVersion
            // Added for instrumental tests
            testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"

            javaCompileOptions {
                annotationProcessorOptions {
                    arguments = ["room.schemaLocation": "$projectDir/schemas".toString()]
                }
            }
        }

        buildTypes {
            release {
                minifyEnabled true
                proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
            }
            debug {
                debuggable true
                testCoverageEnabled true
            }
        }

        compileOptions {
            sourceCompatibility JavaVersion.VERSION_1_8
            targetCompatibility JavaVersion.VERSION_1_8
        }

        testOptions {
            unitTests.returnDefaultValues = true
        }

    }

    dependencies {

        implementation "com.android.support:support-v4:$rootProject.supportLibraryVersion"
        implementation "com.google.code.gson:gson:$rootProject.gsonVersion"
        implementation "commons-io:commons-io:$rootProject.commonsioVersion"

        // Dependencies for RxBinding
        implementation 'com.jakewharton.rxbinding2:rxbinding:2.0.0'

        // Dependencies for Rxjava2 and RxAndroid
        implementation 'io.reactivex.rxjava2:rxandroid:2.0.1'
        implementation 'io.reactivex.rxjava2:rxjava:2.1.8'

        // Dependencies for RxBinding
        implementation 'com.jakewharton.rxbinding2:rxbinding:2.0.0'

        // Dependencies for Retrofit
        implementation 'com.squareup.retrofit2:retrofit:2.3.0'
        implementation 'com.squareup.retrofit2:converter-simplexml:2.3.0'
        implementation 'com.jakewharton.retrofit:retrofit2-rxjava2-adapter:1.0.0'
        implementation 'com.squareup.okhttp:okhttp:2.7.0'

        // Dependencies for Room
        // Room (use 1.1.0-alpha1 for latest alpha)
        implementation 'android.arch.persistence.room:runtime:1.1.1'
        implementation 'android.arch.persistence.room:rxjava2:1.1.1'
        annotationProcessor "android.arch.persistence.room:compiler:1.1.1"

        implementation "com.android.support:support-annotations:$rootProject.supportLibraryVersion"
        implementation "org.apache.httpcomponents:httpcore:$rootProject.httpcoreVersion"
        testImplementation 'junit:junit:4.12'

    //    // [Android Instrumented test]
        androidTestImplementation 'junit:junit:4.12'
    //    androidTestImplementation 'androidx.test:runner:1.1.0'
    //    androidTestImplementation 'androidx.test:rules:1.1.0'
    //    // Optional -- Hamcrest library
        androidTestImplementation 'org.hamcrest:hamcrest-library:1.3'
    //    // Optional -- UI testing with Espresso
    //    androidTestImplementation 'androidx.test.espresso:espresso-core:3.1.0'
    //    // Optional -- UI testing with UI Automator
    //    androidTestImplementation 'androidx.test.uiautomator:uiautomator:2.2.0'

        api files('libs/lifesense_ble_v3.3.7.jar')
        api files('libs/MedPixel.jar')

    }

和应用的:

apply plugin: 'com.android.application'

android {
    compileSdkVersion 28
    buildToolsVersion '28.0.3'

    defaultConfig {
        applicationId "it.*******.********"
        minSdkVersion rootProject.ext.minSdkVersion
        targetSdkVersion rootProject.ext.targetSdkVersion
        versionCode 2
        versionName "1.0.2"
    }

    buildTypes {
        release {
            shrinkResources true
            minifyEnabled true
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
        debug {
            applicationIdSuffix ".debug"
            debuggable true
        }
    }

    lintOptions {
        checkReleaseBuilds false
    }

    compileOptions {
        sourceCompatibility JavaVersion.VERSION_1_8
        targetCompatibility JavaVersion.VERSION_1_8
    }
}

dependencies {
    implementation project(':..:Libraries:app')
    implementation "com.android.support:support-v4:$rootProject.supportLibraryVersion"
    implementation "com.google.code.gson:gson:$rootProject.gsonVersion"
    implementation "commons-io:commons-io:$rootProject.commonsioVersion"

    // Dependencies for ButterKnife
    implementation 'com.jakewharton:butterknife:8.8.1'
    annotationProcessor 'com.jakewharton:butterknife-compiler:8.8.1'

    // Dependencies for Dagger
    implementation 'com.google.dagger:dagger:2.11'
    annotationProcessor 'com.google.dagger:dagger-compiler:2.11'

    // Dependencies for RxBinding
    implementation 'com.jakewharton.rxbinding2:rxbinding:2.0.0'

    // Dependencies for Rxjava2 and RxAndroid
    implementation 'io.reactivex.rxjava2:rxandroid:2.0.1'
    implementation 'io.reactivex.rxjava2:rxjava:2.1.8'

    // Dependencies for Retrofit
    implementation 'com.squareup.retrofit2:retrofit:2.3.0'
    implementation 'com.squareup.retrofit2:converter-simplexml:2.3.0'
    implementation 'com.jakewharton.retrofit:retrofit2-rxjava2-adapter:1.0.0'
    implementation 'com.squareup.okhttp:okhttp:2.7.0'

    // Dependencies for Room
    implementation 'android.arch.persistence.room:runtime:1.1.1'

    // Dependencies for Roboelecttic
    testImplementation 'org.robolectric:robolectric:3.6.1'
    testImplementation 'junit:junit:4.12'

    // Dependencies for Mockito
    testImplementation 'org.mockito:mockito-core:2.7.22'
    androidTestImplementation 'com.android.support.test:runner:1.0.2'
    androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'
    androidTestImplementation 'com.squareup.retrofit2:retrofit-mock:2.0.0'
    androidTestImplementation 'com.squareup.okhttp:mockwebserver:2.7.0'
    androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'
    androidTestImplementation 'com.android.support.test:runner:1.0.2'
    androidTestImplementation 'org.mockito:mockito-android:2.7.22'
}

configurations.all {
    resolutionStrategy {
        force "android.arch.core:runtime:1.1.1"
        force "com.android.support:support-core-utils:28.0.0"
        /*force "com.android.support:support-compat:28.0.0"*/
    }
}

感谢关注!

我 运行 遇到了同样的问题,不是 Parcel 而是其他 android 组件,例如 Context.

对于这个问题,

Empty test suite

您在 中的仪器化测试 class 必须用 @RunWith(AndroidJUnit4.class).

进行注释

关于库的代码覆盖率

最初我确实在 app 中编写了 Instrumented 测试。但是在 app 中编写 库的 InstrumentationTests 的问题是,您不会获得 library.

的覆盖率报告

因此,我将 InstrumentedTests 移回了 library/androidTest 文件夹。经过大量研究,我找到了一个解决方案来获取 library

的完整覆盖率报告

在图书馆的 build.gradle 文件中执行以下操作以获得报道

  • 添加apply plugin: 'jacoco'

  • 如下启用测试覆盖率

    android {
        buildTypes {
            debug {
                testCoverageEnabled true
            }
        }
        ...
    }
    
  • 添加以下依赖项

    androidTestImplementation 'com.android.support.test:runner:1.0.2'
    androidTestImplementation 'com.android.support.test:rules:1.0.2'
    

我相信,对于您项目中的任何模块,如果您的 module/library 中有 androidTests,就会有一个名为 combinedTestReportDebug 的默认任务。

您可以在 Android 工作室右上角的垂直栏中的 Gradle 菜单中找到它。

Gradle menu > :your-library > Tasks > reporting > combinedTestReportDebug

来自 运行 这个命令在你的 Terminal > ./gradlew clean combinedTestReportDebug -p :your-library

测试完成后,您可以在以下位置找到覆盖率或失败的测试

对于失败的测试: your-library/build/reports/androidTests/connected/index.html

覆盖率报告: your-library/build/reports/coverage/debug/index.html

希望对您有所帮助。

更新:

抱歉,我忘了提到下面需要的步骤,否则 combinedTestReportDebug 将不可见。感谢@nicoabie。

在项目的 build.gradle 文件中,执行以下操作使 combinedTestReportDebug 任务在库 gradle 任务中可见。

buildscript {
    repositories {
        jcenter()
    }
    dependencies {
        ...
        classpath 'com.vanniktech:gradle-android-junit-jacoco-plugin:0.13.0'
        ...
    }
}
...

apply plugin: 'com.vanniktech.android.junit.jacoco'