name: Release
on:
push:
tags:
- 'v*'
concurrency:
group: release-${{ github.ref_name }}
cancel-in-progress: false
permissions:
contents: write
id-token: write
env:
CARGO_TERM_COLOR: always
jobs:
validate:
name: Validate tag
runs-on: ubuntu-latest
timeout-minutes: 5
outputs:
version: ${{ steps.resolve.outputs.version }}
tag: ${{ steps.resolve.outputs.tag }}
steps:
- uses: actions/checkout@v4
- name: Resolve version from tag
id: resolve
run: |
TAG="${GITHUB_REF_NAME}"
if ! [[ "$TAG" =~ ^v[0-9]+\.[0-9]+\.[0-9]+(-[a-zA-Z0-9.]+)?$ ]]; then
echo "::error::Invalid tag format: $TAG (expected vX.Y.Z or vX.Y.Z-pre.N)"
exit 1
fi
VERSION="${TAG#v}"
echo "tag=$TAG" >> "$GITHUB_OUTPUT"
echo "version=$VERSION" >> "$GITHUB_OUTPUT"
echo "Resolved tag=$TAG version=$VERSION"
- name: Check Cargo.toml version matches tag
env:
EXPECTED: ${{ steps.resolve.outputs.version }}
run: |
ACTUAL=$(grep -m1 '^version = ' Cargo.toml | cut -d'"' -f2)
echo "Cargo.toml version: $ACTUAL"
echo "Expected: $EXPECTED"
if [ "$ACTUAL" != "$EXPECTED" ]; then
echo "::error::Cargo.toml version ($ACTUAL) does not match tag ($EXPECTED)"
exit 1
fi
- name: Check CHANGELOG has entry for this version
env:
VERSION: ${{ steps.resolve.outputs.version }}
run: scripts/extract-changelog.sh "$VERSION" CHANGELOG.md > /dev/null
build:
name: Build (${{ matrix.target }})
needs: validate
runs-on: ${{ matrix.os }}
timeout-minutes: 30
strategy:
fail-fast: true
matrix:
include:
- target: x86_64-unknown-linux-musl
os: ubuntu-latest
archive: tar.gz
- target: aarch64-unknown-linux-musl
os: ubuntu-latest
archive: tar.gz
- target: x86_64-apple-darwin
os: macos-latest
archive: tar.gz
- target: aarch64-apple-darwin
os: macos-latest
archive: tar.gz
- target: x86_64-pc-windows-msvc
os: windows-latest
archive: zip
steps:
- uses: actions/checkout@v4
- uses: dtolnay/rust-toolchain@stable
with:
targets: ${{ matrix.target }}
- name: Install musl tools (x86_64)
if: matrix.target == 'x86_64-unknown-linux-musl'
run: |
sudo apt-get update
sudo apt-get install -y musl-tools
- name: Install zig (aarch64-musl)
if: matrix.target == 'aarch64-unknown-linux-musl'
uses: mlugg/setup-zig@v2
- name: Install cargo-zigbuild (aarch64-musl)
if: matrix.target == 'aarch64-unknown-linux-musl'
uses: taiki-e/install-action@cargo-zigbuild
- name: Build
if: matrix.target != 'aarch64-unknown-linux-musl'
run: cargo build --release --target ${{ matrix.target }} --features full
- name: Build (zigbuild)
if: matrix.target == 'aarch64-unknown-linux-musl'
run: cargo zigbuild --release --target ${{ matrix.target }} --features full
- name: Package (Unix)
if: matrix.archive == 'tar.gz'
run: |
cd target/${{ matrix.target }}/release
tar czf ../../../dynoxide-${{ matrix.target }}.tar.gz dynoxide
cd ../../..
- name: Package (Windows)
if: matrix.archive == 'zip'
shell: pwsh
run: |
Compress-Archive -Path "target/${{ matrix.target }}/release/dynoxide.exe" -DestinationPath "dynoxide-${{ matrix.target }}.zip"
- name: Upload artifact
uses: actions/upload-artifact@v4
with:
name: dynoxide-${{ matrix.target }}
path: dynoxide-${{ matrix.target }}.${{ matrix.archive }}
github-release:
name: Create GitHub Release
needs: [validate, build]
runs-on: ubuntu-latest
timeout-minutes: 10
steps:
- uses: actions/checkout@v4
- name: Download all artifacts
uses: actions/download-artifact@v4
with:
path: artifacts
- name: Create checksums
run: |
cd artifacts
find . -type f \( -name "*.tar.gz" -o -name "*.zip" \) -exec mv {} . \;
sha256sum *.tar.gz *.zip > sha256sums.txt
cat sha256sums.txt
- name: Extract changelog body
env:
VERSION: ${{ needs.validate.outputs.version }}
run: scripts/extract-changelog.sh "$VERSION" CHANGELOG.md > /tmp/release_body.md
- name: Create Release
uses: softprops/action-gh-release@v2
with:
tag_name: ${{ needs.validate.outputs.tag }}
name: ${{ needs.validate.outputs.tag }}
body_path: /tmp/release_body.md
files: |
artifacts/*.tar.gz
artifacts/*.zip
artifacts/sha256sums.txt
publish-crate:
name: Publish to crates.io
needs: [validate, github-release]
environment: production-publish
runs-on: ubuntu-latest
timeout-minutes: 15
steps:
- uses: actions/checkout@v4
- uses: dtolnay/rust-toolchain@stable
- uses: Swatinem/rust-cache@v2
- name: cargo publish --dry-run
env:
CARGO_REGISTRY_TOKEN: ${{ secrets.CARGO_REGISTRY_TOKEN }}
run: cargo publish --dry-run
- name: cargo publish
env:
CARGO_REGISTRY_TOKEN: ${{ secrets.CARGO_REGISTRY_TOKEN }}
run: cargo publish
publish-homebrew:
name: Update Homebrew tap
needs: [validate, publish-crate]
runs-on: ubuntu-latest
timeout-minutes: 10
steps:
- name: Download checksums from release
env:
GH_TOKEN: ${{ secrets.HOMEBREW_TAP_TOKEN }}
TAG: ${{ needs.validate.outputs.tag }}
run: |
gh release download "$TAG" \
--repo nubo-db/dynoxide \
--pattern 'sha256sums.txt' \
--output sha256sums.txt
- name: Build formula
env:
VERSION: ${{ needs.validate.outputs.version }}
run: |
SHA_DARWIN_ARM64=$(grep 'aarch64-apple-darwin' sha256sums.txt | awk '{print $1}')
SHA_DARWIN_X64=$(grep 'x86_64-apple-darwin' sha256sums.txt | awk '{print $1}')
SHA_LINUX_ARM64=$(grep 'aarch64-unknown-linux-musl' sha256sums.txt | awk '{print $1}')
SHA_LINUX_X64=$(grep 'x86_64-unknown-linux-musl' sha256sums.txt | awk '{print $1}')
for var in SHA_DARWIN_ARM64 SHA_DARWIN_X64 SHA_LINUX_ARM64 SHA_LINUX_X64; do
if [[ -z "${!var}" ]]; then
echo "::error::Missing checksum for $var"
exit 1
fi
done
BASE="https://github.com/nubo-db/dynoxide/releases/download/v${VERSION}"
cat <<FORMULA | sed 's/^ //' > formula.rb
class Dynoxide < Formula
desc "Fast, lightweight drop-in replacement for DynamoDB Local, backed by SQLite"
homepage "https://dynoxide.dev"
version "${VERSION}"
license any_of: ["MIT", "Apache-2.0"]
on_macos do
on_arm do
url "${BASE}/dynoxide-aarch64-apple-darwin.tar.gz"
sha256 "${SHA_DARWIN_ARM64}"
end
on_intel do
url "${BASE}/dynoxide-x86_64-apple-darwin.tar.gz"
sha256 "${SHA_DARWIN_X64}"
end
end
on_linux do
on_arm do
url "${BASE}/dynoxide-aarch64-unknown-linux-musl.tar.gz"
sha256 "${SHA_LINUX_ARM64}"
end
on_intel do
url "${BASE}/dynoxide-x86_64-unknown-linux-musl.tar.gz"
sha256 "${SHA_LINUX_X64}"
end
end
def install
bin.install "dynoxide"
end
test do
assert_match version.to_s, shell_output("#{bin}/dynoxide --version")
end
end
FORMULA
- name: Push formula to tap
env:
GH_TOKEN: ${{ secrets.HOMEBREW_TAP_TOKEN }}
VERSION: ${{ needs.validate.outputs.version }}
run: |
CURRENT_SHA=$(gh api repos/nubo-db/homebrew-tap/contents/Formula/dynoxide.rb --jq '.sha')
gh api repos/nubo-db/homebrew-tap/contents/Formula/dynoxide.rb \
-X PUT \
--field message="dynoxide $VERSION" \
--field content="$(base64 -w0 formula.rb)" \
--field sha="$CURRENT_SHA" \
--jq '.commit.html_url'
publish-npm:
name: Trigger npm publish
needs: [validate, publish-crate]
runs-on: ubuntu-latest
timeout-minutes: 5
permissions:
actions: write
contents: read
steps:
- name: Dispatch npm.yml with the release tag
env:
GH_TOKEN: ${{ github.token }}
TAG: ${{ needs.validate.outputs.tag }}
run: |
gh api "repos/${{ github.repository }}/actions/workflows/npm.yml/dispatches" \
-f ref=main \
-f "inputs[tag]=$TAG"
echo "Dispatched npm.yml with tag $TAG"
notify-site:
name: Trigger dynoxide.dev rebuild
needs: [validate, publish-crate, publish-homebrew, publish-npm]
runs-on: ubuntu-latest
timeout-minutes: 5
steps:
- name: Dispatch dynoxide.dev deploy
env:
GH_TOKEN: ${{ secrets.DYNOXIDE_SITE_DISPATCH_TOKEN }}
VERSION: ${{ needs.validate.outputs.version }}
TAG: ${{ needs.validate.outputs.tag }}
run: |
gh api repos/nubo-db/dynoxide.dev/dispatches \
-X POST \
-f event_type=dynoxide-release \
-f "client_payload[version]=$VERSION" \
-f "client_payload[tag]=$TAG"
echo "Dispatched dynoxide-release event for $TAG"