diff --git a/Assemblies/Scripts/Borealis/Remote_Agent_Update_WIN.json b/Assemblies/Scripts/Borealis/Remote_Agent_Update_WIN.json index 689fd54..ff0e0e6 100644 --- a/Assemblies/Scripts/Borealis/Remote_Agent_Update_WIN.json +++ b/Assemblies/Scripts/Borealis/Remote_Agent_Update_WIN.json @@ -4,7 +4,7 @@ "description": "Reaches out to the remote Borealis agent and triggers an automatic unattended update from the Github repository.", "category": "script", "type": "powershell", - "script": "[CmdletBinding()]
param(
    [Parameter()]
    [string]$TaskName = "Borealis Agent",

    [Parameter()]
    [string]$TaskPath
)

# --- Environment-controlled mode ---
$updateMode = ($env:update_mode).ToLowerInvariant()
if (-not $updateMode) { $updateMode = "update" }  # default behavior
$forceUpdate = $updateMode -eq "force_update"

# --- Repository info (fixed) ---
$RepoOwner = "bunny-lab-io"
$RepoName  = "Borealis"
$Branch    = "main"

# region: helper - fetch cached repository SHA from Borealis server
function Get-BorealisServerRepoSha {
    param(
        [Parameter(Mandatory = $true)]
        [string]$BaseUrl,

        [Parameter(Mandatory = $true)]
        [string]$Owner,

        [Parameter(Mandatory = $true)]
        [string]$Repo,

        [Parameter(Mandatory = $true)]
        [string]$Branch
    )

    if ([string]::IsNullOrWhiteSpace($BaseUrl)) {
        throw "Server URL is blank; cannot query repo hash."
    }

    $base = $BaseUrl.TrimEnd('/')
    $repoParam = [System.Uri]::EscapeDataString("$Owner/$Repo")
    $branchParam = [System.Uri]::EscapeDataString($Branch)
    $uri = "$base/api/agent/repo_hash?repo=$repoParam&branch=$branchParam"

    $headers = @{ "User-Agent" = "borealis-agent-updater" }

    $resp = Invoke-WebRequest -Uri $uri -Method GET -Headers $headers -UseBasicParsing -ErrorAction Stop
    $json = $resp.Content | ConvertFrom-Json

    if ($resp.StatusCode -ne 200) {
        $message = $json.error
        if (-not $message) { $message = "HTTP $($resp.StatusCode)" }
        throw "Borealis server responded with an error: $message"
    }

    $sha = ($json.sha)
    if (-not $sha) {
        $message = $json.error
        if ($message) {
            throw "Borealis server did not return a repository hash: $message"
        }
        throw "Borealis server did not return a repository hash."
    }

    return ($sha.ToString()).Trim()
}
# endregion

# --- Locate Agent folder via scheduled task ---
$taskParams = @{ TaskName = $TaskName; ErrorAction = 'Stop' }
if ($TaskPath) { $taskParams.TaskPath = $TaskPath }

try {
    $task = Get-ScheduledTask @taskParams
} catch {
    throw "Scheduled task '$TaskName' was not found."
}

$execAction = $task.Actions | Where-Object { $_.CimClass.CimClassName -eq 'MSFT_TaskExecAction' } | Select-Object -First 1
if (-not $execAction) { throw "Scheduled task '$TaskName' does not contain an executable action." }

$workingDirectory = $execAction.WorkingDirectory
if ([string]::IsNullOrWhiteSpace($workingDirectory)) {
    $candidate = Split-Path -Path $execAction.Execute -Parent
    if ([string]::IsNullOrWhiteSpace($candidate)) {
        throw "Unable to determine working directory for '$TaskName'."
    }
    $workingDirectory = $candidate
}

try {
    $agentRoot = Resolve-Path -Path $workingDirectory -ErrorAction Stop
} catch {
    throw "The working directory '$workingDirectory' does not exist."
}

