class Context { [string]$EntryPoint; [string]$RootDir; [string]$BackupName; [string]$UserName; [string]$AdminName = "Admin"; [string]$ConfigRoot = "HKLM:\Software\PortValhalla"; [string]$RunOnceName = "PortValhalla"; [string]$StagePropertyName = "Stage"; [string] ProjectRoot() { return Resolve-Path (Join-Path $PSScriptRoot ".." ".." ".."); } [string] BackupRoot() { if (-not $this.RootDir) { return Join-Path $this.ProjectRoot() $this.BackupName; } else { return $this.RootDir; } } [Microsoft.Win32.RegistryKey] EnsureConfigKey() { if (-not (Test-Path $this.ConfigRoot)) { $null = New-Item $this.ConfigRoot; $acl = Get-Acl $this.ConfigRoot; $acl.AddAccessRule( [System.Security.AccessControl.RegistryAccessRule]::new( [System.Security.Principal.SecurityIdentifier]::new([System.Security.Principal.WellKnownSidType]::BuiltinUsersSid, $null), [System.Security.AccessControl.RegistryRights]::FullControl, [System.Security.AccessControl.InheritanceFlags]::ObjectInherit -bor [System.Security.AccessControl.InheritanceFlags]::ContainerInherit, [System.Security.AccessControl.PropagationFlags]::None, [System.Security.AccessControl.AccessControlType]::Allow)); Set-Acl $this.ConfigRoot $acl; } return Get-Item $this.ConfigRoot; } [object] Get([string] $key) { $configKey = $this.EnsureConfigKey(); if ($configKey.GetValueNames().Contains($key)) { return $configKey.GetValue($key); } else { return $null; } } [void] Set([string] $key, $value, [Microsoft.Win32.RegistryValueKind] $type) { $configKey = $this.EnsureConfigKey(); $null = Set-ItemProperty -Path $configKey.PSPath -Name $key -Value $value -Type $type; } [void] Remove([string] $key) { $configKey = $this.EnsureConfigKey(); $null = Remove-ItemProperty -Path $configKey.PSPath -Name $key; } [void] SetStage([string] $name) { $this.Set($this.StagePropertyName, $name, "ExpandString"); } [string] GetStage() { return $this.Get($this.StagePropertyName); } [void] RemoveStage() { $this.Remove($this.StagePropertyName); } [string] ArchivePath($name) { return Join-Path $this.BackupRoot() "$name.7z"; } [string] FileArchivePath($name) { return $this.ArchivePath($(Join-Path "Files" $name)); } [string] SoftwareArchive([string]$softwareName) { return $this.ArchivePath($softwareName); } [void] Backup([string]$sourcePath, [string]$archivePath, [string[]]$arguments) { if (Test-Path $archivePath) { Remove-Item -Recurse $archivePath; } Start-Process -WorkingDirectory "$sourcePath" ` -FilePath "7z" ` -ArgumentList ( @( "a", "-xr!desktop.ini", "-xr!thumbs.db", "-xr!Thumbs.db", $archivePath) + $arguments) ` -Wait ` -NoNewWindow; } [void] Restore([string]$archivePath, [string]$destinationPath) { if (-not (Test-Path -PathType Leaf $archivePath)) { Write-Information ( [string]::Join( "`n", @( "An archive at the specified path $archivePath does not exist.", "No restoration will be performed."))); } else { if (-not (Test-Path -PathType Container $destinationPath)) { New-Item -ItemType Directory "$destinationPath"; } Start-Process -WorkingDirectory "$destinationPath" ` -FilePath "7z" -ArgumentList "x" ` -Wait ` -NoNewWindow; } } [string] GetTempDirectory() { $tempDir = Join-Path $([System.IO.Path]::GetTempPath()) $([System.IO.Path]::GetRandomFileName()); $null = New-Item -ItemType Directory $tempDir; return $tempDir; } [void] ProcessDefaultUserKey([System.Action[Microsoft.Win32.RegistryKey]] $action) { $rootPath = "HKLM:\DefaultUser"; $regRootPath = $rootPath.Replace(":", ""); $hivePath = "$env:SystemDrive\Users\Default\NTUSER.dat" $null = & reg load $regRootPath $hivePath; $root = Get-Item $rootPath; $action.Invoke($root); $root.Handle.Close(); [System.GC]::Collect(); & reg unload $regRootPath; } [void] ProcessLogonKey([System.Action[Microsoft.Win32.RegistryKey]] $action) { $key = Get-Item "HKLM:\Software\Microsoft\Windows NT\CurrentVersion\Winlogon"; $action.Invoke($key); } [Microsoft.Win32.RegistryKey] GetRunOnceKey() { return $this.GetRunOnceKey($null); } [Microsoft.Win32.RegistryKey] GetRunOnceKey([Microsoft.Win32.RegistryKey] $userKey) { if (-not $userKey) { $userKey = Get-Item "HKCU:\"; } Push-Location $userKey.PSPath; $runOncePath = "Software\Microsoft\Windows\CurrentVersion\RunOnce"; if (-not (Test-Path $runOncePath)) { New-Item $runOncePath; } $result = Get-Item $runOncePath; Pop-Location; return $result; } [void] RegisterReboot() { $this.RegisterReboot($null); } [void] RegisterReboot([Microsoft.Win32.RegistryKey] $userKey) { $runOnceKey = $this.GetRunOnceKey($userKey); Set-ItemProperty -Path $runOnceKey.PSPath -Name $this.RunOnceName -Value "pwsh `"$($this.EntryPoint)`"" -Type "ExpandString"; $runOnceKey.Handle.Close(); } [void] RegisterNewUserReboot() { $this.ProcessDefaultUserKey({ param ($root) $this.RegisterReboot($root); }); } [void] DeregisterNewUserReboot() { $this.ProcessDefaultUserKey({ param ($root) Remove-Item -Path $this.GetRunOnceKey($root).PSPath }); } [void] SetAutologin($user, $pw) { $this.ProcessLogonKey( { param ($logon) $path = $logon.PSPath; Set-ItemProperty $path -Name "AutoAdminLogon" 1; Set-ItemProperty $path -Name "DefaultUserName" $user; if ($pw) { Set-ItemProperty $path -Name "DefaultPassword" $pw; } else { Remove-ItemProperty $path -Name "DefaultPassword"; } }); } [void] RemoveAutologin() { $this.ProcessLogonKey( { param ($logon) $path = $logon.PSPath; Set-ItemProperty $path -Name "AutoAdminLogon" 0; Remove-ItemProperty $path -Name "DefaultDomainName" -ErrorAction SilentlyContinue; Remove-ItemProperty $path -Name "DefaultUserName" -ErrorAction SilentlyContinue; Remove-ItemProperty $path -Name "DefaultPassword" -ErrorAction SilentlyContinue; }); } [string] GetNextcloudConfigFile() { return "$env:APPDATA/Nextcloud/nextcloud.cfg"; } [void] AddNextcloudSync([string] $localPath, [string] $targetPath) { $this.AddNextcloudSync($localPath, $targetPath, $false); } [void] AddNextcloudSync([string] $localPath, [string] $targetPath, [bool] $virtualFiles) { $pattern = "^\d+\\Folders(?:WithPlaceholders)?\\(\d+)"; $folderID = ( Get-Content $($this.GetNextcloudConfigFile()) | ` Where-Object { $_ -match "$pattern" } | ` ForEach-Object { $_ -replace "$pattern.*$","`$1" } | ` Sort-Object -Unique | ` Measure-Object -Maximum).Maximum + 1; $configName = "Folders"; $localPath = $localPath.Replace("\", "/"); $targetPath = $targetPath.Replace("\", "/"); if ($virtualFiles) { $configName += "WithPlaceholders"; } $nextcloudProcess = Get-Process nextcloud; $nextcloudPath = $nextcloudProcess.Path; $nextcloudProcess | Stop-Process -Force; $accountSectionEntered = $false; (Get-Content $($this.GetNextcloudConfigFile())) | ` ForEach-Object { if ($_ -eq "[Accounts]") { $accountSectionEntered = $true; } if ($_ -eq "" -and $accountSectionEntered) { [string]::Join( "`n", @( "0\$configName\$folderID\localPath=$localPath", "0\$configName\$folderID\targetPath=$targetPath")); } $_; } | Set-Content $this.GetNextcloudConfigFile(); Start-Process $nextcloudPath; } [void] Reboot() { Write-Host "Restarting Computer..."; $this.RegisterReboot(); Restart-Computer; exit; } [void] Cleanup() { $this.DeregisterNewUserReboot(); $this.RemoveAutologin(); Remove-Item $($this.EnsureConfigKey().PSPath); } }