Skip to content

Commit

Permalink
feat: Support arm64 and amd64 builds of nushell
Browse files Browse the repository at this point in the history
  • Loading branch information
gmpinder committed Jan 4, 2025
1 parent 579f8f1 commit 0d51e64
Show file tree
Hide file tree
Showing 3 changed files with 73 additions and 19 deletions.
3 changes: 3 additions & 0 deletions .github/workflows/build.yml
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,9 @@ jobs:

- uses: docker/setup-buildx-action@v3

- name: Set up QEMU
uses: docker/setup-qemu-action@v2

- name: Login to GitHub Container Registry
uses: docker/login-action@v3
with:
Expand Down
10 changes: 6 additions & 4 deletions Containerfile
Original file line number Diff line number Diff line change
@@ -1,16 +1,18 @@
FROM ubuntu:20.04 as cached
FROM ubuntu:22.04 as cached

RUN apt-get update && apt-get install -y wget

FROM cached as build

ARG URL=https://github.com/nushell/nushell/releases/download/0.98.0/nu-0.98.0-x86_64-unknown-linux-gnu.tar.gz
ARG VERSION=0.99.1
ARG DOWNLOAD_ARCH=x86_64
ARG URL=https://github.com/nushell/nushell/releases/download/${VERSION}/nu-${VERSION}-${DOWNLOAD_ARCH}-unknown-linux-musl.tar.gz

RUN wget -O /tmp/archive.tar.gz "$URL" && \
RUN wget -q -O /tmp/archive.tar.gz "$URL" && \
mkdir /tmp/extract/ && \
tar -xzf /tmp/archive.tar.gz -C /tmp/extract/ && \
mv /tmp/extract/nu* /nu/

FROM scratch

COPY --from=build /nu/ /nu/
COPY --from=build /nu/ /nu/
79 changes: 64 additions & 15 deletions build.nu
100644 → 100755
Original file line number Diff line number Diff line change
Expand Up @@ -4,12 +4,31 @@
# version to be tagged with `default` and used by BlueBuild modules
const DEFAULT_VERSION = "0.99.1"

# architectures to build and the corresponding download arch string
const BUILD_ARCHS = [
{
docker: "linux/amd64",
download: "x86_64",
},
{
docker: "linux/arm64",
download: "aarch64",
}
]

print $"(ansi green_bold)Gathering images"

let images = http get https://api.github.com/repos/nushell/nushell/releases | enumerate | each { |arrayEl|
let release = $arrayEl.item

let url = ($release.assets | where name ends-with "x86_64-unknown-linux-musl.tar.gz").browser_download_url.0
if !($release.assets | any {|asset|
($BUILD_ARCHS | any {|arch|
$asset.name | str contains $"($arch.download)-unknown-linux-musl"
})
}) {
return
}

let version = $release.name

let tags = (
Expand All @@ -32,7 +51,6 @@ let images = http get https://api.github.com/repos/nushell/nushell/releases | en
print $"(ansi cyan)Found version & generated tags:(ansi reset) ($tags | str join ' ')"

{
url: $url
version: $version
tags: $tags
}
Expand All @@ -41,22 +59,53 @@ let images = http get https://api.github.com/repos/nushell/nushell/releases | en
print $"(ansi green_bold)Starting image build(ansi reset)"

$images | each { |img|
let base_image = $"($env.REGISTRY)/nushell-image"

print $"(ansi cyan)Building image for version:(ansi reset) ($img.version)"
(docker build .
-f ./Containerfile
...($img.tags | each { |tag| ["-t", $"($env.REGISTRY)/nushell-image:($tag)"] } | flatten) # generate and spread list of tags
--build-arg $"URL=($img.url)")
$BUILD_ARCHS | each { |arch|
print $"(ansi cyan)Building image for version:(ansi reset) ($img.version)"

}
let tag = $"($base_image):($img.version)-($arch.download)"

try {
(docker build .
-f ./Containerfile
--platform $arch.docker
-t $tag
--build-arg $"VERSION=($img.version)"
--build-arg $"DOWNLOAD_ARCH=($arch.download)")

print $"(ansi cyan)Pushing image for platform ($arch.download) and version ($img.version):(ansi reset) ($tag)"
docker push $tag
} catch {
print $"(ansi red_bold)Failed to build image(ansi reset) ($tag)"
exit 1
}
}

$img.tags | each { |tag|
let final_image = $"($base_image):($tag)"

try {
print $"(ansi cyan)Creating multi-platform manifest:(ansi reset) ($final_image)"
(docker manifest create $final_image
...($BUILD_ARCHS | each { |arch|
["--amend", $"($base_image):($img.version)-($arch.download)"]
} | flatten))

print $"(ansi cyan)Pushing multi-platform manifest:(ansi reset) ($final_image)"
docker manifest push $final_image

print $"(ansi cyan)Pushing images:(ansi reset)"
let digest = (
docker push --all-tags $"($env.REGISTRY)/nushell-image"
| split row "\n" | last | split row " " | get 2 # parse push output to get digest for signing
)
(docker manifest inspect $final_image | from json).manifests | each { |manifest|
let digest_image = $"($base_image)@($manifest.digest)"

print $"(ansi cyan)Signing image:(ansi reset) ($env.REGISTRY)/nushell-image@($digest)"
cosign sign -y --key env://COSIGN_PRIVATE_KEY $"($env.REGISTRY)/nushell-image@($digest)"
print $"(ansi cyan)Signing image:(ansi reset) ($digest_image)"
cosign sign -y --key env://COSIGN_PRIVATE_KEY $digest_image
}
} catch {
print $"(ansi red_bold)Failed to create and sign manifest(ansi reset) ($final_image)"
exit 1
}
}
}

print $"(ansi green_bold)DONE!(ansi reset)"

0 comments on commit 0d51e64

Please sign in to comment.