Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add prebuild for macos arm64, enable prebuilds to be run individually. #2451

Open
wants to merge 15 commits into
base: prebuilds
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 6 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
104 changes: 98 additions & 6 deletions .github/workflows/prebuild.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,21 @@
# "edit" and delete any assets that are in purgatory (have a /!\ next to them).

name: Make Prebuilds
on: workflow_dispatch
on:
workflow_dispatch:
inputs:
os:
description: 'Target os'
required: true
type: choice
options:
- '["ubuntu-latest"]'
- '["windows-latest"]'
- '["macos-13"]'
- '["macos-latest"]'
- '["ubuntu-latest", "windows-latest", "macos-13", "macos-latest"]'
default: '["ubuntu-latest", "windows-latest", "macos-13", "macos-latest"]'


# UPLOAD_TO can be specified to upload the release assets under a different tag
# name (e.g. for testing). If omitted, the assets are published under the same
Expand All @@ -50,10 +64,11 @@ jobs:
strategy:
fail-fast: false
matrix:
node: [21]
node: [22]
btsimonh marked this conversation as resolved.
Show resolved Hide resolved
canvas_tag: ["v3.0.0-rc2"] # e.g. "v2.6.1"
name: ${{ matrix.canvas_tag }}, Node.js ${{ matrix.node }}, Linux
runs-on: ubuntu-latest
if: contains(fromJson(inputs.os), 'ubuntu-latest')
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

GitHub Actions docs on detecting the operating system: https://docs.github.com/en/actions/writing-workflows/choosing-what-your-workflow-does/store-information-in-variables#detecting-the-operating-system

- if: runner.os == 'Linux'
- if: runner.os == 'macOS'
- if: runner.os == 'Windows'
- if: runner.os != 'Windows'

container:
image: ${{ matrix.node < 18 && 'chearon/canvas-prebuilt:9' || 'zbbjornson/canvas-prebuilt:11' }}
env:
Expand Down Expand Up @@ -137,15 +152,90 @@ jobs:
name: assetName,
data: fs.readFileSync(assetName)
});
macOSarm:
strategy:
fail-fast: false
matrix:
node: [22]
btsimonh marked this conversation as resolved.
Show resolved Hide resolved
canvas_tag: ["v3.0.0-rc2"] # e.g. "v2.6.1"
name: ${{ matrix.canvas_tag}}, Node.js ${{ matrix.node }}, macOSarm64
btsimonh marked this conversation as resolved.
Show resolved Hide resolved
runs-on: macos-latest # macos-14+ is M1
if: contains(fromJson(inputs.os), 'macos-latest')
env:
CANVAS_VERSION_TO_BUILD: ${{ matrix.canvas_tag }}
steps:
- uses: actions/checkout@v2
btsimonh marked this conversation as resolved.
Show resolved Hide resolved
with:
ref: ${{ matrix.canvas_tag }}
# Fetch all commits/all branches so we can checkout the prebuild
# branch's files
fetch-depth: 0

- uses: actions/setup-node@v1
btsimonh marked this conversation as resolved.
Show resolved Hide resolved
with:
node-version: ${{ matrix.node }}

- name: Build
run: |
set -Eeuxo pipefail
git checkout ${{ matrix.canvas_tag }}
git checkout $GITHUB_SHA -- prebuild/
npm install -g node-gyp
npm install --ignore-scripts
. prebuild/macOS/preinstall.sh
cp prebuild/macOS/binding.gyp binding.gyp
node-gyp rebuild -j 2
. prebuild/macOS/bundle.sh

- name: Test binary
run: |
brew uninstall --force --ignore-dependencies cairo pango librsvg giflib harfbuzz
npm test

- name: Make bundle
id: make_bundle
run: . prebuild/tarball.sh

- name: Upload
uses: actions/github-script@v2
btsimonh marked this conversation as resolved.
Show resolved Hide resolved
with:
script: |
const fs = require("fs");
const assetName = "${{ steps.make_bundle.outputs.asset_name }}";
const tagName = process.env.UPLOAD_TO || process.env.CANVAS_VERSION_TO_BUILD;
const [owner, repo] = process.env.GITHUB_REPOSITORY.split("/");

const releases = await github.repos.listReleases({owner, repo});
const release = releases.data.find(r => r.tag_name === tagName);
if (!release)
throw new Error(`Tag ${tagName} not found. Did you make the GitHub release?`);

const oldAsset = release.assets.find(a => a.name === assetName);
if (oldAsset)
await github.repos.deleteReleaseAsset({owner, repo, asset_id: oldAsset.id});

// (This is equivalent to actions/upload-release-asset. We're
// already in a script, so might as well do it here.)
const r = await github.repos.uploadReleaseAsset({
url: release.upload_url,
headers: {
"content-type": "application/x-gzip",
"content-length": `${fs.statSync(assetName).size}`
},
name: assetName,
data: fs.readFileSync(assetName)
});


macOS:
strategy:
fail-fast: false
matrix:
node: [21]
node: [22]
btsimonh marked this conversation as resolved.
Show resolved Hide resolved
canvas_tag: ["v3.0.0-rc2"] # e.g. "v2.6.1"
name: ${{ matrix.canvas_tag}}, Node.js ${{ matrix.node }}, macOS
runs-on: macos-12 # macos-14+ is M1
runs-on: macos-13 # macos-14+ is M1
if: contains(fromJson(inputs.os), 'macos-13')
env:
CANVAS_VERSION_TO_BUILD: ${{ matrix.canvas_tag }}
steps:
Expand Down Expand Up @@ -211,14 +301,16 @@ jobs:
data: fs.readFileSync(assetName)
});


