Antediluvian / reckon

Infer a project's version from your Git repository.

Geek Repo:Geek Repo

Github PK Tool:Github PK Tool

reckon

Bintray Travis Quality Gate GitHub license

Why do you care?

Get that version number out of my build file!

Most build tools and release systems require you to hardcode a version number into a file in your source repository. This results in commit messages like "Bumping version number.". Even if you don't have to do this manually, your release plugin probably modifies your build file and commits the new version.

Git already contains tags with a version number pointing to a specific commit, illustrating that power of this with the git describe command that creates a version number based on the amount of change since the previous tag (e.g. v0.1.0-22-g26f678e).

Git also contains branches for specific stages of development or maintenance for a specific subset of versions.

With this much information available, there's little the user should have to provide to get the next version number. And it certainly doesn't need to be hardcoded anywhere.

What does this version number mean?

Semantic versioning is the best answer to this question so far. It specifies a pretty stringent meaning for what a consumer of an API should expect based on the difference between two versions numbers.

Additionally, it describes methods for encoding pre-release and build-metadata and how those should be sorted by tools.

With that specification and some conventions related to encoding your stage of development into the pre-release information, you can end up with a very easy to understand versioning scheme.

For example, this API's scheme includes 3 stages:

  • final (e.g. 1.0.0) the fully-tested version ready for end-user consumption
  • rc (e.g. 1.1.0-rc.1) release candidates, versions believed to be ready for release after final testing
  • milestone (e.g. 1.1.0-milestone.4) versions containing a significant piece of functionality on the road to the next version

What is it?

Reckon is two things:

  • an API to infer your next version from a Git repository
  • applications of that API in various tools (initially, just Gradle)

Reckon Versioning

Reckon uses an opinionated subset of SemVer, meant to provide more structure around how the pre-release versions are managed.

There are three types of versions:

Type Scheme Example Description
final <major>.<minor>.<patch> 1.2.3 A version ready for end-user consumption
significant <major>.<minor>.<patch>-<stage>.<num> 1.3.0-rc.1 A version indicating an important stage has been reached on the way to the next final release (e.g. alpha, beta, rc, milestone)
insignificant <major>.<minor>.<patch>-<stage>.<num>.<commits>+<hash> 1.3.0-rc.1.8+3bb416187b0a478677b274ae29fb4deb664acda3 A general build in-between significant releases.
  • <major> a postive integer incremented when incompatible API changes are made
  • <minor> a positive integer incremented when functionality is added while preserving backwards-compatibility
  • <patch> a positive integer incremented when fixes are made that preserve backwards-compatibility
  • <stage> an alphabetical identifier indicating a level of maturity on the way to a final release. They should make logical sense to a human, but alphabetical order must be the indicator of maturity to ensure they sort correctly. (e.g. milestone, rc, snapshot would not make sense because snapshot would sort after rc)
  • <num> a positive integer incremented when a significant release is made
  • <commits> a positive integer indicating the number of commits since the last significant or final release was made
  • <hash> a full commit hash of the current HEAD

NOTE: This approach is tuned to ensure it sorts correctly both with SemVer rules and Gradle's built in version sorting (which is subtly different).

Maven Versioning

Reckon can alternately use SNAPSHOT versions instead of the stage concept.

Type Scheme Example Description
final <major>.<minor>.<patch> 1.2.3 A version ready for end-user consumption
snapshot <major>.<minor>.<patch>-SNAPSHOT 1.3.0-SNAPSHOT An intermediate version before the final release is ready.

Inferring the Version

In order to infer the next version, reckon needs two pieces of input:

  • scope - one of major, minor, or patch (defaults to minor), indicating which component of the version should be incremented. If the previous version was 1.2.3, a scope of minor would result in 1.3.0.
  • stage - if not present,

These inputs can be provided directly by the user or using a custom implementation that might detect them from elsewhere.

Reckon will use the history of your repository to determine what version your changes are based on and the inputs above

How do I use it?

NOTE: Check the Release Notes for details on compatibility and changes.

Gradle

NOTE: This plugin only calculates a version, it will not tag it or otherwise modify your Git repository.

Apply the plugin:

plugins {
  id 'org.ajoberstar.grgit' version '<version>'
  id 'org.ajoberstar.reckon' version '<version>'
}

reckon {
  normal = scopeFromProp()
  preRelease = stageFromProp('milestone', 'rc', 'final')
  // alternately
  // preRelease = snapshotFromProp()
}

Execute Gradle providing the properties, as needed:

  • reckon.scope - one of major, minor, or patch (defaults to minor) to specify which component of the previous release should be incremented
  • reckon.stage - (if you used stageFromProp) one of the values passed to stageFromProp (defaults to the first alphabetically) to specify what phase of development you are in
  • reckon.snapshot - (if you used snapshotFromProp) one of true or false (defaults to true) to determine whether a snapshot should be made
./gradlew build -Preckon.scope=minor -Preckon.stage=milestone
Reckoned version 1.3.0-milestone.1
...

How does it work?

Axioms

These are the rules that reckon presumes are true, both informing how it reads a repo's history and how it calculates the next version:

  1. NO duplicates. A single version MUST not be produced from two different commits.
  2. Version numbers MUST increase. If version X's commit is an ancestor of version Y's commit, X < Y.
  3. NO skipping final versions. Final versions MUST increase using only the rules from SemVer 6, 7, 8. e.g. If X=1.2.3, Y must be one of 1.2.4, 1.3.0, 2.0.0.
  4. Two branches MUST NOT create tagged pre-releases for the same targeted final version.
    • If the branches have a shared merge base the version being inferred must skip the targeted final version and increment a second time.
    • If no shared merge base the inference should fail.

Contributing

Contributions are very welcome and are accepted through pull requests.

Smaller changes can come directly as a PR, but larger or more complex ones should be discussed in an issue first to flesh out the approach.

If you're interested in implementing a feature on the issues backlog, add a comment to make sure it's not already in progress and for any needed discussion.

Acknowledgements

Thanks to everyone who contributed to previous iterations of this library and to Zafar Khaja for the very helpful jsemver library.

About

Infer a project's version from your Git repository.

License:Apache License 2.0


Languages

Language:Java 54.5%Language:Groovy 44.2%Language:Shell 1.3%