-
Notifications
You must be signed in to change notification settings - Fork 24
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
POC pushing & pulling trunk from an OSI registry
Modify `trunk.mk` to build JSON files required to build [OCI image manifests] and an [OCI image index], including new `make` variables `TITLE`, `DESCRIPTION`, `VENDOR`, `URL`, and `REPO_URL`. Add `push_trunk`, a shell script to take those JSON files and use `oras` to build the images and manifests and push them to a registry. Add `docker_compose.yml` to spin up a PGXN tools image for AMD64 and running Postgres 16, as well as [zot], a vendor-neutral OCI registry. Finally, modify `install_trunk` to download an artifact from a registry address, rather than build from a local file. Use `oras` to fetch the image for the current platform. There's no support for distinguishing between artifacts built for different Postgres versions, but there are annotations in the image index that would allow it. [OCI image manifests]: https://github.com/opencontainers/image-spec/blob/main/manifest.md [OCI image index]: https://github.com/opencontainers/image-spec/blob/main/image-index.md [zot]: https://github.com/project-zot/zot
- Loading branch information
Showing
6 changed files
with
243 additions
and
6 deletions.
There are no files selected for viewing
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
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
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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,57 @@ | ||
version: "3.9" | ||
name: trunking | ||
|
||
# Start a Zot service for pushing and pulling OCI images, and a | ||
# pgxn/pgxn-tools image for building and testing Linux/AMD binaries. | ||
# | ||
# docker compose up -d | ||
# docker compose exec linux bash | ||
# make clean && make && make trunk | ||
# | ||
# On macOS: | ||
# make clean && make && make trunk | ||
# ./push_trunk localhost:5000/theory/semver semver-0.32.1+pg16-darwin-23.5.0-arm64 semver-0.32.1+pg16-linux-amd64 | ||
# clean -dfx --exclude=.vscode | ||
# find "$(pg_config --sharedir)" "$(pg_config --pkglibdir)" "$(pg_config --docdir)" -name '*semver*' -exec rm -rf "{}" \; | ||
# | ||
# ./install_trunk localhost:5000/theory/semver:v1 | ||
# find "$(pg_config --sharedir)" "$(pg_config --pkglibdir)" "$(pg_config --docdir)" -name '*semver*' | ||
# | ||
# Back on Linux | ||
# ./install_trunk zot:5000/theory/semver:v1 | ||
# find "$(pg_config --sharedir)" "$(pg_config --pkglibdir)" "$(pg_config --docdir)" -name '*semver*' | ||
|
||
# Name the network for all of the services to join. | ||
networks: | ||
default: | ||
name: pgxnnet | ||
|
||
services: | ||
zot: | ||
image: ghcr.io/project-zot/zot-linux-arm64:latest | ||
container_name: zot | ||
ports: | ||
- 5000:5000 | ||
hostname: zot | ||
|
||
linux: | ||
image: pgxn/pgxn-tools | ||
platform: "linux/amd64" | ||
container_name: linux | ||
hostname: linux | ||
working_dir: /work | ||
volumes: | ||
- ".:/work" | ||
# Install oras, then start Postgres 16 and install rsync and jq | ||
entrypoint: | ||
- "/bin/sh" | ||
- -ecx | ||
- | | ||
cd /tmp | ||
curl -LO "https://github.com/oras-project/oras/releases/download/v1.2.0/oras_1.2.0_linux_amd64.tar.gz" | ||
mkdir -p oras-install/ | ||
tar -zxf oras_1.2.0_*.tar.gz -C oras-install/ | ||
sudo mv oras-install/oras /usr/local/bin/ | ||
rm -rf oras_1.2.0_*.tar.gz oras-install/ | ||
pg-start 16 rsync jq | ||
tail -f /dev/null |
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
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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,114 @@ | ||
#!/bin/bash | ||
|
||
# POC for publishing Trunk packages to an OCI repository with an image index | ||
# to allow pulling a platform-specific binary. Requires: | ||
# | ||
# * oras | ||
# * jq | ||
# * zot: docker run -d -p 5000:5000 --name oras-quickstart ghcr.io/project-zot/zot-linux-arm64:latest | ||
# | ||
# Inspired by the Homebrew implementation of the pattern as referenced in | ||
# https://github.com/oras-project/oras/issues/237, and with thanks to the | ||
# denizens of the #oras and #zot channels on the CNCF Slack. | ||
|
||
trap 'exit' ERR | ||
set -E | ||
|
||
# OCI types to create. | ||
ARTIFACT_TYPE=application/vnd.pgxn.trunk.layer.v1 | ||
MEDIA_TYPE=application/vnd.oci.image.layer.v1.tar+gzip | ||
CONFIG_TYPE=application/vnd.oci.image.config.v1+json | ||
OCI_DIR=oci_dir | ||
INDEX_FILE=image_index.json | ||
|
||
push_image() { | ||
# Push the image into the OCI layout directory $OCI_DIR. | ||
local trunk=$1 | ||
oras push --no-tty \ | ||
--oci-layout "$OCI_DIR" \ | ||
--artifact-type "${ARTIFACT_TYPE}" \ | ||
--config "${trunk}_config.json":"$CONFIG_TYPE" \ | ||
--format go-template='{{.digest}}' \ | ||
--annotation-file "${trunk}_annotations.json" \ | ||
"$trunk.trunk":"$MEDIA_TYPE" | ||
} | ||
|
||
make_manifest() { | ||
local trunk=$1 | ||
local digest=$2 | ||
local anno platform | ||
|
||
# Extract just the pgxn.trunk annotations. | ||
anno=$(jq -c \ | ||
'.["$manifest"] | with_entries(select(.key | startswith("org.pgxn.trunk.")))' \ | ||
"${trunk}_annotations.json" | ||
) | ||
|
||
# Extract the platform config. | ||
platform=$(jq \ | ||
'pick(.os, .["os.version"], .architecture)| with_entries(select(.value |. !=null and . != ""))' \ | ||
"$trunk"_config.json | ||
) | ||
|
||
# Create and return the image manifest. | ||
oras manifest fetch --oci-layout "$OCI_DIR@${digest}" --descriptor \ | ||
| jq --argjson anno "$anno" --argjson platform "$platform" \ | ||
'{ | ||
mediaType: .mediaType, | ||
size: .size, | ||
digest: .digest, | ||
platform: $platform, | ||
annotations: $anno | ||
}' | ||
} | ||
|
||
write_index() { | ||
darwin_manifest=$1 | ||
linux_manifest=$2 | ||
|
||
# Build the image index with the two manifests. | ||
jq -n --argjson linux "$linux_manifest" \ | ||
--argjson darwin "$darwin_manifest" \ | ||
--argjson annotations "$(cat semver_annotations.json)" \ | ||
'{ | ||
schemaVersion: 2, | ||
mediaType: "application/vnd.oci.image.index.v1+json", | ||
manifests: [$linux, $darwin], | ||
annotations: $annotations | ||
}' > "$INDEX_FILE" | ||
} | ||
|
||
push_trunk() { | ||
# Only testing for Darwin and Linux rn. | ||
local repo=$1 | ||
local darwin_trunk=$2 | ||
local linux_trunk=$3 | ||
|
||
if [ -z "$repo" ] || [ -z "$darwin_trunk" ] || [ -z "$linux_trunk" ]; then | ||
printf "Usage:\n\n %s REPO DARWIN.trunk LINUX.trunk\n" "$0" | ||
exit 1 | ||
fi | ||
|
||
# Push the images and grab the resulting digests. | ||
darwin_digest=$(push_image "$darwin_trunk") | ||
linux_digest=$(push_image "$linux_trunk") | ||
|
||
# Create the image manifests. | ||
darwin_manifest=$(make_manifest "$darwin_trunk" "$darwin_digest") | ||
linux_manifest=$(make_manifest "$linux_trunk" "$linux_digest") | ||
|
||
# Write out and push the image index. | ||
write_index "$darwin_manifest" "$linux_manifest" | ||
oras manifest push --oci-layout ./"$OCI_DIR":image-index "$INDEX_FILE" | ||
|
||
# Push everything from the local layout to the remote registry. | ||
oras cp --from-oci-layout ./"$OCI_DIR":image-index --to-plain-http "${repo}:v1" | ||
|
||
# View the remote image index manifest. | ||
oras manifest get --plain-http "${repo}:v1" | jq | ||
|
||
# Cleanup. | ||
rm -rf "$OCI_DIR" "$INDEX_FILE" | ||
} | ||
|
||
push_trunk "$@" |
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