Files
rmtPocketWatcher/.gitea/workflows/release.yml
HRiggs e82255d8a1
All checks were successful
Flutter Release / get-version (push) Successful in 10s
Flutter Release / build-windows (push) Successful in 1m58s
Flutter Release / build-android (push) Successful in 11m36s
Flutter Release / create-release (push) Successful in 30s
Disable cache
2025-12-15 13:21:50 -05:00

305 lines
12 KiB
YAML

name: Flutter Release
on:
workflow_dispatch:
push:
paths:
- 'flutter_app/pubspec.yaml'
branches:
- main
jobs:
get-version:
runs-on: ubuntu-latest
outputs:
version: ${{ steps.version.outputs.VERSION }}
steps:
- name: Checkout repository
uses: actions/checkout@v4
- name: Get version from pubspec.yaml
id: version
working-directory: flutter_app
shell: bash
run: |
VERSION=$(grep '^version:' pubspec.yaml | sed 's/version: //' | sed 's/+.*//')
echo "VERSION=$VERSION" >> $GITHUB_OUTPUT
echo "Version: $VERSION"
build-windows:
runs-on: windows-latest
needs: get-version
steps:
- name: Checkout repository
uses: actions/checkout@v4
- name: Verify Flutter setup
shell: powershell
run: |
flutter --version
flutter doctor -v
- name: Create production .env file
working-directory: flutter_app
shell: powershell
env:
WS_URL: ${{ secrets.WS_URL }}
API_URL: ${{ secrets.API_URL }}
run: |
"WS_URL=$env:WS_URL" | Out-File -FilePath .env -Encoding utf8
"API_URL=$env:API_URL" | Out-File -FilePath .env -Append -Encoding utf8
- name: Install dependencies
working-directory: flutter_app
shell: powershell
run: flutter pub get
- name: Setup Certificate for Signing
working-directory: flutter_app
shell: powershell
env:
CERT_BASE64: ${{ secrets.CERT_BASE64 }}
CERT_PASSWORD: ${{ secrets.CERT_PASSWORD }}
run: |
if ($env:CERT_BASE64) {
Write-Host "Setting up certificate for code signing..." -ForegroundColor Green
# Create certificates directory if it doesn't exist
if (-not (Test-Path "certificates")) {
New-Item -ItemType Directory -Path "certificates" -Force | Out-Null
}
# Decode base64 certificate and save as PFX
$certBytes = [System.Convert]::FromBase64String($env:CERT_BASE64)
[System.IO.File]::WriteAllBytes("certificates\rmtPocketWatcher.pfx", $certBytes)
Write-Host "✅ Certificate installed successfully" -ForegroundColor Green
} else {
Write-Host "⚠️ No certificate provided - building unsigned" -ForegroundColor Yellow
}
- name: Build Windows release with installer
working-directory: flutter_app
shell: powershell
env:
CERT_PASSWORD: ${{ secrets.CERT_PASSWORD }}
run: |
# Set certificate password environment variable for build script
if ($env:CERT_PASSWORD) {
$env:MSIX_CERTIFICATE_PASSWORD = $env:CERT_PASSWORD
}
# Run our custom build script
.\build_windows.ps1 -Release
# The build script creates: build\rmtPocketWatcher-Windows-v{version}-release.zip
# Rename to simpler format for release
$version = "${{ needs.get-version.outputs.version }}"
# Find the generated zip and rename it
$sourceZip = "build\rmtPocketWatcher-Windows-v$version-release.zip"
if (Test-Path $sourceZip) {
Move-Item $sourceZip "rmtPocketWatcher-Windows-v$version.zip" -Force
Write-Host "Created rmtPocketWatcher-Windows-v$version.zip"
} else {
# Fallback: find any matching zip
$zipFiles = Get-ChildItem -Path "build" -Filter "rmtPocketWatcher-Windows-*.zip" -ErrorAction SilentlyContinue
if ($zipFiles) {
Move-Item $zipFiles[0].FullName "rmtPocketWatcher-Windows-v$version.zip" -Force
Write-Host "Created rmtPocketWatcher-Windows-v$version.zip from $($zipFiles[0].Name)"
}
}
# Build self-extracting portable exe (single file distribution)
Write-Host "Building self-extracting portable executable..."
.\build_sfx.ps1
# Copy SFX exe to root for upload
if (Test-Path "build\windows\sfx\rmtPocketWatcher-v$version-Portable.exe") {
Copy-Item "build\windows\sfx\rmtPocketWatcher-v$version-Portable.exe" "rmtPocketWatcher-Windows-Portable-v$version.exe" -Force
Write-Host "Created rmtPocketWatcher-Windows-Portable-v$version.exe"
}
# Copy MSIX to root for easier upload
$msixFile = Get-ChildItem -Path "build\windows\x64\runner\Release" -Filter "*.msix" -ErrorAction SilentlyContinue | Select-Object -First 1
if ($msixFile) {
Copy-Item $msixFile.FullName "rmtPocketWatcher-Windows-v$version.msix" -Force
Write-Host "Created rmtPocketWatcher-Windows-v$version.msix"
}
# Export certificate for user installation (if certificate exists)
if (Test-Path "certificates\rmtPocketWatcher.pfx") {
Write-Host "Exporting certificate for user installation..."
try {
# Use PowerShell to export the certificate
$pfxPath = "certificates\rmtPocketWatcher.pfx"
$cerPath = "rmtPocketWatcher-Certificate.cer"
$certPassword = if ($env:CERT_PASSWORD) { $env:CERT_PASSWORD } else { "rmtPocketWatcher2024!" }
# Load PFX and export public certificate
$cert = New-Object System.Security.Cryptography.X509Certificates.X509Certificate2($pfxPath, $certPassword)
$certBytes = $cert.Export([System.Security.Cryptography.X509Certificates.X509ContentType]::Cert)
[System.IO.File]::WriteAllBytes($cerPath, $certBytes)
Write-Host "✅ Certificate exported: $cerPath" -ForegroundColor Green
} catch {
Write-Warning "Failed to export certificate: $($_.Exception.Message)"
}
}
# List created artifacts
Write-Host "Artifacts created:"
Get-ChildItem -Filter "*.zip" | ForEach-Object { Write-Host " - $($_.Name)" }
Get-ChildItem -Filter "*Portable*.exe" | ForEach-Object { Write-Host " - $($_.Name)" }
Get-ChildItem -Filter "*.msix" | ForEach-Object { Write-Host " - $($_.Name)" }
Get-ChildItem -Filter "*.cer" | ForEach-Object { Write-Host " - $($_.Name)" }
- name: Upload Windows artifacts
uses: actions/upload-artifact@v3
with:
name: rmtPocketWatcher-Windows
path: |
flutter_app/rmtPocketWatcher-Windows-v${{ needs.get-version.outputs.version }}.zip
flutter_app/rmtPocketWatcher-Windows-Portable-v${{ needs.get-version.outputs.version }}.exe
flutter_app/rmtPocketWatcher-Windows-v${{ needs.get-version.outputs.version }}.msix
flutter_app/rmtPocketWatcher-Certificate.cer
retention-days: 30
build-android:
runs-on: ubuntu-latest
needs: get-version
steps:
- name: Checkout repository
uses: actions/checkout@v4
- name: Setup Java
uses: actions/setup-java@v4
with:
distribution: 'temurin'
java-version: '17'
- name: Setup Flutter
uses: subosito/flutter-action@v2
with:
channel: 'stable'
cache: false
- name: Setup Android SDK
uses: android-actions/setup-android@v3
- name: Accept Android licenses
shell: bash
run: yes | sdkmanager --licenses || true
- name: Verify Flutter setup
shell: bash
run: flutter doctor -v
- name: Create production .env file
working-directory: flutter_app
shell: bash
env:
WS_URL: ${{ secrets.WS_URL }}
API_URL: ${{ secrets.API_URL }}
run: |
echo "WS_URL=$WS_URL" > .env
echo "API_URL=$API_URL" >> .env
echo "Created .env file:"
cat .env | sed 's/=.*/=***/' # Show keys but mask values
- name: Install dependencies
working-directory: flutter_app
shell: bash
run: flutter pub get
- name: Build Android APK
working-directory: flutter_app
shell: bash
run: flutter build apk --release --verbose
- name: Rename APK
working-directory: flutter_app
shell: bash
run: |
cp build/app/outputs/flutter-apk/app-release.apk rmtPocketWatcher-Android-v${{ needs.get-version.outputs.version }}.apk
- name: Upload Android artifact
uses: actions/upload-artifact@v3
with:
name: rmtPocketWatcher-Android
path: flutter_app/rmtPocketWatcher-Android-v${{ needs.get-version.outputs.version }}.apk
retention-days: 30
create-release:
runs-on: ubuntu-latest
needs: [get-version, build-windows, build-android]
steps:
- name: Checkout repository
uses: actions/checkout@v4
- name: Download Windows artifact
uses: actions/download-artifact@v3
with:
name: rmtPocketWatcher-Windows
path: ./artifacts
- name: Download Android artifact
uses: actions/download-artifact@v3
with:
name: rmtPocketWatcher-Android
path: ./artifacts
- name: Create Release
uses: softprops/action-gh-release@v1
with:
tag_name: v${{ needs.get-version.outputs.version }}
name: rmtPocketWatcher v${{ needs.get-version.outputs.version }}
draft: false
prerelease: false
body: |
## rmtPocketWatcher v${{ needs.get-version.outputs.version }}
**Lambda Banking Conglomerate** - Star Citizen AUEC Price Tracker
### Downloads
- **Windows (Full)**: `rmtPocketWatcher-Windows-v${{ needs.get-version.outputs.version }}.zip` - Complete standalone package
- **Windows (Portable)**: `rmtPocketWatcher-Windows-Portable-v${{ needs.get-version.outputs.version }}.exe` - Single self-extracting executable
- **Windows (Installer)**: `rmtPocketWatcher-Windows-v${{ needs.get-version.outputs.version }}.msix` - Windows Store-style installer
- **Certificate**: `rmtPocketWatcher-Certificate.cer` - Required for signed executables (see installation notes)
- **Android**: `rmtPocketWatcher-Android-v${{ needs.get-version.outputs.version }}.apk`
### Features
- Real-time AUEC price tracking from multiple vendors
- Bloomberg-style terminal interface
- Cross-platform native notifications with custom sound
- Historical price charts and trend analysis
- Client-side price alerts
- Vendor comparison tables
### Installation
#### Certificate Installation (Required for signed executables)
If Windows shows "Unknown publisher" warnings, install the certificate first:
1. Download `rmtPocketWatcher-Certificate.cer`
2. Right-click → "Install Certificate"
3. Choose "Local Machine" → "Place all certificates in the following store"
4. Browse → Select "Trusted Root Certification Authorities" → OK
5. Complete the installation
#### Application Installation
**Windows (Full)**: Extract the ZIP file and run `rmtpocketwatcher.exe` - includes all dependencies
**Windows (Portable)**: Just run the .exe - auto-extracts to AppData and launches
**Windows (Installer)**: Double-click the MSIX file for Windows Store-style installation
**Android**: Install the APK file (enable "Install from unknown sources")
---
*Built with Flutter for cross-platform compatibility*
files: |
./artifacts/rmtPocketWatcher-Windows-v${{ needs.get-version.outputs.version }}.zip
./artifacts/rmtPocketWatcher-Windows-Portable-v${{ needs.get-version.outputs.version }}.exe
./artifacts/rmtPocketWatcher-Windows-v${{ needs.get-version.outputs.version }}.msix
./artifacts/rmtPocketWatcher-Certificate.cer
./artifacts/rmtPocketWatcher-Android-v${{ needs.get-version.outputs.version }}.apk
env:
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}