Skip to content

Commit

Permalink
[BDGR-173] Add new automated release workflow (#654)
Browse files Browse the repository at this point in the history
* Use the public tus endpoint

* Add new release workflow

* Uncomment testing steps

* Add canary job and replace version with 0.0.0

* Better release notes + try to fix version setting

* Fix release set

* Fix release setting again

* Fix tests checking version

* Properly upload

* Sort out artifact uploads

* Add clobber to release upload

* Debug

* More fiddling with artifact uploading

* Fix Linear closing
  • Loading branch information
markspolakovs authored Sep 25, 2024
1 parent 7c9767d commit 6d6953b
Show file tree
Hide file tree
Showing 18 changed files with 264 additions and 88 deletions.
35 changes: 35 additions & 0 deletions .github/steps/setup-playwright/action.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
name: Set Up Playwright
description: Sets up Playwright
inputs:
working-directory:
description: Where to run
required: false
runs:
using: composite
steps:
# Adapted from https://playwrightsolutions.com/playwright-github-action-to-cache-the-browser-binaries/
- name: Get installed Playwright version
id: playwright-version
run: echo version=$(yarn info --json @playwright/test | jq -r '.children.Version') >> $GITHUB_OUTPUT
working-directory: ${{ inputs.working-directory }}
shell: bash

- name: Cache playwright binaries
uses: actions/cache@v4
id: playwright-cache
with:
path: |
~/.cache/ms-playwright
key: ${{ runner.os }}-playwright-${{ steps.playwright-version.outputs.version }}

- name: Install Playwright Browsers
run: yarn playwright install --with-deps
working-directory: ${{ inputs.working-directory }}
if: steps.playwright-cache.outputs.cache-hit != 'true'
shell: bash

- name: Install Playwright OS dependencies
run: npx playwright install-deps
working-directory: ${{ inputs.working-directory }}
if: steps.playwright-cache.outputs.cache-hit == 'true'
shell: bash
15 changes: 15 additions & 0 deletions .github/workflows/build_canary.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
on:
push:
branches: [main]

permissions:
contents: write
packages: write

jobs:
build:
uses: ./.github/workflows/build_shared.yml
with:
ref: ${{ github.ref }}
tag: ${{ github.ref_name }}
strip_rc: true
42 changes: 36 additions & 6 deletions .github/workflows/build_shared.yml
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,10 @@ on:
tag:
type: string
required: true
description: Docker tag to push
description: Version to build, also the docker tag
strip_rc:
type: boolean
description: Strip -rc suffix from version number

permissions:
contents: read
Expand All @@ -34,7 +37,17 @@ jobs:

- run: yarn install --immutable

- run: "yarn package"
- name: Set version
shell: pwsh
run: |
$pkg = Get-Content ./desktop/package.json | ConvertFrom-Json
$pkg.version = "${{inputs.tag }}" -replace "^v", ""
if ("${{ inputs.strip_rc }}" -eq "true") {
$pkg.version = $pkg.version -replace "-rc.*", ""
}
$pkg | ConvertTo-Json -Depth 32 | Set-Content ./desktop/package.json
- run: "yarn package --win --publish never"
working-directory: ./desktop
env:
GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}
Expand Down Expand Up @@ -67,18 +80,27 @@ jobs:
registry: ghcr.io
username: ${{ github.actor }}
password: ${{ secrets.GITHUB_TOKEN }}
- name: Set version
run: |
set -x
version=$(echo '${{ inputs.tag }}' | sed 's/^v//')
if [[ '${{ inputs.strip_rc }}' == 'true' ]]; then
version=$(echo $version | sed 's/-rc.*//')
fi
sed -i "s/0.0.0/$version/g" package.json
working-directory: ./server
- name: Docker metadata
id: meta
uses: docker/metadata-action@v5
with:
images: ghcr.io/ystv/badger/server
flavor: latest=true
tags: |
type=semver,pattern={{version}},enable=${{ github.event_name == 'workflow_dispatch' }}
type=raw,value=${{ inputs.tag }},enable=${{ github.event_name == 'workflow_call' }}
type=raw,value=${{ inputs.tag }}
- name: Build and push
uses: docker/build-push-action@v6
with:
context: .
push: true
tags: ${{ steps.meta.outputs.tags }}
labels: ${{ steps.meta.outputs.labels }}
Expand All @@ -101,18 +123,26 @@ jobs:
registry: ghcr.io
username: ${{ github.actor }}
password: ${{ secrets.GITHUB_TOKEN }}
- name: Set version
run: |
version=$(echo '${{ inputs.tag }}' | sed 's/^v//')
if [[ '${{ inputs.strip_rc }}' == 'true' ]]; then
version=$(echo $version | sed 's/-rc.*//')
fi
sed -i "s/0.0.0/$version/g" package.json
working-directory: ./jobrunner
- name: Docker metadata
id: jr_meta
uses: docker/metadata-action@v5
with:
images: ghcr.io/ystv/badger/jobrunner
flavor: latest=true
tags: |
type=semver,pattern={{version}},enable=${{ github.event_name == 'workflow_dispatch' }}
type=raw,value=${{ inputs.tag }},enable=${{ github.event_name == 'workflow_call' }}
type=raw,value=${{ inputs.tag }}
- name: Build and push
uses: docker/build-push-action@v6
with:
context: .
push: true
tags: ${{ steps.jr_meta.outputs.tags }}
labels: ${{ steps.jr_meta.outputs.labels }}
Expand Down
179 changes: 136 additions & 43 deletions .github/workflows/release.yml
Original file line number Diff line number Diff line change
Expand Up @@ -3,16 +3,15 @@ on:
tags:
- "*-rc*"

