mirror of
https://github.com/rbalsleyMSFT/FFU.git
synced 2026-06-14 02:09:35 -06:00
dcb7957d15
Removes the parameter for installing preview cumulative updates to simplify the script's interface. Improves logging during the driver copy process to provide better visibility into which drivers are being used. The UI no longer loads the selected device 'Make' from the configuration on startup.
5346 lines
239 KiB
PowerShell
5346 lines
239 KiB
PowerShell
|
|
#Requires -Modules Hyper-V, Storage
|
|
#Requires -RunAsAdministrator
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
A PowerShell script to create a Windows 10/11 FFU file.
|
|
|
|
.DESCRIPTION
|
|
This script creates a Windows 10/11 FFU and USB drive to help quickly get a Windows device reimaged. FFU can be customized with drivers, apps, and additional settings.
|
|
|
|
.PARAMETER AllowExternalHardDiskMedia
|
|
When set to $true, will allow the use of media identified as External Hard Disk media via WMI class Win32_DiskDrive. Default is not defined.
|
|
|
|
.PARAMETER AllowVHDXCaching
|
|
When set to $true, will cache the VHDX file to the $FFUDevelopmentPath\VHDXCache folder and create a config json file that will keep track of the Windows build information, the updates installed, and the logical sector byte size information. Default is $false.
|
|
|
|
.PARAMETER AppListPath
|
|
Path to a JSON file containing a list of applications to install using WinGet. Default is $FFUDevelopmentPath\Apps\AppList.json.
|
|
|
|
.PARAMETER AppsScriptVariables
|
|
When passed a hashtable, the script will create an AppsScriptVariables.json file in the OrchestrationPath. This file will be used to pass variables to the Apps script. The hashtable should contain key-value pairs where the key is the variable name and the value is the variable value.
|
|
|
|
.PARAMETER BuildUSBDrive
|
|
When set to $true, will partition and format a USB drive and copy the captured FFU to the drive.
|
|
|
|
.PARAMETER CleanupAppsISO
|
|
When set to $true, will remove the Apps ISO after the FFU has been captured. Default is $true.
|
|
|
|
.PARAMETER CleanupCaptureISO
|
|
When set to $true, will remove the WinPE capture ISO after the FFU has been captured. Default is $true.
|
|
|
|
.PARAMETER CleanupDeployISO
|
|
When set to $true, will remove the WinPE deployment ISO after the FFU has been captured. Default is $true.
|
|
|
|
.PARAMETER CleanupDrivers
|
|
When set to $true, will remove the drivers folder after the FFU has been captured. Default is $true.
|
|
|
|
.PARAMETER CompactOS
|
|
When set to $true, will compact the OS when building the FFU. Default is $true.
|
|
|
|
.PARAMETER ConfigFile
|
|
Path to a JSON file containing parameters to use for the script. Default is $null.
|
|
|
|
.PARAMETER CopyAutopilot
|
|
When set to $true, will copy the $FFUDevelopmentPath\Autopilot folder to the Deployment partition of the USB drive. Default is $false.
|
|
|
|
.PARAMETER CopyDrivers
|
|
When set to $true, will copy the drivers from the $FFUDevelopmentPath\Drivers folder to the Drivers folder on the deploy partition of the USB drive. Default is $false.
|
|
|
|
.PARAMETER CopyPEDrivers
|
|
When set to $true, will copy the drivers from the $FFUDevelopmentPath\PEDrivers folder to the WinPE deployment media. Default is $false.
|
|
|
|
.PARAMETER CopyPPKG
|
|
When set to $true, will copy the provisioning package from the $FFUDevelopmentPath\PPKG folder to the Deployment partition of the USB drive. Default is $false.
|
|
|
|
.PARAMETER CopyUnattend
|
|
When set to $true, will copy the $FFUDevelopmentPath\Unattend folder to the Deployment partition of the USB drive. Default is $false.
|
|
|
|
.PARAMETER CreateCaptureMedia
|
|
When set to $true, this will create WinPE capture media for use when $InstallApps is set to $true. This capture media will be automatically attached to the VM, and the boot order will be changed to automate the capture of the FFU.
|
|
|
|
.PARAMETER CreateDeploymentMedia
|
|
When set to $true, this will create WinPE deployment media for use when deploying to a physical device.
|
|
|
|
.PARAMETER CustomFFUNameTemplate
|
|
Sets a custom FFU output name with placeholders. Allowed placeholders are: {WindowsRelease}, {WindowsVersion}, {SKU}, {BuildDate}, {yyyy}, {MM}, {dd}, {H}, {hh}, {mm}, {tt}.
|
|
|
|
.PARAMETER Disksize
|
|
Size of the virtual hard disk for the virtual machine. Default is a 30GB dynamic disk.
|
|
|
|
.PARAMETER DriversFolder
|
|
Path to the drivers folder. Default is $FFUDevelopmentPath\Drivers.
|
|
|
|
.PARAMETER ExportConfigFile
|
|
Path to a JSON file to export the parameters used for the script.
|
|
|
|
.PARAMETER FFUCaptureLocation
|
|
Path to the folder where the captured FFU will be stored. Default is $FFUDevelopmentPath\FFU.
|
|
|
|
.PARAMETER FFUDevelopmentPath
|
|
Path to the FFU development folder. Default is C:\FFUDevelopment.
|
|
|
|
.PARAMETER FFUPrefix
|
|
Prefix for the generated FFU file. Default is _FFU.
|
|
|
|
.PARAMETER Headers
|
|
Headers to use when downloading files. Not recommended to modify.
|
|
|
|
.PARAMETER InstallApps
|
|
When set to $true, the script will create an Apps.iso file from the $FFUDevelopmentPath\Apps folder. It will also create a VM, mount the Apps.iso, install the apps, sysprep, and capture the VM. When set to $false, the FFU is created from a VHDX file, and no VM is created.
|
|
|
|
.PARAMETER InstallDrivers
|
|
Install device drivers from the specified $FFUDevelopmentPath\Drivers folder if set to $true. Download the drivers and put them in the Drivers folder. The script will recurse the drivers folder and add the drivers to the FFU.
|
|
|
|
.PARAMETER InstallOffice
|
|
Install Microsoft Office if set to $true. The script will download the latest ODT and Office files in the $FFUDevelopmentPath\Apps\Office folder and install Office in the FFU via VM.
|
|
|
|
.PARAMETER ISOPath
|
|
Path to the Windows 10/11 ISO file.
|
|
|
|
.PARAMETER LogicalSectorSizeBytes
|
|
Unit32 value of 512 or 4096. Useful for 4Kn drives or devices shipping with UFS drives. Default is 512.
|
|
|
|
.PARAMETER Make
|
|
Make of the device to download drivers. Accepted values are: 'Microsoft', 'Dell', 'HP', 'Lenovo'.
|
|
|
|
.PARAMETER MediaType
|
|
String value of either 'business' or 'consumer'. This is used to identify which media type to download. Default is 'consumer'.
|
|
|
|
.PARAMETER Memory
|
|
Amount of memory to allocate for the virtual machine. Recommended to use 8GB if possible, especially for Windows 11. Default is 4GB.
|
|
|
|
.PARAMETER Model
|
|
Model of the device to download drivers. This is required if Make is set.
|
|
|
|
.PARAMETER Optimize
|
|
When set to $true, will optimize the FFU file. Default is $true.
|
|
|
|
.PARAMETER OptionalFeatures
|
|
Provide a semicolon-separated list of Windows optional features you want to include in the FFU (e.g., netfx3;TFTP).
|
|
|
|
.PARAMETER PEDriversFolder
|
|
Path to the folder containing drivers to be injected into the WinPE deployment media. Default is $FFUDevelopmentPath\PEDrivers.
|
|
|
|
.PARAMETER Processors
|
|
Number of virtual processors for the virtual machine. Recommended to use at least 4.
|
|
|
|
.PARAMETER ProductKey
|
|
Product key for the Windows edition specified in WindowsSKU. This will overwrite whatever SKU is entered for WindowsSKU. Recommended to use if you want to use a MAK or KMS key to activate Enterprise or Education. If using VL media instead of consumer media, you'll want to enter a MAK or KMS key here.
|
|
|
|
.PARAMETER PromptExternalHardDiskMedia
|
|
When set to $true, will prompt the user to confirm the use of media identified as External Hard Disk media via WMI class Win32_DiskDrive. Default is $true.
|
|
|
|
.PARAMETER RemoveFFU
|
|
When set to $true, will remove the FFU file from the $FFUDevelopmentPath\FFU folder after it has been copied to the USB drive. Default is $false.
|
|
|
|
.PARAMETER RemoveApps
|
|
When set to $true, will remove the application content in the Apps folder after the FFU has been captured. Default is $true.
|
|
|
|
.PARAMETER RemoveUpdates
|
|
When set to $true, will remove the downloaded CU, MSRT, Defender, Edge, OneDrive, and .NET files downloaded. Default is $true.
|
|
|
|
.PARAMETER ShareName
|
|
Name of the shared folder for FFU capture. The default is FFUCaptureShare. This share will be created with rights for the user account. When finished, the share will be removed.
|
|
|
|
.PARAMETER UpdateADK
|
|
When set to $true, the script will check for and install the latest Windows ADK and WinPE add-on if they are not already installed or up-to-date. Default is $true.
|
|
|
|
.PARAMETER UpdateEdge
|
|
When set to $true, will download and install the latest Microsoft Edge for Windows 10/11. Default is $false.
|
|
|
|
.PARAMETER UpdateLatestCU
|
|
When set to $true, will download and install the latest cumulative update for Windows 10/11. Default is $false.
|
|
|
|
.PARAMETER UpdateLatestDefender
|
|
When set to $true, will download and install the latest Windows Defender definitions and Defender platform update. Default is $false.
|
|
|
|
.PARAMETER UpdateLatestMicrocode
|
|
When set to $true, will download and install the latest microcode updates for applicable Windows releases (e.g., Windows Server 2016/2019, Windows 10 LTSC 2016/2019) into the FFU. Default is $false.
|
|
|
|
.PARAMETER UpdateLatestMSRT
|
|
When set to $true, will download and install the latest Windows Malicious Software Removal Tool. Default is $false.
|
|
|
|
.PARAMETER UpdateLatestNet
|
|
When set to $true, will download and install the latest .NET Framework for Windows 10/11. Default is $false.
|
|
|
|
.PARAMETER UpdateOneDrive
|
|
When set to $true, will download and install the latest OneDrive for Windows 10/11 and install it as a per-machine installation instead of per-user. Default is $false.
|
|
|
|
.PARAMETER UpdatePreviewCU
|
|
When set to $true, will download and install the latest Preview cumulative update for Windows 10/11. Default is $false.
|
|
|
|
.PARAMETER USBDriveList
|
|
A hashtable containing USB drives from win32_diskdrive where:
|
|
- Key: USB drive model name (partial match supported)
|
|
- Value: USB drive serial number (trailing partial match supported due to some serial numbers ending with blank spaces)
|
|
|
|
Example: @{ "SanDisk Ultra" = "1234567890"; "Kingston DataTraveler" = "0987654321" }
|
|
|
|
.PARAMETER UserAgent
|
|
User agent string to use when downloading files.
|
|
|
|
.PARAMETER Username
|
|
Username for accessing the shared folder. The default is ffu_user. The script will auto-create the account and password. When finished, it will remove the account.
|
|
|
|
.PARAMETER VMHostIPAddress
|
|
IP address of the Hyper-V host for FFU capture. If $InstallApps is set to $true, this parameter must be configured. You must manually configure this. The script will not auto-detect your IP (depending on your network adapters, it may not find the correct IP).
|
|
|
|
.PARAMETER VMLocation
|
|
Default is $FFUDevelopmentPath\VM. This is the location of the VHDX that gets created where Windows will be installed to.
|
|
|
|
.PARAMETER VMSwitchName
|
|
Name of the Hyper-V virtual switch. If $InstallApps is set to $true, this must be set. This is required to capture the FFU from the VM. The default is '*external*', but you will likely need to change this.
|
|
|
|
.PARAMETER WindowsArch
|
|
String value of 'x86' or 'x64'. This is used to identify which architecture of Windows to download. Default is 'x64'.
|
|
|
|
.PARAMETER WindowsLang
|
|
String value in language-region format (e.g., 'en-us'). This is used to identify which language of media to download. Default is 'en-us'.
|
|
|
|
.PARAMETER WindowsRelease
|
|
Integer value of 10 or 11. This is used to identify which release of Windows to download. Default is 11.
|
|
|
|
.PARAMETER WindowsSKU
|
|
Edition of Windows 10/11 to be installed. Accepted values are: 'Home', 'Home N', 'Home Single Language', 'Education', 'Education N', 'Pro', 'Pro N', 'Pro Education', 'Pro Education N', 'Pro for Workstations', 'Pro N for Workstations', 'Enterprise', 'Enterprise N'.
|
|
|
|
.PARAMETER WindowsVersion
|
|
String value of the Windows version to download. This is used to identify which version of Windows to download. Default is '24h2'.
|
|
|
|
.EXAMPLE
|
|
Command line for most people who want to download the latest Windows 11 Pro x64 media in English (US) with the latest Windows Cumulative Update, .NET Framework, Defender platform and definition updates, Edge, OneDrive, and Office/M365 Apps. It will also copy drivers to the FFU. This can take about 40 minutes to create the FFU due to the time it takes to download and install the updates.
|
|
.\BuildFFUVM.ps1 -WindowsSKU 'Pro' -Installapps $true -InstallOffice $true -InstallDrivers $true -VMSwitchName 'Name of your VM Switch in Hyper-V' -VMHostIPAddress 'Your IP Address' -CreateCaptureMedia $true -CreateDeploymentMedia $true -BuildUSBDrive $true -UpdateLatestCU $true -UpdateLatestNet $true -UpdateLatestDefender $true -UpdateEdge $true -UpdateOneDrive $true -verbose
|
|
|
|
Command line for most people who want to create an FFU with Office and drivers and have downloaded their own ISO. This assumes you have copied this script and associated files to the C:\FFUDevelopment folder. If you need to use another drive or folder, change the -FFUDevelopment parameter (e.g. -FFUDevelopment 'D:\FFUDevelopment')
|
|
.\BuildFFUVM.ps1 -ISOPath 'C:\path_to_iso\Windows.iso' -WindowsSKU 'Pro' -Installapps $true -InstallOffice $true -InstallDrivers $true -VMSwitchName 'Name of your VM Switch in Hyper-V' -VMHostIPAddress 'Your IP Address' -CreateCaptureMedia $true -CreateDeploymentMedia $true -BuildUSBDrive $true -verbose
|
|
|
|
Command line for those who just want a FFU with no drivers, apps, or Office and have downloaded their own ISO.
|
|
.\BuildFFUVM.ps1 -ISOPath 'C:\path_to_iso\Windows.iso' -WindowsSKU 'Pro' -Installapps $false -InstallOffice $false -InstallDrivers $false -CreateCaptureMedia $false -CreateDeploymentMedia $true -BuildUSBDrive $true -verbose
|
|
|
|
Command line for those who just want a FFU with Apps and drivers, no Office and have downloaded their own ISO.
|
|
.\BuildFFUVM.ps1 -ISOPath 'C:\path_to_iso\Windows.iso' -WindowsSKU 'Pro' -Installapps $true -InstallOffice $false -InstallDrivers $true -VMSwitchName 'Name of your VM Switch in Hyper-V' -VMHostIPAddress 'Your IP Address' -CreateCaptureMedia $true -CreateDeploymentMedia $true -BuildUSBDrive $true -verbose
|
|
|
|
Command line for those who want to download the latest Windows 11 Pro x64 media in English (US) and install the latest version of Office and drivers.
|
|
.\BuildFFUVM.ps1 -WindowsSKU 'Pro' -Installapps $true -InstallOffice $true -InstallDrivers $true -VMSwitchName 'Name of your VM Switch in Hyper-V' -VMHostIPAddress 'Your IP Address' -CreateCaptureMedia $true -CreateDeploymentMedia $true -BuildUSBDrive $true -verbose
|
|
|
|
Command line for those who want to download the latest Windows 11 Pro x64 media in French (CA) and install the latest version of Office and drivers.
|
|
.\BuildFFUVM.ps1 -WindowsSKU 'Pro' -Installapps $true -InstallOffice $true -InstallDrivers $true -VMSwitchName 'Name of your VM Switch in Hyper-V' -VMHostIPAddress 'Your IP Address' -CreateCaptureMedia $true -CreateDeploymentMedia $true -BuildUSBDrive $true -WindowsRelease 11 -WindowsArch 'x64' -WindowsLang 'fr-ca' -MediaType 'consumer' -verbose
|
|
|
|
Command line for those who want to download the latest Windows 11 Pro x64 media in English (US) and install the latest version of Office and drivers.
|
|
.\BuildFFUVM.ps1 -WindowsSKU 'Pro' -Installapps $true -InstallOffice $true -InstallDrivers $true -VMSwitchName 'Name of your VM Switch in Hyper-V' -VMHostIPAddress 'Your IP Address' -CreateCaptureMedia $true -CreateDeploymentMedia $true -BuildUSBDrive $true -verbose
|
|
|
|
.NOTES
|
|
Additional notes about your script.
|
|
|
|
.LINK
|
|
https://github.com/rbalsleyMSFT/FFU
|
|
#>
|
|
|
|
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(Mandatory = $false, Position = 0)]
|
|
[ValidateScript({ Test-Path $_ })]
|
|
[string]$ISOPath,
|
|
[ValidateSet(
|
|
'Home',
|
|
'Home N',
|
|
'Home Single Language',
|
|
'Education',
|
|
'Education N',
|
|
'Pro',
|
|
'Pro N',
|
|
'Pro Education',
|
|
'Pro Education N',
|
|
'Pro for Workstations',
|
|
'Pro N for Workstations',
|
|
'Enterprise',
|
|
'Enterprise N',
|
|
'Enterprise 2016 LTSB',
|
|
'Enterprise N 2016 LTSB',
|
|
'Enterprise LTSC',
|
|
'Enterprise N LTSC',
|
|
'IoT Enterprise LTSC',
|
|
'IoT Enterprise N LTSC',
|
|
'Standard',
|
|
'Standard (Desktop Experience)',
|
|
'Datacenter',
|
|
'Datacenter (Desktop Experience)'
|
|
)]
|
|
[string]$WindowsSKU = 'Pro',
|
|
[ValidateScript({ Test-Path $_ })]
|
|
[string]$FFUDevelopmentPath = $PSScriptRoot,
|
|
[bool]$InstallApps,
|
|
[string]$AppListPath,
|
|
[string]$UserAppListPath,
|
|
|
|
[hashtable]$AppsScriptVariables,
|
|
[bool]$InstallOffice,
|
|
[ValidateSet('Microsoft', 'Dell', 'HP', 'Lenovo')]
|
|
[string]$Make,
|
|
[string]$Model,
|
|
[bool]$InstallDrivers,
|
|
[uint64]$Memory = 4GB,
|
|
[uint64]$Disksize = 30GB,
|
|
[int]$Processors = 4,
|
|
[string]$VMSwitchName,
|
|
[string]$VMLocation,
|
|
[string]$FFUPrefix = '_FFU',
|
|
[string]$FFUCaptureLocation,
|
|
[string]$ShareName = "FFUCaptureShare",
|
|
[string]$Username = "ffu_user",
|
|
[string]$CustomFFUNameTemplate,
|
|
[Parameter(Mandatory = $false)]
|
|
[string]$VMHostIPAddress,
|
|
[bool]$CreateCaptureMedia = $true,
|
|
[bool]$CreateDeploymentMedia,
|
|
[ValidateScript({
|
|
$allowedFeatures = @("Windows-Defender-Default-Definitions", "Printing-PrintToPDFServices-Features", "Printing-XPSServices-Features", "TelnetClient", "TFTP",
|
|
"TIFFIFilter", "LegacyComponents", "DirectPlay", "MSRDC-Infrastructure", "Windows-Identity-Foundation", "MicrosoftWindowsPowerShellV2Root", "MicrosoftWindowsPowerShellV2",
|
|
"SimpleTCP", "NetFx4-AdvSrvs", "NetFx4Extended-ASPNET45", "WCF-Services45", "WCF-HTTP-Activation45", "WCF-TCP-Activation45", "WCF-Pipe-Activation45", "WCF-MSMQ-Activation45",
|
|
"WCF-TCP-PortSharing45", "IIS-WebServerRole", "IIS-WebServer", "IIS-CommonHttpFeatures", "IIS-HttpErrors", "IIS-HttpRedirect", "IIS-ApplicationDevelopment", "IIS-Security",
|
|
"IIS-RequestFiltering", "IIS-NetFxExtensibility", "IIS-NetFxExtensibility45", "IIS-HealthAndDiagnostics", "IIS-HttpLogging", "IIS-LoggingLibraries", "IIS-RequestMonitor",
|
|
"IIS-HttpTracing", "IIS-URLAuthorization", "IIS-IPSecurity", "IIS-Performance", "IIS-HttpCompressionDynamic", "IIS-WebServerManagementTools", "IIS-ManagementScriptingTools",
|
|
"IIS-IIS6ManagementCompatibility", "IIS-Metabase", "WAS-WindowsActivationService", "WAS-ProcessModel", "WAS-NetFxEnvironment", "WAS-ConfigurationAPI", "IIS-HostableWebCore",
|
|
"WCF-HTTP-Activation", "WCF-NonHTTP-Activation", "IIS-StaticContent", "IIS-DefaultDocument", "IIS-DirectoryBrowsing", "IIS-WebDAV", "IIS-WebSockets", "IIS-ApplicationInit",
|
|
"IIS-ISAPIFilter", "IIS-ISAPIExtensions", "IIS-ASPNET", "IIS-ASPNET45", "IIS-ASP", "IIS-CGI", "IIS-ServerSideIncludes", "IIS-CustomLogging", "IIS-BasicAuthentication",
|
|
"IIS-HttpCompressionStatic", "IIS-ManagementConsole", "IIS-ManagementService", "IIS-WMICompatibility", "IIS-LegacyScripts", "IIS-LegacySnapIn", "IIS-FTPServer", "IIS-FTPSvc",
|
|
"IIS-FTPExtensibility", "MSMQ-Container", "MSMQ-DCOMProxy", "MSMQ-Server", "MSMQ-ADIntegration", "MSMQ-HTTP", "MSMQ-Multicast", "MSMQ-Triggers", "IIS-CertProvider",
|
|
"IIS-WindowsAuthentication", "IIS-DigestAuthentication", "IIS-ClientCertificateMappingAuthentication", "IIS-IISCertificateMappingAuthentication", "IIS-ODBCLogging",
|
|
"NetFx3", "SMB1Protocol-Deprecation", "MediaPlayback", "WindowsMediaPlayer", "Client-DeviceLockdown", "Client-EmbeddedShellLauncher", "Client-EmbeddedBootExp",
|
|
"Client-EmbeddedLogon", "Client-KeyboardFilter", "Client-UnifiedWriteFilter", "HostGuardian", "MultiPoint-Connector", "MultiPoint-Connector-Services", "MultiPoint-Tools"
|
|
, "AppServerClient", "SearchEngine-Client-Package", "WorkFolders-Client", "Printing-Foundation-Features", "Printing-Foundation-InternetPrinting-Client",
|
|
"Printing-Foundation-LPDPrintService", "Printing-Foundation-LPRPortMonitor", "HypervisorPlatform", "VirtualMachinePlatform", "Microsoft-Windows-Subsystem-Linux",
|
|
"Client-ProjFS", "Containers-DisposableClientVM", 'Containers-DisposableClientVM', 'Microsoft-Hyper-V-All', 'Microsoft-Hyper-V', 'Microsoft-Hyper-V-Tools-All',
|
|
'Microsoft-Hyper-V-Management-PowerShell', 'Microsoft-Hyper-V-Hypervisor', 'Microsoft-Hyper-V-Services', 'Microsoft-Hyper-V-Management-Clients', 'DataCenterBridging',
|
|
'DirectoryServices-ADAM-Client', 'Windows-Defender-ApplicationGuard', 'ServicesForNFS-ClientOnly', 'ClientForNFS-Infrastructure', 'NFS-Administration', 'Containers', 'Containers-HNS',
|
|
'Containers-SDN', 'SMB1Protocol', 'SMB1Protocol-Client', 'SMB1Protocol-Server', 'SmbDirect')
|
|
$inputFeatures = $_ -split ';'
|
|
foreach ($feature in $inputFeatures) {
|
|
if (-not ($allowedFeatures -contains $feature)) {
|
|
throw "Invalid optional feature '$feature'. Allowed values: $($allowedFeatures -join ', ')"
|
|
}
|
|
}
|
|
return $true
|
|
})]
|
|
[string]$OptionalFeatures,
|
|
[string]$ProductKey,
|
|
[bool]$BuildUSBDrive,
|
|
[hashtable]$USBDriveList,
|
|
[Parameter(Mandatory = $false)]
|
|
[ValidateSet(10, 11, 2016, 2019, 2021, 2022, 2024, 2025)]
|
|
[int]$WindowsRelease = 11,
|
|
[Parameter(Mandatory = $false)]
|
|
[string]$WindowsVersion = '24h2',
|
|
[Parameter(Mandatory = $false)]
|
|
[ValidateSet('x86', 'x64', 'arm64')]
|
|
[string]$WindowsArch = 'x64',
|
|
[ValidateScript({
|
|
$allowedLang = @('ar-sa', 'bg-bg', 'cs-cz', 'da-dk', 'de-de', 'el-gr', 'en-gb', 'en-us', 'es-es', 'es-mx', 'et-ee', 'fi-fi', 'fr-ca', 'fr-fr', 'he-il', 'hr-hr', 'hu-hu',
|
|
'it-it', 'ja-jp', 'ko-kr', 'lt-lt', 'lv-lv', 'nb-no', 'nl-nl', 'pl-pl', 'pt-br', 'pt-pt', 'ro-ro', 'ru-ru', 'sk-sk', 'sl-si', 'sr-latn-rs', 'sv-se', 'th-th', 'tr-tr', 'uk-ua',
|
|
'zh-cn', 'zh-tw')
|
|
if ($allowedLang -contains $_) { $true } else { throw "Invalid WindowsLang value. Allowed values: $($allowedLang -join ', ')" }
|
|
return $true
|
|
})]
|
|
[Parameter(Mandatory = $false)]
|
|
[string]$WindowsLang = 'en-us',
|
|
[Parameter(Mandatory = $false)]
|
|
[ValidateSet('consumer', 'business')]
|
|
[string]$MediaType = 'consumer',
|
|
[ValidateSet(512, 4096)]
|
|
[uint32]$LogicalSectorSizeBytes = 512,
|
|
[bool]$Optimize = $true,
|
|
[string]$DriversJsonPath,
|
|
[bool]$CompressDownloadedDriversToWim = $false,
|
|
[bool]$CopyDrivers,
|
|
[bool]$CopyPEDrivers,
|
|
[bool]$RemoveFFU,
|
|
[bool]$UpdateLatestCU,
|
|
[bool]$UpdatePreviewCU,
|
|
[bool]$UpdateLatestMicrocode,
|
|
[bool]$UpdateLatestNet,
|
|
[bool]$UpdateLatestDefender,
|
|
[bool]$UpdateLatestMSRT,
|
|
[bool]$UpdateEdge,
|
|
[bool]$UpdateOneDrive,
|
|
[bool]$AllowVHDXCaching,
|
|
[bool]$CopyPPKG,
|
|
[bool]$CopyUnattend,
|
|
[bool]$CopyAutopilot,
|
|
[bool]$CompactOS = $true,
|
|
[bool]$CleanupCaptureISO = $true,
|
|
[bool]$CleanupDeployISO = $true,
|
|
[bool]$CleanupAppsISO = $true,
|
|
[bool]$RemoveUpdates = $true,
|
|
[bool]$RemoveApps = $true,
|
|
[string]$DriversFolder,
|
|
[string]$PEDriversFolder,
|
|
[bool]$CleanupDrivers = $true,
|
|
[string]$UserAgent = 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/125.0.0.0 Safari/537.36 Edg/125.0.0.0',
|
|
#Microsoft sites will intermittently fail on downloads. These headers are to help with that.
|
|
$Headers = @{
|
|
"Accept" = "text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.7"
|
|
"Accept-Encoding" = "gzip, deflate, br, zstd"
|
|
"Accept-Language" = "en-US,en;q=0.9"
|
|
"Priority" = "u=0, i"
|
|
"Sec-Ch-Ua" = "`"Microsoft Edge`";v=`"125`", `"Chromium`";v=`"125`", `"Not.A/Brand`";v=`"24`""
|
|
"Sec-Ch-Ua-Mobile" = "?0"
|
|
"Sec-Ch-Ua-Platform" = "`"Windows`""
|
|
"Sec-Fetch-Dest" = "document"
|
|
"Sec-Fetch-Mode" = "navigate"
|
|
"Sec-Fetch-Site" = "none"
|
|
"Sec-Fetch-User" = "?1"
|
|
"Upgrade-Insecure-Requests" = "1"
|
|
},
|
|
[bool]$AllowExternalHardDiskMedia,
|
|
[bool]$PromptExternalHardDiskMedia = $true,
|
|
[Parameter(Mandatory = $false)]
|
|
[ValidateScript({ $_ -eq $null -or (Test-Path $_) })]
|
|
[string]$ConfigFile,
|
|
[Parameter(Mandatory = $false)]
|
|
[string]$ExportConfigFile,
|
|
[string]$orchestrationPath,
|
|
[bool]$UpdateADK = $true
|
|
)
|
|
$version = '2505.1'
|
|
|
|
# Remove any existing modules to avoid conflicts
|
|
if (Get-Module -Name 'FFU.Common.Core' -ErrorAction SilentlyContinue) {
|
|
Remove-Module -Name 'FFU.Common.Core' -Force
|
|
}
|
|
if (Get-Module -Name 'FFU.Common.Winget' -ErrorAction SilentlyContinue) {
|
|
Remove-Module -Name 'FFU.Common.Winget' -Force
|
|
}
|
|
if (Get-Module -Name 'FFU.Common.Drivers' -ErrorAction SilentlyContinue) {
|
|
Remove-Module -Name 'FFU.Common.Drivers' -Force
|
|
}
|
|
# Import the required modules
|
|
Import-Module "$PSScriptRoot\FFU.Common" -Force
|
|
|
|
# If a config file is specified and it exists, load it
|
|
if ($ConfigFile -and (Test-Path -Path $ConfigFile)) {
|
|
$configData = Get-Content $ConfigFile -Raw | ConvertFrom-Json
|
|
$keys = $configData.psobject.Properties.Name
|
|
|
|
# Iterate through the keys in the config data
|
|
foreach ($key in $keys) {
|
|
$value = $configdata.$key
|
|
|
|
# If $value is empty, skip
|
|
if ($null -eq $value -or
|
|
([string]::IsNullOrEmpty([string]$value)) -or
|
|
($value -is [System.Collections.Hashtable] -and $value.Count -eq 0) -or
|
|
($value -is [System.UInt32] -and $value -eq 0) -or
|
|
($value -is [System.UInt64] -and $value -eq 0) -or
|
|
($value -is [System.Int32] -and $value -eq 0)) {
|
|
continue
|
|
}
|
|
|
|
# If this is the Headers parameter, convert PSCustomObject to hashtable
|
|
if ((($key -eq 'Headers') -or ($key -eq 'AppsScriptVariables') -or ($key -eq 'USBDriveList')) -and ($value -is [System.Management.Automation.PSCustomObject])) {
|
|
$hashtableValue = [hashtable]::new()
|
|
foreach ($prop in $value.psobject.Properties) {
|
|
$hashtableValue[$prop.Name] = $prop.Value
|
|
}
|
|
$value = $hashtableValue
|
|
}
|
|
|
|
# Check if this key matches a parameter in the script
|
|
# and if the user did NOT explicitly supply it on the command line
|
|
if ($MyInvocation.MyCommand.Parameters.ContainsKey($key) -and -not $PSBoundParameters.ContainsKey($key)) {
|
|
# Set the parameter's value to what's in the config file
|
|
Set-Variable -Name $key -Value $value -Scope 0
|
|
}
|
|
}
|
|
}
|
|
|
|
# Validate that the selected Windows SKU is compatible with the chosen Windows release and ensure an ISO is provided for unsupported releases
|
|
$clientSKUs = @(
|
|
'Home',
|
|
'Home N',
|
|
'Home Single Language',
|
|
'Education',
|
|
'Education N',
|
|
'Pro',
|
|
'Pro N',
|
|
'Pro Education',
|
|
'Pro Education N',
|
|
'Pro for Workstations',
|
|
'Pro N for Workstations',
|
|
'Enterprise',
|
|
'Enterprise N'
|
|
)
|
|
$LTSCSKUs = @(
|
|
'Enterprise 2016 LTSB',
|
|
'Enterprise N 2016 LTSB',
|
|
'Enterprise LTSC',
|
|
'Enterprise N LTSC',
|
|
'IoT Enterprise LTSC',
|
|
'IoT Enterprise N LTSC'
|
|
)
|
|
$ServerSKUs = @(
|
|
'Standard',
|
|
'Standard (Desktop Experience)',
|
|
'Datacenter',
|
|
'Datacenter (Desktop Experience)'
|
|
)
|
|
$releaseToSKUMapping = @{
|
|
10 = $clientSKUs
|
|
11 = $clientSKUs
|
|
2016 = $LTSCSKUs + $ServerSKUs
|
|
2019 = $LTSCSKUs + $ServerSKUs
|
|
2021 = $LTSCSKUs
|
|
2022 = $ServerSKUs
|
|
2024 = $LTSCSKUs
|
|
2025 = $ServerSKUs
|
|
}
|
|
if ($releaseToSKUMapping.ContainsKey($WindowsRelease) -and $WindowsSKU -notin $releaseToSKUMapping[$WindowsRelease]) {
|
|
throw "Selected SKU is $WindowsSKU. Windows $WindowsRelease requires one of these SKUs: $($releaseToSKUMapping[$WindowsRelease] -join ', ')"
|
|
}
|
|
if ($WindowsRelease -notin 10, 11 -and -not $ISOPath) {
|
|
throw "Windows $WindowsRelease cannot automatically be downloaded. Please specify your own ISO using the -ISOPath parameter."
|
|
}
|
|
|
|
#Class definition for vhdx cache
|
|
class VhdxCacheUpdateItem {
|
|
[string]$Name
|
|
VhdxCacheUpdateItem([string]$Name) {
|
|
$this.Name = $Name
|
|
}
|
|
}
|
|
|
|
class VhdxCacheItem {
|
|
[string]$VhdxFileName = ""
|
|
[uint32]$LogicalSectorSizeBytes = ""
|
|
[string]$WindowsSKU = ""
|
|
[string]$WindowsRelease = ""
|
|
[string]$WindowsVersion = ""
|
|
[string]$OptionalFeatures = ""
|
|
[VhdxCacheUpdateItem[]]$IncludedUpdates = @()
|
|
}
|
|
|
|
#Check if Hyper-V feature is installed (requires only checks the module)
|
|
$osInfo = Get-WmiObject -Class Win32_OperatingSystem
|
|
$isServer = $osInfo.Caption -match 'server'
|
|
|
|
if ($isServer) {
|
|
$hyperVFeature = Get-WindowsFeature -Name Hyper-V
|
|
if ($hyperVFeature.InstallState -ne "Installed") {
|
|
Write-Host "Hyper-V feature is not installed. Please install it before running this script."
|
|
exit
|
|
}
|
|
}
|
|
else {
|
|
$hyperVFeature = Get-WindowsOptionalFeature -Online -FeatureName Microsoft-Hyper-V-All
|
|
if ($hyperVFeature.State -ne "Enabled") {
|
|
Write-Host "Hyper-V feature is not enabled. Please enable it before running this script."
|
|
exit
|
|
}
|
|
}
|
|
|
|
# Set default values for variables that depend on other parameters
|
|
if (-not $AppsISO) { $AppsISO = "$FFUDevelopmentPath\Apps.iso" }
|
|
if (-not $AppsPath) { $AppsPath = "$FFUDevelopmentPath\Apps" }
|
|
if (-not $AppListPath) { $AppListPath = "$AppsPath\AppList.json" }
|
|
if (-not $UserAppListPath) { $UserAppListPath = "$AppsPath\UserAppList.json" }
|
|
if (-not $OrchestrationPath) { $OrchestrationPath = "$AppsPath\Orchestration" }
|
|
if (-not $wingetWin32jsonFile) { $wingetWin32jsonFile = "$OrchestrationPath\WinGetWin32Apps.json" }
|
|
if (-not $InstallDefenderPath) { $installDefenderPath = "$OrchestrationPath\Update-Defender.ps1" }
|
|
if (-not $InstallMSRTPath) { $installMSRTPath = "$OrchestrationPath\Update-MSRT.ps1" }
|
|
if (-not $InstallODPath) { $installODPath = "$OrchestrationPath\Update-OneDrive.ps1" }
|
|
if (-not $InstallEdgePath) { $installEdgePath = "$OrchestrationPath\Update-Edge.ps1" }
|
|
if (-not $AppsScriptVarsJsonPath) { $AppsScriptVarsJsonPath = "$OrchestrationPath\AppsScriptVariables.json" }
|
|
if (-not $DeployISO) { $DeployISO = "$FFUDevelopmentPath\WinPE_FFU_Deploy_$WindowsArch.iso" }
|
|
if (-not $CaptureISO) { $CaptureISO = "$FFUDevelopmentPath\WinPE_FFU_Capture_$WindowsArch.iso" }
|
|
if (-not $OfficePath) { $OfficePath = "$AppsPath\Office" }
|
|
if (-not $OfficeDownloadXML) { $OfficeDownloadXML = "$OfficePath\DownloadFFU.xml" }
|
|
if (-not $OfficeInstallXML) { $OfficeInstallXML = "DeployFFU.xml" }
|
|
if (-not $rand) { $rand = Get-Random }
|
|
if (-not $VMLocation) { $VMLocation = "$FFUDevelopmentPath\VM" }
|
|
if (-not $VMName) { $VMName = "$FFUPrefix-$rand" }
|
|
if (-not $VMPath) { $VMPath = "$VMLocation\$VMName" }
|
|
if (-not $VHDXPath) { $VHDXPath = "$VMPath\$VMName.vhdx" }
|
|
if (-not $FFUCaptureLocation) { $FFUCaptureLocation = "$FFUDevelopmentPath\FFU" }
|
|
if (-not $LogFile) { $LogFile = "$FFUDevelopmentPath\FFUDevelopment.log" }
|
|
if (-not $KBPath) { $KBPath = "$FFUDevelopmentPath\KB" }
|
|
if (-not $MicrocodePath) { $MicrocodePath = "$KBPath\Microcode" }
|
|
if (-not $DefenderPath) { $DefenderPath = "$AppsPath\Defender" }
|
|
if (-not $MSRTPath) { $MSRTPath = "$AppsPath\MSRT" }
|
|
if (-not $OneDrivePath) { $OneDrivePath = "$AppsPath\OneDrive" }
|
|
if (-not $EdgePath) { $EdgePath = "$AppsPath\Edge" }
|
|
if (-not $DriversFolder) { $DriversFolder = "$FFUDevelopmentPath\Drivers" }
|
|
if (-not $PPKGFolder) { $PPKGFolder = "$FFUDevelopmentPath\PPKG" }
|
|
if (-not $UnattendFolder) { $UnattendFolder = "$FFUDevelopmentPath\Unattend" }
|
|
if (-not $AutopilotFolder) { $AutopilotFolder = "$FFUDevelopmentPath\Autopilot" }
|
|
if (-not $PEDriversFolder) { $PEDriversFolder = "$FFUDevelopmentPath\PEDrivers" }
|
|
if (-not $VHDXCacheFolder) { $VHDXCacheFolder = "$FFUDevelopmentPath\VHDXCache" }
|
|
if (-not $installationType) { $installationType = if ($WindowsSKU -like "Standard*" -or $WindowsSKU -like "Datacenter*") { 'Server' } else { 'Client' } }
|
|
if ($installationType -eq 'Server') {
|
|
#Map $WindowsRelease to $WindowsVersion for Windows Server
|
|
switch ($WindowsRelease) {
|
|
2016 { $WindowsVersion = '1607' }
|
|
2019 { $WindowsVersion = '1809' }
|
|
2022 { $WindowsVersion = '21H2' }
|
|
2025 { $WindowsVersion = '24H2' }
|
|
}
|
|
}
|
|
if (-not $AppListPath) { $AppListPath = "$AppsPath\AppList.json" }
|
|
|
|
if ($WindowsSKU -like "*LTS*") {
|
|
switch ($WindowsRelease) {
|
|
2016 { $WindowsVersion = '1607' }
|
|
2019 { $WindowsVersion = '1809' }
|
|
2021 { $WindowsVersion = '21H2' }
|
|
2024 { $WindowsVersion = '24H2' }
|
|
}
|
|
$isLTSC = $true
|
|
}
|
|
|
|
# Set the log path for the common logger
|
|
Set-CommonCoreLogPath -Path $LogFile
|
|
# Set critical paths and configuration as global variables for module access
|
|
# This is done after Set-CommonCoreLogPath so this action itself can be logged.
|
|
# Ensure $AppsPath, $orchestrationPath, and $WindowsArch are fully initialized before this point.
|
|
$global:AppsPath = $AppsPath
|
|
$global:orchestrationPath = $orchestrationPath
|
|
$global:WindowsArch = $WindowsArch
|
|
WriteLog "Global script variables set for module access: AppsPath='$global:AppsPath', orchestrationPath='$global:orchestrationPath', WindowsArch='$global:WindowsArch'"
|
|
|
|
|
|
|
|
#FUNCTIONS
|
|
|
|
function Get-Parameters {
|
|
[CmdletBinding()]
|
|
param (
|
|
[Parameter()]
|
|
$ParamNames
|
|
)
|
|
# Define unwanted parameters
|
|
$excludedParams = 'Debug', 'ErrorAction', 'ErrorVariable', 'InformationAction', 'InformationVariable', 'OutBuffer', 'OutVariable', 'PipelineVariable', 'Verbose', 'WarningAction', 'WarningVariable', 'ProgressAction'
|
|
|
|
# Filter out the unwanted parameters
|
|
$filteredParamNames = $paramNames | Where-Object { $excludedParams -notcontains $_ }
|
|
return $filteredParamNames
|
|
}
|
|
|
|
function LogVariableValues {
|
|
$excludedVariables = @(
|
|
'PSBoundParameters',
|
|
'PSScriptRoot',
|
|
'PSCommandPath',
|
|
'MyInvocation',
|
|
'?',
|
|
'ConsoleFileName',
|
|
'ExecutionContext',
|
|
'false',
|
|
'HOME',
|
|
'Host',
|
|
'hyperVFeature',
|
|
'input',
|
|
'MaximumAliasCount',
|
|
'MaximumDriveCount',
|
|
'MaximumErrorCount',
|
|
'MaximumFunctionCount',
|
|
'MaximumVariableCount',
|
|
'null',
|
|
'PID',
|
|
'PSCmdlet',
|
|
'PSCulture',
|
|
'PSUICulture',
|
|
'PSVersionTable',
|
|
'ShellId',
|
|
'true'
|
|
)
|
|
|
|
$allVariables = Get-Variable -Scope Script | Where-Object { $_.Name -notin $excludedVariables }
|
|
Writelog "Script version: $version"
|
|
WriteLog 'Logging variables'
|
|
foreach ($variable in $allVariables) {
|
|
$variableName = $variable.Name
|
|
$variableValue = $variable.Value
|
|
if ($null -ne $variableValue) {
|
|
WriteLog "[VAR]$variableName`: $variableValue"
|
|
}
|
|
else {
|
|
WriteLog "[VAR]Variable $variableName not found or not set"
|
|
}
|
|
}
|
|
WriteLog 'End logging variables'
|
|
}
|
|
|
|
function Get-ChildProcesses($parentId) {
|
|
$result = @()
|
|
$children = Get-CimInstance Win32_Process -Filter "ParentProcessId = $parentId"
|
|
foreach ($child in $children) {
|
|
$result += $child
|
|
$result += Get-ChildProcesses $child.ProcessId
|
|
}
|
|
return $result
|
|
}
|
|
|
|
function Test-Url {
|
|
param (
|
|
[Parameter(Mandatory = $true)]
|
|
[string]$Url
|
|
)
|
|
try {
|
|
# Create a web request and check the response
|
|
$request = [System.Net.WebRequest]::Create($Url)
|
|
$request.Method = 'HEAD'
|
|
$response = $request.GetResponse()
|
|
return $true
|
|
}
|
|
catch {
|
|
return $false
|
|
}
|
|
}
|
|
|
|
# Function to download a file using BITS with retry and error handling
|
|
function Start-BitsTransferWithRetry {
|
|
param (
|
|
[Parameter(Mandatory = $true)]
|
|
[string]$Source,
|
|
[Parameter(Mandatory = $true)]
|
|
[string]$Destination,
|
|
[int]$Retries = 3
|
|
)
|
|
|
|
$attempt = 0
|
|
$lastError = $null
|
|
|
|
while ($attempt -lt $Retries) {
|
|
try {
|
|
$OriginalVerbosePreference = $VerbosePreference
|
|
$VerbosePreference = 'SilentlyContinue'
|
|
$ProgressPreference = 'SilentlyContinue'
|
|
Start-BitsTransfer -Source $Source -Destination $Destination -ErrorAction Stop
|
|
$ProgressPreference = 'Continue'
|
|
$VerbosePreference = $OriginalVerbosePreference
|
|
return
|
|
}
|
|
catch {
|
|
# Capture the error that occurred during the failed download attempt
|
|
$lastError = $_
|
|
$attempt++
|
|
WriteLog "Attempt $attempt of $Retries failed to download $Source with error: $($lastError.Exception.Message). Retrying..."
|
|
Start-Sleep -Seconds 5
|
|
}
|
|
}
|
|
|
|
WriteLog "Failed to download $Source after $Retries attempts. Error: $($lastError.Exception.Message)"
|
|
throw $lastError
|
|
}
|
|
|
|
function Get-MicrosoftDrivers {
|
|
param (
|
|
[string]$Make,
|
|
[string]$Model,
|
|
[int]$WindowsRelease
|
|
)
|
|
|
|
$url = "https://support.microsoft.com/en-us/surface/download-drivers-and-firmware-for-surface-09bb2e09-2a4b-cb69-0951-078a7739e120"
|
|
|
|
### DOWNLOAD DRIVER PAGE CONTENT
|
|
WriteLog "Getting Surface driver information from $url"
|
|
$OriginalVerbosePreference = $VerbosePreference
|
|
$VerbosePreference = 'SilentlyContinue'
|
|
$webContent = Invoke-WebRequest -Uri $url -UseBasicParsing -Headers $Headers -UserAgent $UserAgent
|
|
$VerbosePreference = $OriginalVerbosePreference
|
|
WriteLog "Complete"
|
|
|
|
### PARSE THE DRIVER PAGE CONTENT FOR MODELS AND DOWNLOAD LINKS
|
|
WriteLog "Parsing web content for models and download links"
|
|
$html = $webContent.Content
|
|
|
|
# Regex to match divs with selectable-content-options__option-content classes
|
|
$divPattern = '<div[^>]*class="selectable-content-options__option-content(?: ocHidden)?"[^>]*>(.*?)</div>'
|
|
$divMatches = [regex]::Matches($html, $divPattern, [System.Text.RegularExpressions.RegexOptions]::Singleline)
|
|
|
|
$models = @()
|
|
|
|
foreach ($divMatch in $divMatches) {
|
|
$divContent = $divMatch.Groups[1].Value
|
|
|
|
# Find all tables within the div
|
|
$tablePattern = '<table[^>]*>(.*?)</table>'
|
|
$tableMatches = [regex]::Matches($divContent, $tablePattern, [System.Text.RegularExpressions.RegexOptions]::Singleline)
|
|
|
|
foreach ($tableMatch in $tableMatches) {
|
|
$tableContent = $tableMatch.Groups[1].Value
|
|
|
|
# Find all rows in the table
|
|
$rowPattern = '<tr[^>]*>(.*?)</tr>'
|
|
$rowMatches = [regex]::Matches($tableContent, $rowPattern, [System.Text.RegularExpressions.RegexOptions]::Singleline)
|
|
|
|
foreach ($rowMatch in $rowMatches) {
|
|
$rowContent = $rowMatch.Groups[1].Value
|
|
|
|
# Extract cells from the row
|
|
$cellPattern = '<td[^>]*>\s*(?:<p[^>]*>)?(.*?)(?:</p>)?\s*</td>'
|
|
$cellMatches = [regex]::Matches($rowContent, $cellPattern, [System.Text.RegularExpressions.RegexOptions]::Singleline)
|
|
|
|
if ($cellMatches.Count -ge 2) {
|
|
# Model name in the first TD
|
|
$modelName = ($cellMatches[0].Groups[1].Value).Trim()
|
|
|
|
# # Remove <p> and </p> tags if present
|
|
# $modelName = $modelName -replace '<p[^>]*>', '' -replace '</p>', ''
|
|
# $modelName = $modelName.Trim()
|
|
|
|
|
|
# The second TD might contain a link or just text
|
|
$secondTdContent = $cellMatches[1].Groups[1].Value.Trim()
|
|
|
|
# Look for a link in the second TD
|
|
$linkPattern = '<a[^>]+href="([^"]+)"[^>]*>'
|
|
$linkMatch = [regex]::Match($secondTdContent, $linkPattern, [System.Text.RegularExpressions.RegexOptions]::IgnoreCase)
|
|
|
|
if ($linkMatch.Success) {
|
|
$modelLink = $linkMatch.Groups[1].Value
|
|
}
|
|
else {
|
|
# No link, just text instructions
|
|
$modelLink = $secondTdContent
|
|
}
|
|
|
|
$models += [PSCustomObject]@{ Model = $modelName; Link = $modelLink }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
WriteLog "Parsing complete"
|
|
|
|
### FIND THE MODEL IN THE LIST OF MODELS
|
|
$selectedModel = $models | Where-Object { $_.Model -eq $Model }
|
|
|
|
if ($null -eq $selectedModel) {
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Host "The model '$Model' was not found in the list of available models."
|
|
Write-Host "Please run the script with the -Verbose switch to see the list of available models."
|
|
}
|
|
WriteLog "The model '$Model' was not found in the list of available models."
|
|
WriteLog "Please select a model from the list below by number:"
|
|
|
|
for ($i = 0; $i -lt $models.Count; $i++) {
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Host "$($i + 1). $($models[$i].Model)"
|
|
}
|
|
WriteLog "$($i + 1). $($models[$i].Model)"
|
|
}
|
|
|
|
do {
|
|
$selection = Read-Host "Enter the number of the model you want to select"
|
|
WriteLog "User selected model number: $selection"
|
|
|
|
if ($selection -match '^\d+$' -and [int]$selection -ge 1 -and [int]$selection -le $models.Count) {
|
|
$selectedModel = $models[$selection - 1]
|
|
}
|
|
else {
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Host "Invalid selection. Please try again."
|
|
}
|
|
WriteLog "Invalid selection. Please try again."
|
|
}
|
|
} while ($null -eq $selectedModel)
|
|
}
|
|
|
|
$Model = $selectedModel.Model
|
|
WriteLog "Model: $Model"
|
|
WriteLog "Download Page: $($selectedModel.Link)"
|
|
|
|
### GET THE DOWNLOAD LINK FOR THE SELECTED MODEL
|
|
WriteLog "Getting download page content"
|
|
$OriginalVerbosePreference = $VerbosePreference
|
|
$VerbosePreference = 'SilentlyContinue'
|
|
$downloadPageContent = Invoke-WebRequest -Uri $selectedModel.Link -UseBasicParsing -Headers $Headers -UserAgent $UserAgent
|
|
$VerbosePreference = $OriginalVerbosePreference
|
|
WriteLog "Complete"
|
|
WriteLog "Parsing download page for file"
|
|
$scriptPattern = '<script>window.__DLCDetails__={(.*?)}<\/script>'
|
|
$scriptMatch = [regex]::Match($downloadPageContent.Content, $scriptPattern)
|
|
|
|
if ($scriptMatch.Success) {
|
|
$scriptContent = $scriptMatch.Groups[1].Value
|
|
|
|
# Extract the download file information from the script tag
|
|
$downloadFilePattern = '"name":"(.*?)",.*?"url":"(.*?)"'
|
|
$downloadFileMatches = [regex]::Matches($scriptContent, $downloadFilePattern)
|
|
|
|
$downloadLink = $null
|
|
foreach ($downloadFile in $downloadFileMatches) {
|
|
$fileName = $downloadFile.Groups[1].Value
|
|
$fileUrl = $downloadFile.Groups[2].Value
|
|
|
|
if ($fileName -match "Win$WindowsRelease") {
|
|
$downloadLink = $fileUrl
|
|
break
|
|
}
|
|
}
|
|
|
|
|
|
### CREATE FOLDER STRUCTURE AND DOWNLOAD AND EXTRACT THE FILE
|
|
if ($downloadLink) {
|
|
WriteLog "Download Link for Windows ${WindowsRelease}: $downloadLink"
|
|
|
|
# Create directory structure
|
|
if (-not (Test-Path -Path $DriversFolder)) {
|
|
WriteLog "Creating Drivers folder: $DriversFolder"
|
|
New-Item -Path $DriversFolder -ItemType Directory -Force | Out-Null
|
|
WriteLog "Drivers folder created"
|
|
}
|
|
$surfaceDriversPath = Join-Path -Path $DriversFolder -ChildPath $Make
|
|
$modelPath = Join-Path -Path $surfaceDriversPath -ChildPath $Model
|
|
if (-Not (Test-Path -Path $modelPath)) {
|
|
WriteLog "Creating model folder: $modelPath"
|
|
New-Item -Path $modelPath -ItemType Directory | Out-Null
|
|
WriteLog "Complete"
|
|
}
|
|
|
|
### DOWNLOAD THE FILE
|
|
$filePath = Join-Path -Path $surfaceDriversPath -ChildPath ($fileName)
|
|
WriteLog "Downloading $Model driver file to $filePath"
|
|
Start-BitsTransferWithRetry -Source $downloadLink -Destination $filePath
|
|
WriteLog "Download complete"
|
|
|
|
# Determine file extension
|
|
$fileExtension = [System.IO.Path]::GetExtension($filePath).ToLower()
|
|
|
|
### EXTRACT THE FILE
|
|
if ($fileExtension -eq ".msi") {
|
|
# Extract the MSI file using an administrative install
|
|
WriteLog "Extracting MSI file to $modelPath"
|
|
$arguments = "/a `"$($filePath)`" /qn TARGETDIR=`"$($modelPath)`""
|
|
Invoke-Process -FilePath "msiexec.exe" -ArgumentList $arguments | Out-Null
|
|
WriteLog "Extraction complete"
|
|
}
|
|
elseif ($fileExtension -eq ".zip") {
|
|
# Extract the ZIP file
|
|
WriteLog "Extracting ZIP file to $modelPath"
|
|
$ProgressPreference = 'SilentlyContinue'
|
|
Expand-Archive -Path $filePath -DestinationPath $modelPath -Force
|
|
$ProgressPreference = 'Continue'
|
|
WriteLog "Extraction complete"
|
|
}
|
|
else {
|
|
WriteLog "Unsupported file type: $fileExtension"
|
|
}
|
|
# Remove the downloaded file
|
|
WriteLog "Removing $filePath"
|
|
Remove-Item -Path $filePath -Force
|
|
WriteLog "Complete"
|
|
}
|
|
else {
|
|
WriteLog "No download link found for Windows $WindowsRelease."
|
|
}
|
|
}
|
|
else {
|
|
WriteLog "Failed to parse the download page for the MSI file."
|
|
}
|
|
}
|
|
|
|
function Get-HPDrivers {
|
|
[CmdletBinding()]
|
|
param (
|
|
[Parameter()]
|
|
[string]$Make,
|
|
[Parameter()]
|
|
[string]$Model,
|
|
[Parameter()]
|
|
[ValidateSet("x64", "x86", "ARM64")]
|
|
[string]$WindowsArch,
|
|
[Parameter()]
|
|
[ValidateSet(10, 11)]
|
|
[int]$WindowsRelease,
|
|
[Parameter()]
|
|
[string]$WindowsVersion
|
|
)
|
|
|
|
# Download and extract the PlatformList.cab
|
|
$PlatformListUrl = 'https://hpia.hpcloud.hp.com/ref/platformList.cab'
|
|
$DriversFolder = "$DriversFolder\$Make"
|
|
$PlatformListCab = "$DriversFolder\platformList.cab"
|
|
$PlatformListXml = "$DriversFolder\PlatformList.xml"
|
|
|
|
if (-not (Test-Path -Path $DriversFolder)) {
|
|
WriteLog "Creating Drivers folder: $DriversFolder"
|
|
New-Item -Path $DriversFolder -ItemType Directory -Force | Out-Null
|
|
WriteLog "Drivers folder created"
|
|
}
|
|
WriteLog "Downloading $PlatformListUrl to $PlatformListCab"
|
|
Start-BitsTransferWithRetry -Source $PlatformListUrl -Destination $PlatformListCab
|
|
WriteLog "Download complete"
|
|
WriteLog "Expanding $PlatformListCab to $PlatformListXml"
|
|
Invoke-Process -FilePath expand.exe -ArgumentList "$PlatformListCab $PlatformListXml" | Out-Null
|
|
WriteLog "Expansion complete"
|
|
|
|
# Parse the PlatformList.xml to find the SystemID based on the ProductName
|
|
[xml]$PlatformListContent = Get-Content -Path $PlatformListXml
|
|
$ProductNodes = $PlatformListContent.ImagePal.Platform | Where-Object { $_.ProductName.'#text' -match $Model }
|
|
|
|
# Create a list of unique ProductName entries
|
|
$ProductNames = @()
|
|
foreach ($node in $ProductNodes) {
|
|
foreach ($productName in $node.ProductName) {
|
|
if ($productName.'#text' -match $Model) {
|
|
$ProductNames += [PSCustomObject]@{
|
|
ProductName = $productName.'#text'
|
|
SystemID = $node.SystemID
|
|
OSReleaseID = $node.OS.OSReleaseIdFileName -replace 'H', 'h'
|
|
IsWindows11 = $node.OS.IsWindows11 -contains 'true'
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if ($ProductNames.Count -gt 1) {
|
|
Write-Output "More than one model found matching '$Model':"
|
|
WriteLog "More than one model found matching '$Model':"
|
|
$ProductNames | ForEach-Object -Begin { $i = 1 } -Process {
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Output "$i. $($_.ProductName)"
|
|
}
|
|
WriteLog "$i. $($_.ProductName)"
|
|
$i++
|
|
}
|
|
$selection = Read-Host "Please select the number corresponding to the correct model"
|
|
WriteLog "User selected model number: $selection"
|
|
if ($selection -match '^\d+$' -and [int]$selection -le $ProductNames.Count) {
|
|
$SelectedProduct = $ProductNames[[int]$selection - 1]
|
|
$ProductName = $SelectedProduct.ProductName
|
|
WriteLog "Selected model: $ProductName"
|
|
$SystemID = $SelectedProduct.SystemID
|
|
WriteLog "SystemID: $SystemID"
|
|
$ValidOSReleaseIDs = $SelectedProduct.OSReleaseID
|
|
WriteLog "Valid OSReleaseIDs: $ValidOSReleaseIDs"
|
|
$IsWindows11 = $SelectedProduct.IsWindows11
|
|
WriteLog "IsWindows11 supported: $IsWindows11"
|
|
}
|
|
else {
|
|
WriteLog "Invalid selection. Exiting."
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Host "Invalid selection. Exiting."
|
|
}
|
|
exit
|
|
}
|
|
}
|
|
elseif ($ProductNames.Count -eq 1) {
|
|
$SelectedProduct = $ProductNames[0]
|
|
$ProductName = $SelectedProduct.ProductName
|
|
WriteLog "Selected model: $ProductName"
|
|
$SystemID = $SelectedProduct.SystemID
|
|
WriteLog "SystemID: $SystemID"
|
|
$ValidOSReleaseIDs = $SelectedProduct.OSReleaseID
|
|
WriteLog "OSReleaseID: $ValidOSReleaseIDs"
|
|
$IsWindows11 = $SelectedProduct.IsWindows11
|
|
WriteLog "IsWindows11: $IsWindows11"
|
|
}
|
|
else {
|
|
WriteLog "No models found matching '$Model'. Exiting."
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Host "No models found matching '$Model'. Exiting."
|
|
}
|
|
exit
|
|
}
|
|
|
|
if (-not $SystemID) {
|
|
WriteLog "SystemID not found for model: $Model Exiting."
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Host "SystemID not found for model: $Model Exiting."
|
|
}
|
|
exit
|
|
}
|
|
|
|
# Validate if WindowsRelease is 11 and there is no IsWindows11 element set to true
|
|
if ($WindowsRelease -eq 11 -and -not $IsWindows11) {
|
|
WriteLog "WindowsRelease is set to 11, but no drivers are available for this Windows release. Please set the -WindowsRelease parameter to 10, or provide your own drivers to the FFUDevelopment\Drivers folder."
|
|
Write-Output "WindowsRelease is set to 11, but no drivers are available for this Windows release. Please set the -WindowsRelease parameter to 10, or provide your own drivers to the FFUDevelopment\Drivers folder."
|
|
exit
|
|
}
|
|
|
|
# Validate WindowsVersion against OSReleaseID
|
|
$OSReleaseIDs = $ValidOSReleaseIDs -split ' '
|
|
$MatchingReleaseID = $OSReleaseIDs | Where-Object { $_ -eq "$WindowsVersion" }
|
|
|
|
if (-not $MatchingReleaseID) {
|
|
Write-Output "The specified WindowsVersion value '$WindowsVersion' is not valid for the selected model. Please select a valid OSReleaseID:"
|
|
$OSReleaseIDs | ForEach-Object -Begin { $i = 1 } -Process {
|
|
Write-Output "$i. $_"
|
|
$i++
|
|
}
|
|
$selection = Read-Host "Please select the number corresponding to the correct OSReleaseID"
|
|
WriteLog "User selected OSReleaseID number: $selection"
|
|
if ($selection -match '^\d+$' -and [int]$selection -le $OSReleaseIDs.Count) {
|
|
$WindowsVersion = $OSReleaseIDs[[int]$selection - 1]
|
|
WriteLog "Selected OSReleaseID: $WindowsVersion"
|
|
}
|
|
else {
|
|
WriteLog "Invalid selection. Exiting."
|
|
exit
|
|
}
|
|
}
|
|
|
|
# Modify WindowsArch for URL
|
|
$Arch = $WindowsArch -replace "^x", ""
|
|
|
|
# Construct the URL to download the driver XML cab for the model
|
|
# The HPcloud reference site is case sensitve so we must convert the Windowsversion to lower 'h' first
|
|
$WindowsVersionHP = $WindowsVersion -replace 'H', 'h'
|
|
$ModelRelease = $SystemID + "_$Arch" + "_$WindowsRelease" + ".0.$WindowsVersionHP"
|
|
$DriverCabUrl = "https://hpia.hpcloud.hp.com/ref/$SystemID/$ModelRelease.cab"
|
|
$DriverCabFile = "$DriversFolder\$ModelRelease.cab"
|
|
$DriverXmlFile = "$DriversFolder\$ModelRelease.xml"
|
|
|
|
if (-not (Test-Url -Url $DriverCabUrl)) {
|
|
WriteLog "HP Driver cab URL is not accessible: $DriverCabUrl Exiting"
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Host "HP Driver cab URL is not accessible: $DriverCabUrl Exiting"
|
|
}
|
|
exit
|
|
}
|
|
|
|
# Download and extract the driver XML cab
|
|
Writelog "Downloading HP Driver cab from $DriverCabUrl to $DriverCabFile"
|
|
Start-BitsTransferWithRetry -Source $DriverCabUrl -Destination $DriverCabFile
|
|
WriteLog "Expanding HP Driver cab to $DriverXmlFile"
|
|
Invoke-Process -FilePath expand.exe -ArgumentList "$DriverCabFile $DriverXmlFile" | Out-Null
|
|
|
|
# Parse the extracted XML file to download individual drivers
|
|
[xml]$DriverXmlContent = Get-Content -Path $DriverXmlFile
|
|
$baseUrl = "https://ftp.hp.com/pub/softpaq/sp"
|
|
|
|
WriteLog "Downloading drivers for $ProductName"
|
|
foreach ($update in $DriverXmlContent.ImagePal.Solutions.UpdateInfo) {
|
|
if ($update.Category -notmatch '^Driver') {
|
|
continue
|
|
}
|
|
|
|
$Name = $update.Name
|
|
# Fix the name for drivers that contain illegal characters for folder name purposes
|
|
$Name = $Name -replace '[\\\/\:\*\?\"\<\>\|]', '_'
|
|
WriteLog "Downloading driver: $Name"
|
|
$Category = $update.Category
|
|
$Category = $Category -replace '[\\\/\:\*\?\"\<\>\|]', '_'
|
|
$Version = $update.Version
|
|
$Version = $Version -replace '[\\\/\:\*\?\"\<\>\|]', '_'
|
|
$DriverUrl = "https://$($update.URL)"
|
|
WriteLog "Driver URL: $DriverUrl"
|
|
$DriverFileName = [System.IO.Path]::GetFileName($DriverUrl)
|
|
$downloadFolder = "$DriversFolder\$ProductName\$Category"
|
|
$DriverFilePath = Join-Path -Path $downloadFolder -ChildPath $DriverFileName
|
|
|
|
if (Test-Path -Path $DriverFilePath) {
|
|
WriteLog "Driver already downloaded: $DriverFilePath, skipping"
|
|
continue
|
|
}
|
|
|
|
if (-not (Test-Path -Path $downloadFolder)) {
|
|
WriteLog "Creating download folder: $downloadFolder"
|
|
New-Item -Path $downloadFolder -ItemType Directory -Force | Out-Null
|
|
WriteLog "Download folder created"
|
|
}
|
|
|
|
# Download the driver with retry
|
|
WriteLog "Downloading driver to: $DriverFilePath"
|
|
Start-BitsTransferWithRetry -Source $DriverUrl -Destination $DriverFilePath
|
|
WriteLog 'Driver downloaded'
|
|
|
|
# Make folder for extraction
|
|
$extractFolder = "$downloadFolder\$Name\$Version\" + $DriverFileName.TrimEnd('.exe')
|
|
Writelog "Creating extraction folder: $extractFolder"
|
|
New-Item -Path $extractFolder -ItemType Directory -Force | Out-Null
|
|
WriteLog 'Extraction folder created'
|
|
|
|
# Extract the driver
|
|
$arguments = "/s /e /f `"$extractFolder`""
|
|
WriteLog "Extracting driver"
|
|
Invoke-Process -FilePath $DriverFilePath -ArgumentList $arguments | Out-Null
|
|
WriteLog "Driver extracted to: $extractFolder"
|
|
|
|
# Delete the .exe driver file after extraction
|
|
Remove-Item -Path $DriverFilePath -Force
|
|
WriteLog "Driver installation file deleted: $DriverFilePath"
|
|
}
|
|
# Clean up the downloaded cab and xml files
|
|
Remove-Item -Path $DriverCabFile, $DriverXmlFile, $PlatformListCab, $PlatformListXml -Force
|
|
WriteLog "Driver cab and xml files deleted"
|
|
}
|
|
function Get-LenovoDrivers {
|
|
param (
|
|
[Parameter()]
|
|
[string]$Model,
|
|
[Parameter()]
|
|
[ValidateSet("x64", "x86", "ARM64")]
|
|
[string]$WindowsArch,
|
|
[Parameter()]
|
|
[ValidateSet(10, 11)]
|
|
[int]$WindowsRelease
|
|
)
|
|
|
|
function Get-LenovoPSREF {
|
|
param (
|
|
[string]$ModelName
|
|
)
|
|
|
|
$url = "https://psref.lenovo.com/api/search/DefinitionFilterAndSearch/Suggest?kw=$ModelName"
|
|
WriteLog "Querying Lenovo PSREF API for model: $ModelName"
|
|
$OriginalVerbosePreference = $VerbosePreference
|
|
$VerbosePreference = 'SilentlyContinue'
|
|
$response = Invoke-WebRequest -Uri $url -UseBasicParsing -Headers $Headers -UserAgent $UserAgent
|
|
$VerbosePreference = $OriginalVerbosePreference
|
|
WriteLog "Complete"
|
|
|
|
$jsonResponse = $response.Content | ConvertFrom-Json
|
|
|
|
$products = @()
|
|
foreach ($item in $jsonResponse.data) {
|
|
$productName = $item.ProductName
|
|
$machineTypes = $item.MachineType -split " / "
|
|
|
|
foreach ($machineType in $machineTypes) {
|
|
if ($machineType -eq $ModelName) {
|
|
WriteLog "Model name entered is a matching machine type"
|
|
$products = @()
|
|
$products += [pscustomobject]@{
|
|
ProductName = $productName
|
|
MachineType = $machineType
|
|
}
|
|
WriteLog "Product Name: $productName Machine Type: $machineType"
|
|
return $products
|
|
}
|
|
$products += [pscustomobject]@{
|
|
ProductName = $productName
|
|
MachineType = $machineType
|
|
}
|
|
}
|
|
}
|
|
|
|
return , $products
|
|
}
|
|
|
|
# Parse the Lenovo PSREF page for the model
|
|
$machineTypes = Get-LenovoPSREF -ModelName $Model
|
|
if ($machineTypes.ProductName.Count -eq 0) {
|
|
WriteLog "No machine types found for model: $Model"
|
|
WriteLog "Enter a valid model or machine type in the -model parameter"
|
|
exit
|
|
}
|
|
elseif ($machineTypes.ProductName.Count -eq 1) {
|
|
$machineType = $machineTypes[0].MachineType
|
|
$model = $machineTypes[0].ProductName
|
|
}
|
|
else {
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Output "Multiple machine types found for model: $Model"
|
|
}
|
|
WriteLog "Multiple machine types found for model: $Model"
|
|
for ($i = 0; $i -lt $machineTypes.ProductName.Count; $i++) {
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Output "$($i + 1). $($machineTypes[$i].ProductName) ($($machineTypes[$i].MachineType))"
|
|
}
|
|
WriteLog "$($i + 1). $($machineTypes[$i].ProductName) ($($machineTypes[$i].MachineType))"
|
|
}
|
|
$selection = Read-Host "Enter the number of the model you want to select"
|
|
$machineType = $machineTypes[$selection - 1].MachineType
|
|
WriteLog "Selected machine type: $machineType"
|
|
$model = $machineTypes[$selection - 1].ProductName
|
|
WriteLog "Selected model: $model"
|
|
}
|
|
|
|
|
|
# Construct the catalog URL based on Windows release and machine type
|
|
$ModelRelease = $machineType + "_Win" + $WindowsRelease
|
|
$CatalogUrl = "https://download.lenovo.com/catalog/$ModelRelease.xml"
|
|
WriteLog "Lenovo Driver catalog URL: $CatalogUrl"
|
|
|
|
if (-not (Test-Url -Url $catalogUrl)) {
|
|
Write-Error "Lenovo Driver catalog URL is not accessible: $catalogUrl"
|
|
WriteLog "Lenovo Driver catalog URL is not accessible: $catalogUrl"
|
|
exit
|
|
}
|
|
|
|
# Create the folder structure for the Lenovo drivers
|
|
$driversFolder = "$DriversFolder\$Make"
|
|
if (-not (Test-Path -Path $DriversFolder)) {
|
|
WriteLog "Creating Drivers folder: $DriversFolder"
|
|
New-Item -Path $DriversFolder -ItemType Directory -Force | Out-Null
|
|
WriteLog "Drivers folder created"
|
|
}
|
|
|
|
# Download and parse the Lenovo catalog XML
|
|
$LenovoCatalogXML = "$DriversFolder\$ModelRelease.xml"
|
|
WriteLog "Downloading $catalogUrl to $LenovoCatalogXML"
|
|
Start-BitsTransferWithRetry -Source $catalogUrl -Destination $LenovoCatalogXML
|
|
WriteLog "Download Complete"
|
|
$xmlContent = [xml](Get-Content -Path $LenovoCatalogXML)
|
|
|
|
WriteLog "Parsing Lenovo catalog XML"
|
|
# Process each package in the catalog
|
|
foreach ($package in $xmlContent.packages.package) {
|
|
$packageUrl = $package.location
|
|
$category = $package.category
|
|
|
|
#If category starts with BIOS, skip the package
|
|
if ($category -like 'BIOS*') {
|
|
continue
|
|
}
|
|
|
|
#If category name is 'Motherboard Devices Backplanes core chipset onboard video PCIe switches', truncate to 'Motherboard Devices' to shorten path
|
|
if ($category -eq 'Motherboard Devices Backplanes core chipset onboard video PCIe switches') {
|
|
$category = 'Motherboard Devices'
|
|
}
|
|
|
|
$packageName = [System.IO.Path]::GetFileName($packageUrl)
|
|
#Remove the filename from the $packageURL
|
|
$baseURL = $packageUrl -replace $packageName, ""
|
|
|
|
# Download the package XML
|
|
$packageXMLPath = "$DriversFolder\$packageName"
|
|
WriteLog "Downloading $category package XML $packageUrl to $packageXMLPath"
|
|
If ((Start-BitsTransferWithRetry -Source $packageUrl -Destination $packageXMLPath) -eq $false) {
|
|
Write-Output "Failed to download $category package XML: $packageXMLPath"
|
|
WriteLog "Failed to download $category package XML: $packageXMLPath"
|
|
continue
|
|
}
|
|
|
|
# Load the package XML content
|
|
$packageXmlContent = [xml](Get-Content -Path $packageXMLPath)
|
|
$packageType = $packageXmlContent.Package.PackageType.type
|
|
$packageTitle = $packageXmlContent.Package.title.InnerText
|
|
|
|
# Fix the name for drivers that contain illegal characters for folder name purposes
|
|
$packageTitle = $packageTitle -replace '[\\\/\:\*\?\"\<\>\|]', '_'
|
|
|
|
# If ' - ' is in the package title, truncate the title to the first part of the string.
|
|
$packageTitle = $packageTitle -replace ' - .*', ''
|
|
|
|
#Check if packagetype = 2. If packagetype is not 2, skip the package. $packageType is a System.Xml.XmlElement.
|
|
#This filters out Firmware, BIOS, and other non-INF drivers
|
|
if ($packageType -ne 2) {
|
|
Remove-Item -Path $packageXMLPath -Force
|
|
continue
|
|
}
|
|
|
|
# Extract the driver file name and the extract command
|
|
$driverFileName = $packageXmlContent.Package.Files.Installer.File.Name
|
|
$extractCommand = $packageXmlContent.Package.ExtractCommand
|
|
|
|
#if extract command is empty/missing, skip the package
|
|
if (!($extractCommand)) {
|
|
Remove-Item -Path $packageXMLPath -Force
|
|
continue
|
|
}
|
|
|
|
# Create the download URL and folder structure
|
|
$driverUrl = $baseUrl + $driverFileName
|
|
$downloadFolder = "$DriversFolder\$Model\$Category\$packageTitle"
|
|
$driverFilePath = Join-Path -Path $downloadFolder -ChildPath $driverFileName
|
|
|
|
# Check if file has already been downloaded
|
|
if (Test-Path -Path $driverFilePath) {
|
|
Write-Output "Driver already downloaded: $driverFilePath skipping"
|
|
WriteLog "Driver already downloaded: $driverFilePath skipping"
|
|
continue
|
|
}
|
|
|
|
if (-not (Test-Path -Path $downloadFolder)) {
|
|
WriteLog "Creating download folder: $downloadFolder"
|
|
New-Item -Path $downloadFolder -ItemType Directory -Force | Out-Null
|
|
WriteLog "Download folder created"
|
|
}
|
|
|
|
# Download the driver with retry
|
|
WriteLog "Downloading driver: $driverUrl to $driverFilePath"
|
|
Start-BitsTransferWithRetry -Source $driverUrl -Destination $driverFilePath
|
|
WriteLog "Driver downloaded"
|
|
|
|
# Make folder for extraction
|
|
$extractFolder = $downloadFolder + "\" + $driverFileName.TrimEnd($driverFileName[-4..-1])
|
|
WriteLog "Creating extract folder: $extractFolder"
|
|
New-Item -Path $extractFolder -ItemType Directory -Force | Out-Null
|
|
WriteLog "Extract folder created"
|
|
|
|
# Modify the extract command
|
|
$modifiedExtractCommand = $extractCommand -replace '%PACKAGEPATH%', "`"$extractFolder`""
|
|
|
|
# Extract the driver
|
|
# Start-Process -FilePath $driverFilePath -ArgumentList $modifiedExtractCommand -Wait -NoNewWindow
|
|
WriteLog "Extracting driver: $driverFilePath to $extractFolder"
|
|
Invoke-Process -FilePath $driverFilePath -ArgumentList $modifiedExtractCommand | Out-Null
|
|
WriteLog "Driver extracted"
|
|
|
|
# Delete the .exe driver file after extraction
|
|
WriteLog "Deleting driver installation file: $driverFilePath"
|
|
Remove-Item -Path $driverFilePath -Force
|
|
WriteLog "Driver installation file deleted: $driverFilePath"
|
|
|
|
# Delete the package XML file after extraction
|
|
WriteLog "Deleting package XML file: $packageXMLPath"
|
|
Remove-Item -Path $packageXMLPath -Force
|
|
WriteLog "Package XML file deleted"
|
|
}
|
|
|
|
#Delete the catalog XML file after processing
|
|
WriteLog "Deleting catalog XML file: $LenovoCatalogXML"
|
|
Remove-Item -Path $LenovoCatalogXML -Force
|
|
WriteLog "Catalog XML file deleted"
|
|
}
|
|
|
|
function Get-DellDrivers {
|
|
param (
|
|
[Parameter(Mandatory = $true)]
|
|
[string]$Model,
|
|
[Parameter(Mandatory = $true)]
|
|
[ValidateSet("x64", "x86", "ARM64")]
|
|
[string]$WindowsArch,
|
|
[Parameter(Mandatory = $true)]
|
|
[int]$WindowsRelease
|
|
)
|
|
|
|
if (-not (Test-Path -Path $DriversFolder)) {
|
|
WriteLog "Creating Drivers folder: $DriversFolder"
|
|
New-Item -Path $DriversFolder -ItemType Directory -Force | Out-Null
|
|
WriteLog "Drivers folder created"
|
|
}
|
|
|
|
$DriversFolder = "$DriversFolder\$Make"
|
|
WriteLog "Creating Dell Drivers folder: $DriversFolder"
|
|
New-Item -Path $DriversFolder -ItemType Directory -Force | Out-Null
|
|
WriteLog "Dell Drivers folder created"
|
|
|
|
#CatalogPC.cab is the catalog for Windows client PCs, Catalog.cab is the catalog for Windows Server
|
|
if ($WindowsRelease -le 11) {
|
|
$catalogUrl = "http://downloads.dell.com/catalog/CatalogPC.cab"
|
|
$DellCabFile = "$DriversFolder\CatalogPC.cab"
|
|
$DellCatalogXML = "$DriversFolder\CatalogPC.XML"
|
|
}
|
|
else {
|
|
$catalogUrl = "https://downloads.dell.com/catalog/Catalog.cab"
|
|
$DellCabFile = "$DriversFolder\Catalog.cab"
|
|
$DellCatalogXML = "$DriversFolder\Catalog.xml"
|
|
}
|
|
|
|
if (-not (Test-Url -Url $catalogUrl)) {
|
|
WriteLog "Dell Catalog cab URL is not accessible: $catalogUrl Exiting"
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Host "Dell Catalog cab URL is not accessible: $catalogUrl Exiting"
|
|
}
|
|
exit
|
|
}
|
|
|
|
WriteLog "Downloading Dell Catalog cab file: $catalogUrl to $DellCabFile"
|
|
Start-BitsTransferWithRetry -Source $catalogUrl -Destination $DellCabFile
|
|
WriteLog "Dell Catalog cab file downloaded"
|
|
|
|
WriteLog "Extracting Dell Catalog cab file to $DellCatalogXML"
|
|
Invoke-Process -FilePath Expand.exe -ArgumentList "$DellCabFile $DellCatalogXML" | Out-Null
|
|
WriteLog "Dell Catalog cab file extracted"
|
|
|
|
$xmlContent = [xml](Get-Content -Path $DellCatalogXML)
|
|
$baseLocation = "https://" + $xmlContent.manifest.baseLocation + "/"
|
|
$latestDrivers = @{}
|
|
|
|
$softwareComponents = $xmlContent.Manifest.SoftwareComponent | Where-Object { $_.ComponentType.value -eq "DRVR" }
|
|
foreach ($component in $softwareComponents) {
|
|
$models = $component.SupportedSystems.Brand.Model
|
|
foreach ($item in $models) {
|
|
if ($item.Display.'#cdata-section' -match $Model) {
|
|
|
|
if ($WindowsRelease -le 11) {
|
|
$validOS = $component.SupportedOperatingSystems.OperatingSystem | Where-Object { $_.osArch -eq $WindowsArch }
|
|
}
|
|
elseif ($WindowsRelease -eq 2016) {
|
|
$validOS = $component.SupportedOperatingSystems.OperatingSystem | Where-Object { ($_.osArch -eq $WindowsArch) -and ($_.osCode -match "W14") }
|
|
}
|
|
elseif ($WindowsRelease -eq 2019) {
|
|
$validOS = $component.SupportedOperatingSystems.OperatingSystem | Where-Object { ($_.osArch -eq $WindowsArch) -and ($_.osCode -match "W19") }
|
|
}
|
|
elseif ($WindowsRelease -eq 2022) {
|
|
$validOS = $component.SupportedOperatingSystems.OperatingSystem | Where-Object { ($_.osArch -eq $WindowsArch) -and ($_.osCode -match "W22") }
|
|
}
|
|
elseif ($WindowsRelease -eq 2025) {
|
|
$validOS = $component.SupportedOperatingSystems.OperatingSystem | Where-Object { ($_.osArch -eq $WindowsArch) -and ($_.osCode -match "W25") }
|
|
}
|
|
else {
|
|
$validOS = $component.SupportedOperatingSystems.OperatingSystem | Where-Object { ($_.osArch -eq $WindowsArch) -and ($_.osCode -match "W22") }
|
|
}
|
|
|
|
if ($validOS) {
|
|
$driverPath = $component.path
|
|
$downloadUrl = $baseLocation + $driverPath
|
|
$driverFileName = [System.IO.Path]::GetFileName($driverPath)
|
|
$name = $component.Name.Display.'#cdata-section'
|
|
$name = $name -replace '[\\\/\:\*\?\"\<\>\| ]', '_'
|
|
$name = $name -replace '[\,]', '-'
|
|
$category = $component.Category.Display.'#cdata-section'
|
|
$category = $category -replace '[\\\/\:\*\?\"\<\>\| ]', '_'
|
|
$version = [version]$component.vendorVersion
|
|
$namePrefix = ($name -split '-')[0]
|
|
|
|
# Use hash table to store the latest driver for each category to prevent downloading older driver versions
|
|
if ($latestDrivers[$category]) {
|
|
if ($latestDrivers[$category][$namePrefix]) {
|
|
if ($latestDrivers[$category][$namePrefix].Version -lt $version) {
|
|
$latestDrivers[$category][$namePrefix] = [PSCustomObject]@{
|
|
Name = $name;
|
|
DownloadUrl = $downloadUrl;
|
|
DriverFileName = $driverFileName;
|
|
Version = $version;
|
|
Category = $category
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
$latestDrivers[$category][$namePrefix] = [PSCustomObject]@{
|
|
Name = $name;
|
|
DownloadUrl = $downloadUrl;
|
|
DriverFileName = $driverFileName;
|
|
Version = $version;
|
|
Category = $category
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
$latestDrivers[$category] = @{}
|
|
$latestDrivers[$category][$namePrefix] = [PSCustomObject]@{
|
|
Name = $name;
|
|
DownloadUrl = $downloadUrl;
|
|
DriverFileName = $driverFileName;
|
|
Version = $version;
|
|
Category = $category
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
foreach ($category in $latestDrivers.Keys) {
|
|
foreach ($driver in $latestDrivers[$category].Values) {
|
|
$downloadFolder = "$DriversFolder\$Model\$($driver.Category)"
|
|
$driverFilePath = Join-Path -Path $downloadFolder -ChildPath $driver.DriverFileName
|
|
|
|
if (Test-Path -Path $driverFilePath) {
|
|
WriteLog "Driver already downloaded: $driverFilePath skipping"
|
|
continue
|
|
}
|
|
|
|
WriteLog "Downloading driver: $($driver.Name)"
|
|
if (-not (Test-Path -Path $downloadFolder)) {
|
|
WriteLog "Creating download folder: $downloadFolder"
|
|
New-Item -Path $downloadFolder -ItemType Directory -Force | Out-Null
|
|
WriteLog "Download folder created"
|
|
}
|
|
|
|
WriteLog "Downloading driver: $($driver.DownloadUrl) to $driverFilePath"
|
|
try {
|
|
Start-BitsTransferWithRetry -Source $driver.DownloadUrl -Destination $driverFilePath
|
|
WriteLog "Driver downloaded"
|
|
}
|
|
catch {
|
|
WriteLog "Failed to download driver: $($driver.DownloadUrl) to $driverFilePath"
|
|
continue
|
|
}
|
|
|
|
|
|
$extractFolder = $downloadFolder + "\" + $driver.DriverFileName.TrimEnd($driver.DriverFileName[-4..-1])
|
|
# WriteLog "Creating extraction folder: $extractFolder"
|
|
# New-Item -Path $extractFolder -ItemType Directory -Force | Out-Null
|
|
# WriteLog "Extraction folder created"
|
|
|
|
# $arguments = "/s /e /f `"$extractFolder`""
|
|
$arguments = "/s /drivers=`"$extractFolder`""
|
|
WriteLog "Extracting driver: $driverFilePath $arguments"
|
|
try {
|
|
#If Category is Chipset, must add -wait $false to the Invoke-Process command line to prevent the script from hanging on the Intel chipset driver which leaves a Window open
|
|
if ($driver.Category -eq "Chipset") {
|
|
$process = Invoke-Process -FilePath $driverFilePath -ArgumentList $arguments -Wait $false
|
|
|
|
#Wait 5 seconds to allow for the extraction process to finish
|
|
Start-Sleep -Seconds 5
|
|
|
|
$childProcesses = Get-ChildProcesses $process.Id
|
|
|
|
# Find and stop the last created child process
|
|
if ($childProcesses) {
|
|
$latestProcess = $childProcesses | Sort-Object CreationDate -Descending | Select-Object -First 1
|
|
Stop-Process -Id $latestProcess.ProcessId -Force
|
|
# Sleep 1 second to let process finish exiting so its installer can be removed
|
|
Start-Sleep -Seconds 1
|
|
}
|
|
#If Category is Network and $isServer is $false, must add -wait $false to the Invoke-Process command line to prevent the script from hanging on the Intel network driver which leaves a Window open
|
|
}
|
|
elseif ($driver.Category -eq "Network" -and $isServer -eq $false) {
|
|
|
|
$process = Invoke-Process -FilePath $driverFilePath -ArgumentList $arguments -Wait $false
|
|
|
|
#Sometimes the network drivers will extract on client OS, wait 5 seconds and check if the process is still running
|
|
Start-Sleep -Seconds 5
|
|
if ($process.HasExited -eq $false) {
|
|
$childProcesses = Get-ChildProcesses $process.Id
|
|
|
|
# Find and stop the last created child process
|
|
if ($childProcesses) {
|
|
$latestProcess = $childProcesses | Sort-Object CreationDate -Descending | Select-Object -First 1
|
|
Stop-Process -Id $latestProcess.ProcessId -Force
|
|
#Move on to the next driver and skip this one - it won't extract on a client OS even with /s /e switches
|
|
continue
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
Invoke-Process -FilePath $driverFilePath -ArgumentList $arguments | Out-Null
|
|
}
|
|
# If $extractFolder is empty, try alternative extraction method
|
|
if (!(Get-ChildItem -Path $extractFolder -Recurse | Where-Object { -not $_.PSIsContainer })) {
|
|
WriteLog 'Extraction with /drivers= switch failed. Removing folder and retrying with /s /e switches'
|
|
Remove-Item -Path $extractFolder -Force -Recurse -ErrorAction SilentlyContinue
|
|
$arguments = "/s /e=`"$extractFolder`""
|
|
WriteLog "Extracting driver: $driverFilePath $arguments"
|
|
Invoke-Process -FilePath $driverFilePath -ArgumentList $arguments | Out-Null
|
|
}
|
|
}
|
|
catch {
|
|
WriteLog 'Extraction with /drivers= switch failed. Retrying with /s /e switches'
|
|
$arguments = "/s /e=`"$extractFolder`""
|
|
WriteLog "Extracting driver: $driverFilePath $arguments"
|
|
Invoke-Process -FilePath $driverFilePath -ArgumentList $arguments | Out-Null
|
|
}
|
|
WriteLog "Driver extracted"
|
|
|
|
WriteLog "Deleting driver file: $driverFilePath"
|
|
Remove-Item -Path $driverFilePath -Force
|
|
WriteLog "Driver file deleted"
|
|
}
|
|
}
|
|
}
|
|
function Get-ADKURL {
|
|
param (
|
|
[ValidateSet("Windows ADK", "WinPE add-on")]
|
|
[string]$ADKOption
|
|
)
|
|
|
|
# Define base pattern for URL scraping
|
|
$basePattern = '<li><a href="(https://[^"]+)" data-linktype="external">Download the '
|
|
|
|
# Define specific URL patterns based on ADK options
|
|
$ADKUrlPattern = @{
|
|
"Windows ADK" = $basePattern + "Windows ADK"
|
|
"WinPE add-on" = $basePattern + "Windows PE add-on for the Windows ADK"
|
|
}[$ADKOption]
|
|
|
|
try {
|
|
# Retrieve content of Microsoft documentation page
|
|
$OriginalVerbosePreference = $VerbosePreference
|
|
$VerbosePreference = 'SilentlyContinue'
|
|
$ADKWebPage = Invoke-RestMethod "https://learn.microsoft.com/en-us/windows-hardware/get-started/adk-install" -Headers $Headers -UserAgent $UserAgent
|
|
$VerbosePreference = $OriginalVerbosePreference
|
|
|
|
# Extract download URL based on specified pattern
|
|
$ADKMatch = [regex]::Match($ADKWebPage, $ADKUrlPattern)
|
|
|
|
if (-not $ADKMatch.Success) {
|
|
WriteLog "Failed to retrieve ADK download URL. Pattern match failed."
|
|
return
|
|
}
|
|
|
|
# Extract FWlink from the matched pattern
|
|
$ADKFWLink = $ADKMatch.Groups[1].Value
|
|
|
|
if ($null -eq $ADKFWLink) {
|
|
WriteLog "FWLink for $ADKOption not found."
|
|
return
|
|
}
|
|
|
|
# Retrieve headers of the FWlink URL
|
|
$OriginalVerbosePreference = $VerbosePreference
|
|
$VerbosePreference = 'SilentlyContinue'
|
|
$FWLinkRequest = Invoke-WebRequest -Uri $ADKFWLink -Method Head -MaximumRedirection 0 -ErrorAction SilentlyContinue
|
|
$VerbosePreference = $OriginalVerbosePreference
|
|
|
|
if ($FWLinkRequest.StatusCode -ne 302) {
|
|
WriteLog "Failed to retrieve ADK download URL. Unexpected status code: $($FWLinkRequest.StatusCode)"
|
|
return
|
|
}
|
|
|
|
# Get the ADK link redirected to by the FWlink
|
|
$ADKUrl = $FWLinkRequest.Headers.Location
|
|
return $ADKUrl
|
|
}
|
|
catch {
|
|
WriteLog $_
|
|
Write-Error "Error occurred while retrieving ADK download URL"
|
|
throw $_
|
|
}
|
|
}
|
|
function Install-ADK {
|
|
param (
|
|
[ValidateSet("Windows ADK", "WinPE add-on")]
|
|
[string]$ADKOption
|
|
)
|
|
|
|
try {
|
|
$ADKUrl = Get-ADKURL -ADKOption $ADKOption
|
|
|
|
if ($null -eq $ADKUrl) {
|
|
throw "Failed to retrieve URL for $ADKOption. Please manually install it."
|
|
}
|
|
|
|
# Select the installer based on the ADK option specified
|
|
$installer = @{
|
|
"Windows ADK" = "adksetup.exe"
|
|
"WinPE add-on" = "adkwinpesetup.exe"
|
|
}[$ADKOption]
|
|
|
|
# Select the feature based on the ADK option specified
|
|
$feature = @{
|
|
"Windows ADK" = "OptionId.DeploymentTools"
|
|
"WinPE add-on" = "OptionId.WindowsPreinstallationEnvironment"
|
|
}[$ADKOption]
|
|
|
|
$installerLocation = Join-Path $env:TEMP $installer
|
|
|
|
WriteLog "Downloading $ADKOption from $ADKUrl to $installerLocation"
|
|
Start-BitsTransferWithRetry -Source $ADKUrl -Destination $installerLocation -ErrorAction Stop
|
|
WriteLog "$ADKOption downloaded to $installerLocation"
|
|
|
|
WriteLog "Installing $ADKOption with $feature enabled"
|
|
Invoke-Process $installerLocation "/quiet /installpath ""%ProgramFiles(x86)%\Windows Kits\10"" /features $feature" | Out-Null
|
|
|
|
WriteLog "$ADKOption installation completed."
|
|
WriteLog "Removing $installer from $installerLocation"
|
|
# Clean up downloaded installation file
|
|
Remove-Item -Path $installerLocation -Force -ErrorAction SilentlyContinue
|
|
}
|
|
catch {
|
|
WriteLog $_
|
|
Write-Error "Error occurred while installing $ADKOption. Please manually install it."
|
|
throw $_
|
|
}
|
|
}
|
|
function Get-InstalledProgramRegKey {
|
|
param (
|
|
[string]$DisplayName
|
|
)
|
|
|
|
$uninstallRegPath = "HKLM:\SOFTWARE\WOW6432Node\Microsoft\Windows\CurrentVersion\Uninstall"
|
|
$uninstallRegKeys = Get-ChildItem -Path $uninstallRegPath -Recurse
|
|
|
|
foreach ($regKey in $uninstallRegKeys) {
|
|
try {
|
|
$regValue = $regKey.GetValue("DisplayName")
|
|
if ($regValue -eq $DisplayName) {
|
|
return $regKey
|
|
}
|
|
}
|
|
catch {
|
|
WriteLog $_
|
|
throw "Error retrieving installed program info for $DisplayName : $_"
|
|
}
|
|
}
|
|
}
|
|
|
|
function Uninstall-ADK {
|
|
param (
|
|
[ValidateSet("Windows ADK", "WinPE add-on")]
|
|
[string]$ADKOption
|
|
)
|
|
|
|
# Match name as it appears in the registry
|
|
$displayName = switch ($ADKOption) {
|
|
"Windows ADK" { "Windows Assessment and Deployment Kit" }
|
|
"WinPE add-on" { "Windows Assessment and Deployment Kit Windows Preinstallation Environment Add-ons" }
|
|
}
|
|
|
|
try {
|
|
$adkRegKey = Get-InstalledProgramRegKey -DisplayName $displayName
|
|
|
|
if (-not $adkRegKey) {
|
|
WriteLog "$ADKOption is not installed."
|
|
return
|
|
}
|
|
|
|
$adkBundleCachePath = $adkRegKey.GetValue("BundleCachePath")
|
|
WriteLog "Uninstalling $ADKOption..."
|
|
Invoke-Process $adkBundleCachePath "/uninstall /quiet" | Out-Null
|
|
WriteLog "$ADKOption uninstalled successfully."
|
|
}
|
|
catch {
|
|
WriteLog $_
|
|
Write-Error "Error occurred while uninstalling $ADKOption. Please manually uninstall it."
|
|
throw $_
|
|
}
|
|
}
|
|
|
|
function Confirm-ADKVersionIsLatest {
|
|
param (
|
|
[ValidateSet("Windows ADK", "WinPE add-on")]
|
|
[string]$ADKOption
|
|
)
|
|
|
|
$displayName = switch ($ADKOption) {
|
|
"Windows ADK" { "Windows Assessment and Deployment Kit" }
|
|
"WinPE add-on" { "Windows Assessment and Deployment Kit Windows Preinstallation Environment Add-ons" }
|
|
}
|
|
|
|
try {
|
|
$adkRegKey = Get-InstalledProgramRegKey -DisplayName $displayName
|
|
|
|
if (-not $adkRegKey) {
|
|
return $false
|
|
}
|
|
|
|
$installedADKVersion = $adkRegKey.GetValue("DisplayVersion")
|
|
|
|
# Retrieve content of Microsoft documentation page
|
|
$adkWebPage = Invoke-RestMethod "https://learn.microsoft.com/en-us/windows-hardware/get-started/adk-install" -Headers $Headers -UserAgent $UserAgent
|
|
# Specify regex pattern for ADK version
|
|
$adkVersionPattern = 'ADK\s+(\d+(\.\d+)+)'
|
|
# Check for regex pattern match
|
|
$adkVersionMatch = [regex]::Match($adkWebPage, $adkVersionPattern)
|
|
|
|
if (-not $adkVersionMatch.Success) {
|
|
WriteLog "Failed to retrieve latest ADK version from web page."
|
|
return $false
|
|
}
|
|
|
|
# Extract ADK version from the matched pattern
|
|
$latestADKVersion = $adkVersionMatch.Groups[1].Value
|
|
|
|
if ($installedADKVersion -eq $latestADKVersion) {
|
|
WriteLog "Installed $ADKOption version $installedADKVersion is the latest."
|
|
return $true
|
|
}
|
|
else {
|
|
WriteLog "Installed $ADKOption version $installedADKVersion is not the latest ($latestADKVersion)"
|
|
return $false
|
|
}
|
|
}
|
|
catch {
|
|
WriteLog "An error occurred while confirming the ADK version: $_"
|
|
return $false
|
|
}
|
|
}
|
|
|
|
function Get-ADK {
|
|
# Check if latest ADK and WinPE add-on are installed
|
|
if ($UpdateADK) {
|
|
WriteLog "Checking if latest ADK and WinPE add-on are installed"
|
|
$latestADKInstalled = Confirm-ADKVersionIsLatest -ADKOption "Windows ADK"
|
|
$latestWinPEInstalled = Confirm-ADKVersionIsLatest -ADKOption "WinPE add-on"
|
|
|
|
# Uninstall older versions and install latest versions if necessary
|
|
if (-not $latestADKInstalled) {
|
|
Uninstall-ADK -ADKOption "Windows ADK"
|
|
Install-ADK -ADKOption "Windows ADK"
|
|
}
|
|
|
|
if (-not $latestWinPEInstalled) {
|
|
Uninstall-ADK -ADKOption "WinPE add-on"
|
|
Install-ADK -ADKOption "WinPE add-on"
|
|
}
|
|
}
|
|
|
|
# Define registry path
|
|
$adkPathKey = "HKLM:\SOFTWARE\WOW6432Node\Microsoft\Windows Kits\Installed Roots"
|
|
$adkPathName = "KitsRoot10"
|
|
|
|
# Check if ADK installation path exists in registry
|
|
$adkPathNameExists = (Get-ItemProperty -Path $adkPathKey -Name $adkPathName -ErrorAction SilentlyContinue)
|
|
|
|
if ($adkPathNameExists) {
|
|
# Get the ADK installation path
|
|
WriteLog 'Get ADK Path'
|
|
$adkPath = (Get-ItemProperty -Path $adkPathKey -Name $adkPathName).$adkPathName
|
|
WriteLog "ADK located at $adkPath"
|
|
}
|
|
else {
|
|
throw "Windows ADK installation path could not be found."
|
|
}
|
|
|
|
# If ADK was already installed, then check if the Windows Deployment Tools feature is also installed
|
|
$deploymentToolsRegKey = Get-InstalledProgramRegKey -DisplayName "Windows Deployment Tools"
|
|
|
|
if (-not $deploymentToolsRegKey) {
|
|
WriteLog "ADK is installed, but the Windows Deployment Tools feature is not installed."
|
|
$adkRegKey = Get-InstalledProgramRegKey -DisplayName "Windows Assessment and Deployment Kit"
|
|
$adkBundleCachePath = $adkRegKey.GetValue("BundleCachePath")
|
|
if ($adkBundleCachePath) {
|
|
WriteLog "Installing Windows Deployment Tools..."
|
|
$adkInstallPath = $adkPath.TrimEnd('\')
|
|
Invoke-Process $adkBundleCachePath "/quiet /installpath ""$adkInstallPath"" /features OptionId.DeploymentTools" | Out-Null
|
|
WriteLog "Windows Deployment Tools installed successfully."
|
|
}
|
|
else {
|
|
throw "Failed to retrieve path to adksetup.exe to install the Windows Deployment Tools. Please manually install it."
|
|
}
|
|
}
|
|
return $adkPath
|
|
}
|
|
function Get-WindowsESD {
|
|
param(
|
|
[Parameter(Mandatory = $false)]
|
|
[ValidateSet(10, 11)]
|
|
[int]$WindowsRelease,
|
|
|
|
[Parameter(Mandatory = $false)]
|
|
[ValidateSet('x86', 'x64', 'ARM64')]
|
|
[string]$WindowsArch,
|
|
|
|
[Parameter(Mandatory = $false)]
|
|
[string]$WindowsLang,
|
|
|
|
[Parameter(Mandatory = $false)]
|
|
[ValidateSet('consumer', 'business')]
|
|
[string]$MediaType
|
|
)
|
|
WriteLog "Downloading Windows $WindowsRelease ESD file"
|
|
WriteLog "Windows Architecture: $WindowsArch"
|
|
WriteLog "Windows Language: $WindowsLang"
|
|
WriteLog "Windows Media Type: $MediaType"
|
|
|
|
# Select cab file URL based on Windows Release
|
|
$cabFileUrl = if ($WindowsRelease -eq 10) {
|
|
'https://go.microsoft.com/fwlink/?LinkId=841361'
|
|
}
|
|
elseif ($WindowsRelease -eq 11) {
|
|
'https://go.microsoft.com/fwlink/?LinkId=2156292'
|
|
}
|
|
else {
|
|
throw "Downloading Windows $WindowsRelease is not supported. Please use the -ISOPath parameter to specify the path to the Windows $WindowsRelease ISO file."
|
|
}
|
|
|
|
# Download cab file
|
|
WriteLog "Downloading Cab file"
|
|
$cabFilePath = Join-Path $PSScriptRoot "tempCabFile.cab"
|
|
$OriginalVerbosePreference = $VerbosePreference
|
|
$VerbosePreference = 'SilentlyContinue'
|
|
Invoke-WebRequest -Uri $cabFileUrl -OutFile $cabFilePath -Headers $Headers -UserAgent $UserAgent
|
|
$VerbosePreference = $OriginalVerbosePreference
|
|
WriteLog "Download succeeded"
|
|
|
|
# Extract XML from cab file
|
|
WriteLog "Extracting Products XML from cab"
|
|
$xmlFilePath = Join-Path $PSScriptRoot "products.xml"
|
|
Invoke-Process Expand "-F:*.xml $cabFilePath $xmlFilePath" | Out-Null
|
|
WriteLog "Products XML extracted"
|
|
|
|
# Load XML content
|
|
[xml]$xmlContent = Get-Content -Path $xmlFilePath
|
|
|
|
# Define the client type to look for in the FilePath
|
|
$clientType = if ($MediaType -eq 'consumer') { 'CLIENTCONSUMER' } else { 'CLIENTBUSINESS' }
|
|
|
|
# Find FilePath values based on WindowsArch, WindowsLang, and MediaType
|
|
foreach ($file in $xmlContent.MCT.Catalogs.Catalog.PublishedMedia.Files.File) {
|
|
if ($file.Architecture -eq $WindowsArch -and $file.LanguageCode -eq $WindowsLang -and $file.FilePath -like "*$clientType*") {
|
|
$esdFilePath = Join-Path $PSScriptRoot (Split-Path $file.FilePath -Leaf)
|
|
#Download if ESD file doesn't already exist
|
|
If (-not (Test-Path $esdFilePath)) {
|
|
#Required to fix slow downloads
|
|
$ProgressPreference = 'SilentlyContinue'
|
|
WriteLog "Downloading $($file.filePath) to $esdFIlePath"
|
|
$OriginalVerbosePreference = $VerbosePreference
|
|
$VerbosePreference = 'SilentlyContinue'
|
|
Invoke-WebRequest -Uri $file.FilePath -OutFile $esdFilePath -Headers $Headers -UserAgent $UserAgent
|
|
$VerbosePreference = $OriginalVerbosePreference
|
|
WriteLog "Download succeeded"
|
|
#Set back to show progress
|
|
$ProgressPreference = 'Continue'
|
|
WriteLog "Cleanup cab and xml file"
|
|
Remove-Item -Path $cabFilePath -Force
|
|
Remove-Item -Path $xmlFilePath -Force
|
|
WriteLog "Cleanup done"
|
|
}
|
|
return $esdFilePath
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
function Get-ODTURL {
|
|
try {
|
|
[String]$ODTPage = Invoke-WebRequest 'https://www.microsoft.com/en-us/download/details.aspx?id=49117' -Headers $Headers -UserAgent $UserAgent -ErrorAction Stop
|
|
|
|
# Extract JSON data from the webpage
|
|
if ($ODTPage -match '<script>window\.__DLCDetails__=(.*?)<\/script>') {
|
|
# Parse JSON content
|
|
$jsonContent = $matches[1] | ConvertFrom-Json
|
|
$ODTURL = $jsonContent.dlcDetailsView.downloadFile[0].url
|
|
|
|
if ($ODTURL) {
|
|
return $ODTURL
|
|
}
|
|
else {
|
|
WriteLog 'Cannot find the ODT download URL in the JSON content'
|
|
throw 'Cannot find the ODT download URL in the JSON content'
|
|
}
|
|
}
|
|
else {
|
|
WriteLog 'Failed to extract JSON content from the ODT webpage'
|
|
throw 'Failed to extract JSON content from the ODT webpage'
|
|
}
|
|
}
|
|
catch {
|
|
WriteLog $_.Exception.Message
|
|
throw 'An error occurred while retrieving the ODT URL.'
|
|
}
|
|
}
|
|
|
|
function Get-Office {
|
|
#Download ODT
|
|
$ODTUrl = Get-ODTURL
|
|
$ODTInstallFile = "$FFUDevelopmentPath\odtsetup.exe"
|
|
WriteLog "Downloading Office Deployment Toolkit from $ODTUrl to $ODTInstallFile"
|
|
$OriginalVerbosePreference = $VerbosePreference
|
|
$VerbosePreference = 'SilentlyContinue'
|
|
Invoke-WebRequest -Uri $ODTUrl -OutFile $ODTInstallFile -Headers $Headers -UserAgent $UserAgent
|
|
$VerbosePreference = $OriginalVerbosePreference
|
|
|
|
# Extract ODT
|
|
WriteLog "Extracting ODT to $OfficePath"
|
|
Invoke-Process $ODTInstallFile "/extract:$OfficePath /quiet" | Out-Null
|
|
|
|
# Run setup.exe with config.xml and modify xml file to download to $OfficePath
|
|
$xmlContent = [xml](Get-Content $OfficeDownloadXML)
|
|
$xmlContent.Configuration.Add.SourcePath = $OfficePath
|
|
$xmlContent.Save($OfficeDownloadXML)
|
|
WriteLog "Downloading M365 Apps/Office to $OfficePath"
|
|
Invoke-Process $OfficePath\setup.exe "/download $OfficeDownloadXML" | Out-Null
|
|
|
|
WriteLog "Cleaning up ODT default config files"
|
|
#Clean up default configuration files
|
|
Remove-Item -Path "$OfficePath\configuration*" -Force
|
|
|
|
#Create Install-Office.ps1 in $orchestrationpath
|
|
WriteLog "Creating $orchestrationpath\Install-Office.ps1"
|
|
$installOfficePath = Join-Path -Path $orchestrationpath -ChildPath "Install-Office.ps1"
|
|
# Create the Install-Office.ps1 file
|
|
$installOfficeCommand = "& d:\Office\setup.exe /configure d:\office\$OfficeInstallXML"
|
|
Set-Content -Path $installOfficePath -Value $installOfficeCommand -Force
|
|
WriteLog "Install-Office.ps1 created successfully at $installOfficePath"
|
|
|
|
#Remove the ODT setup file
|
|
WriteLog "Removing ODT setup file"
|
|
Remove-Item -Path $ODTInstallFile -Force
|
|
WriteLog "ODT setup file removed"
|
|
}
|
|
|
|
function Get-KBLink {
|
|
param(
|
|
[Parameter(Mandatory)]
|
|
[string]$Name
|
|
)
|
|
$OriginalVerbosePreference = $VerbosePreference
|
|
$VerbosePreference = 'SilentlyContinue'
|
|
$results = Invoke-WebRequest -Uri "http://www.catalog.update.microsoft.com/Search.aspx?q=$Name" -Headers $Headers -UserAgent $UserAgent
|
|
$VerbosePreference = $OriginalVerbosePreference
|
|
|
|
# Extract the first KB article ID from the HTML content and store it globally
|
|
# Edge and Defender do not have KB article IDs
|
|
if ($Name -notmatch 'Defender|Edge') {
|
|
if ($results.Content -match '>\s*([^\(<]+)\(KB(\d+)\)\s*<') {
|
|
$kbArticleID = "KB$($matches[2])"
|
|
$global:LastKBArticleID = $kbArticleID
|
|
WriteLog "Found KB article ID: $kbArticleID"
|
|
}
|
|
else {
|
|
WriteLog "No KB article ID found in search results."
|
|
$global:LastKBArticleID = $null
|
|
}
|
|
}
|
|
|
|
$kbids = $results.InputFields |
|
|
Where-Object { $_.type -eq 'Button' -and $_.Value -eq 'Download' } |
|
|
Select-Object -ExpandProperty ID
|
|
|
|
if (-not $kbids) {
|
|
Write-Warning -Message "No results found for $Name"
|
|
return
|
|
}
|
|
|
|
$guids = $results.Links |
|
|
Where-Object ID -match '_link' |
|
|
Where-Object { $_.OuterHTML -match ( "(?=.*" + ( $Filter -join ")(?=.*" ) + ")" ) } |
|
|
Select-Object -First 1 |
|
|
ForEach-Object { $_.id.replace('_link', '') } |
|
|
Where-Object { $_ -in $kbids }
|
|
|
|
if (-not $guids) {
|
|
Write-Warning -Message "No file found for $Name"
|
|
return
|
|
}
|
|
|
|
foreach ($guid in $guids) {
|
|
# Write-Verbose -Message "Downloading information for $guid"
|
|
$post = @{ size = 0; updateID = $guid; uidInfo = $guid } | ConvertTo-Json -Compress
|
|
$body = @{ updateIDs = "[$post]" }
|
|
$OriginalVerbosePreference = $VerbosePreference
|
|
$VerbosePreference = 'SilentlyContinue'
|
|
$links = Invoke-WebRequest -Uri 'https://www.catalog.update.microsoft.com/DownloadDialog.aspx' -Method Post -Body $body -Headers $Headers -UserAgent $UserAgent |
|
|
Select-Object -ExpandProperty Content |
|
|
Select-String -AllMatches -Pattern "http[s]?://[^']*\.microsoft\.com/[^']*|http[s]?://[^']*\.windowsupdate\.com/[^']*" |
|
|
Select-Object -Unique
|
|
$VerbosePreference = $OriginalVerbosePreference
|
|
|
|
foreach ($link in $links) {
|
|
$link.matches.value
|
|
#Filter out cab files
|
|
# #if ($link -notmatch '\.cab') {
|
|
# $link.matches.value
|
|
# }
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
function Save-KB {
|
|
[CmdletBinding()]
|
|
param(
|
|
[string[]]$Name,
|
|
[string]$Path
|
|
)
|
|
foreach ($kb in $name) {
|
|
$links = Get-KBLink -Name $kb
|
|
foreach ($link in $links) {
|
|
# if (!($link -match 'x64' -or $link -match 'amd64' -or $link -match 'x86' -or $link -match 'arm64')) {
|
|
# WriteLog "No architecture found in $link, skipping"
|
|
# continue
|
|
# }
|
|
|
|
if ($link -match 'x64' -or $link -match 'amd64') {
|
|
if ($WindowsArch -eq 'x64') {
|
|
Writelog "Downloading $link for $WindowsArch to $Path"
|
|
Start-BitsTransferWithRetry -Source $link -Destination $Path
|
|
$fileName = ($link -split '/')[-1]
|
|
Writelog "Returning $fileName"
|
|
}
|
|
|
|
}
|
|
elseif ($link -match 'arm64') {
|
|
if ($WindowsArch -eq 'arm64') {
|
|
Writelog "Downloading $Link for $WindowsArch to $Path"
|
|
Start-BitsTransferWithRetry -Source $link -Destination $Path
|
|
$fileName = ($link -split '/')[-1]
|
|
Writelog "Returning $fileName"
|
|
}
|
|
}
|
|
elseif ($link -match 'x86') {
|
|
if ($WindowsArch -eq 'x86') {
|
|
Writelog "Downloading $link for $WindowsArch to $Path"
|
|
Start-BitsTransferWithRetry -Source $link -Destination $Path
|
|
$fileName = ($link -split '/')[-1]
|
|
Writelog "Returning $fileName"
|
|
}
|
|
|
|
}
|
|
else {
|
|
WriteLog "No architecture found in $link"
|
|
|
|
#If no architecture is found, download the file and run it through Get-PEArchitecture to determine the architecture
|
|
Writelog "Downloading $link to $Path and analyzing file for architecture"
|
|
Start-BitsTransferWithRetry -Source $link -Destination $Path
|
|
|
|
#Take the file and run it through Get-PEArchitecture to determine the architecture
|
|
$fileName = ($link -split '/')[-1]
|
|
$filePath = Join-Path -Path $Path -ChildPath $fileName
|
|
$arch = Get-PEArchitecture -FilePath $filePath
|
|
Writelog "$fileName is $arch"
|
|
#If the architecture matches $WindowsArch, keep the file, otherwise delete it
|
|
if ($arch -eq $WindowsArch) {
|
|
Writelog "Architecture for $fileName matches $WindowsArch, keeping file"
|
|
return $fileName
|
|
}
|
|
else {
|
|
Writelog "Deleting $fileName, architecture does not match"
|
|
Remove-Item -Path $filePath -Force
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
return $fileName
|
|
}
|
|
|
|
function New-AppsISO {
|
|
#Create Apps ISO file
|
|
$OSCDIMG = "$adkpath`Assessment and Deployment Kit\Deployment Tools\amd64\Oscdimg\oscdimg.exe"
|
|
#Adding Long Path support for AppsPath to prevent issues with oscdimg
|
|
$AppsPath = '\\?\' + $AppsPath
|
|
Invoke-Process $OSCDIMG "-n -m -d $Appspath $AppsISO" | Out-Null
|
|
}
|
|
function Get-WimFromISO {
|
|
#Mount ISO, get Wim file
|
|
$mountResult = Mount-DiskImage -ImagePath $isoPath -PassThru
|
|
$sourcesFolder = ($mountResult | Get-Volume).DriveLetter + ":\sources\"
|
|
|
|
# Check for install.wim or install.esd
|
|
$wimPath = (Get-ChildItem $sourcesFolder\install.* | Where-Object { $_.Name -match "install\.(wim|esd)" }).FullName
|
|
|
|
if ($wimPath) {
|
|
WriteLog "The path to the install file is: $wimPath"
|
|
}
|
|
else {
|
|
WriteLog "No install.wim or install.esd file found in: $sourcesFolder"
|
|
}
|
|
|
|
return $wimPath
|
|
}
|
|
|
|
function Get-Index {
|
|
param(
|
|
[Parameter(Mandatory = $true)]
|
|
[string]$WindowsImagePath,
|
|
|
|
[Parameter(Mandatory = $true)]
|
|
[string]$WindowsSKU
|
|
)
|
|
|
|
|
|
# Get the available indexes using Get-WindowsImage
|
|
$imageIndexes = Get-WindowsImage -ImagePath $WindowsImagePath
|
|
|
|
# Get the ImageName of ImageIndex 1 if an ISO was specified, else use ImageIndex 4 - this is usually Home or Education SKU on ESD MCT media
|
|
if ($ISOPath) {
|
|
if ($WindowsSKU -notmatch "Standard|Datacenter") {
|
|
$imageIndex = $imageIndexes | Where-Object ImageIndex -eq 1
|
|
$WindowsImage = $imageIndex.ImageName.Substring(0, 10)
|
|
}
|
|
else {
|
|
$imageIndex = $imageIndexes | Where-Object ImageIndex -eq 1
|
|
$WindowsImage = $imageIndex.ImageName.Substring(0, 19)
|
|
}
|
|
}
|
|
else {
|
|
$imageIndex = $imageIndexes | Where-Object ImageIndex -eq 4
|
|
$WindowsImage = $imageIndex.ImageName.Substring(0, 10)
|
|
}
|
|
|
|
# Concatenate $WindowsImage and $WindowsSKU (E.g. Windows 11 Pro)
|
|
$ImageNameToFind = "$WindowsImage $WindowsSKU"
|
|
|
|
# Find the ImageName in all of the indexes in the image
|
|
$matchingImageIndex = $imageIndexes | Where-Object ImageName -eq $ImageNameToFind
|
|
|
|
# Return the index that matches exactly
|
|
if ($matchingImageIndex) {
|
|
return $matchingImageIndex.ImageIndex
|
|
}
|
|
else {
|
|
# Look for the numbers 10, 11, 2016, 2019, 2022+ in the ImageName
|
|
$relevantImageIndexes = $imageIndexes | Where-Object { ($_.ImageName -match "(10|11|2016|2019|202\d)") }
|
|
|
|
while ($true) {
|
|
# Present list of ImageNames to the end user if no matching ImageIndex is found
|
|
Write-Host "No matching ImageIndex found for $ImageNameToFind. Please select an ImageName from the list below:"
|
|
|
|
$i = 1
|
|
$relevantImageIndexes | ForEach-Object {
|
|
Write-Host "$i. $($_.ImageName)"
|
|
$i++
|
|
}
|
|
|
|
# Ask for user input
|
|
$inputValue = Read-Host "Enter the number of the ImageName you want to use"
|
|
|
|
# Get selected ImageName based on user input
|
|
$selectedImage = $relevantImageIndexes[$inputValue - 1]
|
|
|
|
if ($selectedImage) {
|
|
return $selectedImage.ImageIndex
|
|
}
|
|
else {
|
|
Write-Host "Invalid selection, please try again."
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#Create VHDX
|
|
function New-ScratchVhdx {
|
|
param(
|
|
[Parameter(Mandatory = $true)]
|
|
[string]$VhdxPath,
|
|
[uint64]$SizeBytes = 30GB,
|
|
[uint32]$LogicalSectorSizeBytes,
|
|
[switch]$Dynamic,
|
|
[Microsoft.PowerShell.Cmdletization.GeneratedTypes.Disk.PartitionStyle]$PartitionStyle = [Microsoft.PowerShell.Cmdletization.GeneratedTypes.Disk.PartitionStyle]::GPT
|
|
)
|
|
|
|
WriteLog "Creating new Scratch VHDX..."
|
|
|
|
$newVHDX = New-VHD -Path $VhdxPath -SizeBytes $disksize -LogicalSectorSizeBytes $LogicalSectorSizeBytes -Dynamic:($Dynamic.IsPresent)
|
|
$toReturn = $newVHDX | Mount-VHD -Passthru | Initialize-Disk -PassThru -PartitionStyle GPT
|
|
|
|
#Remove auto-created partition so we can create the correct partition layout
|
|
remove-partition $toreturn.DiskNumber -PartitionNumber 1 -Confirm:$False
|
|
|
|
Writelog "Done."
|
|
return $toReturn
|
|
}
|
|
#Add System Partition
|
|
function New-SystemPartition {
|
|
param(
|
|
[Parameter(Mandatory = $true)]
|
|
[ciminstance]$VhdxDisk,
|
|
[uint64]$SystemPartitionSize = 260MB
|
|
)
|
|
|
|
WriteLog "Creating System partition..."
|
|
|
|
$sysPartition = $VhdxDisk | New-Partition -DriveLetter 'S' -Size $SystemPartitionSize -GptType "{c12a7328-f81f-11d2-ba4b-00a0c93ec93b}" -IsHidden
|
|
$sysPartition | Format-Volume -FileSystem FAT32 -Force -NewFileSystemLabel "System"
|
|
|
|
WriteLog 'Done.'
|
|
return $sysPartition.DriveLetter
|
|
}
|
|
#Add MSRPartition
|
|
function New-MSRPartition {
|
|
param(
|
|
[Parameter(Mandatory = $true)]
|
|
[ciminstance]$VhdxDisk
|
|
)
|
|
|
|
WriteLog "Creating MSR partition..."
|
|
|
|
# $toReturn = $VhdxDisk | New-Partition -AssignDriveLetter -Size 16MB -GptType "{e3c9e316-0b5c-4db8-817d-f92df00215ae}" -IsHidden | Out-Null
|
|
$toReturn = $VhdxDisk | New-Partition -Size 16MB -GptType "{e3c9e316-0b5c-4db8-817d-f92df00215ae}" -IsHidden | Out-Null
|
|
|
|
WriteLog "Done."
|
|
|
|
return $toReturn
|
|
}
|
|
#Add OS Partition
|
|
function New-OSPartition {
|
|
param(
|
|
[Parameter(Mandatory = $true)]
|
|
[ciminstance]$VhdxDisk,
|
|
[Parameter(Mandatory = $true)]
|
|
[string]$WimPath,
|
|
[uint32]$WimIndex,
|
|
[uint64]$OSPartitionSize = 0
|
|
)
|
|
|
|
WriteLog "Creating OS partition..."
|
|
|
|
if ($OSPartitionSize -gt 0) {
|
|
$osPartition = $vhdxDisk | New-Partition -DriveLetter 'W' -Size $OSPartitionSize -GptType "{ebd0a0a2-b9e5-4433-87c0-68b6b72699c7}"
|
|
}
|
|
else {
|
|
$osPartition = $vhdxDisk | New-Partition -DriveLetter 'W' -UseMaximumSize -GptType "{ebd0a0a2-b9e5-4433-87c0-68b6b72699c7}"
|
|
}
|
|
|
|
$osPartition | Format-Volume -FileSystem NTFS -Confirm:$false -Force -NewFileSystemLabel "Windows"
|
|
WriteLog 'Done'
|
|
Writelog "OS partition at drive $($osPartition.DriveLetter):"
|
|
|
|
WriteLog "Writing Windows at $WimPath to OS partition at drive $($osPartition.DriveLetter):..."
|
|
|
|
#Server 2019 is missing the Windows Overlay Filter (wof.sys), likely other Server SKUs are missing it as well. Script will error if trying to use the -compact switch on Server OSes
|
|
if ((Get-CimInstance Win32_OperatingSystem).Caption -match "Server") {
|
|
WriteLog (Expand-WindowsImage -ImagePath $WimPath -Index $WimIndex -ApplyPath "$($osPartition.DriveLetter):\")
|
|
}
|
|
elseif ($CompactOS) {
|
|
WriteLog '$CompactOS is set to true, using -Compact switch to apply the WIM file to the OS partition.'
|
|
WriteLog (Expand-WindowsImage -ImagePath $WimPath -Index $WimIndex -ApplyPath "$($osPartition.DriveLetter):\" -Compact)
|
|
}
|
|
else {
|
|
WriteLog (Expand-WindowsImage -ImagePath $WimPath -Index $WimIndex -ApplyPath "$($osPartition.DriveLetter):\")
|
|
}
|
|
|
|
WriteLog 'Done'
|
|
return $osPartition
|
|
}
|
|
#Add Recovery partition
|
|
function New-RecoveryPartition {
|
|
param(
|
|
[Parameter(Mandatory = $true)]
|
|
[ciminstance]$VhdxDisk,
|
|
[Parameter(Mandatory = $true)]
|
|
$OsPartition,
|
|
[uint64]$RecoveryPartitionSize = 0,
|
|
[ciminstance]$DataPartition
|
|
)
|
|
|
|
WriteLog "Creating empty Recovery partition (to be filled on first boot automatically)..."
|
|
|
|
$calculatedRecoverySize = 0
|
|
$recoveryPartition = $null
|
|
|
|
if ($RecoveryPartitionSize -gt 0) {
|
|
$calculatedRecoverySize = $RecoveryPartitionSize
|
|
}
|
|
else {
|
|
$winReWim = Get-ChildItem "$($OsPartition.DriveLetter):\Windows\System32\Recovery\Winre.wim" -Attributes Hidden -ErrorAction SilentlyContinue
|
|
|
|
if (($null -ne $winReWim) -and ($winReWim.Count -eq 1)) {
|
|
# Wim size + 100MB is minimum WinRE partition size.
|
|
# NTFS and other partitioning size differences account for about 17MB of space that's unavailable.
|
|
# Adding 32MB as a buffer to ensure there's enough space to account for NTFS file system overhead.
|
|
# Adding 250MB as per recommendations from
|
|
# https://learn.microsoft.com/en-us/windows-hardware/manufacture/desktop/configure-uefigpt-based-hard-drive-partitions?view=windows-11#recovery-tools-partition
|
|
$calculatedRecoverySize = $winReWim.Length + 250MB + 32MB
|
|
|
|
WriteLog "Calculated space needed for recovery in bytes: $calculatedRecoverySize"
|
|
|
|
if ($null -ne $DataPartition) {
|
|
$DataPartition | Resize-Partition -Size ($DataPartition.Size - $calculatedRecoverySize)
|
|
WriteLog "Data partition shrunk by $calculatedRecoverySize bytes for Recovery partition."
|
|
}
|
|
else {
|
|
$newOsPartitionSize = [math]::Floor(($OsPartition.Size - $calculatedRecoverySize) / 4096) * 4096
|
|
$OsPartition | Resize-Partition -Size $newOsPartitionSize
|
|
WriteLog "OS partition shrunk by $calculatedRecoverySize bytes for Recovery partition."
|
|
}
|
|
|
|
$recoveryPartition = $VhdxDisk | New-Partition -DriveLetter 'R' -UseMaximumSize -GptType "{de94bba4-06d1-4d40-a16a-bfd50179d6ac}" `
|
|
| Format-Volume -FileSystem NTFS -Confirm:$false -Force -NewFileSystemLabel 'Recovery'
|
|
|
|
WriteLog "Done. Recovery partition at drive $($recoveryPartition.DriveLetter):"
|
|
}
|
|
else {
|
|
WriteLog "No WinRE.WIM found in the OS partition under \Windows\System32\Recovery."
|
|
WriteLog "Skipping creating the Recovery partition."
|
|
WriteLog "If a Recovery partition is desired, please re-run the script setting the -RecoveryPartitionSize flag as appropriate."
|
|
}
|
|
}
|
|
|
|
return $recoveryPartition
|
|
}
|
|
#Add boot files
|
|
function Add-BootFiles {
|
|
param(
|
|
[Parameter(Mandatory = $true)]
|
|
[string]$OsPartitionDriveLetter,
|
|
[Parameter(Mandatory = $true)]
|
|
[string]$SystemPartitionDriveLetter,
|
|
[string]$FirmwareType = 'UEFI'
|
|
)
|
|
|
|
WriteLog "Adding boot files for `"$($OsPartitionDriveLetter):\Windows`" to System partition `"$($SystemPartitionDriveLetter):`"..."
|
|
Invoke-Process bcdboot "$($OsPartitionDriveLetter):\Windows /S $($SystemPartitionDriveLetter): /F $FirmwareType" | Out-Null
|
|
WriteLog "Done."
|
|
}
|
|
|
|
function Enable-WindowsFeaturesByName {
|
|
param (
|
|
[Parameter(Mandatory = $true)]
|
|
[string]$FeatureNames,
|
|
[Parameter(Mandatory = $true)]
|
|
[string]$Source
|
|
)
|
|
|
|
$FeaturesArray = $FeatureNames.Split(';')
|
|
|
|
# Looping through each feature and enabling it
|
|
foreach ($FeatureName in $FeaturesArray) {
|
|
WriteLog "Enabling Windows Optional feature: $FeatureName"
|
|
Enable-WindowsOptionalFeature -Path $WindowsPartition -FeatureName $FeatureName -All -Source $Source | Out-Null
|
|
WriteLog "Done"
|
|
}
|
|
}
|
|
|
|
#Dismount VHDX
|
|
function Dismount-ScratchVhdx {
|
|
param(
|
|
[Parameter(Mandatory = $true)]
|
|
[string]$VhdxPath
|
|
)
|
|
|
|
if (Test-Path $VhdxPath) {
|
|
WriteLog "Dismounting scratch VHDX..."
|
|
Dismount-VHD -Path $VhdxPath
|
|
WriteLog "Done."
|
|
}
|
|
}
|
|
|
|
function New-FFUVM {
|
|
#Create new Gen2 VM
|
|
$VM = New-VM -Name $VMName -Path $VMPath -MemoryStartupBytes $memory -VHDPath $VHDXPath -Generation 2
|
|
Set-VMProcessor -VMName $VMName -Count $processors
|
|
|
|
#Mount AppsISO
|
|
Add-VMDvdDrive -VMName $VMName -Path $AppsISO
|
|
|
|
#Set Hard Drive as boot device
|
|
$VMHardDiskDrive = Get-VMHarddiskdrive -VMName $VMName
|
|
Set-VMFirmware -VMName $VMName -FirstBootDevice $VMHardDiskDrive
|
|
Set-VM -Name $VMName -AutomaticCheckpointsEnabled $false -StaticMemory
|
|
|
|
#Configure TPM
|
|
New-HgsGuardian -Name $VMName -GenerateCertificates
|
|
$owner = get-hgsguardian -Name $VMName
|
|
$kp = New-HgsKeyProtector -Owner $owner -AllowUntrustedRoot
|
|
Set-VMKeyProtector -VMName $VMName -KeyProtector $kp.RawData
|
|
Enable-VMTPM -VMName $VMName
|
|
|
|
#Connect to VM
|
|
WriteLog "Starting vmconnect localhost $VMName"
|
|
& vmconnect localhost "$VMName"
|
|
|
|
#Start VM
|
|
Start-VM -Name $VMName
|
|
|
|
return $VM
|
|
}
|
|
|
|
Function Set-CaptureFFU {
|
|
$CaptureFFUScriptPath = "$FFUDevelopmentPath\WinPECaptureFFUFiles\CaptureFFU.ps1"
|
|
|
|
If (-not (Test-Path -Path $FFUCaptureLocation)) {
|
|
WriteLog "Creating FFU capture location at $FFUCaptureLocation"
|
|
New-Item -Path $FFUCaptureLocation -ItemType Directory -Force
|
|
WriteLog "Successfully created FFU capture location at $FFUCaptureLocation"
|
|
}
|
|
|
|
# Create a standard user
|
|
$UserExists = Get-LocalUser -Name $UserName -ErrorAction SilentlyContinue
|
|
if (-not $UserExists) {
|
|
WriteLog "Creating FFU_User account as standard user"
|
|
New-LocalUser -Name $UserName -AccountNeverExpires -NoPassword | Out-null
|
|
WriteLog "Successfully created FFU_User account"
|
|
}
|
|
|
|
# Create a random password for the standard user
|
|
$Password = New-Guid | Select-Object -ExpandProperty Guid
|
|
$SecurePassword = ConvertTo-SecureString -String $Password -AsPlainText -Force
|
|
Set-LocalUser -Name $UserName -Password $SecurePassword -PasswordNeverExpires:$true
|
|
|
|
# Create a share of the $FFUCaptureLocation variable
|
|
$ShareExists = Get-SmbShare -Name $ShareName -ErrorAction SilentlyContinue
|
|
if (-not $ShareExists) {
|
|
WriteLog "Creating $ShareName and giving access to $UserName"
|
|
New-SmbShare -Name $ShareName -Path $FFUCaptureLocation -FullAccess $UserName | Out-Null
|
|
WriteLog "Share created"
|
|
}
|
|
|
|
# Return the share path in the format of \\<IPAddress>\<ShareName> /user:<UserName> <password>
|
|
$SharePath = "\\$VMHostIPAddress\$ShareName /user:$UserName $Password"
|
|
$SharePath = "net use W: " + $SharePath + ' 2>&1'
|
|
|
|
# Update CaptureFFU.ps1 script
|
|
if (Test-Path -Path $CaptureFFUScriptPath) {
|
|
$ScriptContent = Get-Content -Path $CaptureFFUScriptPath
|
|
#Update variables in CaptureFFU.ps1 script ($VMHostIPAddress, $ShareName, $UserName, $Password)
|
|
WriteLog 'Updating CaptureFFU.ps1 script with new share information'
|
|
$ScriptContent = $ScriptContent -replace '(\$VMHostIPAddress = ).*', "`$1'$VMHostIPAddress'"
|
|
$ScriptContent = $ScriptContent -replace '(\$ShareName = ).*', "`$1'$ShareName'"
|
|
$ScriptContent = $ScriptContent -replace '(\$UserName = ).*', "`$1'$UserName'"
|
|
$ScriptContent = $ScriptContent -replace '(\$Password = ).*', "`$1'$Password'"
|
|
if (![string]::IsNullOrEmpty($CustomFFUNameTemplate)) {
|
|
$ScriptContent = $ScriptContent -replace '(\$CustomFFUNameTemplate = ).*', "`$1'$CustomFFUNameTemplate'"
|
|
WriteLog 'Updating CaptureFFU.ps1 script with new ffu name template information'
|
|
}
|
|
Set-Content -Path $CaptureFFUScriptPath -Value $ScriptContent
|
|
WriteLog 'Update complete'
|
|
}
|
|
else {
|
|
throw "CaptureFFU.ps1 script not found at $CaptureFFUScriptPath"
|
|
}
|
|
}
|
|
|
|
function New-PEMedia {
|
|
param (
|
|
[Parameter()]
|
|
[bool]$Capture,
|
|
[Parameter()]
|
|
[bool]$Deploy
|
|
)
|
|
#Need to use the Demployment and Imaging tools environment to create winPE media
|
|
$DandIEnv = "$adkPath`Assessment and Deployment Kit\Deployment Tools\DandISetEnv.bat"
|
|
$WinPEFFUPath = "$FFUDevelopmentPath\WinPE"
|
|
|
|
If (Test-path -Path "$WinPEFFUPath") {
|
|
WriteLog "Removing old WinPE path at $WinPEFFUPath"
|
|
Remove-Item -Path "$WinPEFFUPath" -Recurse -Force | out-null
|
|
}
|
|
|
|
WriteLog "Copying WinPE files to $WinPEFFUPath"
|
|
# Use & cmd over invoke-process as Invoke-process has issues with the winpe media folder copying all contents
|
|
if ($WindowsArch -eq 'x64') {
|
|
& cmd /c """$DandIEnv"" && copype amd64 $WinPEFFUPath" | Out-Null
|
|
# Invoke-Process cmd "/c ""$DandIEnv"" && copype amd64 $WinPEFFUPath" | Out-Null
|
|
}
|
|
elseif ($WindowsArch -eq 'arm64') {
|
|
& cmd /c """$DandIEnv"" && copype arm64 $WinPEFFUPath" | Out-Null
|
|
# Invoke-Process cmd "/c ""$DandIEnv"" && copype arm64 $WinPEFFUPath" | Out-Null
|
|
}
|
|
WriteLog 'Files copied successfully'
|
|
|
|
WriteLog 'Mounting WinPE media to add WinPE optional components'
|
|
Mount-WindowsImage -ImagePath "$WinPEFFUPath\media\sources\boot.wim" -Index 1 -Path "$WinPEFFUPath\mount" | Out-Null
|
|
WriteLog 'Mounting complete'
|
|
|
|
$Packages = @(
|
|
"WinPE-WMI.cab",
|
|
"en-us\WinPE-WMI_en-us.cab",
|
|
"WinPE-NetFX.cab",
|
|
"en-us\WinPE-NetFX_en-us.cab",
|
|
"WinPE-Scripting.cab",
|
|
"en-us\WinPE-Scripting_en-us.cab",
|
|
"WinPE-PowerShell.cab",
|
|
"en-us\WinPE-PowerShell_en-us.cab",
|
|
"WinPE-StorageWMI.cab",
|
|
"en-us\WinPE-StorageWMI_en-us.cab",
|
|
"WinPE-DismCmdlets.cab",
|
|
"en-us\WinPE-DismCmdlets_en-us.cab"
|
|
)
|
|
|
|
if ($WindowsArch -eq 'x64') {
|
|
$PackagePathBase = "$adkPath`Assessment and Deployment Kit\Windows Preinstallation Environment\amd64\WinPE_OCs\"
|
|
}
|
|
elseif ($WindowsArch -eq 'arm64') {
|
|
$PackagePathBase = "$adkPath`Assessment and Deployment Kit\Windows Preinstallation Environment\arm64\WinPE_OCs\"
|
|
}
|
|
|
|
|
|
foreach ($Package in $Packages) {
|
|
$PackagePath = Join-Path $PackagePathBase $Package
|
|
WriteLog "Adding Package $Package"
|
|
Add-WindowsPackage -Path "$WinPEFFUPath\mount" -PackagePath $PackagePath | Out-Null
|
|
WriteLog "Adding package complete"
|
|
}
|
|
If ($Capture) {
|
|
WriteLog "Copying $FFUDevelopmentPath\WinPECaptureFFUFiles\* to WinPE capture media"
|
|
Copy-Item -Path "$FFUDevelopmentPath\WinPECaptureFFUFiles\*" -Destination "$WinPEFFUPath\mount" -Recurse -Force | out-null
|
|
WriteLog "Copy complete"
|
|
#Remove Bootfix.bin - for BIOS systems, shouldn't be needed, but doesn't hurt to remove for our purposes
|
|
#Remove-Item -Path "$WinPEFFUPath\media\boot\bootfix.bin" -Force | Out-null
|
|
# $WinPEISOName = 'WinPE_FFU_Capture.iso'
|
|
$WinPEISOFile = $CaptureISO
|
|
# $Capture = $false
|
|
}
|
|
If ($Deploy) {
|
|
WriteLog "Copying $FFUDevelopmentPath\WinPEDeployFFUFiles\* to WinPE deploy media"
|
|
Copy-Item -Path "$FFUDevelopmentPath\WinPEDeployFFUFiles\*" -Destination "$WinPEFFUPath\mount" -Recurse -Force | Out-Null
|
|
WriteLog 'Copy complete'
|
|
#If $CopyPEDrivers = $true, add drivers to WinPE media using dism
|
|
if ($CopyPEDrivers) {
|
|
WriteLog "Adding drivers to WinPE media"
|
|
try {
|
|
Add-WindowsDriver -Path "$WinPEFFUPath\Mount" -Driver "$PEDriversFolder" -Recurse -ErrorAction SilentlyContinue | Out-null
|
|
}
|
|
catch {
|
|
WriteLog 'Some drivers failed to be added to the FFU. This can be expected. Continuing.'
|
|
}
|
|
WriteLog "Adding drivers complete"
|
|
}
|
|
# $WinPEISOName = 'WinPE_FFU_Deploy.iso'
|
|
$WinPEISOFile = $DeployISO
|
|
|
|
# $Deploy = $false
|
|
}
|
|
WriteLog 'Dismounting WinPE media'
|
|
Dismount-WindowsImage -Path "$WinPEFFUPath\mount" -Save | Out-Null
|
|
WriteLog 'Dismount complete'
|
|
#Make ISO
|
|
if ($WindowsArch -eq 'x64') {
|
|
$OSCDIMGPath = "$adkPath`Assessment and Deployment Kit\Deployment Tools\amd64\Oscdimg"
|
|
}
|
|
elseif ($WindowsArch -eq 'arm64') {
|
|
$OSCDIMGPath = "$adkPath`Assessment and Deployment Kit\Deployment Tools\arm64\Oscdimg"
|
|
}
|
|
$OSCDIMG = "$OSCDIMGPath\oscdimg.exe"
|
|
WriteLog "Creating WinPE ISO at $WinPEISOFile"
|
|
# & "$OSCDIMG" -m -o -u2 -udfver102 -bootdata:2`#p0,e,b$OSCDIMGPath\etfsboot.com`#pEF,e,b$OSCDIMGPath\Efisys_noprompt.bin $WinPEFFUPath\media $FFUDevelopmentPath\$WinPEISOName | Out-null
|
|
if ($WindowsArch -eq 'x64') {
|
|
if ($Capture) {
|
|
$OSCDIMGArgs = "-m -o -u2 -udfver102 -bootdata:2`#p0,e,b`"$OSCDIMGPath\etfsboot.com`"`#pEF,e,b`"$OSCDIMGPath\Efisys_noprompt.bin`" `"$WinPEFFUPath\media`" `"$WinPEISOFile`""
|
|
}
|
|
if ($Deploy) {
|
|
$OSCDIMGArgs = "-m -o -u2 -udfver102 -bootdata:2`#p0,e,b`"$OSCDIMGPath\etfsboot.com`"`#pEF,e,b`"$OSCDIMGPath\Efisys.bin`" `"$WinPEFFUPath\media`" `"$WinPEISOFile`""
|
|
}
|
|
}
|
|
elseif ($WindowsArch -eq 'arm64') {
|
|
if ($Capture) {
|
|
$OSCDIMGArgs = "-m -o -u2 -udfver102 -bootdata:1`#pEF,e,b`"$OSCDIMGPath\Efisys_noprompt.bin`" `"$WinPEFFUPath\media`" `"$WinPEISOFile`""
|
|
}
|
|
if ($Deploy) {
|
|
$OSCDIMGArgs = "-m -o -u2 -udfver102 -bootdata:1`#pEF,e,b`"$OSCDIMGPath\Efisys.bin`" `"$WinPEFFUPath\media`" `"$WinPEISOFile`""
|
|
}
|
|
|
|
}
|
|
Invoke-Process $OSCDIMG $OSCDIMGArgs | Out-Null
|
|
WriteLog "ISO created successfully"
|
|
WriteLog "Cleaning up $WinPEFFUPath"
|
|
Remove-Item -Path "$WinPEFFUPath" -Recurse -Force
|
|
WriteLog 'Cleanup complete'
|
|
}
|
|
|
|
function Optimize-FFUCaptureDrive {
|
|
param (
|
|
[string]$VhdxPath
|
|
)
|
|
try {
|
|
WriteLog 'Mounting VHDX for volume optimization'
|
|
$mountedDisk = Mount-VHD -Path $VhdxPath -Passthru | Get-Disk
|
|
$osPartition = $mountedDisk | Get-Partition | Where-Object { $_.GptType -eq "{ebd0a0a2-b9e5-4433-87c0-68b6b72699c7}" }
|
|
WriteLog 'Defragmenting Windows partition...'
|
|
Optimize-Volume -DriveLetter $osPartition.DriveLetter -Defrag -NormalPriority
|
|
WriteLog 'Performing slab consolidation on Windows partition...'
|
|
Optimize-Volume -DriveLetter $osPartition.DriveLetter -SlabConsolidate -NormalPriority
|
|
WriteLog 'Dismounting VHDX'
|
|
Dismount-ScratchVhdx -VhdxPath $VhdxPath
|
|
WriteLog 'Mounting VHDX as read-only for optimization'
|
|
Mount-VHD -Path $VhdxPath -NoDriveLetter -ReadOnly
|
|
WriteLog 'Optimizing VHDX in full mode...'
|
|
Optimize-VHD -Path $VhdxPath -Mode Full
|
|
WriteLog 'Dismounting VHDX'
|
|
Dismount-ScratchVhdx -VhdxPath $VhdxPath
|
|
}
|
|
catch {
|
|
throw $_
|
|
}
|
|
}
|
|
|
|
function Get-ShortenedWindowsSKU {
|
|
param (
|
|
[string]$WindowsSKU
|
|
)
|
|
$shortenedWindowsSKU = switch ($WindowsSKU) {
|
|
'Core' { 'Home' }
|
|
'Home' { 'Home' }
|
|
'CoreN' { 'Home_N' }
|
|
'Home N' { 'Home_N' }
|
|
'CoreSingleLanguage' { 'Home_SL' }
|
|
'Home Single Language' { 'Home_SL' }
|
|
'Education' { 'Edu' }
|
|
'EducationN' { 'Edu_N' }
|
|
'Education N' { 'Edu_N' }
|
|
'Professional' { 'Pro' }
|
|
'Pro' { 'Pro' }
|
|
'ProfessionalN' { 'Pro_N' }
|
|
'Pro N' { 'Pro_N' }
|
|
'ProfessionalEducation' { 'Pro_Edu' }
|
|
'Pro Education' { 'Pro_Edu' }
|
|
'ProfessionalEducationN' { 'Pro_Edu_N' }
|
|
'Pro Education N' { 'Pro_Edu_N' }
|
|
'ProfessionalWorkstation' { 'Pro_WKS' }
|
|
'Pro for Workstations' { 'Pro_WKS' }
|
|
'ProfessionalWorkstationN' { 'Pro_WKS_N' }
|
|
'Pro N for Workstations' { 'Pro_WKS_N' }
|
|
'Enterprise' { 'Ent' }
|
|
'EnterpriseN' { 'Ent_N' }
|
|
'Enterprise N' { 'Ent_N' }
|
|
'Enterprise N LTSC' { 'Ent_N_LTSC' }
|
|
'EnterpriseS' { 'Ent_LTSC' }
|
|
'EnterpriseSN' { 'Ent_N_LTSC' }
|
|
'Enterprise LTSC' { 'Ent_LTSC' }
|
|
'Enterprise 2016 LTSB' { 'Ent_LTSC' }
|
|
'Enterprise N 2016 LTSB' { 'Ent_N_LTSC' }
|
|
'IoT Enterprise LTSC' { 'IoT_Ent_LTSC' }
|
|
'IoTEnterpriseS' { 'IoT_Ent_LTSC' }
|
|
'IoT Enterprise N LTSC' { 'IoT_Ent_N_LTSC' }
|
|
'ServerStandard' { 'Srv_Std' }
|
|
'Standard' { 'Srv_Std' }
|
|
'ServerDatacenter' { 'Srv_Dtc' }
|
|
'Datacenter' { 'Srv_Dtc' }
|
|
'Standard (Desktop Experience)' { 'Srv_Std_DE' }
|
|
'Datacenter (Desktop Experience)' { 'Srv_Dtc_DE' }
|
|
}
|
|
return $shortenedWindowsSKU
|
|
|
|
}
|
|
function New-FFUFileName {
|
|
|
|
# $Winverinfo.name will be either Win10 or Win11 for client OSes
|
|
# Since WindowsRelease now includes dates, it breaks default name template in the config file
|
|
# This should keep in line with the naming that's done via VM Captures
|
|
if ($installationType -eq 'Client' -and $winverinfo) {
|
|
$WindowsRelease = $winverinfo.name
|
|
}
|
|
|
|
$BuildDate = Get-Date -uformat %b%Y
|
|
# Replace '{WindowsRelease}' with the Windows release (e.g., 10, 11, 2016, 2019, 2022, 2025)
|
|
$CustomFFUNameTemplate = $CustomFFUNameTemplate -replace '{WindowsRelease}', $WindowsRelease
|
|
# Replace '{WindowsVersion}' with the Windows version (e.g., 1607, 1809, 21h2, 22h2, 23h2, 24h2, etc)
|
|
$CustomFFUNameTemplate = $CustomFFUNameTemplate -replace '{WindowsVersion}', $WindowsVersion
|
|
# Replace '{SKU}' with the SKU of the Windows image (e.g., Pro, Enterprise, etc.)
|
|
$CustomFFUNameTemplate = $CustomFFUNameTemplate -replace '{SKU}', $shortenedWindowsSKU
|
|
# Replace '{BuildDate}' with the current month and year (e.g., Jan2023)
|
|
$CustomFFUNameTemplate = $CustomFFUNameTemplate -replace '{BuildDate}', $BuildDate
|
|
# Replace '{yyyy}' with the current year in 4-digit format (e.g., 2023)
|
|
$CustomFFUNameTemplate = $CustomFFUNameTemplate -replace '{yyyy}', (Get-Date -UFormat '%Y')
|
|
# Replace '{MM}' with the current month in 2-digit format (e.g., 01 for January)
|
|
$CustomFFUNameTemplate = $CustomFFUNameTemplate -creplace '{MM}', (Get-Date -UFormat '%m')
|
|
# Replace '{dd}' with the current day of the month in 2-digit format (e.g., 05)
|
|
$CustomFFUNameTemplate = $CustomFFUNameTemplate -replace '{dd}', (Get-Date -UFormat '%d')
|
|
# Replace '{HH}' with the current hour in 24-hour format (e.g., 14 for 2 PM)
|
|
$CustomFFUNameTemplate = $CustomFFUNameTemplate -creplace '{HH}', (Get-Date -UFormat '%H')
|
|
# Replace '{hh}' with the current hour in 12-hour format (e.g., 02 for 2 PM)
|
|
$CustomFFUNameTemplate = $CustomFFUNameTemplate -creplace '{hh}', (Get-Date -UFormat '%I')
|
|
# Replace '{mm}' with the current minute in 2-digit format (e.g., 09)
|
|
$CustomFFUNameTemplate = $CustomFFUNameTemplate -creplace '{mm}', (Get-Date -UFormat '%M')
|
|
# Replace '{tt}' with the current AM/PM designator (e.g., AM or PM)
|
|
$CustomFFUNameTemplate = $CustomFFUNameTemplate -replace '{tt}', (Get-Date -UFormat '%p')
|
|
if ($CustomFFUNameTemplate -notlike '*.ffu') {
|
|
$CustomFFUNameTemplate += '.ffu'
|
|
}
|
|
return $CustomFFUNameTemplate
|
|
}
|
|
|
|
function New-FFU {
|
|
param (
|
|
[Parameter(Mandatory = $false)]
|
|
[string]$VMName
|
|
)
|
|
#If $InstallApps = $true, configure the VM
|
|
If ($InstallApps) {
|
|
WriteLog 'Creating FFU from VM'
|
|
WriteLog "Setting $CaptureISO as first boot device"
|
|
$VMDVDDrive = Get-VMDvdDrive -VMName $VMName
|
|
Set-VMFirmware -VMName $VMName -FirstBootDevice $VMDVDDrive
|
|
Set-VMDvdDrive -VMName $VMName -Path $CaptureISO
|
|
$VMSwitch = Get-VMSwitch -name $VMSwitchName
|
|
WriteLog "Setting $($VMSwitch.Name) as VMSwitch"
|
|
get-vm $VMName | Get-VMNetworkAdapter | Connect-VMNetworkAdapter -SwitchName $VMSwitch.Name
|
|
WriteLog "Configuring VM complete"
|
|
|
|
#Start VM
|
|
WriteLog "Starting VM"
|
|
Start-VM -Name $VMName
|
|
|
|
# Wait for the VM to turn off
|
|
do {
|
|
$FFUVM = Get-VM -Name $VMName
|
|
Start-Sleep -Seconds 5
|
|
} while ($FFUVM.State -ne 'Off')
|
|
WriteLog "VM Shutdown"
|
|
# Check for .ffu files in the FFUDevelopment folder
|
|
WriteLog "Checking for FFU Files"
|
|
$FFUFiles = Get-ChildItem -Path $FFUCaptureLocation -Filter "*.ffu" -File
|
|
|
|
# If there's more than one .ffu file, get the most recent and store its path in $FFUFile
|
|
if ($FFUFiles.Count -gt 0) {
|
|
WriteLog 'Getting the most recent FFU file'
|
|
$FFUFile = ($FFUFiles | Sort-Object -Property LastWriteTime -Descending | Select-Object -First 1).FullName
|
|
WriteLog "Most recent .ffu file: $FFUFile"
|
|
}
|
|
else {
|
|
WriteLog "No .ffu files found in $FFUCaptureLocation"
|
|
throw $_
|
|
}
|
|
}
|
|
elseif (-not $InstallApps -and (-not $AllowVHDXCaching)) {
|
|
#Get Windows Version Information from the VHDX
|
|
$winverinfo = Get-WindowsVersionInfo
|
|
WriteLog 'Creating FFU File Name'
|
|
if ($CustomFFUNameTemplate) {
|
|
$FFUFileName = New-FFUFileName
|
|
}
|
|
else {
|
|
$FFUFileName = "$($winverinfo.Name)`_$($winverinfo.DisplayVersion)`_$($shortenedWindowsSKU)`_$($winverinfo.BuildDate).ffu"
|
|
}
|
|
WriteLog "FFU file name: $FFUFileName"
|
|
$FFUFile = "$FFUCaptureLocation\$FFUFileName"
|
|
#Capture the FFU
|
|
WriteLog 'Capturing FFU'
|
|
Invoke-Process cmd "/c ""$DandIEnv"" && dism /Capture-FFU /ImageFile:$FFUFile /CaptureDrive:\\.\PhysicalDrive$($vhdxDisk.DiskNumber) /Name:$($winverinfo.Name)$($winverinfo.DisplayVersion)$($shortenedWindowsSKU) /Compress:Default" | Out-Null
|
|
WriteLog 'FFU Capture complete'
|
|
Dismount-ScratchVhdx -VhdxPath $VHDXPath
|
|
}
|
|
elseif (-not $InstallApps -and $AllowVHDXCaching) {
|
|
# Make $FFUFileName based on values in the config.json file
|
|
WriteLog 'Creating FFU File Name'
|
|
if ($CustomFFUNameTemplate) {
|
|
$FFUFileName = New-FFUFileName
|
|
}
|
|
else {
|
|
$BuildDate = Get-Date -UFormat %b%Y
|
|
# Get Windows Information to make the FFU file name from the cachedVHDXInfo file
|
|
if ($installationType -eq 'Client') {
|
|
$FFUFileName = "Win$($cachedVHDXInfo.WindowsRelease)`_$($cachedVHDXInfo.WindowsVersion)`_$($shortenedWindowsSKU)`_$BuildDate.ffu"
|
|
}
|
|
else {
|
|
$FFUFileName = "Server$($cachedVHDXInfo.WindowsRelease)`_$($cachedVHDXInfo.WindowsVersion)`_$($shortenedWindowsSKU)`_$BuildDate.ffu"
|
|
}
|
|
}
|
|
WriteLog "FFU file name: $FFUFileName"
|
|
$FFUFile = "$FFUCaptureLocation\$FFUFileName"
|
|
#Capture the FFU
|
|
WriteLog 'Capturing FFU'
|
|
Invoke-Process cmd "/c ""$DandIEnv"" && dism /Capture-FFU /ImageFile:$FFUFile /CaptureDrive:\\.\PhysicalDrive$($vhdxDisk.DiskNumber) /Name:$($cachedVHDXInfo.WindowsRelease)$($cachedVHDXInfo.WindowsVersion)$($shortenedWindowsSKU) /Compress:Default" | Out-Null
|
|
WriteLog 'FFU Capture complete'
|
|
Dismount-ScratchVhdx -VhdxPath $VHDXPath
|
|
}
|
|
|
|
#Without this 120 second sleep, we sometimes see an error when mounting the FFU due to a file handle lock. Needed for both driver and optimize steps.
|
|
|
|
If ($InstallDrivers -or $Optimize) {
|
|
WriteLog 'Sleeping 2 minutes to prevent file handle lock'
|
|
Start-Sleep 120
|
|
}
|
|
|
|
#Add drivers
|
|
If ($InstallDrivers) {
|
|
WriteLog 'Adding drivers'
|
|
WriteLog "Creating $FFUDevelopmentPath\Mount directory"
|
|
New-Item -Path "$FFUDevelopmentPath\Mount" -ItemType Directory -Force | Out-Null
|
|
WriteLog "Created $FFUDevelopmentPath\Mount directory"
|
|
WriteLog "Mounting $FFUFile to $FFUDevelopmentPath\Mount"
|
|
Mount-WindowsImage -ImagePath $FFUFile -Index 1 -Path "$FFUDevelopmentPath\Mount" | Out-null
|
|
WriteLog 'Mounting complete'
|
|
WriteLog 'Adding drivers - This will take a few minutes, please be patient'
|
|
try {
|
|
Add-WindowsDriver -Path "$FFUDevelopmentPath\Mount" -Driver "$DriversFolder" -Recurse -ErrorAction SilentlyContinue | Out-null
|
|
}
|
|
catch {
|
|
WriteLog 'Some drivers failed to be added to the FFU. This can be expected. Continuing.'
|
|
}
|
|
WriteLog 'Adding drivers complete'
|
|
WriteLog "Dismount $FFUDevelopmentPath\Mount"
|
|
Dismount-WindowsImage -Path "$FFUDevelopmentPath\Mount" -Save | Out-Null
|
|
WriteLog 'Dismount complete'
|
|
WriteLog "Remove $FFUDevelopmentPath\Mount folder"
|
|
Remove-Item -Path "$FFUDevelopmentPath\Mount" -Recurse -Force | Out-null
|
|
WriteLog 'Folder removed'
|
|
}
|
|
#Optimize FFU
|
|
if ($Optimize -eq $true) {
|
|
WriteLog 'Optimizing FFU - This will take a few minutes, please be patient'
|
|
#Need to use ADK version of DISM to address bug in DISM - perhaps Windows 11 24H2 will fix this
|
|
Invoke-Process cmd "/c ""$DandIEnv"" && dism /optimize-ffu /imagefile:$FFUFile" | Out-Null
|
|
#Invoke-Process cmd "/c dism /optimize-ffu /imagefile:$FFUFile" | Out-Null
|
|
WriteLog 'Optimizing FFU complete'
|
|
}
|
|
|
|
|
|
}
|
|
function Remove-FFUVM {
|
|
param (
|
|
[Parameter(Mandatory = $false)]
|
|
[string]$VMName
|
|
)
|
|
#Get the VM object and remove the VM, the HGSGuardian, and the certs
|
|
If ($VMName) {
|
|
$FFUVM = get-vm $VMName | Where-Object { $_.state -ne 'running' }
|
|
}
|
|
If ($null -ne $FFUVM) {
|
|
WriteLog 'Cleaning up VM'
|
|
$certPath = 'Cert:\LocalMachine\Shielded VM Local Certificates\'
|
|
$VMName = $FFUVM.Name
|
|
WriteLog "Removing VM: $VMName"
|
|
Remove-VM -Name $VMName -Force
|
|
WriteLog 'Removal complete'
|
|
WriteLog "Removing $VMPath"
|
|
Remove-Item -Path $VMPath -Force -Recurse
|
|
WriteLog 'Removal complete'
|
|
WriteLog "Removing HGSGuardian for $VMName"
|
|
Remove-HgsGuardian -Name $VMName -WarningAction SilentlyContinue
|
|
WriteLog 'Removal complete'
|
|
WriteLog 'Cleaning up HGS Guardian certs'
|
|
$certs = Get-ChildItem -Path $certPath -Recurse | Where-Object { $_.Subject -like "*$VMName*" }
|
|
foreach ($cert in $Certs) {
|
|
Remove-item -Path $cert.PSPath -force | Out-Null
|
|
}
|
|
WriteLog 'Cert removal complete'
|
|
}
|
|
#If just building the FFU from vhdx, remove the vhdx path
|
|
If (-not $InstallApps -and $vhdxDisk) {
|
|
WriteLog 'Cleaning up VHDX'
|
|
WriteLog "Removing $VMPath"
|
|
Remove-Item -Path $VMPath -Force -Recurse | Out-Null
|
|
WriteLog 'Removal complete'
|
|
}
|
|
|
|
#Remove orphaned mounted images
|
|
$mountedImages = Get-WindowsImage -Mounted
|
|
if ($mountedImages) {
|
|
foreach ($image in $mountedImages) {
|
|
$mountPath = $image.Path
|
|
WriteLog "Dismounting image at $mountPath"
|
|
Dismount-WindowsImage -Path $mountPath -discard
|
|
WriteLog "Successfully dismounted image at $mountPath"
|
|
}
|
|
}
|
|
#Remove Mount folder if it exists
|
|
If (Test-Path -Path $FFUDevelopmentPath\Mount) {
|
|
WriteLog "Remove $FFUDevelopmentPath\Mount folder"
|
|
Remove-Item -Path "$FFUDevelopmentPath\Mount" -Recurse -Force
|
|
WriteLog 'Folder removed'
|
|
}
|
|
#Remove unused mountpoints
|
|
WriteLog 'Remove unused mountpoints'
|
|
Invoke-Process cmd "/c mountvol /r" | Out-Null
|
|
WriteLog 'Removal complete'
|
|
}
|
|
Function Remove-FFUUserShare {
|
|
WriteLog "Removing $ShareName"
|
|
Remove-SmbShare -Name $ShareName -Force | Out-null
|
|
WriteLog 'Removal complete'
|
|
WriteLog "Removing $Username"
|
|
Remove-LocalUser -Name $Username | Out-Null
|
|
WriteLog 'Removal complete'
|
|
}
|
|
|
|
Function Get-WindowsVersionInfo {
|
|
#This sleep prevents CBS/CSI corruption which causes issues with Windows update after deployment. Capturing from very fast disks (NVME) can cause the capture to happen faster than Windows is ready for. This seems to affect VHDX-only captures, not VM captures.
|
|
WriteLog 'Sleep 60 seconds before opening registry to grab Windows version info '
|
|
Start-sleep 60
|
|
WriteLog "Getting Windows Version info"
|
|
#Load Registry Hive
|
|
$Software = "$osPartitionDriveLetter`:\Windows\System32\config\software"
|
|
WriteLog "Loading Software registry hive: $Software"
|
|
Invoke-Process reg "load HKLM\FFU $Software" | Out-Null
|
|
|
|
#Find Windows version values
|
|
# $WindowsSKU = Get-ItemPropertyValue -Path 'HKLM:\FFU\Microsoft\Windows NT\CurrentVersion\' -Name 'EditionID'
|
|
# WriteLog "Windows SKU: $WindowsSKU"
|
|
[int]$CurrentBuild = Get-ItemPropertyValue -Path 'HKLM:\FFU\Microsoft\Windows NT\CurrentVersion\' -Name 'CurrentBuild'
|
|
WriteLog "Windows Build: $CurrentBuild"
|
|
#DisplayVersion does not exist for 1607 builds (RS1 and Server 2016) and Server 2019
|
|
if ($CurrentBuild -notin (14393, 17763)) {
|
|
$DisplayVersion = Get-ItemPropertyValue -Path 'HKLM:\FFU\Microsoft\Windows NT\CurrentVersion\' -Name 'DisplayVersion'
|
|
WriteLog "Windows Version: $DisplayVersion"
|
|
}
|
|
# For Windows 10 LTSC 2019, set DisplayVersion to 2019
|
|
if ($CurrentBuild -eq 17763 -and $InstallationType -eq "Client") {
|
|
$DisplayVersion = '2019'
|
|
}
|
|
|
|
$BuildDate = Get-Date -uformat %b%Y
|
|
|
|
if ($shortenedWindowsSKU -notmatch "Srv") {
|
|
if ($CurrentBuild -ge 22000) {
|
|
$Name = 'Win11'
|
|
}
|
|
else {
|
|
$Name = 'Win10'
|
|
}
|
|
}
|
|
else {
|
|
$Name = switch ($CurrentBuild) {
|
|
26100 { '2025' }
|
|
20348 { '2022' }
|
|
17763 { '2019' }
|
|
14393 { '2016' }
|
|
Default { $DisplayVersion }
|
|
}
|
|
}
|
|
|
|
WriteLog "Unloading registry"
|
|
Invoke-Process reg "unload HKLM\FFU" | Out-Null
|
|
#This prevents Critical Process Died errors you can have during deployment of the FFU. Capturing from very fast disks (NVME) can cause the capture to happen faster than Windows is ready for.
|
|
WriteLog 'Sleep 60 seconds to allow registry to completely unload'
|
|
Start-sleep 60
|
|
|
|
return @{
|
|
|
|
DisplayVersion = $DisplayVersion
|
|
BuildDate = $buildDate
|
|
Name = $Name
|
|
# SKU = $WindowsSKU
|
|
}
|
|
}
|
|
Function Get-USBDrive {
|
|
# Log the start of the USB drive check
|
|
WriteLog 'Checking for USB drives'
|
|
|
|
# Check if external hard disk media is allowed and user has not specified USB drives
|
|
If ($AllowExternalHardDiskMedia -and (-not($USBDriveList))) {
|
|
# Get all removable and external hard disk media drives
|
|
[array]$USBDrives = (Get-WmiObject -Class Win32_DiskDrive -Filter "MediaType='Removable Media' OR MediaType='External hard disk media'")
|
|
[array]$ExternalHardDiskDrives = $USBDrives | Where-Object { $_.MediaType -eq 'External hard disk media' }
|
|
$ExternalCount = $ExternalHardDiskDrives.Count
|
|
$USBDrivesCount = $USBDrives.Count
|
|
|
|
# Check if user should be prompted for external hard disk media
|
|
if ($PromptExternalHardDiskMedia) {
|
|
if ($ExternalHardDiskDrives) {
|
|
# Log and warn about found external hard disk media drives
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Warning 'Found external hard disk media drives'
|
|
Write-Warning 'Will prompt for user input to select the drive to use to prevent accidental data loss'
|
|
Write-Warning 'If you do not want to be prompted for this in the future, set -PromptExternalHardDiskMedia to $false'
|
|
}
|
|
WriteLog 'Found external hard disk media drives'
|
|
WriteLog 'Will prompt for user input to select the drive to use to prevent accidental data loss'
|
|
WriteLog 'If you do not want to be prompted for this in the future, set -PromptExternalHardDiskMedia to $false'
|
|
|
|
# Prepare output for user selection
|
|
$Output = @()
|
|
for ($i = 0; $i -lt $ExternalHardDiskDrives.Count; $i++) {
|
|
$ExternalDiskNumber = $ExternalHardDiskDrives[$i].Index
|
|
$ExternalDisk = Get-Disk -Number $ExternalDiskNumber
|
|
$Index = $i + 1
|
|
$Name = $ExternalDisk.FriendlyName
|
|
$SerialNumber = $ExternalHardDiskDrives[$i].serialnumber
|
|
$PartitionStyle = $ExternalDisk.PartitionStyle
|
|
$Status = $ExternalDisk.OperationalStatus
|
|
$Properties = [ordered]@{
|
|
'Drive Number' = $Index
|
|
'Drive Name' = $Name
|
|
'Serial Number' = $SerialNumber
|
|
'Partition Style' = $PartitionStyle
|
|
'Status' = $Status
|
|
}
|
|
$Output += New-Object PSObject -Property $Properties
|
|
}
|
|
|
|
# Format and display the output
|
|
$FormattedOutput = $Output | Format-Table -AutoSize -Property 'Drive Number', 'Drive Name', 'Serial Number', 'Partition Style', 'Status' | Out-String
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
$FormattedOutput | Out-Host
|
|
}
|
|
WriteLog $FormattedOutput
|
|
|
|
# Prompt user to select a drive
|
|
do {
|
|
$inputChoice = Read-Host "Enter the number corresponding to the external hard disk media drive you want to use"
|
|
if ($inputChoice -match '^\d+$') {
|
|
$inputChoice = [int]$inputChoice
|
|
if ($inputChoice -ge 1 -and $inputChoice -le $ExternalCount) {
|
|
$SelectedIndex = $inputChoice - 1
|
|
$ExternalDiskNumber = $ExternalHardDiskDrives[$SelectedIndex].Index
|
|
$ExternalDisk = Get-Disk -Number $ExternalDiskNumber
|
|
$USBDrives = $ExternalHardDiskDrives[$SelectedIndex]
|
|
$USBDrivesCount = $USBDrives.Count
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Host "Drive $inputChoice was selected"
|
|
}
|
|
WriteLog "Drive $inputChoice was selected"
|
|
}
|
|
else {
|
|
# Handle invalid selection
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Host "Invalid selection. Please try again."
|
|
}
|
|
WriteLog "Invalid selection. Please try again."
|
|
}
|
|
|
|
# Check if the selected drive is offline
|
|
if ($ExternalDisk.OperationalStatus -eq 'Offline') {
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Error "Selected Drive is in an Offline State. Please check the drive status in Disk Manager and try again."
|
|
}
|
|
WriteLog "Selected Drive is in an Offline State. Please check the drive status in Disk Manager and try again."
|
|
exit 1
|
|
}
|
|
}
|
|
else {
|
|
# Handle invalid input
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Host "Invalid selection. Please try again."
|
|
}
|
|
WriteLog "Invalid selection. Please try again."
|
|
}
|
|
} while ($null -eq $selectedIndex)
|
|
}
|
|
}
|
|
else {
|
|
# Log the count of found USB drives
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Host "Found $USBDrivesCount total USB drives"
|
|
If ($ExternalCount -gt 0) {
|
|
Write-Host "$ExternalCount external drives"
|
|
}
|
|
}
|
|
WriteLog "Found $USBDrivesCount total USB drives"
|
|
If ($ExternalCount -gt 0) {
|
|
WriteLog "$ExternalCount external drives"
|
|
}
|
|
}
|
|
}
|
|
elseif ($USBDriveList) {
|
|
# Log the count of specified USB drives
|
|
$USBDriveListCount = $USBDriveList.Count
|
|
WriteLog "Looking for $USBDriveListCount USB drives from USB Drive List"
|
|
# Get only the specified USB drives based on both model and serial number
|
|
$USBDrives = @()
|
|
foreach ($model in $USBDriveList.Keys) {
|
|
$serialNumber = $USBDriveList[$model]
|
|
Writelog "Looking for USB drive model $model with serial number $serialNumber"
|
|
$USBDrive = Get-CimInstance -ClassName Win32_DiskDrive -Filter "Model LIKE '%$model%' AND SerialNumber LIKE '$serialNumber%' AND (MediaType='Removable Media' OR MediaType='External hard disk media')"
|
|
if ($USBDrive) {
|
|
WriteLog "Found USB drive model $($USBDrive.model) with serial number $($USBDrive.serialNumber)"
|
|
$USBDrives += $USBDrive
|
|
}
|
|
else {
|
|
WriteLog "USB drive model $model with serial number $serialNumber not found"
|
|
}
|
|
}
|
|
$USBDrivesCount = $USBDrives.Count
|
|
WriteLog "Found $USBDrivesCount of $USBDriveListCount USB drives from USB Drive List"
|
|
}
|
|
else {
|
|
# Get only removable media drives
|
|
[array]$USBDrives = (Get-WmiObject -Class Win32_DiskDrive -Filter "MediaType='Removable Media'")
|
|
$USBDrivesCount = $USBDrives.Count
|
|
WriteLog "Found $USBDrivesCount Removable USB drives"
|
|
}
|
|
|
|
# Check if any USB drives were found
|
|
if ($null -eq $USBDrives) {
|
|
WriteLog "No USB drive found. Exiting"
|
|
Write-Error "No USB drive found. Exiting"
|
|
exit 1
|
|
}
|
|
|
|
# Return the found USB drives and their count
|
|
return $USBDrives, $USBDrivesCount
|
|
}
|
|
Function New-DeploymentUSB {
|
|
param(
|
|
[switch]$CopyFFU
|
|
)
|
|
WriteLog "CopyFFU is set to $CopyFFU"
|
|
$BuildUSBPath = $PSScriptRoot
|
|
WriteLog "BuildUSBPath is $BuildUSBPath"
|
|
|
|
$SelectedFFUFile = $null
|
|
|
|
# 1. Get FFU File(s)
|
|
# Check if the CopyFFU switch is present
|
|
if ($CopyFFU.IsPresent) {
|
|
# Get all FFU files in the specified directory
|
|
$FFUFiles = Get-ChildItem -Path "$BuildUSBPath\FFU" -Filter "*.ffu"
|
|
$FFUCount = $FFUFiles.count
|
|
|
|
# If there is exactly one FFU file, select it
|
|
if ($FFUCount -eq 1) {
|
|
$SelectedFFUFile = $FFUFiles.FullName
|
|
}
|
|
# If there are multiple FFU files, prompt the user to select one
|
|
elseif ($FFUCount -gt 1) {
|
|
WriteLog "Found $FFUCount FFU files"
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Host "Found $FFUCount FFU files"
|
|
}
|
|
$output = @()
|
|
# Create a table of FFU files with their index, name, and last modified date
|
|
for ($i = 0; $i -lt $FFUCount; $i++) {
|
|
$index = $i + 1
|
|
$name = $FFUFiles[$i].Name
|
|
$modified = $FFUFiles[$i].LastWriteTime
|
|
$Properties = [ordered]@{
|
|
'FFU Number' = $index
|
|
'FFU Name' = $name
|
|
'Last Modified' = $modified
|
|
}
|
|
$output += New-Object PSObject -Property $Properties
|
|
}
|
|
$output | Format-Table -AutoSize -Property 'FFU Number', 'FFU Name', 'Last Modified'
|
|
|
|
# Loop until a valid FFU file is selected
|
|
do {
|
|
$inputChoice = Read-Host "Enter the number corresponding to the FFU file you want to copy or 'A' to copy all FFU files"
|
|
# Check if the input is a valid number or 'A'
|
|
if ($inputChoice -match '^\d+$' -or $inputChoice -eq 'A') {
|
|
if ($inputChoice -eq 'A') {
|
|
# Select all FFU files
|
|
$SelectedFFUFile = $FFUFiles.FullName
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Host 'Will copy all FFU files'
|
|
}
|
|
WriteLog 'Will copy all FFU Files'
|
|
}
|
|
else {
|
|
# Convert input to integer and validate the selection
|
|
$inputChoice = [int]$inputChoice
|
|
if ($inputChoice -ge 1 -and $inputChoice -le $FFUCount) {
|
|
$selectedIndex = $inputChoice - 1
|
|
$SelectedFFUFile = $FFUFiles[$selectedIndex].FullName
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Host "$SelectedFFUFile was selected"
|
|
}
|
|
WriteLog "$SelectedFFUFile was selected"
|
|
}
|
|
else {
|
|
# Handle invalid selection
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Host "Invalid selection. Please try again."
|
|
}
|
|
WriteLog "Invalid selection. Please try again."
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
# Handle invalid input
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Host "Invalid selection. Please try again."
|
|
}
|
|
WriteLog "Invalid selection. Please try again."
|
|
}
|
|
} while ($null -eq $SelectedFFUFile)
|
|
|
|
}
|
|
else {
|
|
# Handle case where no FFU files are found
|
|
WriteLog "No FFU files found in the current directory."
|
|
Write-Error "No FFU files found in the current directory."
|
|
Return
|
|
}
|
|
}
|
|
# 2. Partition and format USB drives
|
|
$counter = 0
|
|
|
|
foreach ($USBDrive in $USBDrives) {
|
|
$Counter++
|
|
WriteLog "Formatting USB drive $Counter out of $USBDrivesCount"
|
|
WriteLog "USB Drive Model: $($USBDrive.Model)"
|
|
WriteLog "USB Drive Serial Number: $($USBDrive.SerialNumber)"
|
|
$DiskNumber = $USBDrive.DeviceID.Replace("\\.\PHYSICALDRIVE", "")
|
|
WriteLog "Physical Disk number is $DiskNumber for USB drive $Counter out of $USBDrivesCount"
|
|
|
|
$ScriptBlock = {
|
|
param($DiskNumber)
|
|
$Disk = Get-Disk -Number $DiskNumber
|
|
# Clear-Disk -Number $DiskNumber -RemoveData -RemoveOEM -Confirm:$false
|
|
# Clear-disk has an unusual behavior where it sets external hard disk media as RAW, however removable media is set as MBR.
|
|
if ($Disk.PartitionStyle -ne "RAW") {
|
|
$Disk | Clear-Disk -RemoveData -RemoveOEM -Confirm:$false
|
|
$Disk = Get-Disk -Number $DiskNumber
|
|
}
|
|
|
|
if ($Disk.PartitionStyle -eq "RAW") {
|
|
$Disk | Initialize-Disk -PartitionStyle MBR -Confirm:$false
|
|
}
|
|
elseif ($Disk.PartitionStyle -ne "RAW") {
|
|
$Disk | Get-Partition | Remove-Partition -Confirm:$false
|
|
$Disk | Set-Disk -PartitionStyle MBR
|
|
}
|
|
# Get-Disk $DiskNumber | Get-Partition | Remove-Partition
|
|
$BootPartition = $Disk | New-Partition -Size 2GB -IsActive -AssignDriveLetter
|
|
$DeployPartition = $Disk | New-Partition -UseMaximumSize -AssignDriveLetter
|
|
Format-Volume -Partition $BootPartition -FileSystem FAT32 -NewFileSystemLabel "TempBoot" -Confirm:$false
|
|
Format-Volume -Partition $DeployPartition -FileSystem NTFS -NewFileSystemLabel "TempDeploy" -Confirm:$false
|
|
}
|
|
|
|
WriteLog 'Partitioning USB Drive'
|
|
Invoke-Command -ScriptBlock $ScriptBlock -ArgumentList $DiskNumber | Out-null
|
|
WriteLog 'Done'
|
|
|
|
# 3. Copy WinPE files to USB drive boot partition
|
|
|
|
# $BootPartitionDriveLetter = (Get-WmiObject -Class win32_volume -Filter "Label='TempBoot' AND DriveType=2 AND DriveLetter IS NOT NULL").Name
|
|
$BootPartitionDriveLetter = (Get-WmiObject -Class win32_volume -Filter "Label='TempBoot' AND DriveLetter IS NOT NULL").Name
|
|
$ISOMountPoint = (Mount-DiskImage -ImagePath $DeployISO -PassThru | Get-Volume).DriveLetter + ":\"
|
|
WriteLog "Copying WinPE files to $BootPartitionDriveLetter"
|
|
robocopy "$ISOMountPoint" "$BootPartitionDriveLetter" /E /COPYALL /R:5 /W:5 /J
|
|
Dismount-DiskImage -ImagePath $DeployISO | Out-Null
|
|
|
|
#4. Copy FFU file(s), drivers, PPKG, Autopilot, unattend to USB drive deploy partition
|
|
|
|
if ($CopyFFU.IsPresent) {
|
|
if ($null -ne $SelectedFFUFile) {
|
|
# $DeployPartitionDriveLetter = (Get-WmiObject -Class win32_volume -Filter "Label='TempDeploy' AND DriveType=2 AND DriveLetter IS NOT NULL").Name
|
|
$DeployPartitionDriveLetter = (Get-WmiObject -Class win32_volume -Filter "Label='TempDeploy' AND DriveLetter IS NOT NULL").Name
|
|
if ($SelectedFFUFile -is [array]) {
|
|
WriteLog "Copying multiple FFU files to $DeployPartitionDriveLetter. This could take a few minutes."
|
|
foreach ($FFUFile in $SelectedFFUFile) {
|
|
robocopy $(Split-Path $FFUFile -Parent) $DeployPartitionDriveLetter $(Split-Path $FFUFile -Leaf) /COPYALL /R:5 /W:5 /J
|
|
}
|
|
}
|
|
else {
|
|
WriteLog ("Copying " + $SelectedFFUFile + " to $DeployPartitionDriveLetter. This could take a few minutes.")
|
|
robocopy $(Split-Path $SelectedFFUFile -Parent) $DeployPartitionDriveLetter $(Split-Path $SelectedFFUFile -Leaf) /COPYALL /R:5 /W:5 /J
|
|
}
|
|
#Copy drivers using robocopy due to potential size
|
|
if ($CopyDrivers) {
|
|
WriteLog "Copying drivers to $DeployPartitionDriveLetter\Drivers"
|
|
if ($Make) {
|
|
WriteLog "Copying drivers for make: $Make"
|
|
robocopy "$DriversFolder\$Make" "$DeployPartitionDriveLetter\Drivers" /E /R:5 /W:5 /J
|
|
}
|
|
else {
|
|
WriteLog "No make specified, copying all drivers"
|
|
robocopy "$DriversFolder" "$DeployPartitionDriveLetter\Drivers" /E /R:5 /W:5 /J
|
|
}
|
|
|
|
}
|
|
#Copy Unattend file to the USB drive.
|
|
if ($CopyUnattend) {
|
|
# WriteLog "Copying Unattend folder to $DeployPartitionDriveLetter"
|
|
# Copy-Item -Path "$FFUDevelopmentPath\Unattend" -Destination $DeployPartitionDriveLetter -Recurse -Force
|
|
$DeployUnattendPath = "$DeployPartitionDriveLetter\unattend"
|
|
WriteLog "Copying unattend file to $DeployUnattendPath"
|
|
New-Item -Path $DeployUnattendPath -ItemType Directory | Out-Null
|
|
if ($WindowsArch -eq 'x64') {
|
|
Copy-Item -Path "$FFUDevelopmentPath\unattend\unattend_x64.xml" -Destination "$DeployUnattendPath\Unattend.xml" -Force | Out-Null
|
|
}
|
|
if ($WindowsArch -eq 'arm64') {
|
|
Copy-Item -Path "$FFUDevelopmentPath\unattend\unattend_arm64.xml" -Destination "$DeployUnattendPath\Unattend.xml" -Force | Out-Null
|
|
}
|
|
#Check for prefixes.txt file and copy it to the USB drive
|
|
if (Test-Path "$FFUDevelopmentPath\unattend\prefixes.txt") {
|
|
WriteLog "Copying prefixes.txt file to $DeployUnattendPath"
|
|
Copy-Item -Path "$FFUDevelopmentPath\unattend\prefixes.txt" -Destination "$DeployUnattendPath\prefixes.txt" -Force | Out-Null
|
|
}
|
|
WriteLog 'Copy completed'
|
|
}
|
|
#Copy PPKG folder in the FFU folder to the USB drive. Can use copy-item as it's a small folder
|
|
if ($CopyPPKG) {
|
|
WriteLog "Copying PPKG folder to $DeployPartitionDriveLetter"
|
|
Copy-Item -Path "$FFUDevelopmentPath\PPKG" -Destination $DeployPartitionDriveLetter -Recurse -Force
|
|
}
|
|
#Copy Autopilot folder in the FFU folder to the USB drive. Can use copy-item as it's a small folder
|
|
if ($CopyAutopilot) {
|
|
WriteLog "Copying Autopilot folder to $DeployPartitionDriveLetter"
|
|
Copy-Item -Path "$FFUDevelopmentPath\Autopilot" -Destination $DeployPartitionDriveLetter -Recurse -Force
|
|
}
|
|
}
|
|
else {
|
|
WriteLog "No FFU file selected. Skipping copy."
|
|
}
|
|
}
|
|
|
|
Set-Volume -FileSystemLabel "TempBoot" -NewFileSystemLabel "Boot"
|
|
Set-Volume -FileSystemLabel "TempDeploy" -NewFileSystemLabel "Deploy"
|
|
|
|
if ($USBDrivesCount -gt 1) {
|
|
& mountvol $BootPartitionDriveLetter /D
|
|
& mountvol $DeployPartitionDriveLetter /D
|
|
}
|
|
|
|
WriteLog "Drive $counter completed"
|
|
}
|
|
|
|
WriteLog "USB Drives completed"
|
|
}
|
|
|
|
|
|
function Get-FFUEnvironment {
|
|
WriteLog 'Dirty.txt file detected. Last run did not complete succesfully. Will clean environment'
|
|
# Check for running VMs that start with '_FFU-' and are in the 'Off' state
|
|
$vms = Get-VM
|
|
|
|
# Loop through each VM
|
|
foreach ($vm in $vms) {
|
|
if ($vm.Name.StartsWith("_FFU-")) {
|
|
if ($vm.State -eq 'Running') {
|
|
Stop-VM -Name $vm.Name -TurnOff -Force
|
|
}
|
|
# If conditions are met, delete the VM
|
|
Remove-FFUVM -VMName $vm.Name
|
|
}
|
|
}
|
|
# Check for MSFT Virtual disks where location contains FFUDevelopment in the path
|
|
$disks = Get-Disk -FriendlyName *virtual*
|
|
foreach ($disk in $disks) {
|
|
$diskNumber = $disk.Number
|
|
$vhdLocation = $disk.Location
|
|
if ($vhdLocation -like "*FFUDevelopment*") {
|
|
WriteLog "Dismounting Virtual Disk $diskNumber with Location $vhdLocation"
|
|
Dismount-ScratchVhdx -VhdxPath $vhdLocation
|
|
$parentFolder = Split-Path -Parent $vhdLocation
|
|
WriteLog "Removing folder $parentFolder"
|
|
Remove-Item -Path $parentFolder -Recurse -Force
|
|
}
|
|
}
|
|
|
|
# Check for mounted DiskImages
|
|
$volumes = Get-Volume | Where-Object { $_.DriveType -eq 'CD-ROM' }
|
|
foreach ($volume in $volumes) {
|
|
$letter = $volume.DriveLetter
|
|
WriteLog "Dismounting DiskImage for volume $letter"
|
|
Get-Volume $letter | Get-DiskImage | Dismount-DiskImage | Out-Null
|
|
WriteLog "Dismounting complete"
|
|
}
|
|
|
|
# Remove unused mountpoints
|
|
WriteLog 'Remove unused mountpoints'
|
|
Invoke-Process cmd "/c mountvol /r" | Out-Null
|
|
WriteLog 'Removal complete'
|
|
|
|
# Check for content in the VM folder and delete any folders that start with _FFU-
|
|
$folders = Get-ChildItem -Path $VMLocation -Directory
|
|
foreach ($folder in $folders) {
|
|
if ($folder.Name -like '_FFU-*') {
|
|
WriteLog "Removing folder $($folder.FullName)"
|
|
Remove-Item -Path $folder.FullName -Recurse -Force
|
|
}
|
|
}
|
|
|
|
# Remove orphaned mounted images
|
|
$mountedImages = Get-WindowsImage -Mounted
|
|
if ($mountedImages) {
|
|
foreach ($image in $mountedImages) {
|
|
$mountPath = $image.Path
|
|
WriteLog "Dismounting image at $mountPath"
|
|
try {
|
|
Dismount-WindowsImage -Path $mountPath -discard | Out-null
|
|
WriteLog "Successfully dismounted image at $mountPath"
|
|
}
|
|
catch {
|
|
WriteLog "Failed to dismount image at $mountPath with error: $_"
|
|
}
|
|
}
|
|
}
|
|
|
|
# Remove Mount folder if it exists
|
|
if (Test-Path -Path "$FFUDevelopmentPath\Mount") {
|
|
WriteLog "Remove $FFUDevelopmentPath\Mount folder"
|
|
Remove-Item -Path "$FFUDevelopmentPath\Mount" -Recurse -Force
|
|
WriteLog 'Folder removed'
|
|
}
|
|
|
|
#Clear any corrupt Windows mount points
|
|
WriteLog 'Clearing any corrupt Windows mount points'
|
|
Clear-WindowsCorruptMountPoint | Out-null
|
|
WriteLog 'Complete'
|
|
|
|
#Clean up registry
|
|
if (Test-Path -Path 'HKLM:\FFU') {
|
|
Writelog 'Found HKLM:\FFU, removing it'
|
|
Invoke-Process reg "unload HKLM\FFU" | Out-Null
|
|
}
|
|
|
|
#Remove FFU User and Share
|
|
$UserExists = Get-LocalUser -Name $UserName -ErrorAction SilentlyContinue
|
|
if ($UserExists) {
|
|
WriteLog "Removing FFU User and Share"
|
|
Remove-FFUUserShare
|
|
WriteLog 'Removal complete'
|
|
}
|
|
if ($RemoveApps) {
|
|
WriteLog "Removing Apps in $AppsPath"
|
|
Remove-Apps
|
|
}
|
|
#Remove updates
|
|
if ($RemoveUpdates) {
|
|
WriteLog "Removing updates"
|
|
Remove-Updates
|
|
}
|
|
#Clean up $KBPath
|
|
If (Test-Path -Path $KBPath) {
|
|
WriteLog "Removing $KBPath"
|
|
Remove-Item -Path $KBPath -Recurse -Force -ErrorAction SilentlyContinue
|
|
WriteLog 'Removal complete'
|
|
}
|
|
Writelog 'Removing dirty.txt file'
|
|
Remove-Item -Path "$FFUDevelopmentPath\dirty.txt" -Force
|
|
WriteLog "Cleanup complete"
|
|
}
|
|
function Remove-FFU {
|
|
#Remove all FFU files in the FFUCaptureLocation
|
|
WriteLog "Removing all FFU files in $FFUCaptureLocation"
|
|
Remove-Item -Path $FFUCaptureLocation\*.ffu -Force
|
|
WriteLog "Removal complete"
|
|
}
|
|
|
|
Function Remove-Updates {
|
|
if ($UpdateLatestDefender) {
|
|
#Clean up $installDefenderPath
|
|
WriteLog "Removing $installDefenderPath"
|
|
If (Test-Path -Path $installDefenderPath) {
|
|
Remove-Item -Path $installDefenderPath -Force -ErrorAction SilentlyContinue
|
|
WriteLog 'Removal complete'
|
|
}
|
|
#Clean up $DefenderPath
|
|
If (Test-Path -Path $DefenderPath) {
|
|
WriteLog "Removing $DefenderPath"
|
|
Remove-Item -Path $DefenderPath -Recurse -Force -ErrorAction SilentlyContinue
|
|
WriteLog 'Removal complete'
|
|
}
|
|
}
|
|
if ($UpdateLatestMSRT) {
|
|
# Clean up Update-MSRT.ps1
|
|
WriteLog "Removing $installMSRTPath"
|
|
If (Test-Path -Path $installMSRTPath) {
|
|
Remove-Item -Path $installMSRTPath -Force -ErrorAction SilentlyContinue
|
|
WriteLog 'Removal complete'
|
|
}
|
|
#Clean up $MSRTPath
|
|
If (Test-Path -Path $MSRTPath) {
|
|
WriteLog "Removing $MSRTPath"
|
|
Remove-Item -Path $MSRTPath -Recurse -Force -ErrorAction SilentlyContinue
|
|
WriteLog 'Removal complete'
|
|
}
|
|
}
|
|
if ($UpdateOneDrive) {
|
|
# Clean up Update-OneDrive.ps1
|
|
WriteLog "Removing $installODPath"
|
|
If (Test-Path -Path $installODPath) {
|
|
Remove-Item -Path $installODPath -Force -ErrorAction SilentlyContinue
|
|
WriteLog 'Removal complete'
|
|
}
|
|
#Clean up $OneDrivePath
|
|
If (Test-Path -Path $OneDrivePath) {
|
|
WriteLog "Removing $OneDrivePath"
|
|
Remove-Item -Path $OneDrivePath -Recurse -Force -ErrorAction SilentlyContinue
|
|
WriteLog 'Removal complete'
|
|
}
|
|
}
|
|
if ($UpdateEdge) {
|
|
# Clean up Update-Edge.ps1
|
|
WriteLog "Removing $installEdgePath"
|
|
If (Test-Path -Path $installEdgePath) {
|
|
Remove-Item -Path $installEdgePath -Force -ErrorAction SilentlyContinue
|
|
WriteLog 'Removal complete'
|
|
}
|
|
#Clean up $EdgePath
|
|
If (Test-Path -Path $EdgePath) {
|
|
WriteLog "Removing $EdgePath"
|
|
Remove-Item -Path $EdgePath -Recurse -Force -ErrorAction SilentlyContinue
|
|
WriteLog 'Removal complete'
|
|
}
|
|
}
|
|
|
|
}
|
|
function Remove-Apps {
|
|
|
|
# Check if the file exists before attempting to clear it
|
|
if (Test-Path -Path $wingetWin32jsonFile) {
|
|
WriteLog "Removing $wingetWin32jsonFile"
|
|
Remove-Item -Path $wingetWin32jsonFile -Force -ErrorAction SilentlyContinue
|
|
WriteLog 'Removal complete'
|
|
}
|
|
# Clean up Win32 and MSStore folders
|
|
if (Test-Path -Path "$AppsPath\Win32" -PathType Container) {
|
|
WriteLog "Cleaning up Win32 folder"
|
|
Remove-Item -Path "$AppsPath\Win32" -Recurse -Force
|
|
}
|
|
if (Test-Path -Path "$AppsPath\MSStore" -PathType Container) {
|
|
WriteLog "Cleaning up MSStore folder"
|
|
Remove-Item -Path "$AppsPath\MSStore" -Recurse -Force
|
|
}
|
|
|
|
#Remove the Office Download and ODT
|
|
if ($InstallOffice) {
|
|
$ODTPath = "$AppsPath\Office"
|
|
$OfficeDownloadPath = "$ODTPath\Office"
|
|
WriteLog 'Removing Office and ODT download'
|
|
Remove-Item -Path $OfficeDownloadPath -Recurse -Force
|
|
Remove-Item -Path "$ODTPath\setup.exe"
|
|
Remove-Item -Path "$orchestrationPath\Install-Office.ps1"
|
|
WriteLog 'Removal complete'
|
|
}
|
|
|
|
#Remove AppsISO
|
|
if ($CleanupAppsISO) {
|
|
WriteLog "Removing $AppsISO"
|
|
Remove-Item -Path $AppsISO -Force -ErrorAction SilentlyContinue
|
|
WriteLog 'Removal complete'
|
|
}
|
|
}
|
|
function Export-ConfigFile {
|
|
[CmdletBinding()]
|
|
param (
|
|
[Parameter()]
|
|
$paramNames
|
|
)
|
|
$filteredParamNames = Get-Parameters -ParamNames $paramNames
|
|
|
|
# Retrieve their values
|
|
$paramsToExport = @{}
|
|
foreach ($paramName in $filteredParamNames) {
|
|
$paramsToExport[$paramName] = Get-Variable -Name $paramName -ValueOnly
|
|
}
|
|
|
|
# Sort the keys alphabetically
|
|
$orderedParams = [ordered]@{}
|
|
foreach ($key in ($paramsToExport.Keys | Sort-Object)) {
|
|
$orderedParams[$key] = $paramsToExport[$key]
|
|
}
|
|
|
|
# Convert to JSON and save
|
|
$orderedParams | ConvertTo-Json | Out-File $ExportConfigFile -Force
|
|
}
|
|
function Get-PEArchitecture {
|
|
param(
|
|
[string]$FilePath
|
|
)
|
|
|
|
# Read the entire file as bytes.1
|
|
$bytes = [System.IO.File]::ReadAllBytes($FilePath)
|
|
|
|
# Check for the 'MZ' signature.
|
|
if ($bytes[0] -ne 0x4D -or $bytes[1] -ne 0x5A) {
|
|
throw "The file is not a valid PE file."
|
|
}
|
|
|
|
# The PE header offset is stored at offset 0x3C.
|
|
$peHeaderOffset = [System.BitConverter]::ToInt32($bytes, 0x3C)
|
|
|
|
# Verify the PE signature "PE\0\0".
|
|
if ($bytes[$peHeaderOffset] -ne 0x50 -or $bytes[$peHeaderOffset + 1] -ne 0x45) {
|
|
throw "Invalid PE header."
|
|
}
|
|
|
|
# The Machine field is located immediately after the PE signature.
|
|
$machine = [System.BitConverter]::ToUInt16($bytes, $peHeaderOffset + 4)
|
|
|
|
switch ($machine) {
|
|
0x014c { return "x86" }
|
|
0x8664 { return "x64" }
|
|
0xAA64 { return "ARM64" }
|
|
default { return ("Unknown architecture: 0x{0:X}" -f $machine) }
|
|
}
|
|
}
|
|
|
|
###END FUNCTIONS
|
|
|
|
|
|
#Remove old log file if found
|
|
if (Test-Path -Path $Logfile) {
|
|
Remove-item -Path $LogFile -Force
|
|
}
|
|
|
|
$startTime = Get-Date
|
|
Write-Host "FFU build process started at" $startTime
|
|
Write-Host "This process can take 20 minutes or more. Please do not close this window or any additional windows that pop up"
|
|
Write-Host "To track progress, please open the log file $Logfile or use the -Verbose parameter next time"
|
|
|
|
WriteLog 'Begin Logging'
|
|
|
|
####### Generate Config File #######
|
|
|
|
if ($ExportConfigFile) {
|
|
WriteLog 'Exporting Config File'
|
|
# Get the parameter names from the script and exclude ExportConfigFile
|
|
$paramNames = $MyInvocation.MyCommand.Parameters.Keys | Where-Object { $_ -ne 'ExportConfigFile' }
|
|
try {
|
|
Export-ConfigFile($paramNames)
|
|
WriteLog "Config file exported to $ExportConfigFile"
|
|
}
|
|
catch {
|
|
WriteLog 'Failed to export config file'
|
|
throw $_
|
|
}
|
|
}
|
|
|
|
####### End Generate Config File #######
|
|
|
|
|
|
#Setting long path support - this prevents issues where some applications have deep directory structures
|
|
#and oscdimg fails to create the Apps ISO
|
|
try {
|
|
$LongPathsEnabled = Get-ItemPropertyValue -Path 'HKLM:\SYSTEM\CurrentControlSet\Control\FileSystem' -Name 'LongPathsEnabled' -ErrorAction Stop
|
|
}
|
|
catch {
|
|
$LongPathsEnabled = $null
|
|
}
|
|
if ($LongPathsEnabled -ne 1) {
|
|
WriteLog 'LongPathsEnabled is not set to 1. Setting it to 1'
|
|
Set-ItemProperty -Path 'HKLM:\SYSTEM\CurrentControlSet\Control\FileSystem' -Name 'LongPathsEnabled' -Value 1
|
|
WriteLog 'LongPathsEnabled set to 1'
|
|
}
|
|
|
|
|
|
###PARAMETER VALIDATION
|
|
|
|
#Validate drivers folder
|
|
if ($InstallDrivers -or $CopyDrivers) {
|
|
WriteLog 'Doing driver validation'
|
|
if ($DriversFolder -match '\s') {
|
|
WriteLog "Driver folder path $DriversFolder contains spaces. Please remove spaces from the path and try again."
|
|
throw "Driver folder path $DriversFolder contains spaces. Please remove spaces from the path and try again."
|
|
}
|
|
if ($Make -and $Model) {
|
|
WriteLog "Make and Model are set to $Make and $Model, will attempt to download drivers"
|
|
}
|
|
elseif ($DriversJsonPath -and (Test-Path -Path $DriversJsonPath)) {
|
|
WriteLog "Drivers JSON path is set to $DriversJsonPath, will attempt to download drivers"
|
|
}
|
|
else {
|
|
if (!(Test-Path -Path $DriversFolder)) {
|
|
WriteLog "-InstallDrivers or -CopyDrivers is set to `$true, but the $DriversFolder folder is missing"
|
|
throw "-InstallDrivers or -CopyDrivers is set to `$true, but the $DriversFolder folder is missing"
|
|
}
|
|
if ((Get-ChildItem -Path $DriversFolder -Recurse | Measure-Object -Property Length -Sum).Sum -lt 1MB) {
|
|
WriteLog "-InstallDrivers or -CopyDrivers is set to `$true, but the $DriversFolder folder is empty"
|
|
throw "-InstallDrivers or -CopyDrivers is set to `$true, but the $DriversFolder folder is empty"
|
|
}
|
|
if (!(Test-Path -Path $DriversJsonPath)) {
|
|
WriteLog "-InstallDrivers or -CopyDrivers is set to `$true, but the $DriversJsonPath file is missing"
|
|
throw "-InstallDrivers or -CopyDrivers is set to `$true, but the $DriversJsonPath file is missing"
|
|
}
|
|
WriteLog 'Driver validation complete'
|
|
}
|
|
}
|
|
#Validate PEDrivers folder
|
|
if ($CopyPEDrivers) {
|
|
WriteLog 'Doing PEDriver validation'
|
|
# Check if $PEdriversFolder has spaces in the path, if it does, throw an error
|
|
if ($PEDriversFolder -match '\s') {
|
|
WriteLog "Driver folder path $PEDriversFolder contains spaces. Please remove spaces from the path and try again."
|
|
throw "Driver folder path $PEDriversFolder contains spaces. Please remove spaces from the path and try again."
|
|
}
|
|
if (!(Test-Path -Path $PEDriversFolder)) {
|
|
WriteLog "-CopyPEDrivers is set to `$true, but the $PEDriversFolder folder is missing"
|
|
throw "-CopyPEDrivers is set to `$true, but the $PEDriversFolder folder is missing"
|
|
}
|
|
if ((Get-ChildItem -Path $PEDriversFolder -Recurse | Measure-Object -Property Length -Sum).Sum -lt 1MB) {
|
|
WriteLog "-CopyPEDrivers is set to `$true, but the $PEDriversFolder folder is empty"
|
|
throw "-CopyPEDrivers is set to `$true, but the $PEDriversFolder folder is empty"
|
|
}
|
|
WriteLog 'PEDriver validation complete'
|
|
}
|
|
|
|
#Validate PPKG folder
|
|
if ($CopyPPKG) {
|
|
WriteLog 'Doing PPKG validation'
|
|
if (!(Test-Path -Path $PPKGFolder)) {
|
|
WriteLog "-CopyPPKG is set to `$true, but the $PPKGFolder folder is missing"
|
|
throw "-CopyPPKG is set to `$true, but the $PPKGFolder folder is missing"
|
|
}
|
|
#Check for at least one .PPKG file
|
|
if (!(Get-ChildItem -Path $PPKGFolder -Filter *.ppkg)) {
|
|
WriteLog "-CopyPPKG is set to `$true, but the $PPKGFolder folder is missing a .PPKG file"
|
|
throw "-CopyPPKG is set to `$true, but the $PPKGFolder folder is missing a .PPKG file"
|
|
}
|
|
WriteLog 'PPKG validation complete'
|
|
}
|
|
|
|
#Validate Autopilot folder
|
|
if ($CopyAutopilot) {
|
|
WriteLog 'Doing Autopilot validation'
|
|
if (!(Test-Path -Path $AutopilotFolder)) {
|
|
WriteLog "-CopyAutopilot is set to `$true, but the $AutopilotFolder folder is missing"
|
|
throw "-CopyAutopilot is set to `$true, but the $AutopilotFolder folder is missing"
|
|
}
|
|
#Check for .JSON file
|
|
if (!(Get-ChildItem -Path $AutopilotFolder -Filter *.json)) {
|
|
WriteLog "-CopyAutopilot is set to `$true, but the $AutopilotFolder folder is missing a .JSON file"
|
|
throw "-CopyAutopilot is set to `$true, but the $AutopilotFolder folder is missing a .JSON file"
|
|
}
|
|
WriteLog 'Autopilot validation complete'
|
|
}
|
|
|
|
#Validate Unattend folder
|
|
if ($CopyUnattend) {
|
|
WriteLog 'Doing Unattend validation'
|
|
if (!(Test-Path -Path $UnattendFolder)) {
|
|
WriteLog "-CopyUnattend is set to `$true, but the $UnattendFolder folder is missing"
|
|
throw "-CopyUnattend is set to `$true, but the $UnattendFolder folder is missing"
|
|
}
|
|
#Check for .XML file
|
|
if (!(Get-ChildItem -Path $UnattendFolder -Filter unattend_*.xml)) {
|
|
WriteLog "-CopyUnattend is set to `$true, but the $UnattendFolder folder is missing a .XML file"
|
|
throw "-CopyUnattend is set to `$true, but the $UnattendFolder folder is missing a .XML file"
|
|
}
|
|
WriteLog 'Unattend validation complete'
|
|
}
|
|
|
|
#Override $InstallApps value if using ESD to build FFU. This is due to a strange issue where building the FFU
|
|
#from vhdx doesn't work (you get an older style OOBE screen and get stuck in an OOBE reboot loop when hitting next).
|
|
#This behavior doesn't happen with WIM files.
|
|
If (-not ($ISOPath) -and (-not ($InstallApps))) {
|
|
$InstallApps = $true
|
|
WriteLog "Script will download Windows media. Setting `$InstallApps to `$true to build VM to capture FFU. Must do this when using MCT ESD."
|
|
}
|
|
|
|
if (($InstallOffice -eq $true) -and ($InstallApps -eq $false)) {
|
|
throw "If variable InstallOffice is set to `$true, InstallApps must also be set to `$true."
|
|
}
|
|
if (($InstallApps -and ($VMSwitchName -eq ''))) {
|
|
throw "If variable InstallApps is set to `$true, VMSwitchName must also be set to capture the FFU. Please set -VMSwitchName and try again."
|
|
}
|
|
|
|
if (($InstallApps -and ($VMHostIPAddress -eq ''))) {
|
|
throw "If variable InstallApps is set to `$true, VMHostIPAddress must also be set to capture the FFU. Please set -VMHostIPAddress and try again."
|
|
}
|
|
|
|
if (($VMHostIPAddress) -and ($VMSwitchName)) {
|
|
WriteLog "Validating -VMSwitchName $VMSwitchName and -VMHostIPAddress $VMHostIPAddress"
|
|
#Check $VMSwitchName by using Get-VMSwitch
|
|
$VMSwitch = Get-VMSwitch -Name $VMSwitchName -ErrorAction SilentlyContinue
|
|
if (-not $VMSwitch) {
|
|
throw "-VMSwitchName $VMSwitchName not found. Please check the -VMSwitchName parameter and try again."
|
|
}
|
|
#Find the IP address of $VMSwitch and check if it matches $VMHostIPAddress
|
|
$interfaceAlias = "vEthernet ($VMSwitchName)"
|
|
$VMSwitchIPAddress = (Get-NetIPAddress -InterfaceAlias $interfaceAlias -AddressFamily 'IPv4' -ErrorAction SilentlyContinue).IPAddress
|
|
if (-not $VMSwitchIPAddress) {
|
|
throw "IP address for -VMSwitchName $VMSwitchName not found. Please check the -VMSwitchName parameter and try again."
|
|
}
|
|
if ($VMSwitchIPAddress -ne $VMHostIPAddress) {
|
|
try {
|
|
# Bypass the check for systems that could have a Hyper-V NAT switch
|
|
$null = Get-NetNat -ErrorAction Stop
|
|
$NetNat = @(Get-NetNat -ErrorAction Stop)
|
|
}
|
|
catch {
|
|
throw "IP address for -VMSwitchName $VMSwitchName is $VMSwitchIPAddress, which does not match the -VMHostIPAddress $VMHostIPAddress. Please check the -VMHostIPAddress parameter and try again."
|
|
}
|
|
if ($NetNat.Count -gt 0) {
|
|
WriteLog "IP address for -VMSwitchName $VMSwitchName is $VMSwitchIPAddress, which does not match the -VMHostIPAddress $VMHostIPAddress!"
|
|
WriteLog "NAT setup detected, remember to configure NATing if the FFU image can't be captured to the network share on the host."
|
|
}
|
|
else {
|
|
throw "IP address for -VMSwitchName $VMSwitchName is $VMSwitchIPAddress, which does not match the -VMHostIPAddress $VMHostIPAddress. Please check the -VMHostIPAddress parameter and try again."
|
|
}
|
|
}
|
|
WriteLog '-VMSwitchName and -VMHostIPAddress validation complete'
|
|
}
|
|
|
|
if (-not ($ISOPath) -and ($OptionalFeatures -like '*netfx3*')) {
|
|
throw "netfx3 specified as an optional feature, however Windows ISO isn't defined. Unable to get netfx3 source files from downloaded ESD media. Please specify a Windows ISO in the ISOPath parameter."
|
|
}
|
|
if (($LogicalSectorSizeBytes -eq 4096) -and ($installdrivers -eq $true)) {
|
|
$installdrivers = $false
|
|
$CopyDrivers = $true
|
|
WriteLog 'LogicalSectorSizeBytes is set to 4096, which is not supported for driver injection. Setting $installdrivers to $false'
|
|
WriteLog 'As a workaround, setting -copydrivers $true to copy drivers to the deploy partition drivers folder'
|
|
WriteLog 'We are investigating this issue and will update the script if/when we have a fix'
|
|
}
|
|
if ($BuildUSBDrive -eq $true) {
|
|
$USBDrives, $USBDrivesCount = Get-USBDrive
|
|
}
|
|
if (($InstallApps -eq $false) -and (($UpdateLatestDefender -eq $true) -or ($UpdateOneDrive -eq $true) -or ($UpdateEdge -eq $true) -or ($UpdateLatestMSRT -eq $true))) {
|
|
WriteLog 'You have selected to update Defender, Malicious Software Removal Tool, OneDrive, or Edge, however you are setting InstallApps to false. These updates require the InstallApps variable to be set to true. Please set InstallApps to true and try again.'
|
|
throw "InstallApps variable must be set to `$true to update Defender, OneDrive, or Edge"
|
|
}
|
|
if (($WindowsArch -eq 'ARM64') -and ($InstallOffice -eq $true)) {
|
|
$InstallOffice = $false
|
|
WriteLog 'M365 Apps/Office currently fails to install on ARM64 VMs without an internet connection. Setting InstallOffice to false'
|
|
}
|
|
|
|
if (($WindowsArch -eq 'ARM64') -and ($UpdateOneDrive -eq $true)) {
|
|
$UpdateOneDrive = $false
|
|
WriteLog 'OneDrive currently fails to install on ARM64 VMs (even with the OneDrive ARM setup files). Setting UpdateOneDrive to false'
|
|
}
|
|
|
|
if (($WindowsArch -eq 'ARM64') -and ($UpdateLatestMSRT -eq $true)) {
|
|
$UpdateLatestMSRT = $false
|
|
WriteLog 'Windows Malicious Software Removal Tool is not available for the ARM64 architecture.'
|
|
}
|
|
#If downloading ESD from MCT, hardcode WindowsVersion to 22H2 for Windows 10 and 24H2 for Windows 11
|
|
#MCT media only provides 22H2 and 24H2 media
|
|
#This prevents issues with VHDX Caching unecessarily and with searching for CUs
|
|
if ($ISOPath -eq '') {
|
|
if ($WindowsRelease -eq '10') {
|
|
$WindowsVersion = '22H2'
|
|
}
|
|
if ($WindowsRelease -eq '11') {
|
|
$WindowsVersion = '24H2'
|
|
}
|
|
}
|
|
|
|
###END PARAMETER VALIDATION
|
|
|
|
#Get script variable values
|
|
LogVariableValues
|
|
|
|
#Check if environment is dirty
|
|
If (Test-Path -Path "$FFUDevelopmentPath\dirty.txt") {
|
|
Get-FFUEnvironment
|
|
}
|
|
WriteLog 'Creating dirty.txt file'
|
|
New-Item -Path .\ -Name "dirty.txt" -ItemType "file" | Out-Null
|
|
|
|
#Get drivers first since user could be prompted for additional info
|
|
if ($driversJsonPath -and (Test-Path $driversJsonPath) -and ($InstallDrivers -or $CopyDrivers)) {
|
|
WriteLog "Processing drivers from JSON file: $driversJsonPath"
|
|
Import-Module "$PSScriptRoot\FFUUI.Core\FFUUI.Core.psm1"
|
|
# FFU.Common.Drivers.psm1 is imported by FFUUI.Core.psm1
|
|
|
|
$driversToProcess = @()
|
|
$jsonData = Get-Content -Path $driversJsonPath -Raw | ConvertFrom-Json
|
|
|
|
foreach ($makeEntry in $jsonData.PSObject.Properties) {
|
|
$makeName = $makeEntry.Name
|
|
if ($makeEntry.Value.PSObject.Properties['Models']) {
|
|
foreach ($modelEntry in $makeEntry.Value.Models) {
|
|
# Construct the PSCustomObject exactly as the Save-*DriversTask functions expect $DriverItemData
|
|
$driverItem = [PSCustomObject]@{
|
|
Make = $makeName
|
|
Model = $modelEntry.Name # This is the display name, e.g., "Surface Book 3" or "Lenovo 500w (83LH)"
|
|
Link = if ($modelEntry.PSObject.Properties['Link']) { $modelEntry.Link } else { $null }
|
|
ProductName = if ($modelEntry.PSObject.Properties['ProductName']) { $modelEntry.ProductName } else { $null } # Specifically for Lenovo
|
|
MachineType = if ($modelEntry.PSObject.Properties['MachineType']) { $modelEntry.MachineType } else { $null } # Specifically for Lenovo
|
|
# Ensure all properties potentially accessed by any Save-*DriversTask via $DriverItemData are present
|
|
}
|
|
$driversToProcess += $driverItem
|
|
}
|
|
}
|
|
}
|
|
|
|
if ($driversToProcess.Count -eq 0) {
|
|
WriteLog "No drivers found to process in $driversJsonPath."
|
|
}
|
|
else {
|
|
WriteLog "Found $($driversToProcess.Count) driver entries to process from $driversJsonPath."
|
|
|
|
$taskArguments = @{
|
|
DriversFolder = $DriversFolder
|
|
WindowsRelease = $WindowsRelease
|
|
WindowsArch = $WindowsArch
|
|
WindowsVersion = $WindowsVersion
|
|
Headers = $Headers
|
|
UserAgent = $UserAgent
|
|
CompressToWim = $CompressDownloadedDriversToWim
|
|
}
|
|
|
|
WriteLog "Starting parallel driver processing using Invoke-ParallelProcessing..."
|
|
$parallelResults = Invoke-ParallelProcessing -ItemsToProcess $driversToProcess `
|
|
-TaskType 'DownloadDriverByMake' `
|
|
-TaskArguments $taskArguments `
|
|
-IdentifierProperty 'Model' `
|
|
-WindowObject $null `
|
|
-ListViewControl $null `
|
|
-MainThreadLogPath $LogFile
|
|
|
|
# After processing, update the driver mapping file
|
|
$successfullyDownloaded = [System.Collections.Generic.List[PSCustomObject]]::new()
|
|
if ($null -ne $parallelResults) {
|
|
# Create a lookup table from the original items to get the 'Make'
|
|
$makeLookup = @{}
|
|
$driversToProcess | ForEach-Object { $makeLookup[$_.Model] = $_.Make }
|
|
|
|
# Filter for objects that could be results, avoiding stray log strings
|
|
foreach ($result in ($parallelResults | Where-Object { $_ -is [hashtable] })) {
|
|
if ($null -eq $result) { continue }
|
|
|
|
# The result from Invoke-ParallelProcessing is a hashtable.
|
|
# Access properties using their keys.
|
|
$modelName = $result['Identifier']
|
|
$resultCode = $result['ResultCode']
|
|
$driverPath = $result['DriverPath']
|
|
|
|
if ([string]::IsNullOrWhiteSpace($modelName)) {
|
|
WriteLog "Could not determine model name from result object: $($result | ConvertTo-Json -Compress -Depth 3)"
|
|
continue
|
|
}
|
|
|
|
if ($resultCode -eq 0 -and -not [string]::IsNullOrWhiteSpace($driverPath)) {
|
|
# The task was successful and returned a driver path.
|
|
$makeJson = $makeLookup[$modelName]
|
|
if ($makeJson) {
|
|
$successfullyDownloaded.Add([PSCustomObject]@{
|
|
Make = $makeJson
|
|
Model = $modelName
|
|
DriverPath = $driverPath
|
|
})
|
|
}
|
|
else {
|
|
WriteLog "Warning: Could not find 'Make' for successful download of model '$modelName'. Skipping from DriverMapping.json."
|
|
}
|
|
}
|
|
else {
|
|
$logMessage = "Driver download failed or did not return a path for model '$modelName'. Status: $($result['Status'])"
|
|
WriteLog $logMessage
|
|
Write-Warning $logMessage
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
WriteLog "Invoke-ParallelProcessing returned null or no results."
|
|
}
|
|
|
|
# Update the driver mapping JSON if there are any successful downloads
|
|
if ($successfullyDownloaded.Count -gt 0) {
|
|
try {
|
|
WriteLog "Updating DriverMapping.json with $($successfullyDownloaded.Count) successfully downloaded drivers."
|
|
Update-DriverMappingJson -DownloadedDrivers $successfullyDownloaded -DriversFolder $DriversFolder
|
|
}
|
|
catch {
|
|
WriteLog "Warning: Failed to update DriverMapping.json: $($_.Exception.Message)"
|
|
# This is not a fatal error for the build process itself, so just show a warning.
|
|
Write-Warning "The driver download process completed, but failed to update the DriverMapping.json file. Please check the log for details."
|
|
}
|
|
}
|
|
WriteLog "Finished processing drivers from $driversJsonPath."
|
|
|
|
# After processing, update the driver mapping file
|
|
$successfullyDownloaded = [System.Collections.Generic.List[PSCustomObject]]::new()
|
|
if ($null -ne $parallelResults) {
|
|
# Create a lookup table from the original items to get the 'Make'
|
|
$makeLookup = @{}
|
|
$driversToProcess | ForEach-Object { $makeLookup[$_.Model] = $_.Make }
|
|
|
|
foreach ($result in $parallelResults) {
|
|
if ($null -ne $result) {
|
|
# Collect successful results for driver mapping
|
|
if ($result.PSObject.Properties['Success'] -and $result.Success -and $result.PSObject.Properties['DriverPath'] -and -not [string]::IsNullOrWhiteSpace($result.DriverPath)) {
|
|
$modelName = if ($result.PSObject.Properties.Name -contains 'Identifier') { $result.Identifier } else { $result.Model }
|
|
|
|
# Find the 'Make' from the original list
|
|
$makeJson = $makeLookup[$modelName]
|
|
|
|
if ($makeJson) {
|
|
$successfullyDownloaded.Add([PSCustomObject]@{
|
|
Make = $makeJson
|
|
Model = $modelName
|
|
DriverPath = $result.DriverPath
|
|
})
|
|
}
|
|
else {
|
|
WriteLog "Warning: Could not find 'Make' for successful download of model '$modelName'. Skipping from DriverMapping.json."
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
# Update the driver mapping JSON if there are any successful downloads
|
|
if ($successfullyDownloaded.Count -gt 0) {
|
|
try {
|
|
WriteLog "Updating DriverMapping.json with $($successfullyDownloaded.Count) successfully downloaded drivers."
|
|
Update-DriverMappingJson -DownloadedDrivers $successfullyDownloaded -DriversFolder $DriversFolder
|
|
}
|
|
catch {
|
|
WriteLog "Warning: Failed to update DriverMapping.json: $($_.Exception.Message)"
|
|
# This is not a fatal error for the build process itself, so just show a warning.
|
|
Write-Warning "The driver download process completed, but failed to update the DriverMapping.json file. Please check the log for details."
|
|
}
|
|
}
|
|
}
|
|
}
|
|
# Existing single-model driver download logic
|
|
elseif (($Make -and $Model) -and ($InstallDrivers -or $CopyDrivers)) {
|
|
try {
|
|
if ($Make -eq 'HP') {
|
|
WriteLog 'Getting HP drivers'
|
|
Get-HPDrivers -Make $Make -Model $Model -WindowsArch $WindowsArch -WindowsRelease $WindowsRelease -WindowsVersion $WindowsVersion
|
|
WriteLog 'Getting HP drivers completed successfully'
|
|
}
|
|
if ($make -eq 'Microsoft') {
|
|
WriteLog 'Getting Microsoft drivers'
|
|
Get-MicrosoftDrivers -Make $Make -Model $Model -WindowsArch $WindowsArch -WindowsRelease $WindowsRelease
|
|
WriteLog 'Getting Microsoft drivers completed successfully'
|
|
}
|
|
if ($make -eq 'Lenovo') {
|
|
WriteLog 'Getting Lenovo drivers'
|
|
Get-LenovoDrivers -Model $Model -WindowsArch $WindowsArch -WindowsRelease $WindowsRelease
|
|
WriteLog 'Getting Lenovo drivers completed successfully'
|
|
}
|
|
if ($make -eq 'Dell') {
|
|
WriteLog 'Getting Dell drivers'
|
|
#Dell mixes Win10 and 11 drivers, hence no WindowsRelease parameter
|
|
Get-DellDrivers -Model $Model -WindowsArch $WindowsArch -WindowsRelease $WindowsRelease
|
|
WriteLog 'Getting Dell drivers completed successfully'
|
|
}
|
|
}
|
|
catch {
|
|
Writelog "Getting drivers failed with error $_"
|
|
throw $_
|
|
}
|
|
}
|
|
|
|
|
|
#Get Windows ADK
|
|
try {
|
|
$adkPath = Get-ADK
|
|
#Need to use the Deployment and Imaging tools environment to use dism from the Sept 2023 ADK to optimize FFU
|
|
$DandIEnv = Join-Path $adkPath "Assessment and Deployment Kit\Deployment Tools\DandISetEnv.bat"
|
|
|
|
}
|
|
catch {
|
|
WriteLog 'ADK not found'
|
|
throw $_
|
|
}
|
|
|
|
#Create apps ISO for Office and/or 3rd party apps
|
|
if ($InstallApps) {
|
|
if (Test-Path -Path $AppsISO) {
|
|
WriteLog "Apps ISO exists at: $AppsISO"
|
|
WriteLog "Will use existing ISO"
|
|
}
|
|
else {
|
|
try {
|
|
#Check for and download WinGet applications
|
|
if (Test-Path -Path $AppListPath) {
|
|
$appList = Get-Content -Path $AppListPath -Raw | ConvertFrom-Json
|
|
|
|
WriteLog 'Checking for previously downloaded Winget applications'
|
|
|
|
# Initialize variables
|
|
$missingApps = @()
|
|
$existingMSStoreApps = @()
|
|
$hasExistingApps = $false
|
|
|
|
# Check for WinGetWin32Apps.json
|
|
$wingetAppsJson = $null
|
|
if (Test-Path -Path $wingetWin32jsonFile) {
|
|
WriteLog "$wingetWin32jsonFile found"
|
|
$wingetAppsJson = Get-Content -Path $wingetWin32jsonFile -Raw | ConvertFrom-Json
|
|
$hasExistingApps = $true
|
|
}
|
|
|
|
# Check MSStore folder for existing apps
|
|
if (Test-Path -Path "$AppsPath\MSStore") {
|
|
WriteLog "$AppsPath\MSStore folder found"
|
|
|
|
# Get root folder names in MSStore directory
|
|
$MSStoreFolder = Get-ChildItem -Path "$AppsPath\MSStore" -Directory
|
|
|
|
# Check content size of each folder
|
|
foreach ($folder in $MSStoreFolder) {
|
|
$folderSize = (Get-ChildItem -Path $folder.FullName -Recurse | Measure-Object -Property Length -Sum).Sum
|
|
if ($folderSize -gt 1MB) {
|
|
$existingMSStoreApps += $folder.Name
|
|
$hasExistingApps = $true
|
|
}
|
|
}
|
|
}
|
|
|
|
# If there are no existing apps, use the original AppList.json directly
|
|
if (-not $hasExistingApps) {
|
|
WriteLog "No existing applications found. Using original AppList.json for all apps."
|
|
Get-Apps -AppList $AppListPath
|
|
}
|
|
else {
|
|
# Compare apps in AppList.json with existing installations
|
|
foreach ($app in $appList.apps) {
|
|
$appFound = $false
|
|
|
|
# Check Win32 apps regardless of source
|
|
if ($wingetAppsJson) {
|
|
$wingetApp = $wingetAppsJson | Where-Object { $_.Name -eq $app.name }
|
|
|
|
if ($wingetApp) {
|
|
# Verify content exists in Win32 folder
|
|
$appFolder = Join-Path -Path "$AppsPath\Win32" -ChildPath $app.name
|
|
if (Test-Path -Path $appFolder) {
|
|
$folderSize = (Get-ChildItem -Path $appFolder -Recurse | Measure-Object -Property Length -Sum).Sum
|
|
if ($folderSize -gt 1MB) {
|
|
$appFound = $true
|
|
WriteLog "Found existing Win32 app: $($app.name)"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
# If not found in Win32, check MSStore folder regardless of source
|
|
if (-not $appFound) {
|
|
if ($existingMSStoreApps -contains $app.name) {
|
|
$appFound = $true
|
|
WriteLog "Found existing MSStore app: $($app.name)"
|
|
}
|
|
}
|
|
|
|
# If app not found in either location, add to missing apps list
|
|
if (-not $appFound) {
|
|
$missingApps += $app
|
|
WriteLog "App not found, will download: $($app.name)"
|
|
}
|
|
}
|
|
|
|
# If missing apps found, create modified AppList.json
|
|
if ($missingApps.Count -gt 0) {
|
|
$modifiedAppList = @{
|
|
apps = $missingApps
|
|
}
|
|
|
|
$modifiedAppListPath = Join-Path -Path $AppsPath -ChildPath "ModifiedAppList.json"
|
|
$modifiedAppList | ConvertTo-Json | Set-Content -Path $modifiedAppListPath
|
|
WriteLog "Created ModifiedAppList.json with $($missingApps.Count) apps to download"
|
|
|
|
# Download missing apps
|
|
WriteLog "Downloading missing applications"
|
|
Get-Apps -AppList $modifiedAppListPath
|
|
|
|
# Cleanup modified app list
|
|
Remove-Item -Path $modifiedAppListPath -Force
|
|
}
|
|
else {
|
|
WriteLog "All applications already downloaded, skipping downloads"
|
|
}
|
|
}
|
|
}
|
|
# Check is UserAppList.json exists and output to the user which apps will be installed
|
|
# It's expected that the user will have already copied the applications and created the UserAppList.json file
|
|
if (Test-Path -Path $UserAppListPath) {
|
|
$userAppList = Get-Content -Path $UserAppListPath -Raw | ConvertFrom-Json
|
|
WriteLog "UserAppList.json found, the following apps will be installed:"
|
|
foreach ($app in $userAppList) {
|
|
WriteLog "$($app.name)"
|
|
}
|
|
}
|
|
|
|
#Install Office
|
|
if ($InstallOffice) {
|
|
#Check if Office has already been downloaded, if so, skip download
|
|
WriteLog 'Checking for M365 Apps/Office download'
|
|
$officeDataFolder = "$AppsPath\Office\Office\Data"
|
|
if (Test-Path -Path $officeDataFolder) {
|
|
# Check the size of the $officeDataFolder folder
|
|
$OfficeSize = (Get-ChildItem -Path $officeDataFolder -Recurse | Measure-Object -Property Length -Sum).Sum
|
|
if ($OfficeSize -gt 1MB) {
|
|
WriteLog "Found Office download in $officeDataFolder, skipping download"
|
|
}
|
|
else {
|
|
WriteLog 'Downloading M365 Apps/Office'
|
|
Get-Office
|
|
WriteLog 'Downloading M365 Apps/Office completed successfully'
|
|
}
|
|
|
|
}
|
|
else {
|
|
WriteLog 'Downloading M365 Apps/Office'
|
|
Get-Office
|
|
WriteLog 'Downloading M365 Apps/Office completed successfully'
|
|
}
|
|
|
|
}
|
|
|
|
#Update Latest Defender Platform and Definitions - these can't be serviced into the VHDX, will be saved to AppsPath
|
|
if ($UpdateLatestDefender) {
|
|
# Check if Defender has already been downloaded, if so, skip download
|
|
WriteLog "`$UpdateLatestDefender is set to true, checking for latest Defender Platform and Security updates"
|
|
if (Test-Path -Path $DefenderPath) {
|
|
# Check the size of the $DefenderPath folder
|
|
$DefenderSize = (Get-ChildItem -Path $DefenderPath -Recurse | Measure-Object -Property Length -Sum).Sum
|
|
if ($DefenderSize -gt 1MB) {
|
|
WriteLog "Found Defender download in $DefenderPath, skipping download"
|
|
$DefenderDownloaded = $true
|
|
}
|
|
}
|
|
if (-not $DefenderDownloaded) {
|
|
WriteLog "Creating $DefenderPath"
|
|
New-Item -Path $DefenderPath -ItemType Directory -Force | Out-Null
|
|
|
|
# Define array of updates to download
|
|
$defenderUpdates = @(
|
|
@{
|
|
Name = "Update for Microsoft Defender Antivirus antimalware platform"
|
|
Description = "Defender Platform"
|
|
},
|
|
@{
|
|
Name = "Windows Security Platform"
|
|
Description = "Windows Security Platform"
|
|
}
|
|
)
|
|
|
|
# Download each update
|
|
foreach ($update in $defenderUpdates) {
|
|
WriteLog "Searching for $($update.Name) from Microsoft Update Catalog and saving to $DefenderPath"
|
|
$KBFilePath = Save-KB -Name $update.Name -Path $DefenderPath
|
|
WriteLog "Latest $($update.Description) saved to $DefenderPath\$KBFilePath"
|
|
# Add the KB file path to the installDefenderCommand
|
|
$installDefenderCommand += "& d:\Defender\$KBFilePath`r`n"
|
|
}
|
|
|
|
# Download latest Defender Definitions
|
|
WriteLog "Downloading latest Defender Definitions"
|
|
# Defender def updates can be found https://www.microsoft.com/en-us/wdsi/defenderupdates
|
|
if ($WindowsArch -eq 'x64') {
|
|
$DefenderDefURL = 'https://go.microsoft.com/fwlink/?LinkID=121721&arch=x64'
|
|
}
|
|
if ($WindowsArch -eq 'ARM64') {
|
|
$DefenderDefURL = 'https://go.microsoft.com/fwlink/?LinkID=121721&arch=arm64'
|
|
}
|
|
try {
|
|
WriteLog "Defender definitions URL is $DefenderDefURL"
|
|
Start-BitsTransferWithRetry -Source $DefenderDefURL -Destination "$DefenderPath\mpam-fe.exe"
|
|
WriteLog "Defender Definitions downloaded to $DefenderPath\mpam-fe.exe"
|
|
$installDefenderCommand += "& d:\Defender\mpam-fe.exe"
|
|
}
|
|
catch {
|
|
Write-Host "Downloading Defender Definitions Failed"
|
|
WriteLog "Downloading Defender Definitions Failed with error $_"
|
|
throw $_
|
|
}
|
|
|
|
# Create Update-Defender.ps1
|
|
WriteLog "Creating $installDefenderPath"
|
|
Set-Content -Path $installDefenderPath -Value $installDefenderCommand -Force
|
|
if (Test-Path -Path $installDefenderPath) {
|
|
WriteLog "$installDefenderPath created successfully"
|
|
}
|
|
else {
|
|
WriteLog "$installDefenderPath failed to create"
|
|
throw "$installDefenderPath failed to create"
|
|
}
|
|
}
|
|
}
|
|
# Download latest MSRT
|
|
if ($UpdateLatestMSRT) {
|
|
WriteLog "`$UpdateLatestMSRT is set to true, checking for latest Windows Malicious Software Removal Tool"
|
|
# Check if MSRT has already been downloaded, if so, skip download
|
|
if (Test-Path -Path $MSRTPath) {
|
|
# Check the size of the $MSRTPath folder
|
|
$MSRTSize = (Get-ChildItem -Path $MSRTPath -Recurse | Measure-Object -Property Length -Sum).Sum
|
|
if ($MSRTSize -gt 1MB) {
|
|
WriteLog "Found MSRT download in $MSRTPath, skipping download"
|
|
$MSRTDownloaded = $true
|
|
}
|
|
}
|
|
if (-Not $MSRTDownloaded) {
|
|
# Create the search string for MSRT based on Windows architecture and release
|
|
if ($WindowsArch -eq 'x64') {
|
|
if ($installationType -eq 'client' -and (-not $isLTSC)) {
|
|
$Name = """Windows Malicious Software Removal Tool x64""" + " " + """Windows $WindowsRelease"""
|
|
}
|
|
# Handle LTSB/LTSC
|
|
elseif ($installationType -eq 'client' -and $isLTSC -eq $true) {
|
|
$Name = """Windows Malicious Software Removal Tool x64""" + " " + """LTSB"""
|
|
}
|
|
#Windows Server 2025 isn't listed as a product in the Microsoft Update Catalog, so we'll use the 2019 version
|
|
elseif ($installationType -eq 'server' -and $WindowsRelease -eq '24H2') {
|
|
$Name = """Windows Malicious Software Removal Tool x64""" + " " + """Windows Server 2019"""
|
|
}
|
|
else {
|
|
$Name = """Windows Malicious Software Removal Tool x64""" + " " + """Windows Server $WindowsRelease"""
|
|
}
|
|
}
|
|
if ($WindowsArch -eq 'x86') {
|
|
$Name = """Windows Malicious Software Removal Tool""" + " " + """Windows $WindowsRelease"""
|
|
}
|
|
#Check if $MSRTPath exists, if not, create it
|
|
if (-not (Test-Path -Path $MSRTPath)) {
|
|
WriteLog "Creating $MSRTPath"
|
|
New-Item -Path $MSRTPath -ItemType Directory -Force | Out-Null
|
|
}
|
|
WriteLog "Searching for $Name from Microsoft Update Catalog and saving to $MSRTPath"
|
|
WriteLog "Getting Windows Malicious Software Removal Tool URL"
|
|
$MSRTFileName = Save-KB -Name $Name -Path $MSRTPath
|
|
WriteLog "Latest Windows Malicious Software Removal Tool saved to $MSRTPath\$MSRTFileName"
|
|
|
|
# Create Update-MSRT.ps1
|
|
$installMSRTPath = Join-Path -Path $orchestrationPath -ChildPath "Update-MSRT.ps1"
|
|
WriteLog "Creating $installMSRTPath"
|
|
$installMSRTCommand = "& d:\MSRT\$MSRTFileName /quiet"
|
|
Set-Content -Path $installMSRTPath -Value $installMSRTCommand -Force
|
|
# Validate that the file created successfully
|
|
if (Test-Path -Path $installMSRTPath) {
|
|
WriteLog "$installMSRTPath created successfully"
|
|
}
|
|
else {
|
|
WriteLog "$installMSRTPath failed to create"
|
|
throw "$installMSRTPath failed to create"
|
|
}
|
|
}
|
|
}
|
|
|
|
#Download and Install OneDrive Per Machine
|
|
if ($UpdateOneDrive) {
|
|
WriteLog "`$UpdateOneDrive is set to true, checking for latest OneDrive client"
|
|
# Check if OneDrive has already been downloaded, if so, skip download
|
|
if (Test-Path -Path $OneDrivePath) {
|
|
# Check the size of the $OneDrivePath folder
|
|
$OneDriveSize = (Get-ChildItem -Path $OneDrivePath -Recurse | Measure-Object -Property Length -Sum).Sum
|
|
if ($OneDriveSize -gt 1MB) {
|
|
WriteLog "Found OneDrive download in $OneDrivePath, skipping download"
|
|
$OneDriveDownloaded = $true
|
|
}
|
|
}
|
|
if (-not $OneDriveDownloaded) {
|
|
#Check if $OneDrivePath exists, if not, create it
|
|
If (-not (Test-Path -Path $OneDrivePath)) {
|
|
WriteLog "Creating $OneDrivePath"
|
|
New-Item -Path $OneDrivePath -ItemType Directory -Force | Out-Null
|
|
}
|
|
WriteLog "Downloading latest OneDrive client"
|
|
if ($WindowsArch -eq 'x64') {
|
|
$OneDriveURL = 'https://go.microsoft.com/fwlink/?linkid=844652'
|
|
}
|
|
elseif ($WindowsArch -eq 'ARM64') {
|
|
$OneDriveURL = 'https://go.microsoft.com/fwlink/?linkid=2271260'
|
|
}
|
|
try {
|
|
Start-BitsTransferWithRetry -Source $OneDriveURL -Destination "$OneDrivePath\OneDriveSetup.exe"
|
|
WriteLog "OneDrive client downloaded to $OneDrivePath\OneDriveSetup.exe"
|
|
}
|
|
catch {
|
|
Write-Host "Downloading OneDrive client Failed"
|
|
WriteLog "Downloading OneDrive client Failed with error $_"
|
|
throw $_
|
|
}
|
|
|
|
# Create Update-OneDrive.ps1
|
|
$installODPath = Join-Path -Path $orchestrationPath -ChildPath "Update-OneDrive.ps1"
|
|
WriteLog "Creating $installODPath"
|
|
$installODCommand = "& d:\OneDrive\OneDriveSetup.exe /allusers /silent"
|
|
Set-Content -Path $installODPath -Value $installODCommand -Force
|
|
# Validate that the file created successfully
|
|
if (Test-Path -Path $installODPath) {
|
|
WriteLog "$installODPath created successfully"
|
|
}
|
|
else {
|
|
WriteLog "$installODPath failed to create"
|
|
throw "$installODPath failed to create"
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
#Download and Install Edge Stable
|
|
if ($UpdateEdge) {
|
|
WriteLog "`$UpdateEdge is set to true, checking for latest Edge Stable $WindowsArch release"
|
|
# Check if Edge has already been downloaded, if so, skip download
|
|
if (Test-Path -Path $EdgePath) {
|
|
# Check the size of the $EdgePath folder
|
|
$EdgeSize = (Get-ChildItem -Path $EdgePath -Recurse | Measure-Object -Property Length -Sum).Sum
|
|
if ($EdgeSize -gt 1MB) {
|
|
WriteLog "Found Edge download in $EdgePath, skipping download"
|
|
$EdgeDownloaded = $true
|
|
}
|
|
}
|
|
if (-not $EdgeDownloaded) {
|
|
# Create the search string for Edge based on Windows architecture
|
|
$Name = "microsoft edge stable -extended $WindowsArch"
|
|
#Check if $EdgePath exists, if not, create it
|
|
If (-not (Test-Path -Path $EdgePath)) {
|
|
WriteLog "Creating $EdgePath"
|
|
New-Item -Path $EdgePath -ItemType Directory -Force | Out-Null
|
|
}
|
|
WriteLog "Searching for $Name from Microsoft Update Catalog and saving to $EdgePath"
|
|
$KBFilePath = Save-KB -Name $Name -Path $EdgePath
|
|
$EdgeCABFilePath = "$EdgePath\$KBFilePath"
|
|
WriteLog "Latest Edge Stable $WindowsArch release saved to $EdgeCABFilePath"
|
|
|
|
#Extract Edge cab file to same folder as $EdgeFilePath
|
|
$EdgeMSIFileName = "MicrosoftEdgeEnterprise$WindowsArch.msi"
|
|
$EdgeFullFilePath = "$EdgePath\$EdgeMSIFileName"
|
|
WriteLog "Expanding $EdgeCABFilePath"
|
|
Invoke-Process Expand "$EdgeCABFilePath -F:*.msi $EdgeFullFilePath" | Out-Null
|
|
WriteLog "Expansion complete"
|
|
|
|
#Remove Edge CAB file
|
|
WriteLog "Removing $EdgeCABFilePath"
|
|
Remove-Item -Path $EdgeCABFilePath -Force
|
|
WriteLog "Removal complete"
|
|
|
|
# Create Update-Edge.ps1
|
|
$installEdgePath = Join-Path -Path $orchestrationPath -ChildPath "Update-Edge.ps1"
|
|
WriteLog "Creating $installEdgePath"
|
|
$installEdgeCommand = "& d:\Edge\$EdgeMSIFileName /quiet /norestart"
|
|
Set-Content -Path $installEdgePath -Value $installEdgeCommand -Force
|
|
# Validate that the file created successfully
|
|
if (Test-Path -Path $installEdgePath) {
|
|
WriteLog "$installEdgePath created successfully"
|
|
}
|
|
else {
|
|
WriteLog "$installEdgePath failed to create"
|
|
throw "$installEdgePath failed to create"
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
# Process AppsScriptVariables - Create json file
|
|
if ($AppsScriptVariables) {
|
|
$AppsScriptVariables | ConvertTo-Json | Out-File -FilePath $appsScriptVarsJsonPath -Encoding UTF8
|
|
WriteLog "AppsScriptVariables exported to $appsScriptVarsJsonPath for use during orchestration"
|
|
}
|
|
|
|
#Create Apps ISO
|
|
WriteLog "Creating $AppsISO file"
|
|
New-AppsISO
|
|
WriteLog "$AppsISO created successfully"
|
|
}
|
|
catch {
|
|
Write-Host "Creating Apps ISO Failed"
|
|
WriteLog "Creating Apps ISO Failed with error $_"
|
|
throw $_
|
|
}
|
|
}
|
|
}
|
|
|
|
#Create VHDX
|
|
try {
|
|
|
|
#Update latest Cumulative Update if both $UpdateLatestCU is $true and $UpdatePreviewCU is $false
|
|
#Changed to use MU Catalog instead of using Get-LatestWindowsKB
|
|
#The Windows release info page is updated later than the MU Catalog
|
|
if ($UpdateLatestCU -and -not $UpdatePreviewCU) {
|
|
Writelog "`$UpdateLatestCU is set to true, checking for latest CU"
|
|
if ($WindowsRelease -in 10, 11) {
|
|
$Name = """Cumulative update for Windows $WindowsRelease Version $WindowsVersion for $WindowsArch"""
|
|
}
|
|
if ($WindowsRelease -eq 2025) {
|
|
$Name = """Cumulative Update for Microsoft server operating system, version 24h2 for $WindowsArch"""
|
|
}
|
|
if ($WindowsRelease -eq 2022) {
|
|
$Name = """Cumulative Update for Microsoft server operating system, version 21h2 for $WindowsArch"""
|
|
}
|
|
if ($WindowsRelease -in 2016, 2019 -and $installationType -eq "Server") {
|
|
$Name = """Cumulative update for Windows Server $WindowsRelease for $WindowsArch"""
|
|
}
|
|
if ($WindowsRelease -in 2016, 2019, 2021 -and $isLTSC) {
|
|
$today = Get-Date
|
|
$firstDayOfMonth = Get-Date -Year $today.Year -Month $today.Month -Day 1
|
|
$secondTuesday = $firstDayOfMonth.AddDays(((2 - [int]$firstDayOfMonth.DayOfWeek + 7) % 7) + 7)
|
|
$updateDate = if ($today -gt $secondTuesday) { $today } else { $today.AddMonths(-1) }
|
|
# More precise search to prevent Dynamic cumulative update from being chosen.
|
|
$Name = """$($updateDate.ToString('yyyy-MM')) Cumulative update for Windows 10 Version $WindowsVersion for $WindowsArch"""
|
|
}
|
|
if ($WindowsRelease -eq 2024 -and $isLTSC) {
|
|
$Name = """Cumulative update for Windows 11 Version $WindowsVersion for $WindowsArch"""
|
|
}
|
|
#Check if $KBPath exists, if not, create it
|
|
If (-not (Test-Path -Path $KBPath)) {
|
|
WriteLog "Creating $KBPath"
|
|
New-Item -Path $KBPath -ItemType Directory -Force | Out-Null
|
|
}
|
|
#Get latest Servicing Stack Update for Windows Server 2016
|
|
if ($WindowsRelease -eq 2016 -and $installationType -eq "Server") {
|
|
$SSUName = """Servicing stack update for Windows Server $WindowsRelease for $WindowsArch"""
|
|
WriteLog "Searching for $SSUName from Microsoft Update Catalog and saving to $KBPath"
|
|
$SSUFile = Save-KB -Name $SSUName -Path $KBPath
|
|
$SSUFilePath = "$KBPath\$SSUFile"
|
|
WriteLog "Latest SSU saved to $SSUFilePath"
|
|
}
|
|
if ($WindowsRelease -in 2016, 2019, 2021 -and $isLTSC) {
|
|
$SSUName = """Servicing Stack Update for Windows 10 Version $WindowsVersion for $WindowsArch"""
|
|
WriteLog "Searching for $SSUName from Microsoft Update Catalog and saving to $KBPath"
|
|
$SSUFile = Save-KB -Name $SSUName -Path $KBPath
|
|
$SSUFilePath = "$KBPath\$SSUFile"
|
|
WriteLog "Latest SSU saved to $SSUFilePath"
|
|
}
|
|
WriteLog "Searching for $name from Microsoft Update Catalog and saving to $KBPath"
|
|
$CUFileName = Save-KB -Name $Name -Path $KBPath
|
|
# Check if $CUFileName contains the string in $global:LastKBArticleID
|
|
# If it does not, look in $KBPath for the file that contains the string in $global:LastKBArticleID
|
|
# and set that as the $CUFileName
|
|
# This is because checkpoint CUs download indeterministically
|
|
WriteLog "Checking if $CUFileName contains $global:LastKBArticleID"
|
|
if ($CUFileName -notmatch $global:LastKBArticleID) {
|
|
WriteLog "$CUFileName does not contain $global:LastKBArticleID, searching for file that contains it"
|
|
$CUFileName = $null
|
|
# Get the file that contains the string in $global:LastKBArticleID
|
|
$CUFileName = (Get-ChildItem -Path $KBPath -Filter "*$global:LastKBArticleID*" | Select-Object -First 1).Name
|
|
if ($null -ne $CUFileName) {
|
|
WriteLog "Found $CUFileName"
|
|
}
|
|
else {
|
|
WriteLog "Could not find file that contains $global:LastKBArticleID"
|
|
throw "Could not find file that contains $global:LastKBArticleID"
|
|
}
|
|
}
|
|
$CUPath = "$KBPath\$CUFileName"
|
|
WriteLog "Latest CU saved to $CUPath"
|
|
}
|
|
|
|
#Update Latest Preview Cumlative Update for Client OS only
|
|
#will take Precendence over $UpdateLatestCU if both were set to $true
|
|
if ($UpdatePreviewCU -and $installationType -eq 'Client' -and $WindowsSKU -notlike "*LTSC") {
|
|
Writelog "`$UpdatePreviewCU is set to true, checking for latest Preview CU"
|
|
$Name = """Cumulative update Preview for Windows $WindowsRelease Version $WindowsVersion for $WindowsArch"""
|
|
#Check if $KBPath exists, if not, create it
|
|
If (-not (Test-Path -Path $KBPath)) {
|
|
WriteLog "Creating $KBPath"
|
|
New-Item -Path $KBPath -ItemType Directory -Force | Out-Null
|
|
}
|
|
WriteLog "Searching for $name from Microsoft Update Catalog and saving to $KBPath"
|
|
$CUPFileName = Save-KB -Name $Name -Path $KBPath
|
|
# Check if $CUPFileName contains the string in $global:LastKBArticleID
|
|
# If it does not, look in $KBPath for the file that contains the string in $global:LastKBArticleID
|
|
# and set that as the $CUPFileName
|
|
# This is because checkpoint CUs download indeterministically
|
|
WriteLog "Checking if $CUPFileName contains $global:LastKBArticleID"
|
|
if ($CUPFileName -notmatch $global:LastKBArticleID) {
|
|
WriteLog "$CUPFileName does not contain $global:LastKBArticleID, searching for file that contains it"
|
|
$CUPFileName = $null
|
|
# Get the file that contains the string in $global:LastKBArticleID
|
|
$CUPFileName = (Get-ChildItem -Path $KBPath -Filter "*$global:LastKBArticleID*" | Select-Object -First 1).Name
|
|
if ($null -ne $CUPFileName) {
|
|
WriteLog "Found $CUPFileName"
|
|
}
|
|
else {
|
|
WriteLog "Could not find file that contains $global:LastKBArticleID"
|
|
throw "Could not find file that contains $global:LastKBArticleID"
|
|
}
|
|
}
|
|
$CUPPath = "$KBPath\$CUPFileName"
|
|
WriteLog "Latest CU saved to $CUPPath"
|
|
}
|
|
|
|
#Update Latest .NET Framework
|
|
if ($UpdateLatestNet) {
|
|
Writelog "`$UpdateLatestNet is set to true, checking for latest .NET Framework"
|
|
#Check if $KBPath exists, if not, create it
|
|
if (-not (Test-Path -Path $KBPath)) {
|
|
WriteLog "Creating $KBPath"
|
|
New-Item -Path $KBPath -ItemType Directory -Force | Out-Null
|
|
}
|
|
|
|
######
|
|
#LTSC#
|
|
######
|
|
|
|
# For Windows 10 LTSC editions (2016, 2019, 2021), download and save the latest Servicing Stack Update (SSU) and .NET Framework cumulative update(s)
|
|
if ($WindowsRelease -in 2016, 2019, 2021 -and $isLTSC) {
|
|
# SSU likely was downloaded via CU, but still needed here if .net is being updated, no need to download twice though
|
|
if ($null -eq $SSUFile) {
|
|
$SSUName = """Servicing Stack Update for Windows 10 Version $WindowsVersion for $WindowsArch"""
|
|
WriteLog "Searching for $SSUName from Microsoft Update Catalog and saving to $KBPath"
|
|
$SSUFile = Save-KB -Name $SSUName -Path $KBPath
|
|
$SSUFilePath = "$KBPath\$SSUFile"
|
|
WriteLog "Latest SSU saved to $SSUFilePath"
|
|
}
|
|
|
|
# For Windows 10 LTSC editions (2016, 2019, 2021), download and save the latest .NET Framework cumulative update(s)
|
|
# to a dedicated NET subdirectory, as these editions may include multiple .NET updates that need to be installed together.
|
|
if ($WindowsRelease -in 2016) {
|
|
$name = """Cumulative Update for .NET Framework 4.8 for Windows 10 version $WindowsVersion for $WindowsArch"""
|
|
}
|
|
if ($WindowsRelease -eq 2019) {
|
|
$name = """Cumulative Update for .NET Framework 3.5, 4.7.2 and 4.8 for Windows 10 Version $WindowsVersion for $WindowsArch"""
|
|
}
|
|
if ($WindowsRelease -eq 2021) {
|
|
$name = """Cumulative Update for .NET Framework 3.5, 4.8 and 4.8.1 for Windows 10 Version $WindowsVersion for $WindowsArch"""
|
|
}
|
|
|
|
$NETPath = Join-Path -Path $KBPath -ChildPath "NET"
|
|
if (-not (Test-Path -Path $NETPath)) {
|
|
WriteLog "Creating $NETPath"
|
|
New-Item -Path $NETPath -ItemType Directory -Force | Out-Null
|
|
}
|
|
WriteLog "Searching for $name from Microsoft Update Catalog and saving to $NETPath"
|
|
$NETFileName = Save-KB -Name $name -Path $NETPath
|
|
WriteLog "Latest .NET Framework cumulative update saved to $NETPath\$NETFileName"
|
|
}
|
|
|
|
# For Windows 11 LTSC 2024, set the update name to search for the latest .NET Framework cumulative update in the Microsoft Update Catalog
|
|
if ($WindowsRelease -eq 2024 -and $isLTSC) {
|
|
$Name = "Cumulative update for .NET framework windows 11 $WindowsVersion $WindowsArch -preview"
|
|
}
|
|
|
|
# For Windows 10 LTSC 2021, download and save the latest .NET Framework 4.8.1 feature pack to the NET subdirectory.
|
|
if ($WindowsRelease -eq 2021 -and $isLTSC) {
|
|
WriteLog "Checking for latest .NET Framework feature pack for Windows $WindowsRelease $WindowsSKU"
|
|
$NETFeatureName = """Microsoft .NET Framework 4.8.1 for Windows 10 Version 21H2 for x64"""
|
|
$NETFeaturePackFile = Save-KB -Name $NETFeatureName -Path $NETPath
|
|
WriteLog "Latest .NET Framework Feature pack saved to $NETPath\$NETFeaturePackFile"
|
|
}
|
|
# For Windows 10 LTSC 2016 and 2019, download and save the latest .NET Framework 4.8 feature pack to the NET subdirectory.
|
|
if ($WindowsRelease -in 2016, 2019 -and $isLTSC) {
|
|
WriteLog "Checking for latest .NET Framework feature pack for Windows $WindowsRelease $WindowsSKU"
|
|
$NETFeatureName = """Microsoft .NET Framework 4.8 for Windows 10 Version $WindowsVersion and Windows Server $WindowsRelease for x64"""
|
|
$NETFeaturePackFile = Save-KB -Name $NETFeatureName -Path $NETPath
|
|
WriteLog "Latest .NET Framework Feature pack saved to $NETPath\$NETFeaturePackFile"
|
|
}
|
|
|
|
########
|
|
#CLIENT#
|
|
########
|
|
|
|
# For Windows 10 and 11, set the update name to search for the latest .NET Framework cumulative update (excluding preview) in the Microsoft Update Catalog
|
|
if ($WindowsRelease -in 10, 11) {
|
|
$Name = "Cumulative update for .NET framework windows $WindowsRelease $WindowsVersion $WindowsArch -preview"
|
|
}
|
|
|
|
########
|
|
#SERVER#
|
|
########
|
|
|
|
# For Windows Server 2025, set the update name to search for the latest .NET Framework cumulative update (excluding preview) in the Microsoft Update Catalog
|
|
if ($WindowsRelease -eq 2025 -and $installationType -eq "Server") {
|
|
$Name = """Cumulative Update for .NET Framework"" ""3.5 and 4.8.1"" for Windows 11 24H2 x64 -preview"
|
|
}
|
|
|
|
# For Windows Server 2022, set the update name to search for the latest .NET Framework cumulative update (3.5, 4.8, and 4.8.1) for OS version 21H2 x64
|
|
if ($WindowsRelease -eq 2022 -and $installationType -eq "Server") {
|
|
$Name = """Cumulative Update for .NET Framework 3.5, 4.8 and 4.8.1"" ""operating system version 21H2 for x64"""
|
|
}
|
|
# For Windows Server 2019, set the update name to search for the latest .NET Framework cumulative update (3.5, 4.7.2, and 4.8) for x64
|
|
if ($WindowsRelease -eq 2019 -and $installationType -eq "Server") {
|
|
$Name = """Cumulative Update for .NET Framework 3.5, 4.7.2 and 4.8 for Windows Server 2019 for x64"""
|
|
}
|
|
|
|
# For Windows Server 2016, set the update name to search for the latest .NET Framework 4.8 cumulative update for x64
|
|
if ($WindowsRelease -eq 2016 -and $installationType -eq "Server") {
|
|
$Name = """Cumulative Update for .NET Framework 4.8 for Windows Server 2016 for x64"""
|
|
}
|
|
|
|
# For all editions except Windows 10 LTSC (2016, 2019, 2021), search for the latest .NET Framework cumulative update in the Microsoft Update Catalog,
|
|
# download it to $KBPath, and verify the correct file was downloaded by matching the KB article ID. If not found, search for the file by KB article ID.
|
|
if (-not ($WindowsRelease -in 2016, 2019, 2021 -and $isLTSC)) {
|
|
WriteLog "Searching for $name from Microsoft Update Catalog and saving to $KBPath"
|
|
$NETFileName = Save-KB -Name $Name -Path $KBPath
|
|
# Check if $NETFileName contains the string in $global:LastKBArticleID
|
|
# If it does not, look in $KBPath for the file that contains the string in $global:LastKBArticleID
|
|
# and set that as the $NETFileName
|
|
WriteLog "Checking if $NETFileName contains $global:LastKBArticleID"
|
|
if ($NETFileName -notmatch $global:LastKBArticleID) {
|
|
WriteLog "$NETFileName does not contain $global:LastKBArticleID, searching for file that contains it"
|
|
$NETFileName = $null
|
|
# Get the file that contains the string in $global:LastKBArticleID
|
|
$NETFileName = (Get-ChildItem -Path $KBPath -Filter "*$global:LastKBArticleID*" | Select-Object -First 1).Name
|
|
if ($null -ne $NETFileName) {
|
|
WriteLog "Found $NETFileName"
|
|
}
|
|
else {
|
|
WriteLog "Could not find file that contains $global:LastKBArticleID"
|
|
throw "Could not find file that contains $global:LastKBArticleID"
|
|
}
|
|
}
|
|
$NETPath = "$KBPath\$NETFileName"
|
|
WriteLog "Latest .NET Framework saved to $NETPath"
|
|
}
|
|
}
|
|
# Update latest Microcode
|
|
if ($UpdateLatestMicrocode -and $WindowsRelease -in 2016, 2019) {
|
|
WriteLog "`$UpdateLatestMicrocode is set to true, checking for latest Microcode"
|
|
#Check if $MicrocodePath exists, if not, create it
|
|
If (-not (Test-Path -Path $MicrocodePath)) {
|
|
WriteLog "Creating $MicrocodePath"
|
|
New-Item -Path $MicrocodePath -ItemType Directory -Force | Out-Null
|
|
}
|
|
|
|
# Windows 10 LTSC 2016 (1607) and Windows Server 2016
|
|
if ($WindowsRelease -eq 2016) {
|
|
$name = "KB4589210 $windowsArch"
|
|
}
|
|
|
|
# Windows 10 LTSC 2019 (1809) and Windows Server 2019
|
|
if ($WindowsRelease -eq 2019) {
|
|
$name = "KB4589208 $windowsArch"
|
|
}
|
|
WriteLog "Searching for $name from Microsoft Update Catalog and saving to $MicrocodePath"
|
|
$MicrocodeFileName = Save-KB -Name $name -Path $MicrocodePath
|
|
WriteLog "Latest Microcode saved to $MicrocodePath\$MicrocodeFileName"
|
|
}
|
|
|
|
#Search for cached VHDX and skip VHDX creation if there's a cached version
|
|
if ($AllowVHDXCaching) {
|
|
WriteLog 'AllowVHDXCaching is true, checking for cached VHDX file'
|
|
if (Test-Path -Path $VHDXCacheFolder) {
|
|
WriteLog "Found $VHDXCacheFolder"
|
|
$vhdxJsons = @(Get-ChildItem -File -Path $VHDXCacheFolder -Filter '*_config.json' | Sort-Object -Property CreationTime -Descending)
|
|
WriteLog "Found $($vhdxJsons.Count) cached VHDX files"
|
|
if (Test-Path -Path $KBPath) {
|
|
$downloadedKBs = @(Get-ChildItem -File -Path $KBPath -Recurse)
|
|
}
|
|
else {
|
|
$downloadedKBs = @()
|
|
}
|
|
#$jsonDeserializer = [System.Web.Script.Serialization.JavaScriptSerializer]::new()
|
|
|
|
foreach ($vhdxJson in $vhdxJsons) {
|
|
try {
|
|
WriteLog "Processing $($vhdxJson.FullName)"
|
|
#$vhdxCacheItem = $jsonDeserializer.Deserialize((Get-Content -Path $vhdxJson.FullName -Raw), [VhdxCacheItem])
|
|
$vhdxCacheItem = Get-Content -Path $vhdxJson.FullName -Raw | ConvertFrom-Json
|
|
|
|
if ((($vhdxCacheItem.WindowsSKU -ne $WindowsSKU) -or
|
|
([string]::IsNullOrEmpty($vhdxCacheItem.WindowsSKU) -xor [string]::IsNullOrEmpty($WindowsSKU)))) {
|
|
WriteLog 'WindowsSKU mismatch, continuing'
|
|
continue
|
|
}
|
|
|
|
if ((($vhdxCacheItem.LogicalSectorSizeBytes -ne $LogicalSectorSizeBytes) -or
|
|
([string]::IsNullOrEmpty($vhdxCacheItem.LogicalSectorSizeBytes) -xor [string]::IsNullOrEmpty($LogicalSectorSizeBytes)))) {
|
|
WriteLog 'LogicalSectorSizeBytes mismatch, continuing'
|
|
continue
|
|
}
|
|
|
|
if ((($vhdxCacheItem.WindowsRelease -ne $WindowsRelease) -or
|
|
([string]::IsNullOrEmpty($vhdxCacheItem.WindowsRelease) -xor [string]::IsNullOrEmpty($WindowsRelease)))) {
|
|
WriteLog 'WindowsRelease mismatch, continuing'
|
|
continue
|
|
}
|
|
|
|
if ((($vhdxCacheItem.WindowsVersion -ne $WindowsVersion) -or
|
|
([string]::IsNullOrEmpty($vhdxCacheItem.WindowsVersion) -xor [string]::IsNullOrEmpty($WindowsVersion)))) {
|
|
Writelog 'WindowsVersion mismatch, continuing'
|
|
continue
|
|
}
|
|
|
|
if ((($vhdxCacheItem.OptionalFeatures -ne $OptionalFeatures) -or
|
|
([string]::IsNullOrEmpty($vhdxCacheItem.OptionalFeatures) -xor [string]::IsNullOrEmpty($OptionalFeatures)))) {
|
|
WriteLog 'OptionalFeatures mismatch, continuing'
|
|
continue
|
|
}
|
|
|
|
if ((Compare-Object -ReferenceObject $downloadedKBs -DifferenceObject $vhdxCacheItem.IncludedUpdates -Property Name).Length -gt 0) {
|
|
# (Compare-Object -ReferenceObject $downloadedKBs -DifferenceObject $vhdxCacheItem.IncludedUpdates -Property Name)
|
|
# $downloadedKBs.Name
|
|
# $vhdxCacheItem.IncludedUpdates.Name
|
|
WriteLog 'IncludedUpdates mismatch, continuing'
|
|
continue
|
|
}
|
|
|
|
WriteLog "Found cached VHDX file $vhdxCacheFolder\$($vhdxCacheItem.VhdxFileName) with matching parameters and included updates"
|
|
$cachedVHDXFileFound = $true
|
|
$cachedVHDXInfo = $vhdxCacheItem
|
|
break
|
|
}
|
|
catch {
|
|
WriteLog "Reading $vhdxJson Failed with error $_"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (-Not $cachedVHDXFileFound) {
|
|
if ($ISOPath) {
|
|
$wimPath = Get-WimFromISO
|
|
}
|
|
else {
|
|
$wimPath = Get-WindowsESD -WindowsRelease $WindowsRelease -WindowsArch $WindowsArch -WindowsLang $WindowsLang -MediaType $mediaType
|
|
}
|
|
#If index not specified by user, try and find based on WindowsSKU
|
|
if (-not($index) -and ($WindowsSKU)) {
|
|
$index = Get-Index -WindowsImagePath $wimPath -WindowsSKU $WindowsSKU
|
|
}
|
|
|
|
$vhdxDisk = New-ScratchVhdx -VhdxPath $VHDXPath -SizeBytes $disksize -LogicalSectorSizeBytes $LogicalSectorSizeBytes
|
|
|
|
$systemPartitionDriveLetter = New-SystemPartition -VhdxDisk $vhdxDisk
|
|
|
|
New-MSRPartition -VhdxDisk $vhdxDisk
|
|
|
|
$osPartition = New-OSPartition -VhdxDisk $vhdxDisk -OSPartitionSize $OSPartitionSize -WimPath $WimPath -WimIndex $index
|
|
$osPartitionDriveLetter = $osPartition[1].DriveLetter
|
|
$WindowsPartition = $osPartitionDriveLetter + ':\'
|
|
|
|
#$recoveryPartition = New-RecoveryPartition -VhdxDisk $vhdxDisk -OsPartition $osPartition[1] -RecoveryPartitionSize $RecoveryPartitionSize -DataPartition $dataPartition
|
|
$recoveryPartition = New-RecoveryPartition -VhdxDisk $vhdxDisk -OsPartition $osPartition[1] -RecoveryPartitionSize $RecoveryPartitionSize -DataPartition $dataPartition
|
|
|
|
WriteLog 'All necessary partitions created.'
|
|
|
|
Add-BootFiles -OsPartitionDriveLetter $osPartitionDriveLetter -SystemPartitionDriveLetter $systemPartitionDriveLetter[1]
|
|
|
|
#Add Windows packages
|
|
if ($UpdateLatestCU -or $UpdateLatestNet -or $UpdatePreviewCU ) {
|
|
try {
|
|
WriteLog "Adding KBs to $WindowsPartition"
|
|
WriteLog 'This can take 10+ minutes depending on how old the media is and the size of the KB. Please be patient'
|
|
# If WindowsRelease is 2016, we need to add the SSU first
|
|
if ($WindowsRelease -eq 2016 -and $installationType -eq "Server") {
|
|
WriteLog 'WindowsRelease is 2016, adding SSU first'
|
|
WriteLog "Adding SSU to $WindowsPartition"
|
|
# Add-WindowsPackage -Path $WindowsPartition -PackagePath $SSUFilePath -PreventPending | Out-Null
|
|
# Commenting out -preventpending as it causes an issue with the SSU being applied
|
|
# Seems to be because of the registry being mounted per dism.log
|
|
Add-WindowsPackage -Path $WindowsPartition -PackagePath $SSUFilePath | Out-Null
|
|
WriteLog "SSU added to $WindowsPartition"
|
|
WriteLog "Removing $SSUFilePath"
|
|
Remove-Item -Path $SSUFilePath -Force | Out-Null
|
|
WriteLog 'SSU removed'
|
|
}
|
|
if ($WindowsRelease -in 2016, 2019, 2021 -and $isLTSC) {
|
|
WriteLog "WindowsRelease is $WindowsRelease and is $WindowsSKU, adding SSU first"
|
|
WriteLog "Adding SSU to $WindowsPartition"
|
|
Add-WindowsPackage -Path $WindowsPartition -PackagePath $SSUFilePath | Out-Null
|
|
WriteLog "SSU added to $WindowsPartition"
|
|
WriteLog "Removing $SSUFilePath"
|
|
Remove-Item -Path $SSUFilePath -Force | Out-Null
|
|
WriteLog 'SSU removed'
|
|
}
|
|
# Break out CU and NET updates to be added separately to abide by Checkpoint Update recommendations
|
|
if ($UpdateLatestCU) {
|
|
WriteLog "Adding $CUPath to $WindowsPartition"
|
|
Add-WindowsPackage -Path $WindowsPartition -PackagePath $CUPath | Out-Null
|
|
WriteLog "$CUPath added to $WindowsPartition"
|
|
}
|
|
if ($UpdatePreviewCU) {
|
|
WriteLog "Adding $CUPPath to $WindowsPartition"
|
|
Add-WindowsPackage -Path $WindowsPartition -PackagePath $CUPPath | Out-Null
|
|
WriteLog "$CUPPath added to $WindowsPartition"
|
|
}
|
|
if ($UpdateLatestNet) {
|
|
WriteLog "Adding $NETPath to $WindowsPartition"
|
|
Add-WindowsPackage -Path $WindowsPartition -PackagePath $NETPath | Out-Null
|
|
WriteLog "$NETPath added to $WindowsPartition"
|
|
}
|
|
if ($UpdateLatestMicrocode -and $WindowsRelease -in 2016, 2019) {
|
|
WriteLog "Adding $MicrocodePath to $WindowsPartition"
|
|
Add-WindowsPackage -Path $WindowsPartition -PackagePath $MicrocodePath | Out-Null
|
|
WriteLog "$MicrocodePath added to $WindowsPartition"
|
|
}
|
|
WriteLog "KBs added to $WindowsPartition"
|
|
if ($AllowVHDXCaching) {
|
|
$cachedVHDXInfo = [VhdxCacheItem]::new()
|
|
$includedUpdates = Get-ChildItem -Path $KBPath -File -Recurse
|
|
|
|
foreach ($includedUpdate in $includedUpdates) {
|
|
$cachedVHDXInfo.IncludedUpdates += ([VhdxCacheUpdateItem]::new($includedUpdate.Name))
|
|
}
|
|
}
|
|
WriteLog "Removing $KBPath"
|
|
Remove-Item -Path $KBPath -Recurse -Force | Out-Null
|
|
WriteLog 'Clean Up the WinSxS Folder'
|
|
WriteLog 'This can take 10+ minutes depending on how old the media is and the size of the KB. Please be patient'
|
|
Dism /Image:$WindowsPartition /Cleanup-Image /StartComponentCleanup /ResetBase | Out-Null
|
|
WriteLog 'Clean Up the WinSxS Folder completed'
|
|
}
|
|
catch {
|
|
Write-Host "Adding KB to VHDX failed with error $_"
|
|
WriteLog "Adding KB to VHDX failed with error $_"
|
|
if ($_.Exception.HResult -eq -2146498525) {
|
|
Write-Host 'Missing latest Servicing Stack Update'
|
|
Write-Host 'Media likely older than 2023-09 for Windows Server 2022 (KB5030216), or 2021-08 for Windows Server 2019 (KB5005112)'
|
|
Write-Host 'Recommended to use the latest media'
|
|
WriteLog 'Missing latest Servicing Stack Update'
|
|
WriteLog 'Media likely older than 2023-09 for Windows Server 2022 (KB5030216), or 2021-08 for Windows Server 2019 (KB5005112)'
|
|
WriteLog 'Recommended to use the latest media'
|
|
}
|
|
throw $_
|
|
}
|
|
}
|
|
|
|
#Enable Windows Optional Features (e.g. .Net3, etc)
|
|
If ($OptionalFeatures) {
|
|
$Source = Join-Path (Split-Path $wimpath) 'sxs'
|
|
Enable-WindowsFeaturesByName -FeatureNames $OptionalFeatures -Source $Source
|
|
}
|
|
If ($ISOPath) {
|
|
WriteLog 'Dismounting Windows ISO'
|
|
Dismount-DiskImage -ImagePath $ISOPath | Out-null
|
|
WriteLog 'Done'
|
|
}
|
|
#If $wimPath is an esd file, remove it
|
|
If ($wimPath -match '.esd') {
|
|
WriteLog "Deleting $wimPath file"
|
|
Remove-Item -Path $wimPath -Force
|
|
WriteLog "$wimPath deleted"
|
|
}
|
|
|
|
}
|
|
else {
|
|
#Use cached vhdx file
|
|
WriteLog 'Using cached VHDX file to speed up build proces'
|
|
WriteLog "VHDX file is: $($cachedVHDXInfo.VhdxFileName)"
|
|
|
|
Robocopy.exe $($VHDXCacheFolder) $($VMPath) $($cachedVHDXInfo.VhdxFileName) /E /COPY:DAT /R:5 /W:5 /J
|
|
$VHDXPath = Join-Path $($VMPath) $($cachedVHDXInfo.VhdxFileName)
|
|
|
|
$vhdxDisk = Get-VHD -Path $VHDXPath | Mount-VHD -Passthru | Get-Disk
|
|
$osPartition = $vhdxDisk | Get-Partition | Where-Object { $_.GptType -eq '{ebd0a0a2-b9e5-4433-87c0-68b6b72699c7}' }
|
|
$osPartitionDriveLetter = $osPartition.DriveLetter
|
|
$WindowsPartition = $osPartitionDriveLetter + ':\'
|
|
|
|
}
|
|
|
|
#Set Product key
|
|
If ($ProductKey) {
|
|
WriteLog "Setting Windows Product Key"
|
|
Set-WindowsProductKey -Path $WindowsPartition -ProductKey $ProductKey
|
|
}
|
|
|
|
|
|
If ($InstallApps) {
|
|
#Copy Unattend file so VM Boots into Audit Mode
|
|
WriteLog 'Copying unattend file to boot to audit mode'
|
|
New-Item -Path "$($osPartitionDriveLetter):\Windows\Panther\unattend" -ItemType Directory -Force | Out-Null
|
|
if ($WindowsArch -eq 'x64') {
|
|
Copy-Item -Path "$FFUDevelopmentPath\BuildFFUUnattend\unattend_x64.xml" -Destination "$($osPartitionDriveLetter):\Windows\Panther\Unattend\Unattend.xml" -Force | Out-Null
|
|
}
|
|
else {
|
|
Copy-Item -Path "$FFUDevelopmentPath\BuildFFUUnattend\unattend_arm64.xml" -Destination "$($osPartitionDriveLetter):\Windows\Panther\Unattend\Unattend.xml" -Force | Out-Null
|
|
}
|
|
WriteLog 'Copy completed'
|
|
}
|
|
|
|
if ($AllowVHDXCaching -and !$cachedVHDXFileFound) {
|
|
WriteLog 'Caching VHDX file'
|
|
|
|
WriteLog 'Defragmenting Windows partition...'
|
|
Optimize-Volume -DriveLetter $osPartition.DriveLetter -Defrag -NormalPriority
|
|
WriteLog 'Performing slab consolidation on Windows partition...'
|
|
Optimize-Volume -DriveLetter $osPartition.DriveLetter -SlabConsolidate -NormalPriority
|
|
WriteLog 'Dismounting VHDX'
|
|
Dismount-ScratchVhdx -VhdxPath $VHDXPath
|
|
|
|
WriteLog 'Copying to cache dir'
|
|
|
|
#Assuming there are now name collisons
|
|
Robocopy.exe $($VMPath) $($VHDXCacheFolder) $("$VMName.vhdx") /E /COPY:DAT /R:5 /W:5 /J
|
|
|
|
#Only create new instance if not created during patching
|
|
if ($null -eq $cachedVHDXInfo) {
|
|
$cachedVHDXInfo = [VhdxCacheItem]::new()
|
|
}
|
|
$cachedVHDXInfo.VhdxFileName = $("$VMName.vhdx")
|
|
$cachedVHDXInfo.LogicalSectorSizeBytes = $LogicalSectorSizeBytes
|
|
$cachedVHDXInfo.WindowsSKU = $WindowsSKU
|
|
$cachedVHDXInfo.WindowsRelease = $WindowsRelease
|
|
$cachedVHDXInfo.WindowsVersion = $WindowsVersion
|
|
$cachedVHDXInfo.OptionalFeatures = $OptionalFeatures
|
|
|
|
$cachedVHDXInfo | ConvertTo-Json | Out-File -FilePath ("{0}\{1}_config.json" -f $($VHDXCacheFolder), $VMName)
|
|
WriteLog "Cached VHDX file $("$VMName.vhdx")"
|
|
|
|
#Remount the VHDX file if $installapps is false so the VHDX can be captured to an FFU
|
|
If (-not $InstallApps) {
|
|
Mount-Vhd -Path $VHDXPath
|
|
}
|
|
}
|
|
else {
|
|
if ($InstallApps) {
|
|
Dismount-ScratchVhdx -VhdxPath $VHDXPath
|
|
}
|
|
}
|
|
}
|
|
catch {
|
|
Write-Host 'Creating VHDX Failed'
|
|
WriteLog "Creating VHDX Failed with error $_"
|
|
WriteLog "Dismounting $VHDXPath"
|
|
Dismount-ScratchVhdx -VhdxPath $VHDXPath
|
|
WriteLog "Removing $VMPath"
|
|
Remove-Item -Path $VMPath -Force -Recurse | Out-Null
|
|
WriteLog 'Removal complete'
|
|
If ($ISOPath) {
|
|
WriteLog 'Dismounting Windows ISO'
|
|
Dismount-DiskImage -ImagePath $ISOPath | Out-null
|
|
WriteLog 'Done'
|
|
}
|
|
else {
|
|
#Remove ESD file
|
|
WriteLog "Deleting ESD file"
|
|
Remove-Item -Path $wimPath -Force
|
|
WriteLog "ESD File deleted"
|
|
}
|
|
throw $_
|
|
|
|
}
|
|
|
|
#If installing apps (Office or 3rd party), we need to build a VM and capture that FFU, if not, just cut the FFU from the VHDX file
|
|
if ($InstallApps) {
|
|
#Create VM and attach VHDX
|
|
try {
|
|
WriteLog 'Creating new FFU VM'
|
|
$FFUVM = New-FFUVM
|
|
WriteLog 'FFU VM Created'
|
|
}
|
|
catch {
|
|
Write-Host 'VM creation failed'
|
|
Writelog "VM creation failed with error $_"
|
|
Remove-FFUVM -VMName $VMName
|
|
throw $_
|
|
|
|
}
|
|
#Create ffu user and share to capture FFU to
|
|
try {
|
|
Set-CaptureFFU
|
|
}
|
|
catch {
|
|
Write-Host 'Set-CaptureFFU function failed'
|
|
WriteLog "Set-CaptureFFU function failed with error $_"
|
|
Remove-FFUVM -VMName $VMName
|
|
throw $_
|
|
|
|
}
|
|
If ($CreateCaptureMedia) {
|
|
#Create Capture Media
|
|
try {
|
|
#This should happen while the FFUVM is building
|
|
New-PEMedia -Capture $true
|
|
}
|
|
catch {
|
|
Write-Host 'Creating capture media failed'
|
|
WriteLog "Creating capture media failed with error $_"
|
|
Remove-FFUVM -VMName $VMName
|
|
throw $_
|
|
|
|
}
|
|
}
|
|
}
|
|
#Capture FFU file
|
|
try {
|
|
#Check for FFU Folder and create it if it's missing
|
|
If (-not (Test-Path -Path $FFUCaptureLocation)) {
|
|
WriteLog "Creating FFU capture location at $FFUCaptureLocation"
|
|
New-Item -Path $FFUCaptureLocation -ItemType Directory -Force
|
|
WriteLog "Successfully created FFU capture location at $FFUCaptureLocation"
|
|
}
|
|
#Check if VM is done provisioning
|
|
If ($InstallApps) {
|
|
do {
|
|
$FFUVM = Get-VM -Name $FFUVM.Name
|
|
Start-Sleep -Seconds 10
|
|
WriteLog 'Waiting for VM to shutdown'
|
|
} while ($FFUVM.State -ne 'Off')
|
|
WriteLog 'VM Shutdown'
|
|
Optimize-FFUCaptureDrive -VhdxPath $VHDXPath
|
|
#Capture FFU file
|
|
New-FFU $FFUVM.Name
|
|
}
|
|
else {
|
|
#Shorten Windows SKU for use in FFU file name to remove spaces and long names
|
|
WriteLog "Shortening Windows SKU: $WindowsSKU for FFU file name"
|
|
$shortenedWindowsSKU = Get-ShortenedWindowsSKU -WindowsSKU $WindowsSKU
|
|
WriteLog "Shortened Windows SKU: $shortenedWindowsSKU"
|
|
#Create FFU file
|
|
New-FFU
|
|
}
|
|
}
|
|
Catch {
|
|
Write-Host 'Capturing FFU file failed'
|
|
Writelog "Capturing FFU file failed with error $_"
|
|
If ($InstallApps) {
|
|
Remove-FFUVM -VMName $VMName
|
|
}
|
|
else {
|
|
Remove-FFUVM
|
|
}
|
|
|
|
throw $_
|
|
|
|
}
|
|
#Clean up ffu_user and Share and clean up apps
|
|
If ($InstallApps) {
|
|
try {
|
|
Remove-FFUUserShare
|
|
}
|
|
catch {
|
|
Write-Host 'Cleaning up FFU User and/or share failed'
|
|
WriteLog "Cleaning up FFU User and/or share failed with error $_"
|
|
Remove-FFUVM -VMName $VMName
|
|
throw $_
|
|
}
|
|
#Clean up Apps
|
|
if ($RemoveApps) {
|
|
try {
|
|
WriteLog "Cleaning up $AppsPath"
|
|
Remove-Apps
|
|
}
|
|
catch {
|
|
Write-Host 'Cleaning up Apps failed'
|
|
Writelog "Cleaning up Apps failed with error $_"
|
|
throw $_
|
|
}
|
|
}
|
|
#Clean up Updates
|
|
if ($RemoveUpdates) {
|
|
try {
|
|
WriteLog "Cleaning up downloaded update files"
|
|
Remove-Updates
|
|
}
|
|
catch {
|
|
Write-Host 'Cleaning up downloaded update files failed'
|
|
Writelog "Cleaning up downloaded update files failed with error $_"
|
|
throw $_
|
|
}
|
|
}
|
|
}
|
|
#Clean up VM or VHDX
|
|
try {
|
|
Remove-FFUVM
|
|
WriteLog 'FFU build complete!'
|
|
}
|
|
catch {
|
|
Write-Host 'VM or vhdx cleanup failed'
|
|
Writelog "VM or vhdx cleanup failed with error $_"
|
|
throw $_
|
|
}
|
|
|
|
|
|
#Create Deployment Media
|
|
If ($CreateDeploymentMedia) {
|
|
try {
|
|
New-PEMedia -Deploy $true
|
|
}
|
|
catch {
|
|
Write-Host 'Creating deployment media failed'
|
|
WriteLog "Creating deployment media failed with error $_"
|
|
throw $_
|
|
|
|
}
|
|
}
|
|
If ($BuildUSBDrive) {
|
|
try {
|
|
If (Test-Path -Path $DeployISO) {
|
|
New-DeploymentUSB -CopyFFU
|
|
}
|
|
else {
|
|
WriteLog "$BuildUSBDrive set to true, however unable to find $DeployISO. USB drive not built."
|
|
}
|
|
|
|
}
|
|
catch {
|
|
Write-Host 'Building USB deployment drive failed'
|
|
Writelog "Building USB deployment drive failed with error $_"
|
|
throw $_
|
|
}
|
|
}
|
|
If ($RemoveFFU) {
|
|
try {
|
|
Remove-FFU
|
|
}
|
|
catch {
|
|
Write-Host 'Removing FFU files failed'
|
|
Writelog "Removing FFU files failed with error $_"
|
|
throw $_
|
|
}
|
|
|
|
}
|
|
If ($CleanupCaptureISO) {
|
|
try {
|
|
If (Test-Path -Path $CaptureISO) {
|
|
WriteLog "Removing $CaptureISO"
|
|
Remove-Item -Path $CaptureISO -Force
|
|
WriteLog "Removal complete"
|
|
}
|
|
}
|
|
catch {
|
|
Writelog "Removing $CaptureISO failed with error $_"
|
|
throw $_
|
|
}
|
|
}
|
|
If ($CleanupDeployISO) {
|
|
try {
|
|
If (Test-Path -Path $DeployISO) {
|
|
WriteLog "Removing $DeployISO"
|
|
Remove-Item -Path $DeployISO -Force
|
|
WriteLog "Removal complete"
|
|
}
|
|
}
|
|
catch {
|
|
Writelog "Removing $DeployISO failed with error $_"
|
|
throw $_
|
|
}
|
|
}
|
|
If ($CleanupAppsISO) {
|
|
try {
|
|
If (Test-Path -Path $AppsISO) {
|
|
WriteLog "Removing $AppsISO"
|
|
Remove-Item -Path $AppsISO -Force
|
|
WriteLog "Removal complete"
|
|
}
|
|
}
|
|
catch {
|
|
Writelog "Removing $AppsISO failed with error $_"
|
|
throw $_
|
|
}
|
|
}
|
|
If ($CleanupDrivers) {
|
|
try {
|
|
#Remove files in $Driversfolder, but keep $DriversFolder
|
|
If (Test-Path -Path $Driversfolder) {
|
|
WriteLog "Removing files in $Driversfolder"
|
|
Remove-Item -Path $Driversfolder\* -Force -Recurse
|
|
WriteLog "Removal complete"
|
|
}
|
|
}
|
|
catch {
|
|
Writelog "Removing $Driversfolder\* failed with error $_"
|
|
throw $_
|
|
}
|
|
}
|
|
if ($AllowVHDXCaching) {
|
|
try {
|
|
If (Test-Path -Path $KBPath) {
|
|
WriteLog "Removing $KBPath"
|
|
Remove-Item -Path $KBPath -Recurse -Force -ErrorAction SilentlyContinue
|
|
WriteLog 'Removal complete'
|
|
}
|
|
}
|
|
catch {
|
|
Writelog "Removing $KBPath failed with error $_"
|
|
throw $_
|
|
}
|
|
}
|
|
#Set $LongPathsEnabled registry value back to original value. $LongPathsEnabled could be $null if the registry value was not found
|
|
if ($null -eq $LongPathsEnabled) {
|
|
Remove-ItemProperty -Path 'HKLM:\SYSTEM\CurrentControlSet\Control\FileSystem' -Name 'LongPathsEnabled' -ErrorAction SilentlyContinue
|
|
}
|
|
else {
|
|
Set-ItemProperty -Path 'HKLM:\SYSTEM\CurrentControlSet\Control\FileSystem' -Name 'LongPathsEnabled' -Value $LongPathsEnabled
|
|
}
|
|
|
|
#Clean up dirty.txt file
|
|
Remove-Item -Path .\dirty.txt -Force | out-null
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Host 'Script complete'
|
|
}
|
|
# Record the end time
|
|
$endTime = Get-Date
|
|
Write-Host "FFU build process completed at" $endTime
|
|
|
|
# Calculate the total run time
|
|
$runTime = $endTime - $startTime
|
|
|
|
# Format the runtime with hours, minutes, and seconds
|
|
if ($runTime.TotalHours -ge 1) {
|
|
$runTimeFormatted = 'Duration: {0:hh} hr {0:mm} min {0:ss} sec' -f $runTime
|
|
}
|
|
else {
|
|
$runTimeFormatted = 'Duration: {0:mm} min {0:ss} sec' -f $runTime
|
|
}
|
|
|
|
if ($VerbosePreference -ne 'Continue') {
|
|
Write-Host $runTimeFormatted
|
|
}
|
|
WriteLog 'Script complete'
|
|
WriteLog $runTimeFormatted
|