Spring Cloud Release Tools

Spring Cloud projects reuse the same pattern of building and deploying the applications. That’s why this tool makes it easy to automate the release / dependency update process of our applications.

What does it do?

Single project

For a single project, by default if you opt in to all tasks

  • Clones the BOM project and picks all versions

  • Modifies the project versions with values from a BOM (e.g. for Spring Cloud it’s Spring Cloud Release)

    • throws an exception when we bump versions to release and there’s a SNAPSHOT version referenced in the POM

  • Performs the build and checks if the docs modules have properly created the documentation

    • throws an exception when in the docs module there’s an unresolved tag in any HTML file

  • Commits changed poms (ONLY FOR NON-SNAPSHOT VERSIONS)

  • Creates a tag for the release / milestone (ONLY FOR NON-SNAPSHOT VERSIONS)

  • Runs the deployment of the artifacts

  • Publishes the docs (for Spring Cloud to spring-cloud-static for non-snapshots, to gh-pages for snapshots)

  • Reverts back to snapshots, bumps the version by a patch (1.0.1.RELEASE1.0.2.BUILD-SNAPSHOT) (ONLY FOR RELEASE VERSIONS)

  • Closes the milestone on Github (e.g. v1.0.1.RELEASE) (ONLY FOR NON-SNAPSHOT VERSIONS)

Starting with version that does Sagan integration, you MUST pass the OAuth token, otherwise the application will fail to start

After project release

Meta-release

