Покрытие кода Jacoco в Android Studio с ароматами

Я пытаюсь запустить тестовое покрытие Jacoco на некоторое время. Я пробовал несколько возможных решений, описанных в этих темах:

Оболочка тестового кода Android с плагином JaCoCo Gradle

Как получить отчет о покрытии jacoco с помощью плагина Android Gradle 0.10.0 или новее?

Я запускаю тесты в эмуляционном устройстве с использованием genymotion. Вот что я добавил в build.gradle:

apply plugin: 'jacoco'

android{       
    jacoco {
        version "0.7.1.201405082137"
    }        
    buildTypes{
        debug{
                    testCoverageEnabled = true
        }
    }
}

jacoco {
    toolVersion "0.7.1.201405082137"
}

Чтобы запустить его, я использую что-то вроде

./gradlew clean
./gradlew createFLAVOR_NAMEDebugCoverageReport

Соответствующие сгенерированные файлы/папки:

/build/intermediates/coverage-instrumented-classes
/build/intermediates/jacoco
/build/outputs/code-coverage/connected/flavors/MyFlavor/coverage.ec

Однако нет ничего @build/reports/jacoco/test/html/index.html или любого отчета html-страницы/кода покрытия @/build/output.

Я также попытался создать выделенную задачу для создания отчета о покрытии:

def coverageSourceDirs = [
    'src/main/java',
]

task jacocoTestReport(type: JacocoReport, dependsOn: "connectedAndroidTestFLAVOR_NAMEDebug") {
    group = "Reporting"
    description = "Generate Jacoco coverage reports after running tests."
    reports {
        xml.enabled = true
        html.enabled = true
    }
    classDirectories = fileTree(
        dir: './build/intermediates/classes/debug',
        excludes: ['**/R*.class',
                   '**/*$InjectAdapter.class',
                   '**/*$ModuleAdapter.class',
                   '**/*$ViewInjector*.class'
        ])
    sourceDirectories = files(coverageSourceDirs)
    executionData = files("$buildDir/jacoco/connectedAndroidTestMyFlavorDebug.exec")
    // Bit hacky but fixes https://code.google.com/p/android/issues/detail?id=69174.
    // We iterate through the compiled .class tree and rename $$ to $.
    doFirst {
       new File("$buildDir/intermediates/classes/").eachFileRecurse { file ->
            if (file.name.contains('$$')) {
                file.renameTo(file.path.replace('$$', '$'))
            }
        }
    }
}

Затем ./gradlew clean и ./gradlew jacocoTestReport. Результат такой же, как и выше, поэтому нет html-страницы с отчет о покрытии или любого другого файла покрытия.

В настоящее время я использую Android Studio v1.0.2 с последней версией Gradle. Im довольно новичок в gradle, поэтому возможно, что здесь отсутствует что-то основное.

Спасибо

Ответы

Ответ 1

Проведя целый день, преследуя эту проблему, я узнал, в чем проблема. В отличие от примеров, которые я видел, файл, сгенерированный сборкой testDebug, не является файлом .exec @ $buildDir/jacoco/testDebug.exec.

С моей версией gradle и студийной версией созданный файл представляет собой .ec @ сборка/выходы/кодирование/присоединение/флейворы/myFlavor/coverage.ec

Я не нашел никакой соответствующей информации, связанной с этим. Это может быть недавнее изменение, однако, создавая пользовательскую задачу JacocoReport и изменяя переменную executeData, я решил проблему. Вот моя реализация:

task jacocoTestReport(type: JacocoReport) {

  def coverageSourceDirs = [
        'src/main/java'
  ]

  group = "Reporting"
  description = "Generates Jacoco coverage reports"
  reports {
      xml{
          enabled = true
          destination "${buildDir}/reports/jacoco/jacoco.xml"
      }
      csv.enabled false
      html{
          enabled true
          destination "${buildDir}/jacocoHtml"
      }
  }

  classDirectories = fileTree(
          dir: 'build/intermediates/classes',
          excludes: ['**/R.class',
                     '**/R$*.class',
                     '**/BuildConfig.*',
                     '**/Manifest*.*',
                     '**/*Activity*.*',
                     '**/*Fragment*.*'
          ]
  )

  sourceDirectories = files(coverageSourceDirs)
  additionalSourceDirs = files(coverageSourceDirs)
  executionData = files('build/outputs/code-coverage/connected/flavors/smartcompanion/coverage.ec')
}

Ответ 2

Отчет о тестировании с использованием Jacoco с Android Flavors:

Предположим, что у вас есть ароматы с именем "бесплатно" и "платные"

  • Шаг 1. Создайте файл jacoco.gradle в каталоге модулей проектов (по умолчанию), где build.gradle существует, он должен быть рядом с файлом build.gradle. структура каталога, как показано ниже

    app > jacoco.gradle

  • Шаг 2: вставьте под кодом в файл, который мы создали на шаге 1, код имеет пояснительные комментарии для понимания

apply plugin: 'jacoco'

jacoco {
    toolVersion = "0.7.5.201505241946"
}
project.afterEvaluate {
    // Grab all build types and product flavors
    def buildTypes = android.buildTypes.collect { type ->
        type.name
    }
    def productFlavors = android.productFlavors.collect { flavor ->
        flavor.name
    }
    // When no product flavors defined, use empty
    if (!productFlavors) productFlavors.add('')

    //iterate over the flavors

    productFlavors.each {

        productFlavorName ->
//iterate over build types like debug,release,prod etc.
        buildTypes.each {

            buildTypeName ->
                //sourceName — e.g. freeDebug ,sourcePath — e.g. free/debug
            def sourceName, sourcePath
            if (!productFlavorName) {
                sourceName = sourcePath = "${buildTypeName}"
            } else {
                sourceName = "${productFlavorName}${buildTypeName.capitalize()}"
                sourcePath = "${productFlavorName}/${buildTypeName}"
            }
                // testTaskName —  e.g. testFreeDebugtest task that the coverage task depends on,
            def testTaskName = "test${sourceName.capitalize()}UnitTest"
            // Create coverage task of form 'testFlavorTypeCoverage' depending on 'testFlavorTypeUnitTest'
            task "${testTaskName}Coverage" (type:JacocoReport, dependsOn: "$testTaskName") {
                group = "Reporting"
                description = "Generate Jacoco coverage reports on the ${sourceName.capitalize()} build."
                classDirectories = fileTree(
                        dir: "${project.buildDir}/intermediates/classes/${sourcePath}",
                        excludes: [
                                '**/R.class',
                                '**/R$*.class',
                                '**/*$ViewInjector*.*',
                                '**/*$ViewBinder*.*',
                                '**/BuildConfig.*',
                                '**/Manifest*.*'
                        ]
                )
                def coverageSourceDirs = [
                        "src/main/java",
                        "src/$productFlavorName/java",
                        "src/$buildTypeName/java"
                ]
                additionalSourceDirs = files(coverageSourceDirs)
                sourceDirectories = files(coverageSourceDirs)
                executionData = files("${project.buildDir}/jacoco/${testTaskName}.exec")
                reports {
                    //enables and disable the type of file you need
                    xml.enabled = false
                    html.enabled = true
                }
            }
        }
    }
}