add service, service-container, service-process-resource, and test-integration

This commit is contained in:
2025-09-09 14:06:25 +02:00
parent 2e264a7ea9
commit 238e4b0a66
6 changed files with 194 additions and 0 deletions

View File

@@ -13,6 +13,13 @@
* [Plugin Library.](#plugin-library) * [Plugin Library.](#plugin-library)
* [Plugin Library Manifest.](#plugin-library-manifest) * [Plugin Library Manifest.](#plugin-library-manifest)
* [Plugin Library publish.](#plugin-library-publish) * [Plugin Library publish.](#plugin-library-publish)
* [Service](#service)
* [Plugin Service](#plugin-service)
* [Plugin Service Container](#plugin-service-container)
* [Configuration](#configuration)
* [Gradle Tasks](#gradle-tasks)
* [Plugin Service Test Integration](#plugin-service-test-integration)
* [Plugin Service Process Resource](#plugin-service-process-resource)
* [Publish gradle plugin locally.](#publish-gradle-plugin-locally) * [Publish gradle plugin locally.](#publish-gradle-plugin-locally)
* [Releasing gradle plugin.](#releasing-gradle-plugin) * [Releasing gradle plugin.](#releasing-gradle-plugin)
* [Publish gradle plugin to repository.](#publish-gradle-plugin-to-repository) * [Publish gradle plugin to repository.](#publish-gradle-plugin-to-repository)
@@ -73,6 +80,55 @@ id `ltd.lulz.plugin.common-plugin.library-publish`
Configuration for publishing project artifacts to a remote Maven repository. Configuration for publishing project artifacts to a remote Maven repository.
### Service
#### Plugin Service
id `ltd.lulz.plugin.common-plugin.service`
Default setting and tasks for services.
#### Plugin Service Container
id `ltd.lulz.plugin.common-plugin.service-container`
Configuration for running project in docker locally during development.
##### Configuration
* properties `container.network`, environment `CONTAINER_NETWORK`, or default `develop`
* properties `container.port.expose`, environment `CONTAINER_PORT_EXPOSE`, or default `8080`
* properties `container.port.host`, environment `CONTAINER_PORT_HOST`, or default `8080`
* properties `container.profiles`, environment `CONTAINER_PROFILES`, or default `docker`
* properties `docker.port.expose`, environment `DOCKER_PORT_EXPOSE`, or default `8080`
container and docker ports can be a single port (e.g., 8080) or multiple ports separated by commas (e.g., 8080,8443)
##### Gradle Tasks
* `containerCreate` create docker container with network and spring boot profile.
* `containerStart` starts docker container.
* `containerStop` stops docker container.
* `containerNetworkCheck` check if network exist.
* `containerNetworkCreate` creates network.
* `containerNetworkRemove` removes network.
#### Plugin Service Test Integration
id `ltd.lulz.plugin.common-plugin.service-test-integration`
Adding task `testIntegration` to run integration test, add to `verification` group and add to task `check`.
Adding intellij support `src/test-integration/java`, `src/test-integration/kotlin`, and `src/test-integration/resources` as test module in intellij.
Adding dependencies support `testIntegrationImplementation()`, and `testIntegrationRuntimeOnly()` as part of Gradle.
#### Plugin Service Process Resource
id `ltd.lulz.plugin.common-plugin.service-process-resource`
Assign values to `application.yml` from Gradle.
## Publish gradle plugin locally. ## Publish gradle plugin locally.
```shell ```shell

View File

@@ -11,6 +11,7 @@ plugins {
} }
dependencies { dependencies {
implementation(aa.plugin.docker)
implementation(aa.plugin.core) implementation(aa.plugin.core)
implementation(aa.plugin.detekt) implementation(aa.plugin.detekt)
implementation(aa.plugin.ktlint) implementation(aa.plugin.ktlint)

View File

@@ -0,0 +1,66 @@
import com.bmuschko.gradle.docker.tasks.container.DockerCreateContainer
import com.bmuschko.gradle.docker.tasks.container.DockerStartContainer
import com.bmuschko.gradle.docker.tasks.container.DockerStopContainer
import com.bmuschko.gradle.docker.tasks.network.DockerCreateNetwork
import com.bmuschko.gradle.docker.tasks.network.DockerInspectNetwork
import com.bmuschko.gradle.docker.tasks.network.DockerRemoveNetwork
plugins {
id("com.bmuschko.docker-spring-boot-application")
id("ltd.lulz.plugin.core-plugin")
}
fun configurationPorts(
property: String,
environment: String,
): List<String> = config.findOrDefault(property, environment, "8080").split(',')
fun exposeDockerPorts(): List<Int> = configurationPorts("docker.port.expose", "DOCKER_PORT_EXPOSE")
.mapNotNull { it.toIntOrNull() }
fun exposeContainerPorts(): List<String> = configurationPorts("container.port.host", "CONTAINER_PORT_HOST")
.zip(configurationPorts("container.port.expose", "CONTAINER_PORT_EXPOSE"))
.map { (containerPort, exposedPort) -> "$containerPort:$exposedPort" }
docker.springBootApplication {
baseImage.set("eclipse-temurin:17-jre-alpine")
ports.set(exposeDockerPorts())
images.set(listOf("${project.name}:${project.version}"))
}
tasks {
register("containerCreate", DockerCreateContainer::class) {
group = "container"
targetImageId("${project.name}:${project.version}")
containerName.set(project.name)
hostConfig.autoRemove.set(true)
hostConfig.network.set(config.findOrDefault("container.network", "CONTAINER_NETWORK", "develop"))
hostConfig.portBindings.set(exposeContainerPorts())
withEnvVar(
"SPRING_PROFILES_ACTIVE",
config.findOrDefault("container.profiles", "CONTAINER_PROFILES", "docker"),
)
}
register("containerStart", DockerStartContainer::class) {
group = "container"
dependsOn(findByPath("containerCreate"))
targetContainerId(project.name)
}
register("containerStop", DockerStopContainer::class) {
group = "container"
targetContainerId(project.name)
}
register("containerNetworkCheck", DockerInspectNetwork::class) {
group = "container"
targetNetworkId(config.findOrDefault("container.network", "CONTAINER_NETWORK", "develop"))
onError { println("Network does not exist.") }
}
register("containerNetworkCreate", DockerCreateNetwork::class) {
group = "container"
networkName.set(config.findOrDefault("container.network", "CONTAINER_NETWORK", "develop"))
}
register("containerNetworkRemove", DockerRemoveNetwork::class) {
group = "container"
targetNetworkId(config.findOrDefault("container.network", "CONTAINER_NETWORK", "develop"))
}
}

View File

@@ -0,0 +1,27 @@
import org.gradle.kotlin.dsl.support.uppercaseFirstChar
plugins {
id("ltd.lulz.plugin.core-plugin")
}
tasks {
withType<ProcessResources> {
val projectName = project.name
val projectVersion = project.version
filesMatching("**/application.yml") {
filter {
it.replace(
"%APP_NAME%",
projectName.split("-").joinToString(" ") { word -> word.uppercaseFirstChar() },
)
}
filter { it.replace("%APP_VERSION%", projectVersion as String) }
filter { it.replace("%APP_BUILD_TIME%", info.utcTimestamp) }
filter { it.replace("%APP_BUILD_OS_NAME%", System.getProperty("os.name")) }
filter { it.replace("%APP_BUILD_OS_VERSION%", System.getProperty("os.version")) }
filter { it.replace("%APP_BUILD_GIT_COMMIT%", git.currentShortHash()) }
filter { it.replace("%APP_BUILD_GIT_BRANCH%", git.currentBranch()) }
}
onlyIf { file("src/main/resources/application.yml").exists() }
}
}

View File

@@ -0,0 +1,38 @@
plugins {
id("idea")
kotlin("jvm")
}
@Suppress("unused")
fun DependencyHandler.testIntegrationImplementation(
dependencyNotation: Any,
): Dependency? = add("testIntegrationImplementation", dependencyNotation)
@Suppress("unused")
fun DependencyHandler.testIntegrationRuntimeOnly(
dependencyNotation: Any,
): Dependency? = add("testIntegrationRuntimeOnly", dependencyNotation)
sourceSets.create("test-integration") {
compileClasspath += sourceSets["main"].output
runtimeClasspath += sourceSets["main"].output
idea.module {
testSources.from(sourceSets["test-integration"].kotlin.srcDirs, sourceSets["test-integration"].java.srcDirs)
testResources.from(sourceSets["test-integration"].resources.srcDirs)
}
configurations.let {
it["testIntegrationImplementation"].extendsFrom(configurations.implementation.get())
it["testIntegrationRuntimeOnly"].extendsFrom(configurations.runtimeOnly.get())
}
}
tasks {
register<Test>("testIntegration") {
description = "Runs test integration."
group = "verification"
testClassesDirs = sourceSets["test-integration"].output.classesDirs
classpath = sourceSets["test-integration"].runtimeClasspath
}
check { dependsOn(getByName("testIntegration")) }
}

View File

@@ -0,0 +1,6 @@
plugins {
id("ltd.lulz.plugin.common-plugin.common")
id("ltd.lulz.plugin.common-plugin.service-container")
id("ltd.lulz.plugin.common-plugin.service-process-resource")
id("ltd.lulz.plugin.common-plugin.service-test-integration")
}