Even though this project adheres to semver, this guide aims to explain how semantic versioning is used exactly and how we aim to provide stability within an ever-changing codebase.
Please note that this guide isn't stable itself and may be adjusted to fit changes in requirements or new ways are discovered.
- dependent crate
- A crate which depends on a crate in this workspace directly.
- downstream crate
- A crate which directly or indirectly depends on a crate in this workspace.
- workspace crate
- A crate which is a member of this workspace and hence is stored in this repository
- breaking change
- A change in code that requires a
dependent crate
to adjust their code to fix compile errors.
- A change in code that requires a
- release
- A new version of a crate is published to crates.io
- development version
- A crate version whose major version is 0.
- release version
- A crate version whose major version is 1 or higher.
- initial development phase (IDP)
- The phase of development leading up to producing a crate with a major version of 1 or greater, as per
semver
. - Not to be confused with the term pre-release, which is used to indicate any release version prior to an actual release, like
1.1.0-beta.1
.
- The phase of development leading up to producing a crate with a major version of 1 or greater, as per
The project uses three stability tiers for all of its crates, and all crates use semver for their version numbers.
Tiers differ primarily in the time between breaking changes, which always have to be announced with PRs
as per
our collaboration guide.
The following schematic helps to visualize what follows.
Release Software v1.X
Stability Tier 1 ═════════════════════════════╗
║ ║
║ gix──────────────┐ ein──────────────┐ ║
║ │ plumbing app │ │ porcelain app │ ║
║ └────────────────┘ └────────────────┘ ║
║ │ │ ║
║ ▼ ▼ ║
║ gitoxide-core───────────────────────┐ ║
║ │ application functionality │ ║
║ └───────────────────────────────────┘ ║
║ │ ║
║ ▼ ║
║ git-repository──────────────────────┐ ║
║ │ application crate │─ ─ ╬ ─
║ └───────────────────────────────────┘ ║ │
║ │ ║
║ ▼ ║ │
║ Foundation Crates───────────────────┐ ║
║ │ ┌─────────────┐ ┌─────────────┐ │ ║ │
║ │ │ git-hash │ │ git-actor │ │ ║
║ │ └─────────────┘ └─────────────┘ │ ║ │
║ │ ┌─────────────┐ ┌─────────────┐ │ ║
║ │ │ git-ref │ │ git-config │ │ ║ │
║ │ └─────────────┘ └─────────────┘ │ ║
║ │ ┌─────────────┐ ┌─────────────┐ │ ║ │
║ │ │ git-object │ │ git-lock │ │ ║
║ │ └─────────────┘ └─────────────┘ │ ║ │
║ │ ┌───────────────────────────────┐ │ ║
║ │ │ git-features │ │ ║ │
║ │ └───────────────────────────────┘ │ ║
║ └───────────────────────────────────┘ ║ │
║ ║
╚═════════════════════════════════════════════╝ │
Stability Tier 2 ─────────────────────────────┐ │
│ │
│ Plumbing Crates─────────────────────┐ │ │
│ │ ┌─────────────┐ ┌─────────────┐ │ │
│ │ │ git-odb │ │ git-diff │ │ │ │
│ │ └─────────────┘ └─────────────┘ │ │
│ │ ┌─────────────┐ ┌─────────────┐ │ │ │
│ │ │git-traverse │ │ git-pack │ │◀ ─ ┼ ─
│ │ └─────────────┘ └─────────────┘ │ │
│ │ ┌ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ┐ │ │
│ │ …many more… │ │
│ │ └ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ┘ │ │
│ └───────────────────────────────────┘ │
│ │
└─────────────────────────────────────────────┘
Crates in initial development are marked with the major version number zero, for example 0.1.4
and live in stability tier 3 (->ST3).
It's acceptable to let each breaking change be immediately followed by a minor version release.
Released plumbing crates are marked with major version number 1 or above, for example 1.2.4
and live in stability tier 2 (->ST2).
Breaking changes are collected and may be released no more often than every 4 weeks by incrementing the major version number.
For example, git-odb
and git-ref
both have breaking changes, where git-odb
's change is on August 1st and git-ref
's dependent change
is on September 10th. git-odb
's breaking change can be released earliest on September 1st, whereas git-ref
can be released earliest at October 10th.
If there are additional breaking changes without a release, these push back the earliest release date accordingly.
Released apps and application crates are marked with major version number 1 or above, like 2.3.0+21.06
and live in tier 1 (->ST1),
with the build identifiers for year (21
) and and month 06
appended, based on the actual release year and month.
Breaking changes are collected and may be released no more often than every 6 months by incrementing the major version number. If there are additional breaking changes, these push bac the release date so that they can be tested at least for 3 months. For example, a breaking change happens in January 01, and another breaking change in February 15. The earliest release date is July 1st. Had the second breaking change happened in April 01, the release date would have to be pushed to August 1st.
Intermediate pre-releases may be created at most every 4 weeks by appending -alpha.X
where X
is the sequential release number. These should help testing
breaking changes or new features without forcing the use of git
sources for dependencies in cargo manifests. Pre-release releases must pin all the pre-release
crates they depend on to prevent automatic upgrades. Dependent external crates are advised to pin their alpha
dependencies with =<version>
version requirements to avoid
automatic updates which may be breaking.
Once breaking changes are known to be planned, deprecation warnings should be provided in intermediate pre-releases.
Minor version updates for new features can be released when needed assuming there are no other breaking changes, updating the build identifiers for year and month accordingly.
The MSRV is automatically assumed to be the latest stable version for all crates with the following exception: git-repository
and all it's dependencies must
adhere to an MSRV, as validated by the msrv.yml
GitHub workflow.
Increasing the MSRV is considered a breaking change and warrants a major version bump itself for stable crates and a minor version bump for IDP crates.
Please let us know if you have other requirement and we see if we can provide stability guarantees for it or reduce the MSRV to a given version.
How do we avoid staying in the initial development phase (IDP) forever?
There is a couple of questions to ask and answer positively:
- Does the crate fulfill its intended purpose well enough?
- Do the dependent workspace crates fulfill their intended purposes well enough?
- Do they hide types and functionality of lower-tier workspace crates and external IDP crates?
For plumbing crates, the intended purpose is narrow which would allow them to transition earlier. For plumbing crates, if in doubt or fear of future requirements especially if dependent crates are still early in development, prefer to release them anyway and live with requirements of ST2.
Apps and application crates may take longer as they are larger in scope. A good indicator for them to get to a release may be maturing plumbing crates they use. Their scope should also be narrowed to a minimal viable product.