Maven & Git: Releases and Hotfixes

When you write an application you want a simple way to release a new major version, but also to release bugfixes (here i call them hotfixes) to a released version. If you write Java applications, chances are, you are using Maven and then as a source control you might be using GIT. Maven offers a nice way to set versions, and Git makes it easy to create releases and tags.

I personally use the git branch model described by Vincent Griessen. There development is done on the develop branch, and when a release is done, then a release branch is created. This branch is called release/x.x.x. There the version is updated and a tag is created from this release. If you then create a hotfix, you do that on this release branch, create the new tag, and then merge those changes back down to the develop branch (and perhaps any feature branches).

I have created to scripts which help you to easily create these releases and hotfixes.

release.sh: This is the first script to do a release. It’s input is the source branch, mostly develop, and the version to created. It will create a new release/x.x.x branch, create a temp branch to create the tag, create the tag, cleanup and then push to remote:

#!/bin/bash

# usage
if [ $# != 2 ] ; then
 echo -e "Usage: ${0} <source_branch> <version>"
 exit 1
fi

# Confirm
sourceBranch=${1}
releaseVersion=${2}
echo -e "INFO: Do you want to release version ${releaseVersion} from branch ${sourceBranch}? y/n"
read a
if [[ "${a}" != "y" && "${a}" != "Y" ]] ; then
 exit 0;
fi

# validate tag and release branch do not exist
echo -e "INFO: Fetching tags and branches and validating they do not exist..."
if ! git fetch --all --tags > /dev/null ; then
 echo -e "ERROR: Tags and branches could not be fetched!"
 exit 1
fi
if git tag | grep "${releaseVersion}" > /dev/null ; then
 echo -e "ERROR: Tag already exists!"
 exit 1
fi
if git branch --all | grep "release/${releaseVersion}" > /dev/null ; then
 echo -e "ERROR: Tag already exists!"
 exit 1
fi

# validate tag does not exist
if git tag | grep "${releaseVersion}" > /dev/null ; then
 echo -e "ERROR: Tag already exists!"
 exit 1
fi

# make sure gpg-agent is available and loaded
echo -e "\nINFO: Searching for gpg-agent..."
if ! which gpg-agent ; then
 echo -e "ERROR: gpg-agent missing!"
fi
if ! gpg-agent 2>&1 | grep "running and available" ; then
 echo -e "WARN: gpg-agent not running, trying to start..."
 if ! gpg-agent --daemon ; then
 echo -e "ERROR: Failed to initialize gpg-agent, please make sure it is up and running before continuing!"
 exit 1
 fi
 if ! gpg-agent 2>&1 | grep "running and available" ; then
 echo -e "ERROR: Failed to initialize gpg-agent, please make sure it is up and running before continuing!"
 exit 1
 fi
fi

# Checkout source branch
echo -e "\nINFO: Checking out source branch ${sourceBranch}"
if ! git checkout ${sourceBranch} ; then
 echo -e "ERROR: Failed to checkout branch ${sourceBranch}"
 exit 1
fi

# create new release branch, and temp tag
echo -e "\nINFO: Creating release and temp branches..."
if ! git checkout -b release/${releaseVersion} ; then 
 echo -e "ERROR: Failed to create release branch!"
 exit 1
fi
if ! git checkout -b temp ; then
 echo -e "ERROR: Failed to create temp branch!"
 exit 1
fi

# set release version
echo -e "\nINFO: Setting version..."
if ! mvn versions:set -DgenerateBackupPoms=false -DnewVersion=${releaseVersion} > /dev/null ; then
 echo -e "ERROR: Failed to set new version!"
 exit 1
fi

# build
echo -e "\nINFO: Doing a build with new version..."
if ! mvn clean package -DskipTests > /dev/null ; then
 echo -e "ERROR: Failed to build with new version!"
 exit 1
fi

# commit to tag
echo -e "\nINFO: Creating tag..."
if ! git add . ; then
 echo -e "ERROR: Failed to git add"
 exit 1
fi
if ! git commit -m "[Project] Set new version ${releaseVersion}" ; then
 echo -e "ERROR: Failed to git commit"
 exit 1
fi
if ! git tag --sign --message "[Project] New Version ${releaseVersion}" ${releaseVersion} ; then
 echo -e "ERROR: Failed to git tag"
 exit 1
fi

# cleanup
echo -e "\nINFO: Cleaning up..."
if ! git checkout release/${releaseVersion} ; then
 echo -e "ERROR: Failed to checkout release branch"
 exit 1
fi
if ! git branch -D temp ; then
 echo -e "ERROR: Failed to delete temp branch"
 exit 1
fi

# git push
echo -e "\nINFO: Release ${releaseVersion} created. Do you want to push to origin? y/n"
read a
if [[ "${a}" == "y" || "${a}" == "Y" ]] ; then
 echo -e "INFO: Pushing to origin..."
 if ! git push origin release/${releaseVersion} ; then
 echo -e "ERROR: Failed to push release branch"
 exit 1
 fi
 if ! git push origin ${releaseVersion} ; then
 echo -e "ERROR: Failed to push tag"
 exit 1
 fi
 echo -e "\nINFO: Pushed release branch and tag for release version ${releaseVersion}"
else
 echo -e "WARN: Release not pushed!"
fi

echo -e "\nINFO: Release ${releaseVersion} created."
echo -e "INFO: Don't forget to update snapshot version!"

exit 0

hotfix.sh: This is the second script to do a hotfix to a release. It’s input is the release version, i.e. 1.0.0 or 1.1.0, and the hotfix version to created i.e. 1.0.1 or 1.1.1. It will check out the release branch i.e. release/1.0.0, create a temp branch to create the tag, create the tag, cleanup and then push to remote:

#!/bin/bash

# usage
if [ $# != 2 ] ; then
 echo -e "Usage: ${0} <release_version> <hotfix_version>"
 exit 1
fi

# Confirm
releaseVersion="${1}"
releaseBranch="release/${1}"
hotfixVersion="${2}"
echo -e "INFO: Do you want to make hotfix version ${hotfixVersion} from release branch ${releaseBranch}? y/n"
read a
if [[ "${a}" != "y" && "${a}" != "Y" ]] ; then
 exit 0;
fi

# validate tag does not exist
echo -e "INFO: Fetching tags and branches and validating they do not exist..."
if ! git fetch --all --tags > /dev/null ; then
 echo -e "ERROR: Tags and branches could not be fetched!"
 exit 1
fi
if git tag | grep ${hotfixVersion} > /dev/null ; then
 echo -e "ERROR: Tag already exists!"
 exit 1
fi

# make sure gpg-agent is available and loaded
echo -e "\nINFO: Searching for gpg-agent..."
if ! which gpg-agent ; then
 echo -e "ERROR: gpg-agent missing!"
fi
if ! gpg-agent 2>&1 | grep "running and available" ; then
 echo -e "WARN: gpg-agent not running, trying to start..."
 if ! gpg-agent --daemon ; then
 echo -e "ERROR: Failed to initialize gpg-agent, please make sure it is up and running before continuing!"
 exit 1
 fi
 if ! gpg-agent 2>&1 | grep "running and available" ; then
 echo -e "ERROR: Failed to initialize gpg-agent, please make sure it is up and running before continuing!"
 exit 1
 fi
fi

# Checkout release branch
echo -e "\nINFO: Checking out release branch ${releaseBranch}"
if ! git checkout ${releaseBranch} ; then
 echo -e "ERROR: Failed to checkout branch ${releaseBranch}"
 exit 1
fi

# create temp branch
echo -e "\nINFO: Creating temp branch..."
if ! git checkout -b temp ; then
 echo -e "ERROR: Failed to create temp branch!"
 exit 1
fi

# set hotfix version
echo -e "\nINFO: Setting version..."
if ! mvn versions:set -DgenerateBackupPoms=false -DnewVersion=${hotfixVersion} > /dev/null ; then
 echo -e "ERROR: Failed to set new version!"
 exit 1
fi

# build
echo -e "\nINFO: Doing a build with new version..."
if ! mvn clean package -DskipTests > /dev/null ; then
 echo -e "ERROR: Failed to build with new version!"
 exit 1
fi

# commit to tag
echo -e "\nINFO: Creating tag..."
if ! git add . ; then
 echo -e "ERROR: Failed to git add"
 exit 1
fi
if ! git commit -m "[Project] Set new version ${hotfixVersion}" ; then
 echo -e "ERROR: Failed to git commit"
 exit 1
fi
if ! git tag --sign --message "[Project] New Version ${hotfixVersion}" ${hotfixVersion} ; then
 echo -e "ERROR: Failed to git tag"
 exit 1
fi

# cleanup
echo -e "\nINFO: Cleaning up..."
if ! git checkout ${releaseBranch} ; then
 echo -e "ERROR: Failed to checkout release branch"
 exit 1
fi
if ! git branch -D temp ; then
 echo -e "ERROR: Failed to delete temp branch"
 exit 1
fi

# git push
echo -e "\nINFO: Hotfix ${hotfixVersion} created. Do you want to push to origin? y/n"
read a
if [[ "${a}" == "y" || "${a}" == "Y" ]] ; then
 echo -e "INFO: Pushing to origin..."
 if ! git push origin ${hotfixVersion} ; then
 echo -e "ERROR: Failed to push tag"
 exit 1
 fi
 echo -e "\nINFO: Pushed hotfix tag ${hotfixVersion}"
else
 echo -e "WARN: Release not pushed!"
fi

echo -e "\nINFO: Hotfix ${hotfixVersion} created."

exit 0

Have fun using the scripts, and don’t hesitate to give me comments!