mirror of
https://github.com/wwiinnddyy/LanMountainDesktop.git
synced 2026-06-20 23:54:26 +08:00
1133 lines
43 KiB
YAML
1133 lines
43 KiB
YAML
name: Release
|
|
|
|
on:
|
|
push:
|
|
tags:
|
|
- 'v*'
|
|
workflow_dispatch:
|
|
inputs:
|
|
tag:
|
|
description: 'Release tag'
|
|
required: true
|
|
type: string
|
|
is_prerelease:
|
|
description: 'Pre-release'
|
|
required: false
|
|
type: boolean
|
|
default: false
|
|
|
|
env:
|
|
DOTNET_VERSION: '10.0.x'
|
|
Solution_Name: LanMountainDesktop.slnx
|
|
DOTNET_gcServer: 1
|
|
|
|
jobs:
|
|
prepare:
|
|
runs-on: ubuntu-latest
|
|
outputs:
|
|
version: ${{ steps.version.outputs.version }}
|
|
assembly_version: ${{ steps.version.outputs.assembly_version }}
|
|
informational_version: ${{ steps.version.outputs.informational_version }}
|
|
tag: ${{ steps.version.outputs.tag }}
|
|
checkout_ref: ${{ steps.version.outputs.checkout_ref }}
|
|
|
|
steps:
|
|
- name: Get release info
|
|
id: version
|
|
run: |
|
|
if [[ "${{ github.event_name }}" == "push" ]]; then
|
|
TAG="${GITHUB_REF#refs/tags/}"
|
|
CHECKOUT_REF="${GITHUB_REF}"
|
|
else
|
|
RAW_TAG="${{ github.event.inputs.tag }}"
|
|
if [[ "${RAW_TAG}" == refs/tags/* ]]; then
|
|
TAG="${RAW_TAG#refs/tags/}"
|
|
elif [[ "${RAW_TAG}" == v* ]]; then
|
|
TAG="${RAW_TAG}"
|
|
else
|
|
TAG="v${RAW_TAG}"
|
|
fi
|
|
CHECKOUT_REF="${GITHUB_SHA}"
|
|
fi
|
|
VERSION="${TAG#v}"
|
|
IFS='.' read -r -a VERSION_PARTS <<< "${VERSION}"
|
|
while [ "${#VERSION_PARTS[@]}" -lt 4 ]; do
|
|
VERSION_PARTS+=("0")
|
|
done
|
|
ASSEMBLY_VERSION="${VERSION_PARTS[0]}.${VERSION_PARTS[1]}.${VERSION_PARTS[2]}.${VERSION_PARTS[3]}"
|
|
echo "tag=${TAG}" >> $GITHUB_OUTPUT
|
|
echo "version=${VERSION}" >> $GITHUB_OUTPUT
|
|
echo "assembly_version=${ASSEMBLY_VERSION}" >> $GITHUB_OUTPUT
|
|
echo "informational_version=${VERSION}" >> $GITHUB_OUTPUT
|
|
echo "checkout_ref=${CHECKOUT_REF}" >> $GITHUB_OUTPUT
|
|
|
|
build-windows:
|
|
needs: prepare
|
|
runs-on: windows-latest
|
|
strategy:
|
|
fail-fast: false
|
|
matrix:
|
|
include:
|
|
- arch: x64
|
|
self_contained: true
|
|
suffix: ''
|
|
- arch: x86
|
|
self_contained: true
|
|
suffix: ''
|
|
name: Build_Windows_${{ matrix.arch }}${{ matrix.suffix }}
|
|
|
|
steps:
|
|
- name: Checkout
|
|
uses: actions/checkout@v4
|
|
with:
|
|
fetch-depth: 0
|
|
submodules: recursive
|
|
ref: ${{ needs.prepare.outputs.checkout_ref }}
|
|
|
|
- name: Setup .NET
|
|
uses: actions/setup-dotnet@v4
|
|
with:
|
|
dotnet-version: ${{ env.DOTNET_VERSION }}
|
|
dotnet-quality: 'preview'
|
|
|
|
- name: Restore
|
|
run: dotnet restore ${{ env.Solution_Name }}
|
|
|
|
- name: Build
|
|
run: >
|
|
dotnet build ${{ env.Solution_Name }} -c Release --no-restore -v minimal
|
|
-p:Version=${{ needs.prepare.outputs.version }}
|
|
-p:AssemblyVersion=${{ needs.prepare.outputs.assembly_version }}
|
|
-p:FileVersion=${{ needs.prepare.outputs.assembly_version }}
|
|
-p:InformationalVersion=${{ needs.prepare.outputs.informational_version }}
|
|
|
|
- name: Publish Launcher (AOT)
|
|
run: |
|
|
$version = "${{ needs.prepare.outputs.version }}"
|
|
$arch = "${{ matrix.arch }}"
|
|
$launcherPublishDir = "publish/launcher-win-$arch"
|
|
|
|
Write-Host "Publishing Launcher with AOT for Windows $arch..."
|
|
|
|
# AOT publish
|
|
dotnet publish LanMountainDesktop.Launcher/LanMountainDesktop.Launcher.csproj `
|
|
-c Release `
|
|
-o ./$launcherPublishDir `
|
|
--self-contained `
|
|
-r win-$arch `
|
|
-p:PublishAot=true `
|
|
-p:PublishSingleFile=true `
|
|
-p:IncludeNativeLibrariesForSelfExtract=true `
|
|
-p:EnableCompressionInSingleFile=true `
|
|
-p:DebugType=none `
|
|
-p:DebugSymbols=false
|
|
|
|
if ($LASTEXITCODE -ne 0) {
|
|
Write-Error "Launcher AOT publish failed"
|
|
exit 1
|
|
}
|
|
|
|
# 鏄剧ず鍙戝竷缁撴灉
|
|
Write-Host "Launcher published to: $launcherPublishDir"
|
|
$exeFile = Get-ChildItem -Path $launcherPublishDir -Filter "*.exe" | Select-Object -First 1
|
|
if ($exeFile) {
|
|
$size = [Math]::Round($exeFile.Length / 1MB, 2)
|
|
Write-Host "Launcher executable: $($exeFile.Name) ($size MB)"
|
|
}
|
|
|
|
# Warn if unexpected extra files are produced
|
|
$files = Get-ChildItem -Path $launcherPublishDir -File
|
|
if ($files.Count -gt 1) {
|
|
Write-Host "Warning: Expected single file but found $($files.Count) files"
|
|
$files | ForEach-Object { Write-Host " - $($_.Name)" }
|
|
}
|
|
shell: pwsh
|
|
|
|
- name: Publish Main App
|
|
run: |
|
|
$selfContained = "${{ matrix.self_contained }}" -eq "true"
|
|
$publishDir = if ($selfContained) { "publish/windows-${{ matrix.arch }}" } else { "publish/windows-${{ matrix.arch }}-lite" }
|
|
|
|
if ($selfContained) {
|
|
dotnet publish LanMountainDesktop/LanMountainDesktop.csproj `
|
|
-c Release `
|
|
-o ./$publishDir `
|
|
--self-contained `
|
|
-r win-${{ matrix.arch }} `
|
|
-p:PublishSingleFile=false `
|
|
-p:DebugType=none `
|
|
-p:DebugSymbols=false `
|
|
-p:PublishTrimmed=false `
|
|
-p:PublishReadyToRun=false `
|
|
-p:Version=${{ needs.prepare.outputs.version }} `
|
|
-p:AssemblyVersion=${{ needs.prepare.outputs.assembly_version }} `
|
|
-p:FileVersion=${{ needs.prepare.outputs.assembly_version }} `
|
|
-p:InformationalVersion=${{ needs.prepare.outputs.informational_version }}
|
|
} else {
|
|
dotnet publish LanMountainDesktop/LanMountainDesktop.csproj `
|
|
-c Release `
|
|
-o ./$publishDir `
|
|
--self-contained:false `
|
|
-p:PublishSingleFile=false `
|
|
-p:DebugType=none `
|
|
-p:DebugSymbols=false `
|
|
-p:PublishTrimmed=false `
|
|
-p:PublishReadyToRun=false `
|
|
-p:Version=${{ needs.prepare.outputs.version }} `
|
|
-p:AssemblyVersion=${{ needs.prepare.outputs.assembly_version }} `
|
|
-p:FileVersion=${{ needs.prepare.outputs.assembly_version }} `
|
|
-p:InformationalVersion=${{ needs.prepare.outputs.informational_version }}
|
|
}
|
|
|
|
Write-Host "Published to: $publishDir"
|
|
Write-Host "Self-contained: $selfContained"
|
|
shell: pwsh
|
|
|
|
- name: Restructure for Launcher
|
|
run: |
|
|
$version = "${{ needs.prepare.outputs.version }}"
|
|
$arch = "${{ matrix.arch }}"
|
|
$selfContained = "${{ matrix.self_contained }}" -eq "true"
|
|
$publishDir = if ($selfContained) { "publish/windows-$arch" } else { "publish/windows-$arch-lite" }
|
|
$launcherPublishDir = "publish/launcher-win-$arch"
|
|
$appDir = "app-$version"
|
|
|
|
Write-Host "Restructuring for Launcher mode..."
|
|
Write-Host "Version: $version"
|
|
Write-Host "Publish dir: $publishDir"
|
|
|
|
$newStructure = "publish-launcher/windows-$arch"
|
|
New-Item -ItemType Directory -Path $newStructure -Force | Out-Null
|
|
|
|
$appPath = Join-Path $newStructure $appDir
|
|
Move-Item -Path $publishDir -Destination $appPath -Force
|
|
|
|
$launcherSource = $launcherPublishDir
|
|
if (Test-Path $launcherSource) {
|
|
Write-Host "Copying Launcher to root..."
|
|
Copy-Item -Path "$launcherSource\*" -Destination $newStructure -Recurse -Force
|
|
} else {
|
|
Write-Warning "Launcher publish dir not found: $launcherSource"
|
|
}
|
|
|
|
New-Item -ItemType File -Path (Join-Path $appPath ".current") -Force | Out-Null
|
|
|
|
Write-Host "New directory structure:"
|
|
Get-ChildItem -Path $newStructure -Recurse -Depth 2 | Select-Object FullName
|
|
|
|
Remove-Item -Path $publishDir -Recurse -Force -ErrorAction SilentlyContinue
|
|
Remove-Item -Path $launcherPublishDir -Recurse -Force -ErrorAction SilentlyContinue
|
|
Move-Item -Path $newStructure -Destination $publishDir -Force
|
|
shell: pwsh
|
|
|
|
- name: Install Inno Setup and 7z
|
|
run: |
|
|
choco install innosetup -y --no-progress
|
|
choco install 7zip -y --no-progress
|
|
shell: pwsh
|
|
|
|
- name: Build Installer
|
|
run: |
|
|
$version = "${{ needs.prepare.outputs.version }}"
|
|
$arch = "${{ matrix.arch }}"
|
|
$selfContained = "${{ matrix.self_contained }}" -eq "true"
|
|
$suffix = "${{ matrix.suffix }}"
|
|
$publishDir = if ($selfContained) { "publish\windows-$arch" } else { "publish\windows-$arch-lite" }
|
|
$installerScript = "LanMountainDesktop\installer\LanMountainDesktop.iss"
|
|
$outputDir = "build-installer"
|
|
|
|
if (-not (Test-Path -Path $publishDir)) {
|
|
Write-Error "Publish directory not found: $publishDir"
|
|
Get-ChildItem -Path "publish" -Directory -ErrorAction SilentlyContinue | Select-Object Name
|
|
exit 1
|
|
}
|
|
|
|
New-Item -ItemType Directory -Path $outputDir -Force | Out-Null
|
|
|
|
if (-not (Test-Path -Path $installerScript)) {
|
|
Write-Error "Installer script not found: $installerScript"
|
|
exit 1
|
|
}
|
|
|
|
$isccPath = $null
|
|
$isccCommand = Get-Command ISCC.exe -ErrorAction SilentlyContinue
|
|
if ($isccCommand) {
|
|
$isccPath = $isccCommand.Source
|
|
}
|
|
|
|
$candidatePaths = @(
|
|
"C:\Program Files (x86)\Inno Setup 6\ISCC.exe",
|
|
"C:\Program Files\Inno Setup 6\ISCC.exe",
|
|
"$env:ChocolateyInstall\bin\ISCC.exe",
|
|
"$env:ChocolateyInstall\lib\innosetup\tools\ISCC.exe"
|
|
)
|
|
|
|
if (-not $isccPath) {
|
|
foreach ($candidate in $candidatePaths) {
|
|
if ($candidate -and (Test-Path -Path $candidate)) {
|
|
$isccPath = $candidate
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
if (-not $isccPath) {
|
|
Write-Host "ISCC.exe was not found in PATH or known locations."
|
|
Write-Host "Checked locations:"
|
|
$candidatePaths | ForEach-Object { Write-Host " - $_" }
|
|
Write-Host "Chocolatey bin listing (if exists):"
|
|
Get-ChildItem "$env:ChocolateyInstall\bin" -Filter "*iscc*" -ErrorAction SilentlyContinue | Select-Object FullName
|
|
Write-Error "Inno Setup compiler not found."
|
|
exit 1
|
|
}
|
|
|
|
Write-Host "Found Inno Setup at: $isccPath"
|
|
|
|
Write-Host "Building installer for Windows $arch with version $version..."
|
|
|
|
$publishDir = (Resolve-Path $publishDir).Path
|
|
$outputDir = (Resolve-Path $outputDir).Path
|
|
$installerScript = (Resolve-Path $installerScript).Path
|
|
|
|
$compileArgs = @(
|
|
"/DMyAppVersion=$version",
|
|
"/DPublishDir=$publishDir",
|
|
"/DMyOutputDir=$outputDir",
|
|
"/DMyAppArch=$arch",
|
|
"/DMyAppSuffix=$suffix",
|
|
"/DIsSelfContained=$selfContained",
|
|
$installerScript
|
|
)
|
|
|
|
Write-Host "Compile command: `"$isccPath`" $($compileArgs -join ' ')"
|
|
|
|
& $isccPath @compileArgs
|
|
if ($LASTEXITCODE -ne 0) {
|
|
Write-Error "Inno Setup compiler exited with code $LASTEXITCODE"
|
|
exit 1
|
|
}
|
|
|
|
$installerFile = Get-ChildItem -Path $outputDir -Filter "*.exe" -ErrorAction SilentlyContinue | Select-Object -First 1
|
|
if (-not $installerFile) {
|
|
Write-Error "Failed to create installer"
|
|
exit 1
|
|
}
|
|
|
|
Write-Host "Successfully created: $($installerFile.Name)"
|
|
Write-Host "Installer size: $([Math]::Round($installerFile.Length / 1MB, 2)) MB"
|
|
shell: pwsh
|
|
|
|
- name: Upload App Payload
|
|
uses: actions/upload-artifact@v4
|
|
with:
|
|
name: app-payload-windows-${{ matrix.arch }}
|
|
path: |
|
|
publish/windows-${{ matrix.arch }}/**
|
|
if-no-files-found: error
|
|
retention-days: 30
|
|
|
|
- name: Upload Installer
|
|
uses: actions/upload-artifact@v4
|
|
with:
|
|
name: installer-windows-${{ matrix.arch }}
|
|
path: build-installer/*.exe
|
|
if-no-files-found: error
|
|
retention-days: 30
|
|
|
|
build-linux:
|
|
needs: prepare
|
|
runs-on: ubuntu-latest
|
|
name: Build_Linux
|
|
|
|
steps:
|
|
- name: Checkout
|
|
uses: actions/checkout@v4
|
|
with:
|
|
fetch-depth: 0
|
|
submodules: recursive
|
|
ref: ${{ needs.prepare.outputs.checkout_ref }}
|
|
|
|
- name: Install dependencies
|
|
run: |
|
|
sudo apt-get update
|
|
sudo apt-get install -y \
|
|
libfontconfig1 libfreetype6 \
|
|
libx11-6 libxrandr2 libxinerama1 \
|
|
libxi6 libxcursor1 libxext6 \
|
|
libxrender1 libxkbcommon-x11-0 \
|
|
clang zlib1g-dev
|
|
|
|
# Ubuntu 24.04+ moved several packages to t64 names.
|
|
sudo apt-get install -y libasound2t64 || sudo apt-get install -y libasound2
|
|
sudo apt-get install -y libportaudio2t64 || sudo apt-get install -y libportaudio2
|
|
|
|
# Prefer modern WebKit package, fallback for older images.
|
|
sudo apt-get install -y libwebkit2gtk-4.1-dev || sudo apt-get install -y libwebkit2gtk-4.0-dev
|
|
|
|
- name: Setup .NET
|
|
uses: actions/setup-dotnet@v4
|
|
with:
|
|
dotnet-version: ${{ env.DOTNET_VERSION }}
|
|
dotnet-quality: 'preview'
|
|
|
|
- name: Restore
|
|
run: dotnet restore ${{ env.Solution_Name }}
|
|
|
|
- name: Build
|
|
run: >
|
|
dotnet build ${{ env.Solution_Name }} -c Release --no-restore -v minimal
|
|
-p:Version=${{ needs.prepare.outputs.version }}
|
|
-p:AssemblyVersion=${{ needs.prepare.outputs.assembly_version }}
|
|
-p:FileVersion=${{ needs.prepare.outputs.assembly_version }}
|
|
-p:InformationalVersion=${{ needs.prepare.outputs.informational_version }}
|
|
|
|
- name: Publish Launcher (AOT)
|
|
run: |
|
|
echo "Publishing Launcher with AOT for Linux x64..."
|
|
|
|
dotnet publish LanMountainDesktop.Launcher/LanMountainDesktop.Launcher.csproj \
|
|
-c Release \
|
|
-o ./publish/launcher-linux-x64 \
|
|
--self-contained \
|
|
-r linux-x64 \
|
|
-p:PublishAot=true \
|
|
-p:PublishSingleFile=true \
|
|
-p:IncludeNativeLibrariesForSelfExtract=true \
|
|
-p:EnableCompressionInSingleFile=true \
|
|
-p:DebugType=none \
|
|
-p:DebugSymbols=false
|
|
|
|
if [ $? -ne 0 ]; then
|
|
echo "Launcher AOT publish failed"
|
|
exit 1
|
|
fi
|
|
|
|
echo "Launcher published to: ./publish/launcher-linux-x64"
|
|
ls -lh ./publish/launcher-linux-x64/
|
|
|
|
- name: Publish Main App
|
|
run: |
|
|
dotnet publish LanMountainDesktop/LanMountainDesktop.csproj \
|
|
-c Release \
|
|
-o ./publish/linux-x64-app \
|
|
--self-contained \
|
|
-r linux-x64 \
|
|
-p:PublishSingleFile=false \
|
|
-p:SelfContained=true \
|
|
-p:DebugType=none \
|
|
-p:DebugSymbols=false \
|
|
-p:PublishTrimmed=false \
|
|
-p:PublishReadyToRun=false \
|
|
-p:Version=${{ needs.prepare.outputs.version }} \
|
|
-p:AssemblyVersion=${{ needs.prepare.outputs.assembly_version }} \
|
|
-p:FileVersion=${{ needs.prepare.outputs.assembly_version }} \
|
|
-p:InformationalVersion=${{ needs.prepare.outputs.informational_version }}
|
|
|
|
- name: Restructure for Launcher
|
|
run: |
|
|
version="${{ needs.prepare.outputs.version }}"
|
|
publishDir="publish/linux-x64"
|
|
appDir="app-$version"
|
|
launcherDir="publish/launcher-linux-x64"
|
|
|
|
echo "Restructuring for Launcher mode..."
|
|
echo "Version: $version"
|
|
|
|
mkdir -p "$publishDir"
|
|
mv "publish/linux-x64-app" "$publishDir/$appDir"
|
|
|
|
if [ -d "$launcherDir" ]; then
|
|
echo "Copying Launcher to root..."
|
|
cp -r "$launcherDir"/* "$publishDir/"
|
|
chmod +x "$publishDir/LanMountainDesktop.Launcher" 2>/dev/null || true
|
|
else
|
|
echo "Warning: Launcher publish dir not found: $launcherDir"
|
|
fi
|
|
|
|
touch "$publishDir/$appDir/.current"
|
|
|
|
echo "New directory structure:"
|
|
find "$publishDir" -maxdepth 2 | head -50
|
|
|
|
rm -rf "$launcherDir"
|
|
|
|
- name: Package as DEB
|
|
run: |
|
|
version="${{ needs.prepare.outputs.version }}"
|
|
source="publish/linux-x64"
|
|
package_name="LanMountainDesktop"
|
|
package_version="${version}"
|
|
arch="amd64"
|
|
desktop_template="LanMountainDesktop/packaging/linux/LanMountainDesktop.desktop"
|
|
icon_source="LanMountainDesktop/packaging/linux/lanmountaindesktop.png"
|
|
|
|
if [ ! -d "$source" ]; then
|
|
echo "Error: Source directory not found: $source"
|
|
ls -la publish/ || echo "publish directory not found"
|
|
exit 1
|
|
fi
|
|
|
|
mkdir -p "build-deb/DEBIAN"
|
|
mkdir -p "build-deb/usr/local/bin"
|
|
mkdir -p "build-deb/usr/share/applications"
|
|
mkdir -p "build-deb/usr/share/pixmaps"
|
|
mkdir -p "build-deb/usr/share/icons/hicolor/256x256/apps"
|
|
|
|
cp -r "$source"/* "build-deb/usr/local/bin/"
|
|
|
|
item_count=$(find build-deb/usr/local/bin -type f 2>/dev/null | wc -l)
|
|
echo "DEB package contains $item_count files"
|
|
|
|
if [ "$item_count" -eq 0 ]; then
|
|
echo "Error: DEB package is empty after copy"
|
|
exit 1
|
|
fi
|
|
|
|
if [ ! -f "$desktop_template" ] || [ ! -f "$icon_source" ]; then
|
|
echo "Error: Linux desktop resources are missing"
|
|
ls -la "LanMountainDesktop/packaging/linux" || true
|
|
exit 1
|
|
fi
|
|
|
|
sed \
|
|
-e "s|@@EXEC@@|/usr/local/bin/LanMountainDesktop.Launcher|g" \
|
|
-e "s|@@ICON@@|lanmountaindesktop|g" \
|
|
"$desktop_template" > "build-deb/usr/share/applications/LanMountainDesktop.desktop"
|
|
|
|
cp "$icon_source" "build-deb/usr/share/pixmaps/lanmountaindesktop.png"
|
|
cp "$icon_source" "build-deb/usr/share/icons/hicolor/256x256/apps/lanmountaindesktop.png"
|
|
|
|
{
|
|
printf '%s\n' '#!/bin/sh'
|
|
printf '%s\n' 'set -e'
|
|
printf '%s\n' 'if command -v update-desktop-database >/dev/null 2>&1; then'
|
|
printf '%s\n' ' update-desktop-database /usr/share/applications >/dev/null 2>&1 || true'
|
|
printf '%s\n' 'fi'
|
|
printf '%s\n' 'if command -v gtk-update-icon-cache >/dev/null 2>&1; then'
|
|
printf '%s\n' ' gtk-update-icon-cache /usr/share/icons/hicolor >/dev/null 2>&1 || true'
|
|
printf '%s\n' 'fi'
|
|
} > "build-deb/DEBIAN/postinst"
|
|
|
|
{
|
|
printf '%s\n' "Package: $package_name"
|
|
printf '%s\n' "Version: $package_version"
|
|
printf '%s\n' "Architecture: $arch"
|
|
printf '%s\n' "Maintainer: LanMountain Team <dev@example.com>"
|
|
printf '%s\n' "Description: LanMountain Desktop Application"
|
|
printf '%s\n' " A desktop application for LanMountain."
|
|
} > "build-deb/DEBIAN/control"
|
|
|
|
chmod 755 "build-deb/usr/local/bin/LanMountainDesktop.Launcher" 2>/dev/null || chmod 755 "build-deb/usr/local/bin"/*
|
|
chmod 644 "build-deb/usr/share/applications/LanMountainDesktop.desktop"
|
|
chmod 644 "build-deb/usr/share/pixmaps/lanmountaindesktop.png"
|
|
chmod 644 "build-deb/usr/share/icons/hicolor/256x256/apps/lanmountaindesktop.png"
|
|
chmod 755 "build-deb/DEBIAN/postinst"
|
|
|
|
if dpkg-deb --build "build-deb" "${package_name}_${package_version}_${arch}.deb"; then
|
|
echo "Successfully created: ${package_name}_${package_version}_${arch}.deb"
|
|
ls -lh "${package_name}_${package_version}_${arch}.deb"
|
|
else
|
|
echo "Error: Failed to build DEB package"
|
|
exit 1
|
|
fi
|
|
|
|
- name: Upload App Payload
|
|
uses: actions/upload-artifact@v4
|
|
with:
|
|
name: app-payload-linux-x64
|
|
path: |
|
|
publish/linux-x64/**
|
|
if-no-files-found: error
|
|
retention-days: 30
|
|
|
|
- name: Upload Installer
|
|
uses: actions/upload-artifact@v4
|
|
with:
|
|
name: installer-linux-x64
|
|
path: "*.deb"
|
|
if-no-files-found: error
|
|
retention-days: 30
|
|
|
|
build-macos:
|
|
needs: prepare
|
|
runs-on: macos-latest
|
|
strategy:
|
|
matrix:
|
|
arch: [x64, arm64]
|
|
name: Build_macOS_${{ matrix.arch }}
|
|
|
|
steps:
|
|
- name: Checkout
|
|
uses: actions/checkout@v4
|
|
with:
|
|
fetch-depth: 0
|
|
submodules: recursive
|
|
ref: ${{ needs.prepare.outputs.checkout_ref }}
|
|
|
|
- name: Install dependencies
|
|
run: brew install portaudio
|
|
|
|
- name: Setup .NET
|
|
uses: actions/setup-dotnet@v4
|
|
with:
|
|
dotnet-version: ${{ env.DOTNET_VERSION }}
|
|
dotnet-quality: 'preview'
|
|
|
|
- name: Restore
|
|
run: dotnet restore ${{ env.Solution_Name }}
|
|
|
|
- name: Build
|
|
run: >
|
|
dotnet build ${{ env.Solution_Name }} -c Release --no-restore -v minimal
|
|
-p:Version=${{ needs.prepare.outputs.version }}
|
|
-p:AssemblyVersion=${{ needs.prepare.outputs.assembly_version }}
|
|
-p:FileVersion=${{ needs.prepare.outputs.assembly_version }}
|
|
-p:InformationalVersion=${{ needs.prepare.outputs.informational_version }}
|
|
|
|
- name: Publish Launcher (AOT)
|
|
run: |
|
|
echo "Publishing Launcher with AOT for macOS ${{ matrix.arch }}..."
|
|
|
|
dotnet publish LanMountainDesktop.Launcher/LanMountainDesktop.Launcher.csproj \
|
|
-c Release \
|
|
-o ./publish/launcher-macos-${{ matrix.arch }} \
|
|
--self-contained \
|
|
-r osx-${{ matrix.arch }} \
|
|
-p:PublishAot=true \
|
|
-p:PublishSingleFile=true \
|
|
-p:IncludeNativeLibrariesForSelfExtract=true \
|
|
-p:EnableCompressionInSingleFile=true \
|
|
-p:DebugType=none \
|
|
-p:DebugSymbols=false
|
|
|
|
if [ $? -ne 0 ]; then
|
|
echo "Launcher AOT publish failed"
|
|
exit 1
|
|
fi
|
|
|
|
echo "Launcher published to: ./publish/launcher-macos-${{ matrix.arch }}"
|
|
ls -lh ./publish/launcher-macos-${{ matrix.arch }}/
|
|
|
|
- name: Publish Main App
|
|
run: |
|
|
dotnet publish LanMountainDesktop/LanMountainDesktop.csproj \
|
|
-c Release \
|
|
-o ./publish/macos-${{ matrix.arch }}-app \
|
|
--self-contained \
|
|
-r osx-${{ matrix.arch }} \
|
|
-p:PublishSingleFile=false \
|
|
-p:SelfContained=true \
|
|
-p:DebugType=none \
|
|
-p:DebugSymbols=false \
|
|
-p:PublishTrimmed=false \
|
|
-p:PublishReadyToRun=false \
|
|
-p:Version=${{ needs.prepare.outputs.version }} \
|
|
-p:AssemblyVersion=${{ needs.prepare.outputs.assembly_version }} \
|
|
-p:FileVersion=${{ needs.prepare.outputs.assembly_version }} \
|
|
-p:InformationalVersion=${{ needs.prepare.outputs.informational_version }}
|
|
|
|
- name: Restructure and Package as DMG
|
|
run: |
|
|
version="${{ needs.prepare.outputs.version }}"
|
|
arch="${{ matrix.arch }}"
|
|
app_name="LanMountainDesktop"
|
|
package_name="${app_name}-${version}-macos-${arch}"
|
|
launcherDir="publish/launcher-macos-$arch"
|
|
appSourceDir="publish/macos-$arch-app"
|
|
|
|
echo "Restructuring for Launcher mode..."
|
|
echo "Version: $version"
|
|
|
|
mkdir -p "${app_name}.app/Contents/MacOS"
|
|
|
|
appDir="app-$version"
|
|
mkdir -p "${app_name}.app/Contents/MacOS/$appDir"
|
|
|
|
if [ -d "$appSourceDir" ]; then
|
|
cp -r "$appSourceDir"/* "${app_name}.app/Contents/MacOS/$appDir/"
|
|
else
|
|
echo "Error: Main app source directory not found: $appSourceDir"
|
|
exit 1
|
|
fi
|
|
|
|
if [ -d "$launcherDir" ]; then
|
|
echo "Copying Launcher to root..."
|
|
cp -r "$launcherDir"/* "${app_name}.app/Contents/MacOS/"
|
|
chmod +x "${app_name}.app/Contents/MacOS/LanMountainDesktop.Launcher" 2>/dev/null || true
|
|
else
|
|
echo "Warning: Launcher publish dir not found: $launcherDir"
|
|
fi
|
|
|
|
touch "${app_name}.app/Contents/MacOS/$appDir/.current"
|
|
|
|
mkdir -p "${app_name}.app/Contents/Resources"
|
|
|
|
item_count=$(find "${app_name}.app/Contents/MacOS" -type f | wc -l)
|
|
echo "App bundle contains $item_count files"
|
|
|
|
if [ "$item_count" -eq 0 ]; then
|
|
echo "Error: App bundle is empty after copy"
|
|
exit 1
|
|
fi
|
|
|
|
{
|
|
printf '%s\n' '<?xml version="1.0" encoding="UTF-8"?>'
|
|
printf '%s\n' '<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">'
|
|
printf '%s\n' '<plist version="1.0">'
|
|
printf '%s\n' '<dict>'
|
|
printf '%s\n' ' <key>CFBundleExecutable</key>'
|
|
printf '%s\n' ' <string>LanMountainDesktop.Launcher</string>'
|
|
printf '%s\n' ' <key>CFBundleName</key>'
|
|
printf '%s\n' ' <string>LanMountain Desktop</string>'
|
|
printf '%s\n' ' <key>CFBundleVersion</key>'
|
|
printf '%s\n' " <string>$version</string>"
|
|
printf '%s\n' ' <key>CFBundleShortVersionString</key>'
|
|
printf '%s\n' " <string>$version</string>"
|
|
printf '%s\n' ' <key>CFBundleIdentifier</key>'
|
|
printf '%s\n' ' <string>com.lanmountain.desktop</string>'
|
|
printf '%s\n' ' <key>CFBundlePackageType</key>'
|
|
printf '%s\n' ' <string>APPL</string>'
|
|
printf '%s\n' '</dict>'
|
|
printf '%s\n' '</plist>'
|
|
} > "${app_name}.app/Contents/Info.plist"
|
|
|
|
mkdir -p dmg-temp
|
|
cp -r "${app_name}.app" dmg-temp/
|
|
|
|
if hdiutil create -volname "${app_name}" -srcfolder dmg-temp -ov -format UDZO "${package_name}.dmg" 2>&1; then
|
|
echo "Successfully created: ${package_name}.dmg"
|
|
ls -lh "${package_name}.dmg"
|
|
else
|
|
echo "Error: Failed to create DMG"
|
|
exit 1
|
|
fi
|
|
|
|
rm -rf dmg-temp "${app_name}.app"
|
|
|
|
- name: Upload
|
|
uses: actions/upload-artifact@v4
|
|
with:
|
|
name: installer-macos-${{ matrix.arch }}
|
|
path: "*.dmg"
|
|
if-no-files-found: error
|
|
retention-days: 30
|
|
|
|
publish-pdc:
|
|
needs: [ prepare, build-windows, build-linux, build-macos ]
|
|
runs-on: ubuntu-latest
|
|
permissions:
|
|
contents: read
|
|
env:
|
|
VERSION: ${{ needs.prepare.outputs.version }}
|
|
PRIMARY_VERSION: ${{ needs.prepare.outputs.version }}
|
|
PDCC_primaryVersion: ${{ needs.prepare.outputs.version }}
|
|
PDCC_version: ${{ needs.prepare.outputs.version }}
|
|
PDC_CLIENT_VERSION: ${{ vars.PDC_CLIENT_VERSION }}
|
|
S3_ENDPOINT: ${{ vars.S3_ENDPOINT }}
|
|
S3_BUCKET: ${{ vars.S3_BUCKET }}
|
|
S3_REGION: ${{ vars.S3_REGION }}
|
|
PDC_ENDPOINT: ${{ vars.PDC_ENDPOINT }}
|
|
PDC_TOKEN: ${{ secrets.PDC_TOKEN }}
|
|
PDC_SIGNING_KEY: ${{ secrets.PDC_SIGNING_KEY }}
|
|
PDC_SIGNING_KEY_PS: ${{ secrets.PDC_SIGNING_KEY_PS }}
|
|
UPDATE_PRIVATE_KEY_PEM: ${{ secrets.UPDATE_PRIVATE_KEY_PEM }}
|
|
S3_ACCESS_KEY: ${{ secrets.S3_ACCESS_KEY }}
|
|
S3_SECRET_KEY: ${{ secrets.S3_SECRET_KEY }}
|
|
steps:
|
|
- name: Checkout
|
|
uses: actions/checkout@v4
|
|
with:
|
|
fetch-depth: 0
|
|
submodules: recursive
|
|
ref: ${{ needs.prepare.outputs.checkout_ref }}
|
|
|
|
- name: Download payload artifacts
|
|
uses: actions/download-artifact@v4
|
|
with:
|
|
path: payload-artifacts
|
|
pattern: app-payload-*
|
|
|
|
- name: Download installer artifacts
|
|
uses: actions/download-artifact@v4
|
|
with:
|
|
path: installer-artifacts
|
|
pattern: installer-*
|
|
|
|
- name: Prepare PDC environment
|
|
shell: pwsh
|
|
run: |
|
|
$ErrorActionPreference = "Stop"
|
|
|
|
function Resolve-PgpPrivateKey([string]$value) {
|
|
if ([string]::IsNullOrWhiteSpace($value)) {
|
|
return $null
|
|
}
|
|
|
|
$trimmed = $value.Trim()
|
|
if ($trimmed -match '-----BEGIN PGP PRIVATE KEY BLOCK-----') {
|
|
return $trimmed
|
|
}
|
|
|
|
try {
|
|
$decoded = [System.Text.Encoding]::UTF8.GetString([Convert]::FromBase64String($trimmed)).Trim()
|
|
if ($decoded -match '-----BEGIN PGP PRIVATE KEY BLOCK-----') {
|
|
return $decoded
|
|
}
|
|
}
|
|
catch {
|
|
}
|
|
|
|
return $trimmed
|
|
}
|
|
|
|
if ([string]::IsNullOrWhiteSpace($env:S3_ENDPOINT) -or
|
|
[string]::IsNullOrWhiteSpace($env:S3_BUCKET)) {
|
|
throw "Missing required S3 variables."
|
|
}
|
|
|
|
$resolvedSigningKey = Resolve-PgpPrivateKey $env:PDC_SIGNING_KEY
|
|
if ([string]::IsNullOrWhiteSpace($resolvedSigningKey)) {
|
|
$resolvedSigningKey = Resolve-PgpPrivateKey $env:UPDATE_PRIVATE_KEY_PEM
|
|
}
|
|
if ([string]::IsNullOrWhiteSpace($resolvedSigningKey)) {
|
|
throw "Missing PDC_SIGNING_KEY (PGP private key)."
|
|
}
|
|
if ($resolvedSigningKey -notmatch '-----BEGIN PGP PRIVATE KEY BLOCK-----') {
|
|
throw "PDC signing key format is invalid. Please provide armored OpenPGP private key in PDC_SIGNING_KEY."
|
|
}
|
|
Add-Content -Path $env:GITHUB_ENV -Value "PDC_SIGNING_KEY<<EOF`n$resolvedSigningKey`nEOF"
|
|
|
|
$workRoot = Join-Path $PWD "pdc-work"
|
|
if (Test-Path $workRoot) {
|
|
Remove-Item -LiteralPath $workRoot -Recurse -Force
|
|
}
|
|
New-Item -ItemType Directory -Path $workRoot -Force | Out-Null
|
|
|
|
$template = Get-Content -Path "phainon.yml" -Raw
|
|
$resolved = $template `
|
|
-replace '__FILE_REPO_ROOT__', "$($env:S3_ENDPOINT.TrimEnd('/'))/$($env:S3_BUCKET)/lanmountain/update/repo/" `
|
|
-replace '__ARCHIVE_ROOT__', "$($env:S3_ENDPOINT.TrimEnd('/'))/$($env:S3_BUCKET)/lanmountain/update/archive"
|
|
|
|
Set-Content -Path (Join-Path $workRoot "phainon.resolved.yml") -Value $resolved -NoNewline
|
|
|
|
python3 -m pip install --user --upgrade awscli
|
|
Add-Content -Path $env:GITHUB_PATH -Value "$HOME/.local/bin"
|
|
|
|
- name: Verify S3 credentials and endpoint
|
|
shell: pwsh
|
|
run: |
|
|
$ErrorActionPreference = "Stop"
|
|
$probeDir = Join-Path $PWD "pdc-work"
|
|
New-Item -ItemType Directory -Path $probeDir -Force | Out-Null
|
|
|
|
$probeFile = Join-Path $probeDir "s3-probe.txt"
|
|
Set-Content -Path $probeFile -Value "lanmountain pdc probe $(Get-Date -Format o)" -NoNewline
|
|
|
|
$probeKey = "lanmountain/update/probe/$($env:GITHUB_RUN_ID)-$($env:GITHUB_RUN_ATTEMPT).txt"
|
|
aws --endpoint-url "$env:S3_ENDPOINT" s3 cp $probeFile "s3://$env:S3_BUCKET/$probeKey" --only-show-errors
|
|
aws --endpoint-url "$env:S3_ENDPOINT" s3 rm "s3://$env:S3_BUCKET/$probeKey" --only-show-errors
|
|
Write-Host "S3 probe succeeded."
|
|
|
|
- name: Bootstrap PDC Endpoint and Token
|
|
shell: pwsh
|
|
run: |
|
|
$ErrorActionPreference = "Stop"
|
|
|
|
$endpoint = $env:PDC_ENDPOINT
|
|
if ([string]::IsNullOrWhiteSpace($endpoint)) {
|
|
$endpoint = "http://127.0.0.1:18765"
|
|
}
|
|
|
|
$token = $env:PDC_TOKEN
|
|
if ([string]::IsNullOrWhiteSpace($token)) {
|
|
$token = "lmd-pdc-local-token"
|
|
}
|
|
|
|
Add-Content -Path $env:GITHUB_ENV -Value "PDC_ENDPOINT=$endpoint"
|
|
Add-Content -Path $env:GITHUB_ENV -Value "PDC_TOKEN=$token"
|
|
Write-Host "Using PDC endpoint: $endpoint"
|
|
|
|
- name: Start Local PDC Mock (Fallback)
|
|
shell: pwsh
|
|
run: |
|
|
$ErrorActionPreference = "Stop"
|
|
|
|
if ([string]::IsNullOrWhiteSpace($env:PDC_ENDPOINT)) {
|
|
throw "PDC_ENDPOINT is empty after bootstrap."
|
|
}
|
|
|
|
$uri = [Uri]$env:PDC_ENDPOINT
|
|
$isLocalHost = $uri.Host -eq "127.0.0.1" -or $uri.Host -eq "localhost"
|
|
if (-not $isLocalHost) {
|
|
Write-Host "Using external PDC endpoint: $($env:PDC_ENDPOINT)"
|
|
exit 0
|
|
}
|
|
|
|
if ([string]::IsNullOrWhiteSpace($env:PDC_TOKEN)) {
|
|
throw "PDC_TOKEN is empty after bootstrap."
|
|
}
|
|
|
|
$port = if ($uri.Port -gt 0) { $uri.Port } else { 18765 }
|
|
$dataDir = Join-Path $PWD "pdc-output/mock-pdc"
|
|
$workDir = Join-Path $PWD "pdc-work"
|
|
$logPath = Join-Path $workDir "pdc-mock.out.log"
|
|
$errLogPath = Join-Path $workDir "pdc-mock.err.log"
|
|
|
|
New-Item -ItemType Directory -Path $workDir -Force | Out-Null
|
|
New-Item -ItemType Directory -Path $dataDir -Force | Out-Null
|
|
if (Test-Path $logPath) {
|
|
Remove-Item -LiteralPath $logPath -Force
|
|
}
|
|
if (Test-Path $errLogPath) {
|
|
Remove-Item -LiteralPath $errLogPath -Force
|
|
}
|
|
|
|
$args = @(
|
|
"scripts/pdc-mock-server.py",
|
|
"--host", "127.0.0.1",
|
|
"--port", $port.ToString(),
|
|
"--token", $env:PDC_TOKEN,
|
|
"--data-dir", $dataDir
|
|
)
|
|
$process = Start-Process -FilePath "python3" -ArgumentList $args -PassThru -RedirectStandardOutput $logPath -RedirectStandardError $errLogPath
|
|
if (-not $process) {
|
|
throw "Failed to launch PDC mock server."
|
|
}
|
|
|
|
$healthUrl = "http://127.0.0.1:$port/healthz"
|
|
$ready = $false
|
|
for ($i = 0; $i -lt 20; $i++) {
|
|
Start-Sleep -Seconds 1
|
|
try {
|
|
$response = Invoke-WebRequest -Uri $healthUrl -Method Get -TimeoutSec 2
|
|
if ($response.StatusCode -eq 200) {
|
|
$ready = $true
|
|
break
|
|
}
|
|
}
|
|
catch {
|
|
}
|
|
}
|
|
|
|
if (-not $ready) {
|
|
if (Test-Path $logPath) {
|
|
Write-Host "===== pdc-mock stdout ====="
|
|
Get-Content -LiteralPath $logPath -ErrorAction SilentlyContinue | Write-Host
|
|
}
|
|
if (Test-Path $errLogPath) {
|
|
Write-Host "===== pdc-mock stderr ====="
|
|
Get-Content -LiteralPath $errLogPath -ErrorAction SilentlyContinue | Write-Host
|
|
}
|
|
throw "PDC mock server did not become ready in time. See $logPath and $errLogPath."
|
|
}
|
|
|
|
Write-Host "Local PDC mock is running at http://127.0.0.1:$port"
|
|
|
|
- name: Install PDCC
|
|
shell: pwsh
|
|
env:
|
|
GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
|
run: |
|
|
./scripts/Install-Pdcc.ps1 -Repository "ClassIsland/PhainonDistributionCenter" -OutputDir "./pdcc"
|
|
|
|
- name: Publish with PDCC
|
|
shell: pwsh
|
|
run: |
|
|
$ErrorActionPreference = "Stop"
|
|
# Map CI vars to the naming convention expected by PDCC tooling.
|
|
$env:S3_Endpoint = $env:S3_ENDPOINT
|
|
$env:S3_Bucket = $env:S3_BUCKET
|
|
$env:S3_Region = $env:S3_REGION
|
|
$env:PDC_Endpoint = $env:PDC_ENDPOINT
|
|
$env:PDC_Token = $env:PDC_TOKEN
|
|
$env:S3_AccessKey = $env:S3_ACCESS_KEY
|
|
$env:S3_SecretKey = $env:S3_SECRET_KEY
|
|
$signingKeyPs = $env:PDC_SIGNING_KEY_PS
|
|
if ([string]::IsNullOrWhiteSpace($signingKeyPs)) {
|
|
# Keep a non-empty value so PDCC required-env check passes on Linux runners.
|
|
$signingKeyPs = " "
|
|
}
|
|
$env:PDC_SigningKeyPs = $signingKeyPs
|
|
# Map config variables with exact names required by phainon placeholders.
|
|
$env:PDCC_version = $env:VERSION
|
|
$env:PDCC_primaryVersion = $env:PRIMARY_VERSION
|
|
$signingKey = $env:PDC_SIGNING_KEY
|
|
if ([string]::IsNullOrWhiteSpace($signingKey)) {
|
|
$signingKey = $env:UPDATE_PRIVATE_KEY_PEM
|
|
}
|
|
if ([string]::IsNullOrWhiteSpace($signingKey)) {
|
|
throw "Missing PDC signing key: PDC_SIGNING_KEY or UPDATE_PRIVATE_KEY_PEM."
|
|
}
|
|
if ($signingKey -notmatch '-----BEGIN PGP PRIVATE KEY BLOCK-----') {
|
|
throw "PDC signing key is not an armored OpenPGP private key."
|
|
}
|
|
$env:PDC_SigningKey = $signingKey
|
|
|
|
$stageRoot = Join-Path $PWD "pdc-stage"
|
|
$payloadRoot = Join-Path $PWD "payload-artifacts"
|
|
$installerRoot = Join-Path $PWD "installer-artifacts"
|
|
$outRoot = Join-Path $PWD "pdc-output"
|
|
$client = Join-Path $PWD "pdcc/PhainonDistributionCenter.Client"
|
|
$config = Join-Path $PWD "pdc-work/phainon.resolved.yml"
|
|
|
|
if (Test-Path $stageRoot) {
|
|
Remove-Item -LiteralPath $stageRoot -Recurse -Force
|
|
}
|
|
if (Test-Path $outRoot) {
|
|
Remove-Item -LiteralPath $outRoot -Recurse -Force
|
|
}
|
|
New-Item -ItemType Directory -Path $stageRoot -Force | Out-Null
|
|
New-Item -ItemType Directory -Path $outRoot -Force | Out-Null
|
|
|
|
$payloadArtifacts = Get-ChildItem -LiteralPath $payloadRoot -Directory
|
|
if (-not $payloadArtifacts) {
|
|
throw "No payload artifacts were downloaded."
|
|
}
|
|
|
|
$installerArtifacts = Get-ChildItem -LiteralPath $installerRoot -Directory
|
|
if (-not $installerArtifacts) {
|
|
throw "No installer artifacts were downloaded."
|
|
}
|
|
|
|
foreach ($installerArtifact in $installerArtifacts) {
|
|
$stagedInstallerDir = Join-Path $stageRoot "installers/$($installerArtifact.Name)"
|
|
./scripts/Prepare-PdccOut.ps1 -SourceDir $installerArtifact.FullName -OutputDir $stagedInstallerDir
|
|
}
|
|
|
|
foreach ($payloadArtifact in $payloadArtifacts) {
|
|
$platformKey = $payloadArtifact.Name -replace '^app-payload-', ''
|
|
$stagedPayloadDir = Join-Path $stageRoot "payloads/$platformKey"
|
|
./scripts/Prepare-PdccOut.ps1 -SourceDir $payloadArtifact.FullName -OutputDir $stagedPayloadDir
|
|
|
|
$publishRoot = Join-Path $outRoot "published/$platformKey"
|
|
New-Item -ItemType Directory -Path $publishRoot -Force | Out-Null
|
|
$parts = $platformKey.Split('-', 2)
|
|
if ($parts.Count -lt 2) {
|
|
throw "Invalid platform key format: $platformKey"
|
|
}
|
|
$os = $parts[0]
|
|
$arch = $parts[1]
|
|
$packageName = "LanMountainDesktop_app_${os}_${arch}_release_folder.zip"
|
|
$packagePath = Join-Path $publishRoot $packageName
|
|
|
|
& $client $config GenerateFileMap $stagedPayloadDir
|
|
if ($LASTEXITCODE -ne 0) {
|
|
throw "PDCC GenerateFileMap failed for $platformKey."
|
|
}
|
|
|
|
if (Test-Path $packagePath) {
|
|
Remove-Item -LiteralPath $packagePath -Force
|
|
}
|
|
Compress-Archive -Path (Join-Path $stagedPayloadDir '*') -DestinationPath $packagePath -Force
|
|
|
|
Push-Location $publishRoot
|
|
try {
|
|
& $client $config Publish $env:PRIMARY_VERSION $env:VERSION $publishRoot
|
|
if ($LASTEXITCODE -ne 0) {
|
|
throw "PDCC Publish failed for $platformKey."
|
|
}
|
|
}
|
|
finally {
|
|
Pop-Location
|
|
}
|
|
}
|
|
|
|
if (Test-Path (Join-Path $stageRoot "installers")) {
|
|
aws --endpoint-url "$env:S3_ENDPOINT" s3 sync (Join-Path $stageRoot "installers") "s3://$env:S3_BUCKET/lanmountain/update/installers/" --only-show-errors
|
|
}
|
|
|
|
- name: Upload PDC Assets
|
|
uses: actions/upload-artifact@v4
|
|
with:
|
|
name: pdc-assets
|
|
path: |
|
|
pdc-output/published/**
|
|
if-no-files-found: error
|
|
retention-days: 90
|
|
|
|
github-release:
|
|
needs: [ prepare, build-windows, build-linux, build-macos, publish-pdc ]
|
|
runs-on: ubuntu-latest
|
|
permissions:
|
|
contents: write
|
|
|
|
steps:
|
|
- name: Download installer artifacts
|
|
uses: actions/download-artifact@v4
|
|
with:
|
|
path: artifacts/installers
|
|
pattern: installer-*
|
|
|
|
- name: Download PDC artifacts
|
|
uses: actions/download-artifact@v4
|
|
with:
|
|
path: artifacts/pdc
|
|
pattern: pdc-assets
|
|
|
|
- name: List artifacts structure
|
|
run: |
|
|
echo "Artifact directory structure:"
|
|
find artifacts -type f -o -type d | sort
|
|
echo ""
|
|
echo "Files found:"
|
|
find artifacts -type f -exec ls -lh {} \;
|
|
echo ""
|
|
echo "Full tree:"
|
|
tree artifacts || find artifacts -print | sed -e 's;[^/]*/;|____;g;s;____|; |;g'
|
|
|
|
- name: Flatten artifacts for release
|
|
run: |
|
|
echo "Organizing artifacts..."
|
|
mkdir -p release-files
|
|
find artifacts/installers -type f \( -name "*.exe" -o -name "*.deb" -o -name "*.dmg" \) -exec cp -v {} release-files/ \;
|
|
find artifacts/pdc -type f \( -name "files-*.json" -o -name "files-*.json.sig" -o -name "update-*.zip" \) -exec cp -v {} release-files/ \;
|
|
echo ""
|
|
echo "Files ready for release:"
|
|
ls -lh release-files/ || echo "No files found in release-files"
|
|
echo ""
|
|
echo "Total files:"
|
|
file_count=$(find release-files -type f | wc -l)
|
|
echo "$file_count"
|
|
if [ "$file_count" -eq 0 ]; then
|
|
echo "Error: No release files found"
|
|
exit 1
|
|
fi
|
|
|
|
- name: Create Release
|
|
uses: ncipollo/release-action@v1
|
|
with:
|
|
tag: ${{ needs.prepare.outputs.tag }}
|
|
name: ${{ needs.prepare.outputs.tag }}
|
|
commit: ${{ github.sha }}
|
|
allowUpdates: true
|
|
draft: false
|
|
prerelease: ${{ github.event.inputs.is_prerelease == 'true' }}
|
|
artifacts: "release-files/**"
|
|
body: |
|
|
## Release ${{ needs.prepare.outputs.version }}
|
|
|
|
### Windows
|
|
- **LanMountainDesktop-Setup-${{ needs.prepare.outputs.version }}-x64.exe** - 64-bit installer (includes .NET runtime)
|
|
- **LanMountainDesktop-Setup-${{ needs.prepare.outputs.version }}-x86.exe** - 32-bit installer (includes .NET runtime)
|
|
|
|
**Note:** The Launcher is now built with AOT (Ahead-of-Time) compilation as a single executable file for faster startup and smaller footprint.
|
|
|
|
Installation: Double-click the .exe file and follow the wizard.
|
|
|
|
### Incremental Update Assets
|
|
- **files-windows-x64.json / files-windows-x64.json.sig / update-windows-x64.zip**
|
|
- **files-windows-x86.json / files-windows-x86.json.sig / update-windows-x86.zip**
|
|
- **files-linux-x64.json / files-linux-x64.json.sig / update-linux-x64.zip**
|
|
|
|
Existing users: Launcher will detect platform-matching signed assets and apply update on next startup.
|
|
|
|
### Linux
|
|
- **LanMountainDesktop-${{ needs.prepare.outputs.version }}-linux-x64.deb** - Debian package (x64)
|
|
|
|
### macOS
|
|
- **LanMountainDesktop-${{ needs.prepare.outputs.version }}-macos-x64.dmg** - Intel processor
|
|
- **LanMountainDesktop-${{ needs.prepare.outputs.version }}-macos-arm64.dmg** - Apple Silicon (M1/M2/M3)
|
|
|
|
See commits for changes.
|
|
token: ${{ secrets.GITHUB_TOKEN }}
|