try {
    $repoRoot = Resolve-Path -Path (Join-Path $agentRoot '..\..') -ErrorAction Stop
} catch {
    throw "Unable to resolve Borealis repository root from '$agentRoot'."
}

$updateScript = Join-Path $repoRoot 'Borealis.ps1'
if (-not (Test-Path $updateScript -PathType Leaf)) {
    throw "Borealis.ps1 not found at '$updateScript'."
}

Write-Verbose "Agent root: $agentRoot"
Write-Verbose "Repo root: $repoRoot"

# --- Determine Borealis server URL ---
$serverBaseUrl = $env:BOREALIS_SERVER_URL
if (-not $serverBaseUrl) {
    $settingsDir = Join-Path $agentRoot 'Settings'
    $serverUrlFile = Join-Path $settingsDir 'server_url.txt'
    if (Test-Path $serverUrlFile) {
        $serverBaseUrl = (Get-Content $serverUrlFile -Raw).Trim()
    }
}
if (-not $serverBaseUrl) { $serverBaseUrl = 'http://localhost:5000' }
$serverBaseUrl = $serverBaseUrl.Trim()
Write-Verbose "Using Borealis server URL: $serverBaseUrl"

# --- Single-file SHA state at agent root ---
$hashFile = Join-Path $agentRoot 'github_repo_hash.txt'

# --- Load last-known SHA (PS 5.1 safe) ---
$lastSha = $null
if (Test-Path $hashFile) {
    $lastSha = (Get-Content $hashFile -Raw).Trim()
}

# --- Query Borealis server for current SHA ---
$newSha = $null
try {
    $newSha = Get-BorealisServerRepoSha -BaseUrl $serverBaseUrl -Owner $RepoOwner -Repo $RepoName -Branch $Branch
} catch {
    throw "Failed to query Borealis server for latest SHA: $($_.Exception.Message)"
}

# --- Determine change ---
$changeDetected = $false
if (-not $lastSha) {
    Write-Verbose "No prior SHA on disk; treating as first run."
    $changeDetected = $true
} elseif ($lastSha -ne $newSha) {
    $changeDetected = $true
    Write-Verbose "Repo updated: $lastSha -> $newSha"
} else {
    Write-Verbose "SHA unchanged: $newSha"
}

# --- Decide if update should run ---
$shouldUpdate = $false
if ($forceUpdate) {
    Write-Verbose "update_mode=force_update → forcing update."
    $shouldUpdate = $true
} elseif ($changeDetected) {
    Write-Verbose "update_mode=update → repo changed or first run."
    $shouldUpdate = $true
} else {
    Write-Verbose "No change; skipping Borealis.ps1 -SilentUpdate."
}

if (-not $shouldUpdate) {
    Write-Host "=============================================="
    Write-Host "Borealis Agent Already Up-to-Date"
    Write-Host "=============================================="
    return
}