permissions:
contents: write
packages: write
permissions: write-all

jobs:
build:
uses: ./.github/workflows/build_shared.yml
with:
ref: ${{ github.event.before }}
tag: ${{ github.event.ref }}
ref: ${{ github.ref }}
tag: ${{ github.ref_name }}
strip_rc: true

test-e2e-server:
needs: [build]
Expand All @@ -31,36 +30,24 @@ jobs:
cache-dependency-path: "yarn.lock"

- name: Set ref in docker-compose
run: sed -i "s/__RC_REF__/${{ github.event.ref }}/g" docker-compose-rc-test.yml
run: sed -i "s/__RC_REF__/${{ github.ref_name }}/g" docker-compose-rc-test.yml

- name: Start services
run: docker compose up -d -f docker-compose.yml -f docker-compose-rc-test.yml
run: docker compose -f docker-compose.yml -f docker-compose-rc-test.yml up -d

# Adapted from https://playwrightsolutions.com/playwright-github-action-to-cache-the-browser-binaries/
- name: Get installed Playwright version
id: playwright-version
run: echo version=$(yarn info --json @playwright/test | jq -r '.children.Version') >> $GITHUB_OUTPUT
working-directory: ./server
- run: yarn install --immutable --inline-builds

- name: Cache playwright binaries
uses: actions/cache@v4
id: playwright-cache
- uses: ./.github/steps/setup-playwright
with:
path: |
~/.cache/ms-playwright
key: ${{ runner.os }}-playwright-${{ steps.playwright-version.outputs.version }}

- name: Install Playwright Browsers
run: yarn playwright install --with-deps
working-directory: ./server
if: steps.playwright-cache.outputs.cache-hit != 'true'

- name: Install Playwright OS dependencies
run: npx playwright install-deps
if: steps.playwright-cache.outputs.cache-hit == 'true'
working-directory: ./server

- name: Migrate database
run: yarn prisma:migrateProd
run: |
yarn prisma:migrateProd
- name: Retart services
run: |
docker compose -f docker-compose.yml -f docker-compose-rc-test.yml restart server jobrunner
- name: Run Playwright tests
run: yarn ${{ runner.debug && 'test:e2e:debug' || 'test:e2e' }}
Expand All @@ -75,31 +62,114 @@ jobs:
path: ./server/playwright-report/
retention-days: 30

test-desktop:
runs-on: windows-latest
needs: [build]
steps:
- uses: actions/checkout@v4
- name: Use Node.js 18.x
uses: actions/setup-node@v4
with:
node-version: 18.x
cache: "yarn"
cache-dependency-path: "yarn.lock"
- name: Download Desktop build
uses: actions/download-artifact@v4
with:
name: badger-desktop-windows
- name: Install Badger
run: |
$version = "${{ github.ref_name }}" -replace "^v", "" -replace "-rc.*", ""
Start-Process -FilePath "Badger Desktop-$version.exe" -ArgumentList "/S","/D=${{ runner.temp }}\badger" -Wait
shell: pwsh
- run: yarn install --immutable --inline-builds
- name: Run tests
run: yarn test:e2e --project=standalone
working-directory: ./desktop
env:
TEST_APPLICATION_PATH: ${{ runner.temp }}\badger\Badger Desktop.exe

