using namespace Microsoft.Win32; $null = New-Module { . "$PSScriptRoot/../Scripts/Registry.ps1"; . "$PSScriptRoot/../../Common/Scripts/Config.ps1"; . "$PSScriptRoot/../../Common/Scripts/Scripting.ps1"; [RegistryKey] $key = $null; $runOncePath = "Software\Microsoft\Windows\CurrentVersion\RunOnce"; $systemRunOncePath = "HKLM:\$runOncePath"; $logonPath = "HKLM:\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Winlogon" $runOnceName = "PortValhalla"; $autologinOption = "AutoAdminLogon"; $domainOption = "DefaultDomainName"; $userOption = "DefaultUserName"; $passwordOption = "DefaultPassword"; <# .SYNOPSIS Gets the reghistry key containing the `RunOnce` commands. #> function Get-RunOnceKey { param( [RegistryKey] $UserKey ) [string] $path = $null; if ($UserKey) { $path = "$($UserKey.PSPath)\$runOncePath"; } else { $path = $systemRunOncePath; } if (-not (Test-Path $path)) { New-Item $path; } else { Get-Item $path; } } <# .SYNOPSIS Generates a script for executing the installer. #> function Get-StartupScript { "pwsh " + (Get-StartupArguments); } <# .SYNOPSIS Generates arguments for running the installer using `pwsh`. #> function Get-StartupArguments { "-Command " + (& { if ($env:PWSH_PATH) { "`$env:PWSH_PATH = $(ConvertTo-Injection $env:PWSH_PATH);" } else { "" } }) + "`$env:INSTALLER_SCRIPT = $(ConvertTo-Injection (Resolve-Path $env:INSTALLER_SCRIPT));" + "`$env:CONFIG_MODULE = $(ConvertTo-Injection (Resolve-Path $env:CONFIG_MODULE));" + "& `$env:INSTALLER_SCRIPT;"; } <# .SYNOPSIS Registers a task to run the setup once after the next reboot. .PARAMETER UserKey The regtistry key of the user to register the task for. #> function Register-Setup { param( [Parameter(ParameterSetName="System")] [switch] $System, [Parameter(ParameterSetName="User", Mandatory)] [switch] $User, [Parameter(ParameterSetName="User")] [Parameter(ParameterSetName="SpecificUser", Mandatory)] [RegistryKey] $UserKey ) if ($User.IsPresent -or $UserKey) { if (-not $UserKey) { $UserKey = Get-Item "HKCU:\"; } $key = Get-RunOnceKey $UserKey; } else { $key = Get-RunOnceKey; } Set-ItemProperty -Path $key.PSPath -Name $runOnceName -Type "ExpandString" -Value (Get-StartupScript); $key.Handle.Close(); } <# .SYNOPSIS Clears leftovers from past registrations. #> function Clear-SetupRegistration { Edit-DefaultUserKey { param( [RegistryKey] $Key ) $runOnceKey = Get-RunOnceKey $Key; Remove-Item $runOnceKey.PSPath; } } <# .SYNOPSIS Sets the user to login automatically on boot. .PARAMETER Name The name of the user to login automatically. #> function Set-AutologinUser { param( [string] $Name ) Set-ItemProperty $logonPath -Name $autologinOption "1"; if (-not $Name) { $Name = Get-SetupUser; } $options = @{ $domainOption = ""; $userOption = $Name; $passwordOption = ""; }; foreach ($key in $options.Keys) { Set-ItemProperty $logonPath -Name $key -Value $options[$key]; } } <# .SYNOPSIS Disables the automatic login. #> function Disable-Autologin { Set-ItemProperty $logonPath -Name $autologinOption "0"; foreach ($key in @($domainOption, $userOption, $passwordOption)) { Remove-ItemProperty $logonPath -Name $key -ErrorAction SilentlyContinue; } } <# .SYNOPSIS Reboots the machine intermediately and restarts the setup after the next login. #> function Restart-Intermediate { param( [switch] $DefaultUser ) $register = { param($UserKey) Register-Setup @PSBoundParameters; }; if ($DefaultUser) { Edit-DefaultUserKey { param( [RegistryKey] $Key ) & $register $Key; } } else { & $register; } Restart-Computer -Force; } }