/lovely-gradle-plugin

Gradle plugin used by various Lovely Systems projects for versioning and Docker based release management.

Primary LanguageKotlinApache License 2.0Apache-2.0

Lovely Gradle Plugin

CircleCI Build Status Gradle Plugin

This repository provides a Gradle plugin used by Lovely Systems Projects. It is only tested with the kotlin-dsl for Gradle, Groovy is not supported.

The newest version of this plugin can be found on the Gradle Plugin Portal and can be applied to your project by adding it to the plugins section of your build.gradle.kts file like this:

plugins {
    id("com.lovelysystems.gradle") version ("1.0.0")
}

Git Project

The Git project support automatically sets the project version from the Git state of the project by using git describe internally. It also allows to automatically generate a new version tag and push it based on the changelog.

To enable this functionality add the following to your build.gradle.kts file:

lovely {
  gitProject()
}

Tasks

  • printVersion: Prints out the current version of the project
  • printChangeLogVersion: Parses the changes file and prints out the latest defined version
  • createTag - Creates a new git tag for the current version and pushes the tag to the upstream

The createTag task creates a version tag for the latest version defined in the CHANGES.(rst|md|txt) file in the root of the project. The task validates the state of the project tree and only allows to tag the version if the validation passes. The following conditions need to be met in order for createTag to operate:

  • The latest changelog entry in the changes file needs to have a valid version number (e.g: 0.1.2).
  • There are no uncommitted changes or unknown files in the work tree.
  • Your current branch is the default branch or a release branch (e.g: release/0.1)
  • The same or a newer git tag does not exist.

Docker Project

INFO: Requires Docker >= 20.10.1 with buildx >= v0.5.0

This functionality is enabled by defining the docker repository for the project in the build.gradle.kts file like this::

lovely {
  dockerProject("hub.example.com/lovely/exampleproject")
}

As for any docker repository the hostname of the registry is not required and defaults to docker.io

Optionally a list of stages to build can be supplied to build multiple stages from a single Dockerfile. The default is to just build the default stage, which is actually an empty string, so the above example naming the stage explicitly looks like this:

lovely {
  dockerProject("hub.example.com/lovely/exampleproject", stages=listOf(""))
}

See the example Project to see stage builds in action.

Platforms

By default, docker images are pushed for target platforms linux/amd64 and linux/arm64 (e.g. Apple M1). If not pushed, the image is build and loaded to the container registry for the platform only that runs the build. However, the target platforms can be configured by passing a list of supported platform identifiers. In case there are build platforms set, the local docker builds the image for these platforms. Otherwise the local system architecture is used as build platform.

lovely {
  dockerProject(
    "hub.example.com/lovely/exampleproject", 
    platforms = listOf("linux/amd64", "linux/arm/v7"),
    buildPlatforms = listOf("linux/amd64")
  )
}

The buildDockerImage task use BuildKit in order to build Docker images for different target platforms. In order to run the following build tasks you need to create the docker-container builder on your host.

Note: Run ./gradlew prepareDockerContainerBuilder or ./gradlew pDCB in order to bootstrap the required docker-container builder that supports the enhanced buildx features.

Tasks

  • printDockerTag - Prints out the currently generated Docker tag
  • prepareDockerContainerBuilder - Bootstraps a new docker-container builder
  • prepareDockerImage - Prepares all files required for a Docker build
  • buildDockerImage - Builds a Docker image and tags it with current version and dev
  • pushDockerImage - Pushes the Docker image to the registry
  • pushDockerDevImage - Pushes the Docker image to the registry and tag it as dev

How to release/publish a new development version?

The following steps explain how to release a new development version of your project:

  1. There are no uncommitted changes and everything is pushed to the remote
  2. Run ./gradlew pushDockerDevImage to build a docker image and push it to the registry
  3. Run ./gradle printDockerTag to get the tag of the pushed image

How to release/publish a new production version?

The following steps explain how to release a new stable version of your project:

  1. You are on the default branch or a release branch
  2. The latest changelog entry is updated with a valid (e.g. 1.0.0) version number
  3. There are no uncommitted changes and everything is pushed to the remote
  4. Create a new tag with the ./gradlew createTag task
  5. Run ./gradlew pushDockerImage to build a docker image and push it to the registry

Python Project

The Python project support ads an opinionated project setup for Python projects with the use of pip-tools. To enable it add the following to your build.gradle.kts file:

lovely {
  pythonProject()
}

AWS Project

The AWS project add support for fetching temporary AWS SSO credentials for a configurable profile. In background it uses the official AWS CLI to fetch the credentials, for installation of the CLI read here. The issued commands are described in the AWS documentation.

Before using the plugin you should configure sso on your machine to create a profile that sources temporary AWS credentials from AWS IAM Identity Center:

aws configure sso --profile <my-profile>

When prompted, give the SSO session a meaningful name. If you need to setup other profiles later on then you can reference the existing SSO session config (an SSO session can be referenced by multiple profiles, read the official documentation for further details).

To enable the plugin add the following to your build.gradle.kts file:

lovely {
  awsProject("<my-profile (defaults to 'default')>", "<region (defaults to eu-central-1)>")
}

The tasks also support loading credentials from the environment variables: AWS_ACCESS_KEY_ID, AWS_SECRET_ACCESS_KEY and AWS_SESSION_TOKEN. When credentials can be loaded from these then the profile and the sso session is ignored.

Tasks

  • ssoCredentials - Fetches temporary AWS SSO credentials for the configured profile. If the profile doesn't exist on your machine you will be prompted to create it.

Task types

  • S3UploadDirectory - Uploads a directory to an S3 bucket, e.g.:
    val uploadMyContent by tasks.registering(UploadDirectoryToS3::class) {
        bucket = "<bucket-name>"
        sourceDirectory = File("build/content")
        prefix = "content/$version" // optional (defaults to "")
        overwrite = true // optional (defaults to false)
    }
  • S3DownloadFile - Downloads a single file. It should only be used to download files which aren't changing, e.g. downloading example data, as newer versions from S3 will not be downloaded if one already exists. Usage:
    val downloadMyContent by tasks.registering(S3DownloadFile::class) {
      bucket = "<bucket-name>"
      key = "samples.zip"
      targetFile = File("samples/content.zip")
    }

License

This plugin is made available under the Apache 2.0 License.