# Releases ## Branches * The `release` branch is intended to preserve the most recent release (including, if applicable, Hackage-only dependency compatibility patch releases). * The `stable` branch is intended to be a copy of the `release` branch together with any subsequent commits that (only) revise the documentation for the most recent release. That documentation is presented at . * The `master` branch is the current development branch. It is intended that a working version of Stack can be built from the branch. The release process begins with a copy of the branch. * A `rc/vX.Y` branch (named after a release in the Stack X.Y.* series) is intended to be for release candidates and final releases. It begins as a copy of the `master` branch. ## Pre-release checks 1. Check that Stack is built against the most recent version of `hpack` released on Hackage. If not, create an issue and/or raise a pull request accordingly. 2. Check for any important issues that should be dealt with before release. 3. Check for un-merged pull requests that should be merged before release. 4. Ensure the `release` and `stable` branches are merged to the `master` branch. 5. Check the copyright dates, and update if needed. 6. Check the backwards compatibility section of `CONTRIBUTING.md` is up to date. 7. Ensure CI matrices in docs (travis-complex, appveyor, azure) have current stackage snapshots and GHC versions (e.g. https://github.com/commercialhaskell/stack/pull/4565/files) 8. Update any `stack-*.yaml` that uses a `nightly` snapshot to the latest nightly (go over the extra-deps too) and ensure the project builds and tests pass. For example, command: ~~~text stack build --stack-yaml=… --haddock --test --bench --no-run-benchmarks ~~~ 9. The Windows installer is built using an [NSIS compiler](https://nsis.sourceforge.io/Main_Page). Check that the NSIS compiler that will be used is capable of handling [large strings](https://nsis.sourceforge.io/Special_Builds). 10. Ensure the integration tests pass on Linux, macOS and Windows. 11. Some people prefer, or need, to build Stack with Cabal (the tool). Check that `cabal.project` is up to date (the specified `with-compiler:`). Check that `cabal.config` is up to date and is not missing dependencies relevant on Windows and non-Windows operating systems, following the instructions in `cabal.project`. ## Release preparation ### A: In the `master` branch 1. `package.yaml`: bump to the next release candidate version (bump the second component to the next odd number, ensure the third component is `0`, and add patchlevel `0`; e.g. from `3.4.0` to `3.5.0.0`). !!! attention Be sure to update also `stack.cabal` (for example by using `stack build --dry-run`). 2. `cabal.config`: Ensure the `stack` constraint is set to the same version as in the `package.yaml`. 3. `ChangeLog.md`: Check for any entries that snuck into the previous version's changes due to merges (`git diff origin/stable HEAD ChangeLog.md`) 4. Commit the changes to the `master` branch. ### B: Create a new release candidate branch From the `master` branch, checkout a new release candidate (RC) branch named `rc/vX.Y` (replacing `X.Y` with the first and second components of the release version). ~~~text git checkout -b rc/vX.Y ~~~ ### C: Return to the `master` branch 1. `package.yaml`: bump version to the next unstable version (bump the second component to the next even number, ensure the third component is `0`; e.g. from `3.5.0` to `3.6.0`). !!! attention Be sure to update also `stack.cabal` (for example by using `stack build --dry-run`). 2. `cabal.config`: Ensure the `stack` constraint is set to the same version as in the `package.yaml`. 3. `Changelog.md`: * Change the title of the existing **Unreleased changes** section to what will be the next final (non-RC) release (e.g. `v3.5.1`). * Add new "Unreleased changes" section: ~~~markdown ## Unreleased changes Release notes: **Changes since vX.Y.Z:** Major changes: Behavior changes: Other enhancements: Bug fixes: ~~~ 4. Commit the changes to the `master` branch. ### D: For each release candidate, in the release candidate branch 1. Review documentation for any changes that need to be made: * Ensure all the documentation pages are listed in the `mkdocs.yaml` file. Use `git diff --stat origin/stable..HEAD doc/` to look for new or deleted files. * Any new documentation pages should have the "may not be correct for the released version of Stack" warning at the top. * Search for old Stack version, unstable Stack version, and the next "obvious" possible versions in sequence, and `UNRELEASED` and replace with next release version (`X.Y.1`, where Y is odd). !!! attention Do **NOT** update the repository's issue and pull request templates (in the `.github` directory) to point at the new release version yet! * Search for old snapshots, set to latest snapshot (e.g. in documentation where it references the "currently the latest LTS") * Look for any links to "latest" (`latest/`) documentation, replace with version tag 2. Check for any platform entries that need to be added to (or removed from): * [releases.yaml](https://github.com/fpco/stackage-content/blob/master/stack/releases.yaml), * [install_and_upgrade.md](https://github.com/commercialhaskell/stack/blob/master/doc/install_and_upgrade.md), * [get-stack.sh](https://github.com/commercialhaskell/stack/blob/master/etc/scripts/get-stack.sh), * [doc/README.md](https://github.com/commercialhaskell/stack/blob/master/doc/README.md), and * `get.haskellstack.org` redirects. 3. Re-do the pre-release checks (see the section above). 4. Update `package.yaml` and `ChangeLog.md`. This step differs between a first, second etc release candidate and a final release. === "First RC" * `package.yaml`: bump to first odd patchlevel version (e.g. `X.Y.0.1`). * `ChangeLog.md`: Rename the “Unreleased changes” section to the same version as `package.yaml`, and mark it clearly as a release candidate (e.g. `vX.Y.0.1 (release candidate)`). Remove any empty sections. === "Second, third etc RC" * `package.yaml`: bump to next odd patchlevel version (e.g. `X.Y.0.3`). * `ChangeLog.md`: Rename the “Unreleased changes” section to the same version as `package.yaml`, and mark it clearly as a release candidate (e.g. `vX.Y.0.3 (release candidate)`). Remove any empty sections. === "Final Release" * `package.yaml`: bump version to odd last component and no patchlevel (e.g. from `X.Y.0.2` to `X.Y.1`). * `ChangeLog.md`: consolidate all the release candidate changes into a single section for the final release version. !!! attention After updating `package.yaml`, be sure to update also `stack.cabal` (for example by using `stack build --dry-run`). 5. Ensure the `stack ==` constraint in `cabal.config` is set to be equal to the same version as `package.yaml`. 6. Commit the changes to the release candidate branch. 7. Follow the steps in the *Release process* section below that apply. ## Release process The release process differs between a first, second etc release candidate and a final release. === "First, second etc RC" ### A: Integration tests workflow passes Ensure that the GitHub [Integration Tests workflow](https://github.com/commercialhaskell/stack/actions?query=workflow%3A%22Integration+tests%22) passes on the branch that you are releasing. This workflow will run automatically for the `rc/*` branch. ### B: Push a Git tag Push a Git tag. The tag should be `rc/vX.Y.Z.A`, with `X.Y.Z.A` matching the version in `package.yaml`. For example, command: ~~~text git tag -m rc/vX.Y.Z.A rc/vX.Y.Z.A git push origin rc/vX.Y.Z.A ~~~ ### C: Edit the draft GitHub release, and publish it Wait for the GitHub [Integration Tests workflow](https://github.com/commercialhaskell/stack/actions?query=workflow%3A%22Integration+tests%22) to complete for the branch you just created. This will create a draft GitHub release and upload the bindists (plus signatures and hashes) to it. Edit the draft [GitHub release](https://github.com/commercialhaskell/stack/releases/): * Add `(release candidate)` to the name field and ensure that *This is a pre-release* is checked. * Add the ChangeLog to the description. Publish the GitHub release. ### D: Consider adding other platforms to the GitHub release The [Integration Tests workflow](https://github.com/commercialhaskell/stack/actions?query=workflow%3A%22Integration+tests%22) is limited to the platforms supported by the GitHub-hosted runners (currently, x86_64 and macOS/AArch64) and any self-hosted runners (currently, only Linux/AArch64). However, it is possible to edit the GitHub release to include binary distributions for other platforms. The prerequisites are: * a computer with that platform (operating system, machine architecture); * a sufficiently-recent existing version of Stack for that platform; * a tool to print SHA checksums, such as `shasum` on Linux and macOS; and * the GNU Privacy Guard tool (`gpg`), which has had imported the private key used to sign Stack executables (see further below). The steps are similar to those in the workflow: 1. Change to the root directory of the Stack project. 2. `stack etc/scripts/release.hs check`, to check before building. 3. `stack etc/scripts/release.hs build`, to build. The output 'assets' (`stack--- ...`) will be in the `_release` directory in the root directory of the Stack project. 4. For each of the output assets, create a corresponding SHA 256 file with a `.sha256` extension. For example (where `` is the name of the file): ~~~text shasum -a 256 > .sha256 ~~~ 5. For each of the output assets, create a corresponding ASCII-armored signature file with an `.asc` extension using `gpg`. For example (where `` is the name of the file): ~~~text gpg --digest-algo=sha512 --detach-sig --armor -u 0x575159689BEFB442 ~~~ 6. Edit the GitHub release to include the output assets and their corresponding `.sha256` and `.asc` files. The private key used to sign Stack executables can be exported from a version of `gpg` to which it has previously been imported with: ~~~text gpg --armor --export-secret-key 0x575159689BEFB442 ~~~ The private key, so obtained, can be imported into `gpg` by: 1. Commanding `gpg --import`. 2. Pasting the private key. 3. Entering Ctrl+D and Enter. ### E: Update versions and `ChangeLog.md` for 'unreleased' In the `rc/vX.Y` branch: * `package.yaml`: bump the version number. Bump the fourth component to an even number (e.g. from `3.5.0.1` to `3.5.0.2`). !!! attention Be sure to update also `stack.cabal` (for example by using `stack build --dry-run`). * `cabal.config`: Ensure the `stack` constraint is set to the same version as in the `package.yaml`. * `ChangeLog.md`: Add an “Unreleased changes” section (update the “changes since” version): ~~~markdown ## Unreleased changes Release notes: **Changes since vX.Y.Z:** Major changes: Behavior changes: Other enhancements: Bug fixes: ~~~ ### F: Request update of GHCup's metadata Raise a pull request at the [`haskell/ghcup-metadata`](https://github.com/haskell/ghcup-metadata) GitHub repository to request an addition to GHCup's latest metadata configuration file for prereleases, tagged as the latest prerelease. In the metadata, change the tags for any past Stack prereleases to indicate that they are no longer the latest prerelease. ### G: Announce the release candidate Announce the release candidate: * on the [Haskell Community](https://discourse.haskell.org/c/announcements/10/l/latest) forum; * in the Haskell Foundation's [general](https://matrix.to/#/#haskell-foundation-general:matrix.org) room (address `#haskell-foundation-general:matrix.org`) on [Matrix](https://matrix.org/); * in the Haskell [Stack and Stackage](https://matrix.to/#/#haskell-stack:matrix.org) room (address `#haskell-stack:matrix.org`) on [Matrix](https://matrix.org/); and * in Reddit's [Haskell](https://www.reddit.com/r/haskell/) community. In each case, use the subject (change 'first' to 'second' etc for subsequent release candidates): * `[ANN] First release candidate for Stack X.Y.Z` In the message, include: * a link to the release on GitHub (`https://github.com/commercialhaskell/stack/releases/tag/rc/vX.Y.Z.A`) to download it * the release description from Github. === "Final Release" ### A: Integration tests workflow passes Ensure that the GitHub [Integration Tests workflow](https://github.com/commercialhaskell/stack/actions?query=workflow%3A%22Integration+tests%22) passes on the branch that you are releasing. This workflow will run automatically for `rc/*` branches. ### B: Push a Git tag Push a Git tag. The tag should be `vX.Y.Z`, where `X.Y.Z` matches the version in `package.yaml`. For example, command: ~~~text git tag -m vX.Y.Z vX.Y.Z git push origin vX.Y.Z ~~~ ### C: Edit the draft GitHub release, and publish it Wait for the GitHub [Integration Tests workflow](https://github.com/commercialhaskell/stack/actions?query=workflow%3A%22Integration+tests%22) to complete for the tag you just created. This will create a draft GitHub release and upload the bindists (plus signatures and hashes) to it. Edit the draft [GitHub release](https://github.com/commercialhaskell/stack/releases/): * Add the ChangeLog to the description. * Get the list of contributors to the release and add it to the description. For example, command: === "Unix-like" ~~~text git shortlog -s origin/release..HEAD|sed 's/^[0-9 \t]*/* /'|LC_ALL=C sort -f ~~~ === "Windows" ~~~text (git shortlog -s origin/release..HEAD) -Replace '^[0-9 \t]*', '* ' | Sort-Object ~~~ in PowerShell. Publish the GitHub release. ### D: Consider adding other platforms to the GitHub release The [Integration Tests workflow](https://github.com/commercialhaskell/stack/actions?query=workflow%3A%22Integration+tests%22) is limited to the platforms supported by the GitHub-hosted runners (currently, x86_64 and macOS/AArch64) and any self-hosted runners (currently, only Linux/AArch64). However, it is possible to edit the GitHub release to include binary distributions for other platforms. The prerequisites are: * a computer with that platform (operating system, machine architecture); * a sufficiently-recent existing version of Stack for that platform; * a tool to print SHA checksums, such as `shasum` on Linux and macOS; and * the GNU Privacy Guard tool (`gpg`), which has had imported the private key used to sign Stack executables (see further below). The steps are similar to those in the workflow: 1. Change to the root directory of the Stack project. 2. `stack etc/scripts/release.hs check`, to check before building. 3. `stack etc/scripts/release.hs build`, to build. The output 'assets' (`stack--- ...`) will be in the `_release` directory in the root directory of the Stack project. 4. For each of the output assets, create a corresponding SHA 256 file with a `.sha256` extension. For example (where `` is the name of the file): ~~~text shasum -a 256 > .sha256 ~~~ 5. For each of the output assets, create a corresponding ASCII-armored signature file with an `.asc` extension using `gpg`. For example (where `` is the name of the file): ~~~text gpg --digest-algo=sha512 --detach-sig --armor -u 0x575159689BEFB442 ~~~ 6. Edit the GitHub release to include the output assets and their corresponding `.sha256` and `.asc` files. The private key used to sign Stack executables can be exported from a version of `gpg` to which it has previously been imported with: ~~~text gpg --armor --export-secret-key 0x575159689BEFB442 ~~~ The private key, so obtained, can be imported into `gpg` by: 1. Commanding `gpg --import`. 2. Pasting the private key. 3. Entering Ctrl+D and Enter. ### E: Upload to Hackage and reset branches Upload the `stack` package to Hackage with the command: ~~~text stack upload . --pvp-bounds=lower ~~~ Reset the `release` branch to the released commit. For example, with the commands: ~~~text git checkout release git merge --ff-only vX.Y.Z git push origin release ~~~ Update the `stable` branch to the released commit. For example, with the commands: ~~~text git checkout stable git merge --ff-only vX.Y.Z git push origin stable ~~~ Merge any changes made in the RC, `release` or `stable` branches to the `master` branch. Be careful about version and `ChangeLog.md`. It is best to do this by making a `ci/merge-stable-to-master` branch and waiting for CI to pass, then merging. If anything is complicated to merge, consider making it a pull request and getting it reviewed rather than merging immediately. Delete the RC branch, both locally and on the remote. For example with the commands: ~~~text git branch -d rc/vX.Y git push origin :rc/vX.Y ~~~ ### F: Activate the version on Read The Docs Activate the version for new release tag, on [readthedocs.org](https://readthedocs.org/projects/stack/versions/). Ensure that the `stable` documentation has updated. ### G: Update get.haskellstack.org redirects Update the https://get.haskellstack.org redirects by updating the `_redirects` file in the root of the `commercialhaskell/get-haskellstack-org` GitHub [repository](https://github.com/commercialhaskell/get-haskellstack-org). For further information, see the [get.haskellstack.org redirects](haskellstack.org.md#gethaskellstackorg-redirects) documentation. Test with the commands: === "Unix-like" ~~~text curl -vL https://get.haskellstack.org/stable/linux-x86_64.tar.gz >/dev/null curl -vL https://get.haskellstack.org/upgrade/linux-x86_64.tar.gz >/dev/null ~~~ === "Windows" ~~~text curl -vL https://get.haskellstack.org/stable/linux-x86_64.tar.gz >NUL curl -vL https://get.haskellstack.org/upgrade/linux-x86_64.tar.gz >NUL ~~~ and make sure it redirects to the new version. ### H: Update versions and `ChangeLog.md` for 'unreleased' In the `stable` branch: * `package.yaml`: bump the version number. Bump the third component to an even number (e.g. from `3.5.1` to `3.5.2`). !!! attention Be sure to update also `stack.cabal` (for example by using `stack build --dry-run`). * `cabal.config`: Ensure the `stack` constraint is set to the same version as in the `package.yaml`. ### I: Update the repository's issue and pull request templates The repository's issue and pull request templates are the `.github` directory. Update them to refer to the new release version (`X.Y.Z`). ### J: Request update of GHCup's metadata Raise a pull request at the [`haskell/ghcup-metadata`](https://github.com/haskell/ghcup-metadata) GitHub repository to request an addition to GHCup's latest metadata configuration files for releases and 'vanilla' releases, tagged as the latest release. (The GHCup project will decide whether, and when, to recommend the release.) In the metadata, change the tags for any past Stack releases to indicate that they are no longer the latest release. ### K: Announce the release Announce the release: * on the [Haskell Community](https://discourse.haskell.org/c/announcements/10/l/latest) forum. * in the Haskell Foundation's [general](https://matrix.to/#/#haskell-foundation-general:matrix.org) room (address `#haskell-foundation-general:matrix.org`) on [Matrix](https://matrix.org/). * in the Haskell [Stack and Stackage](https://matrix.to/#/#haskell-stack:matrix.org) room (address `#haskell-stack:matrix.org`) on [Matrix](https://matrix.org/). * in Reddit's [Haskell](https://www.reddit.com/r/haskell/) community. In each case, use the subject: * `[ANN] Stack X.Y.Z` In the message, include: * the release description from Github.