All the tasks are opt in, so if you do opt in for everything you’ll get:

  • Uses the fixed versions to clone and check out each project (e.g. spring-cloud-sleuth: 2.1.0.RELEASE)

  • From the version analyzes the branch and checks it out. E.g.

    • for spring-cloud-release’s `Finchley.RELEASE version will resolve either Finchley branch or will fallback to master if there’s no Finchley branch.

    • for spring-cloud-sleuth’s `2.1.0.RELEASE version will resolve 2.1.x branch

  • Performs the release tasks per each project

  • Performs the post release tasks at the end of the release

  • Will update and run smoke test samples (for Spring Cloud it will be https://github.com/spring-cloud/spring-cloud-core-tests)

  • Will clone provided test samples and will update all versions to the latest ones

  • Will clone the release train wiki and update it with the latest release versions (for Spring Cloud it will be https://github.com/spring-projects/spring-cloud.wiki.git)

For the meta-releaser to work we assume that the path to the custom configuration file for each project is always config/releaser.yml.
If you want to run some projects in parallel you have to set the releaser.meta-release.release-groups property to state which projects should be released in parallel. Example:
releaser.meta-release.release-groups[0]=projectA,projectB,projectC

If in the list of projects you have projectA,projectB,projectC,projectD,projectE, then projectA,projectB,projectC will be released in parallel and then projectD and projectE sequentially.

How can I extend it?

The project consists of the following main modules

  • releaser-core - with the core logic for doing releases

  • relaser-spring - with the Spring setup of tasks and a flow execution

  • projects

    • where each project has their configuration properties and additional tasks

You can create your own project’s module and

  • if you want to completely rewrite the flow of the release, just set releaser.flow.default-enabled to false and create the whole flow from scratch

  • if you want to modify the current flow, you can add new tasks by just creating a bean of a given type that extends the ReleaserTask and set its order accordingly.

Example of creating a new ReleaseReleaserTask called BuildCustomStuffTask:

package releaser.my_project;

//...

import releaser.internal.Releaser;
import releaser.internal.spring.Arguments;
import releaser.internal.spring.ExecutionResult;
import releaser.internal.tasks.ReleaseReleaserTask;

@Component
public class BuildCustomStuffTask implements ReleaseReleaserTask {

	/**
	 * Order of this task. The higher value, the lower order.
	 */
	public static final int ORDER = 45;

	@Override
	public String name() {
		return "build_custom_stuff";
	}

	@Override
	public String shortName() {
		return "bcf";
	}

	@Override
	public String header() {
		return "BUILDING CUSTOM STUFF";
	}

	@Override
	public String description() {
		return "Builds custom stuff";
	}

	@Override
	public ExecutionResult runTask(Arguments args) {
		// do some custom stuff basing on the arguments
		return ExecutionResult.success();
	}

	@Override
	public int getOrder() {
		return BuildCustomStuffTask.ORDER;
	}

}

Each release or post release task can implement one of the following interfaces

  • ReleaserTask - marker interface for all release tasks

  • ReleaseReleaserTask - if a task is part of the main release process. That means that if it breaks, the whole release process should stop at once.

  • PostReleaseReleaserTask - marker interface for a post release task. If a post release task fails - the build continues but will be unstable.

  • SingleProjectReleaserTask - a release task for a single project.

  • ProjectPostReleaseReleaserTask - a post release task for a single project.

  • DryRunReleaseReleaserTask - a release task that should be executed during dry run mode.

  • TrainPostReleaseReleaserTask - a post release task that should be executed after the whole release train.

  • CompositeReleaserTask - a task that delegates work to other tasks.

In addition, your project can provide the following beans:

  • CustomBomParser - if you need to perform some additional BOM parsing. E.g. Spring Cloud adds spring-boot and spring-cloud-build versions when parsing the BOM project.

  • CustomProjectDocumentationUpdater - if you need to perform some custom logic when updating the project’s documentation.

  • CustomGithubIssues - if you need to perform additional logic when dealing with Github issues.

To run the project you should create your main class preferably under the releaser package and extend the ReleaserCommandLineRunner class.

package releaser;

import releaser.internal.options.Parser;
import releaser.internal.spring.ExecutionResultHandler;
import releaser.internal.spring.SpringReleaser;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.WebApplicationType;
import org.springframework.boot.autoconfigure.SpringBootApplication;

@SpringBootApplication
public class ReleaserApplication extends ReleaserCommandLineRunner {

	public ReleaserApplication(SpringReleaser releaser,
			ExecutionResultHandler executionResultHandler, Parser parser) {
		super(releaser, executionResultHandler, parser);
	}

	public static void main(String[] args) {
		SpringApplication application = new SpringApplication(ReleaserApplication.class);
		application.setWebApplicationType(WebApplicationType.NONE);
		application.run(args);
	}

}

You can also extend the way the projects and tasks are parsed, flows are executed and the result is analyzed together with the printed report. To do that you should implement the following interfaces as beans:

  • SpringReleaser - performs the release, given the provided options

  • FlowRunner - knows how to execute a release and post release flow

  • ExecutionResultHandler - handles the result of the release

What should I do first?

Members of the Spring Cloud Team typically use this tool as follows. They first clone the releaser locally and build the jar manually

$ git clone [email protected]:spring-cloud/spring-cloud-release-tools.git
$ cd spring-cloud-release-tools
$ ./mvnw clean install
You must set the value of the OAuth token. You can do it either via the command line --releaser.git.oauth-token=…​ or put it as an env variable in .bashrc or .zshrc e.g. export RELEASER_GIT_OAUTH_TOKEN=…​

How to run it (interactive mode)

Go to your project (e.g. Spring Cloud Sleuth)

$ git clone [email protected]:spring-cloud/spring-cloud-sleuth.git
$ cd spring-cloud-sleuth
$ # example of running the releaser agains Dalston.SR1 tag with 1.0.0.BUILD-SNAPSHOT version of the releaser
$ java -jar ~/repo/spring-cloud-release-tools/projects/spring-cloud/target/spring-cloud-1.0.0.BUILD-SNAPSHOT.jar --releaser.pom.branch=vDalston.SR1 --spring.config.name=releaser

The application will start running from your working directory. Running this code follows the convention that you have the OAuth token environment variable set. It also assumes that you might have some custom configuration in config/releaser.yml file. This setting is optional - if you don’t have that file, nothing will happen.

It is important that you clone the repository you are going to release using SSH in order for the releaser to be able to push tags and commit changes automatically.

You will see text similar to this one

=== WHAT DO YOU WANT TO DO? ===

0) Perform a full release of this project without interruptions
1) Perform a full release of this project in a verbose mode (you'll be asked about skipping steps)
2) Update poms with versions from Spring Cloud Release
3) Build the project
4) Commit, tag and push the tag
5) Deploy the artifacts
6) Publish the docs
7) Go back to snapshots and bump originalVersion by patch
8) Push the commits
9) Close the milestone at Github
10) Create email / blog / tweet etc. templates

You can pick a range of options by using the hyphen - e.g. '2-4' will execute jobs [2,3,4]
You can execute all tasks starting from a job by using a hyphen and providing only one number - e.g. '8-' will execute jobs [8,9,10]
You can execute given tasks by providing a comma separated list of tasks - e.g. '3,7,8' will execute jobs [3,7,8]

You can press 'q' to quit

Just pick a number and continue! Pick either a full release or single steps. You can also pick ranges or multiple steps. You can also provide the range only with the starting step - that you will execute all steps starting from the given one.

Read before picking a number cause it might have changed between tool releases ;)

How to run it (automatic mode)

Go to your project (e.g. Spring Cloud Sleuth) and execute the application with -h or --help flag.

