Skip to content

Patch Installer

Patch Installer #151

name: Patch Installer
# Builds the FieldWorks Patch Installer on the specified `base_release`
# If `make_release` is true, uploads installers to https://flex-updates.s3.amazonaws.com/?prefix=jobs/FieldWorks-Win-all-Release-Patch.
# Saves the build log as an artifact of the workflow run.
# Note: FW_BUILD_NUMBER is higher than GITHUB_RUN_NUMBER because it needs to be higher than the build number on artifacts from our previous system.
# Note: FW_BUILD_NUMBER must be higher for patches than their base so that the patch is considered an upgrade.
on:
push:
branches: ["release/9.3"]
schedule:
# Runs every Monday at 03:30 UTC (which is 8:30pm MST/PDT Sunday evening)
- cron: "30 3 * * 1"
workflow_dispatch:
inputs:
fw_ref:
description: 'Commit-ish (branch, tag, SHA) to checkout for the main repository'
required: false
default: ''
helps_ref:
description: 'Commit-ish for helps repository'
required: false
default: 'develop'
installer_ref:
description: 'Commit-ish for PatchableInstaller repository'
required: false
default: 'master'
localizations_ref:
description: 'Commit-ish for localization repository'
required: false
default: 'develop'
lcm_ref:
description: 'Commit-ish for liblcm repository'
required: false
default: 'master'
# default should be changed to the most recent base release for the release cycle.
# After 9.3 is stable, base_release can be computed from base_build_number, but they need to be independent for bootstrapping patches
# on Jenkins-built bases.
base_release:
description: 'The github release for the base build artifacts (separate only for bootstrapping; should be removed after 9.3 is the stable)'
default: 'build-1188' # When updating this, update base_build_number and their fallbacks below, too.
base_build_number:
description: 'The base build number'
required: false
default: '1188'
make_release:
description: 'Should the release be pushed to s3 - use false for test builds'
required: false
default: 'true'
concurrency:
group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}
cancel-in-progress: true
jobs:
debug_build_and_test:
permissions:
contents: read
env:
CROWDIN_API_KEY: ${{ secrets.FLEX_CROWDIN_API }}
LcmRootDir: ${{ github.workspace }}/Localizations/LCM
FILESTOSIGNLATER: ./signExternally
GH_TOKEN: ${{ github.token }}
BASE_BUILD_NUMBER: ${{ inputs.base_build_number || '1188' }}
name: Build Debug and run Tests
runs-on: windows-latest
steps:
- name: Compute build number for archival
id: build_number
run: |
$lastJenkins = 2000 # The last patch build from jenkins, rounded to the next hundred
$githubRun = $env:GITHUB_RUN_NUMBER
$combined = $lastJenkins + $githubRun
echo "Calculated build number: $combined"
echo "FW_BUILD_NUMBER=$combined" >> $env:GITHUB_ENV
echo "BuildVersionSegment=$combined" >> $env:GITHUB_ENV
- name: Checkout Files
uses: actions/checkout@v4
id: checkout
with:
ref: ${{ github.event.inputs.fw_ref || github.ref }}
fetch-depth: 0
- name: Checkout Helps
uses: actions/checkout@v4
id: helps-checkout
with:
repository: 'sillsdev/FwHelps'
ref: ${{ github.event.inputs.helps_ref || 'develop' }}
fetch-depth: 0
path: 'DistFiles/Helps'
- name: Checkout PatchableInstaller
uses: actions/checkout@v4
id: installer-checkout
with:
repository: 'sillsdev/genericinstaller'
ref: ${{ github.event.inputs.installer_ref || 'master' }}
fetch-depth: 0
path: 'PatchableInstaller'
- name: Checkout Localizations
uses: actions/checkout@v4
id: loc-checkout
with:
repository: 'sillsdev/FwLocalizations'
ref: ${{ github.event.inputs.localizations_ref || 'develop' }}
fetch-depth: 0
path: 'Localizations'
- name: Checkout liblcm
uses: actions/checkout@v4
id: liblcm-checkout
with:
repository: 'sillsdev/liblcm'
ref: ${{ github.event.inputs.lcm_ref || 'master' }}
fetch-depth: 0
path: 'Localizations/LCM'
- name: Download .NET 461 targeting pack
uses: suisei-cn/actions-download-file@818d6b7dc8fe73f2f924b6241f2b1134ca1377d9 # 1.6.0
id: downloadfile # Remember to give an ID if you need the output filename
with:
url: "https://download.microsoft.com/download/F/1/D/F1DEB8DB-D277-4EF9-9F48-3A65D4D8F965/NDP461-DevPack-KB3105179-ENU.exe"
target: public/
- name: Install .NET 461 targeting pack
shell: cmd
working-directory: public
run: NDP461-DevPack-KB3105179-ENU.exe /q
- name: Setup dotnet
uses: actions/setup-dotnet@v4
with:
dotnet-version: |
3.1.x
5.0.x
- name: Install Overcrowdin
shell: cmd
run: |
dotnet tool update -g overcrowdin || dotnet tool install -g overcrowdin
- name: Downgrade Wix Toolset - remove when runner has 3.14.2
run: |
choco uninstall wixtoolset
choco install wixtoolset --version 3.11.2 --allow-downgrade --force
echo "C:\Program Files (x86)\WiX Toolset v3.11\bin" | Out-File -FilePath $env:GITHUB_PATH -Encoding utf8 -Append
if: github.event_name != 'pull_request'
- name: Import Base Build Artifacts
shell: pwsh
run: |
$release = "${{ inputs.base_release || 'build-1188' }}"
$repo = "${{ github.repository }}"
Write-Host "Downloading artifacts for release $release from $repo"
# Download BuildDir.zip
gh release download $release --repo $repo --pattern "BuildDir.zip" --dir base-artifacts
if (-not (Test-Path "base-artifacts/BuildDir.zip")) {
throw "BuildDir.zip not found in release '$release'."
}
# Download ProcRunner.zip
gh release download $release --repo $repo --pattern "ProcRunner.zip" --dir base-artifacts
if (-not (Test-Path "base-artifacts/ProcRunner.zip")) {
throw "ProcRunner.zip not found in release '$release'."
}
# Expand BuildDir.zip into BuildDir
Expand-Archive -Path "base-artifacts/BuildDir.zip" -DestinationPath "BuildDir" -Force
Write-Host "Expanded BuildDir.zip -> ./BuildDir"
# Ensure ProcRunner target path exists
$procTarget = "PatchableInstaller/ProcRunner/ProcRunner/bin/Release/net48"
if (-not (Test-Path $procTarget)) {
New-Item -ItemType Directory -Path $procTarget -Force | Out-Null
}
# Expand ProcRunner.zip into the exact net48 folder
Expand-Archive -Path "base-artifacts/ProcRunner.zip" -DestinationPath $procTarget -Force
Write-Host "Expanded ProcRunner.zip -> $procTarget"
# Write out the properties file (a first build on a system requires a prompt response otherwise)
# and set an OS feature in the registry that will allow Wix v3 to use temporary files without error
- name: Prepare for build
working-directory: Build
run: |
# Define paths and the key/value to set
$regPaths = @(
"HKLM:\SOFTWARE\Microsoft\.NETFramework\AppContext",
"HKLM:\SOFTWARE\WOW6432Node\Microsoft\.NETFramework\AppContext"
)
$valueName = "Switch.System.DisableTempFileCollectionDirectoryFeature"
$expectedValue = "true"
foreach ($path in $regPaths) {
Write-Host "Adding or updating registry value in $path..."
if (-not (Test-Path $path)) {
New-Item -Path $path -Force | Out-Null
}
New-ItemProperty -Path $path -Name $valueName -Value $expectedValue -Type String -Force
}
.\build64.bat /t:WriteNonlocalDevelopmentPropertiesFile
- name: Build Debug and run tests
id: build_installer
shell: powershell
run: |
Get-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\.NETFramework\AppContext"
cd Build
.\build64.bat /t:BuildPatchInstaller "/property:config=release;action=test;desktopNotAvailable=true" /v:d /bl ^| tee-object -FilePath build.log
cd ..
cd BuildDir
- name: Scan Debug Build Output
shell: powershell
working-directory: Build
run: |
$results = Select-String -Path "build.log" -Pattern "^\s*[1-9][0-9]* Error\(s\)"
if ($results) {
foreach ($result in $results) {
Write-Host "Found errors in build.log $($result.LineNumber): $($result.Line)" -ForegroundColor red
}
exit 1
} else {
Write-Host "No errors found" -ForegroundColor green
exit 0
}
- name: Find patch installer
id: find_patch
shell: pwsh
run: |
$patch = Get-ChildItem -Path BuildDir -Filter "FieldWorks_*.msp" | Select-Object -First 1
if (-not $patch) { throw "Patch (.msp) file not found in BuildDir" }
$patchPath = $patch.FullName
"patch_file=$patchPath" >> $env:GITHUB_OUTPUT
- name: Sign Patch
if: github.event_name != 'pull_request'
uses: sillsdev/codesign/trusted-signing-action@v3
with:
credentials: ${{ secrets.TRUSTED_SIGNING_CREDENTIALS }}
files-folder: BuildDir
files-folder-filter: '*.msp'
description: 'FieldWorks Patch Installer'
description-url: 'https://software.sil.org/fieldworks/'
- name: Configure AWS credentials
# Use inputs if defined but default to true
if: ${{ (inputs.make_release || 'true') == 'true' }}
uses: aws-actions/configure-aws-credentials@v4
with:
aws-access-key-id: ${{ secrets.AWS_ACCESS_KEY_ID }}
aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }}
aws-region: us-east-1
- name: Upload Patch to S3
# Use inputs if defined but default to true
if: ${{ (inputs.make_release || 'true') == 'true' }}
shell: pwsh
run: |
$patchPath = "${{ steps.find_patch.outputs.patch_file }}"
if (-not (Test-Path $patchPath)) {
throw "Patch file not found at $patchPath"
}
$patchFile = Split-Path $patchPath -Leaf
$s3Key = "jobs/FieldWorks-Win-all-Release-Patch/$($env:FW_BUILD_NUMBER)/$patchFile"
aws s3 cp $patchPath "s3://flex-updates/$s3Key"
Write-Host "Uploaded to s3://flex-updates/$s3Key"
- name: Upload Build Logs
uses: actions/upload-artifact@v4
if: always()
with:
if-no-files-found: warn
name: build-logs
path: |
Build/*.log
Build/*.binlog