179 lines
7.7 KiB
PowerShell
179 lines
7.7 KiB
PowerShell
<#
|
|
.SYNOPSIS
|
|
End-to-end installer build: publish service + GUI, then run Inno Setup
|
|
to produce dist/WebhookServer-Setup-{version}.exe.
|
|
|
|
.DESCRIPTION
|
|
Reads the version from Directory.Build.props. Requires Inno Setup 6 (ISCC.exe)
|
|
on PATH or in the standard install location. CI runs this same script after
|
|
setup-dotnet + winget install Inno Setup.
|
|
#>
|
|
[CmdletBinding()]
|
|
param(
|
|
[string]$Configuration = 'Release',
|
|
[string]$VersionOverride
|
|
)
|
|
|
|
$ErrorActionPreference = 'Stop'
|
|
$repoRoot = Split-Path -Parent $PSScriptRoot
|
|
|
|
function Get-RepoVersion {
|
|
$propsPath = Join-Path $repoRoot 'Directory.Build.props'
|
|
[xml]$props = Get-Content $propsPath
|
|
return $props.Project.PropertyGroup.Version
|
|
}
|
|
|
|
function Find-InnoCompiler {
|
|
$candidates = @(
|
|
'ISCC.exe', # on PATH
|
|
'C:\Program Files (x86)\Inno Setup 6\ISCC.exe',
|
|
'C:\Program Files\Inno Setup 6\ISCC.exe'
|
|
)
|
|
foreach ($c in $candidates) {
|
|
$cmd = Get-Command $c -ErrorAction SilentlyContinue
|
|
if ($cmd) { return $cmd.Path }
|
|
if (Test-Path $c) { return $c }
|
|
}
|
|
throw "Inno Setup compiler not found. Install with: winget install JRSoftware.InnoSetup"
|
|
}
|
|
|
|
$version = if ($VersionOverride) { $VersionOverride } else { Get-RepoVersion }
|
|
Write-Host "Building Webhook Server installer v$version" -ForegroundColor Cyan
|
|
|
|
# 1. Publish both projects.
|
|
$publishSvc = Join-Path $repoRoot 'publish\service'
|
|
$publishGui = Join-Path $repoRoot 'publish\gui'
|
|
Remove-Item -Recurse -Force $publishSvc, $publishGui -ErrorAction SilentlyContinue
|
|
|
|
& dotnet publish (Join-Path $repoRoot 'src\WebhookServer.Service\WebhookServer.Service.csproj') `
|
|
-c $Configuration -r win-x64 --self-contained false -o $publishSvc | Out-Host
|
|
if ($LASTEXITCODE -ne 0) { throw 'service publish failed' }
|
|
|
|
& dotnet publish (Join-Path $repoRoot 'src\WebhookServer.Gui\WebhookServer.Gui.csproj') `
|
|
-c $Configuration -r win-x64 --self-contained false -o $publishGui | Out-Host
|
|
if ($LASTEXITCODE -ne 0) { throw 'GUI publish failed' }
|
|
|
|
# 2. Pre-flight: confirm every source path the .iss references exists, and
|
|
# surface the longest path so MAX_PATH issues are obvious in the log.
|
|
function Show-SourcePath($label, $path, [switch]$Recursive) {
|
|
if (-not (Test-Path $path)) { Write-Warning "MISSING $label : $path"; return }
|
|
$items = if ($Recursive) {
|
|
Get-ChildItem $path -Recurse -File -ErrorAction SilentlyContinue
|
|
} else {
|
|
Get-ChildItem $path -File -ErrorAction SilentlyContinue
|
|
}
|
|
$count = ($items | Measure-Object).Count
|
|
$longest = ($items | Measure-Object -Maximum -Property { $_.FullName.Length }).Maximum
|
|
Write-Host (" {0,-30} files={1,-5} longestPath={2,-5} root={3}" -f $label, $count, $longest, $path)
|
|
}
|
|
|
|
Write-Host ""
|
|
Write-Host "--- pre-flight: source paths the .iss will read ---" -ForegroundColor Cyan
|
|
Show-SourcePath 'publish\service' $publishSvc -Recursive
|
|
Show-SourcePath 'publish\gui' $publishGui -Recursive
|
|
Show-SourcePath 'scripts' (Join-Path $repoRoot 'scripts')
|
|
Show-SourcePath 'scripts\examples' (Join-Path $repoRoot 'scripts\examples') -Recursive
|
|
Show-SourcePath 'docs' (Join-Path $repoRoot 'docs') -Recursive
|
|
Show-SourcePath 'resources' (Join-Path $repoRoot 'resources')
|
|
Show-SourcePath 'README.md (file)' (Join-Path $repoRoot 'README.md')
|
|
|
|
$lpe = (Get-ItemProperty 'HKLM:\SYSTEM\CurrentControlSet\Control\FileSystem' `
|
|
-Name LongPathsEnabled -ErrorAction SilentlyContinue).LongPathsEnabled
|
|
Write-Host " LongPathsEnabled (HKLM): $lpe"
|
|
Write-Host ""
|
|
|
|
# 3. Compile installer.
|
|
$iscc = Find-InnoCompiler
|
|
$iss = Join-Path $repoRoot 'installer\webhook-server.iss'
|
|
$dist = Join-Path $repoRoot 'dist'
|
|
New-Item -ItemType Directory -Path $dist -Force | Out-Null
|
|
|
|
Write-Host "Compiling installer with $iscc"
|
|
# Run ISCC from the .iss directory with just the bare filename. When invoked
|
|
# with a deeply-nested absolute path on the act-runner host (under
|
|
# %SystemRoot%\System32\config\systemprofile\...), ISCC sometimes prints a
|
|
# generic "The system cannot find the path specified." before it touches any
|
|
# source files. cd-ing first sidesteps it.
|
|
$issDir = Split-Path $iss -Parent
|
|
$issName = Split-Path $iss -Leaf
|
|
|
|
# Extra pre-flight: confirm the specific files our .iss references that a
|
|
# trivial test .iss wouldn't (icon, README, scripts) actually exist relative
|
|
# to the .iss directory the way ISCC will resolve them (RepoRoot = ..\).
|
|
Write-Host "--- pre-flight: paths the .iss references via {#RepoRoot} ---" -ForegroundColor Cyan
|
|
$issRefs = @(
|
|
'resources\webhook-server.ico',
|
|
'README.md',
|
|
'scripts\install-service.ps1',
|
|
'scripts\uninstall-service.ps1',
|
|
'publish\service',
|
|
'publish\gui',
|
|
'docs',
|
|
'scripts\examples'
|
|
)
|
|
foreach ($ref in $issRefs) {
|
|
$abs = Join-Path $repoRoot $ref
|
|
$exists = Test-Path $abs
|
|
Write-Host (" {0,-40} exists={1} ({2})" -f $ref, $exists, $abs)
|
|
}
|
|
Write-Host ""
|
|
|
|
Write-Host "--- runtime context ---" -ForegroundColor Cyan
|
|
Write-Host " identity: $([Security.Principal.WindowsIdentity]::GetCurrent().Name)"
|
|
Write-Host " USERPROFILE: $env:USERPROFILE"
|
|
Write-Host " APPDATA: $env:APPDATA"
|
|
Write-Host " LOCALAPPDATA: $env:LOCALAPPDATA"
|
|
Write-Host " TEMP: $env:TEMP"
|
|
$isccDir = Split-Path $iscc -Parent
|
|
Write-Host " ISCC dir: $isccDir"
|
|
foreach ($f in @('ISCC.exe','ISCmplr.dll','ISPP.dll','Default.isl','Compil32.exe')) {
|
|
$p = Join-Path $isccDir $f
|
|
Write-Host (" {0,-15} exists={1}" -f $f, (Test-Path $p))
|
|
}
|
|
Write-Host ""
|
|
|
|
Write-Host " PS location (pre): $((Get-Location).Path)"
|
|
Write-Host " .NET cwd (pre): $([System.IO.Directory]::GetCurrentDirectory())"
|
|
|
|
Push-Location $issDir
|
|
$savedDotNetCwd = [System.IO.Directory]::GetCurrentDirectory()
|
|
[System.IO.Directory]::SetCurrentDirectory($issDir)
|
|
try {
|
|
Write-Host " PS location (post): $((Get-Location).Path)"
|
|
Write-Host " .NET cwd (post): $([System.IO.Directory]::GetCurrentDirectory())"
|
|
|
|
# Bake the version into a temp .iss and override OutputDir to an absolute
|
|
# path so nothing in the build depends on cwd resolution.
|
|
$tempIss = Join-Path $issDir "webhook-server.gen.iss"
|
|
$issBody = Get-Content $issName -Raw
|
|
$pattern = '(?s)#ifndef AppVersion\s+#define AppVersion "[^"]*"\s+#endif'
|
|
if ($issBody -notmatch $pattern) { throw "Could not find #ifndef AppVersion block in $issName" }
|
|
$issBody = $issBody -replace $pattern, "#define AppVersion `"$version`""
|
|
Set-Content -Path $tempIss -Value $issBody -Encoding ascii
|
|
Write-Host " using $tempIss"
|
|
|
|
# Capture stdout+stderr together so any error line ISCC emits is visible
|
|
# in the runner log even if the runner's console capture drops one stream.
|
|
# /O<absolute> overrides OutputDir so ..\dist isn't resolved relative to
|
|
# whatever cwd ISCC actually inherits.
|
|
$logPath = Join-Path $env:TEMP "iscc-$version.log"
|
|
& $iscc "/O$dist" (Split-Path $tempIss -Leaf) *>&1 | Tee-Object -FilePath $logPath | ForEach-Object { Write-Host $_ }
|
|
$exit = $LASTEXITCODE
|
|
Write-Host " ISCC exit code: $exit"
|
|
Write-Host " ISCC log path: $logPath"
|
|
if (Test-Path $logPath) {
|
|
Write-Host " --- iscc log file contents ---"
|
|
Get-Content $logPath | ForEach-Object { Write-Host " $_" }
|
|
Write-Host " --- end iscc log ---"
|
|
}
|
|
Remove-Item $tempIss -ErrorAction SilentlyContinue
|
|
} finally {
|
|
[System.IO.Directory]::SetCurrentDirectory($savedDotNetCwd)
|
|
Pop-Location
|
|
}
|
|
if ($exit -ne 0) { throw "Inno Setup compile failed (exit $exit)" }
|
|
|
|
$out = Get-Item (Join-Path $dist "WebhookServer-Setup-$version.exe")
|
|
Write-Host ""
|
|
Write-Host ("Built: {0} ({1:n0} bytes)" -f $out.FullName, $out.Length) -ForegroundColor Green
|