Win:
strategy:
fail-fast: false
matrix:
node: [21]
node: [22]
btsimonh marked this conversation as resolved.
Show resolved Hide resolved
canvas_tag: ["v3.0.0-rc2"] # e.g. "v2.6.1"
name: ${{ matrix.canvas_tag}}, Node.js ${{ matrix.node }}, Windows
runs-on: windows-2019
runs-on: windows-latest
if: contains(fromJson(inputs.os), 'windows-latest')
env:
CANVAS_VERSION_TO_BUILD: ${{ matrix.canvas_tag }}
steps:
Expand Down
95 changes: 94 additions & 1 deletion README.md
Original file line number Diff line number Diff line change
@@ -1 +1,94 @@
This is the prebuild branch. The only directory used in it is prebuild/. Otherwise files are taken from the branch being built.
This is the prebuild branch.

The only directory used in it is prebuild/.

Otherwise files are taken from the branch being built.

# Changes:

## 2024-11-01 btsimonh

Added the below notes.

Added macos arm64 prebuilds.



# notes from prebuild.yml:

```
# Release procedure:
# 1. On the master branch, update the changelog and stage it in git.
# 2. Run `yarn version --[major|minor|patch]`. (This tags the commit.)
# 3. Push both the commit and tags to GitHub (`git push --tags Automattic` and
# `git push Automattic HEAD:master`, where "Automattic" is the remote name
# for [email protected]:Automattic/node-canvas.git).
# 4. Create a draft release manually using the GitHub UI for the new tag.
# 5. Switch to the prebuilds branch.
# 6. Set the `jobs.*.strategy.matrix.node` arrays to the set of Node.js versions
# to build for.
# 7. Set the `jobs.*.strategy.matrix.canvas_tag` arrays to the set of Canvas
# tags to build. (Usually this is a single tag, but can be an array when a
# new version of Node.js is released and older versions of Canvas need to be
# built.)
# 8. Commit this file and push.
# 9. In the Actions tab, navigate to the "Make Prebuilds" workflow and click
# "Run workflow". Select the `prebuilds` branch so that that branch's
# prebuild.yaml file is used.
# 10.Once the builds succeed, promote the draft release to a full release and
# run npm publish.

# Note that the files in the prebuild/ directory will be used from the commit
# that was used to trigger the prebuild workflow. They will not be from the
# commit/tag that you are building. Because of differences in the provided
# versions of git, this is achieved in a different way on Linux than on Mac and
# Win.
#
# Pay particular attention to changes to binding.gyp on master. That file is not
# used! The ones in prebuild/**/ are.

# Tip: If uploads are inexplicably failing, open the release in GitHub, click
# "edit" and delete any assets that are in purgatory (have a /!\ next to them).
```

### Prebuilding only a specific platform:

When manually running `Make Prebuilds`, you should be asked for `Target os`.

Select the single target os OR the default multiple OSes.


### prebuilding for your own repo.

If you fork, and prebuild on your own repo, you will need to create a release based on a tag.

Prebuild WILL checkout the tag you specify in the cavnas tag property for each OS, and will build THAT tag from master - not your master HEAD or another branch.

e.g.:
```
canvas_tag: ["vdev"] # e.g. "v2.6.1"`
```

Prebuild will upload the output to YOUR repo into the release you created.

IF you want to USE your prebuilds, you would need to change (in package.json):
```
"repository": "git://github.com/Automattic/node-canvas.git",
```
and
```
"version": "3.0.0-rc2",
```
in the installed branch to reflect the repo you want to pull released prebuilds from,

and for version, this MUST match your tag without 'v' prefix (e.g. a version of 'dev' needs a tag of 'vdev')










btsimonh marked this conversation as resolved.
Show resolved Hide resolved
5 changes: 4 additions & 1 deletion prebuild/macOS/binding.gyp
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,8 @@
'<!@(pkg-config pangocairo --libs)',
'<!@(pkg-config freetype2 --libs)',
'<!@(pkg-config librsvg-2.0 --libs)',
'<!@(pkg-config libjpeg --libs)',
'-L/opt/homebrew/lib',
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Running brew --prefix or echo $HOMEBREW_PREFIX will generate /opt/homebrew on macOS ARM (Apple Silicon) CPUs but will generate /usr/local on Intel Macs. https://docs.brew.sh/Installation

We should support both paths.

'-ljpeg',
'-lgif'
],
Expand All @@ -43,7 +45,8 @@
'<!@(pkg-config libpng --cflags-only-I | sed s/-I//g)',
'<!@(pkg-config pangocairo --cflags-only-I | sed s/-I//g)',
'<!@(pkg-config freetype2 --cflags-only-I | sed s/-I//g)',
'<!@(pkg-config librsvg-2.0 --cflags-only-I | sed s/-I//g)'
'<!@(pkg-config librsvg-2.0 --cflags-only-I | sed s/-I//g)',
'/opt/homebrew/include'
],
'cflags+': ['-fvisibility=hidden'],
'xcode_settings': {
Expand Down
7 changes: 5 additions & 2 deletions prebuild/tarball.sh
Original file line number Diff line number Diff line change
@@ -1,8 +1,11 @@
# Generate the prebuild-install formatted filename from the node environment
# match prebuild-install style.
FILENAME=$(
node -e "
var p = process, v = p.versions, libc = require('detect-libc').familySync() || 'unknown';
if (libc === 'glibc') libc = '';
var p = process, v = p.versions;
const detectLibc = require('detect-libc');
const libc = process.env.LIBC || process.env.npm_config_libc ||
(detectLibc.isNonGlibcLinuxSync() && detectLibc.familySync()) || ''
const tagName = p.env.UPLOAD_TO || p.env.CANVAS_VERSION_TO_BUILD;
console.log('canvas-' + tagName + '-napi-v7-' + p.platform + libc + '-' + p.arch);
"
Expand Down