$ git clone [email protected]:spring-cloud/spring-cloud-sleuth.git
$ cd spring-cloud-sleuth
$ # example of running the releaser agains Dalston.SR1 tag with 1.0.0.BUILD-SNAPSHOT version of the releaser
$ java -jar ~/repo/spring-cloud-release-tools/releaser-spring/target/releaser-spring-1.0.0.BUILD-SNAPSHOT.jar --releaser.pom.branch=vDalston.SR1 --spring.config.name=releaser -h

You will see a help screen looking like more or less like this

Here you can find the list of tasks in order

[release,releaseVerbose,metaRelease,postRelease,updatePoms,build,commit,deploy,docs,snapshots,push,closeMilestone,updateSagan,createTemplates,updateGuides,updateDocumentation]


Option                                Description
------                                -----------
-a, --start-from <String>             Starts all release task starting from the
                                        given task. Requires passing the task
                                        name (either one letter or the full
                                        name)
-b, --build [String]                  Build the project
-c, --commit [String]                 Commit, tag and push the tag
-d, --deploy [String]                 Deploy the artifacts
-f, --full-release [Boolean]          Do you want to do the full release of a
                                        single project? (default: false)
-g, --updateSagan [String]            Updating Sagan with release info
-h, --help [String]
-i, --interactive <Boolean>           Do you want to set the properties from
                                        the command line of a single project?
                                        (default: true)
-m, --closeMilestone [String]         Close the milestone at Github
-o, --docs [String]                   Publish the docs
-p, --push [String]                   Push the commits
-r, --range <String>                  Runs release tasks from the given range.
                                        Requires passing the task names with a
                                        hyphen. The first task is inclusive,
                                        the second inclusive. E.g. 's-m' would
                                        mean running 'snapshot', 'push' and
                                        'milestone' tasks
-s, --snapshots [String]              Go back to snapshots and bump
                                        originalVersion by patch
-t, --createTemplates [String]        Create email / blog / tweet etc. templates
--task-names, --tn <String>           Starts all release task for the given
                                        task names
-u, --updatePoms [String]             Update poms with versions from Spring
                                        Cloud Release
--ud, --updateDocumentation [String]  Updating documentation repository
--ug, --updateGuides [String]         Updating Spring Guides
-x, --meta-release <Boolean>          Do you want to do the meta release?
                                        (default: false)

Examples of usage:

Run 'build' & 'commit' & 'deploy'
java -jar jar.jar -b -c -d

Start from 'push'
java -jar releaser.jar -a push

Range 'docs' -> 'push'
java -jar releaser.jar -r o-p

The Releaser can use two sets of options. The configuration options like releaser.pom.branch and the task switches. For the tasks you can use either the full names or short switches. For example providing range of tasks via switches o-p is equivalent to full name docs-push.

A couple of examples:

Doing the full release in interactive mode (asking for skipping steps)
$ git clone [email protected]:spring-cloud/spring-cloud-sleuth.git
$ cd spring-cloud-sleuth
$ # example of running the releaser agains Dalston.SR1 tag with 1.0.0.BUILD-SNAPSHOT version of the releaser
$ java -jar ~/repo/spring-cloud-release-tools/releaser-spring/target/releaser-spring-1.0.0.BUILD-SNAPSHOT.jar --releaser.pom.branch=vDalston.SR1 --spring.config.name=releaser --full-release
Doing the full release in non interactive mode (automatic release)
$ java -jar ~/repo/spring-cloud-release-tools/releaser-spring/target/releaser-spring-1.0.0.BUILD-SNAPSHOT.jar --releaser.pom.branch=vDalston.SR1 --spring.config.name=releaser --full-release --interactive=false
Updating pom, closing milestone & createTemplates in interactive mode
$ java -jar ~/repo/spring-cloud-release-tools/releaser-spring/target/releaser-spring-1.0.0.BUILD-SNAPSHOT.jar --releaser.pom.branch=vDalston.SR1 --spring.config.name=releaser -u -m -t
Running all tasks starting from 'push' (automatic)
$ java -jar ~/repo/spring-cloud-release-tools/releaser-spring/target/releaser-spring-1.0.0.BUILD-SNAPSHOT.jar --releaser.pom.branch=vDalston.SR1 --spring.config.name=releaser -a push -i=false
Running tasks from 'docs' (inclusive) to 'push' (inclusive) (automatic)
$ java -jar ~/repo/spring-cloud-release-tools/releaser-spring/target/releaser-spring-1.0.0.BUILD-SNAPSHOT.jar --releaser.pom.branch=vDalston.SR1 --spring.config.name=releaser -r d-p -i=false
Running single task 'closeMilestone' (automatic)
$ java -jar ~/repo/spring-cloud-release-tools/releaser-spring/target/releaser-spring-1.0.0.BUILD-SNAPSHOT.jar --releaser.pom.branch=vDalston.SR1 --spring.config.name=releaser --closeMilestone -i=false

