gpkg/libacl: build to apt #55
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 gpkg | |
on: | |
push: | |
branches: | |
- main | |
paths: | |
- 'gpkg/**' | |
pull_request: | |
paths: | |
- 'gpkg/**' | |
workflow_dispatch: | |
inputs: | |
packages: | |
description: "A space-separated names of packages selected for rebuilding" | |
required: true | |
permissions: {} # none | |
jobs: | |
build: | |
permissions: | |
contents: read # actions/upload-artifact doesn't need contents: write | |
runs-on: ubuntu-22.04 | |
strategy: | |
matrix: | |
target_arch: [aarch64, arm, i686, x86_64] | |
fail-fast: false | |
steps: | |
- name: Clone repository | |
uses: actions/checkout@v3 | |
with: | |
fetch-depth: 1000 | |
- name: Set up QEMU | |
uses: docker/[email protected] | |
- name: Get files from termux-packages repository | |
run: ./get-build-package.sh | |
- name: Gather build summary | |
run: | | |
if [ "${{ github.event_name }}" != "workflow_dispatch" ]; then | |
BASE_COMMIT=$(jq --raw-output .pull_request.base.sha "$GITHUB_EVENT_PATH") | |
OLD_COMMIT=$(jq --raw-output .commits[0].id "$GITHUB_EVENT_PATH") | |
HEAD_COMMIT=$(jq --raw-output .commits[-1].id "$GITHUB_EVENT_PATH") | |
if [ "$BASE_COMMIT" = "null" ]; then | |
if [ "$OLD_COMMIT" = "$HEAD_COMMIT" ]; then | |
# Single-commit push. | |
echo "Processing commit: ${HEAD_COMMIT}" | |
CHANGED_FILES=$(git diff-tree --no-commit-id --name-only -r "${HEAD_COMMIT}") | |
else | |
# Multi-commit push. | |
OLD_COMMIT="${OLD_COMMIT}~1" | |
echo "Processing commit range: ${OLD_COMMIT}..${HEAD_COMMIT}" | |
CHANGED_FILES=$(git diff-tree --no-commit-id --name-only -r "${OLD_COMMIT}" "${HEAD_COMMIT}") | |
fi | |
else | |
# Pull requests. | |
echo "Processing pull request #$(jq --raw-output .pull_request.number "$GITHUB_EVENT_PATH"): ${BASE_COMMIT}..HEAD" | |
CHANGED_FILES=$(git diff-tree --no-commit-id --name-only -r "${BASE_COMMIT}" "HEAD") | |
fi | |
fi | |
mkdir -p ./artifacts ./debs | |
touch ./debs/.placeholder | |
if [ "${{ github.event_name }}" != "workflow_dispatch" ]; then | |
# Process tag '%ci:no-build' that may be added as line to commit message. | |
# Forces CI to cancel current build with status 'passed' | |
if grep -qiP '^\s*%ci:no-build\s*$' <(git log --format="%B" -n 1 "HEAD"); then | |
tar cf artifacts/debs-${{ matrix.target_arch }}.tar debs | |
echo "[!] Force exiting as tag '%ci:no-build' was applied to HEAD commit message." | |
exit 0 | |
fi | |
# Build local Docker image if setup scripts were changed. | |
# Useful for pull requests submitting changes for both build environment and packages. | |
if grep -qP '^scripts/(Dockerfile|properties\.sh|setup-android-sdk\.sh|setup-cgct\.sh|setup-ubuntu\.sh)$' <<< "$CHANGED_FILES"; then | |
echo "Detected changes for environment setup scripts. Building custom Docker image now." | |
docker build -t ghcr.io/termux/package-builder-cgct:latest . | |
fi | |
for repo_path in $(jq --raw-output 'del(.pkg_format) | keys | .[]' repo.json); do | |
repo=$(jq --raw-output '.["'${repo_path}'"].name' repo.json) | |
# Parse changed files and identify new packages and deleted packages. | |
# Create lists of those packages that will be passed to upload job for | |
# further processing. | |
while read -r file; do | |
if ! [[ $file == ${repo_path}/* ]]; then | |
# This file does not belong to a package, so ignore it | |
continue | |
fi | |
if [[ $file =~ ^${repo_path}/([.a-z0-9+-]*)/([.a-z0-9+-]*).subpackage.sh$ ]]; then | |
# A subpackage was modified, check if it was deleted or just updated | |
pkg=${BASH_REMATCH[1]} | |
subpkg=${BASH_REMATCH[2]} | |
if [ ! -f "${repo_path}/${pkg}/${subpkg}.subpackage.sh" ]; then | |
echo "$subpkg" >> ./deleted_${repo}_packages.txt | |
fi | |
elif [[ $file =~ ^${repo_path}/([.a-z0-9+-]*)/.*$ ]]; then | |
# package, check if it was deleted or updated | |
pkg=${BASH_REMATCH[1]} | |
if [ -d "${repo_path}/${pkg}" ]; then | |
echo "$pkg" >> ./built_${repo}_packages.txt | |
# If there are subpackages we want to create a list of those | |
# as well | |
for file in $(find "${repo_path}/${pkg}/" -maxdepth 1 -type f -name \*.subpackage.sh | sort); do | |
echo "$(basename "${file%%.subpackage.sh}")" >> ./built_${repo}_subpackages.txt | |
done | |
else | |
echo "$pkg" >> ./deleted_${repo}_packages | |
fi | |
fi | |
done<<<${CHANGED_FILES} | |
done | |
else | |
for pkg in ${{ github.event.inputs.packages }}; do | |
repo_paths=$(jq --raw-output 'del(.pkg_format) | keys | .[]' repo.json) | |
found=false | |
for repo_path in $repo_paths; do | |
repo=$(jq --raw-output '.["'${repo_path}'"].name' repo.json) | |
if [ -d "${repo_path}/${pkg}" ]; then | |
found=true | |
echo "$pkg" >> ./built_${repo}_packages.txt | |
for subpkg in $(find "${repo_path}/${pkg}/" -maxdepth 1 -type f -name \*.subpackage.sh | sort); do | |
echo "$(basename "${subpkg%%.subpackage.sh}")" >> ./built_${repo}_subpackages.txt | |
done | |
fi | |
done | |
if [ "$found" != true ]; then | |
echo "Package '${pkg}' not found in any of the repo" | |
exit 1 | |
fi | |
done | |
fi | |
for repo in $(jq --raw-output 'del(.pkg_format) | .[].name' repo.json); do | |
# Fix so that lists do not contain duplicates | |
if [ -f ./built_${repo}_packages.txt ]; then | |
sort ./built_${repo}_packages.txt | uniq > ./built_${repo}_packages.txt.tmp | |
mv ./built_${repo}_packages.txt.tmp ./built_${repo}_packages.txt | |
fi | |
if [ -f ./built_${repo}_subpackages.txt ]; then | |
sort ./built_${repo}_subpackages.txt | uniq > ./built_${repo}_subpackages.txt.tmp | |
mv ./built_${repo}_subpackages.txt.tmp ./built_${repo}_subpackages.txt | |
fi | |
if [ -f ./deleted_${repo}_packages.txt ]; then | |
sort ./deleted_${repo}_packages.txt | uniq > ./deleted_${repo}_packages.txt.tmp | |
mv ./deleted_${repo}_packages.txt.tmp ./deleted_${repo}_packages.txt | |
fi | |
done | |
- name: Lint packages | |
run: | | |
declare -a package_recipes | |
for repo_path in $(jq --raw-output 'del(.pkg_format) | keys | .[]' repo.json); do | |
repo=$(jq --raw-output '.["'${repo_path}'"].name' repo.json) | |
if [ -f ./built_${repo}_packages.txt ]; then | |
package_recipes="$package_recipes $(cat ./built_${repo}_packages.txt | repo_path=${repo_path} awk '{print ENVIRON["repo_path"]"/"$1"/build.sh"}')" | |
fi | |
done | |
if [ ! -z "$package_recipes" ]; then | |
./scripts/lint-packages.sh $package_recipes | |
fi | |
- name: Build packages | |
run: | | |
declare -a packages | |
for repo_path in $(jq --raw-output 'del(.pkg_format) | keys | .[]' repo.json); do | |
repo=$(jq --raw-output '.["'${repo_path}'"].name' repo.json) | |
if [ -f ./built_${repo}_packages.txt ]; then | |
packages="$packages $(cat ./built_${repo}_packages.txt)" | |
fi | |
done | |
if [ ! -z "$packages" ]; then | |
if grep -qP "(^|\s)${packages// /($|\s)|(^|\s)}($|\s)" ./big-pkgs.list; then | |
./scripts/setup-ubuntu.sh | |
./scripts/setup-cgct.sh | |
sudo apt install ninja-build | |
sudo apt purge -yq $(dpkg -l | grep '^ii' | awk '{ print $2 }' | grep -P '(aspnetcore|cabal-|dotnet-|ghc-|libmono|php)') \ | |
firefox google-chrome-stable microsoft-edge-stable mono-runtime-common monodoc-manual ruby | |
sudo apt autoremove -yq | |
sudo rm -fr /opt/hostedtoolcache /usr/share/dotnet /usr/share/swift | |
NDK=$ANDROID_NDK_LATEST_HOME ANDROID_HOME=$ANDROID_SDK_ROOT ./build-package.sh -I -a ${{ matrix.target_arch }} --library glibc $packages | |
else | |
TERMUX_BUILDER_IMAGE_NAME="ghcr.io/termux/package-builder-cgct" ./scripts/run-docker.sh ./build-package.sh -I -a ${{ matrix.target_arch }} --library glibc $packages | |
fi | |
fi | |
- name: Generate build artifacts | |
if: always() | |
run: | | |
for repo in $(jq --raw-output 'del(.pkg_format) | .[].name' repo.json); do | |
# Put package lists into directory with *.pkg.* files so they will be transferred to | |
# upload job. | |
test -f ./built_${repo}_packages.txt && mv ./built_${repo}_packages.txt ./debs/ | |
test -f ./built_${repo}_subpackages.txt && cat ./built_${repo}_subpackages.txt >> ./debs/built_${repo}_packages.txt \ | |
&& rm ./built_${repo}_subpackages.txt | |
test -f ./deleted_${repo}_packages.txt && mv ./deleted_${repo}_packages.txt ./debs/ | |
# Move only pkgs from built_packages into pkgs/ folder before | |
# creating an archive. | |
while read -r pkg; do | |
# Match both $pkg.pkg.* and $pkg-static.pkg.*. | |
find output \( -name "$pkg_*.deb" -o -name "$pkg-static_*.deb" \) -type f -print0 | xargs -0r mv -t debs/ | |
done < <(cat ./debs/built_${repo}_packages.txt) | |
done | |
# Files containing certain symbols (e.g. ":") will cause failure in actions/upload-artifact. | |
# Archiving *.deb files in a tarball to avoid issues with uploading. | |
tar cf artifacts/debs-${{ matrix.target_arch }}-${{ github.sha }}.tar debs | |
- name: Checksums for built *.deb files | |
if: always() | |
run: | | |
find debs -type f -name "*.deb" -exec sha256sum "{}" \; | sort -k2 | |
- name: Store *.deb files | |
if: always() | |
uses: actions/upload-artifact@v3 | |
with: | |
name: debs-${{ matrix.target_arch }}-${{ github.sha }} | |
path: ./artifacts | |
upload: | |
concurrency: ${{ github.workflow }} | |
permissions: | |
contents: read | |
if: github.event_name != 'pull_request' | |
needs: build | |
runs-on: ubuntu-latest | |
steps: | |
- name: Clone repository | |
uses: actions/checkout@v4 | |
- name: Get *.deb files | |
uses: actions/download-artifact@v3 | |
with: | |
path: ./ | |
- name: Get files from termux-packages repository | |
run: ./get-build-package.sh | |
- name: Upload to packages.termux.dev | |
env: | |
REPOSITORY_URL: https://packages.termux.dev/aptly-api | |
run: | | |
GITHUB_SHA=${{ github.sha }} | |
APTLY_API_AUTH=${{ secrets.APTLY_API_AUTH }} | |
GPG_PASSPHRASE=${{ secrets.GPG_PASSPHRASE }} | |
source scripts/aptly_api.sh | |
#source scripts/utils/package/package.sh | |
for archive in debs-*/debs-{aarch64,arm,i686,x86_64}-${{ github.sha }}.tar; do | |
tar xf "$archive" | |
done | |
for repo in $(jq --raw-output 'del(.pkg_format) | keys | .[]' repo.json); do | |
export REPOSITORY_NAME=$(jq --raw-output '.["'$repo'"].name' repo.json) | |
export REPOSITORY_DISTRIBUTION=$(jq --raw-output '.["'$repo'"].distribution' repo.json) | |
# Upload file to temporary directory. | |
uploaded_files=false | |
shopt -s nullglob | |
if [ -f debs/built_${REPOSITORY_NAME}_packages.txt ]; then | |
for filename in $(cat debs/built_${REPOSITORY_NAME}_packages.txt | sed -E 's/(..*)/\1\n\1-static/g' | awk '{system("bash -c '\''source scripts/utils/package/package.sh; ls debs/$(package__is_package_name_have_glibc_prefix "$1" && echo "$1" || package__add_prefix_glibc_to_package_name "$1")_*.deb 2>/dev/null'\''")}'); do | |
if ! aptly_upload_file "$filename"; then | |
exit 1 | |
fi | |
uploaded_files=true | |
done | |
shopt -u nullglob | |
# Publishing repository changes. | |
if [ "$uploaded_files" = "true" ]; then | |
#if ! aptly_add_to_repo; then | |
# exit 1 | |
#fi | |
aptly_add_to_repo || true | |
# Usually temporary directory is deleted automatically, but in certain cases it is left. | |
aptly_delete_dir | |
# Final part to make changes appear in web root. | |
if ! aptly_publish_repo; then | |
exit 1 | |
fi | |
fi | |
fi | |
done |