Build and upload Mac app artifact #178
Workflow file for this run
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
name: Build and upload Mac app artifact | |
on: | |
workflow_dispatch: | |
inputs: | |
buildBranch: | |
description: 'Headlamp ref/branch/tag' | |
required: true | |
default: 'main' | |
signBinaries: | |
description: Notarize app | |
default: true | |
type: boolean | |
permissions: | |
contents: read | |
jobs: | |
build-mac: | |
runs-on: macos-latest | |
permissions: | |
contents: read | |
actions: write # needed to upload artifacts | |
steps: | |
- uses: actions/checkout@692973e3d937129bcbf40652eb9f2f61becf3332 # v4.1.7 | |
with: | |
ref: ${{ github.event.inputs.buildBranch }} | |
- name: Setup nodejs | |
uses: actions/setup-node@60edb5dd545a775178f52524783378180af0d1f8 # v4.0.2 | |
with: | |
node-version: 20.x | |
cache: 'npm' | |
cache-dependency-path: | | |
app/package-lock.json | |
frontend/package-lock.json | |
- uses: actions/setup-go@cdcb36043654635271a94b9a6d1392de5bb323a7 # v5.0.1 | |
with: | |
go-version: '1.22.*' | |
cache-dependency-path: | | |
backend/go.sum | |
- name: Dependencies | |
run: brew install make | |
- name: Build Backend and Frontend | |
run: | | |
make | |
- name: Add MacOS certs | |
run: cd ./app/mac/scripts/ && sh ./setup-certificate.sh | |
env: | |
APPLE_CERTIFICATE: ${{ secrets.APPLE_CERTIFICATE }} | |
APPLE_CERTIFICATE_PASSWORD: ${{ secrets.APPLE_CERTIFICATE_PASSWORD }} | |
- name: Build App Mac | |
if: ${{ inputs.signBinaries }} | |
env: | |
# This will trigger codesign. See app/mac/scripts/codeSign.js | |
APPLE_TEAM_ID: ${{ secrets.APPLE_TEAM_ID }} | |
run: | | |
make app-mac | |
- name: Early staple (only if we are not notarizing the app) | |
if: ${{ ! inputs.signBinaries }} | |
run: | | |
xcrun stapler staple ./app/dist/Headlamp*.dmg | |
- name: Upload artifact | |
uses: actions/upload-artifact@65462800fd760344b1a7b4382951275a0abb4808 # v4.3.3 | |
with: | |
name: dmgs | |
path: ./app/dist/Headlamp*.dmg | |
if-no-files-found: error | |
retention-days: 1 | |
notarize: | |
permissions: | |
id-token: write # For fetching an OpenID Connect (OIDC) token | |
contents: read | |
runs-on: windows-latest | |
needs: build-mac | |
if: ${{ inputs.signBinaries }} | |
steps: | |
- uses: actions/checkout@692973e3d937129bcbf40652eb9f2f61becf3332 # v4.1.7 | |
with: | |
ref: ${{ github.event.inputs.buildBranch }} | |
- name: Setup nodejs | |
uses: actions/setup-node@60edb5dd545a775178f52524783378180af0d1f8 # v4.0.2 | |
with: | |
node-version: 18.x | |
cache: 'npm' | |
cache-dependency-path: | | |
app/package-lock.json | |
frontend/package-lock.json | |
- name: Download artifact | |
uses: actions/download-artifact@65a9edc5881444af0b9093a5e628f2fe47ea3b2e # v4.1.7 | |
with: | |
name: dmgs | |
path: ./dmgs | |
- name: Azure login | |
if: ${{ inputs.signBinaries }} | |
uses: azure/login@6c251865b4e6290e7b78be643ea2d005bc51f69a # v2.1.1 | |
with: | |
client-id: ${{ secrets.WINDOWS_CLIENT_ID }} | |
tenant-id: ${{ secrets. AZ_TENANT_ID }} | |
subscription-id: ${{ secrets.AZ_SUBSCRIPTION_ID }} | |
- name: Fetch certificates | |
if: ${{ inputs.signBinaries }} | |
shell: pwsh | |
run: | | |
az keyvault secret download --subscription ${{ secrets.AZ_SUBSCRIPTION_ID }} --vault-name headlamp --name HeadlampAuthCert --file c:\HeadlampAuthCert.pfx --encoding base64 | |
az keyvault secret download --subscription ${{ secrets.AZ_SUBSCRIPTION_ID }} --vault-name headlamp --name ESRPHeadlampReqCert --file c:\HeadlampReqCert.pfx --encoding base64 | |
- name: Set up certificates | |
if: ${{ inputs.signBinaries }} | |
shell: pwsh | |
run: | | |
Import-PfxCertificate -FilePath c:\HeadlampAuthCert.pfx -CertStoreLocation Cert:\LocalMachine\My -Exportable | |
Import-PfxCertificate -FilePath c:\HeadlampReqCert.pfx -CertStoreLocation Cert:\LocalMachine\My -Exportable | |
- name: Download and Set up ESRPClient | |
if: ${{ inputs.signBinaries }} | |
shell: pwsh | |
run: | | |
nuget.exe sources add -name esrp -source ${{ secrets.ESRP_NUGET_INDEX_URL }} -username headlamp -password ${{ secrets.AZ_DEVOPS_TOKEN }} | |
nuget.exe install Microsoft.EsrpClient -Version 1.2.87 -source ${{ secrets.ESRP_NUGET_INDEX_URL }} | out-null | |
- name: Sign App | |
shell: pwsh | |
run: | | |
$env:ESRP_PATH="$(Get-Location)\Microsoft.EsrpClient.1.2.87\tools\EsrpClient.exe" | |
$env:HEADLAMP_WINDOWS_CLIENT_ID="${{ secrets.WINDOWS_CLIENT_ID }}" | |
$env:HEADLAMP_WINDOWS_SIGN_EMAIL="${{ secrets.WINDOWS_SIGN_EMAIL }}" | |
cd ./app/scripts | |
node ./esrp.js apple-sign ../../dmgs/ | |
- name: Notarize App | |
shell: pwsh | |
run: | | |
$env:ESRP_PATH="$(Get-Location)\Microsoft.EsrpClient.1.2.87\tools\EsrpClient.exe" | |
$env:HEADLAMP_WINDOWS_CLIENT_ID="${{ secrets.WINDOWS_CLIENT_ID }}" | |
$env:HEADLAMP_WINDOWS_SIGN_EMAIL="${{ secrets.WINDOWS_SIGN_EMAIL }}" | |
cd ./app/scripts | |
node ./esrp.js apple-notarize ../../dmgs/ | |
- name: Upload Notarized | |
uses: actions/upload-artifact@65462800fd760344b1a7b4382951275a0abb4808 # v4.3.3 | |
with: | |
name: dmgs | |
path: ./dmgs/Headlamp*.dmg | |
if-no-files-found: error | |
overwrite: true | |
retention-days: 2 | |
verify-notarization: | |
runs-on: macos-latest | |
needs: notarize | |
permissions: | |
actions: write # for downloading and uploading artifacts | |
contents: read | |
if: ${{ inputs.signBinaries }} | |
strategy: | |
matrix: | |
arch: [x86, arm64] | |
steps: | |
- name: Download artifact | |
uses: actions/download-artifact@65a9edc5881444af0b9093a5e628f2fe47ea3b2e # v4.1.7 | |
with: | |
name: dmgs | |
path: ./dmgs | |
- name: Verify Notarization | |
run: | | |
cd ./dmgs | |
# Map x86 to x64 | |
ARCH=${{ matrix.arch }} | |
if [ "$ARCH" = "x86" ]; then | |
ARCH="x64" | |
fi | |
echo "Verifying notarization of the app: $(ls ./Headlamp*${ARCH}*.dmg)" | |
MOUNT_OUTPUT="$(hdiutil attach ./Headlamp*${ARCH}*.dmg)" | |
echo "MOUNT_OUTPUT: $MOUNT_OUTPUT" | |
VOLUME_NAME="$(echo "$MOUNT_OUTPUT" | grep -o '/Volumes/[^\s]*')" | |
# Check if the app is notarized | |
echo "CHECK VOL: $VOLUME_NAME" | |
spctl -a -v "$VOLUME_NAME/Headlamp.app/Contents/MacOS/Headlamp" | |
echo "CHECK SYMLINKS!" | |
# Check if the app has symlinks | |
SYMLINKS=$(find "$VOLUME_NAME" -type l -ls || true) | |
NODE_MODULES_AS_SYMLINKS=$(echo "$SYMLINKS" | grep node_modules || true) | |
echo "CHECK SYMLINKS: $SYMLINKS $NODE_MODULES_AS_SYMLINKS" | |
if [ -n "$NODE_MODULES_AS_SYMLINKS" ]; then | |
echo "Symlinks found in the DMG:" | |
echo "$NODE_MODULES_AS_SYMLINKS" | |
exit 1 | |
else | |
echo "No symlinks found in the DMG" | |
fi | |
hdiutil detach "$VOLUME_NAME" || true | |
exit 0 | |
stapler: | |
runs-on: macos-latest | |
needs: verify-notarization | |
permissions: | |
actions: write # for downloading and uploading artifacts | |
contents: read | |
if: ${{ inputs.signBinaries }} | |
steps: | |
- name: Download artifact | |
uses: actions/download-artifact@65a9edc5881444af0b9093a5e628f2fe47ea3b2e # v4.1.7 | |
with: | |
name: dmgs | |
path: ./dmgs | |
- name: Staple | |
run: | | |
xcrun stapler staple ./dmgs/Headlamp*.dmg | |
- name: Upload Stapled | |
uses: actions/upload-artifact@65462800fd760344b1a7b4382951275a0abb4808 # v4.3.3 | |
with: | |
name: dmgs | |
path: ./dmgs/Headlamp*.dmg | |
if-no-files-found: error | |
overwrite: true | |
retention-days: 2 |