# --- SINGLETON MUTEX: prevent concurrent updaters from stomping the same zip ---
$mutex = $null
$gotMutex = $false
try {
    $mutex = New-Object System.Threading.Mutex($false, "Global\BorealisUpdate")
    $gotMutex = $mutex.WaitOne(0)
    if (-not $gotMutex) {
        Write-Verbose "Another update is already running (mutex held). Exiting quietly."
        Write-Host "=============================================="
        Write-Host "Borealis Agent Already Up-to-Date"
        Write-Host "=============================================="
        return
    }

    # --- PRE-FLIGHT: purge stale 'main.zip' with retries to avoid 'in use' errors ---
    $staging = Join-Path $repoRoot 'Update_Staging'
    $zipPath = Join-Path $staging 'main.zip'
    if (Test-Path $zipPath) {
        Write-Verbose "Pre-flight: removing existing $zipPath"
        for ($i=1; $i -le 10; $i++) {
            try {
                Remove-Item -LiteralPath $zipPath -Force -ErrorAction Stop
                break
            } catch {
                Start-Sleep -Milliseconds (100 * $i)  # backoff: 100..1000ms
                if ($i -eq 10) { throw "Pre-flight delete failed; $zipPath appears locked by another process." }
            }
        }
    }

    # Optional: give child script the SHA; Borealis.ps1 can choose to name unique archives if desired
    $env:BOREALIS_EXPECTED_SHA = $newSha

    # --- Perform update ---
    Write-Verbose "Invoking Borealis.ps1 -SilentUpdate..."
    Push-Location $repoRoot
    $updateSucceeded = $false
    try {
        & $updateScript -SilentUpdate
        $updateSucceeded = $?
    } finally {
        Pop-Location
    }

    if (-not $updateSucceeded) {
        throw "Borealis.ps1 -SilentUpdate failed; not advancing stored SHA."
    }

    # --- Save new SHA only after success ---
    if ($newSha) {
        Set-Content -Path $hashFile -Value $newSha -Encoding ASCII
    }

    Write-Host "=============================================="
    Write-Host ("Borealis Agent Updated - New Github Repository Hash: {0}" -f $newSha)
    Write-Host "=============================================="
}
finally {
    if ($mutex -and $gotMutex) { $mutex.ReleaseMutex() | Out-Null }
    if ($mutex) { $mutex.Dispose() }
}
", + "script": "W0NtZGxldEJpbmRpbmcoKV0KcGFyYW0oCiAgICBbUGFyYW1ldGVyKCldCiAgICBbc3RyaW5nXSRUYXNrTmFtZSA9ICJCb3JlYWxpcyBBZ2VudCIsCgogICAgW1BhcmFtZXRlcigpXQogICAgW3N0cmluZ10kVGFza1BhdGgKKQoKJHRhc2tQYXJhbXMgPSBAeyBUYXNrTmFtZSA9ICRUYXNrTmFtZTsgRXJyb3JBY3Rpb24gPSAnU3RvcCcgfQppZiAoJFRhc2tQYXRoKSB7ICR0YXNrUGFyYW1zLlRhc2tQYXRoID0gJFRhc2tQYXRoIH0KCnRyeSB7CiAgICAkdGFzayA9IEdldC1TY2hlZHVsZWRUYXNrIEB0YXNrUGFyYW1zCn0gY2F0Y2ggewogICAgdGhyb3cgIlNjaGVkdWxlZCB0YXNrICckVGFza05hbWUnIHdhcyBub3QgZm91bmQuIgp9CgokZXhlY0FjdGlvbiA9ICR0YXNrLkFjdGlvbnMgfCBXaGVyZS1PYmplY3QgeyAkXy5DaW1DbGFzcy5DaW1DbGFzc05hbWUgLWVxICdNU0ZUX1Rhc2tFeGVjQWN0aW9uJyB9IHwgU2VsZWN0LU9iamVjdCAtRmlyc3QgMQppZiAoLW5vdCAkZXhlY0FjdGlvbikgeyB0aHJvdyAiU2NoZWR1bGVkIHRhc2sgJyRUYXNrTmFtZScgZG9lcyBub3QgY29udGFpbiBhbiBleGVjdXRhYmxlIGFjdGlvbi4iIH0KCiR3b3JraW5nRGlyZWN0b3J5ID0gJGV4ZWNBY3Rpb24uV29ya2luZ0RpcmVjdG9yeQppZiAoW3N0cmluZ106OklzTnVsbE9yV2hpdGVTcGFjZSgkd29ya2luZ0RpcmVjdG9yeSkpIHsKICAgICRjYW5kaWRhdGUgPSBTcGxpdC1QYXRoIC1QYXRoICRleGVjQWN0aW9uLkV4ZWN1dGUgLVBhcmVudAogICAgaWYgKFtzdHJpbmddOjpJc051bGxPcldoaXRlU3BhY2UoJGNhbmRpZGF0ZSkpIHsKICAgICAgICB0aHJvdyAiVW5hYmxlIHRvIGRldGVybWluZSB3b3JraW5nIGRpcmVjdG9yeSBmb3IgJyRUYXNrTmFtZScuIgogICAgfQogICAgJHdvcmtpbmdEaXJlY3RvcnkgPSAkY2FuZGlkYXRlCn0KCnRyeSB7CiAgICAkYWdlbnRSb290ID0gUmVzb2x2ZS1QYXRoIC1QYXRoICR3b3JraW5nRGlyZWN0b3J5IC1FcnJvckFjdGlvbiBTdG9wCn0gY2F0Y2ggewogICAgdGhyb3cgIlRoZSB3b3JraW5nIGRpcmVjdG9yeSAnJHdvcmtpbmdEaXJlY3RvcnknIGRvZXMgbm90IGV4aXN0LiIKfQoKdHJ5IHsKICAgICRyZXBvUm9vdCA9IFJlc29sdmUtUGF0aCAtUGF0aCAoSm9pbi1QYXRoICRhZ2VudFJvb3QgJy4uXC4uJykgLUVycm9yQWN0aW9uIFN0b3AKfSBjYXRjaCB7CiAgICB0aHJvdyAiVW5hYmxlIHRvIHJlc29sdmUgQm9yZWFsaXMgcmVwb3NpdG9yeSByb290IGZyb20gJyRhZ2VudFJvb3QnLiIKfQoKJHVwZGF0ZVNjcmlwdCA9IEpvaW4tUGF0aCAkcmVwb1Jvb3QgJ0JvcmVhbGlzLnBzMScKaWYgKC1ub3QgKFRlc3QtUGF0aCAkdXBkYXRlU2NyaXB0IC1QYXRoVHlwZSBMZWFmKSkgewogICAgdGhyb3cgIkJvcmVhbGlzLnBzMSBub3QgZm91bmQgYXQgJyR1cGRhdGVTY3JpcHQnLiIKfQoKV3JpdGUtVmVyYm9zZSAiQWdlbnQgcm9vdDogJGFnZW50Um9vdCIKV3JpdGUtVmVyYm9zZSAiUmVwbyByb290OiAkcmVwb1Jvb3QiCldyaXRlLVZlcmJvc2UgIkludm9raW5nIEJvcmVhbGlzLnBzMSAtU2lsZW50VXBkYXRlLi4uIgoKUHVzaC1Mb2NhdGlvbiAkcmVwb1Jvb3QKJHVwZGF0ZVN1Y2NlZWRlZCA9ICRmYWxzZQp0cnkgewogICAgJiAkdXBkYXRlU2NyaXB0IC1TaWxlbnRVcGRhdGUKICAgICR1cGRhdGVTdWNjZWVkZWQgPSAkPwp9IGZpbmFsbHkgewogICAgUG9wLUxvY2F0aW9uCn0KCmlmICgtbm90ICR1cGRhdGVTdWNjZWVkZWQpIHsKICAgIHRocm93ICdCb3JlYWxpcy5wczEgLVNpbGVudFVwZGF0ZSBmYWlsZWQuJwp9", "timeout_seconds": 3600, "sites": { "mode": "all", diff --git a/Borealis.ps1 b/Borealis.ps1 index f5dba24..ddc036f 100644 --- a/Borealis.ps1 +++ b/Borealis.ps1 @@ -633,6 +633,153 @@ function Start-AgentScheduledTasks { } } +$BorealisRepoOwner = 'bunny-lab-io' +$BorealisRepoName = 'Borealis' +$BorealisRepoBranch = 'main' + +function Get-BorealisServerRepoSha { + param( + [Parameter(Mandatory = $true)] + [string]$BaseUrl, + + [Parameter(Mandatory = $true)] + [string]$Owner, + + [Parameter(Mandatory = $true)] + [string]$Repo, + + [Parameter(Mandatory = $true)] + [string]$Branch + ) + + if ([string]::IsNullOrWhiteSpace($BaseUrl)) { + throw 'Server URL is blank; cannot query repo hash.' + } + + $base = $BaseUrl.TrimEnd('/') + $repoParam = [System.Uri]::EscapeDataString("$Owner/$Repo") + $branchParam = [System.Uri]::EscapeDataString($Branch) + $uri = "$base/api/agent/repo_hash?repo=$repoParam&branch=$branchParam" + + $headers = @{ 'User-Agent' = 'borealis-agent-updater' } + + $resp = Invoke-WebRequest -Uri $uri -Method GET -Headers $headers -UseBasicParsing -ErrorAction Stop + $json = $resp.Content | ConvertFrom-Json + + if ($resp.StatusCode -ne 200) { + $message = $json.error + if (-not $message) { $message = "HTTP $($resp.StatusCode)" } + throw "Borealis server responded with an error: $message" + } + + $sha = ($json.sha) + if (-not $sha) { + $message = $json.error + if ($message) { + throw "Borealis server did not return a repository hash: $message" + } + throw 'Borealis server did not return a repository hash.' + } + + return ($sha.ToString()).Trim() +} + +function Get-BorealisUpdatePlan { + param( + [Parameter(Mandatory = $true)] + [string]$ScriptDirectory, + + [Parameter(Mandatory = $true)] + [string]$RepoOwner, + + [Parameter(Mandatory = $true)] + [string]$RepoName, + + [Parameter(Mandatory = $true)] + [string]$Branch + ) + + if ([string]::IsNullOrWhiteSpace($ScriptDirectory)) { + throw 'Script directory was not provided.' + } + + $updateMode = $env:update_mode + if ($updateMode) { + $updateMode = $updateMode.ToLowerInvariant() + } else { + $updateMode = 'update' + } + $forceUpdate = $updateMode -eq 'force_update' + + $agentRootCandidate = Join-Path $ScriptDirectory 'Agent\Borealis' + $agentRoot = $ScriptDirectory + if (Test-Path $agentRootCandidate -PathType Container) { + try { + $agentRoot = (Resolve-Path -Path $agentRootCandidate -ErrorAction Stop).Path + } catch { + $agentRoot = $agentRootCandidate + } + } + + $hashFile = Join-Path $agentRoot 'github_repo_hash.txt' + $lastSha = $null + if (Test-Path $hashFile -PathType Leaf) { + $lastSha = (Get-Content $hashFile -Raw).Trim() + } + + $serverBaseUrl = $env:BOREALIS_SERVER_URL + if (-not $serverBaseUrl) { + $settingsDir = Join-Path $agentRoot 'Settings' + $serverUrlFile = Join-Path $settingsDir 'server_url.txt' + if (Test-Path $serverUrlFile -PathType Leaf) { + $serverBaseUrl = (Get-Content $serverUrlFile -Raw).Trim() + } + } + if (-not $serverBaseUrl) { $serverBaseUrl = 'http://localhost:5000' } + $serverBaseUrl = $serverBaseUrl.Trim() + Write-Verbose "Using Borealis server URL: $serverBaseUrl" + + try { + $newSha = Get-BorealisServerRepoSha -BaseUrl $serverBaseUrl -Owner $RepoOwner -Repo $RepoName -Branch $Branch + } catch { + throw "Failed to query Borealis server for latest SHA: $($_.Exception.Message)" + } + + $changeDetected = $false + if (-not $lastSha) { + Write-Verbose 'No prior SHA on disk; treating as first run.' + $changeDetected = $true + } elseif ($lastSha -ne $newSha) { + $changeDetected = $true + Write-Verbose "Repo updated: $lastSha -> $newSha" + } else { + Write-Verbose "SHA unchanged: $newSha" + } + + $shouldUpdate = $false + if ($forceUpdate) { + Write-Verbose 'update_mode=force_update → forcing update.' + $shouldUpdate = $true + } elseif ($changeDetected) { + Write-Verbose 'update_mode=update → repo changed or first run.' + $shouldUpdate = $true + } else { + Write-Verbose 'No change; skipping Borealis.ps1 -SilentUpdate.' + } + + return [pscustomobject]@{ + ShouldUpdate = $shouldUpdate + ForceUpdate = $forceUpdate + ChangeDetected = $changeDetected + LastSha = $lastSha + NewSha = $newSha + HashFile = $hashFile + AgentRoot = $agentRoot + ServerBaseUrl = $serverBaseUrl + UpdateMode = $updateMode + } +} + function Invoke-BorealisUpdate { param( [switch]$Silent @@ -702,13 +849,85 @@ function Invoke-BorealisUpdate { function Invoke-BorealisSilentUpdate { Write-Host "Initiating Borealis silent update workflow..." -ForegroundColor DarkCyan - $managedTasks = Stop-AgentScheduledTasks -TaskNames @('Borealis Agent','Borealis Agent (UserHelper)') + + $plan = Get-BorealisUpdatePlan -ScriptDirectory $scriptDir -RepoOwner $BorealisRepoOwner -RepoName $BorealisRepoName -Branch $BorealisRepoBranch + + if (-not $plan.ShouldUpdate) { + Write-Host "==============================================" + Write-Host "Borealis Agent Already Up-to-Date" + Write-Host "==============================================" + return + } + + $mutex = $null + $gotMutex = $false + $managedTasks = @() try { - Invoke-BorealisUpdate -Silent - } finally { - if ($managedTasks.Count -gt 0) { - Start-AgentScheduledTasks -TaskNames $managedTasks + $mutex = New-Object System.Threading.Mutex($false, 'Global\BorealisUpdate') + $gotMutex = $mutex.WaitOne(0) + if (-not $gotMutex) { + Write-Verbose 'Another update is already running (mutex held). Exiting quietly.' + Write-Host "==============================================" + Write-Host "Borealis Agent Already Up-to-Date" + Write-Host "==============================================" + return } + + $staging = Join-Path $scriptDir 'Update_Staging' + $zipPath = Join-Path $staging 'main.zip' + if (Test-Path $zipPath) { + Write-Verbose "Pre-flight: removing existing $zipPath" + for ($i = 1; $i -le 10; $i++) { + try { + Remove-Item -LiteralPath $zipPath -Force -ErrorAction Stop + break + } catch { + Start-Sleep -Milliseconds (100 * $i) + if ($i -eq 10) { + throw "Pre-flight delete failed; $zipPath appears locked by another process." + } + } + } + } + + $managedTasks = Stop-AgentScheduledTasks -TaskNames @('Borealis Agent','Borealis Agent (UserHelper)') + + $updateSucceeded = $false + try { + if ($plan.NewSha) { + $env:BOREALIS_EXPECTED_SHA = $plan.NewSha + } + + Invoke-BorealisUpdate -Silent + $updateSucceeded = $true + } finally { + if ($plan.NewSha) { + Remove-Item Env:BOREALIS_EXPECTED_SHA -ErrorAction SilentlyContinue + } + + if ($managedTasks.Count -gt 0) { + Start-AgentScheduledTasks -TaskNames $managedTasks + } + } + + if (-not $updateSucceeded) { + throw 'Borealis.ps1 -SilentUpdate failed; not advancing stored SHA.' + } + + if ($plan.NewSha) { + $hashDir = Split-Path $plan.HashFile -Parent + if ($hashDir -and -not (Test-Path $hashDir -PathType Container)) { + New-Item -ItemType Directory -Path $hashDir -Force | Out-Null + } + Set-Content -Path $plan.HashFile -Value $plan.NewSha -Encoding ASCII + } + + Write-Host "==============================================" + Write-Host ("Borealis Agent Updated - New Github Repository Hash: {0}" -f $plan.NewSha) + Write-Host "==============================================" + } finally { + if ($mutex -and $gotMutex) { $mutex.ReleaseMutex() | Out-Null } + if ($mutex) { $mutex.Dispose() } } }