Тестируй плагины для Gradle правильно
Как-то при подготовке одного из докладов про разработку плагинов для Gradle встала задача — как свои поделия потестировать. Без тестов вообще жить плохо, а когда твой код реально запускается в отдельном процессе и подавно, потому что хочется дебага, хочется быстрого запуска и не хочется писать миллион example-ов, чтобы протестировать все возможные кейсы. Под катом сравнение нескольких способов тестирования, которые мы успели попробовать.
Подопытный кролик
Нашим подопытным кроликом будет проект, который мы с tolkkv готовили для конференции JPoint 2016. Если кратко, то мы писали плагин, который будет собирать документацию из различных проектов и генерировать обычный html-документ с кросс-референсными ссылками. Но речь не о том, как мы писали сам плагин (хотя это тоже было весело и увлекательно), а как протестировать то, что ты пишешь. Каюсь, но практически весь проект мы тестировали интеграционно, через примеры. И в какой-то момент поняли, что стоит подумать о другом способе тестирования. Итак, наши кандидаты:
Задача везде одна и та же. Просто проверить, что наш плагин для документации подключен, и есть таск, который способен выполниться успешно. Погнали.
Gradle Test Kit
Сейчас находится в стадии инкубации, что было сильно заметно, когда мы пытались его прикрутить. Если взять пример из документации и наивно его применить к нашим реалиям (см. пример ниже), то ничего не заработает. Давайте разбираться, а что мы сделали.
@Slf4j
class TestSpecification extends Specification {
@Rule
final TemporaryFolder testProjectDir = new TemporaryFolder()
def buildFile
def setup() {
buildFile = testProjectDir.newFile('build.gradle')
}
def "execution of documentation distribution task is up to date"() {
given:
buildFile << """
buildscript {
repositories { jcenter() }
dependencies {
classpath 'org.asciidoctor:asciidoctor-gradle-plugin:1.5.3'
}
}
apply plugin: 'org.asciidoctor.convert'
apply plugin: 'ru.jpoint.documentation'
docs {
debug = true
}
dependencies {
asciidoctor 'org.asciidoctor:asciidoctorj:1.5.4'
docs 'org.slf4j:slf4j-api:1.7.2'
}
"""
when:
def result = GradleRunner.create()
.withProjectDir(testProjectDir.root)
.withArguments('documentationDistZip')
.build()
then:
result.task(":documentationDistZip").outcome == TaskOutcome.UP_TO_DATE
}
}
Мы используем Spock, хотя можно использовать и JUnit. Наш проект будет лежать и запускаться во временной папке, которая определяется через testProjectDir
. В методе setup мы создаём новый файл сборки проекта. В given мы определили контент этого файла, подключили к нему необходимые нам плагины и зависимости. В секции when через новый класс GradleRunner
, мы передаём определённую ранее директорию с проектом и говорим, что хотим запустить таск из плагина. В секции then мы проверяем, что таск у нас есть, но так как никаких документов мы не определили, то исполнять его не нужно.
Дак вот, запустив тест, мы узнаем, что тестовый фреймворк не знает что за плагин — ru.jpoint.documentation
— мы подключили. Почему так происходит? Потому что сейчас GradleRunner
не передаёт внутрь себя classpath плагина. А это очень сильно ограничивает нас в тестировании. Идём в документацию и узнаём, что есть метод withPluginClasspath
, в который можно передать нужные нам ресурсы, и они подхвачены в процессе тестирования. Осталось понять — как его сформировать.
Если думаете, что это очевидно, подумайте ещё раз. Чтобы решить проблему, нужно самому через отдельный таск (спасибо Gradle за императивный подход) сформировать текстовый файл с набором ресурсов в build
директории. Пишем:
task createClasspathManifest {
def outputDir = sourceSets.test.output.resourcesDir
inputs.files sourceSets.main.runtimeClasspath
outputs.dir outputDir
doLast {
outputDir.mkdirs()
file("$outputDir/plugin-classpath.txt").text = sourceSets.main.runtimeClasspath.join("\n")
}
}
Запускаем, получаем файлик. Теперь идём в наш тест и в setup добавляем следующий приятный для чтения код:
def pluginClasspathResource = getClass().classLoader.findResource("plugin-classpath.txt")
if (pluginClasspathResource == null) {
throw new IllegalStateException("Did not find plugin classpath resource, run `testClasses` build task.")
}
pluginClasspath = pluginClasspathResource.readLines()
.collect { new File(it) }
Теперь передадим classpath
в GradleRunner
. Запустим, и ничего не работает. Идём на форумы и узнаём, что это работает только с Gradle 2.8+. Проверяем, что у нас 2.12 и грустим. Что делать? Попробуем сделать как советуют делать для Gradle 2.7 и ниже. Мы сами сформируем ещё один classpath
и добавим его напрямую в buildscript
:
def classpathString = pluginClasspath
.collect { it.absolutePath.replace('\\', '\\\\') }
.collect { "'$it'" }
.join(", ")
dependencies {
classpath files($classpathString)
...
}
Запускаем — работает. Это не все проблемы. Можете почитать эпичный трэд и станет совсем грустно.
2.13 update: когда мы экспериментировали, новая версия ещё не вышла. В ней исправили (наконец-то) проблему с подтягиванием ресурсов и теперь код выглядит куда пристойнее и благороднее. Для этого нужно немного по-другому подключить плагин:
plugins {
id 'ru.jpoint.documentation'
}
и запускать GradleRunner
с пустым classpath-ом:
def result = GradleRunner.create()
.withProjectDir(testProjectDir.root)
.withArguments('documentationDistZip')
.withPluginClasspath()
.build()
Осталось лишь огорчение, что из Idea нельзя запустить этот тест через контекстное меню, потому что она не умеет правильно подставлять нужные ресурсы. Через ./gradlew
всё прекрасно работает.
Итог: направление правильно, но использование порой причиняет боль.
Nebula Test
Второй кандидат показал себя куда лучше. Всё что нужно сделать — подключить плагин в свои зависимости:
functionalTestCompile 'com.netflix.nebula:nebula-test:4.0.0'
Затем в спецификации мы можем по аналогии с прошлым примером создать build.gradle
файл:
def setup() {
buildFile << """
buildscript {
repositories { jcenter() }
dependencies {
classpath 'org.asciidoctor:asciidoctor-gradle-plugin:1.5.3'
}
}
apply plugin: 'org.asciidoctor.convert'
apply plugin: info.developerblog.documentation.plugin.DocumentationPlugin
docs {
debug = true
}
dependencies {
asciidoctor 'org.asciidoctor:asciidoctorj:1.5.4'
docs 'org.slf4j:slf4j-api:1.7.2'
}
"""
}
А вот сам тест выглядит легко, понятно, а самое главное — он запускается без приседаний:
def "execution of documentation distribution task is success"() {
when:
createFile("/src/docs/asciidoc/documentation.adoc")
ExecutionResult executionResult = runTasksSuccessfully('documentationDistZip')
then:
executionResult.wasExecuted('documentationDistZip')
executionResult.getSuccess()
}
В этом примере мы ещё и создали файл с документацией, и поэтому результат исполнения нашего таска будет SUCCESS
.
Итог: всё очень здорово. Рекомендуется к использованию.
Unit тестирование
Ок, всё что мы делали ранее это всё-такие интеграционные тесты. Посмотрим, что мы можем сделать через механизм Unit-тестов.
Сначала сконфигурируем проект просто через код:
def setup() {
project = new ProjectBuilder().build()
project.buildscript.repositories {
jcenter()
}
project.buildscript.dependencies {
classpath 'org.asciidoctor:asciidoctor-gradle-plugin:1.5.3'
}
project.plugins.apply('org.asciidoctor.convert')
project.plugins.apply(DocumentationPlugin.class)
project.dependencies {
asciidoctor 'org.asciidoctor:asciidoctorj:1.5.4'
docs 'org.slf4j:slf4j-api:1.7.2'
}
}
Как видно, это практически ничем не отличается от того, что мы писали раньше, только Closure
пишутся несколько длиннее.
Теперь мы можем протестировать, что наш таск из плагина действительно появился в сконфигурированном проекте (и вообще конфигурирование прошло успешно):
def "execution of documentation distribution task is success"() {
when:
project
then:
project.getTasksByName('documentationDistZip', true).size() == 1
}
Но больше этого мы протестировать не можем. То есть через этот способ нам не понять, что таск будет делать то, что ему положено, и, скажем, документ действительно будет сформирован.
Итог: можно использовать для проверки конфигурации проектов. Это быстрее, чем тестирование через реальное выполнение. Но возможности у нас сильно ограничены.
Резюме
Рекомендуется использование Nebula Test
для тестирования плагинов. Если у вас есть развесистая логика при конфигурации проекта, то имеет смысл посмотреть в сторону Unit-тестирования. Ну и ждём допиленный Gradle Test Kit
.
Ссылка на проект с тестами и плагином: https://github.com/aatarasoff/documentation-plugin-demo