mirror of
https://github.com/ScoopInstaller/Scoop.git
synced 2025-10-30 06:07:56 +00:00
* fix (decompress): `Expand-7zipArchive` only delete temp dir / `$extractDir` if it is empty (#6092) Co-authored-by: Hsiao-nan Cheung <niheaven@gmail.com> * refactor(download): Move download-related functions to 'download.ps1' (#6095) * fix(download): Fallback to default downloader when aria2 fails (#4292) * fix(commands): Handling broken aliases (#6141) * fix(shim): properly check `wslpath`/`cygpath` command first (#6114) Co-authored-by: Hsiao-nan Cheung <niheaven@gmail.com> * fix(scoop-bucket): Add missing import for `no_junction` envs (#6181) Signed-off-by: Chawye Hsu <su+git@chawyehsu.com> * docs(chglog): Update to 0.5.3 (#6258) * perf(shim): Update kiennq-shim to v3.1.2 (#6261) * fix(decompress): Replace deprecated 7ZIPEXTRACT_USE_EXTERNAL config (#6327) Co-authored-by: Hsiao-nan Cheung <niheaven@gmail.com> * fix(scoop-uninstall): Fix uninstaller does not gain Global state (#6430) * global arg * changelog * refactor(Get-Manifest): Select actual source for manifest (#6142) * first step * Revert "first step" This reverts commitc5907c3e25. * refactor(Get-Manifest): Select actual source for installed manifest * rework sub-commands, `scoop-depends` is NOT working at this stage * URI manifest * opt * deprecated manifest * source of manifests * source of manifest pt2 - Mark URI(path/URL/UNC/etc.) query as standalone manifest - Drop `installed` and `available update` items for [query] and [installed] are different sources. * remove variable preventing I forget it * scoop-info: fix source of manifest on bucket * fix `scoop-depends` * Fix Standalone and Source detection * fix global install * Fix scoop-cat, scoop-home - Query for remote manifest * scoop-list: info +deprecated * manifest: Fix first selected manifest * gramma.. * Fix61b3259* length * fix(scoop-depends-tests): Mocking `USE_EXTERNAL_7ZIP` as $false (#6431) * fix(scoop-depends-tests): Mocking `USE_EXTERNAL_7ZIP` as $false to avoding error when it is $true * CHANGELOG * feat(autoupdate): GitHub predefined hashes support (#6416) * feat(autoupdate): predefined hash case for GitHub - Remove `sha256:` prefix in `format_hash()` - Add GitHub support in `get_hash_for_app()` Close #6381 * doc(chglog): GitHub auto hash update * fix(autoupdate): remove prefix only * docs(CHANGELOG): Update to 0.5.3 (#6432) * docs(CHANGELOG): Update to 0.5.3 * 6416 --------- Signed-off-by: Chawye Hsu <su+git@chawyehsu.com> Co-authored-by: Olav Rønnestad Birkeland <6450056+o-l-a-v@users.noreply.github.com> Co-authored-by: kiennq <kien.n.quang@gmail.com> Co-authored-by: HUMORCE <humorce@outlook.com> Co-authored-by: Ryan <sitiom@proton.me> Co-authored-by: Chawye Hsu <su+git@chawyehsu.com> Co-authored-by: Bassel Rachid <101208715+basselworkforce@users.noreply.github.com> Co-authored-by: Wordless Echo <wordless@echo.moe>
469 lines
18 KiB
PowerShell
469 lines
18 KiB
PowerShell
# Usage: scoop update <app> [options]
|
|
# Summary: Update apps, or Scoop itself
|
|
# Help: 'scoop update' updates Scoop to the latest version.
|
|
# 'scoop update <app>' installs a new version of that app, if there is one.
|
|
#
|
|
# You can use '*' in place of <app> to update all apps.
|
|
#
|
|
# Options:
|
|
# -f, --force Force update even when there isn't a newer version
|
|
# -g, --global Update a globally installed app
|
|
# -i, --independent Don't install dependencies automatically
|
|
# -k, --no-cache Don't use the download cache
|
|
# -s, --skip-hash-check Skip hash validation (use with caution!)
|
|
# -q, --quiet Hide extraneous messages
|
|
# -a, --all Update all apps (alternative to '*')
|
|
|
|
. "$PSScriptRoot\..\lib\getopt.ps1"
|
|
. "$PSScriptRoot\..\lib\json.ps1" # 'save_install_info' in 'manifest.ps1' (indirectly)
|
|
. "$PSScriptRoot\..\lib\system.ps1"
|
|
. "$PSScriptRoot\..\lib\shortcuts.ps1"
|
|
. "$PSScriptRoot\..\lib\psmodules.ps1"
|
|
. "$PSScriptRoot\..\lib\decompress.ps1"
|
|
. "$PSScriptRoot\..\lib\manifest.ps1"
|
|
. "$PSScriptRoot\..\lib\versions.ps1"
|
|
. "$PSScriptRoot\..\lib\depends.ps1"
|
|
. "$PSScriptRoot\..\lib\install.ps1"
|
|
. "$PSScriptRoot\..\lib\download.ps1"
|
|
if (get_config USE_SQLITE_CACHE) {
|
|
. "$PSScriptRoot\..\lib\database.ps1"
|
|
}
|
|
|
|
$opt, $apps, $err = getopt $args 'gfiksqa' 'global', 'force', 'independent', 'no-cache', 'skip-hash-check', 'quiet', 'all'
|
|
if ($err) { "scoop update: $err"; exit 1 }
|
|
$global = $opt.g -or $opt.global
|
|
$force = $opt.f -or $opt.force
|
|
$check_hash = !($opt.s -or $opt.'skip-hash-check')
|
|
$use_cache = !($opt.k -or $opt.'no-cache')
|
|
$quiet = $opt.q -or $opt.quiet
|
|
$independent = $opt.i -or $opt.independent
|
|
$all = $opt.a -or $opt.all
|
|
|
|
# load config
|
|
$configRepo = get_config SCOOP_REPO
|
|
if (!$configRepo) {
|
|
$configRepo = 'https://github.com/ScoopInstaller/Scoop'
|
|
set_config SCOOP_REPO $configRepo | Out-Null
|
|
}
|
|
|
|
# Find current update channel from config
|
|
$configBranch = get_config SCOOP_BRANCH
|
|
if (!$configBranch) {
|
|
$configBranch = 'master'
|
|
set_config SCOOP_BRANCH $configBranch | Out-Null
|
|
}
|
|
|
|
if (($PSVersionTable.PSVersion.Major) -lt 5) {
|
|
# check powershell version
|
|
Write-Output 'PowerShell 5 or later is required to run Scoop.'
|
|
Write-Output 'Upgrade PowerShell: https://docs.microsoft.com/en-us/powershell/scripting/install/installing-powershell-core-on-windows'
|
|
break
|
|
}
|
|
$show_update_log = get_config SHOW_UPDATE_LOG $true
|
|
|
|
function Sync-Scoop {
|
|
[CmdletBinding()]
|
|
Param (
|
|
[Switch]$Log
|
|
)
|
|
# Test if Scoop Core is hold
|
|
if (Test-ScoopCoreOnHold) {
|
|
return
|
|
}
|
|
|
|
# check for git
|
|
if (!(Test-GitAvailable)) { abort "Scoop uses Git to update itself. Run 'scoop install git' and try again." }
|
|
|
|
Write-Host 'Updating Scoop...'
|
|
$currentdir = versiondir 'scoop' 'current'
|
|
if (!(Test-Path "$currentdir\.git")) {
|
|
$newdir = "$currentdir\..\new"
|
|
$olddir = "$currentdir\..\old"
|
|
|
|
# get git scoop
|
|
Invoke-Git -ArgumentList @('clone', '-q', $configRepo, '--branch', $configBranch, '--single-branch', $newdir)
|
|
|
|
# check if scoop was successful downloaded
|
|
if (!(Test-Path "$newdir\bin\scoop.ps1")) {
|
|
Remove-Item $newdir -Force -Recurse
|
|
abort "Scoop download failed. If this appears several times, try removing SCOOP_REPO by 'scoop config rm SCOOP_REPO'"
|
|
} else {
|
|
# replace non-git scoop with the git version
|
|
try {
|
|
Rename-Item $currentdir 'old' -ErrorAction Stop
|
|
Rename-Item $newdir 'current' -ErrorAction Stop
|
|
} catch {
|
|
Write-Warning $_
|
|
abort "Scoop update failed. Folder in use. Please rename folders $currentdir to ``old`` and $newdir to ``current``."
|
|
}
|
|
}
|
|
} else {
|
|
if (Test-Path "$currentdir\..\old") {
|
|
Remove-Item "$currentdir\..\old" -Recurse -Force -ErrorAction SilentlyContinue
|
|
}
|
|
|
|
$previousCommit = Invoke-Git -Path $currentdir -ArgumentList @('rev-parse', 'HEAD')
|
|
$currentRepo = Invoke-Git -Path $currentdir -ArgumentList @('config', 'remote.origin.url')
|
|
$currentBranch = Invoke-Git -Path $currentdir -ArgumentList @('branch')
|
|
|
|
$isRepoChanged = !($currentRepo -match $configRepo)
|
|
$isBranchChanged = !($currentBranch -match "\*\s+$configBranch")
|
|
|
|
# Stash uncommitted changes
|
|
if (Invoke-Git -Path $currentdir -ArgumentList @('diff', 'HEAD', '--name-only')) {
|
|
if (get_config AUTOSTASH_ON_CONFLICT) {
|
|
warn 'Uncommitted changes detected. Stashing...'
|
|
Invoke-Git -Path $currentdir -ArgumentList @('stash', 'push', '-m', "WIP at $([System.DateTime]::Now.ToString('o'))", '-u', '-q')
|
|
} else {
|
|
warn 'Uncommitted changes detected. Update aborted.'
|
|
return
|
|
}
|
|
}
|
|
|
|
# Change remote url if the repo is changed
|
|
if ($isRepoChanged) {
|
|
Invoke-Git -Path $currentdir -ArgumentList @('config', 'remote.origin.url', $configRepo)
|
|
}
|
|
|
|
# Fetch and reset local repo if the repo or the branch is changed
|
|
if ($isRepoChanged -or $isBranchChanged) {
|
|
# Reset git fetch refs, so that it can fetch all branches (GH-3368)
|
|
Invoke-Git -Path $currentdir -ArgumentList @('config', 'remote.origin.fetch', '+refs/heads/*:refs/remotes/origin/*')
|
|
# fetch remote branch
|
|
Invoke-Git -Path $currentdir -ArgumentList @('fetch', '--force', 'origin', "refs/heads/$configBranch`:refs/remotes/origin/$configBranch", '-q')
|
|
# checkout and track the branch
|
|
Invoke-Git -Path $currentdir -ArgumentList @('checkout', '-B', $configBranch, '-t', "origin/$configBranch", '-q')
|
|
# reset branch HEAD
|
|
Invoke-Git -Path $currentdir -ArgumentList @('reset', '--hard', "origin/$configBranch", '-q')
|
|
} else {
|
|
Invoke-Git -Path $currentdir -ArgumentList @('pull', '-q')
|
|
}
|
|
|
|
$res = $lastexitcode
|
|
if ($Log) {
|
|
Invoke-GitLog -Path $currentdir -CommitHash $previousCommit
|
|
}
|
|
|
|
if ($res -ne 0) {
|
|
abort 'Update failed.'
|
|
}
|
|
}
|
|
|
|
shim "$currentdir\bin\scoop.ps1" $false
|
|
}
|
|
|
|
function Sync-Bucket {
|
|
Param (
|
|
[Switch]$Log
|
|
)
|
|
Write-Host 'Updating Buckets...'
|
|
|
|
if (!(Test-Path (Join-Path (Find-BucketDirectory 'main' -Root) '.git'))) {
|
|
info "Converting 'main' bucket to git repo..."
|
|
$status = rm_bucket 'main'
|
|
if ($status -ne 0) {
|
|
abort "Failed to remove local 'main' bucket."
|
|
}
|
|
$status = add_bucket 'main' (known_bucket_repo 'main')
|
|
if ($status -ne 0) {
|
|
abort "Failed to add remote 'main' bucket."
|
|
}
|
|
}
|
|
|
|
|
|
$buckets = Get-LocalBucket | ForEach-Object {
|
|
$path = Find-BucketDirectory $_ -Root
|
|
return @{
|
|
name = $_
|
|
valid = Test-Path (Join-Path $path '.git')
|
|
path = $path
|
|
}
|
|
}
|
|
|
|
$buckets | Where-Object { !$_.valid } | ForEach-Object { Write-Host "'$($_.name)' is not a git repository. Skipped." }
|
|
|
|
$updatedFiles = [System.Collections.ArrayList]::Synchronized([System.Collections.ArrayList]::new())
|
|
$removedFiles = [System.Collections.ArrayList]::Synchronized([System.Collections.ArrayList]::new())
|
|
if ($PSVersionTable.PSVersion.Major -ge 7) {
|
|
# Parallel parameter is available since PowerShell 7
|
|
$buckets | Where-Object { $_.valid } | ForEach-Object -ThrottleLimit 5 -Parallel {
|
|
. "$using:PSScriptRoot\..\lib\core.ps1"
|
|
. "$using:PSScriptRoot\..\lib\buckets.ps1"
|
|
|
|
$name = $_.name
|
|
$bucketLoc = $_.path
|
|
$innerBucketLoc = Find-BucketDirectory $name
|
|
|
|
$previousCommit = Invoke-Git -Path $bucketLoc -ArgumentList @('rev-parse', 'HEAD')
|
|
Invoke-Git -Path $bucketLoc -ArgumentList @('pull', '-q')
|
|
if ($using:Log) {
|
|
Invoke-GitLog -Path $bucketLoc -Name $name -CommitHash $previousCommit
|
|
}
|
|
if (get_config USE_SQLITE_CACHE) {
|
|
Invoke-Git -Path $bucketLoc -ArgumentList @('diff', '--name-status', $previousCommit) | ForEach-Object {
|
|
$status, $file = $_ -split '\s+', 2
|
|
$filePath = Join-Path $bucketLoc $file
|
|
if ($filePath -match "^$([regex]::Escape($innerBucketLoc)).*\.json$") {
|
|
switch ($status) {
|
|
{ $_ -in 'A', 'M', 'R' } {
|
|
[void]($using:updatedFiles).Add($filePath)
|
|
}
|
|
'D' {
|
|
[void]($using:removedFiles).Add([pscustomobject]@{
|
|
Name = ([System.IO.FileInfo]$file).BaseName
|
|
Bucket = $name
|
|
})
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
$buckets | Where-Object { $_.valid } | ForEach-Object {
|
|
$name = $_.name
|
|
$bucketLoc = $_.path
|
|
$innerBucketLoc = Find-BucketDirectory $name
|
|
|
|
$previousCommit = Invoke-Git -Path $bucketLoc -ArgumentList @('rev-parse', 'HEAD')
|
|
Invoke-Git -Path $bucketLoc -ArgumentList @('pull', '-q')
|
|
if ($Log) {
|
|
Invoke-GitLog -Path $bucketLoc -Name $name -CommitHash $previousCommit
|
|
}
|
|
if (get_config USE_SQLITE_CACHE) {
|
|
Invoke-Git -Path $bucketLoc -ArgumentList @('diff', '--name-status', $previousCommit) | ForEach-Object {
|
|
$status, $file = $_ -split '\s+', 2
|
|
$filePath = Join-Path $bucketLoc $file
|
|
if ($filePath -match "^$([regex]::Escape($innerBucketLoc)).*\.json$") {
|
|
switch ($status) {
|
|
{ $_ -in 'A', 'M', 'R' } {
|
|
[void]($updatedFiles).Add($filePath)
|
|
}
|
|
'D' {
|
|
[void]($removedFiles).Add([pscustomobject]@{
|
|
Name = ([System.IO.FileInfo]$file).BaseName
|
|
Bucket = $name
|
|
})
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if ((get_config USE_SQLITE_CACHE) -and ($updatedFiles.Count -gt 0 -or $removedFiles.Count -gt 0)) {
|
|
info 'Updating cache...'
|
|
Set-ScoopDB -Path $updatedFiles
|
|
$removedFiles | Remove-ScoopDBItem
|
|
}
|
|
}
|
|
|
|
function update($app, $global, $quiet = $false, $independent, $suggested, $use_cache = $true, $check_hash = $true) {
|
|
$old_version = Select-CurrentVersion -AppName $app -Global:$global
|
|
$old_manifest = installed_manifest $app $old_version $global
|
|
$install = install_info $app $old_version $global
|
|
|
|
# re-use architecture, bucket and url from first install
|
|
$architecture = Format-ArchitectureString $install.architecture
|
|
$bucket = $install.bucket
|
|
if ($null -eq $bucket) {
|
|
$bucket = 'main'
|
|
}
|
|
$url = $install.url
|
|
|
|
$manifest = manifest $app $bucket $url
|
|
$version = $manifest.version
|
|
$is_nightly = $version -eq 'nightly'
|
|
if ($is_nightly) {
|
|
$version = nightly_version $quiet
|
|
$check_hash = $false
|
|
}
|
|
|
|
if (!$force -and ($old_version -eq $version)) {
|
|
if (!$quiet) {
|
|
warn "The latest version of '$app' ($version) is already installed."
|
|
}
|
|
return
|
|
}
|
|
if (!$version) {
|
|
# installed from a custom bucket/no longer supported
|
|
error "No manifest available for '$app'."
|
|
return
|
|
}
|
|
|
|
Write-Host "Updating '$app' ($old_version -> $version)"
|
|
|
|
#region Workaround for #2952
|
|
if (test_running_process $app $global) {
|
|
Write-Host 'Running process detected, skip updating.'
|
|
return
|
|
}
|
|
#endregion Workaround for #2952
|
|
|
|
# region Workaround
|
|
# Workaround for https://github.com/ScoopInstaller/Scoop/issues/2220 until install is refactored
|
|
# Remove and replace whole region after proper fix
|
|
Write-Host 'Downloading new version'
|
|
if (Test-Aria2Enabled) {
|
|
Invoke-CachedAria2Download $app $version $manifest $architecture $cachedir $manifest.cookie $true $check_hash
|
|
} else {
|
|
$urls = script:url $manifest $architecture
|
|
|
|
foreach ($url in $urls) {
|
|
Invoke-CachedDownload $app $version $url $null $manifest.cookie $true
|
|
|
|
if ($check_hash) {
|
|
$manifest_hash = hash_for_url $manifest $url $architecture
|
|
$source = cache_path $app $version $url
|
|
$ok, $err = check_hash $source $manifest_hash $(show_app $app $bucket)
|
|
|
|
if (!$ok) {
|
|
error $err
|
|
if (Test-Path $source) {
|
|
# rm cached file
|
|
Remove-Item -Force $source
|
|
}
|
|
if ($url.Contains('sourceforge.net')) {
|
|
Write-Host -f yellow 'SourceForge.net is known for causing hash validation fails. Please try again before opening a ticket.'
|
|
}
|
|
abort $(new_issue_msg $app $bucket 'hash check failed')
|
|
}
|
|
}
|
|
}
|
|
}
|
|
# There is no need to check hash again while installing
|
|
$check_hash = $false
|
|
# endregion Workaround
|
|
|
|
$dir = versiondir $app $old_version $global
|
|
$persist_dir = persistdir $app $global
|
|
|
|
Invoke-HookScript -HookType 'pre_uninstall' -Manifest $old_manifest -Arch $architecture
|
|
|
|
Write-Host "Uninstalling '$app' ($old_version)"
|
|
Invoke-Installer -Path $dir -Manifest $old_manifest -ProcessorArchitecture $architecture -Uninstall
|
|
rm_shims $app $old_manifest $global $architecture
|
|
|
|
# If a junction was used during install, that will have been used
|
|
# as the reference directory. Otherwise it will just be the version
|
|
# directory.
|
|
$refdir = unlink_current $dir
|
|
uninstall_psmodule $old_manifest $refdir $global
|
|
env_rm_path $old_manifest $refdir $global $architecture
|
|
env_rm $old_manifest $global $architecture
|
|
|
|
if ($force -and ($old_version -eq $version)) {
|
|
if (!(Test-Path "$dir/../_$version.old")) {
|
|
Move-Item "$dir" "$dir/../_$version.old"
|
|
} else {
|
|
$i = 1
|
|
While (Test-Path "$dir/../_$version.old($i)") {
|
|
$i++
|
|
}
|
|
Move-Item "$dir" "$dir/../_$version.old($i)"
|
|
}
|
|
}
|
|
|
|
Invoke-HookScript -HookType 'post_uninstall' -Manifest $old_manifest -Arch $architecture
|
|
|
|
if ($bucket) {
|
|
# add bucket name it was installed from
|
|
$app = "$bucket/$app"
|
|
}
|
|
if ($install.url) {
|
|
# use the url of the install json if the application was installed through url
|
|
$app = $install.url
|
|
}
|
|
|
|
if ($independent) {
|
|
install_app $app $architecture $global $suggested $use_cache $check_hash
|
|
} else {
|
|
# Also add missing dependencies
|
|
$apps = @(Get-Dependency $app $architecture) -ne $app
|
|
ensure_none_failed $apps
|
|
$apps.Where({ !(installed $_) }) + $app | ForEach-Object { install_app $_ $architecture $global $suggested $use_cache $check_hash }
|
|
}
|
|
}
|
|
|
|
if (-not ($apps -or $all)) {
|
|
if ($global) {
|
|
error 'scoop update: --global is invalid when <app> is not specified.'
|
|
exit 1
|
|
}
|
|
if (!$use_cache) {
|
|
error 'scoop update: --no-cache is invalid when <app> is not specified.'
|
|
exit 1
|
|
}
|
|
Sync-Scoop -Log:$show_update_log
|
|
Sync-Bucket -Log:$show_update_log
|
|
set_config LAST_UPDATE ([System.DateTime]::Now.ToString('o')) | Out-Null
|
|
success 'Scoop was updated successfully!'
|
|
} else {
|
|
if ($global -and !(is_admin)) {
|
|
'ERROR: You need admin rights to update global apps.'; exit 1
|
|
}
|
|
|
|
$outdated = @()
|
|
$updateScoop = $null -ne ($apps | Where-Object { $_ -eq 'scoop' }) -or (is_scoop_outdated)
|
|
$apps = $apps | Where-Object { $_ -ne 'scoop' }
|
|
$apps_param = $apps
|
|
|
|
if ($updateScoop) {
|
|
Sync-Scoop -Log:$show_update_log
|
|
Sync-Bucket -Log:$show_update_log
|
|
set_config LAST_UPDATE ([System.DateTime]::Now.ToString('o')) | Out-Null
|
|
success 'Scoop was updated successfully!'
|
|
}
|
|
|
|
if ($apps_param -eq '*' -or $all) {
|
|
$apps = applist (installed_apps $false) $false
|
|
if ($global) {
|
|
$apps += applist (installed_apps $true) $true
|
|
}
|
|
} else {
|
|
if ($apps_param) {
|
|
$apps = Confirm-InstallationStatus $apps_param -Global:$global
|
|
}
|
|
}
|
|
if ($apps) {
|
|
$apps | ForEach-Object {
|
|
($app, $global) = $_
|
|
$status = app_status $app $global
|
|
if ($status.installed -and ($force -or $status.outdated)) {
|
|
if (!$status.hold) {
|
|
$outdated += applist $app $global
|
|
Write-Host -f yellow ("$app`: $($status.version) -> $($status.latest_version){0}" -f ('', ' (global)')[$global])
|
|
} else {
|
|
warn "'$app' is held to version $($status.version)"
|
|
}
|
|
} elseif ($apps_param -ne '*' -and !$all) {
|
|
if ($status.installed) {
|
|
ensure_none_failed $app
|
|
Write-Host "$app`: $($status.version) (latest version)" -ForegroundColor Green
|
|
} else {
|
|
info 'Please reinstall it or fix the manifest.'
|
|
}
|
|
}
|
|
}
|
|
|
|
if ($outdated -and ((Test-Aria2Enabled) -and (get_config 'aria2-warning-enabled' $true))) {
|
|
warn "Scoop uses 'aria2c' for multi-connection downloads."
|
|
warn "Should it cause issues, run 'scoop config aria2-enabled false' to disable it."
|
|
warn "To disable this warning, run 'scoop config aria2-warning-enabled false'."
|
|
}
|
|
if ($outdated.Length -gt 1) {
|
|
Write-Host -f DarkCyan "Updating $($outdated.Length) outdated apps:"
|
|
} elseif ($outdated.Length -eq 0) {
|
|
Write-Host -f Green "Latest versions for all apps are installed! For more information try 'scoop status'"
|
|
} else {
|
|
Write-Host -f DarkCyan 'Updating one outdated app:'
|
|
}
|
|
}
|
|
|
|
$suggested = @{}
|
|
# $outdated is a list of ($app, $global) tuples
|
|
$outdated | ForEach-Object { update @_ $quiet $independent $suggested $use_cache $check_hash }
|
|
}
|
|
|
|
exit 0
|