From 101954cccb849e15f6025cb6819b73dd117d5c38 Mon Sep 17 00:00:00 2001 From: Kaleb Luedtke Date: Mon, 29 Nov 2021 18:12:38 -0600 Subject: [PATCH] Installer Detection Enhancements (#35735) * Prepare file for 1.1.0 manifests - Trenly/winget-pkgs#31 * Installer detection improvements (#111) * Begin update in place * Condense installer switch logic * Fix Null values * Prevent null checking * Cleanup * Fix missed variable * Bump version * Update function calls * Fix empty ProductCode * Confirm Architecture when Automatically Detected * Remove Debug Line * Add duplicate url check for Quick update * Remove line used in debugging * Remove unneeded comment * Resolves https://github.com/Trenly/winget-pkgs/issues/121 Co-authored-by: Vedant <83997633+vedantmgoyal2009@users.noreply.github.com> --- Tools/YamlCreate.ps1 | 549 +++++++++++++++++++++---------------------- 1 file changed, 267 insertions(+), 282 deletions(-) diff --git a/Tools/YamlCreate.ps1 b/Tools/YamlCreate.ps1 index 466c777cfc2a8..5b6572c1e64dd 100644 --- a/Tools/YamlCreate.ps1 +++ b/Tools/YamlCreate.ps1 @@ -1,5 +1,5 @@ #Requires -Version 5 -[Diagnostics.CodeAnalysis.SuppressMessageAttribute('PSAvoidUsingWriteHost', '', Justification='This script is not intended to have any outputs piped')] +[Diagnostics.CodeAnalysis.SuppressMessageAttribute('PSAvoidUsingWriteHost', '', Justification = 'This script is not intended to have any outputs piped')] Param ( @@ -38,8 +38,8 @@ if ($Settings) { exit } -$ScriptHeader = '# Created with YamlCreate.ps1 v2.0.0' -$ManifestVersion = '1.0.0' +$ScriptHeader = '# Created with YamlCreate.ps1 v2.0.2' +$ManifestVersion = '1.1.0' $PSDefaultParameterValues = @{ '*:Encoding' = 'UTF8' } $Utf8NoBomEncoding = New-Object System.Text.UTF8Encoding $False $ofs = ', ' @@ -86,11 +86,11 @@ if (-not(Get-Module -ListAvailable -Name powershell-yaml)) { # Fetch Schema data from github for entry validation, key ordering, and automatic commenting try { $ProgressPreference = 'SilentlyContinue' - $LocaleSchema = @(Invoke-WebRequest 'https://raw.githubusercontent.com/microsoft/winget-cli/master/schemas/JSON/manifests/v1.0.0/manifest.defaultLocale.1.0.0.json' -UseBasicParsing | ConvertFrom-Json) + $LocaleSchema = @(Invoke-WebRequest "https://raw.githubusercontent.com/microsoft/winget-cli/master/schemas/JSON/manifests/v$ManifestVersion/manifest.defaultLocale.$ManifestVersion.json" -UseBasicParsing | ConvertFrom-Json) $LocaleProperties = (ConvertTo-Yaml $LocaleSchema.properties | ConvertFrom-Yaml -Ordered).Keys - $VersionSchema = @(Invoke-WebRequest 'https://raw.githubusercontent.com/microsoft/winget-cli/master/schemas/JSON/manifests/v1.0.0/manifest.version.1.0.0.json' -UseBasicParsing | ConvertFrom-Json) + $VersionSchema = @(Invoke-WebRequest "https://raw.githubusercontent.com/microsoft/winget-cli/master/schemas/JSON/manifests/v$ManifestVersion/manifest.version.$ManifestVersion.json" -UseBasicParsing | ConvertFrom-Json) $VersionProperties = (ConvertTo-Yaml $VersionSchema.properties | ConvertFrom-Yaml -Ordered).Keys - $InstallerSchema = @(Invoke-WebRequest 'https://raw.githubusercontent.com/microsoft/winget-cli/master/schemas/JSON/manifests/v1.0.0/manifest.installer.1.0.0.json' -UseBasicParsing | ConvertFrom-Json) + $InstallerSchema = @(Invoke-WebRequest "https://raw.githubusercontent.com/microsoft/winget-cli/master/schemas/JSON/manifests/v$ManifestVersion/manifest.installer.$ManifestVersion.json" -UseBasicParsing | ConvertFrom-Json) $InstallerProperties = (ConvertTo-Yaml $InstallerSchema.properties | ConvertFrom-Yaml -Ordered).Keys $InstallerSwitchProperties = (ConvertTo-Yaml $InstallerSchema.definitions.InstallerSwitches.properties | ConvertFrom-Yaml -Ordered).Keys $InstallerEntryProperties = (ConvertTo-Yaml $InstallerSchema.definitions.Installer.properties | ConvertFrom-Yaml -Ordered).Keys @@ -187,13 +187,13 @@ Function Test-String { if ($PSBoundParameters.ContainsKey('MatchPattern')) { $_isValid = $_isValid -and ($InputString -match $MatchPattern) } - if ($AllowNull -and [string]::IsNullOrEmpty($InputString)) { + if ($AllowNull -and [string]::IsNullOrWhiteSpace($InputString)) { $_isValid = $true - } elseif ($NotNull -and [string]::IsNullOrEmpty($InputString)) { + } elseif ($NotNull -and [string]::IsNullOrWhiteSpace($InputString)) { $_isValid = $false } if ($IsNull) { - $_isValid = [string]::IsNullOrEmpty($InputString) + $_isValid = [string]::IsNullOrWhiteSpace($InputString) } if ($Not) { @@ -343,52 +343,27 @@ Function Get-InstallerFile { ) # Create a filename based on the Package Identifier and Version; Try to get the extension from the URL # If the extension isn't found, use a custom one - $Filename = "$PackageIdentifier v$PackageVersion" + $(if([System.IO.Path]::HasExtension($URI)) { [System.IO.Path]::GetExtension($URI) } else { '.winget-tmp' }) - $script:dest = Join-Path -Path $env:TEMP -ChildPath $Filename + $_Filename = "$PackageIdentifier v$PackageVersion - $(Get-Date -f 'yyyy.MM.dd-hh.mm.ss')" + $(if ([System.IO.Path]::HasExtension($URI)) { [System.IO.Path]::GetExtension($URI) } else { '.winget-tmp' }) + $_OutFile = Join-Path -Path $env:TEMP -ChildPath $_Filename # Create a new web client for downloading the file - $WebClient = [System.Net.WebClient]::new() + $_WebClient = [System.Net.WebClient]::new() # If the system has a default proxy set, use it # Powershell Core will automatically use this, so it's only necessary for PS5 - if ($PSVersionTable.PSVersion.Major -lt 6){ $WebClient.Proxy = [System.Net.WebProxy]::GetDefaultProxy() } + if ($PSVersionTable.PSVersion.Major -lt 6) { $_WebClient.Proxy = [System.Net.WebProxy]::GetDefaultProxy() } # Download the file - $WebClient.DownloadFile($URI, $dest) + $_WebClient.DownloadFile($URI, $_OutFile) # Dispose of the web client to release the resources it uses - $WebClient.Dispose() -} + $_WebClient.Dispose() -# Prompts the user to enter installer values -# Sets the $script:Installers value as an output -# Returns void -Function Read-InstallerEntry { - # Clear prompted variables to ensure data from previous installer entries is not used for new entries - $InstallerValues = @( - 'Architecture' - 'InstallerType' - 'InstallerUrl' - 'InstallerSha256' - 'SignatureSha256' - 'PackageFamilyName' - 'Custom' - 'Silent' - 'SilentWithProgress' - 'ProductCode' - 'Scope' - 'InstallerLocale' - 'UpgradeBehavior' - 'AnotherInstaller' - ) - Foreach ($InstallerValue in $InstallerValues) { Clear-Variable -Name $InstallerValue -Force -ErrorAction SilentlyContinue } - - # Request user enter Installer URL - $InstallerUrl = Request-InstallerUrl + return $_OutFile +} - # Get or request Installer Sha256 - # Check the settings to see if we need to display this menu +Function Get-UserSavePreference { switch ($ScriptSettings.SaveToTemporaryFolder) { - 'always' { $script:SaveOption = '0' } - 'never' { $script:SaveOption = '1' } - 'manual' { $script:SaveOption = '2' } + 'always' { $_Preference = '0' } + 'never' { $_Preference = '1' } + 'manual' { $_Preference = '2' } default { $_menu = @{ entries = @('[Y] Yes'; '*[N] No'; '[M] Manually Enter SHA256') @@ -396,83 +371,132 @@ Function Read-InstallerEntry { DefaultString = 'N' } switch ( Invoke-KeypressMenu -Prompt $_menu['Prompt'] -Entries $_menu['Entries'] -DefaultString $_menu['DefaultString']) { - 'Y' { $script:SaveOption = '0' } - 'N' { $script:SaveOption = '1' } - 'M' { $script:SaveOption = '2' } - default { $script:SaveOption = '1' } + 'Y' { $_Preference = '0' } + 'N' { $_Preference = '1' } + 'M' { $_Preference = '2' } + default { $_Preference = '1' } } } } + return $_Preference +} - # If user did not select manual entry for Sha256, download file and calculate hash - # Also attempt to detect installer type and architecture - if ($script:SaveOption -ne '2') { - Write-Host - $start_time = Get-Date - Write-Host $NewLine - Write-Host 'Downloading URL. This will take a while...' -ForegroundColor Blue - try { - Get-InstallerFile -URI $InstallerUrl -PackageIdentifier $PackageIdentifier -PackageVersion $PackageVersion - } catch { - # Here we also want to pass any exceptions through for potential debugging - throw [System.Net.WebException]::new('The file could not be downloaded. Try running the script again', $_.Exception) - } finally { - Write-Host "Time taken: $((Get-Date).Subtract($start_time).Seconds) second(s)" -ForegroundColor Green - - $InstallerSha256 = (Get-FileHash -Path $script:dest -Algorithm SHA256).Hash +Function Get-PathInstallerType { + Param + ( + [Parameter(Mandatory = $true, Position = 0)] + [string] $Path + ) - if ($script:dest -match '\.msix(bundle){0,1}$') { $InstallerType = 'msix' } - elseif ($script:dest -match '\.msi$') { $InstallerType = 'msi' } - elseif ($script:dest -match '\.appx(bundle){0,1}$') { $InstallerType = 'appx' } - elseif ($script:dest -match '\.zip$') { $InstallerType = 'zip' } + if ($Path -match '\.msix(bundle){0,1}$') { return 'msix' } + if ($Path -match '\.msi$') { return 'msi' } + if ($Path -match '\.appx(bundle){0,1}$') { return 'appx' } + if ($Path -match '\.zip$') { return 'zip' } + return $null +} - if ($InstallerUrl -match '\b(x|win){0,1}64\b') { $architecture = 'x64' } - elseif ($InstallerUrl -match '\b((win|ia)32)|(x{0,1}86)\b') { $architecture = 'x86' } - elseif ($InstallerUrl -match '\b(arm|aarch)64\b') { $architecture = 'arm64' } - elseif ($InstallerUrl -match '\barm\b') { $architecture = 'arm' } +Function Get-UriArchitecture { + Param + ( + [Parameter(Mandatory = $true, Position = 0)] + [string] $URI + ) - $MSIProductCode = $(Get-AppLockerFileInformation -Path $script:dest | Select-Object Publisher | Select-String -Pattern '{[A-Z0-9]{8}-([A-Z0-9]{4}-){3}[A-Z0-9]{12}}').Matches + if ($URI -match '\b(x|win){0,1}64\b') { return 'x64' } + if ($URI -match '\b((win|ia)32)|(x{0,1}86)\b') { return 'x86' } + if ($URI -match '\b(arm|aarch)64\b') { return 'arm64' } + if ($URI -match '\barm\b') { return 'arm' } + return $null +} - if ($script:SaveOption -eq '1' -and -not($script:dest -match '\.(msix|appx)(bundle){0,1}$')) { Remove-Item -Path $script:dest } - } +# Prompts the user to enter installer values +# Sets the $script:Installers value as an output +# Returns void +Function Read-InstallerEntry { + $_Installer = [ordered] @{} + # Request user enter Installer URL + $_Installer['InstallerUrl'] = Request-InstallerUrl + + if ($_Installer.InstallerUrl -in ($script:Installers).InstallerUrl) { + $_MatchingInstaller = $script:Installers | Where-Object { $_.InstallerUrl -eq $_Installer.InstallerUrl } | Select-Object -First 1 + if ($_MatchingInstaller.InstallerSha256) { $_Installer['InstallerSha256'] = $_MatchingInstaller.InstallerSha256 } + if ($_MatchingInstaller.InstallerType) { $_Installer['InstallerType'] = $_MatchingInstaller.InstallerType } + if ($_MatchingInstaller.Architecture) { $_Installer['Architecture'] = $_MatchingInstaller.Architecture } + if ($_MatchingInstaller.ProductCode) { $_Installer['ProductCode'] = $_MatchingInstaller.ProductCode } + if ($_MatchingInstaller.PackageFamilyName) { $_Installer['PackageFamilyName'] = $_MatchingInstaller.PackageFamilyName } + if ($_MatchingInstaller.SignatureSha256) { $_Installer['SignatureSha256'] = $_MatchingInstaller.SignatureSha256 } } - # Manual Entry of Sha256 with validation - else { - Write-Host - do { - Write-Host -ForegroundColor 'Red' $script:_returnValue.ErrorString() - Write-Host -ForegroundColor 'Green' -Object '[Required] Enter the installer SHA256 Hash' - $InstallerSha256 = Read-Host -Prompt 'InstallerSha256' | TrimString - $InstallerSHA256 = $InstallerSha256.toUpper() - if ($InstallerSha256 -match $Patterns.InstallerSha256) { - $script:_returnValue = [ReturnValue]::Success() - } else { - $script:_returnValue = [ReturnValue]::PatternError() + + # Get or request Installer Sha256 + # Check the settings to see if we need to display this menu + if ($_Installer.Keys -notcontains 'InstallerSha256') { + + $script:SaveOption = Get-UserSavePreference + # If user did not select manual entry for Sha256, download file and calculate hash + # Also attempt to detect installer type and architecture + if ($script:SaveOption -ne '2') { + Write-Host + $start_time = Get-Date + Write-Host $NewLine + Write-Host 'Downloading URL. This will take a while...' -ForegroundColor Blue + try { + $script:dest = Get-InstallerFile -URI $_Installer['InstallerUrl'] -PackageIdentifier $PackageIdentifier -PackageVersion $PackageVersion + } catch { + # Here we also want to pass any exceptions through for potential debugging + throw [System.Net.WebException]::new('The file could not be downloaded. Try running the script again', $_.Exception) + } finally { + Write-Host "Time taken: $((Get-Date).Subtract($start_time).Seconds) second(s)" -ForegroundColor Green + $_Installer['InstallerSha256'] = (Get-FileHash -Path $script:dest -Algorithm SHA256).Hash + Get-PathInstallerType -Path $script:dest -OutVariable _ | Out-Null + if ($_) { $_Installer['InstallerType'] = $_ | Select-Object -First 1 } + Get-UriArchitecture -URI $_Installer['InstallerUrl'] -OutVariable _ | Out-Null + if ($_) { $_Installer['Architecture'] = $_ | Select-Object -First 1 } + $ProductCode = $(Get-AppLockerFileInformation -Path $script:dest | Select-Object Publisher | Select-String -Pattern '{[A-Z0-9]{8}-([A-Z0-9]{4}-){3}[A-Z0-9]{12}}').Matches + if (Test-String -Not "$ProductCode" -IsNull) { $_Installer['ProductCode'] = "$ProductCode" } } - } until ($script:_returnValue.StatusCode -eq [ReturnValue]::Success().StatusCode) + } + # Manual Entry of Sha256 with validation + else { + Write-Host + do { + Write-Host -ForegroundColor 'Red' $script:_returnValue.ErrorString() + Write-Host -ForegroundColor 'Green' -Object '[Required] Enter the installer SHA256 Hash' + $_Installer['InstallerSha256'] = Read-Host -Prompt 'InstallerSha256' | TrimString + $_Installer['InstallerSha256'] = $_Installer['InstallerSha256'].toUpper() + if ($_Installer['InstallerSha256'] -match $Patterns.InstallerSha256) { + $script:_returnValue = [ReturnValue]::Success() + } else { + $script:_returnValue = [ReturnValue]::PatternError() + } + } until ($script:_returnValue.StatusCode -eq [ReturnValue]::Success().StatusCode) + } } # Manual Entry of Architecture with validation - if ($architecture -CNotIn @($Patterns.ValidArchitectures)) { - do { - Write-Host -ForegroundColor 'Red' $script:_returnValue.ErrorString() - Write-Host -ForegroundColor 'Green' -Object '[Required] Enter the architecture. Options:' , @($Patterns.ValidArchitectures -join ', ') - $architecture = Read-Host -Prompt 'Architecture' | TrimString - if ($architecture -Cin @($Patterns.ValidArchitectures)) { - $script:_returnValue = [ReturnValue]::Success() - } else { - $script:_returnValue = [ReturnValue]::new(400, 'Invalid Architecture', "Value must exist in the enum - $(@($Patterns.ValidArchitectures -join ', '))", 2) - } - } until ($script:_returnValue.StatusCode -eq [ReturnValue]::Success().StatusCode) - } + do { + Write-Host -ForegroundColor 'Red' $script:_returnValue.ErrorString() + if (Test-String $_Installer['Architecture'] -IsNull) { Write-Host -ForegroundColor 'Green' -Object '[Required] Enter the architecture. Options:' , @($Patterns.ValidArchitectures -join ', ') } + else { + Write-Host -ForegroundColor 'Yellow' -Object '[Optional] Enter the architecture. Options:' , @($Patterns.ValidArchitectures -join ', ') + Write-Host -ForegroundColor 'DarkGray' -Object "Old Variable: $($_Installer['Architecture'])" + } + Read-Host -Prompt 'Architecture' -OutVariable _ | Out-Null + if (Test-String $_ -Not -IsNull) { $_Installer['Architecture'] = $_ | TrimString } + + if ($_Installer['Architecture'] -Cin @($Patterns.ValidArchitectures)) { + $script:_returnValue = [ReturnValue]::Success() + } else { + $script:_returnValue = [ReturnValue]::new(400, 'Invalid Architecture', "Value must exist in the enum - $(@($Patterns.ValidArchitectures -join ', '))", 2) + } + } until ($script:_returnValue.StatusCode -eq [ReturnValue]::Success().StatusCode) # Manual Entry of Installer Type with validation - if ($InstallerType -CNotIn @($Patterns.ValidInstallerTypes)) { + if ($_Installer['InstallerType'] -CNotIn @($Patterns.ValidInstallerTypes)) { do { Write-Host -ForegroundColor 'Red' $script:_returnValue.ErrorString() Write-Host -ForegroundColor 'Green' -Object '[Required] Enter the InstallerType. Options:' , @($Patterns.ValidInstallerTypes -join ', ' ) - $InstallerType = Read-Host -Prompt 'InstallerType' | TrimString - if ($InstallerType -Cin @($Patterns.ValidInstallerTypes)) { + $_Installer['InstallerType'] = Read-Host -Prompt 'InstallerType' | TrimString + if ($_Installer['InstallerType'] -Cin @($Patterns.ValidInstallerTypes)) { $script:_returnValue = [ReturnValue]::Success() } else { $script:_returnValue = [ReturnValue]::new(400, 'Invalid Installer Type', "Value must exist in the enum - $(@($Patterns.ValidInstallerTypes -join ', '))", 2) @@ -480,89 +504,68 @@ Function Read-InstallerEntry { } until ($script:_returnValue.StatusCode -eq [ReturnValue]::Success().StatusCode) } + $_Switches = [ordered] @{} # If Installer Type is `exe`, require the silent switches to be entered - if ($InstallerType -ieq 'exe') { - # Required entry of `Silent` switches with validation - do { - Write-Host -ForegroundColor 'Red' $script:_returnValue.ErrorString() - Write-Host -ForegroundColor 'Green' -Object '[Required] Enter the silent install switch. For example: /S, -verysilent, /qn, --silent, /exenoui' - $Silent = Read-Host -Prompt 'Silent switch' | TrimString - - if (Test-String $Silent -MaxLength $Patterns.SilentSwitchMaxLength -NotNull) { - $script:_returnValue = [ReturnValue]::Success() - } else { - $script:_returnValue = [ReturnValue]::LengthError(1, $Patterns.SilentSwitchMaxLength) - } - } until ($script:_returnValue.StatusCode -eq [ReturnValue]::Success().StatusCode) - - # Required entry of `SilentWithProgress` swtich with validation - do { - Write-Host -ForegroundColor 'Red' $script:_returnValue.ErrorString() - Write-Host -ForegroundColor 'Green' -Object '[Required] Enter the silent with progress install switch. For example: /S, -silent, /qb, /exebasicui' - $SilentWithProgress = Read-Host -Prompt 'Silent with progress switch' | TrimString - - if (Test-String $SilentWithProgress -MaxLength $Patterns.ProgressSwitchMaxLength -NotNull) { - $script:_returnValue = [ReturnValue]::Success() - } else { - $script:_returnValue = [ReturnValue]::LengthError(1, $Patterns.ProgressSwitchMaxLength) - } - } until ($script:_returnValue.StatusCode -eq [ReturnValue]::Success().StatusCode) - } - # If Installer Type is not `exe`, the silent switches are optional - else { - # Optional entry of `Silent` switch with validation - do { - Write-Host -ForegroundColor 'Red' $script:_returnValue.ErrorString() - Write-Host -ForegroundColor 'Yellow' -Object '[Optional] Enter the silent install switch. For example: /S, /s, /VERYSILENT, /qn, --silent' - $Silent = Read-Host -Prompt 'Silent' | TrimString - - if (Test-String $Silent -MaxLength $Patterns.SilentSwitchMaxLength -AllowNull) { - $script:_returnValue = [ReturnValue]::Success() - } else { - $script:_returnValue = [ReturnValue]::LengthError(1, $Patterns.SilentSwitchMaxLength) - } + do { + Write-Host -ForegroundColor 'Red' $script:_returnValue.ErrorString() + if ($_Installer['InstallerType'] -ieq 'exe') { Write-Host -ForegroundColor 'Green' -Object '[Required] Enter the silent install switch. For example: /S, -verysilent, /qn, --silent, /exenoui' } + else { Write-Host -ForegroundColor 'Yellow' -Object '[Optional] Enter the silent install switch. For example: /S, -verysilent, /qn, --silent, /exenoui' } + Read-Host -Prompt 'Silent switch' -OutVariable _ | Out-Null + if ($_) { $_Switches['Silent'] = $_ | TrimString } - } until ($script:_returnValue.StatusCode -eq [ReturnValue]::Success().StatusCode) + if (Test-String $_Switches['Silent'] -MaxLength $Patterns.SilentSwitchMaxLength -NotNull) { + $script:_returnValue = [ReturnValue]::Success() + } elseif ($_Installer['InstallerType'] -ne 'exe' -and (Test-String $_Switches['Silent'] -MaxLength $Patterns.SilentSwitchMaxLength -AllowNull)) { + $script:_returnValue = [ReturnValue]::Success() + } else { + $script:_returnValue = [ReturnValue]::LengthError(1, $Patterns.SilentSwitchMaxLength) + } + } until ($script:_returnValue.StatusCode -eq [ReturnValue]::Success().StatusCode) - # Optional entry of `SilentWithProgress` switch with validation - do { - Write-Host -ForegroundColor 'Red' $script:_returnValue.ErrorString() - Write-Host -ForegroundColor 'Yellow' -Object '[Optional] Enter the silent with progress install switch. For example: /S, /SILENT, /qb' - $SilentWithProgress = Read-Host -Prompt 'SilentWithProgress' | TrimString + do { + Write-Host -ForegroundColor 'Red' $script:_returnValue.ErrorString() + if ($_Installer['InstallerType'] -ieq 'exe') { Write-Host -ForegroundColor 'Green' -Object '[Required] Enter the silent with progress install switch. For example: /S, -silent, /qb, /exebasicui' } + else { Write-Host -ForegroundColor 'Yellow' -Object '[Optional] Enter the silent with progress install switch. For example: /S, -silent, /qb, /exebasicui' } + Read-Host -Prompt 'Silent with progress switch' -OutVariable _ | Out-Null + if ($_) { $_Switches['SilentWithProgress'] = $_ | TrimString } - if (Test-String $SilentWithProgress -MaxLength $Patterns.ProgressSwitchMaxLength -AllowNull) { - $script:_returnValue = [ReturnValue]::Success() - } else { - $script:_returnValue = [ReturnValue]::LengthError(1, $Patterns.ProgressSwitchMaxLength) - } - } until ($script:_returnValue.StatusCode -eq [ReturnValue]::Success().StatusCode) - } + if (Test-String $_Switches['SilentWithProgress'] -MaxLength $Patterns.ProgressSwitchMaxLength -NotNull) { + $script:_returnValue = [ReturnValue]::Success() + } elseif ($_Installer['InstallerType'] -ne 'exe' -and (Test-String $_Switches['SilentWithProgress'] -MaxLength $Patterns.ProgressSwitchMaxLength -AllowNull)) { + $script:_returnValue = [ReturnValue]::Success() + } else { + $script:_returnValue = [ReturnValue]::LengthError(1, $Patterns.ProgressSwitchMaxLength) + } + } until ($script:_returnValue.StatusCode -eq [ReturnValue]::Success().StatusCode) # Optional entry of `Custom` switches with validation for all installer types do { Write-Host -ForegroundColor 'Red' $script:_returnValue.ErrorString() Write-Host -ForegroundColor 'Yellow' -Object '[Optional] Enter any custom switches for the installer. For example: /norestart, -norestart' - $Custom = Read-Host -Prompt 'Custom Switch' | TrimString - - if (Test-String $Custom -MaxLength $Patterns.CustomSwitchMaxLength -AllowNull) { + Read-Host -Prompt 'Custom Switch' -OutVariable _ | Out-Null + if ($_) { $_Switches['Custom'] = $_ | TrimString } + if (Test-String $_Switches['Custom'] -MaxLength $Patterns.CustomSwitchMaxLength -AllowNull) { $script:_returnValue = [ReturnValue]::Success() } else { $script:_returnValue = [ReturnValue]::LengthError(1, $Patterns.CustomSwitchMaxLength) } } until ($script:_returnValue.StatusCode -eq [ReturnValue]::Success().StatusCode) + if ($_Switches.Keys.Count -gt 0) { $_Installer['InstallerSwitches'] = $_Switches } + # If the installer is `msix` or `appx`, prompt for or detect additional fields - if ($InstallerType -ieq 'msix' -or $InstallerType -ieq 'appx') { + if ($_Installer['InstallerType'] -in @('msix'; 'appx')) { # Detect or prompt for Signature Sha256 if (Get-Command 'winget.exe' -ErrorAction SilentlyContinue) { $SignatureSha256 = winget hash -m $script:dest | Select-String -Pattern 'SignatureSha256:' | ConvertFrom-String; if ($SignatureSha256.P2) { $SignatureSha256 = $SignatureSha256.P2.ToUpper() } } - if (Test-String $SignatureSha256 -IsNull) { + if ($SignatureSha256) { $_Installer['SignatureSha256'] = $SignatureSha256 } + if (Test-String $_Installer['SignatureSha256'] -IsNull) { # Manual entry of Signature Sha256 with validation do { Write-Host -ForegroundColor 'Red' $script:_returnValue.ErrorString() Write-Host -ForegroundColor 'Yellow' -Object '[Recommended] Enter the installer SignatureSha256' - $SignatureSha256 = Read-Host -Prompt 'SignatureSha256' | TrimString - - if (Test-String $SignatureSha256 -MatchPattern $Patterns.SignatureSha256 -AllowNull) { + Read-Host -Prompt 'SignatureSha256' -OutVariable _ | Out-Null + if ($_) { $_Installer['SignatureSha256'] = $_ | TrimString } + if (Test-String $_Installer['SignatureSha256'] -MatchPattern $Patterns.SignatureSha256 -AllowNull) { $script:_returnValue = [ReturnValue]::Success() } else { $script:_returnValue = [ReturnValue]::PatternError() @@ -580,7 +583,6 @@ Function Read-InstallerEntry { DefaultString = 'F' } switch ( Invoke-KeypressMenu -Prompt $_menu['Prompt'] -Entries $_menu['Entries'] -DefaultString $_menu['DefaultString']) { - 'F' { $ChoicePfn = '0' } 'M' { $ChoicePfn = '1' } default { $ChoicePfn = '0' } } @@ -592,13 +594,13 @@ Function Read-InstallerEntry { try { Add-AppxPackage -Path $script:dest $InstalledPkg = Get-AppxPackage | Select-Object -Last 1 | Select-Object PackageFamilyName, PackageFullName - $PackageFamilyName = $InstalledPkg.PackageFamilyName + if ($InstalledPkg.PackageFamilyName) { $_Installer['PackageFamilyName'] = $InstalledPkg.PackageFamilyName } Remove-AppxPackage $InstalledPkg.PackageFullName } catch { # Take no action here, we just want to catch the exceptions as a precaution Out-Null } finally { - if (Test-String $PackageFamilyName -IsNull) { + if (Test-String $_Installer['PackageFamilyName'] -IsNull) { $script:_returnValue = [ReturnValue]::new(500, 'Could not find PackageFamilyName', 'Value should be entered manually', 1) } } @@ -610,24 +612,22 @@ Function Read-InstallerEntry { if (($ChoicePfn -ne '0') -or ($script:_returnValue.StatusCode -ne [ReturnValue]::Success().StatusCode)) { Write-Host -ForegroundColor 'Red' $script:_returnValue.ErrorString() Write-Host -ForegroundColor 'Yellow' -Object '[Recommended] Enter the PackageFamilyName' - $PackageFamilyName = Read-Host -Prompt 'PackageFamilyName' | TrimString + Read-Host -Prompt 'PackageFamilyName' -OutVariable _ | Out-Null + if ($_) { $_Installer['PackageFamilyName'] = $_ | TrimString } } - - if (Test-String $PackageFamilyName -MaxLength $Patterns.FamilyNameMaxLength -MatchPattern $Patterns.FamilyName -AllowNull) { - if (Test-String $PackageFamilyName -IsNull) { $PackageFamilyName = "$([char]0x2370)" } + if (Test-String $_Installer['PackageFamilyName'] -MaxLength $Patterns.FamilyNameMaxLength -MatchPattern $Patterns.FamilyName -AllowNull) { + if (Test-String $_Installer['PackageFamilyName'] -IsNull) { $_Installer['PackageFamilyName'] = "$([char]0x2370)" } $script:_returnValue = [ReturnValue]::Success() } else { - if (Test-String -not $PackageFamilyName -MaxLength $Patterns.FamilyNameMaxLength) { + if (Test-String -not $_Installer['PackageFamilyName'] -MaxLength $Patterns.FamilyNameMaxLength) { $script:_returnValue = [ReturnValue]::LengthError(1, $Patterns.FamilyNameMaxLength) - } elseif (Test-String -not $PackageFamilyName -MatchPattern $Patterns.FamilyName) { + } elseif (Test-String -not $_Installer['PackageFamilyName'] -MatchPattern $Patterns.FamilyName) { $script:_returnValue = [ReturnValue]::PatternError() } else { $script:_returnValue = [ReturnValue]::GenericError() } } - } until ($script:_returnValue.StatusCode -eq [ReturnValue]::Success().StatusCode) - if ($script:SaveOption -eq '1') { Remove-Item -Path $script:dest } } # Request installer locale with validation as optional @@ -635,16 +635,17 @@ Function Read-InstallerEntry { Write-Host -ForegroundColor 'Red' $script:_returnValue.ErrorString() Write-Host -ForegroundColor 'Yellow' -Object '[Optional] Enter the installer locale. For example: en-US, en-CA' Write-Host -ForegroundColor 'Blue' -Object 'https://docs.microsoft.com/openspecs/office_standards/ms-oe376/6c085406-a698-4e12-9d4d-c3b0ee3dbc4a' - $InstallerLocale = Read-Host -Prompt 'InstallerLocale' | TrimString + Read-Host -Prompt 'InstallerLocale' -OutVariable _ + if ($_) { $_Installer['InstallerLocale'] = $_ | TrimString } # If user defined a default locale, add it - if ((Test-String $InstallerLocale -IsNull) -and (Test-String -not $ScriptSettings.DefaultInstallerLocale -IsNull)) { $InstallerLocale = $ScriptSettings.DefaultInstallerLocale } + if ((Test-String $_Installer['InstallerLocale'] -IsNull) -and (Test-String -not $ScriptSettings.DefaultInstallerLocale -IsNull)) { $_Installer['InstallerLocale'] = $ScriptSettings.DefaultInstallerLocale } - if (Test-String $InstallerLocale -MaxLength $Patterns.InstallerLocaleMaxLength -MatchPattern $Patterns.PackageLocale -AllowNull) { + if (Test-String $_Installer['InstallerLocale'] -MaxLength $Patterns.InstallerLocaleMaxLength -MatchPattern $Patterns.PackageLocale -AllowNull) { $script:_returnValue = [ReturnValue]::Success() } else { - if (Test-String -not $InstallerLocale -MaxLength $Patterns.InstallerLocaleMaxLength -AllowNull) { + if (Test-String -not $_Installer['InstallerLocale'] -MaxLength $Patterns.InstallerLocaleMaxLength -AllowNull) { $script:_returnValue = [ReturnValue]::LengthError(0, $Patterns.InstallerLocaleMaxLength) - } elseif (Test-String -not $InstallerLocale -MatchPattern $Patterns.PackageLocale) { + } elseif (Test-String -not $_Installer['InstallerLocale'] -MatchPattern $Patterns.PackageLocale) { $script:_returnValue = [ReturnValue]::PatternError() } else { $script:_returnValue = [ReturnValue]::GenericError() @@ -656,11 +657,13 @@ Function Read-InstallerEntry { do { Write-Host -ForegroundColor 'Red' $script:_returnValue.ErrorString() Write-Host -ForegroundColor 'Yellow' -Object '[Optional] Enter the application product code. Looks like {CF8E6E00-9C03-4440-81C0-21FACB921A6B}' - Write-Host -ForegroundColor 'White' -Object "ProductCode found from installer: $MSIProductCode" + Write-Host -ForegroundColor 'White' -Object "ProductCode found from installer: $($_Installer['ProductCode'])" Write-Host -ForegroundColor 'White' -Object 'Can be found with ' -NoNewline; Write-Host -ForegroundColor 'DarkYellow' 'get-wmiobject Win32_Product | Sort-Object Name | Format-Table IdentifyingNumber, Name -AutoSize' - $ProductCode = Read-Host -Prompt 'ProductCode' | TrimString + $NewProductCode = Read-Host -Prompt 'ProductCode' | TrimString + if (Test-String $NewProductCode -Not -IsNull) { $_Installer['ProductCode'] = $NewProductCode } + elseif (Test-String $_Installer['ProductCode'] -Not -IsNull) { $_Installer['ProductCode'] = "$($_Installer['ProductCode'])" } - if (Test-String $ProductCode -MinLength $Patterns.ProductCodeMinLength -MaxLength $Patterns.ProductCodeMaxLength -AllowNull) { + if (Test-String $_Installer['ProductCode'] -MinLength $Patterns.ProductCodeMinLength -MaxLength $Patterns.ProductCodeMaxLength -AllowNull) { $script:_returnValue = [ReturnValue]::Success() } else { $script:_returnValue = [ReturnValue]::LengthError($Patterns.ProductCodeMinLength, $Patterns.ProductCodeMaxLength) @@ -674,10 +677,9 @@ Function Read-InstallerEntry { DefaultString = 'N' } switch ( Invoke-KeypressMenu -Prompt $_menu['Prompt'] -Entries $_menu['Entries'] -DefaultString $_menu['DefaultString']) { - 'M' { $Scope = 'machine' } - 'U' { $Scope = 'user' } - 'N' { $Scope = '' } - default { $Scope = '' } + 'M' { $_Installer['Scope'] = 'machine' } + 'U' { $_Installer['Scope'] = 'user' } + default { } } # Request upgrade behavior @@ -687,53 +689,17 @@ Function Read-InstallerEntry { DefaultString = 'I' } switch ( Invoke-KeypressMenu -Prompt $_menu['Prompt'] -Entries $_menu['Entries'] -DefaultString $_menu['DefaultString']) { - 'I' { $UpgradeBehavior = 'install' } - 'U' { $UpgradeBehavior = 'uninstallPrevious' } - default { $UpgradeBehavior = 'install' } + 'U' { $_Installer['UpgradeBehavior'] = 'uninstallPrevious' } + default { $_Installer['UpgradeBehavior'] = 'install' } } + if ($script:SaveOption -eq '1' -and (Test-Path -Path $script:dest)) { Remove-Item -Path $script:dest } + # If the installers array is empty, create it if (!$script:Installers) { $script:Installers = @() } - # Set up a new empty installer - $_Installer = [ordered] @{} - - # Add the single-line parameters to the installer entry - $_InstallerSingletons = [ordered] @{ - 'InstallerLocale' = $InstallerLocale - 'Architecture' = $Architecture - 'InstallerType' = $InstallerType - 'Scope' = $Scope - 'InstallerUrl' = $InstallerUrl - 'InstallerSha256' = $InstallerSha256 - 'SignatureSha256' = $SignatureSha256 - 'PackageFamilyName' = $PackageFamilyName - } - foreach ($_Item in $_InstallerSingletons.GetEnumerator()) { - If ($_Item.Value) { Add-YamlParameter -Object $_Installer -Parameter $_Item.Name -Value $_Item.Value } - } - - # Add the installer switches to the installer entry, if they exist - If ($Silent -or $SilentWithProgress -or $Custom) { - $_InstallerSwitches = [ordered]@{} - $_Switches = [ordered] @{ - 'Custom' = $Custom - 'Silent' = $Silent - 'SilentWithProgress' = $SilentWithProgress - } - foreach ($_Item in $_Switches.GetEnumerator()) { - If ($_Item.Value) { Add-YamlParameter -Object $_InstallerSwitches -Parameter $_Item.Name -Value $_Item.Value } - } - $_InstallerSwitches = Restore-YamlKeyOrder $_InstallerSwitches $InstallerSwitchProperties -NoComments - $_Installer['InstallerSwitches'] = $_InstallerSwitches - } - - # Add the product code to the installer entry, if it exists - If ($ProductCode) { Add-YamlParameter -Object $_Installer -Parameter 'ProductCode' -Value $ProductCode } - Add-YamlParameter -Object $_Installer -Parameter 'UpgradeBehavior' -Value $UpgradeBehavior - # Add the completed installer to the installers array $_Installer = Restore-YamlKeyOrder $_Installer $InstallerEntryProperties -NoComments $script:Installers += $_Installer @@ -776,6 +742,7 @@ Function Read-QuickInstallerEntry { # This is to ensure all previously entered and un-modified parameters are retained $_iteration += 1 $_NewInstaller = $_OldInstaller + $_NewInstaller.Remove('InstallerSha256'); # Show the user which installer entry they should be entering information for Write-Host -ForegroundColor 'Green' "Installer Entry #$_iteration`:`n" @@ -788,53 +755,68 @@ Function Read-QuickInstallerEntry { # Request user enter the new Installer URL $_NewInstaller['InstallerUrl'] = Request-InstallerUrl - try { - Get-InstallerFile -URI $_NewInstaller['InstallerUrl'] -PackageIdentifier $PackageIdentifier -PackageVersion $PackageVersion - } catch { - # Here we also want to pass any exceptions through for potential debugging - throw [System.Net.WebException]::new('The file could not be downloaded. Try running the script again', $_.Exception) - } finally { - # Get the Sha256 - $_NewInstaller['InstallerSha256'] = (Get-FileHash -Path $script:dest -Algorithm SHA256).Hash - # Update the product code, if a new one exists - # If a new product code doesn't exist, and the installer isn't an `.exe` file, remove the product code if it exists - $MSIProductCode = [string]$(Get-AppLockerFileInformation -Path $script:dest | Select-Object Publisher | Select-String -Pattern '{[A-Z0-9]{8}-([A-Z0-9]{4}-){3}[A-Z0-9]{12}}').Matches - if (Test-String -not $MSIProductCode -IsNull) { - $_NewInstaller['ProductCode'] = $MSIProductCode - } elseif ( ($_NewInstaller.Keys -contains 'ProductCode') -and ($script:dest -notmatch '.exe$')) { - $_NewInstaller.Remove('ProductCode') - } - # If the installer is msix or appx, try getting the new SignatureSha256 - # If the new SignatureSha256 can't be found, remove it if it exists - if ($_NewInstaller.InstallerType -in @('msix', 'appx')) { - if (Get-Command 'winget.exe' -ErrorAction SilentlyContinue) { $NewSignatureSha256 = winget hash -m $script:dest | Select-String -Pattern 'SignatureSha256:' | ConvertFrom-String; if ($NewSignatureSha256.P2) { $NewSignatureSha256 = $NewSignatureSha256.P2.ToUpper() } } - } - if (Test-String -not $NewSignatureSha256 -IsNull) { - $_NewInstaller['SignatureSha256'] = $NewSignatureSha256 - } elseif ($_NewInstaller.Keys -contains 'SignatureSha256') { - $_NewInstaller.Remove('SignatureSha256') - } - # If the installer is msix or appx, try getting the new package family name - # If the new package family name can't be found, remove it if it exists - if ($script:dest -match '\.(msix|appx)(bundle){0,1}$') { - try { - Add-AppxPackage -Path $script:dest - $InstalledPkg = Get-AppxPackage | Select-Object -Last 1 | Select-Object PackageFamilyName, PackageFullName - $PackageFamilyName = $InstalledPkg.PackageFamilyName - Remove-AppxPackage $InstalledPkg.PackageFullName - } catch { - # Take no action here, we just want to catch the exceptions as a precaution - Out-Null - } finally { - if (Test-String -not $PackageFamilyName -IsNull) { - $_NewInstaller['PackageFamilyName'] = $PackageFamilyName - } elseif ($_NewInstaller.Keys -contains 'PackageFamilyName') { - $_NewInstaller.Remove('PackageFamilyName') + if ($_NewInstaller.InstallerUrl -in ($_NewInstallers).InstallerUrl) { + $_MatchingInstaller = $_NewInstallers | Where-Object { $_.InstallerUrl -eq $_NewInstaller.InstallerUrl } | Select-Object -First 1 + if ($_MatchingInstaller.InstallerSha256) { $_NewInstaller['InstallerSha256'] = $_MatchingInstaller.InstallerSha256 } + if ($_MatchingInstaller.InstallerType) { $_NewInstaller['InstallerType'] = $_MatchingInstaller.InstallerType } + if ($_MatchingInstaller.Architecture) { $_NewInstaller['Architecture'] = $_MatchingInstaller.Architecture } + if ($_MatchingInstaller.ProductCode) { $_NewInstaller['ProductCode'] = $_MatchingInstaller.ProductCode } + elseif ( ($_NewInstaller.Keys -contains 'ProductCode') -and ($script:dest -notmatch '.exe$')) { $_NewInstaller.Remove('ProductCode') } + if ($_MatchingInstaller.PackageFamilyName) { $_NewInstaller['PackageFamilyName'] = $_MatchingInstaller.PackageFamilyName } + elseif ($_NewInstaller.Keys -contains 'PackageFamilyName') { $_NewInstaller.Remove('PackageFamilyName') } + if ($_MatchingInstaller.SignatureSha256) { $_NewInstaller['SignatureSha256'] = $_MatchingInstaller.SignatureSha256 } + elseif ($_NewInstaller.Keys -contains 'SignatureSha256') { $_NewInstaller.Remove('SignatureSha256') } + } + + if ($_NewInstaller.Keys -notcontains 'InstallerSha256') { + try { + $script:dest = Get-InstallerFile -URI $_NewInstaller['InstallerUrl'] -PackageIdentifier $PackageIdentifier -PackageVersion $PackageVersion + } catch { + # Here we also want to pass any exceptions through for potential debugging + throw [System.Net.WebException]::new('The file could not be downloaded. Try running the script again', $_.Exception) + } finally { + # Get the Sha256 + $_NewInstaller['InstallerSha256'] = (Get-FileHash -Path $script:dest -Algorithm SHA256).Hash + # Update the product code, if a new one exists + # If a new product code doesn't exist, and the installer isn't an `.exe` file, remove the product code if it exists + $MSIProductCode = [string]$(Get-AppLockerFileInformation -Path $script:dest | Select-Object Publisher | Select-String -Pattern '{[A-Z0-9]{8}-([A-Z0-9]{4}-){3}[A-Z0-9]{12}}').Matches + if (Test-String -not $MSIProductCode -IsNull) { + $_NewInstaller['ProductCode'] = $MSIProductCode + } elseif ( ($_NewInstaller.Keys -contains 'ProductCode') -and ($script:dest -notmatch '.exe$')) { + $_NewInstaller.Remove('ProductCode') + } + # If the installer is msix or appx, try getting the new SignatureSha256 + # If the new SignatureSha256 can't be found, remove it if it exists + if ($_NewInstaller.InstallerType -in @('msix', 'appx')) { + if (Get-Command 'winget.exe' -ErrorAction SilentlyContinue) { $NewSignatureSha256 = winget hash -m $script:dest | Select-String -Pattern 'SignatureSha256:' | ConvertFrom-String; if ($NewSignatureSha256.P2) { $NewSignatureSha256 = $NewSignatureSha256.P2.ToUpper() } } + } + if (Test-String -not $NewSignatureSha256 -IsNull) { + $_NewInstaller['SignatureSha256'] = $NewSignatureSha256 + } elseif ($_NewInstaller.Keys -contains 'SignatureSha256') { + $_NewInstaller.Remove('SignatureSha256') + } + # If the installer is msix or appx, try getting the new package family name + # If the new package family name can't be found, remove it if it exists + if ($script:dest -match '\.(msix|appx)(bundle){0,1}$') { + try { + Add-AppxPackage -Path $script:dest + $InstalledPkg = Get-AppxPackage | Select-Object -Last 1 | Select-Object PackageFamilyName, PackageFullName + $PackageFamilyName = $InstalledPkg.PackageFamilyName + Remove-AppxPackage $InstalledPkg.PackageFullName + } catch { + # Take no action here, we just want to catch the exceptions as a precaution + Out-Null + } finally { + if (Test-String -not $PackageFamilyName -IsNull) { + $_NewInstaller['PackageFamilyName'] = $PackageFamilyName + } elseif ($_NewInstaller.Keys -contains 'PackageFamilyName') { + $_NewInstaller.Remove('PackageFamilyName') + } } } + # Remove the downloaded files + Remove-Item -Path $script:dest } - # Remove the downloaded files - Remove-Item -Path $script:dest } #Add the updated installer to the new installers array $_NewInstaller = Restore-YamlKeyOrder $_NewInstaller $InstallerEntryProperties -NoComments @@ -859,7 +841,7 @@ Function Read-InstallerMetadataValue { ) Write-Host -ForegroundColor 'Red' $script:_returnValue.ErrorString() Write-Host -ForegroundColor 'Yellow' -Object $Prompt - if (Test-String -not $Variable -IsNull) { Write-Host -ForegroundColor 'DarkGray' "Old Value: $Variable" } + if (Test-String -not $Variable -IsNull) { Write-Host -ForegroundColor 'DarkGray' "Old Variable: $Variable" } $NewValue = Read-Host -Prompt $Key | TrimString if (Test-String -not $NewValue -IsNull) { @@ -962,10 +944,10 @@ Function Read-InstallerMetadata { $script:_returnValue = [ReturnValue]::Success() try { #Ensure all values are integers - $script:InstallerSuccessCodes.Split(',').Trim()| ForEach-Object {[long]$_} + $script:InstallerSuccessCodes.Split(',').Trim() | ForEach-Object { [long]$_ } $script:_returnValue = [ReturnValue]::Success() } catch { - $script:_returnValue = [ReturnValue]::new(400,"Invalid Data Type","The value entered does not match the type requirements defined in the manifest schema",2) + $script:_returnValue = [ReturnValue]::new(400, 'Invalid Data Type', 'The value entered does not match the type requirements defined in the manifest schema', 2) } } else { $script:_returnValue = [ReturnValue]::MaxItemsError($Patterns.MaxItemsSuccessCodes) @@ -1533,7 +1515,7 @@ Function Write-VersionManifest { $VersionManifestPath = $AppFolder + "\$PackageIdentifier" + '.yaml' # Write the manifest to the file - $ScriptHeader + "$(Get-DebugString)`n# yaml-language-server: `$schema=https://aka.ms/winget-manifest.version.1.0.0.schema.json`n" > $VersionManifestPath + $ScriptHeader + "$(Get-DebugString)`n# yaml-language-server: `$schema=https://aka.ms/winget-manifest.version.$ManifestVersion.schema.json`n" > $VersionManifestPath ConvertTo-Yaml $VersionManifest >> $VersionManifestPath $(Get-Content $VersionManifestPath -Encoding UTF8) -replace "(.*)$([char]0x2370)", "# `$1" | Out-File -FilePath $VersionManifestPath -Force $MyRawString = Get-Content -Raw $VersionManifestPath | TrimString @@ -1582,7 +1564,7 @@ Function Write-InstallerManifest { $InstallerManifest['Dependencies'] = Restore-YamlKeyOrder $InstallerManifest['Dependencies'] $InstallerDependencyProperties -NoComments } # Move Installer Level Keys to Manifest Level - $_KeysToMove = $InstallerEntryProperties | Where-Object { $_ -in $InstallerProperties -and $_ -ne 'ProductCode'} + $_KeysToMove = $InstallerEntryProperties | Where-Object { $_ -in $InstallerProperties -and $_ -ne 'ProductCode' } foreach ($_Key in $_KeysToMove) { if ($_Key -in $InstallerManifest.Installers[0].Keys) { # Handle the switches specially @@ -1642,7 +1624,7 @@ Function Write-InstallerManifest { $InstallerManifestPath = $AppFolder + "\$PackageIdentifier" + '.installer' + '.yaml' # Write the manifest to the file - $ScriptHeader + "$(Get-DebugString)`n# yaml-language-server: `$schema=https://aka.ms/winget-manifest.installer.1.0.0.schema.json`n" > $InstallerManifestPath + $ScriptHeader + "$(Get-DebugString)`n# yaml-language-server: `$schema=https://aka.ms/winget-manifest.installer.$ManifestVersion.schema.json`n" > $InstallerManifestPath ConvertTo-Yaml $InstallerManifest >> $InstallerManifestPath $(Get-Content $InstallerManifestPath -Encoding UTF8) -replace "(.*)$([char]0x2370)", "# `$1" | Out-File -FilePath $InstallerManifestPath -Force $MyRawString = Get-Content -Raw $InstallerManifestPath | TrimString @@ -1662,7 +1644,7 @@ Function Write-LocaleManifest { if (!$LocaleManifest) { [PSCustomObject]$LocaleManifest = [ordered]@{} } # Set the appropriate langage server depending on if it is a default locale file or generic locale file - if ($LocaleManifest.ManifestType -eq 'defaultLocale') { $yamlServer = '# yaml-language-server: $schema=https://aka.ms/winget-manifest.defaultLocale.1.0.0.schema.json' } else { $yamlServer = '# yaml-language-server: $schema=https://aka.ms/winget-manifest.locale.1.0.0.schema.json' } + if ($LocaleManifest.ManifestType -eq 'defaultLocale') { $yamlServer = "# yaml-language-server: `$schema=https://aka.ms/winget-manifest.defaultLocale.$ManifestVersion.schema.json" } else { $yamlServer = "# yaml-language-server: `$schema=https://aka.ms/winget-manifest.locale.$ManifestVersion.schema.json" } # Add the properties to the manifest $_Singletons = [ordered]@{ @@ -1712,9 +1694,10 @@ Function Write-LocaleManifest { $script:OldLocaleManifest = ConvertFrom-Yaml -Yaml ($(Get-Content -Path $DifLocale.FullName -Encoding UTF8) -join "`n") -Ordered $script:OldLocaleManifest['PackageVersion'] = $PackageVersion if ($script:OldLocaleManifest.Keys -contains 'Moniker') { $script:OldLocaleManifest.Remove('Moniker') } + $script:OldLocaleManifest['ManifestVersion'] = $ManifestVersion $script:OldLocaleManifest = Restore-YamlKeyOrder $script:OldLocaleManifest $LocaleProperties - $yamlServer = '# yaml-language-server: $schema=https://aka.ms/winget-manifest.locale.1.0.0.schema.json' + $yamlServer = "# yaml-language-server: `$schema=https://aka.ms/winget-manifest.locale.$ManifestVersion.schema.json" $ScriptHeader + "$(Get-DebugString)`n$yamlServer`n" > ($AppFolder + '\' + $DifLocale.Name) ConvertTo-Yaml $OldLocaleManifest >> ($AppFolder + '\' + $DifLocale.Name) @@ -1744,6 +1727,8 @@ function Remove-ManifestVersion { } while (@(Get-ChildItem $PathToVersion).Count -eq 0) } +## START OF MAIN SCRIPT ## + # Initialize the return value to be a success $script:_returnValue = [ReturnValue]::new(200) @@ -2129,7 +2114,7 @@ Switch ($script:Option) { Write-Host 'Updating Manifest Information. This may take a while...' -ForegroundColor Blue foreach ($_Installer in $script:OldInstallerManifest.Installers) { try { - Get-InstallerFile -URI $_Installer.InstallerUrl -PackageIdentifier $PackageIdentifier -PackageVersion $PackageVersion + $script:dest = Get-InstallerFile -URI $_Installer.InstallerUrl -PackageIdentifier $PackageIdentifier -PackageVersion $PackageVersion } catch { # Here we also want to pass any exceptions through for potential debugging throw [System.Net.WebException]::new('The file could not be downloaded. Try running the script again', $_.Exception)