modules/WinGet-Initialize.psm1
Set-StrictMode -Version 3 Import-Module "$PSScriptRoot\WinGet-Utils.psm1" [string]$PackageDatabase = "$PSScriptRoot\winget.packages.json" [string]$IgnoreFilePath = "$PSScriptRoot\winget.{HOSTNAME}.ignore" <# .DESCRIPTION This is the back end handler for initializing various user-facing files that typically are expected to resize external to the module install location but symlinked. #> function Initialize-WinGetResource { param( # The path to an existing file. This file will be symbolically linked. [string]$SourceFile, # Points to one of the internally referernced resource files which # the $SourceFile will be SmyLink'd or copied to. [string]$DestinationFile ) $DestinationFilename = $(Split-Path -Leaf $DestinationFile) if (-not([string]::IsNullOrWhiteSpace($SourceFile))) { if ((Test-Path $DestinationFile) -and -not(Test-ReparsePoint $DestinationFile)) { Write-Output "Backing up existing '$DestinationFilename'." Move-Item $DestinationFile -Destination "$DestinationFile.bak" -Force } $SymLinkArgs = @{ ItemType = "SymbolicLink" Path = "$(Split-Path -Parent $DestinationFile)" Name = $DestinationFilename Value = "$($SourceFile | Resolve-Path)" Force = $true } Write-Output "Creating new symlink for '$DestinationFilename'." New-Item @SymLinkArgs } elseif (Test-Path $DestinationFile) { Write-Output "The '$DestinationFilename' is already initialized." return } else { $currentVersion = [version]"0.0" if (-not[version]::TryParse((Split-Path -Leaf (Get-Item $PSScriptRoot/..)), [ref]$currentVersion)) { # Not installed as a module, do not try to migrate Write-Output "Not installed as a module. Migration of '$DestinationFilename' cannot be performed." return } $selectedVersion = [version]"0.0" $selectedPackageFile = $null $moduleVersionPaths = Get-ChildItem -Directory $PSScriptRoot/../.. $moduleVersionPaths | Where-Object { [version]($_.Name) -ne $currentVersion } | ForEach-Object { $packageFile = (Join-Path $_ "modules/$DestinationFilename") if (Test-Path $packageFile) { $version = [version](Split-Path -Leaf $_) if ($version -gt $selectedVersion) { $selectedVersion = $version $selectedPackageFile = $packageFile } } } if ($null -ne $selectedPackageFile) { $source = Get-Item $selectedPackageFile if ($source.Target) { if (-not(Test-Administrator)) { Write-Warning "A symlink to '$DestinationFilename' cannot be created as a non-admin." return } Write-Output "Creating new symlink to '$DestinationFilename'." $SymLinkArgs = @{ ItemType = "SymbolicLink" Path = "$(Split-Path -Parent $DestinationFile)" Name = $DestinationFilename Value = $source.Target } New-Item @SymLinkArgs } else { Write-Output "Copying existing '$DestinationFilename'." Copy-Item -Path $source.FullName -Destination $DestinationFile } } else { Write-Output "No '$DestinationFilename' detected." Write-Output "Create one and provide it as the argument to -SourceFile." } } } <# .DESCRIPTION Initialize the local "winget.software.json" needed by Restore-WinGetSoftware. If -SourceFile is specified, the file will be symbolically linked to the appropriate location. When this parameter is not specified, the cmdlet will auto-detect other installed module versions and attempt to find the latest existing "winget.packages.json". In such cases, it will make a symlink only if one was used previously; otherwise it will copy the previous file. .EXAMPLE PS> Initialize-WinGetRestore -SourceFile "./winget.software.json" .EXAMPLE PS> Initialize-WinGetRestore #> function Initialize-WinGetRestore { param( <# The path to an existing "winget.software.json" file. This file will be symbolically linked. #> [string]$SourceFile, <# When set, the cmdlet will automatically relaunch using an Administrator PowerShell instance. This cmdlet needs such permissions to create Symbolic Links. #> [switch]$Administrator ) if ($Administrator -and -not(Test-Administrator)) { $boundParamsString = $PSBoundParameters.Keys | ForEach-Object { if ($PSBoundParameters[$_] -is [switch]) { if ($PSBoundParameters[$_]) { "-$($_)" } } else { "-$($_) $($PSBoundParameters[$_])" } } $cmdArgs = "-NoLogo -NoExit -Command Initialize-WinGetRestore $($boundParamsString -join ' ')" Start-Process -Verb RunAs -FilePath "pwsh" -ArgumentList $cmdArgs return } Initialize-WinGetResource -SourceFile $SourceFile -DestinationFile $PackageDatabase } <# .DESCRIPTION Initialize the local "winget.{HOSTNAME}.ignore" used by various cmdlets. If -SourceFile is specified, the file will be symbolically linked to the appropriate location. When this parameter is not specified, the cmdlet will auto-detect other installed module versions and attempt to find the latest existing "winget.{HOSTNAME}.ignore". In such cases, it will make a symlink only if one was used previously; otherwise it will copy the previous file. .EXAMPLE PS> Initialize-WinGetIgnore -SourceFile "./winget.example-hostname.ignore" .EXAMPLE PS> Initialize-WinGetIgnore #> function Initialize-WinGetIgnore { param( <# The path to an existing winget-ignore file. This file will be symbolically linked. #> [string]$SourceFile, <# When set, the cmdlet will automatically relaunch using an Administrator PowerShell instance. This cmdlet needs such permissions to create Symbolic Links. #> [switch]$Administrator ) if ($Administrator -and -not(Test-Administrator)) { $boundParamsString = $PSBoundParameters.Keys | ForEach-Object { if ($PSBoundParameters[$_] -is [switch]) { if ($PSBoundParameters[$_]) { "-$($_)" } } else { "-$($_) $($PSBoundParameters[$_])" } } $cmdArgs = "-NoLogo -NoExit -Command Initialize-WinGetIgnore $($boundParamsString -join ' ')" Start-Process -Verb RunAs -FilePath "pwsh" -ArgumentList $cmdArgs return } $ignoreFile = $IgnoreFilePath.Replace('{HOSTNAME}', $(hostname).ToLower()) Initialize-WinGetResource -SourceFile $SourceFile -DestinationFile $ignoreFile } |