Skip to content

Sign binaries with SignPath #69

Sign binaries with SignPath

Sign binaries with SignPath #69

Workflow file for this run

name: Build
on:
pull_request:
paths:
- .github/workflows/build.yml
- build-exe/**
push:
branches:
- main
tags-ignore:
- "**"
paths:
- .github/workflows/build.yml
- build-exe/**
workflow_dispatch:
env:
CLDR_VERSION: 45
ICONV_VERSION: 1.17
GETTEXT_VERSION: 0.22.5a
# To be used for testing: we'll only build iconv
#BUILD_ONLY_ICONV: y
jobs:
exe:
name: Build executables ${{ matrix.link }} ${{ matrix.bits}} bits
runs-on: windows-2022
strategy:
matrix:
bits:
- 32
- 64
link:
- shared
- static
env:
CYGWIN_NOWINPATH: 1
CHERE_INVOKING: 1
defaults:
run:
shell: C:\cygwin\bin\bash.exe --login -o igncr -o errexit -o pipefail {0}
steps:
-
name: Configure git
shell: cmd
run: git config --global core.autocrlf input
-
name: Checkout
uses: actions/checkout@v4
-
name: Restore cache
id: restore-cache
uses: actions/cache/restore@v4
with:
key: ${{ matrix.link }}-${{ matrix.bits }}
path: |
src\downloads
C:\cygwin-packages
-
name: Set variables
id: vars
shell: pwsh
run: ./build-exe/vars.ps1 -Bits ${{ matrix.bits }} -Link ${{ matrix.link }}
-
name: Download Cygwin installer
shell: pwsh
run: Invoke-WebRequest -Uri https://cygwin.com/setup-x86_64.exe -OutFile C:\CygwinInstaller.exe
-
name: Install Cygwin
shell: cmd
run: >
C:\CygwinInstaller.exe
--root C:\cygwin
--local-package-dir C:\cygwin-packages
--packages ${{ steps.vars.outputs.cygwin-packages }}
--site http://mirrors.kernel.org/sourceware/cygwin/
--only-site
--quiet-mode
--upgrade-also
--no-shortcuts
--no-admin
-
name: Setup Cygwin environment
run: |
cd -- "$(/bin/cygpath -au '${{ github.workspace }}')"
/bin/printf 'Working directory: %s\n' "$(/bin/pwd)"
/bin/mkdir -p src/downloads
/bin/mkdir installed
/bin/mkdir files-unsigned
/bin/mkdir files-signed
/bin/mkdir installer-unsigned
/bin/mkdir installer-signed
/bin/perl -pe 's/\r\n|\n|\r/\r\n/g' < build-exe/license-for-distribution.txt > files-unsigned/license.txt
INSTALLED_PATH="$(/bin/cygpath -au ./installed)"
/bin/echo "INSTALLED_PATH=$INSTALLED_PATH" >> $GITHUB_ENV
/bin/mkdir -p "$HOME"
/bin/printf '\nPATH=%s/bin:${{ steps.vars.outputs.cygwin-path }}\nexport PATH\n' "$INSTALLED_PATH" >>$HOME/.bash_profile
-
name: Dump Cygwin PATH
run: |
IFS=:
for p in $PATH; do
printf '%s\n' "$p"
done
-
name: Download CLDR
if: env.BUILD_ONLY_ICONV != 'y'
working-directory: src\downloads
shell: pwsh
run: |
if (Test-Path -LiteralPath "cldr-$env:CLDR_VERSION.zip" -PathType Leaf) {
Write-Host -Object 'Already downloaded'
} else {
Invoke-WebRequest "https://unicode.org/Public/cldr/$env:CLDR_VERSION/core.zip" -OutFile "cldr-$env:CLDR_VERSION.zip"
Write-Host -Object 'Downloaded'
}
-
name: Extract CLDR
if: env.BUILD_ONLY_ICONV != 'y'
working-directory: installed
run: |
unzip -p ../src/downloads/cldr-$CLDR_VERSION.zip LICENSE >cldr-license.txt
unzip -p ../src/downloads/cldr-$CLDR_VERSION.zip common/supplemental/plurals.xml >cldr-plurals.xml
-
name: Download iconv
working-directory: src\downloads
shell: pwsh
run: |
if (Test-Path -LiteralPath "libiconv-$env:ICONV_VERSION.tar.gz" -PathType Leaf) {
Write-Host -Object 'Already downloaded'
} else {
Invoke-WebRequest "https://ftp.gnu.org/pub/gnu/libiconv/libiconv-$env:ICONV_VERSION.tar.gz" -OutFile "libiconv-$env:ICONV_VERSION.tar.gz"
Write-Host -Object 'Downloaded'
}
-
name: Extract iconv
working-directory: src
run: tar x -z -f downloads/libiconv-$ICONV_VERSION.tar.gz
-
name: Configure iconv
id: iconv-configure
working-directory: src\libiconv-${{ env.ICONV_VERSION }}
run: |
mkdir build
cd build
../configure \
CPPFLAGS='${{ steps.vars.outputs.cpp-flags }}' \
LDFLAGS='${{ steps.vars.outputs.ld-flags }}' \
${{ steps.vars.outputs.configure-args }} \
--prefix=$INSTALLED_PATH
-
name: Compile iconv
working-directory: src\libiconv-${{ env.ICONV_VERSION }}\build
run: make --jobs=$(nproc)
-
name: Check iconv
if: env.BUILD_ONLY_ICONV != 'y'
working-directory: src\libiconv-${{ env.ICONV_VERSION }}\build
run: make check --jobs=$(nproc)
-
name: Install iconv
working-directory: src\libiconv-${{ env.ICONV_VERSION }}\build
run: make install && cp ../COPYING $INSTALLED_PATH/iconv-license.txt
-
name: Download gettext
if: env.BUILD_ONLY_ICONV != 'y'
working-directory: src\downloads
shell: pwsh
run: |
if (Test-Path -LiteralPath "gettext-$env:GETTEXT_VERSION.tar.gz" -PathType Leaf) {
Write-Host -Object 'Already downloaded'
} else {
Invoke-WebRequest "${{ steps.vars.outputs.gettext-source-url }}" -OutFile "gettext-$env:GETTEXT_VERSION.tar.gz"
Write-Host -Object 'Downloaded'
}
-
name: Extract gettext
if: env.BUILD_ONLY_ICONV != 'y'
working-directory: src
run: tar x -z -f downloads/gettext-$GETTEXT_VERSION.tar.gz
-
name: Configure gettext
if: env.BUILD_ONLY_ICONV != 'y'
id: gettext-configure
working-directory: src\gettext-${{ env.GETTEXT_VERSION }}
run: |
mkdir build
cd build
../configure \
CPPFLAGS="-I$INSTALLED_PATH/include ${{ steps.vars.outputs.cpp-flags }}" \
LDFLAGS="-L$INSTALLED_PATH/lib ${{ steps.vars.outputs.ld-flags }}" \
${{ steps.vars.outputs.configure-args }} \
--disable-java \
--disable-native-java \
--disable-csharp \
--disable-openmp \
--disable-curses \
--without-emacs \
--with-included-libxml \
--without-bzip2 \
--without-xz \
--prefix=$INSTALLED_PATH
-
name: Ignore gettext C tests
if: steps.vars.outputs.gettext-ignore-tests-c && env.BUILD_ONLY_ICONV != 'y'
working-directory: src\gettext-${{ env.GETTEXT_VERSION }}
run: for f in ${{ steps.vars.outputs.gettext-ignore-tests-c }}; do echo 'int main() { return 0; }' >$f; done
-
name: Install gettext license
if: env.BUILD_ONLY_ICONV != 'y'
working-directory: src\gettext-${{ env.GETTEXT_VERSION }}
run: cp COPYING $INSTALLED_PATH/gettext-license.txt
-
name: Compile gettext/gnulib-local
if: env.BUILD_ONLY_ICONV != 'y'
working-directory: src\gettext-${{ env.GETTEXT_VERSION }}\build\gnulib-local
run: make --jobs=$(nproc)
-
name: Check gettext/gnulib-local
if: env.BUILD_ONLY_ICONV != 'y'
working-directory: src\gettext-${{ env.GETTEXT_VERSION }}\build\gnulib-local
run: make --jobs=$(nproc) check
-
name: Install gettext/gnulib-local
if: env.BUILD_ONLY_ICONV != 'y'
working-directory: src\gettext-${{ env.GETTEXT_VERSION }}\build\gnulib-local
run: make --jobs=$(nproc) install
-
name: Compile gettext/gettext-runtime
if: env.BUILD_ONLY_ICONV != 'y'
working-directory: src\gettext-${{ env.GETTEXT_VERSION }}\build\gettext-runtime
run: make --jobs=$(nproc)
-
name: Check gettext/gettext-runtime
if: env.BUILD_ONLY_ICONV != 'y'
working-directory: src\gettext-${{ env.GETTEXT_VERSION }}\build\gettext-runtime
run: make --jobs=$(nproc) check
-
name: Install gettext/gettext-runtime
if: env.BUILD_ONLY_ICONV != 'y'
working-directory: src\gettext-${{ env.GETTEXT_VERSION }}\build\gettext-runtime
run: make --jobs=$(nproc) install
-
name: Compile gettext/libtextstyle
if: env.BUILD_ONLY_ICONV != 'y'
working-directory: src\gettext-${{ env.GETTEXT_VERSION }}\build\libtextstyle
run: make --jobs=$(nproc)
-
name: Check gettext/libtextstyle
if: env.BUILD_ONLY_ICONV != 'y'
working-directory: src\gettext-${{ env.GETTEXT_VERSION }}\build\libtextstyle
run: make --jobs=$(nproc) check
-
name: Install gettext/libtextstyle
if: env.BUILD_ONLY_ICONV != 'y'
working-directory: src\gettext-${{ env.GETTEXT_VERSION }}\build\libtextstyle
run: make --jobs=$(nproc) install
-
name: Compile gettext/gettext-tools
if: env.BUILD_ONLY_ICONV != 'y'
working-directory: src\gettext-${{ env.GETTEXT_VERSION }}\build\gettext-tools
run: make --jobs=$(nproc)
-
name: Check gettext/gettext-tools
if: env.BUILD_ONLY_ICONV != 'y'
working-directory: src\gettext-${{ env.GETTEXT_VERSION }}\build\gettext-tools
run: XFAIL_TESTS='${{ steps.vars.outputs.gettext-xfail-gettext-tools }}' make --jobs=$(nproc) check
-
name: Install gettext/gettext-tools
if: env.BUILD_ONLY_ICONV != 'y'
working-directory: src\gettext-${{ env.GETTEXT_VERSION }}\build\gettext-tools
run: make --jobs=$(nproc) install
-
name: Prepare build log
id: prepare-build-log
if: (success() || failure()) && steps.iconv-configure.outcome == 'success'
run: |
mkdir build-log
if [ -d src/libiconv-$ICONV_VERSION/build ]; then
tar c -J -f build-log/iconv.tar.xz src/libiconv-$ICONV_VERSION/build
fi
if [ -d src/gettext-$GETTEXT_VERSION/build ]; then
tar c -J -f build-log/gettext.tar.xz src/gettext-$GETTEXT_VERSION/build
fi
if find installed -mindepth 1 -maxdepth 1 | read; then
tar c -J -f build-log/installed.tar.xz installed
fi
ls -al build-log
-
name: Copy built assets
run: ./build-exe/create-output.sh installed files-unsigned '${{ steps.vars.outputs.mingw-host }}'
-
name: Delete install directory
run: rm -rf installed
-
name: Process dependencies
shell: pwsh
run: >
./build-exe/process-dependencies.ps1
-Bits ${{ matrix.bits }}
-Link ${{ matrix.link }}
-OutputPath files-unsigned
-MinGWPath C:\cygwin\usr\${{ steps.vars.outputs.mingw-host }}
-
name: Check bitness
run: ./build-exe/check-bits.sh ${{ matrix.bits }} files-unsigned
-
name: Check if programs load translations correctly
if: env.BUILD_ONLY_ICONV != 'y'
shell: pwsh
run: |
$env:LANGUAGE = 'it'
$stdout = & .\files-unsigned\bin\xgettext.exe --help
if (-not($?)) {
throw "xgettext.exe failed"
}
$stdout = $stdout -join "`n"
if (!$stdout.Contains('impostazione predefinita')) {
throw "xgettext.exe didn't load the translations.`nIts output is:`n$stdout"
}
Write-Host "xgettext.exe correctly loaded the translations when LANGUAGE=$env:LANGUAGE`nIts localized output is`n$stdout"
-
name: Upload unsigned files
if: steps.vars.outputs.signpath-signing-policy
id: upload-files-unsigned
uses: actions/upload-artifact@v4
with:
name: ${{ matrix.link }}-${{ matrix.bits }}-files-unsigned
path: files-unsigned
if-no-files-found: error
retention-days: 1
-
name: Sign files
if: steps.vars.outputs.signpath-signing-policy
id: sign-files
uses: signpath/github-action-submit-signing-request@v1
with:
api-token: '${{ secrets.SIGNPATH_API_TOKEN }}'
organization-id: 98c3accc-92c9-4962-b150-ff1f5c6356b8
project-slug: gettext-iconv-windows
signing-policy-slug: '${{ steps.vars.outputs.signpath-signing-policy }}'
artifact-configuration-slug: gh_sigh_files
github-artifact-id: ${{ steps.upload-files-unsigned.outputs.artifact-id }}
wait-for-completion: true
output-artifact-directory: files-signed
parameters: |
iconvPEVersion: "${{ env.ICONV_VERSION }}"
gettextPEVersion: "${{ env.GETTEXT_VERSION }}"
gettextPEVersionNumeric: "${{ steps.vars.outputs.gettext-peversion-numeric }}"
-
name: Check signatures
if: steps.vars.outputs.signpath-signing-policy
shell: pwsh
run: >
./build-exe/check-signature.ps1
-Path files-signed
-CanBeInvalid ${{ steps.vars.outputs.signatures-canbeinvalid }}
-
name: Create files archive
shell: pwsh
run: |
if ('${{ steps.vars.outputs.signpath-signing-policy }}') {
Set-Location -LiteralPath 'files-signed'
} else {
Set-Location -LiteralPath 'files-unsigned'
}
& 7z.exe a -bd -bt -mx9 -r -sse -tzip ..\gettext${{ env.GETTEXT_VERSION }}-iconv${{ env.ICONV_VERSION }}-${{ matrix.link }}-${{ matrix.bits }}.zip
-
name: Upload files archive
uses: actions/upload-artifact@v4
with:
name: ${{ matrix.link }}-${{ matrix.bits }}-files
path: gettext${{ env.GETTEXT_VERSION }}-iconv${{ env.ICONV_VERSION }}-${{ matrix.link }}-${{ matrix.bits }}.zip
if-no-files-found: error
compression-level: 0
-
name: Prepare installer files
run: |
if [ -n '${{ steps.vars.outputs.signpath-signing-policy }}' ]; then
mv files-signed installer-files
else
mv files-unsigned installer-files
fi
-
name: Create installer
shell: pwsh
run: >
./build-exe/create-installer.ps1
-Bits ${{ matrix.bits }}
-Link ${{ matrix.link }}
-SourceDirectory installer-files
-OutputDirectory installer-unsigned
-
name: Check bitness
run: ./build-exe/check-bits.sh 32 installer-unsigned/gettext${{ env.GETTEXT_VERSION }}-iconv${{ env.ICONV_VERSION }}-${{ matrix.link }}-${{ matrix.bits }}.exe
-
name: Upload unsigned installer
if: steps.vars.outputs.signpath-signing-policy
id: upload-installer-unsigned
uses: actions/upload-artifact@v4
with:
name: ${{ matrix.link }}-${{ matrix.bits }}-installer-unsigned
path: installer-unsigned\gettext${{ env.GETTEXT_VERSION }}-iconv${{ env.ICONV_VERSION }}-${{ matrix.link }}-${{ matrix.bits }}.exe
if-no-files-found: error
compression-level: 0
retention-days: 1
-
name: Sign installer
if: steps.vars.outputs.signpath-signing-policy
id: sign-installer
uses: signpath/github-action-submit-signing-request@v1
with:
api-token: '${{ secrets.SIGNPATH_API_TOKEN }}'
organization-id: 98c3accc-92c9-4962-b150-ff1f5c6356b8
project-slug: gettext-iconv-windows
signing-policy-slug: '${{ steps.vars.outputs.signpath-signing-policy }}'
artifact-configuration-slug: gh_sigh_installer
github-artifact-id: ${{ steps.upload-installer-unsigned.outputs.artifact-id }}
wait-for-completion: true
output-artifact-directory: installer-signed
-
name: Check signatures
if: steps.vars.outputs.signpath-signing-policy
shell: pwsh
run: >
./build-exe/check-signature.ps1
-Path installer-signed
-CanBeInvalid ${{ steps.vars.outputs.signatures-canbeinvalid }}
-
name: Move installer
run: |
if [ -n '${{ steps.vars.outputs.signpath-signing-policy }}' ]; then
mv installer-signed/*.exe .
else
mv installer-unsigned/*.exe .
fi
-
name: Upload installer
uses: actions/upload-artifact@v4
with:
name: ${{ matrix.link }}-${{ matrix.bits }}-installer
path: gettext${{ env.GETTEXT_VERSION }}-iconv${{ env.ICONV_VERSION }}-${{ matrix.link }}-${{ matrix.bits }}.exe
if-no-files-found: error
compression-level: 0
-
name: Upload build log
if: always() && steps.prepare-build-log.outcome == 'success'
uses: actions/upload-artifact@v4
with:
name: ${{ matrix.link }}-${{ matrix.bits }}-log
if-no-files-found: ignore
retention-days: 1
include-hidden-files: true
compression-level: 0
path: build-log
-
name: Persist cache
if: always() && steps.restore-cache.outputs.cache-hit != 'true'
uses: actions/cache/save@v4
with:
key: ${{ steps.restore-cache.outputs.cache-primary-key }}
path: |
src\downloads
C:\cygwin-packages