Динамическое подключение внешних собственных модулей в Gradle

в 6:52, , рубрики: dependency injection, gradle, java, Программирование

Преамбула

Есть своя «внешняя» библиотека и есть своё приложение, использующее эту библиотеку (подгружается через внешний репозитарий). Требуется внести изменение и в библиотеку и в приложение.

Казалось бы, собери библиотеку и выложи её в локальный maven-репозитарий, а потом уже собирай приложение. Но хочется, чтобы можно было поправив код в библиотеке сразу попробовать изменения в приложении и при этом сохранить раздельное хранение кода библиотеки и приложения, включая настройки IDE и прочее.

С помощью gradle и и символических связей в файловой системе такое можно легко устроить.

Библиотека

Для начала приведу пример содержимого build.gradle в библиотеке:

import java.text.SimpleDateFormat

apply plugin: 'java'
apply plugin: 'maven-publish'

sourceCompatibility = JavaVersion.VERSION_1_8
targetCompatibility = JavaVersion.VERSION_1_8

jar.baseName = 'library'

publishing {
    publications {
        mavenJava(MavenPublication) {
            groupId='name.alenkov.habr.gradle-dynamic-dependency'
            
            version = new SimpleDateFormat('yyyyMMddHHmm').format(new Date())

            from components.java
        }
    }
}

Здесь ключевым является строка «version = new SimpleDateFormat('yyyyMMddHHmm').format(new Date())», которая задаёт версию сборки на основе текущего времени в момент её публикации в репозитарий — в остальное время версия для библиотеки нам не требуется.

Disclaimer: В нашем демонстрационном приложении мы не поддерживаем более одной ветки библиотек в продуктовой среде и потому нет потребности в поддержке версионности вида X.Y.Z

Note: в примерах я использую локальный Maven и не привожу примеры с использованием Artifactory, т.к. это не влияет на подход.

Приложение

Теперь перейдём к настройке нашего build.gradle в приложении.

Исходное состояние build.gradle:

apply plugin: 'java'

sourceCompatibility = JavaVersion.VERSION_1_8
targetCompatibility = JavaVersion.VERSION_1_8

repositories {
    mavenLocal()
    jcenter()
}

dependencies {
    compile 'name.alenkov.habr.gradle-dynamic-dependency:library:+'
}

Динамическое подключение внешних собственных модулей в Gradle - 1

Теперь давайте модифицируем конфигурацию gradle так, чтобы получить желаемое — динамическое подключение модулей.

Первый шагом, слинкуем нашу библиотеку в проект приложения в подкаталог ext:

cd ./app/ext
ln -s ../../library/ ./

Вторым шагом добавим небольшой код в settings.gradle, который сканирует каталог "/ext" на наличие gradle-проектов и будет подключать их к нам в проект:

final extDir = new File(rootDir, 'ext')
if (extDir.exists()) {
    extDir.eachDir { dir ->
        if (new File(dir, 'build.gradle').exists()) {
            logger.trace('found ext module: ' + dir.name)

            final String prjName = ':' + dir.name
            logger.lifecycle('include ext project: ' + prjName)
            include prjName
            project(prjName).projectDir = dir
            project(prjName).name = 'ext-' + dir.name
        }
    }
}

И третий, заключительный штрих — модифицируем секцию dependencies в build.gradle:

dependencies {
    compile subprojects.find({ it.name == 'ext-library' }) ? project(':ext-library')
            : 'name.alenkov.habr.gradle-dynamic-dependency:library:+'
}

Динамическое подключение внешних собственных модулей в Gradle - 2

и пример, когда в library есть дополнительные gradle-задачи:
Динамическое подключение внешних собственных модулей в Gradle - 3

Из текущего неудобства — в IDEA динамический модуль подключается не совсем корректно — не подключаются source-каталоги как надо. Завёл баг. Как временное решение — открываю оба проекта в IDEA и в одном окне вношу правки (когда автоподсказки надо), а в другом запускаю

Исходный код обоих модулей можно посмотреть на github

UPD: аналогичным способом можно подключать и плагины gradle, только монтируя их в buildSrc. Если будут желающие, могу написать пример…

Автор: Виктор Аленьков

Источник

Поделиться

* - обязательные к заполнению поля