Functions/Winget.ps1
#Function to get the winget command regarding execution context (User, System...) function Get-WingetCmd { #Get WinGet Path (if admin context) # Includes Workaround for ARM64 (removed X64 and replaces it with a wildcard) $ResolveWingetPath = Resolve-Path "$env:ProgramFiles\WindowsApps\Microsoft.DesktopAppInstaller_*_*__8wekyb3d8bbwe" | Sort-Object { [version]($_.Path -replace '^[^\d]+_((\d+\.)*\d+)_.*', '$1') } if ($ResolveWingetPath) { #If multiple version, pick last one $WingetPath = $ResolveWingetPath[-1].Path } #Get Winget Location in User context $WingetCmd = Get-Command winget.exe -ErrorAction SilentlyContinue if ($WingetCmd) { $Script:Winget = $WingetCmd.Source } #Get Winget Location in System context elseif (Test-Path "$WingetPath\winget.exe") { $Script:Winget = "$WingetPath\winget.exe" } else { Write-ToLog "Winget not installed or detected !" "Red" return $false } #Run winget to list apps and accept source agrements (necessary on first run) & $Winget list --accept-source-agreements -s winget | Out-Null #Log Winget installed version $WingetVer = & $Winget --version Write-ToLog "Winget Version: $WingetVer" return $true } #Function to get the outdated app list, in formatted array function Get-WingetOutdatedApps { class Software { [string]$Name [string]$Id [string]$Version [string]$AvailableVersion } #Get list of available upgrades on winget format $upgradeResult = & $Winget upgrade --source winget | Out-String #Start Convertion of winget format to an array. Check if "-----" exists (Winget Error Handling) if (!($upgradeResult -match "-----")) { return "An unusual thing happened (maybe all apps are upgraded):`n$upgradeResult" } #Split winget output to lines $lines = $upgradeResult.Split([Environment]::NewLine) | Where-Object { $_ } # Find the line that starts with "------" $fl = 0 while (-not $lines[$fl].StartsWith("-----")) { $fl++ } #Get header line $fl = $fl - 1 #Get header titles $index = $lines[$fl] -split '\s+' # Line $fl has the header, we can find char where we find ID and Version $idStart = $lines[$fl].IndexOf($index[1]) $versionStart = $lines[$fl].IndexOf($index[2]) $availableStart = $lines[$fl].IndexOf($index[3]) # Now cycle in real package and split accordingly $upgradeList = @() For ($i = $fl + 2; $i -lt $lines.Length; $i++) { $line = $lines[$i] if ( $line.StartsWith("-----")) { #Get header line $fl = $i - 1 #Get header titles $index = $lines[$fl] -split '\s+' # Line $fl has the header, we can find char where we find ID and Version $idStart = $lines[$fl].IndexOf($index[1]) $versionStart = $lines[$fl].IndexOf($index[2]) $availableStart = $lines[$fl].IndexOf($index[3]) } #(Alphanumeric | Literal . | Alphanumeric) - the only unique thing in common for lines with applications if ($line -match "\w\.\w") { $software = [Software]::new() $software.Name = $line.Substring(0, $idStart).TrimEnd() $software.Id = $line.Substring($idStart, $versionStart - $idStart).TrimEnd() $software.Version = $line.Substring($versionStart, $availableStart - $versionStart).TrimEnd() $software.AvailableVersion = $line.Substring($availableStart).TrimEnd() #add formated soft to list $upgradeList += $software } } return $upgradeList | Sort-Object { Get-Random } } function Get-WingetSystemApps { #Json File, where to export system installed apps $jsonFile = "$WorkingDir\winget_system_apps.txt" #Get list of installed Winget apps to json file & $Winget export -o $jsonFile --accept-source-agreements -s winget | Out-Null #Convert json file to txt file with app ids $InstalledApps = get-content $jsonFile | ConvertFrom-Json #Save app list Set-Content $InstalledApps.Sources.Packages.PackageIdentifier -Path $jsonFile #Sort app list Get-Content $jsonFile | Sort-Object | Set-Content $jsonFile } function Install-Winget { Write-Host "`nChecking if Winget is installed" -ForegroundColor Yellow #Check Package Install $TestWinGet = Get-AppxProvisionedPackage -Online | Where-Object { $_.DisplayName -eq "Microsoft.DesktopAppInstaller" } If ([Version]$TestWinGet.Version -ge "2023.118.406.0") { Write-Host "WinGet is Installed" -ForegroundColor Green } else { #Download WinGet MSIXBundle Write-Host "-> Not installed. Downloading WinGet..." $WinGetURL = "https://github.com/microsoft/winget-cli/releases/download/v1.4.10173/Microsoft.DesktopAppInstaller_8wekyb3d8bbwe.msixbundle" $WebClient = New-Object System.Net.WebClient $WebClient.DownloadFile($WinGetURL, "$PSScriptRoot\Microsoft.DesktopAppInstaller_8wekyb3d8bbwe.msixbundle") #Install WinGet MSIXBundle try { Write-Host "-> Installing Winget MSIXBundle for App Installer..." Add-AppxProvisionedPackage -Online -PackagePath "$PSScriptRoot\Microsoft.DesktopAppInstaller_8wekyb3d8bbwe.msixbundle" -SkipLicense | Out-Null Write-Host "Installed Winget MSIXBundle for App Installer" -ForegroundColor Green } catch { Write-Host "Failed to intall Winget MSIXBundle for App Installer..." -ForegroundColor Red } #Remove WinGet MSIXBundle Remove-Item -Path "$PSScriptRoot\Microsoft.DesktopAppInstaller_8wekyb3d8bbwe.msixbundle" -Force -ErrorAction Continue } } function Install-Prerequisites { Write-Host "`nChecking prerequisites..." -ForegroundColor Yellow #Check if Visual C++ 2019 or 2022 installed $Visual2019 = "Microsoft Visual C++ 2015-2019 Redistributable*" $Visual2022 = "Microsoft Visual C++ 2015-2022 Redistributable*" $path = Get-Item HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\*, HKLM:\SOFTWARE\Wow6432Node\Microsoft\Windows\CurrentVersion\Uninstall\* | Where-Object { $_.GetValue("DisplayName") -like $Visual2019 -or $_.GetValue("DisplayName") -like $Visual2022 } #If not installed, ask for installation if (!($path)) { try { if ((Get-CimInStance Win32_OperatingSystem).OSArchitecture -like "*64*") { $OSArch = "x64" } else { $OSArch = "x86" } Write-host "-> Downloading VC_redist.$OSArch.exe..." $SourceURL = "https://aka.ms/vs/17/release/VC_redist.$OSArch.exe" $Installer = $WingetUpdatePath + "\VC_redist.$OSArch.exe" $ProgressPreference = 'SilentlyContinue' Invoke-WebRequest $SourceURL -UseBasicParsing -OutFile (New-Item -Path $Installer -Force) Write-host "-> Installing VC_redist.$OSArch.exe..." Start-Process -FilePath $Installer -Args "/quiet /norestart" -Wait Remove-Item $Installer -ErrorAction Ignore Write-host "-> MS Visual C++ 2015-2022 installed successfully" -ForegroundColor Green } catch { Write-host "-> MS Visual C++ 2015-2022 installation failed." -ForegroundColor Red Start-Sleep 3 } else { Write-host "-> MS Visual C++ 2015-2022 will not be installed." -ForegroundColor Magenta } } else { Write-Host "Prerequisites checked. OK" -ForegroundColor Green } } |