linear:
needs: [test-e2e-server, test-desktop]
runs-on: ubuntu-latest
outputs:
issue_id: ${{ steps.issue.outputs.issue_id }}
steps:
- name: Determine version number
run: echo "VERSION=$(echo '${{ github.ref_name }}' | sed 's/-rc.*//')" >> $GITHUB_ENV
- uses: actions/setup-node@v3
with:
node-version: "20.x"
- run: npm install @linear/sdk
- name: Create Linear release ticket
id: issue
uses: actions/github-script@v7
with:
script: |
const { LinearClient } = require('@linear/sdk');
const lin = new LinearClient({
accessToken: "${{ secrets.LINEAR_ACCESS_TOKEN }}"
});
const issueCreate = await lin.createIssue({
teamId: "${{ vars.LINEAR_TEAM_ID }}",
templateId: "${{ vars.LINEAR_RELEASE_ISSUE_TEMPLATE_ID }}",
stateId: "${{ vars.LINEAR_TODO_STATE_ID }}",
title: `Release ${{ github.ref_name }}`,
});
if (!issueCreate.success) {
throw new Error(`Failed to create issue`);
}
const issue = await issueCreate.issue;
await lin.createComment({
issueId: issue.id,
body: `Artifacts:
Server Docker image: \`ghcr.io/ystv/badger/server:${{ github.ref_name }}\`
Jobrunner Docker Image: \`ghcr.io/ystv/badger/jobrunner:${{ github.ref_name }}\`
Desktop Windows installer: ${{ github.server_url }}/${{ github.repository }}/actions/runs/${{ github.run_id }}
Once testing is complete, please approve [this workflow](${{ github.server_url }}/${{ github.repository }}/actions/runs/${{ github.run_id }}) to publish the release.
`.replace(/^\s*/gm, ''),
});
core.summary.addHeading('Linear issue');
core.summary.addLink(issue.identifier, issue.url);
core.setOutput('issue_id', issue.id);
release:
needs: [test-e2e-server, build]
needs: [test-e2e-server, test-desktop, linear]
environment: release
runs-on: ubuntu-latest
permissions: write-all
steps:
- name: Determine version number
run: echo "VERSION=$(echo '${{ github.event.ref }}' | sed 's/^refs\/tags\///' | sed 's/-rc.*//')" >> $GITHUB_ENV
run: echo "VERSION=$(echo '${{ github.ref_name }}' | sed 's/-rc.*//')" >> $GITHUB_ENV
- name: Download Desktop build
uses: actions/download-artifact@v4
with:
pattern: badger-desktop-*
path: artifacts
- name: Pull Docker images
run: |
docker pull ghcr.io/ystv/badger/server:${{ github.event.ref }}
docker pull ghcr.io/ystv/badger/jobrunner:${{ github.event.ref }}
- name: Login to GitHub Container Registry
uses: docker/login-action@v3
with:
registry: ghcr.io
username: ${{ github.actor }}
password: ${{ secrets.GITHUB_TOKEN }}
- name: Re-tag and push Docker images
run:
docker tag ghcr.io/ystv/badger/server:${{ github.event.ref }} ghcr.io/ystv/badger/server:$VERSION
docker tag ghcr.io/ystv/badger/jobrunner:${{ github.event.ref }} ghcr.io/ystv/badger/jobrunner:$VERSION
docker push ghcr.io/ystv/badger/server:$VERSION
docker push ghcr.io/ystv/badger/jobrunner:$VERSION
run: |
for img in server jobrunner; do
docker pull ghcr.io/ystv/badger/$img:${{ github.ref_name }}
docker tag ghcr.io/ystv/badger/$img:${{ github.ref_name }} ghcr.io/ystv/badger/$img:$VERSION
docker push ghcr.io/ystv/badger/$img:$VERSION
docker tag ghcr.io/ystv/badger/$img:${{ github.ref_name }} ghcr.io/ystv/badger/$img:latest
docker push ghcr.io/ystv/badger/$img:latest
done
shell: bash
- name: Create GitHub release
uses: actions/github-script@v7
id: release
with:
github-token: ${{ secrets.GH_RELEASE_PAT }}
script: |
const release = await github.rest.repos.createRelease({
owner: context.repo.owner,
Expand All @@ -109,21 +179,44 @@ jobs:
name: process.env.VERSION,
draft: true,
generate_release_notes: true,
make_latest: true
prerelease: false,
body: [
'## Docker images',
`- Server: \`ghcr.io/ystv/badger/server:${process.env.VERSION}\``,
`- Jobrunner: \`ghcr.io/ystv/badger/jobrunner:${process.env.VERSION}\``,
''
].join('\n'),
});
core.setOutput('id', release.data.id)
core.setOutput('upload_url', release.data.upload_url)
core.setOutput('tag_name', release.data.tag_name)
- name: Upload artifacts
run: |
find artifacts -type f -exec gh release upload ${{ steps.release.outputs.tag_name }} {} \;
find artifacts -type f -not -name '*.yml' -not -name '*.yaml' -print0 | xargs -0 -I{} gh release -R ystv/badger upload --clobber ${{ steps.release.outputs.tag_name }} '{}'
env:
GITHUB_TOKEN: ${{ secrets.GH_RELEASE_PAT }}
- name: Publish release
uses: actions/github-script@v7
with:
github-token: ${{ secrets.GITHUB_TOKEN }}
script: |
await github.rest.repos.updateRelease({
owner: context.repo.owner,
repo: context.repo.repo,
release_id: ${{ steps.release.outputs.id }},
release_id: "${{ steps.release.outputs.id }}",
draft: false
})
- uses: actions/setup-node@v3
with:
node-version: "20.x"
- run: npm install @linear/sdk
- name: Close Linear issue
uses: actions/github-script@v7
with:
script: |
const { LinearClient } = require('@linear/sdk');
const lin = new LinearClient({
accessToken: "${{ secrets.LINEAR_ACCESS_TOKEN }}"
});
await lin.updateIssue("${{ needs.linear.outputs.issue_id }}", {
stateId: "${{ vars.LINEAR_DONE_STATE_ID }}",
});
Loading

0 comments on commit 6d6953b

Please sign in to comment.