How to run meta-release (automatic-mode)

All you have to do is run the jar with the releaser and pass the -x=true option to turn on meta-release and a list of fixed versions in the `--"releaser.fixed-versions[project-name]=project-version" format

$ java -jar projects/spring-cloud/target/spring-cloud-1.0.0.BUILD-SNAPSHOT.jar --spring.config.name=releaser -x=true --"releaser.fixed-versions[spring-cloud-sleuth]=2.0.1.BUILD-SNAPSHOT"
For the meta release the startFrom or taskNames take into consideration the project names, not task names. E.g. you can start from spring-cloud-netflix project, or build only tasks with names spring-cloud-build,spring-cloud-sleuth.

Project options

Below you can find a table with all the releaser options.

Name Default Description

releaser.bash.build-command

echo "{{systemProps}}"

Command to be executed to build the project. If present "{{version}}" will be replaced by the provided version. "{{nextVersion}}" with the bumped snapshot version and "{{oldVersion}}" with the version before version updating.

releaser.bash.deploy-command

echo "{{systemProps}}"

Command to be executed to deploy a built project. If present "{{version}}" will be replaced by the provided version. "{{nextVersion}}" with the bumped snapshot version and "{{oldVersion}}" with the version before version updating.

releaser.bash.deploy-guides-command

echo "{{systemProps}}"

Command to be executed to build and deploy guides project only. If present "{{version}}" will be replaced by the provided version. "{{nextVersion}}" with the bumped snapshot version and "{{oldVersion}}" with the version before version updating.

releaser.bash.generate-release-train-docs-command

echo "{{version}}"

Command to be executed to generate release train documentation.

releaser.bash.publish-docs-commands

[mkdir -p target, echo "{{version}}"]

Command to be executed to publish documentation. If present "{{version}}" will be replaced by the provided version. "{{nextVersion}}" with the bumped snapshot version and "{{oldVersion}}" with the version before version updating.

releaser.bash.system-properties

Additional system properties that should be passed to the build / deploy commands. If present in other commands "{{systemProps}}" will be substituted with this property.

releaser.bash.wait-time-in-minutes

20

Max wait time in minutes for the process to finish.

releaser.fixed-versions

Project name to its version - overrides all versions retrieved from a release train repository like Spring Cloud Release.

releaser.flow.default-enabled

true

Should the default flow of jobs be preserved. If set to {@code false} will not register any jobs as beans, and it will be up to you to set the whole configuration of jobs.

releaser.git.all-test-sample-urls

Project to urls mapping. For each project will clone the test project and will update its versions.

releaser.git.clone-destination-dir

Where should the release train repo get cloned to. If {@code null} defaults to a temporary directory.

releaser.git.documentation-branch

Branch to check out for the documentation project.

releaser.git.documentation-url

URL to the documentation Git repository.

releaser.git.fetch-versions-from-git

true

If {@code true} then should fill the map of versions from Git. If {@code false} then picks fixed versions.

releaser.git.number-of-checked-milestones

50

In order not to iterate endlessly over milestones we introduce a threshold of milestones that we will go through to find the matching milestone.

releaser.git.oauth-token

GitHub OAuth token to be used to interact with GitHub repo.

releaser.git.password

Optional Git password. If not passed keys will be used for authentication.

releaser.git.release-train-bom-url

URL to a release train repository.

releaser.git.release-train-docs-branch

Branch to check out for the release train docs.

releaser.git.release-train-docs-url

URL to the release train documentation.

releaser.git.release-train-wiki-page-prefix

Page prefix for the release train wiki. E.g. for [Spring-Cloud-Finchley-Release-Notes] it would be [Spring-Cloud].

releaser.git.release-train-wiki-url

URL to the release train wiki.

releaser.git.run-updated-samples

false

If set to {@code false}, will not update the test samples.

releaser.git.spring-project-branch

Branch to check out for the release train project.

releaser.git.spring-project-url

URL to the release train project page repository.

releaser.git.test-samples-branch

Branch to check out for the test samples.

releaser.git.test-samples-project-url

URL to test samples.

releaser.git.update-all-test-samples

false

If set to {@code false}, will not clone and update the samples for all projects.

releaser.git.update-documentation-repo

false

If {@code false}, will not update the documentation repository.

releaser.git.update-github-milestones

false

If set to {@code false}, will not update Github milestones.

releaser.git.update-release-train-docs

false

If set to {@code false}, will not update the release train docs.

releaser.git.update-release-train-wiki

false

If set to {@code false}, will not clone and update the release train wiki.

releaser.git.update-spring-guides

false

If set to {@code false}, will not update Spring Guides for a release train.

releaser.git.update-spring-project

false

If set to {@code false}, will not update the Spring Project for a release train. E.g. for Spring Cloud will not update https://cloud.spring.io .

releaser.git.update-start-spring-io

false

If set to {@code false}, will not update start.spring.io for a release train.

releaser.git.username

Optional Git username. If not passed keys will be used for authentication.

releaser.gradle.build-command

./gradlew clean build publishToMavenLocal --console=plain -PnextVersion={{nextVersion}} -PoldVersion={{oldVersion}} -PcurrentVersion={{version}} {{systemProps}}

Command to be executed to build the project If present "{{version}}" will be replaced by the provided version. "{{nextVersion}}" with the bumped snapshot version and "{{oldVersion}}" with the version before version updating.

releaser.gradle.deploy-command

./gradlew publish --console=plain -PnextVersion={{nextVersion}} -PoldVersion={{oldVersion}} -PcurrentVersion={{version}} {{systemProps}}

Command to be executed to deploy a built project.

releaser.gradle.deploy-guides-command

./gradlew clean build deployGuides --console=plain -PnextVersion={{nextVersion}} -PoldVersion={{oldVersion}} -PcurrentVersion={{version}} {{systemProps}}

Command to be executed to build and deploy guides project only.

releaser.gradle.generate-release-train-docs-command

./gradlew generateReleaseTrainDocs --console=plain -PnextVersion={{nextVersion}} -PoldVersion={{oldVersion}} -PcurrentVersion={{version}} {{systemProps}}

Command to be executed to generate release train documentation.

releaser.gradle.gradle-props-substitution

A mapping that should be applied to {@code gradle.properties} in order to perform a substitution of properties. The mapping is from a property inside {@code gradle.properties} to the projects name. Example. In {@code gradle.properties} you have {@code verifierVersion=1.0.0} . You want this property to get updated with the value of {@code spring-cloud-contract} version. Then it’s enough to do the mapping like this for this Releaser’s property: {@code verifierVersion=spring-cloud-contract}.

releaser.gradle.ignored-gradle-regex

List of regular expressions of ignored gradle props. Defaults to test projects and samples.

releaser.gradle.publish-docs-commands

[./gradlew publishDocs --console=plain -PnextVersion={{nextVersion}} -PoldVersion={{oldVersion}} -PcurrentVersion={{version}} {{systemProps}}]

Command to be executed to publish documentation. If present "{{version}}" will be replaced by the provided version.

releaser.gradle.system-properties

Additional system properties that should be passed to the build / deploy commands. If present in other commands "{{systemProps}}" will be substituted with this property.

releaser.gradle.wait-time-in-minutes

20

Max wait time in minutes for the process to finish.

releaser.maven.build-command

./mvnw clean install -B -Pdocs {{systemProps}}

Command to be executed to build the project. If present "{{version}}" will be replaced by the provided version. "{{nextVersion}}" with the bumped snapshot version and "{{oldVersion}}" with the version before version updating.

releaser.maven.deploy-command

./mvnw deploy -DskipTests -B -Pfast,deploy {{systemProps}}

Command to be executed to deploy a built project. If present "{{version}}" will be replaced by the provided version. "{{nextVersion}}" with the bumped snapshot version and "{{oldVersion}}" with the version before version updating.

releaser.maven.deploy-guides-command

./mvnw clean verify deploy -B -Pguides,integration -pl guides {{systemProps}}

Command to be executed to build and deploy guides project only. If present "{{version}}" will be replaced by the provided version. "{{nextVersion}}" with the bumped snapshot version and "{{oldVersion}}" with the version before version updating.

releaser.maven.generate-release-train-docs-command

bash release_train.sh --retrieveversions --version {{version}} --ghpages --auto

Command to be executed to generate release train documentation.

releaser.maven.publish-docs-commands

[mkdir -p target, wget https://raw.githubusercontent.com/spring-cloud/spring-cloud-build/master/docs/src/main/asciidoc/ghpages.sh -O target/gh-pages.sh, chmod +x target/gh-pages.sh, ./target/gh-pages.sh -v {{version}} -c]

Command to be executed to publish documentation. If present "{{version}}" will be replaced by the provided version.

releaser.maven.system-properties

Additional system properties that should be passed to the build / deploy commands. If present in other commands "{{systemProps}}" will be substituted with this property.

releaser.maven.wait-time-in-minutes

20

Max wait time in minutes for the process to finish.

releaser.meta-release.enabled

false

Are we releasing the whole suite of apps or only one?

releaser.meta-release.git-org-url

The URL of the Git organization. We’ll append each project’s name to it.

releaser.meta-release.projects-to-skip

Names of projects to skip deployment for meta-release.

releaser.meta-release.release-group-thread-count

4

Number of threads per release group. E.g. for thread count of 4 if there are 6 projects in a release group, 4 of them will be executed in parallel and 2 will wait for their turn.

releaser.meta-release.release-group-timeout-in-minutes

180

Timeout in minutes during which we’re waiting for a single composite task per a project to be executed. That means that if set to e.g. 180 then a release process for a single project should take at most 180 minutes.

releaser.meta-release.release-groups

If provided, allows to provide groups of projects that can be ran in parallel. E.g. {@code --releaser.meta-release.release-groups[0]=projectA,projectB,projectC} {@code --releaser.meta-release.release-groups[1]=projectD,projectE} {@code --releaser.meta-release.release-groups[2]=projectF,projectG} The order is still provided by the list of versions passed to the releaser. Basing on that order, and this value we are able to build a flow with projects.

releaser.meta-release.release-train-dependency-names

All the names of dependencies that should be updated with the release train project version.

releaser.meta-release.release-train-project-name

Name of the release train project.

releaser.pom.bom-version-pattern

The pattern to match a version property in a BOM. Remember to catch the dependency name in a group. E.g. "^(spring-cloud-.*)\\.version$".

releaser.pom.branch

master

Which branch of release train BOM should be checked out. Defaults to {@code master}.

releaser.pom.ignored-pom-regex

^.\.git/.$

List of regular expressions of ignored poms. Defaults to test projects and samples.

releaser.pom.pom-with-boot-starter-parent

Subfolder of the pom that contains the {@code spring-boot-starer-parent} dependency.

releaser.pom.this-train-bom

Subfolder of the pom that contains the versions for the release train.

releaser.post-release-tasks-only

false

If set to {@code true} will run only post release tasks.

releaser.sagan.base-url

https://spring.io

URL to the Sagan API.

releaser.sagan.boot-section-file-name

sagan-boot.adoc

Name of the ascii doc file with boot part of this project’s Sagan project page. Linked with {@link this#docsAdocsFile}.

releaser.sagan.docs-adocs-file

docs/src/main/asciidoc

Folder with asciidoctor files for docs.

releaser.sagan.index-section-file-name

sagan-index.adoc

Name of the ascii doc file with core part of this project’s Sagan project page. Linked with {@link this#docsAdocsFile}.

releaser.sagan.update-sagan

false

If set to {@code false} will not update Sagan.

releaser.skip-post-release-tasks

false

If set to {@code true} will not run post release tasks.

releaser.template.enabled

false

Should template generation be enabled.

releaser.template.template-folder

Folder in which blog, email etc. templates are stored.

releaser.versions.all-versions-file-url

https://raw.githubusercontent.com/spring-io/start.spring.io/master/start-site/src/main/resources/application.yml

Url to a file containing all the versions. Defaults to YAML from start.spring.io.

releaser.versions.bom-name

Name in the YAML from initilizr for BOM mappings.

releaser.working-dir

By default Releaser assumes running the program from the current working directory. If you want to change this behaviour - just change this value.

You can pass the options either via system properties or via application arguments. Example for system properties: java -Dreleaser.pom.branch=Camden.SR6 -jar target/releaser-spring-1.0.0.M1.jar Example for application arguments: java -jar target/releaser-spring-1.0.0.M1.jar --releaser.pom.branch=Camden.SR6
For the GA release to be successful, it’s important that if the build / deploy command run a script (e.g. scripts/foo.sh) then inside foo.sh if you call a Maven build ./mvnw clean install then remember to pass all arguments of the script there too. E.g. ./mvnw clean install ${@}. That’s because the releaser will pass any system properties to the build / deploy command, such as system properties with keys and we need them to be passed inside the command executed by the releaser.

Examples

Keeping configuration in the project

If your project has some custom configuration (e.g. Spring Cloud Contract needs a script to be executed to build the project and properly merge the docs) then you can put a file named e.g. releaser.yml under config folder and run your application like this:

$ wget https://repo.spring.io/libs-milestone/org/springframework/cloud/internal/releaser-spring/1.0.0.M1/releaser-spring-1.0.0.M1.jar -O ../releaser-spring-1.0.0.M1.jar
$ java -jar target/releaser-spring-1.0.0.M1.jar --spring.config.name=releaser
Notice that we’re downloading the jar to a parent folder, not to target. That’s because target get cleaned during the build process
For the meta-releaser to work we assume that the path to the configuration file is always config/releaser.yml.

Specifying A Branch

By deafult the releaser will default to using the master branch of spring-cloud-release. If you would like to use another branch you can specify it using the releaser.pom.branch property.

$ java -jar releaser-spring-1.0.0.M1.jar --releaser.pom.branch=Camden.SR6

Using Environment Variables

In some cases it might be easier to specify environment variables instead of passing parameters to releaser. For example, you might want to use environment variables if you are going to be releasing multiple projects, this keeps you from having to specify the same parameters for each release

$ export RELEASER_POM_BRANCH=Dalston.RELEASE
$ export RELEASER_GIT_OAUTH_TOKEN=...
$ wget https://repo.spring.io/libs-milestone/org/springframework/cloud/internal/releaser-spring/1.0.0.M1/releaser-spring-1.0.0.M1.jar -O releaser-spring-1.0.0.M1.jar
$ java -jar target/releaser-spring-1.0.0.M1.jar --releaser.working-dir=/path/to/project/root

Releasing through Jenkins

Whenever a release process is broken, Jenkins marks it with a red ball and breaks the build. Whenever a post-release action went wrong but the release is successful, Jenkins marks the build with a yellow ball and marks the build as unstable.

Releasing a Single Project

Let us assume that we are to release spring-cloud-build project. We need to do the following steps:

  1. Create a branch (for example, springCloudBuildRelease) in a project that contains a BOM (for example, spring-cloud-release). The following example shows how to do so:

$ git clone [email protected]:spring-cloud/spring-cloud-release.git
$ cd spring-cloud-release
$ git checkout -b springCloudBuildRelease
  1. Update all versions as if you were doing a release train. We need to update the project’s versions, Boot version, and dependencies versions, too. Let us assume that we will eventually be doing a release train for the Hoxton.M1 release, Spring Boot to the latest available one, and spring-cloud-commons to 1.2.3.BUILD-SNAPSHOT. The following example shows how to do so:

// setting the release train value
$ ./mvnw versions:set -DnewVersion=Hoxton.M1 -DgenerateBackupPoms=false -DprocessAllModules=true
// Update all parent versions
$ ./mvnw versions:update-parent -DgenerateBackupPoms=false -DprocessAllModules=true
// Setting the necessary dependencies
$ ./mvnw versions:set-property -Dproperty=spring-cloud-commons.version -DnewVersion=1.2.3.BUILD-SNAPSHOT -pl spring-cloud-dependencies -DgenerateBackupPoms=false
// NOTE!!!
// Verify that the versions set by Maven are correct!!
$ git diff
// commit and push the branch
$ git add . && git commit -m "Updating project for Spring Cloud Build release" && git push origin springCloudBuildRelease
If you’re doing a e.g. M1 release, remember to not have any snapshot versions in this branch.

Since the project is prepared, go to Jenkins and select the Releaser view, which the following image shows:

Releaser view
  1. Pick the proper releaser project (for example, spring-cloud-build-releaser). The following image shows the settings for this example:

Spring Cloud Build Releaser - build with parameters
  1. Next, click Build with parameters. The following image shows the UI for doing so:

Updated `RELEASER_POM_BRANCH`

Pick from which branch you would like the project (for example, spring-cloud-build - defaults to master) to be built and update the RELEASER_POM_BRANCH to point to the checked-out branch of Spring Cloud Release (for example, springCloudBuildRelease). You can pick whether you want to perform only post-release tasks or the whole release.

  1. Finally, click Build.

You are done!

As a post action, do not forget to remove the branch. The following example shows how to do so:

// to synchronize any deleted branches (don't run this if you want leave any deleted branches that were deleted in the origin)
$ git fetch -p
$ git branch -d springCloudBuildRelease
$ git push origin --delete springCloudBuildRelease

Releasing a Release Train

We call a release train a meta-release. In order to perform one, you need to:

  1. In your project (which must contain a BOM, such as spring-cloud-release) you have to have a branch, where you store properties with versions of your projects. For example, the branch name can be jenkins-releaser-config). The folloiwng example shows how to do so:

$ git clone [email protected]:spring-cloud/spring-cloud-release.git
$ cd spring-cloud-release
$ git checkout jenkins-releaser-config
  1. Create a file that contains all properties for a given release train. The name of the release train should be lowercase, and dots should be converted to underscores. For example, for the Greenwich.SR2 release train we need to have a file named greenwich_sr2.properties. The following example shows how to do so:

$ touch greenwich_sr2.properties
  1. We need to update the file with all versions for the release train. The properties file contains an ordered list of releaser.fixed-versions[project-name]=project-version entries, as the following listing shows:

$ echo "releaser.fixed-versions[spring-boot]=2.1.5.RELEASE
releaser.fixed-versions[spring-cloud-build]=2.1.5.RELEASE
releaser.fixed-versions[spring-cloud-commons]=2.1.2.RELEASE
releaser.fixed-versions[spring-cloud-function]=2.0.2.RELEASE
releaser.fixed-versions[spring-cloud-stream]=Fishtown.SR3
releaser.fixed-versions[spring-cloud-aws]=2.1.2.RELEASE
releaser.fixed-versions[spring-cloud-bus]=2.1.2.RELEASE
releaser.fixed-versions[spring-cloud-task]=2.1.2.RELEASE
releaser.fixed-versions[spring-cloud-config]=2.1.3.RELEASE
releaser.fixed-versions[spring-cloud-netflix]=2.1.2.RELEASE
releaser.fixed-versions[spring-cloud-cloudfoundry]=2.1.2.RELEASE
releaser.fixed-versions[spring-cloud-kubernetes]=1.0.2.RELEASE
releaser.fixed-versions[spring-cloud-openfeign]=2.1.2.RELEASE
releaser.fixed-versions[spring-cloud-consul]=2.1.2.RELEASE
releaser.fixed-versions[spring-cloud-gateway]=2.1.2.RELEASE
releaser.fixed-versions[spring-cloud-security]=2.1.3.RELEASE
releaser.fixed-versions[spring-cloud-sleuth]=2.1.2.RELEASE
releaser.fixed-versions[spring-cloud-zookeeper]=2.1.2.RELEASE
releaser.fixed-versions[spring-cloud-contract]=2.1.2.RELEASE
releaser.fixed-versions[spring-cloud-gcp]=1.1.2.RELEASE
releaser.fixed-versions[spring-cloud-vault]=2.1.2.RELEASE
releaser.fixed-versions[spring-cloud-release]=Greenwich.SR2" >> greenwich_sr2.properties
$ git add greenwich_sr2.properties && git commit -m "Added Greenwich.SR2 properties" && git push origin jenkins-releaser-config

Since the project is prepared, go to Jenkins and select the Releaser view, as the following image shows:

Releaser view
  1. Pick the proper meta-releaser project (for example, spring-cloud-meta-releaser), as the following image shows:

Spring Cloud Meta Releaser - build with parameters
  1. Next, click Build with parameters.

Spring Cloud Meta Releaser view

You have quite a few options to pick, but the most important one is to set the value of the RELEASE_VERSION to the given release train version (for example, Greenwich.SR2). Continue updating the rest of the fields if necessary and read the field descriptions and this documentation for more information.

  1. Finally, click Build.

You are done!

FAQ

JSchException: Auth fail

You may get an exception similar to the following:

Caused by: org.eclipse.jgit.errors.TransportException: [email protected]:spring-cloud/spring-cloud-sleuth.git: Auth fail
	at org.eclipse.jgit.transport.JschConfigSessionFactory.getSession(JschConfigSessionFactory.java:160) ~[org.eclipse.jgit-4.6.0.201612231935-r.jar!/:4.6.0.201612231935-r]
	at org.eclipse.jgit.transport.SshTransport.getSession(SshTransport.java:137) ~[org.eclipse.jgit-4.6.0.201612231935-r.jar!/:4.6.0.201612231935-r]
	at org.eclipse.jgit.transport.TransportGitSsh$SshPushConnection.<init>(TransportGitSsh.java:322) ~[org.eclipse.jgit-4.6.0.201612231935-r.jar!/:4.6.0.201612231935-r]
	at org.eclipse.jgit.transport.TransportGitSsh.openPush(TransportGitSsh.java:167) ~[org.eclipse.jgit-4.6.0.201612231935-r.jar!/:4.6.0.201612231935-r]
	at org.eclipse.jgit.transport.PushProcess.execute(PushProcess.java:160) ~[org.eclipse.jgit-4.6.0.201612231935-r.jar!/:4.6.0.201612231935-r]
	at org.eclipse.jgit.transport.Transport.push(Transport.java:1275) ~[org.eclipse.jgit-4.6.0.201612231935-r.jar!/:4.6.0.201612231935-r]
	at org.eclipse.jgit.api.PushCommand.call(PushCommand.java:161) ~[org.eclipse.jgit-4.6.0.201612231935-r.jar!/:4.6.0.201612231935-r]
	... 25 common frames omitted
Caused by: com.jcraft.jsch.JSchException: Auth fail
	at com.jcraft.jsch.Session.connect(Session.java:512) ~[jsch-0.1.53.jar!/:na]
	at org.eclipse.jgit.transport.JschConfigSessionFactory.getSession(JschConfigSessionFactory.java:117) ~[org.eclipse.jgit-4.6.0.201612231935-r.jar!/:4.6.0.201612231935-r]
	... 31 common frames omitted

To fix that just call, run the following commands before running the app:

# to run the agent
$ eval `ssh-agent`
# to store the pass in the agent
$ ssh-add ~/.ssh/id_rsa