AutomatedLabDefinition.psm1
function Get-LabFreeDiskSpace { param( [Parameter(Mandatory)] [string]$Path ) [uint64]$freeBytesAvailable = 0 [uint64]$totalNumberOfBytes = 0 [uint64]$totalNumberOfFreeBytes = 0 $success = [AutomatedLab.DiskSpaceWin32]::GetDiskFreeSpaceEx($Path, [ref]$freeBytesAvailable, [ref]$totalNumberOfBytes, [ref]$totalNumberOfFreeBytes) if (-not $success) { Write-Error "Could not determine free disk space of path '$Path'" } New-Object -TypeName PSObject -Property @{ TotalNumberOfBytes = $totalNumberOfBytes FreeBytesAvailable = $freeBytesAvailable TotalNumberOfFreeBytes = $totalNumberOfFreeBytes } } function Get-OnlineAdapterHardwareAddress { [Diagnostics.CodeAnalysis.SuppressMessageAttribute("PSUseCompatibleCmdlets", "", Justification="Special handling for Linux")] [OutputType([string[]])] [CmdletBinding()] param ( ) if ($IsLinux) { ip link show up | ForEach-Object { if ($_ -match '(\w{2}:?){6}' -and $Matches.0 -ne '00:00:00:00:00:00') { $Matches.0 } } } else { Get-CimInstance Win32_NetworkAdapter | Where-Object { $_.NetEnabled -and $_.NetConnectionID } | Select-Object -ExpandProperty MacAddress } } function Add-LabAzureAppServicePlanDefinition { [CmdletBinding()] [OutputType([AutomatedLab.Azure.AzureRmService])] param ( [Parameter(Mandatory)] [string]$Name, [string]$ResourceGroup, [string]$Location, [ValidateSet('Basic', 'Free', 'Premium', 'Shared', 'Standard')] [string]$Tier = 'Free', [ValidateSet('ExtraLarge', 'Large', 'Medium', 'Small')] [string]$WorkerSize = 'Small', [int]$NumberofWorkers, [switch]$PassThru ) Write-LogFunctionEntry $script:lab = & $MyInvocation.MyCommand.Module { $script:lab } if ($Script:lab.AzureResources.ServicePlans | Where-Object Name -eq $Name) { Write-Error "There is already an Azure App Service Plan with the name $'$Name'" return } if (-not $ResourceGroup) { $ResourceGroup = $lab.AzureSettings.DefaultResourceGroup.ResourceGroupName } if (-not $Location) { $Location = $lab.AzureSettings.DefaultLocation.DisplayName } $servicePlan = New-Object AutomatedLab.Azure.AzureRmServerFarmWithRichSku $servicePlan.Name = $Name $servicePlan.ResourceGroup = $ResourceGroup $servicePlan.Location = $Location $servicePlan.Tier = $Tier $servicePlan.WorkerSize = $WorkerSize $servicePlan.NumberofWorkers = $NumberofWorkers $Script:lab.AzureResources.ServicePlans.Add($servicePlan) Write-LogFunctionExit } function Add-LabAzureWebAppDefinition { [CmdletBinding()] [OutputType([AutomatedLab.Azure.AzureRmService])] param ( [Parameter(Mandatory)] [string]$Name, [string]$ResourceGroup, [string]$Location, [string]$AppServicePlan, [switch]$PassThru ) Write-LogFunctionEntry $script:lab = & $MyInvocation.MyCommand.Module { $script:lab } if ($Script:lab.AzureResources.Services | Where-Object Name -eq $Name) { Write-Error "There is already a Azure Web App with the name $'$Name'" return } if (-not $ResourceGroup) { $ResourceGroup = $lab.AzureSettings.DefaultResourceGroup.ResourceGroupName } if (-not $Location) { $Location = $lab.AzureSettings.DefaultLocation.DisplayName } if (-not $AppServicePlan) { $AppServicePlan = $Name } if (-not ($lab.AzureResources.ServicePlans | Where-Object Name -eq $AppServicePlan)) { Write-ScreenInfo "The Azure Application Service plan '$AppServicePlan' does not exist, creating it with default settings." Add-LabAzureAppServicePlanDefinition -Name $Name -ResourceGroup $ResourceGroup -Location $Location -Tier Free -WorkerSize Small } $webApp = New-Object AutomatedLab.Azure.AzureRmService $webApp.Name = $Name $webApp.ResourceGroup = $ResourceGroup $webApp.Location = $Location $webApp.ApplicationServicePlan = $AppServicePlan $Script:lab.AzureResources.Services.Add($webApp) Write-LogFunctionExit } function Get-LabAzureAppServicePlanDefinition { [CmdletBinding(DefaultParameterSetName = 'All')] [OutputType([AutomatedLab.Azure.AzureRmServerFarmWithRichSku])] param ( [Parameter(Position = 0, ParameterSetName = 'ByName', ValueFromPipeline, ValueFromPipelineByPropertyName)] [string[]]$Name ) begin { Write-LogFunctionEntry $script:lab = & $MyInvocation.MyCommand.Module { $script:lab } if ($PSCmdlet.ParameterSetName -eq 'All') { $lab.AzureResources.ServicePlans break } } process { $sp = $lab.AzureResources.ServicePlans | Where-Object Name -eq $Name if (-not $sp) { Write-Error "The Azure App Service Plan '$Name' does not exist." } else { $sp } } end { Write-LogFunctionExit } } function Get-LabAzureWebAppDefinition { [CmdletBinding(DefaultParameterSetName = 'All')] [OutputType([AutomatedLab.Azure.AzureRmService])] param ( [Parameter(Position = 0, ParameterSetName = 'ByName', ValueFromPipeline, ValueFromPipelineByPropertyName)] [string[]]$Name ) begin { Write-LogFunctionEntry $script:lab = & $MyInvocation.MyCommand.Module { $script:lab } if ($PSCmdlet.ParameterSetName -eq 'All') { $lab.AzureResources.Services break } } process { $sp = $lab.AzureResources.Services | Where-Object Name -eq $Name if (-not $sp) { Write-Error "The Azure App Service '$Name' does not exist." } else { $sp } } end { Write-LogFunctionExit } } function Add-LabDiskDefinition { [CmdletBinding()] [OutputType([AutomatedLab.Disk])] param ( [Parameter(Mandatory, ValueFromPipelineByPropertyName)] [ValidateScript( { $doesAlreadyExist = Test-Path -Path $_ if ($doesAlreadyExist) { Write-ScreenInfo 'The disk does already exist' -Type Warning return $false } else { return $true } } )] [ValidateNotNullOrEmpty()] [string]$Name, [Parameter(Mandatory, ValueFromPipelineByPropertyName)] [ValidateNotNullOrEmpty()] [int]$DiskSizeInGb = 60, [string]$Label, [char]$DriveLetter, [switch]$UseLargeFRS, [long]$AllocationUnitSize = 4KB, [ValidateSet('MBR','GPT')] [string] $PartitionStyle = 'GPT', [switch]$SkipInitialize, [switch]$PassThru ) Write-LogFunctionEntry if ($null -eq $script:disks) { $errorMessage = "Create a new lab first using 'New-LabDefinition' before adding disks" Write-Error $errorMessage Write-LogFunctionExitWithError -Message $errorMessage return } if ($Name) { if ($script:disks | Where-Object Name -eq $Name) { $errorMessage = "A disk with the name '$Name' does already exist" Write-Error $errorMessage Write-LogFunctionExitWithError -Message $errorMessage return } } $disk = New-Object -TypeName AutomatedLab.Disk $disk.Name = $Name $disk.DiskSize = $DiskSizeInGb $disk.SkipInitialization = [bool]$SkipInitialize $disk.AllocationUnitSize = $AllocationUnitSize $disk.UseLargeFRS = $UseLargeFRS $disk.DriveLetter = $DriveLetter $disk.PartitionStyle = $PartitionStyle $disk.Label = if ($Label) { $Label } else { 'ALData' } $script:disks.Add($disk) Write-PSFMessage "Added disk '$Name' with path '$Path'. Lab now has $($Script:disks.Count) disk(s) defined" if ($PassThru) { $disk } Write-LogFunctionExit } function Add-LabDomainDefinition { [CmdletBinding()] param ( [Parameter(Mandatory = $true, ValueFromPipelineByPropertyName = $true)] [string]$Name, [Parameter(Mandatory = $true, ValueFromPipelineByPropertyName = $true)] [string]$AdminUser, [Parameter(Mandatory = $true, ValueFromPipelineByPropertyName = $true)] [string]$AdminPassword, [switch]$PassThru ) Write-LogFunctionEntry if ($script:lab.Domains | Where-Object { $_.Name -eq $Name }) { $errorMessage = "A domain with the name '$Name' is already defined" Write-Error $errorMessage Write-LogFunctionExitWithError -Message $errorMessage return } $domain = New-Object -TypeName AutomatedLab.Domain $domain.Name = $Name $user = New-Object -TypeName AutomatedLab.User $user.UserName = $AdminUser $user.Password = $AdminPassword $domain.Administrator = $user $script:lab.Domains.Add($domain) Write-PSFMessage "Added domain '$Name'. Lab now has $($Script:lab.Domains.Count) domain(s) defined" if ($PassThru) { $domain } Write-LogFunctionExit } function Add-LabIsoImageDefinition { [CmdletBinding()] param ( [string]$Name, [string]$Path, [Switch]$IsOperatingSystem, [switch]$NoDisplay ) Write-LogFunctionEntry if ($IsOperatingSystem) { Write-ScreenInfo -Message 'The -IsOperatingSystem switch parameter is obsolete and thereby ignored' -Type Warning } if (-not $script:lab) { throw 'Please create a lab before using this cmdlet. To create a new lab, call New-LabDefinition' } $type = Get-Type -GenericType AutomatedLab.ListXmlStore -T AutomatedLab.IsoImage #read the cache try { if ($IsLinux -or $IsMacOs) { $cachedIsos = $type::Import((Join-Path -Path (Get-LabConfigurationItem -Name LabAppDataRoot) -ChildPath 'Stores/LocalIsoImages.xml')) } else { $cachedIsos = $type::ImportFromRegistry('Cache', 'LocalIsoImages') } Write-PSFMessage "Read $($cachedIsos.Count) ISO images from the cache" } catch { Write-PSFMessage 'Could not read ISO images info from the cache' $cachedIsos = New-Object $type } $lab = try { Get-Lab -ErrorAction Stop } catch { Get-LabDefinition -ErrorAction Stop } if ($lab.DefaultVirtualizationEngine -eq 'Azure') { if (Test-LabPathIsOnLabAzureLabSourcesStorage -Path $Path) { $isoRoot = 'ISOs' if ($Path -notmatch 'ISOs$') { # Get relative path $isoRoot = $Path.Replace($labSources, '') } if ($isoRoot.StartsWith('\') -or $isoRoot.StartsWith('/') ) { $isoRoot = $isoRoot.Substring(1) } $isoRoot = $isoRoot.Replace('\','/') $isoFiles = Get-LabAzureLabSourcesContent -Path $isoRoot -RegexFilter '\.iso' -File -ErrorAction SilentlyContinue if ( -not $IsLinux -and [System.IO.Path]::HasExtension($Path) -or $IsLinux -and $Path -match '\.iso$') { $isoFiles = $isoFiles | Where-Object {$_.Name -eq (Split-Path -Path $Path -Leaf)} if (-not $isoFiles -and $Name) { $filterPath = (Split-Path -Path $Path -Leaf) -replace '\\','/' # Due to breaking changes introduced in Az.Storage 4.7.0 Write-ScreenInfo -Message "Syncing $filterPath with Azure lab sources storage as it did not exist" Sync-LabAzureLabSources -Filter $filterPath -NoDisplay $isoFiles = Get-LabAzureLabSourcesContent -Path $isoRoot -RegexFilter '\.iso' -File -ErrorAction SilentlyContinue | Where-Object {$_.Name -eq (Split-Path -Path $Path -Leaf)} } } } else { Write-ScreenInfo -Type Warning -Message "$Path is not on Azure LabSources $()! If you intend to use`r`nMount-LabIsoImage it will result in the ISO getting copied to the remote machine!" $isoFiles = Get-ChildItem -Path $Path -Filter *.iso -Recurse -ErrorAction SilentlyContinue } } else { $isoFiles = Get-ChildItem -Path $Path -Filter *.iso -Recurse -ErrorAction SilentlyContinue } if (-not $isoFiles) { throw "The specified iso file could not be found or no ISO file could be found in the given folder: $Path" } $isos = @() foreach ($isoFile in $isoFiles) { if (-not $PSBoundParameters.ContainsKey('Name')) { $Name = [guid]::NewGuid() } else { $cachedIsos.Remove(($cachedIsos | Where-Object Name -eq $name)) | Out-Null } $iso = New-Object -TypeName AutomatedLab.IsoImage $iso.Name = $Name $iso.Path = $isoFile.FullName $iso.Size = $isoFile.Length if ($cachedIsos -contains $iso) { Write-PSFMessage "The ISO '$($iso.Path)' with a size '$($iso.Size)' is already in the cache." $cachedIso = ($cachedIsos -eq $iso)[0] if ($PSBoundParameters.ContainsKey('Name')) { $cachedIso.Name = $Name } $isos += $cachedIso } else { if (-not $script:lab.DefaultVirtualizationEngine -eq 'Azure') { Write-PSFMessage "The ISO '$($iso.Path)' with a size '$($iso.Size)' is not in the cache. Reading the operating systems from ISO." [void] (Mount-DiskImage -ImagePath $isoFile.FullName -StorageType ISO) Get-PSDrive | Out-Null #This is just to refresh the drives. Somehow if this cmdlet is not called, PowerShell does not see the new drives. $letter = (Get-DiskImage -ImagePath $isoFile.FullName | Get-Volume).DriveLetter $isOperatingSystem = (Test-Path "$letter`:\Sources\Install.wim") -or (Test-Path "$letter`:\.discinfo") -or (Test-Path "$letter`:\isolinux") -or (Test-Path "$letter`:\suse") [void] (Dismount-DiskImage -ImagePath $isoFile.FullName) } if ($isOperatingSystem) { $oses = Get-LabAvailableOperatingSystem -Path $isoFile.FullName if ($oses) { foreach ($os in $oses) { if ($isos.OperatingSystems -contains $os) { Write-ScreenInfo "The operating system '$($os.OperatingSystemName)' with version '$($os.Version)' is already added to the lab. If this is an issue with cached information, use Clear-LabCache to solve the issue." -Type Warning } $iso.OperatingSystems.Add($os) | Out-Null } } $cachedIsos.Add($iso) #the new ISO goes into the cache $isos += $iso } else { $cachedIsos.Add($iso) #ISO is not an OS. Add only if 'Name' is specified. Hence, ISO is manually added $isos += $iso } } } $duplicateOperatingSystems = $isos | Where-Object { $_.OperatingSystems } | Group-Object -Property { "$($_.OperatingSystems.OperatingSystemName) $($_.OperatingSystems.Version)" } | Where-Object Count -gt 1 if ($duplicateOperatingSystems) { $duplicateOperatingSystems.Group | ForEach-Object { $_ } -PipelineVariable iso | ForEach-Object { $_.OperatingSystems } | ForEach-Object { Write-ScreenInfo "The operating system $($_.OperatingSystemName) version $($_.Version) defined more than once in '$($iso.Path)'" -Type Warning } } if ($IsLinux -or $IsMacOs) { $cachedIsos.Export((Join-Path -Path (Get-LabConfigurationItem -Name LabAppDataRoot) -ChildPath 'Stores/LocalIsoImages.xml')) } else { $cachedIsos.ExportToRegistry('Cache', 'LocalIsoImages') } foreach ($iso in $isos) { $isosToRemove = $script:lab.Sources.ISOs | Where-Object { $_.Name -eq $iso.Name -or $_.Path -eq $iso.Path } foreach ($isoToRemove in $isosToRemove) { $script:lab.Sources.ISOs.Remove($isoToRemove) | Out-Null } #$script:lab.Sources.ISOs.Remove($iso) | Out-Null $script:lab.Sources.ISOs.Add($iso) Write-ScreenInfo -Message "Added '$($iso.Path)'" } Write-PSFMessage "Final Lab ISO count: $($script:lab.Sources.ISOs.Count)" Write-LogFunctionExit } function Add-LabMachineDefinition { [CmdletBinding(DefaultParameterSetName = 'Network')] [OutputType([AutomatedLab.Machine])] param ( [Parameter(Mandatory = $true, ValueFromPipelineByPropertyName = $true)] [ValidatePattern("^([\'\""a-zA-Z0-9-]){1,15}$")] [string]$Name, [ValidateRange(128MB, 128GB)] [double]$Memory, [ValidateRange(128MB, 128GB)] [double]$MinMemory, [ValidateRange(128MB, 128GB)] [double]$MaxMemory, [ValidateRange(1, 64)] [ValidateNotNullOrEmpty()] [int]$Processors = 0, [ValidatePattern('^([a-zA-Z0-9-_]){2,30}$')] [string[]]$DiskName, [Alias('OS')] [AutomatedLab.OperatingSystem]$OperatingSystem = (Get-LabDefinition).DefaultOperatingSystem, [string]$OperatingSystemVersion, [Parameter(ParameterSetName = 'Network')] [ValidatePattern('^([a-zA-Z0-9])|([ ]){2,244}$')] [string]$Network, [Parameter(ParameterSetName = 'Network')] [ValidatePattern('^(([2]([0-4][0-9]|[5][0-5])|[0-1]?[0-9]?[0-9])[.]){3}(([2]([0-4][0-9]|[5][0-5])|[0-1]?[0-9]?[0-9]))$')] [string]$IpAddress, [Parameter(ParameterSetName = 'Network')] [ValidatePattern('^(([2]([0-4][0-9]|[5][0-5])|[0-1]?[0-9]?[0-9])[.]){3}(([2]([0-4][0-9]|[5][0-5])|[0-1]?[0-9]?[0-9]))$')] [string]$Gateway, [Parameter(ParameterSetName = 'Network')] [ValidatePattern('^(([2]([0-4][0-9]|[5][0-5])|[0-1]?[0-9]?[0-9])[.]){3}(([2]([0-4][0-9]|[5][0-5])|[0-1]?[0-9]?[0-9]))$')] [string]$DnsServer1, [Parameter(ParameterSetName = 'Network')] [ValidatePattern('^(([2]([0-4][0-9]|[5][0-5])|[0-1]?[0-9]?[0-9])[.]){3}(([2]([0-4][0-9]|[5][0-5])|[0-1]?[0-9]?[0-9]))$')] [string]$DnsServer2, [Parameter(ParameterSetName = 'NetworkAdapter')] [AutomatedLab.NetworkAdapter[]]$NetworkAdapter, [switch]$IsDomainJoined, [Parameter(ValueFromPipelineByPropertyName = $true)] [switch]$DefaultDomain, [System.Management.Automation.PSCredential]$InstallationUserCredential, [ValidatePattern("(?=^.{1,254}$)|([\'\""])(^(?:(?!\d+\.|-)[a-zA-Z0-9_\-]{1,63}(?<!-)\.)+(?:[a-zA-Z]{2,})$)")] [string]$DomainName, [AutomatedLab.Role[]]$Roles, #Created ValidateSet using: "'" + ([System.Globalization.CultureInfo]::GetCultures([System.Globalization.CultureTypes]::InstalledWin32Cultures).Name -join "', '") + "'" | clip [ValidateScript( { $_ -in @([System.Globalization.CultureInfo]::GetCultures([System.Globalization.CultureTypes]::AllCultures).Name) })] [string]$UserLocale, [AutomatedLab.InstallationActivity[]]$PostInstallationActivity, [AutomatedLab.InstallationActivity[]]$PreInstallationActivity, [string]$ToolsPath, [string]$ToolsPathDestination, [AutomatedLab.VirtualizationHost]$VirtualizationHost = 'HyperV', [switch]$EnableWindowsFirewall, [string]$AutoLogonDomainName, [string]$AutoLogonUserName, [string]$AutoLogonPassword, [hashtable]$AzureProperties, [hashtable]$HypervProperties, [hashtable]$Notes, [switch]$PassThru, [string]$ResourceName, [switch]$SkipDeployment, [string]$AzureRoleSize, [string]$TimeZone, [string[]]$RhelPackage, [string[]]$SusePackage, [string[]]$UbuntuPackage, [string]$SshPublicKeyPath, [string]$SshPrivateKeyPath, [string]$OrganizationalUnit, [string]$ReferenceDisk, [string]$KmsServerName, [uint16]$KmsPort, [string]$KmsLookupDomain, [switch]$ActivateWindows, [string]$InitialDscConfigurationMofPath, [string]$InitialDscLcmConfigurationMofPath, [ValidateSet(1, 2)] [int] $VmGeneration ) begin { Write-LogFunctionEntry } process { $machineRoles = '' if ($Roles) { $machineRoles = " (Roles: $($Roles.Name -join ', '))" } $azurePropertiesValidKeys = 'ResourceGroupName', 'UseAllRoleSizes', 'RoleSize', 'LoadBalancerRdpPort', 'LoadBalancerWinRmHttpPort', 'LoadBalancerWinRmHttpsPort', 'LoadBalancerAllowedIp', 'SubnetName', 'UseByolImage', 'AutoshutdownTime', 'AutoshutdownTimezoneId', 'StorageSku', 'EnableSecureBoot', 'EnableTpm' $hypervPropertiesValidKeys = 'AutomaticStartAction', 'AutomaticStartDelay', 'AutomaticStopAction', 'EnableSecureBoot', 'SecureBootTemplate', 'EnableTpm' if (-not $VirtualizationHost -and -not (Get-LabDefinition).DefaultVirtualizationEngine) { Throw "Parameter 'VirtualizationHost' is mandatory when calling 'Add-LabMachineDefinition' if no default virtualization engine is specified" } if (-not $PSBoundParameters.ContainsKey('VirtualizationHost') -and (Get-LabDefinition).DefaultVirtualizationEngine) { $VirtualizationHost = (Get-LabDefinition).DefaultVirtualizationEngine } Write-ScreenInfo -Message (("Adding $($VirtualizationHost.ToString().Replace('HyperV', 'Hyper-V')) machine definition '$Name'").PadRight(47) + $machineRoles) -TaskStart if (-not (Get-LabDefinition)) { throw 'Please create a lab definition by calling New-LabDefinition before adding machines' } $script:lab = Get-LabDefinition if (($script:lab.DefaultVirtualizationEngine -eq 'Azure' -or $VirtualizationHost -eq 'Azure') -and -not $script:lab.AzureSettings) { try { Add-LabAzureSubscription } catch { throw "No Azure subscription added yet. Please run 'Add-LabAzureSubscription' first." } } if ($Global:labExported) { throw 'Lab is already exported. Please create a new lab definition by calling New-LabDefinition before adding machines' } if (Get-Lab -ErrorAction SilentlyContinue) { throw 'Lab is already imported. Please create a new lab definition by calling New-LabDefinition before adding machines' } if (-not $OperatingSystem) { $os = Get-LabAvailableOperatingSystem -UseOnlyCache -NoDisplay | Where-Object -Property OperatingSystemType -eq 'Windows' | Sort-Object Version | Select-Object -Last 1 if ($null -ne $os) { Write-ScreenInfo -Message "No operating system specified. Assuming you want $os ($(Split-Path -Leaf -Path $os.IsoPath))." $OperatingSystem = $os } else { throw "No operating system was defined for machine '$Name' and no default operating system defined. Please define either of these and retry. Call 'Get-LabAvailableOperatingSystem' to get a list of operating systems added to the lab." } } if (((Get-Command New-PSSession).Parameters.Values.Name -notcontains 'HostName') -and -not [string]::IsNullOrWhiteSpace($SshPublicKeyPath)) { Write-ScreenInfo -Type Warning -Message "SSH Transport is not available from within Windows PowerShell. Please use PowerShell 6+ if you want to use remoting-cmdlets." } if ((-not [string]::IsNullOrWhiteSpace($SshPublicKeyPath) -and [string]::IsNullOrWhiteSpace($SshPrivateKeyPath)) -or ([string]::IsNullOrWhiteSpace($SshPublicKeyPath) -and -not [string]::IsNullOrWhiteSpace($SshPrivateKeyPath))) { Write-ScreenInfo -Type Warning -Message "Both SshPublicKeyPath and SshPrivateKeyPath need to be used to successfully remote to Linux VMs (Host Windows, Engine Hyper-V) and Windows VMs (Host Linux/WSL, Engine Azure)" } if ($AzureProperties) { $illegalKeys = Compare-Object -ReferenceObject $azurePropertiesValidKeys -DifferenceObject ($AzureProperties.Keys | Sort-Object -Unique) | Where-Object SideIndicator -eq '=>' | Select-Object -ExpandProperty InputObject if ($AzureProperties.ContainsKey('StorageSku') -and ($AzureProperties['StorageSku'] -notin (Get-LabConfigurationItem -Name AzureDiskSkus))) { throw "$($AzureProperties['StorageSku']) is not in $(Get-LabConfigurationItem -Name AzureDiskSkus)" } if ($illegalKeys) { throw "The key(s) '$($illegalKeys -join ', ')' are not supported in AzureProperties. Valid keys are '$($azurePropertiesValidKeys -join ', ')'" } } if ($HypervProperties) { $illegalKeys = Compare-Object -ReferenceObject $hypervPropertiesValidKeys -DifferenceObject ($HypervProperties.Keys | Sort-Object -Unique) | Where-Object SideIndicator -eq '=>' | Select-Object -ExpandProperty InputObject if ($illegalKeys) { throw "The key(s) '$($illegalKeys -join ', ')' are not supported in HypervProperties. Valid keys are '$($hypervPropertiesValidKeys -join ', ')'" } } if ($global:labNamePrefix) { $Name = "$global:labNamePrefix$Name" } if ($null -eq $script:machines) { $errorMessage = "Create a new lab first using 'New-LabDefinition' before adding machines" Write-Error $errorMessage Write-LogFunctionExitWithError -Message $errorMessage return } if ($script:machines | Where-Object Name -eq $Name) { $errorMessage = "A machine with the name '$Name' does already exist" Write-Error $errorMessage Write-LogFunctionExitWithError -Message $errorMessage return } if ($script:machines | Where-Object IpAddress.IpAddress -eq $IpAddress) { $errorMessage = "A machine with the IP address '$IpAddress' does already exist" Write-Error $errorMessage Write-LogFunctionExitWithError -Message $errorMessage return } $machine = New-Object AutomatedLab.Machine if ($ReferenceDisk -and $script:lab.DefaultVirtualizationEngine -eq 'HyperV') { Write-ScreenInfo -Type Warning -Message "Usage of the ReferenceDisk parameter makes your lab essentially unsupportable. Don't be mad at us if we cannot reproduce your random issue if you bring your own images." $machine.ReferenceDiskPath = $ReferenceDisk } if ($ReferenceDisk -and $script:lab.DefaultVirtualizationEngine -ne 'HyperV') { Write-ScreenInfo -Type Warning -Message "Sorry, no custom reference disk allowed on $($script:lab.DefaultVirtualizationEngine). This parameter will be ignored." } if ($VmGeneration) { $machine.VmGeneration = $VmGeneration } $machine.Name = $Name $machine.FriendlyName = $ResourceName $machine.OrganizationalUnit = $OrganizationalUnit $script:machines.Add($machine) if ($SshPublicKeyPath -and -not (Test-Path -Path $SshPublicKeyPath)) { throw "$SshPublicKeyPath does not exist. Rethink your decision." } elseif ($SshPublicKeyPath -and (Test-Path -Path $SshPublicKeyPath)) { $machine.SshPublicKeyPath = $SshPublicKeyPath $machine.SshPublicKey = Get-Content -Raw -Path $SshPublicKeyPath } if ($SshPrivateKeyPath -and -not (Test-Path -Path $SshPrivateKeyPath)) { throw "$SshPrivateKeyPath does not exist. Rethink your decision." } elseif ($SshPrivateKeyPath -and (Test-Path -Path $SshPrivateKeyPath)) { $machine.SshPrivateKeyPath = $SshPrivateKeyPath } if ((Get-LabDefinition).DefaultVirtualizationEngine -and (-not $PSBoundParameters.ContainsKey('VirtualizationHost'))) { $VirtualizationHost = (Get-LabDefinition).DefaultVirtualizationEngine } if ($VirtualizationHost -eq 'Azure') { $script:lab.AzureSettings.LoadBalancerPortCounter++ $machine.LoadBalancerRdpPort = $script:lab.AzureSettings.LoadBalancerPortCounter $script:lab.AzureSettings.LoadBalancerPortCounter++ $machine.LoadBalancerWinRmHttpPort = $script:lab.AzureSettings.LoadBalancerPortCounter $script:lab.AzureSettings.LoadBalancerPortCounter++ $machine.LoadBalancerWinrmHttpsPort = $script:lab.AzureSettings.LoadBalancerPortCounter $script:lab.AzureSettings.LoadBalancerPortCounter++ $machine.LoadBalancerSshPort = $script:lab.AzureSettings.LoadBalancerPortCounter } if ($InstallationUserCredential) { $installationUser = New-Object AutomatedLab.User($InstallationUserCredential.UserName, $InstallationUserCredential.GetNetworkCredential().Password) } else { if ((Get-LabDefinition).DefaultInstallationCredential) { $installationUser = New-Object AutomatedLab.User((Get-LabDefinition).DefaultInstallationCredential.UserName, (Get-LabDefinition).DefaultInstallationCredential.Password) } else { switch ($VirtualizationHost) { 'HyperV' { $installationUser = New-Object AutomatedLab.User('Administrator', 'Somepass1') } 'Azure' { $installationUser = New-Object AutomatedLab.User('Install', 'Somepass1') } Default { $installationUser = New-Object AutomatedLab.User('Administrator', 'Somepass1') } } } } $machine.InstallationUser = $installationUser $machine.IsDomainJoined = $false if ($PSBoundParameters.ContainsKey('DefaultDomain') -and $DefaultDomain) { if (-not (Get-LabDomainDefinition)) { if ($VirtualizationHost -eq 'Azure') { Add-LabDomainDefinition -Name 'contoso.com' -AdminUser Install -AdminPassword 'Somepass1' } else { Add-LabDomainDefinition -Name 'contoso.com' -AdminUser Administrator -AdminPassword 'Somepass1' } } $DomainName = (Get-LabDomainDefinition)[0].Name } if ($DomainName -or ($Roles -and $Roles.Name -match 'DC$')) { $machine.IsDomainJoined = $true if ($script:Lab.DefaultVirtualizationEngine -eq 'HyperV' -and (-not $Roles -or $Roles -and $Roles.Name -notmatch 'DC$')) { $machine.HasDomainJoined = $true # In order to use the correct credentials upon connecting via SSH. Hyper-V VMs join during first boot } if ($Roles.Name -eq 'RootDC' -or $Roles.Name -eq 'DC') { if (-not $DomainName) { if (-not (Get-LabDomainDefinition)) { $DomainName = 'contoso.com' switch ($VirtualizationHost) { 'Azure' { Add-LabDomainDefinition -Name $DomainName -AdminUser Install -AdminPassword Somepass1 } 'HyperV' { Add-LabDomainDefinition -Name $DomainName -AdminUser Administrator -AdminPassword Somepass1 } 'VMware' { Add-LabDomainDefinition -Name $DomainName -AdminUser Administrator -AdminPassword Somepass1 } } } else { throw 'Domain name not specified for Root Domain Controller' } } } elseif ('FirstChildDC' -in $Roles.Name) { $role = $Roles | Where-Object Name -eq FirstChildDC $containsProperties = [boolean]$role.Properties if ($containsProperties) { $parentDomainInProperties = $role.Properties.ParentDomain $newDomainInProperties = $role.Properties.NewDomain Write-PSFMessage -Message "Machine contains custom properties for FirstChildDC: 'ParentDomain'='$parentDomainInProperties', 'NewDomain'='$newDomainInProperties'" } if ((-not $containsProperties) -and (-not $DomainName)) { Write-PSFMessage -Message 'Nothing specified (no DomainName nor ANY Properties). Giving up' throw 'Domain name not specified for Child Domain Controller' } if ((-not $DomainName) -and ((-not $parentDomainInProperties -or (-not $newDomainInProperties)))) { Write-PSFMessage -Message 'No DomainName or Properties for ParentName and NewDomain specified. Giving up' throw 'Domain name not specified for Child Domain Controller' } if ($containsProperties -and $parentDomainInProperties -and $newDomainInProperties -and (-not $DomainName)) { Write-PSFMessage -Message 'Properties specified but DomainName is not. Then populate DomainName based on Properties' $DomainName = "$($role.Properties.NewDomain).$($role.Properties.ParentDomain)" Write-PSFMessage -Message "Machine contains custom properties for FirstChildDC but DomainName parameter is not specified. Setting now to '$DomainName'" } elseif (((-not $containsProperties) -or ($containsProperties -and (-not $parentDomainInProperties) -and (-not $newDomainInProperties))) -and $DomainName) { $newDomainName = $DomainName.Substring(0, $DomainName.IndexOf('.')) $parentDomainName = $DomainName.Substring($DomainName.IndexOf('.') + 1) Write-PSFMessage -Message 'No Properties specified (or properties for ParentName and NewDomain omitted) but DomainName parameter is specified. Calculating/updating ParentDomain and NewDomain properties' if (-not $containsProperties) { $role.Properties = @{ 'NewDomain' = $newDomainName } $role.Properties.Add('ParentDomain', $parentDomainName) } else { if (-not $role.Properties.ContainsKey('NewDomain')) { $role.Properties.Add('NewDomain', $newDomainName) } if (-not $role.Properties.ContainsKey('ParentDomain')) { $role.Properties.Add('ParentDomain', $parentDomainName) } } $parentDomainInProperties = $role.Properties.ParentDomain $newDomainInProperties = $role.Properties.NewDomain Write-PSFMessage -Message "ParentDomain now set to '$parentDomainInProperties'" Write-PSFMessage -Message "NewDomain now set to '$newDomainInProperties'" } } if (-not (Get-LabDomainDefinition | Where-Object Name -eq $DomainName)) { if ($VirtualizationHost -eq 'Azure') { Add-LabDomainDefinition -Name $DomainName -AdminUser Install -AdminPassword 'Somepass1' } else { Add-LabDomainDefinition -Name $DomainName -AdminUser Administrator -AdminPassword 'Somepass1' } } $machine.DomainName = $DomainName } if (-not $OperatingSystem.Version) { if ($OperatingSystemVersion) { $OperatingSystem.Version = $OperatingSystemVersion } else { throw "Could not identify the version of operating system '$($OperatingSystem.OperatingSystemName)' assigned to machine '$Name'. The version is required to continue." } } switch ($OperatingSystem.Version.ToString(2)) { '6.0' { $level = 'Win2008' } '6.1' { $level = 'Win2008R2' } '6.2' { $level = 'Win2012' } '6.3' { $level = 'Win2012R2' } '6.4' { $level = 'WinThreshold' } '10.0' { $level = 'WinThreshold' } } $role = $roles | Where-Object Name -in ('RootDC', 'FirstChildDC', 'DC') if ($role) { if ($role.Properties) { if ($role.Name -eq 'RootDC') { if (-not $role.Properties.ContainsKey('ForestFunctionalLevel')) { $role.Properties.Add('ForestFunctionalLevel', $level) } } if ($role.Name -eq 'RootDC' -or $role.Name -eq 'FirstChildDC') { if (-not $role.Properties.ContainsKey('DomainFunctionalLevel')) { $role.Properties.Add('DomainFunctionalLevel', $level) } } } else { if ($role.Name -eq 'RootDC') { $role.Properties = @{'ForestFunctionalLevel' = $level } $role.Properties.Add('DomainFunctionalLevel', $level) } elseif ($role.Name -eq 'FirstChildDC') { $role.Properties = @{'DomainFunctionalLevel' = $level } } } } #Virtual network detection and automatic creation if ($VirtualizationHost -eq 'Azure') { if (-not (Get-LabVirtualNetworkDefinition)) { #No virtual networks has been specified Write-ScreenInfo -Message 'No virtual networks specified. Creating a network automatically' -Type Warning if (-not ($Global:existingAzureNetworks)) { $Global:existingAzureNetworks = Get-AzVirtualNetwork } #Virtual network name will be same as lab name $autoNetworkName = (Get-LabDefinition).Name #Priority 1. Check for existence of an Azure virtual network with same name as network name $existingNetwork = $Global:existingAzureNetworks | Where-Object { $_.Name -eq $autoNetworkName } if ($existingNetwork) { Write-PSFMessage -Message 'Virtual switch already exists with same name as lab being deployed. Trying to re-use.' $addressSpace = $existingNetwork.AddressSpace.AddressPrefixes Write-ScreenInfo -Message "Creating virtual network '$autoNetworkName' with address spacee '$addressSpace'" -Type Warning Add-LabVirtualNetworkDefinition -Name $autoNetworkName -AddressSpace $addressSpace[0] #First automatically assigned IP address will be following+1 $addressSpaceIpAddress = "$($addressSpace.Split('/')[0].Split('.')[0..2] -Join '.').5" $script:autoIPAddress = [AutomatedLab.IPAddress]$addressSpaceIpAddress $notDone = $false } else { Write-PSFMessage -Message 'No Azure virtual network found with same name as network name. Attempting to find unused network in the range 192.168.2.x-192.168.255.x' $networkFound = $false [int]$octet = 1 do { $octet++ $azureInUse = $false foreach ($azureNetwork in $Global:existingAzureNetworks.AddressSpace.AddressPrefixes) { if (Test-IpInSameSameNetwork -Ip1 "192.168.$octet.0/24" -Ip2 $azureNetwork) { $azureInUse = $true } } if ($azureInUse) { Write-PSFMessage -Message "Network '192.168.$octet.0/24' is in use by an existing Azure virtual network" continue } $networkFound = $true } until ($networkFound -or $octet -ge 255) if ($networkFound) { Write-ScreenInfo "Creating virtual network with name '$autoNetworkName' and address space '192.168.$octet.1/24'" -Type Warning Add-LabVirtualNetworkDefinition -Name $autoNetworkName -AddressSpace "192.168.$octet.1/24" } else { throw 'Virtual network could not be created. Please create virtual network manually by calling Add-LabVirtualNetworkDefinition (after calling New-LabDefinition)' } #First automatically asigned IP address will be following+1 $script:autoIPAddress = ([AutomatedLab.IPAddress]("192.168.$octet.5")).AddressAsString } #throw 'No virtual network is defined. Please call Add-LabVirtualNetworkDefinition before adding machines but after calling New-LabDefinition' } } elseif ($VirtualizationHost -eq 'HyperV') { Write-PSFMessage -Message 'Detect if a virtual switch already exists with same name as lab being deployed. If so, use this switch for defining network name and address space.' #this takes a lot of time hence it should be called only once in a deployment if (-not $script:existingHyperVVirtualSwitches) { $script:existingHyperVVirtualSwitches = Get-LabVirtualNetwork } $networkDefinitions = Get-LabVirtualNetworkDefinition if (-not $networkDefinitions) { #No virtual networks has been specified Write-ScreenInfo -Message 'No virtual networks specified. Creating a network automatically' -Type Warning #Virtual network name will be same as lab name $autoNetworkName = (Get-LabDefinition).Name #Priority 1. Check for existence of Hyper-V virtual switch with same name as network name $existingNetwork = $existingHyperVVirtualSwitches | Where-Object Name -eq $autoNetworkName if ($existingNetwork) { Write-PSFMessage -Message 'Virtual switch already exists with same name as lab being deployed. Trying to re-use.' Write-ScreenInfo -Message "Using virtual network '$autoNetworkName' with address space '$addressSpace'" -Type Info Add-LabVirtualNetworkDefinition -Name $autoNetworkName -AddressSpace $existingNetwork.AddressSpace } else { Write-PSFMessage -Message 'No virtual switch found with same name as network name. Attempting to find unused network' $addressSpace = Get-LabAvailableAddresseSpace if ($addressSpace) { Write-ScreenInfo "Creating network '$autoNetworkName' with address space '$addressSpace'" -Type Warning Add-LabVirtualNetworkDefinition -Name $autoNetworkName -AddressSpace $addressSpace } else { throw 'Virtual network could not be created. Please create virtual network manually by calling Add-LabVirtualNetworkDefinition (after calling New-LabDefinition)' } } } else { Write-PSFMessage -Message 'One or more virtual network(s) has been specified.' #Using first specified virtual network '$($networkDefinitions[0])' with address space '$($networkDefinitions[0].AddressSpace)'." <# if ($script:autoIPAddress) { #Network already created and IP range already found Write-PSFMessage -Message 'Network already created and IP range already found' } else { #> foreach ($networkDefinition in $networkDefinitions) { #check for an virtual switch having already the name of the new network switch $existingNetwork = $existingHyperVVirtualSwitches | Where-Object Name -eq $networkDefinition.ResourceName #does the current network definition has an address space assigned if ($networkDefinition.AddressSpace) { Write-PSFMessage -Message "Virtual network '$($networkDefinition.ResourceName)' specified with address space '$($networkDefinition.AddressSpace)'" #then check if the existing network has the same address space as the new one and throw an exception if not if ($existingNetwork) { if ($existingNetwork.SwitchType -eq 'External') { #Different address spaces for different labs reusing an existing External virtual switch is permitted, however this requires knowledge and support # for switching / routing fabrics external to AL and the host. Note to the screen this is an advanced configuration. if ($networkDefinition.AddressSpace -ne $existingNetwork.AddressSpace) { Write-ScreenInfo "Address space defined '$($networkDefinition.AddressSpace)' for network '$networkDefinition' is different from the address space '$($existingNetwork.AddressSpace)' used by currently existing Hyper-V switch with same name." -Type Warning Write-ScreenInfo "This is an advanced configuration, ensure external switching and routing is configured correctly" -Type Warning Write-PSFMessage -Message 'Existing External Hyper-V virtual switch found with different address space. This is an allowed advanced configuration' } else { Write-PSFMessage -Message 'Existing External Hyper-V virtual switch found with same name and address space as first virtual network specified. Using this.' } } else { if ($networkDefinition.AddressSpace -ne $existingNetwork.AddressSpace) { throw "Address space defined '$($networkDefinition.AddressSpace)' for network '$networkDefinition' is different from the address space '$($existingNetwork.AddressSpace)' used by currently existing Hyper-V switch with same name. Cannot continue." } } } else { #if the network does not already exist, verify if the address space if not already assigned $otherHypervSwitch = $existingHyperVVirtualSwitches | Where-Object AddressSpace -eq $networkDefinition.AddressSpace if ($otherHypervSwitch) { throw "Another Hyper-V virtual switch '$($otherHypervSwitch.Name)' is using address space specified in this lab ($($networkDefinition.AddressSpace)). Cannot continue." } #and also verify that the new address space is not overlapping with an exsiting one $otherHypervSwitch = $existingHyperVVirtualSwitches | Where-Object { $_.AddressSpace } | Where-Object { [AutomatedLab.IPNetwork]::Overlap($_.AddressSpace, $networkDefinition.AddressSpace) } | Select-Object -First 1 if ($otherHypervSwitch) { throw "The Hyper-V virtual switch '$($otherHypervSwitch.Name)' is using an address space ($($otherHypervSwitch.AddressSpace)) that overlaps with the specified one in this lab ($($networkDefinition.AddressSpace)). Cannot continue." } Write-PSFMessage -Message 'Address space specified is valid' } } else { if ($networkDefinition.SwitchType -eq 'External') { Write-PSFMessage 'External network interfaces will not get automatic IP addresses' continue } Write-PSFMessage -Message "Virtual network '$networkDefinition' specified but without address space specified" if ($existingNetwork) { Write-PSFMessage -Message "Existing Hyper-V virtual switch found with same name as first virtual network name. Using it with address space '$($existingNetwork.AddressSpace)'." $networkDefinition.AddressSpace = $existingNetwork.AddressSpace } else { Write-PSFMessage -Message 'No Hyper-V virtual switch found with same name as lab name. Attempting to find unused network.' $addressSpace = Get-LabAvailableAddresseSpace if ($addressSpace) { Write-ScreenInfo "Using network '$networkDefinition' with address space '$addressSpace'" -Type Warning $networkDefinition.AddressSpace = $addressSpace } else { throw 'Virtual network could not be used. Please create virtual network manually by calling Add-LabVirtualNetworkDefinition (after calling New-LabDefinition)' } } } } } } if ($Network) { $networkDefinition = Get-LabVirtualNetworkDefinition -Name $network if (-not $networkDefinition) { throw "A virtual network definition with the name '$Network' could not be found. To get a list of network definitions, use 'Get-LabVirtualNetworkDefinition'" } if ($networkDefinition.SwitchType -eq 'External' -and -not $networkDefinition.AddressSpace -and -not $IpAddress) { $useDhcp = $true } $NetworkAdapter = New-LabNetworkAdapterDefinition -VirtualSwitch $networkDefinition.Name -UseDhcp:$useDhcp } elseif (-not $NetworkAdapter) { if ((Get-LabVirtualNetworkDefinition).Count -eq 1) { $networkDefinition = Get-LabVirtualNetworkDefinition $NetworkAdapter = New-LabNetworkAdapterDefinition -VirtualSwitch $networkDefinition.Name } else { throw "Network cannot be determined for machine '$machine'. Either no networks is defined or more than one network is defined while network is not specified when calling this function" } } $machine.HostType = $VirtualizationHost foreach ($adapter in $NetworkAdapter) { $adapterVirtualNetwork = Get-LabVirtualNetworkDefinition -Name $adapter.VirtualSwitch #if there is no IPV4 address defined on the adapter if (-not $adapter.IpV4Address) { #if there is also no IP address defined on the machine and the adapter is not set to DHCP and the network the adapter is connected to does not know about an address space we cannot continue if (-not $IpAddress -and -not $adapter.UseDhcp -and -not $adapterVirtualNetwork.AddressSpace) { throw "The virtual network '$adapterVirtualNetwork' defined on machine '$machine' does not have an IP address assigned and is not set to DHCP" } elseif ($IpAddress) { if ($AzureProperties.SubnetName -and $adapterVirtualNetwork.Subnets.Count -gt 0) { $chosenSubnet = $adapterVirtualNetwork.Subnets | Where-Object Name -EQ $AzureProperties.SubnetName if (-not $chosenSubnet) { throw ('No fitting subnet available. Subnet {0} could not be found in the list of available subnets {1}' -f $AzureProperties.SubnetName, ($adapterVirtualNetwork.Subnets.Name -join ',')) } $adapter.Ipv4Address.Add([AutomatedLab.IPNetwork]::Parse($IpAddress, $chosenSubnet.AddressSpace.Netmask)) } elseif ($VirtualizationHost -eq 'Azure' -and $adapterVirtualNetwork.Subnets.Count -gt 0 -and -not $AzureProperties.SubnetName) { # No default subnet and no name selected. Chose fitting subnet. $chosenSubnet = $adapterVirtualNetwork.Subnets | Where-Object { $IpAddress -in (Get-NetworkRange -IPAddress $_.AddressSpace.IpAddress -SubnetMask $_.AddressSpace.Netmask) } if (-not $chosenSubnet) { throw ('No fitting subnet available. No subnet was found with a valid address range. {0} was not in the range of these subnets: ' -f $IpAddress, ($adapterVirtualNetwork.Subnets.Name -join ',')) } $adapter.Ipv4Address.Add([AutomatedLab.IPNetwork]::Parse($IpAddress, $chosenSubnet.AddressSpace.Netmask)) } else { $adapter.Ipv4Address.Add([AutomatedLab.IPNetwork]::Parse($IpAddress, $adapterVirtualNetwork.AddressSpace.Netmask)) } } elseif (-not $adapter.UseDhcp) { $ip = $adapterVirtualNetwork.NextIpAddress() if ($AzureProperties.SubnetName -and $adapterVirtualNetwork.Subnets.Count -gt 0) { $chosenSubnet = $adapterVirtualNetwork.Subnets | Where-Object Name -EQ $AzureProperties.SubnetName if (-not $chosenSubnet) { throw ('No fitting subnet available. Subnet {0} could not be found in the list of available subnets {1}' -f $AzureProperties.SubnetName, ($adapterVirtualNetwork.Subnets.Name -join ',')) } $adapter.Ipv4Address.Add([AutomatedLab.IPNetwork]::Parse($ip, $chosenSubnet.AddressSpace.Netmask)) } elseif ($VirtualizationHost -eq 'Azure' -and $adapterVirtualNetwork.Subnets.Count -gt 0 -and -not $AzureProperties.SubnetName) { # No default subnet and no name selected. Chose fitting subnet. $chosenSubnet = $adapterVirtualNetwork.Subnets | Where-Object { $ip -in (Get-NetworkRange -IPAddress $_.AddressSpace.IpAddress -SubnetMask $_.AddressSpace.Netmask) } if (-not $chosenSubnet) { throw ('No fitting subnet available. No subnet was found with a valid address range. {0} was not in the range of these subnets: ' -f $IpAddress, ($adapterVirtualNetwork.Subnets.Name -join ',')) } $adapter.Ipv4Address.Add([AutomatedLab.IPNetwork]::Parse($ip, $chosenSubnet.AddressSpace.Netmask)) } else { $adapter.Ipv4Address.Add([AutomatedLab.IPNetwork]::Parse($ip, $adapterVirtualNetwork.AddressSpace.Netmask)) } } } if ($DnsServer1) { $adapter.Ipv4DnsServers.Add($DnsServer1) } if ($DnsServer2) { $adapter.Ipv4DnsServers.Add($DnsServer2) } #if the virtual network is not external, the machine is not an Azure one, is domain joined and there is no DNS server configured if ($adapter.VirtualSwitch.SwitchType -ne 'External' -and $machine.HostType -ne 'Azure' -and #$machine.IsDomainJoined -and -not $adapter.UseDhcp -and -not ($DnsServer1 -or $DnsServer2 )) { $adapter.Ipv4DnsServers.Add('0.0.0.0') } if ($Gateway) { $adapter.Ipv4Gateway.Add($Gateway) } $machine.NetworkAdapters.Add($adapter) } Repair-LabDuplicateIpAddresses if ($processors -eq 0) { $processors = 1 if (-not $script:processors) { $script:processors = if ($IsLinux -or $IsMacOs) { $coreInf = Get-Content /proc/cpuinfo | Select-String 'siblings\s+:\s+\d+' | Select-Object -Unique [int]($coreInf -replace 'siblings\s+:\s+') } else { (Get-CimInstance -Namespace Root\CIMv2 -Class win32_processor | Measure-Object NumberOfLogicalProcessors -Sum).Sum } } if ($script:processors -ge 2) { $machine.Processors = 2 } } else { $machine.Processors = $Processors } if ($PSBoundParameters.ContainsKey('Memory')) { $machine.Memory = $Memory } else { $machine.Memory = 1 #Memory weight based on role of machine $machine.Memory = 1 foreach ($role in $Roles) { if ((Get-LabConfigurationItem -Name "MemoryWeight_$($role.Name)") -gt $machine.Memory) { $machine.Memory = Get-LabConfigurationItem -Name "MemoryWeight_$($role.Name)" } } } if ($PSBoundParameters.ContainsKey('MinMemory')) { $machine.MinMemory = $MinMemory } if ($PSBoundParameters.ContainsKey('MaxMemory')) { $machine.MaxMemory = $MaxMemory } $machine.EnableWindowsFirewall = $EnableWindowsFirewall $machine.AutoLogonDomainName = $AutoLogonDomainName $machine.AutoLogonUserName = $AutoLogonUserName $machine.AutoLogonPassword = $AutoLogonPassword if ($machine.HostType -eq 'HyperV') { if ($RhelPackage) { $machine.LinuxPackageGroup = $RhelPackage } if ($SusePackage) { $machine.LinuxPackageGroup = $SusePackage } if ($UbuntuPackage) { $machine.LinuxPackageGroup = $UbuntuPackage } if ($OperatingSystem.IsoPath) { $os = $OperatingSystem } if (-not $OperatingSystem.IsoPath -and $OperatingSystemVersion) { $os = Get-LabAvailableOperatingSystem -NoDisplay | Where-Object { $_.OperatingSystemName -eq $OperatingSystem -and $_.Version -eq $OperatingSystemVersion } } elseif (-not $OperatingSystem.IsoPath -and -not $OperatingSystemVersion) { $os = Get-LabAvailableOperatingSystem -NoDisplay | Where-Object OperatingSystemName -eq $OperatingSystem if ($os.Count -gt 1) { $os = $os | Group-Object -Property Version | Sort-Object -Property Name -Descending | Select-Object -First 1 | Select-Object -ExpandProperty Group Write-ScreenInfo "The operating system '$OperatingSystem' is available multiple times. Choosing the one with the highest version ($($os[0].Version))" -Type Warning } if ($os.Count -gt 1) { $os = $os | Sort-Object -Property { (Get-Item -Path $_.IsoPath).LastWriteTime } -Descending | Select-Object -First 1 Write-ScreenInfo "The operating system '$OperatingSystem' with the same version is available on multiple images. Choosing the one with the highest LastWriteTime to honor updated images ($((Get-Item -Path $os.IsoPath).LastWriteTime))" -Type Warning } } if (-not $os) { if ($OperatingSystemVersion) { throw "The operating system '$OperatingSystem' for machine '$Name' with version '$OperatingSystemVersion' could not be found in the available operating systems. Call 'Get-LabAvailableOperatingSystem' to get a list of operating systems added to the lab." } else { throw "The operating system '$OperatingSystem' for machine '$Name' could not be found in the available operating systems. Call 'Get-LabAvailableOperatingSystem' to get a list of operating systems added to the lab." } } $machine.OperatingSystem = $os } elseif ($machine.HostType -eq 'Azure') { $machine.OperatingSystem = $OperatingSystem } elseif ($machine.HostType -eq 'VMWare') { $machine.OperatingSystem = $OperatingSystem } if ($script:lab.DefaultVirtualizationEngine -eq 'HyperV' -and $InitialDscConfigurationMofPath -and -not (Test-Path $InitialDscConfigurationMofPath)) { throw "$InitialDscConfigurationMofPath does not exist. Make sure it exists and is a mof" } elseif ($script:lab.DefaultVirtualizationEngine -eq 'HyperV' -and $InitialDscConfigurationMofPath -and (Test-Path $InitialDscConfigurationMofPath)) { if ($Machine.OperatingSystem.Version -lt 10.0) { Write-ScreenInfo -Type Warning -Message "Integrated PowerShell version of $Machine is less than 5. Please keep in mind that DSC has been introduced in PS4 and some resources may not work with versions older than PS5."} $Machine.InitialDscConfigurationMofPath = $InitialDscConfigurationMofPath } if ($script:lab.DefaultVirtualizationEngine -eq 'HyperV' -and $InitialDscLcmConfigurationMofPath -and -not (Test-Path $InitialDscLcmConfigurationMofPath)) { throw "$InitialDscLcmConfigurationMofPath does not exist. Make sure it exists and is a meta.mof" } elseif ($script:lab.DefaultVirtualizationEngine -eq 'HyperV' -and $InitialDscLcmConfigurationMofPath -and (Test-Path $InitialDscLcmConfigurationMofPath)) { if ($Machine.OperatingSystem.Version -lt 10.0) { Write-ScreenInfo -Type Warning -Message "Integrated PowerShell version of $Machine is less than 5. Please keep in mind that DSC has been introduced in PS4 and some resources may not work with versions older than PS5."} $Machine.InitialDscLcmConfigurationMofPath = $InitialDscLcmConfigurationMofPath } if (-not $TimeZone) { $TimeZone = (Get-TimeZone).StandardName } $machine.Timezone = $TimeZone if (-not $UserLocale) { $UserLocale = (Get-Culture).Name -replace '-POSIX' } $machine.UserLocale = $UserLocale $machine.Roles = $Roles $machine.PostInstallationActivity = $PostInstallationActivity $machine.PreInstallationActivity = $PreInstallationActivity if (($KmsLookupDomain -or $KmsServerName -or $ActivateWindows.IsPresent) -and $null -eq $Notes) { $Notes = @{} } if ($KmsLookupDomain) { $Notes['KmsLookupDomain'] = $KmsLookupDomain } elseif ($KmsServerName) { $Notes['KmsServerName'] = $KmsServerName $Notes['KmsPort'] = $KmsPort -as [string] } if ($ActivateWindows.IsPresent) { $Notes['ActivateWindows'] = '1' } if ($HypervProperties) { $machine.HypervProperties = $HypervProperties } if ($AzureProperties) { if ($AzureRoleSize) { $AzureProperties['RoleSize'] = $AzureRoleSize # Adding keys to properties later did silently fail } $machine.AzureProperties = $AzureProperties } if ($AzureRoleSize -and -not $AzureProperties) { $machine.AzureProperties = @{ RoleSize = $AzureRoleSize } } $machine.ToolsPath = $ToolsPath.Replace('<machinename>', $machine.Name) $machine.ToolsPathDestination = $ToolsPathDestination $type = Get-Type -GenericType AutomatedLab.ListXmlStore -T AutomatedLab.Disk $machine.Disks = New-Object $type if ($DiskName) { foreach ($disk in $DiskName) { $labDisk = $script:disks | Where-Object Name -eq $disk if (-not $labDisk) { throw "The disk with the name '$disk' has not yet been added to the lab. Do this first using the cmdlet 'Add-LabDiskDefinition'" } $machine.Disks.Add($labDisk) } } $machine.SkipDeployment = $SkipDeployment } end { if ($Notes) { $machine.Notes = $Notes } Write-ScreenInfo -Message 'Done' -TaskEnd if ($PassThru) { $machine } Write-LogFunctionExit } } function Export-LabDefinition { [CmdletBinding()] param ( [switch] $Force, [switch] $ExportDefaultUnattendedXml = $true, [switch] $Silent ) Write-LogFunctionEntry if (Get-LabMachineDefinition | Where-Object HostType -eq 'HyperV') { $osesCount = (Get-LabAvailableOperatingSystem -NoDisplay).Count } #Automatic DNS configuration in Azure if no DNS server is specified and an AD is being deployed foreach ($network in (Get-LabVirtualNetworkDefinition | Where HostType -eq Azure)) { $rootDCs = Get-LabMachineDefinition -Role RootDC | Where-Object Network -eq $network $dnsServerIP = $rootDCs.IpV4Address if (-not $network.DnsServers -and $dnsServerIP) { $network.DnsServers = $dnsServerIP if (-not $Silent) { Write-ScreenInfo -Message "No DNS server was defined for Azure virtual network while AD is being deployed. Setting DNS server to IP address of '$($rootDCs -join ',')'" -Type Warning } } } #Automatic DNS (client) configuration of machines $firstRootDc = Get-LabMachineDefinition -Role RootDC | Select-Object -First 1 $firstRouter = Get-LabMachineDefinition -Role Routing | Select-Object -First 1 $firstRouterExternalSwitch = $firstRouter.NetworkAdapters | Where-Object { $_.VirtualSwitch.SwitchType -eq 'External' } if ($firstRootDc -or $firstRouter) { foreach ($machine in (Get-LabMachineDefinition)) { if ($firstRouter) { $mappingNetworks = Compare-Object -ReferenceObject $firstRouter.NetworkAdapters.VirtualSwitch.Name ` -DifferenceObject $machine.NetworkAdapters.VirtualSwitch.Name -ExcludeDifferent -IncludeEqual } foreach ($networkAdapter in $machine.NetworkAdapters) { if ($networkAdapter.IPv4DnsServers -contains '0.0.0.0') { if (-not $machine.IsDomainJoined) #machine is not domain joined, the 1st network adapter's IP of the 1st root DC is used as DNS server { if ($firstRootDc) { $networkAdapter.IPv4DnsServers = $firstRootDc.NetworkAdapters[0].Ipv4Address[0].IpAddress } elseif ($firstRouter) { if ($networkAdapter.VirtualSwitch.Name -in $mappingNetworks.InputObject) { $networkAdapter.IPv4DnsServers = ($firstRouter.NetworkAdapters | Where-Object { $_.VirtualSwitch.Name -eq $networkAdapter.VirtualSwitch.Name }).Ipv4Address.IpAddress } } } elseif ($machine.Roles.Name -contains 'RootDC') #if the machine is RootDC, its 1st network adapter's IP is used for DNS { $networkAdapter.IPv4DnsServers = $machine.NetworkAdapters[0].Ipv4Address[0].IpAddress } elseif ($machine.Roles.Name -contains 'FirstChildDC') #if it is a FirstChildDc, the 1st network adapter's IP of the corresponsing RootDC is used { $firstChildDcRole = $machine.Roles | Where-Object Name -eq 'FirstChildDC' $roleParentDomain = $firstChildDcRole.Properties.ParentDomain $rootDc = Get-LabMachineDefinition -Role RootDC | Where-Object DomainName -eq $roleParentDomain $networkAdapter.IPv4DnsServers = $machine.NetworkAdapters[0].Ipv4Address[0].IpAddress, $rootDc.NetworkAdapters[0].Ipv4Address[0].IpAddress } elseif ($machine.Roles.Name -contains 'DC') { $parentDc = Get-LabMachineDefinition -Role RootDC,FirstChildDc | Where-Object DomainName -eq $machine.DomainName | Select-Object -First 1 $networkAdapter.IPv4DnsServers = $machine.NetworkAdapters[0].Ipv4Address[0].IpAddress, $parentDc.NetworkAdapters[0].Ipv4Address[0].IpAddress } else #machine is domain joined and not a RootDC or FirstChildDC { Write-PSFMessage "Looking for a root DC in the machine's domain '$($machine.DomainName)'" $rootDc = Get-LabMachineDefinition -Role RootDC | Where-Object DomainName -eq $machine.DomainName if ($rootDc) { Write-PSFMessage "RootDC found, using the IP address of '$rootDc' for DNS: " $networkAdapter.IPv4DnsServers = $rootDc.NetworkAdapters[0].Ipv4Address[0].IpAddress } else { Write-PSFMessage "No RootDC found, looking for FirstChildDC in the machine's domain" $firstChildDC = Get-LabMachineDefinition -Role FirstChildDC | Where-Object DomainName -eq $machine.DomainName if ($firstChildDC) { $networkAdapter.IPv4DnsServers = $firstChildDC.NetworkAdapters[0].Ipv4Address[0].IpAddress } else { Write-ScreenInfo "Automatic assignment of DNS server did not work for machine '$machine'. No domain controller could be found for domain '$($machine.DomainName)'" -Type Warning } } } } #if there is a router in the network and no gateways defined, we try to set the gateway automatically. This does not #apply to network adapters that have a gateway manually configured or set to DHCP, any network adapter on a router, #or if there is there wasn't found an external network adapter on the router ($firstRouterExternalSwitch) if ($networkAdapter.Ipv4Gateway.Count -eq 0 -and $firstRouterExternalSwitch -and $machine.Roles.Name -notcontains 'Routing' -and -not $networkAdapter.UseDhcp ) { if ($networkAdapter.VirtualSwitch.Name -in $mappingNetworks.InputObject) { $networkAdapter.Ipv4Gateway.Add(($firstRouter.NetworkAdapters | Where-Object { $_.VirtualSwitch.Name -eq $networkAdapter.VirtualSwitch.Name } | Select-Object -First 1).Ipv4Address.IpAddress) } } } } } if (Get-LabMachineDefinition | Where-Object HostType -eq HyperV) { $hypervMachines = Get-LabMachineDefinition | Where-Object { $_.HostType -eq 'HyperV' -and -not $_.SkipDeployment } $hypervUsedOperatingSystems = Get-LabAvailableOperatingSystem -NoDisplay | Where-Object OperatingSystemImageName -in $hypervMachines.OperatingSystem.OperatingSystemName $spaceNeededBaseDisks = ($hypervUsedOperatingSystems | Measure-Object -Property Size -Sum).Sum $spaceBaseDisksAlreadyClaimed = ($hypervUsedOperatingSystems | Measure-Object -Property size -Sum).Sum $spaceNeededData = ($hypervMachines | Where-Object { -not (Get-LWHypervVM -Name $_.ResourceName -ErrorAction SilentlyContinue) }).Count * 2GB $spaceNeeded = $spaceNeededBaseDisks + $spaceNeededData - $spaceBaseDisksAlreadyClaimed Write-PSFMessage -Message "Space needed by HyperV base disks: $([int]($spaceNeededBaseDisks / 1GB))" Write-PSFMessage -Message "Space needed by HyperV base disks but already claimed: $([int]($spaceBaseDisksAlreadyClaimed / 1GB * -1))" Write-PSFMessage -Message "Space estimated for HyperV data: $([int]($spaceNeededData / 1GB))" if (-not $Silent) { Write-ScreenInfo -Message "Estimated (additional) local drive space needed for all machines: $([System.Math]::Round(($spaceNeeded / 1GB),2)) GB" -Type Info } $labTargetPath = (Get-LabDefinition).Target.Path if ($labTargetPath) { if (-not (Test-Path -Path $labTargetPath)) { try { Write-PSFMessage "Creating new folder '$labTargetPath'" New-Item -ItemType Directory -Path $labTargetPath -ErrorAction Stop | Out-Null } catch { Write-Error -Message "Could not create folder '$labTargetPath'. Please make sure that the folder is accessibe and you have permission to write." return } } Write-PSFMessage "Calling 'Get-LabFreeDiskSpace' targeting path '$labTargetPath'" $freeSpace = (Get-LabFreeDiskSpace -Path $labTargetPath).FreeBytesAvailable Write-PSFMessage "Free disk space is '$([Math]::Round($freeSpace / 1GB, 2))GB'" if ($freeSpace -lt $spaceNeeded) { throw "VmPath parameter is specified for the lab and contains: '$labTargetPath'. However, estimated needed space be $([int]($spaceNeeded / 1GB))GB but drive has only $([System.Math]::Round($freeSpace / 1GB)) GB of free space" } } else { Set-LabLocalVirtualMachineDiskAuto $labTargetPath = (Get-LabDefinition).Target.Path if (-not $labTargetPath) { Throw 'No local drive found matching requirements for free space' } } if (-not $Silent) { Write-ScreenInfo -Message "Location of Hyper-V machines will be '$labTargetPath'" } } if (-not $lab.LabFilePath) { $lab.LabFilePath = Join-Path -Path $script:labPath -ChildPath (Get-LabConfigurationItem LabFileName) $script:lab | Add-Member -Name Path -MemberType NoteProperty -Value $labFilePath -Force } if (-not (Test-Path $script:labPath)) { New-Item -Path $script:labPath -ItemType Directory | Out-Null } if (Test-Path -Path $lab.LabFilePath) { if ($Force) { Remove-Item -Path $lab.LabFilePath } else { Write-Error 'The file does already exist' -TargetObject $lab.LabFilePath return } } try { $script:lab.Export($lab.LabFilePath) } catch { throw $_ } $machineFilePath = $script:lab.MachineDefinitionFiles[0].Path $diskFilePath = $script:lab.DiskDefinitionFiles[0].Path if (Test-Path -Path $machineFilePath) { if ($Force) { Remove-Item -Path $machineFilePath } else { Write-Error 'The file does already exist' -TargetObject $machineFilePath return } } $script:machines.Export($machineFilePath) $script:disks.Export($diskFilePath) if ($ExportDefaultUnattendedXml) { if ($script:machines.Count -eq 0) { Write-ScreenInfo 'There are no machines defined, nothing to export' -Type Warning } else { if ($Script:machines.OperatingSystem | Where-Object Version -lt '6.2') { $unattendedXmlDefaultContent2008 | Out-File -FilePath (Join-Path -Path $script:lab.Sources.UnattendedXml.Value -ChildPath Unattended2008.xml) -Encoding unicode } if ($Script:machines.OperatingSystem | Where-Object Version -ge '6.2') { $unattendedXmlDefaultContent2012 | Out-File -FilePath (Join-Path -Path $script:lab.Sources.UnattendedXml.Value -ChildPath Unattended2012.xml) -Encoding unicode } if ($Script:machines | Where-Object {$_.LinuxType -eq 'RedHat' -and $_.OperatingSystem.Version -ge 9.0}) { $kickstartContent.Replace('install','').Trim() | Out-File -FilePath (Join-Path -Path $script:lab.Sources.UnattendedXml.Value -ChildPath ks_default.cfg) -Encoding unicode $kickstartContent.Replace(' --non-interactive','') | Out-File -FilePath (Join-Path -Path $script:lab.Sources.UnattendedXml.Value -ChildPath ks_defaultLegacy.cfg) -Encoding unicode } elseif ($Script:machines | Where-Object LinuxType -eq 'RedHat') { $kickstartContent | Out-File -FilePath (Join-Path -Path $script:lab.Sources.UnattendedXml.Value -ChildPath ks_default.cfg) -Encoding unicode $kickstartContent.Replace(' --non-interactive','') | Out-File -FilePath (Join-Path -Path $script:lab.Sources.UnattendedXml.Value -ChildPath ks_defaultLegacy.cfg) -Encoding unicode } if ($Script:machines | Where-Object LinuxType -eq 'Suse') { $autoyastContent | Out-File -FilePath (Join-Path -Path $script:lab.Sources.UnattendedXml.Value -ChildPath autoinst_default.xml) -Encoding unicode } if ($Script:machines | Where-Object LinuxType -eq 'Ubuntu') { $cloudInitContent | Out-File -FilePath (Join-Path -Path $script:lab.Sources.UnattendedXml.Value -ChildPath cloudinit_default.yml) -Encoding unicode } } } $Global:labExported = $true Write-LogFunctionExit } function Get-DiskSpeed { [CmdletBinding()] param ( [ValidatePattern('[a-zA-Z]')] [Parameter(Mandatory)] [string]$DriveLetter, [ValidateRange(1, 50)] [int]$Interations = 1 ) Write-LogFunctionEntry if (-not $labSources) { $labSources = Get-LabSourcesLocation } $IsReadOnly = Get-Partition -DriveLetter ($DriveLetter.TrimEnd(':')) | Select-Object -ExpandProperty IsReadOnly if ($IsReadOnly) { Write-ScreenInfo -Message "Drive $DriveLetter is read-only. Skipping disk speed test" -Type Warning $readThroughoutRandom = 0 $writeThroughoutRandom = 0 } else { Write-ScreenInfo -Message "Measuring speed of drive $DriveLetter" -Type Info $tempFileName = [System.IO.Path]::GetTempFileName() & "$labSources\Tools\WinSAT.exe" disk -ran -read -count $Interations -drive $DriveLetter -xml $tempFileName | Out-Null $readThroughoutRandom = (Select-Xml -Path $tempFileName -XPath '/WinSAT/Metrics/DiskMetrics/AvgThroughput').Node.'#text' & "$labSources\Tools\WinSAT.exe" disk -ran -write -count $Interations -drive $DriveLetter -xml $tempFileName | Out-Null $writeThroughoutRandom = (Select-Xml -Path $tempFileName -XPath '/WinSAT/Metrics/DiskMetrics/AvgThroughput').Node.'#text' Remove-Item -Path $tempFileName } $result = New-Object PSObject -Property ([ordered]@{ ReadRandom = $readThroughoutRandom WriteRandom = $writeThroughoutRandom }) $result Write-LogFunctionExit } function Get-LabAvailableAddresseSpace { $defaultAddressSpace = Get-LabConfigurationItem -Name DefaultAddressSpace if (-not $defaultAddressSpace) { Write-Error 'Could not get the PrivateData value DefaultAddressSpace. Cannot find an available address space.' return } $existingHyperVVirtualSwitches = Get-LabVirtualNetwork $networkFound = $false $addressSpace = [AutomatedLab.IPNetwork]$defaultAddressSpace if ($null -eq $reservedAddressSpaces) { $script:reservedAddressSpaces = @() } do { $addressSpace = $addressSpace.Increment() $conflictingSwitch = $existingHyperVVirtualSwitches | Where-Object AddressSpace -eq $addressSpace if ($conflictingSwitch) { Write-PSFMessage -Message "Network '$addressSpace' is in use by existing Hyper-V virtual switch '$conflictingSwitch'" continue } if ($addressSpace -in $reservedAddressSpaces) { Write-PSFMessage -Message "Network '$addressSpace' has already been defined in this lab" continue } $localAddresses = if ($IsLinux) { (ip -4 addr) | grep -oP '(?<=inet\s)\d+(\.\d+){3}' } else { (Get-NetIPAddress -AddressFamily IPv4).IPAddress } if ($addressSpace.IpAddress -in $localAddresses) { Write-PSFMessage -Message "Network '$addressSpace' is in use locally" continue } $route = if ($IsLinux) { (route | Select-Object -First 5 -Skip 2 | ForEach-Object { '{0}/{1}' -f ($_ -split '\s+')[0], (ConvertTo-MaskLength ($_ -split '\s+')[2]) }) } else { Get-NetRoute -DestinationPrefix $addressSpace.ToString() -ErrorAction SilentlyContinue } if ($null -ne $route) { Write-PSFMessage -Message "Network '$addressSpace' is routable" continue } $networkFound = $true } until ($networkFound) $script:reservedAddressSpaces += $addressSpace $addressSpace } function Get-LabDefinition { [CmdletBinding()] [OutputType([AutomatedLab.Lab])] param () Write-LogFunctionEntry return $script:lab Write-LogFunctionExit } function Get-LabDomainDefinition { Write-LogFunctionEntry return $script:lab.Domains Write-LogFunctionExit } function Get-LabInstallationActivity { [CmdletBinding()] param ( [Parameter(Mandatory, ParameterSetName = 'FileContentDependencyRemoteScript')] [Parameter(Mandatory, ParameterSetName = 'FileContentDependencyLocalScript')] [string]$DependencyFolder, [Parameter(Mandatory, ParameterSetName = 'IsoImageDependencyRemoteScript')] [Parameter(Mandatory, ParameterSetName = 'IsoImageDependencyLocalScript')] [string]$IsoImage, [Parameter(ParameterSetName = 'FileContentDependencyRemoteScript')] [Parameter(ParameterSetName = 'FileContentDependencyLocalScript')] [Parameter(ParameterSetName = 'CustomRole')] [switch]$KeepFolder, [Parameter(Mandatory, ParameterSetName = 'FileContentDependencyRemoteScript')] [Parameter(Mandatory, ParameterSetName = 'IsoImageDependencyRemoteScript')] [string]$ScriptFileName, [Parameter(Mandatory, ParameterSetName = 'IsoImageDependencyLocalScript')] [Parameter(Mandatory, ParameterSetName = 'FileContentDependencyLocalScript')] [string]$ScriptFilePath, [Parameter(ParameterSetName = 'CustomRole')] [hashtable]$Properties, [System.Management.Automation.PSVariable[]]$Variable, [System.Management.Automation.FunctionInfo[]]$Function, [switch]$DoNotUseCredSsp, [string]$CustomRole ) begin { Write-LogFunctionEntry $activity = New-Object -TypeName AutomatedLab.InstallationActivity if ($Variable) { $activity.SerializedVariables = $Variable | ConvertTo-PSFClixml} if ($Function) { $activity.SerializedFunctions = $Function | ConvertTo-PSFClixml} if (-not $Properties) { $Properties = @{ } } } process { if ($PSCmdlet.ParameterSetName -like 'FileContentDependency*') { $activity.DependencyFolder = $DependencyFolder $activity.KeepFolder = $KeepFolder.ToBool() if ($ScriptFilePath) { $activity.ScriptFilePath = $ScriptFilePath } else { $activity.ScriptFileName = $ScriptFileName } } elseif ($PSCmdlet.ParameterSetName -like 'IsoImage*') { $activity.IsoImage = $IsoImage if ($ScriptFilePath) { $activity.ScriptFilePath = $ScriptFilePath } else { $activity.ScriptFileName = $ScriptFileName } } elseif ($PSCmdlet.ParameterSetName -eq 'CustomRole') { $activity.DependencyFolder = Join-Path -Path (Join-Path -Path (Get-LabSourcesLocation -Local) -ChildPath 'CustomRoles') -ChildPath $CustomRole $activity.KeepFolder = $KeepFolder.ToBool() $activity.ScriptFileName = "$CustomRole.ps1" $activity.IsCustomRole = $true #The next sections compares the given custom role properties with with the custom role parameters. #Custom role parameters are taken form the main role script as well as the HostStart.ps1 and the HostEnd.ps1 $scripts = $activity.ScriptFileName, 'HostStart.ps1', 'HostEnd.ps1' $unknownParameters = New-Object System.Collections.Generic.List[string] foreach ($script in $scripts) { $scriptFullName = Join-Path -Path $activity.DependencyFolder -ChildPath $script if (-not (Test-Path -Path $scriptFullName)) { continue } $scriptInfo = Get-Command -Name $scriptFullName $commonParameters = [System.Management.Automation.Internal.CommonParameters].GetProperties().Name $parameters = $scriptInfo.Parameters.GetEnumerator() | Where-Object Key -NotIn $commonParameters #If the custom role knows about a ComputerName parameter and if there is no value defined by the user, add add empty value now. #Later that will be filled with the computer name of the computer the role is assigned to when the HostStart and the HostEnd scripts are invoked. if ($Properties) { if (($parameters | Where-Object Key -eq 'ComputerName') -and -not $Properties.ContainsKey('ComputerName')) { $Properties.Add('ComputerName', '') } } #test if all mandatory parameters are defined foreach ($parameter in $parameters) { if ($parameter.Value.Attributes.Mandatory -and -not $properties.ContainsKey($parameter.Key)) { Write-Error "There is no value defined for mandatory property '$($parameter.Key)' and custom role '$CustomRole'" -ErrorAction Stop } } #test if there are custom role properties defined that do not map to the custom role parameters if ($Properties) { foreach ($property in $properties.GetEnumerator()) { if (-not $scriptInfo.Parameters.ContainsKey($property.Key) -and -not $unknownParameters.Contains($property.Key)) { $unknownParameters.Add($property.Key) } } } } #antoher loop is required to remove all unknown parameters that are added due to the order of the first loop foreach ($script in $scripts) { $scriptFullName = Join-Path -Path $activity.DependencyFolder -ChildPath $script if (-not (Test-Path -Path $scriptFullName)) { continue } $scriptInfo = Get-Command -Name $scriptFullName $commonParameters = [System.Management.Automation.Internal.CommonParameters].GetProperties().Name $parameters = $scriptInfo.Parameters.GetEnumerator() | Where-Object Key -NotIn $commonParameters if ($Properties) { foreach ($property in $properties.GetEnumerator()) { if ($scriptInfo.Parameters.ContainsKey($property.Key) -and $unknownParameters.Contains($property.Key)) { $unknownParameters.Remove($property.Key) | Out-Null } } } } if ($unknownParameters.Count -gt 0) { Write-Error "The defined properties '$($unknownParameters -join ', ')' are unknown for custom role '$CustomRole'" -ErrorAction Stop } if ($Properties) { $activity.SerializedProperties = $Properties | ConvertTo-PSFClixml -ErrorAction SilentlyContinue } } $activity.DoNotUseCredSsp = $DoNotUseCredSsp } end { Write-LogFunctionExit -ReturnValue $activity return $activity } } function Get-LabIsoImageDefinition { Write-LogFunctionEntry $script:lab.Sources.ISOs Write-LogFunctionExit } function Get-LabMachineDefinition { [CmdletBinding(DefaultParameterSetName = 'ByName')] [OutputType([AutomatedLab.Machine])] param ( [Parameter(Position = 0, ParameterSetName = 'ByName', ValueFromPipeline, ValueFromPipelineByPropertyName)] [ValidateNotNullOrEmpty()] [string[]]$ComputerName, [Parameter(Mandatory, ParameterSetName = 'ByRole')] [AutomatedLab.Roles]$Role, [Parameter(Mandatory, ParameterSetName = 'All')] [switch]$All ) begin { #required to suporess verbose messages, warnings and errors Get-CallerPreference -Cmdlet $PSCmdlet -SessionState $ExecutionContext.SessionState Write-LogFunctionEntry $result = @() } process { if ($PSCmdlet.ParameterSetName -eq 'ByName') { if ($ComputerName) { foreach ($n in $ComputerName) { $machine = $Script:machines | Where-Object Name -in $n if (-not $machine) { continue } $result += $machine } } else { $result = $Script:machines } } if ($PSCmdlet.ParameterSetName -eq 'ByRole') { $result = $Script:machines | Where-Object { $_.Roles.Name } | Where-Object { $_.Roles | Where-Object { $Role.HasFlag([AutomatedLab.Roles]$_.Name) } } if (-not $result) { return } } if ($PSCmdlet.ParameterSetName -eq 'All') { $result = $Script:machines } } end { $result } } function Get-LabMachineRoleDefinition { [CmdletBinding()] param ( [Parameter(Mandatory)] [AutomatedLab.Roles] $Role, [hashtable] $Properties, [switch] $Syntax ) $roleObjects = @() $availableRoles = [Enum]::GetNames([AutomatedLab.Roles]) $config = Get-LabConfigurationItem -Name ValidationSettings foreach ($availableRole in $availableRoles) { if ($Role.HasFlag([AutomatedLab.Roles]$availableRole)) { if ($Syntax.IsPresent -and $config.ValidRoleProperties.Contains($availableRole.ToString())) { $roleObjects += "Get-LabMachineRoleDefinition -Role $availableRole -Properties @{`r`n$($config.ValidRoleProperties[$availableRole.ToString()] -join `"='value'`r`n`")='value'`r`n}`r`n" } elseif ($Syntax.IsPresent -and -not $config.ValidRoleProperties.Contains($availableRole.ToString())) { $roleObjects += "Get-LabMachineRoleDefinition -Role $availableRole`r`n" } else { $roleObject = New-Object -TypeName AutomatedLab.Role $roleObject.Name = $availableRole $roleObject.Properties = $Properties $roleObjects += $roleObject } } } return $roleObjects } function Get-LabVirtualNetwork { [cmdletBinding(DefaultParameterSetName = 'ByName')] param ( [Parameter(ParameterSetName = 'ByName')] [string]$Name, [Parameter(ParameterSetName = 'All')] [switch]$All ) $virtualnetworks = @() $lab = Get-Lab -ErrorAction SilentlyContinue if (-not $lab) { return } $switches = if ($IsLinux) { return } $switches = if ($Name) { $Name | foreach { Get-VMSwitch -Name $_ } } elseif ($All) { Get-VMSwitch } else { Get-VMSwitch | Where-Object Name -in $lab.VirtualNetworks.Name } foreach ($switch in $switches) { $network = New-Object AutomatedLab.VirtualNetwork $network.Name = $switch.Name $network.SwitchType = $switch.SwitchType.ToString() $ipAddress = Get-NetIPAddress -AddressFamily IPv4 | Where-Object { $_.InterfaceAlias -eq "vEthernet ($($network.Name))" -and $_.PrefixOrigin -eq 'manual' } | Select-Object -First 1 if ($ipAddress) { $network.AddressSpace = "$($ipAddress.IPAddress)/$($ipAddress.PrefixLength)" } $network.Notes = Get-LWHypervNetworkSwitchDescription -NetworkSwitchName $switch.Name $virtualnetworks += $network } $virtualnetworks } function Get-LabVolumesOnPhysicalDisks { [CmdletBinding()] $physicalDisks = Get-PhysicalDisk $disks = Get-CimInstance -Class Win32_DiskDrive $labVolumes = foreach ($disk in $disks) { $query = 'ASSOCIATORS OF {{Win32_DiskDrive.DeviceID="{0}"}} WHERE AssocClass=Win32_DiskDriveToDiskPartition' -f $disk.DeviceID.Replace('\', '\\') $partitions = Get-CimInstance -Query $query foreach ($partition in $partitions) { $query = 'ASSOCIATORS OF {{Win32_DiskPartition.DeviceID="{0}"}} WHERE AssocClass=Win32_LogicalDiskToPartition' -f $partition.DeviceID $volumes = Get-CimInstance -Query $query foreach ($volume in $volumes) { Get-Volume -DriveLetter $volume.DeviceId[0] | Add-Member -Name Serial -MemberType NoteProperty -Value $disk.SerialNumber -PassThru | Add-Member -Name Signature -MemberType NoteProperty -Value $disk.Signature -PassThru } } } $labVolumes | Select-Object -ExpandProperty DriveLetter | Sort-Object | ForEach-Object { $localDisk = New-Object AutomatedLab.LocalDisk($_) $localDisk.Serial = $_.Serial $localDisk.Signature = $_.Signature $localDisk } } function Import-LabDefinition { [CmdletBinding(DefaultParameterSetName = 'ByName')] param ( [Parameter(Mandatory, ParameterSetName = 'ByPath', Position = 1)] [string]$Path, [Parameter(Mandatory, ParameterSetName = 'ByName', Position = 1)] [string]$Name, [switch]$PassThru ) Write-LogFunctionEntry Clear-Lab $type = Get-Type -GenericType AutomatedLab.ListXmlStore -T AutomatedLab.Machine $script:machines = New-Object $type $type = Get-Type -GenericType AutomatedLab.ListXmlStore -T AutomatedLab.Disk $script:disks = New-Object $type $script:labPath = "$((Get-LabConfigurationItem -Name LabAppDataRoot))/Labs/$Name" $machineDefinitionFilePath = Join-Path -Path $script:labPath -ChildPath (Get-LabConfigurationItem MachineFileName) $diskDefinitionFilePath = Join-Path -Path $script:labPath -ChildPath (Get-LabConfigurationItem DiskFileName) $global:labExported = $false if ($PSCmdlet.ParameterSetName -in 'ByPath', 'ByName') { if ($Name) { $Path = "$((Get-LabConfigurationItem -Name LabAppDataRoot))/Labs/$Name" } if (Test-Path -Path $Path -PathType Container) { $newPath = Join-Path -Path $Path -ChildPath Lab.xml if (-not (Test-Path -Path $newPath -PathType Leaf)) { throw "The file '$newPath' is missing. Please point to an existing lab file / folder." } else { $Path = $newPath } } elseif (Test-Path -Path $Path -PathType Leaf) { #file is there, do nothing } else { throw "The file '$Path' is missing. Please point to an existing lab file / folder." } if (-not ($IsLinux -or $IsMacOs) -and -not (Test-IsAdministrator)) { throw 'Import-Lab needs to be called in an elevated PowerShell session.' } if (Test-Path -Path $Path) { $Script:lab = [AutomatedLab.Lab]::Import((Resolve-Path -Path $Path)) $Script:lab | Add-Member -MemberType ScriptMethod -Name GetMachineTargetPath -Value { param ( [string]$MachineName ) (Join-Path -Path $this.Target.Path -ChildPath $MachineName) } } else { throw 'Lab Definition File not found' } #import all the machine files referenced in the lab.xml $type = Get-Type -GenericType AutomatedLab.ListXmlStore -T AutomatedLab.Machine $importMethodInfo = $type.GetMethod('Import',[System.Reflection.BindingFlags]::Public -bor [System.Reflection.BindingFlags]::Static, [System.Type]::DefaultBinder, [Type[]]@([string]), $null) try { $Script:lab.Machines = $importMethodInfo.Invoke($null, $Script:lab.MachineDefinitionFiles[0].Path) if ($Script:lab.MachineDefinitionFiles.Count -gt 1) { foreach ($machineDefinitionFile in $Script:lab.MachineDefinitionFiles[1..($Script:lab.MachineDefinitionFiles.Count - 1)]) { $Script:lab.Machines.AddFromFile($machineDefinitionFile.Path) } } if ($Script:lab.Machines) { $Script:lab.Machines | Add-Member -MemberType ScriptProperty -Name UnattendedXmlContent -Value { if ($this.OperatingSystem.Version -lt '6.2') { $Path = Join-Path -Path (Get-Lab).Sources.UnattendedXml.Value -ChildPath 'Unattended2008.xml' } else { $Path = Join-Path -Path (Get-Lab).Sources.UnattendedXml.Value -ChildPath 'Unattended2012.xml' } if ($this.OperatingSystemType -eq 'Linux' -and $this.LinuxType -eq 'RedHat' -and $this.OperatingSystem.Version -lt 8.0) { $Path = Join-Path -Path (Get-Lab).Sources.UnattendedXml.Value -ChildPath ks_defaultLegacy.cfg } if ($this.OperatingSystemType -eq 'Linux' -and $this.LinuxType -eq 'RedHat' -and $this.OperatingSystem.Version -ge 8.0) { $Path = Join-Path -Path (Get-Lab).Sources.UnattendedXml.Value -ChildPath ks_default.cfg } if ($this.OperatingSystemType -eq 'Linux' -and $this.LinuxType -eq 'Suse') { $Path = Join-Path -Path (Get-Lab).Sources.UnattendedXml.Value -ChildPath autoinst_default.xml } if ($this.OperatingSystemType -eq 'Linux' -and $this.LinuxType -eq 'Suse') { $Path = Join-Path -Path (Get-Lab).Sources.UnattendedXml.Value -ChildPath cloudinit_default.yml } return (Get-Content -Path $Path) } } } catch { Write-Error -Message "No machines imported from file $machineDefinitionFile" -Exception $_.Exception -ErrorAction Stop } #import all the disk files referenced in the lab.xml $type = Get-Type -GenericType AutomatedLab.ListXmlStore -T AutomatedLab.Disk $importMethodInfo = $type.GetMethod('Import',[System.Reflection.BindingFlags]::Public -bor [System.Reflection.BindingFlags]::Static, [System.Type]::DefaultBinder, [Type[]]@([string]), $null) try { $Script:lab.Disks = $importMethodInfo.Invoke($null, $Script:lab.DiskDefinitionFiles[0].Path) if ($Script:lab.DiskDefinitionFiles.Count -gt 1) { foreach ($diskDefinitionFile in $Script:lab.DiskDefinitionFiles[1..($Script:lab.DiskDefinitionFiles.Count - 1)]) { $Script:lab.Disks.AddFromFile($diskDefinitionFile.Path) } } } catch { Write-ScreenInfo "No disks imported from file '$diskDefinitionFile': $($_.Exception.Message)" -Type Warning } if ($Script:lab.VMWareSettings.DataCenterName) { Add-LabVMWareSettings -DataCenterName $Script:lab.VMWareSettings.DataCenterName ` -DataStoreName $Script:lab.VMWareSettings.DataStoreName ` -ResourcePoolName $Script:lab.VMWareSettings.ResourcePoolName ` -VCenterServerName $Script:lab.VMWareSettings.VCenterServerName ` -Credential ([System.Management.Automation.PSSerializer]::Deserialize($Script:lab.VMWareSettings.Credential)) } if (-not ($IsLinux -or $IsMacOs) -and (Get-LabConfigurationItem -Name OverridePowerPlan)) { $powerSchemeBackup = (powercfg.exe -GETACTIVESCHEME).Split(':')[1].Trim().Split()[0] powercfg.exe -setactive 8c5e7fda-e8bf-4a96-9a85-a6e23a8c635c } } elseif($PSCmdlet.ParameterSetName -eq 'ByValue') { $Script:lab = [AutomatedLab.Lab]::Import($LabBytes) } $script:machines = $script:lab.Machines $script:disks = $script:lab.Disks if ($PassThru) { $Script:lab } Write-LogFunctionExit } function New-LabDefinition { [CmdletBinding()] param ( [string]$Name, [string]$VmPath = (Get-LabConfigurationItem -Name VmPath), [int]$ReferenceDiskSizeInGB = 50, [long]$MaxMemory = 0, [hashtable]$Notes, [switch]$UseAllMemory = $false, [switch]$UseStaticMemory = $false, [ValidateSet('Azure', 'HyperV', 'VMWare')] [string]$DefaultVirtualizationEngine, [switch]$Passthru ) Write-LogFunctionEntry $global:PSLog_Indent = 0 $hostOSVersion = ([Environment]::OSVersion).Version if (-Not $IsLinux -and (($hostOSVersion -lt [System.Version]'6.2') -or (($hostOSVersion -ge [System.Version]'6.4') -and ($hostOSVersion.Build -lt '14393')))) { $osName = $(([Environment]::OSVersion).VersionString.PadRight(10)) $osBuild = $(([Environment]::OSVersion).Version.ToString().PadRight(11)) Write-PSFMessage -Level Host '***************************************************************************' Write-PSFMessage -Level Host ' THIS HOST MACHINE IS NOT RUNNING AN OS SUPPORTED BY AUTOMATEDLAB!' Write-PSFMessage -Level Host '' Write-PSFMessage -Level Host ' Operating System detected as:' Write-PSFMessage -Level Host " Name: $osName" Write-PSFMessage -Level Host " Build: $osBuild" Write-PSFMessage -Level Host '' Write-PSFMessage -Level Host ' AutomatedLab is supported on the following virtualization platforms' Write-PSFMessage -Level Host '' Write-PSFMessage -Level Host ' - Microsoft Azure' Write-PSFMessage -Level Host ' - Windows 2016 1607 or newer' Write-PSFMessage -Level Host ' - Windows 10 1607 or newer' Write-PSFMessage -Level Host ' - Windows 8.1 Professional or Enterprise' Write-PSFMessage -Level Host ' - Windows 2012 R2' Write-PSFMessage -Level Host '***************************************************************************' } if ($DefaultVirtualizationEngine -eq 'Azure') { Clear-Lab $null = Test-LabAzureModuleAvailability -ErrorAction SilentlyContinue } #settings for a new log #reset the log and its format $Global:AL_DeploymentStart = $null $Global:taskStart = @() $Global:indent = 0 $global:AL_CurrentLab = $null $Global:labDeploymentNoNewLine = $false $Script:reservedAddressSpaces = $null Write-ScreenInfo -Message 'Initialization' -TimeDelta ([timespan]0) -TimeDelta2 ([timespan]0) -TaskStart $hostOsName = if (($IsLinux -or $IsMacOs) -and (Get-Command -Name lsb_release -ErrorAction SilentlyContinue)) { lsb_release -d -s } elseif (-not ($IsLinux -or $IsMacOs)) # easier than IsWindows, which does not exist in Windows PowerShell... { (Get-CimInstance -ClassName Win32_OperatingSystem).Caption } else { 'Unknown' } Write-ScreenInfo -Message "Host operating system version: '$hostOsName, $($hostOSVersion.ToString())'" if (-not $Name) { $reservedMacAddresses = @() #Microsoft $reservedMacAddresses += '00:03:FF' $reservedMacAddresses += '00:0D:3A' $reservedMacAddresses += '00:12:5A' $reservedMacAddresses += '00:15:5D' $reservedMacAddresses += '00:17:FA' $reservedMacAddresses += '00:50:F2' $reservedMacAddresses += '00:1D:D8' #VMware $reservedMacAddresses += '00:05:69' $reservedMacAddresses += '00:0C:29' $reservedMacAddresses += '00:1C:14' $reservedMacAddresses += '00:50:56' #Citrix $reservedMacAddresses += '00:16:3E' $macAddress = Get-OnlineAdapterHardwareAddress | Where-Object { $_.SubString(0, 8) -notin $reservedMacAddresses } | Select-Object -Unique $Name = "$($env:COMPUTERNAME)$($macAddress.SubString(12,2))$($macAddress.SubString(15,2))" Write-ScreenInfo -Message "Lab name and network name has automatically been generated as '$Name' (if not overridden)" } Write-ScreenInfo -Message "Creating new lab definition with name '$Name'" #remove the current lab from memory if (Get-Lab -ErrorAction SilentlyContinue) { Clear-Lab } $global:labExported = $false $global:firstAzureVMCreated = $false $global:existingAzureNetworks = @() $global:cacheVMs = $null $script:existingHyperVVirtualSwitches = $null #cleanup $PSDefaultParameterValues for entries for AL functions $automatedLabPSDefaultParameterValues = $global:PSDefaultParameterValues.GetEnumerator() | Where-Object { (Get-Command ($_.Name).Split(':')[0] -ErrorAction SilentlyContinue).Module -like 'Automated*' } if ($automatedLabPSDefaultParameterValues) { foreach ($entry in $automatedLabPSDefaultParameterValues) { $global:PSDefaultParameterValues.Remove($entry.Name) Write-ScreenInfo -Message "Entry '$($entry.Name)' with value '$($entry.Value)' was removed from `$PSDefaultParameterValues. If needed, modify `$PSDefaultParameterValues after calling New-LabDefinition'" -Type Warning } } if (Get-Variable -Name 'autoIPAddress' -Scope Script -ErrorAction SilentlyContinue) { Remove-Variable -Name 'AutoIPAddress' -Scope Script } if ($global:labNamePrefix) { $Name = "$global:labNamePrefix$Name" } $script:labPath = "$((Get-LabConfigurationItem -Name LabAppDataRoot))/Labs/$Name" Write-ScreenInfo -Message "Location of lab definition files will be '$($script:labpath)'" $script:lab = New-Object AutomatedLab.Lab $script:lab.Name = $Name Update-LabSysinternalsTools while (Get-LabVirtualNetworkDefinition) { Remove-LabVirtualNetworkDefinition -Name (Get-LabVirtualNetworkDefinition)[0].Name } $machineDefinitionFilePath = Join-Path -Path $script:labPath -ChildPath (Get-LabConfigurationItem MachineFileName) $machineDefinitionFile = New-Object AutomatedLab.MachineDefinitionFile $machineDefinitionFile.Path = $machineDefinitionFilePath $script:lab.MachineDefinitionFiles.Add($machineDefinitionFile) $diskDefinitionFilePath = Join-Path -Path $script:labPath -ChildPath (Get-LabConfigurationItem DiskFileName) $diskDefinitionFile = New-Object AutomatedLab.DiskDefinitionFile $diskDefinitionFile.Path = $diskDefinitionFilePath $script:lab.DiskDefinitionFiles.Add($diskDefinitionFile) $sourcesPath = $labSources if (-not $sourcesPath) { $sourcesPath = New-LabSourcesFolder } Write-ScreenInfo -Message "Location of LabSources folder is '$sourcesPath'" if (-not (Get-LabIsoImageDefinition) -and $DefaultVirtualizationEngine -ne 'Azure') { if (-not (Get-ChildItem -Path "$(Get-LabSourcesLocation)\ISOs" -Filter *.iso -Recurse)) { Write-ScreenInfo -Message "No ISO files found in $(Get-LabSourcesLocation)\ISOs folder. If using Hyper-V for lab machines, please add ISO files manually using 'Add-LabIsoImageDefinition'" -Type Warning } Write-ScreenInfo -Message 'Auto-adding ISO files' -TaskStart Get-LabAvailableOperatingSystem -Path "$(Get-LabSourcesLocation)\ISOs" | Out-Null #for updating the cache if necessary Add-LabIsoImageDefinition -Path "$(Get-LabSourcesLocation)\ISOs" Write-ScreenInfo -Message 'Done' -TaskEnd } if ($DefaultVirtualizationEngine) { $script:lab.DefaultVirtualizationEngine = $DefaultVirtualizationEngine } if ($MaxMemory -ne 0) { $script:lab.MaxMemory = $MaxMemory } if ($UseAllMemory) { $script:lab.MaxMemory = 4TB } $script:lab.UseStaticMemory = $UseStaticMemory $script:lab.Sources.UnattendedXml = $script:labPath if ($VmPath) { $Script:lab.target.Path = $vmPath Write-ScreenInfo -Message "Path for VMs specified as '$($script:lab.Target.Path)'" -Type Info } $script:lab.Target.ReferenceDiskSizeInGB = $ReferenceDiskSizeInGB $type = Get-Type -GenericType AutomatedLab.ListXmlStore -T AutomatedLab.Machine $script:machines = New-Object $type $type = Get-Type -GenericType AutomatedLab.ListXmlStore -T AutomatedLab.Disk $script:disks = New-Object $type $script:lab.Notes = $Notes if ($Passthru) { $script:lab } $global:AL_CurrentLab = $script:lab Register-LabArgumentCompleters Write-LogFunctionExit } function Remove-LabDomainDefinition { [CmdletBinding()] param ( [Parameter(Mandatory)] [string]$Name ) Write-LogFunctionEntry $domain = $script:lab.Domains | Where-Object { $_.Name -eq $Name } if (-not $domain) { Write-ScreenInfo "There is no domain defined with the name '$Name'" -Type Warning } else { [Void]$script:lab.Domains.Remove($domain) Write-PSFMessage "Domain '$Name' removed. Lab has $($Script:lab.Domains.Count) domain(s) defined" } Write-LogFunctionExit } function Remove-LabIsoImageDefinition { [CmdletBinding()] param ( [Parameter(Mandatory = $true, ValueFromPipelineByPropertyName = $true)] [string]$Name ) Write-LogFunctionEntry $iso = $script:lab.Sources.ISOs | Where-Object -FilterScript { $_.Name -eq $Name } if (-not $iso) { Write-ScreenInfo "There is no Iso Image defined with the name '$Name'" -Type Warning } else { [Void]$script:lab.Sources.ISOs.Remove($iso) Write-PSFMessage "Iso Image '$Name' removed. Lab has $($Script:lab.Sources.ISOs.Count) Iso Image(s) defined" } Write-LogFunctionExit } function Remove-LabMachineDefinition { [CmdletBinding()] param ( [Parameter(Mandatory = $true, ValueFromPipelineByPropertyName = $true)] [string]$Name ) Write-LogFunctionEntry $machine = $script:machines | Where-Object Name -eq $Name if (-not $machine) { Write-ScreenInfo "There is no machine defined with the name '$Name'" -Type Warning } else { [Void]$script:machines.Remove($machine) Write-PSFMessage "Machine '$Name' removed. Lab has $($Script:machines.Count) machine(s) defined" } Write-LogFunctionExit } function Repair-LabDuplicateIpAddresses { [CmdletBinding()] param ( ) foreach ($machine in (Get-LabMachineDefinition)) { foreach ($adapter in $machine.NetworkAdapters) { foreach ($ipAddress in $adapter.Ipv4Address | Where-Object { $_.IPAddress.IsAutoGenerated }) { $currentIp = $ipAddress $otherIps = (Get-LabMachineDefinition | Where-Object Name -ne $machine.Name).NetworkAdapters.IPV4Address while ($ipAddress.IpAddress -in $otherIps.IpAddress) { $ipAddress.IpAddress = $ipAddress.IpAddress.Increment() } $adapter.Ipv4Address.Remove($currentIp) | Out-Null $adapter.Ipv4Address.Add($ipAddress) } } } } function Set-LabDefinition { param ( [AutomatedLab.Lab] $Lab, [AutomatedLab.Machine[]] $Machines, [AutomatedLab.Disk[]] $Disks ) if ($Lab) { $script:lab = $Lab } if ($Machines) { if (-not $script:machines) { $script:machines = New-Object 'AutomatedLab.SerializableList[AutomatedLab.Machine]' } $script:machines.Clear() $Machines | ForEach-Object { $script:Machines.Add($_) } } if ($Disks) { $script:Disks.Clear() $Disks | ForEach-Object { $script:Disks.Add($_) } } } function Set-LabLocalVirtualMachineDiskAuto { [CmdletBinding()] param ( [int64] $SpaceNeeded ) $type = Get-Type -GenericType AutomatedLab.ListXmlStore -T AutomatedLab.LocalDisk $drives = New-Object $type #read the cache try { if ($IsLinux -or $IsMacOs) { $cachedDrives = $type::Import((Join-Path -Path (Get-LabConfigurationItem -Name LabAppDataRoot) -ChildPath 'Stores/LocalDisks.xml')) } else { $cachedDrives = $type::ImportFromRegistry('Cache', 'LocalDisks') } Write-PSFMessage "Read $($cachedDrives.Count) drive infos from the cache" } catch { Write-PSFMessage 'Could not read info from the cache' } #Retrieve drives with enough space for placement of VMs foreach ($drive in (Get-LabVolumesOnPhysicalDisks | Where-Object FreeSpace -ge $SpaceNeeded)) { $drives.Add($drive) } if (-not $drives) { return $false } #if the current disk config is different from the is in the cache, wait until the running lab deployment is done. if ($cachedDrives -and (Compare-Object -ReferenceObject $drives.DriveLetter -DifferenceObject $cachedDrives.DriveLetter)) { $labDiskDeploymentInProgressPath = Get-LabConfigurationItem -Name DiskDeploymentInProgressPath if (Test-Path -Path $labDiskDeploymentInProgressPath) { Write-ScreenInfo "Another lab disk deployment seems to be in progress. If this is not correct, please delete the file '$labDiskDeploymentInProgressPath'." -Type Warning Write-ScreenInfo "Waiting with 'Get-DiskSpeed' until other disk deployment is finished. Otherwise a mounted virtual disk could be chosen for deployment." -NoNewLine do { Write-ScreenInfo -Message . -NoNewLine Start-Sleep -Seconds 15 } while (Test-Path -Path $labDiskDeploymentInProgressPath) } Write-ScreenInfo 'done' #refresh the list of drives with enough space for placement of VMs $drives.Clear() foreach ($drive in (Get-LabVolumesOnPhysicalDisks | Where-Object FreeSpace -ge $SpaceNeeded)) { $drives.Add($drive) } if (-not $drives) { return $false } } Write-Debug -Message "Drive letters placed on physical drives: $($drives.DriveLetter -Join ', ')" foreach ($drive in $drives) { Write-Debug -Message "Drive $drive free space: $($drive.FreeSpaceGb)GB)" } #Measure speed on drives found Write-PSFMessage -Message 'Measuring speed on fixed drives...' for ($i = 0; $i -lt $drives.Count; $i++) { $drive = $drives[$i] if ($cachedDrives -contains $drive) { $drive = ($cachedDrives -eq $drive)[0] $drives[$drives.IndexOf($drive)] = $drive Write-PSFMessage -Message "(cached) Measurements for drive $drive (serial: $($drive.Serial)) (signature: $($drive.Signature)): Read=$([int]($drive.ReadSpeed)) MB/s Write=$([int]($drive.WriteSpeed)) MB/s Total=$([int]($drive.TotalSpeed)) MB/s" } else { $result = Get-DiskSpeed -DriveLetter $drive.DriveLetter $drive.ReadSpeed = $result.ReadRandom $drive.WriteSpeed = $result.WriteRandom Write-PSFMessage -Message "Measurements for drive $drive (serial: $($drive.Serial)) (signature: $($drive.Signature)): Read=$([int]($drive.ReadSpeed)) MB/s Write=$([int]($drive.WriteSpeed)) MB/s Total=$([int]($drive.TotalSpeed)) MB/s" } } if ($IsLinux -or $IsMacOs) { $drives.Export((Join-Path -Path (Get-LabConfigurationItem -Name LabAppDataRoot) -ChildPath 'Stores/LocalDisks.xml')) } else { $drives.ExportToRegistry('Cache', 'LocalDisks') } #creating a new list is required as otherwise $drives would be converted into an Object[] $drives = $drives | Sort-Object -Property TotalSpeed -Descending $bootDrive = $drives | Where-Object DriveLetter -eq $env:SystemDrive[0] if ($bootDrive) { Write-PSFMessage -Message "Boot drive is drive '$bootDrive'" } else { Write-PSFMessage -Message 'Boot drive is not part of the selected drive' } if ($drives[0] -ne $bootDrive) { #Fastest drive is not the boot drive. Selecting this drive! Write-PSFMessage -Message "Selecing drive $($drives[0].DriveLetter) for VMs based on speed and NOT being the boot drive" $script:lab.Target.Path = "$($drives[0].DriveLetter):\AutomatedLab-VMs" } else { if ($drives.Count -lt 2) { Write-PSFMessage "Selecing drive $($drives[0].DriveLetter) for VMs as it is the only one" $script:lab.Target.Path = "$($drives[0].DriveLetter):\AutomatedLab-VMs" } #Fastest drive is the boot drive. If speed on next fastest drive is close to the boot drive in speed (within 50%), select this drive now instead of the boot drive #If not, select the boot drive elseif (($drives[1].TotalSpeed * 100 / $drives[0].TotalSpeed) -gt 50) { Write-PSFMessage "Selecing drive $($drives[1].DriveLetter) for VMs based on speed and NOT being the boot drive" Write-PSFMessage "Selected disk speed compared to system disk is $(($drives[1].TotalSpeed * 100 / $drives[0].TotalSpeed))%" $script:lab.Target.Path = "$($drives[1].DriveLetter):\AutomatedLab-VMs" } else { Write-PSFMessage "Selecing drive $($drives[0].DriveLetter) for VMs based on speed though this drive is actually the boot drive but is much faster than second fastest drive ($($drives[1].DriveLetter))" Write-PSFMessage ('Selected system disk, speed of next fastest disk compared to system disk is {0:P}' -f ($drives[1].TotalSpeed / $drives[0].TotalSpeed)) $script:lab.Target.Path = "$($drives[0].DriveLetter):\AutomatedLab-VMs" } } } function Test-LabDefinition { [CmdletBinding()] param ( [string]$Path, [switch]$Quiet ) Write-LogFunctionEntry $lab = Get-LabDefinition if (-not $lab) { $lab = Get-Lab -ErrorAction SilentlyContinue } if (-not $lab -and -not $Path) { Write-Error 'There is no lab loaded and no path specified. Please either import a lab using Import-Lab or point to a lab.xml document using the path parameter' return $false } if (-not $Path) { $Path = Join-Path -Path $lab.LabPath -ChildPath (Get-LabConfigurationItem LabFileName) } $labDefinition = Import-LabDefinition -Path $Path -PassThru $skipHostFileModification = Get-LabConfigurationItem -Name SkipHostFileModification foreach ($machine in (Get-LabMachineDefinition | Where-Object HostType -in 'HyperV', 'VMware' )) { $hostEntry = Get-HostEntry -HostName $machine if ($machine.FriendlyName -or $skipHostFileModification) { continue #if FriendlyName / ResourceName is defined, host file will not be modified } if ($hostEntry -and $hostEntry.IpAddress.IPAddressToString -ne $machine.IpV4Address) { Write-ScreenInfo "There is already an entry for machine '$($machine.Name)' in the hosts file pointing to other IP address(es) ($((Get-HostEntry -HostName $machine).IpAddress.IPAddressToString -join ',')) than the machine '$($machine.Name)' in this lab will have ($($machine.IpV4Address)). Cannot continue." $wrongIpInHostEntry = $true } } if ($wrongIpInHostEntry) { return $false } #we need to get the machine config files as well $machineDefinitionFiles = $labDefinition.MachineDefinitionFiles.Path Write-PSFMessage "There are $($machineDefinitionFiles.Count) machine XML file referenced in the lab xml file" foreach ($machineDefinitionFile in $machineDefinitionFiles) { if (-not (Test-Path -Path $machineDefinitionFile)) { throw 'Error importing the machines. Verify the paths in the section <MachineDefinitionFiles> of the lab definition XML file.' } } $Script:ValidationPass = $true Write-PSFMessage 'Starting validation against all xml files' try { [AutomatedLab.XmlValidatorArgs]::XmlPath = $Path $summaryMessageContainer = New-Object AutomatedLab.ValidationMessageContainer $assembly = [System.Reflection.Assembly]::GetAssembly([AutomatedLab.ValidatorBase]) $validatorCount = 0 foreach ($t in $assembly.GetTypes()) { if ($t.IsSubclassOf([AutomatedLab.ValidatorBase])) { try { $validator = [AutomatedLab.ValidatorBase][System.Activator]::CreateInstance($t) Write-Debug "Validator '$($validator.MessageContainer.ValidatorName)' took $($validator.Runtime.TotalMilliseconds) milliseconds" $summaryMessageContainer += $validator.MessageContainer $validatorCount++ } catch { Write-ScreenInfo "Could not invoke validator $t" -Type Warning } } } $summaryMessageContainer.AddSummary() } catch { throw $_ } Write-PSFMessage -Message "Lab Validation complete, overvall runtime was $($summaryMessageContainer.Runtime)" $messages = $summaryMessageContainer | ForEach-Object { $_.GetFilteredMessages('All') } if (-not $Quiet) { Write-ScreenInfo ($messages | Where-Object { $_.Type -band [AutomatedLab.MessageType]::Default } | Out-String) if ($VerbosePreference -eq 'Continue') { Write-PSFMessage ($messages | Where-Object { $_.Type -band [AutomatedLab.MessageType]::VerboseDebug } | Out-String) } } else { if ($messages | Where-Object { $_.Type -band [AutomatedLab.MessageType]::Warning }) { $messages | Where-Object { $_.Type -band [AutomatedLab.MessageType]::Warning } | ForEach-Object ` { Write-ScreenInfo -Message "Issue: '$($_.TargetObject)'. Cause: $($_.Message)" -Type Warning } } if ($messages | Where-Object { $_.Type -band [AutomatedLab.MessageType]::Error }) { $messages | Where-Object { $_.Type -band [AutomatedLab.MessageType]::Error } | ForEach-Object ` { Write-ScreenInfo -Message "Issue: '$($_.TargetObject)'. Cause: $($_.Message)" -Type Error } } } if ($messages | Where-Object Type -eq ([AutomatedLab.MessageType]::Error)) { $Script:ValidationPass = $false $false } else { $Script:ValidationPass = $true $true } Write-LogFunctionExit } function Add-LabVirtualNetworkDefinition { [CmdletBinding()] param ( [string]$Name = (Get-LabDefinition).Name, [AllowNull()] [AutomatedLab.IPNetwork]$AddressSpace, [AutomatedLab.VirtualizationHost]$VirtualizationEngine, [hashtable[]]$HyperVProperties, [hashtable[]]$AzureProperties, [AutomatedLab.NetworkAdapter]$ManagementAdapter, [string]$ResourceName, [switch]$PassThru ) Write-LogFunctionEntry if ((Get-LabDefinition).DefaultVirtualizationEngine -eq 'Azure' -and -not ((Get-LabDefinition).AzureSettings)) { Add-LabAzureSubscription } $azurePropertiesValidKeys = 'Subnets', 'LocationName', 'DnsServers', 'ConnectToVnets', 'DnsLabel' $hypervPropertiesValidKeys = 'SwitchType', 'AdapterName', 'ManagementAdapter' if (-not (Get-LabDefinition)) { throw 'No lab defined. Please call New-LabDefinition first before calling Add-LabVirtualNetworkDefinition.' } $script:lab = Get-LabDefinition if (-not $VirtualizationEngine) { if ((Get-LabDefinition).DefaultVirtualizationEngine) { $VirtualizationEngine = (Get-LabDefinition).DefaultVirtualizationEngine } else { Throw "Virtualization engine MUST be specified. This can be done:`n - Using parameter 'DefaultVirtualizationEngine' when calling New-LabDefinition`n - Using Set-LabDefaultVirtualizationEngine -Engine <engine>`n - Using parameter 'VirtualizationEngine' when calling Add-LabVirtualNetworkDefinition`n `nRemember to specify VirtualizationEngine parameter when adding machines if no default virtualization engine has been specified`n `n " } } if ($VirtualizationEngine -eq 'HyperV' -and (-not (Get-Module -ListAvailable -Name Hyper-V))) { throw 'The Hyper-V tools are not installed. Please install them first to use AutomatedLab with Hyper-V. Alternatively, you can use AutomatedLab with Microsoft Azure.' } if ($VirtualizationEngine -eq 'Azure' -and -not $script:lab.AzureSettings.DefaultResourceGroup) { Add-LabAzureSubscription } if ($AzureProperties) { $illegalKeys = Compare-Object -ReferenceObject $azurePropertiesValidKeys -DifferenceObject ($AzureProperties.Keys | Sort-Object -Unique) | Where-Object SideIndicator -eq '=>' | Select-Object -ExpandProperty InputObject if ($illegalKeys) { throw "The key(s) '$($illegalKeys -join ', ')' are not supported in AzureProperties. Valid keys are '$($azurePropertiesValidKeys -join ', ')'" } if (($AzureProperties.Keys -eq 'LocationName').Count -ne 1) { throw 'Location must be speficfied exactly once in AzureProperties' } } if ($HyperVProperties) { $illegalKeys = Compare-Object -ReferenceObject $hypervPropertiesValidKeys -DifferenceObject ($HyperVProperties.Keys | Select-Object -Unique) | Where-Object SideIndicator -eq '=>' | Select-Object -ExpandProperty InputObject if ($illegalKeys) { throw "The key(s) '$($illegalKeys -join ', ')' are not supported in HyperVProperties. Valid keys are '$($hypervPropertiesValidKeys -join ', ')'" } if ($HyperVProperties.SwitchType -eq 'External' -and -not $HyperVProperties.AdapterName) { throw 'You have to provide a network adapter if you want to create an external switch' return } if ($HyperVProperties.ManagementAdapter -eq $false -and $HyperVProperties.SwitchType -ne 'External') { throw 'Disabling the Management Adapter for private or internal VM Switch is not supported, as this will result in being unable to build labs' } if ($HyperVProperties.ManagementAdapter -eq $false -and $ManagementAdapter) { throw "A Management Adapter has been specified, however the Management Adapter for '$($Name)' has been disabled. Either re-enable the Management Adapter, or remove the -ManagementAdapter parameter" } if (-not $HyperVProperties.SwitchType) { $HyperVProperties.Add('SwitchType', 'Internal') } } if ($script:lab.VirtualNetworks | Where-Object Name -eq $Name) { $errorMessage = "A network with the name '$Name' is already defined" Write-Error $errorMessage Write-LogFunctionExitWithError -Message $errorMessage return } $network = New-Object -TypeName AutomatedLab.VirtualNetwork $network.AddressSpace = $AddressSpace $network.Name = $Name if ($ResourceName) {$network.FriendlyName = $ResourceName} if ($HyperVProperties.SwitchType) { $network.SwitchType = $HyperVProperties.SwitchType } if ($HyperVProperties.AdapterName) {$network.AdapterName = $HyperVProperties.AdapterName } if ($HyperVProperties.ManagementAdapter -eq $false) {$network.EnableManagementAdapter = $false } if ($ManagementAdapter) {$network.ManagementAdapter = $ManagementAdapter} #VLAN's are not supported on non-external interfaces if ($network.SwitchType -ne 'External' -and $network.ManagementAdapter.AccessVLANID -ne 0) { throw "A Management Adapter for Internal switch '$($network.Name)' has been specified with the -AccessVlanID parameter. This configuration is unsupported." } $network.HostType = $VirtualizationEngine if($AzureProperties.LocationName) { $network.LocationName = $AzureProperties.LocationName } if($AzureProperties.ConnectToVnets) { $network.ConnectToVnets = $AzureProperties.ConnectToVnets } if($AzureProperties.DnsServers) { $network.DnsServers = $AzureProperties.DnsServers } if($AzureProperties.Subnets) { foreach($subnet in $AzureProperties.Subnets.GetEnumerator()) { $temp = New-Object -TypeName AutomatedLab.AzureSubnet $temp.Name = $subnet.Key $temp.AddressSpace = $subnet.Value $network.Subnets.Add($temp) } } if ($AzureProperties.DnsLabel) { $network.AzureDnsLabel = $AzureProperties.DnsLabel } if (-not $network.LocationName) { $network.LocationName = $script:lab.AzureSettings.DefaultLocation } $script:lab.VirtualNetworks.Add($network) Write-PSFMessage "Network '$Name' added. Lab has $($Script:lab.VirtualNetworks.Count) network(s) defined" if ($PassThru) { $network } Write-LogFunctionExit } function Get-LabVirtualNetworkDefinition { [CmdletBinding()] [OutputType([AutomatedLab.VirtualNetwork])] param( [Parameter(ParameterSetName = 'ByName')] [string]$Name, [Parameter(Mandatory, ParameterSetName = 'ByAddressSpace')] [string]$AddressSpace ) $script:lab = Get-LabDefinition -ErrorAction SilentlyContinue Write-LogFunctionEntry if ($PSCmdlet.ParameterSetName -eq 'ByAddressSpace') { return $script:lab.VirtualNetworks | Where-Object AddressSpace -eq $AddressSpace } else { if ($Name) { return $script:lab.VirtualNetworks | Where-Object Name -eq $Name } else { return $script:lab.VirtualNetworks } } Write-LogFunctionExit } function New-LabNetworkAdapterDefinition { [CmdletBinding(DefaultParameterSetName = 'manual')] param ( [Parameter(Mandatory)] [string]$VirtualSwitch, [string]$InterfaceName, [Parameter(ParameterSetName = 'dhcp')] [switch]$UseDhcp, [Parameter(ParameterSetName = 'manual')] [ValidatePattern('^(([2]([0-4][0-9]|[5][0-5])|[0-1]?[0-9]?[0-9])[.]){3}(([2]([0-4][0-9]|[5][0-5])|[0-1]?[0-9]?[0-9]))/([3][0-2]|[1-2][0-9]|[2-9])$')] [AutomatedLab.IPNetwork[]]$Ipv4Address, [Parameter(ParameterSetName = 'manual')] [ValidatePattern('^(([2]([0-4][0-9]|[5][0-5])|[0-1]?[0-9]?[0-9])[.]){3}(([2]([0-4][0-9]|[5][0-5])|[0-1]?[0-9]?[0-9]))$')] [AutomatedLab.IPAddress]$Ipv4Gateway, [ValidatePattern('^(([2]([0-4][0-9]|[5][0-5])|[0-1]?[0-9]?[0-9])[.]){3}(([2]([0-4][0-9]|[5][0-5])|[0-1]?[0-9]?[0-9]))$')] [AutomatedLab.IPAddress[]]$Ipv4DNSServers, [Parameter(ParameterSetName = 'manual')] [AutomatedLab.IPNetwork[]]$IPv6Address, [Parameter(ParameterSetName = 'manual')] [ValidateRange(1, 128)] [int]$IPv6AddressPrefix, [Parameter(ParameterSetName = 'manual')] [string]$IPv6Gateway, [string[]]$IPv6DNSServers, [string]$ConnectionSpecificDNSSuffix, [boolean]$AppendParentSuffixes, [string[]]$AppendDNSSuffixes, [boolean]$RegisterInDNS = $true, [boolean]$DnsSuffixInDnsRegistration, [ValidateSet('Default', 'Enabled', 'Disabled')] [string]$NetBIOSOptions = 'Default', [ValidateRange(0,4096)] [int]$AccessVLANID = 0, [boolean]$ManagementAdapter = $false, [string] $MacAddress, [bool] $Default ) Write-LogFunctionEntry if (-not (Get-LabDefinition)) { throw 'No lab defined. Please call New-LabDefinition first before calling Set-LabDefaultOperatingSystem.' } $adapter = New-Object -TypeName AutomatedLab.NetworkAdapter $adapter.Default = $Default $MacAddress = $MacAddress -replace '[\.\-\:]' #If the defined interface is flagged as being a Management interface, ignore the virtual switch check as it will not exist yet if (-not $ManagementAdapter) { if ($VirtualSwitch) { $adapter.VirtualSwitch = Get-LabVirtualNetworkDefinition | Where-Object Name -eq $VirtualSwitch } else { $adapter.VirtualSwitch = Get-LabVirtualNetworkDefinition | Select-Object -First 1 } if (-not $adapter.VirtualSwitch) { throw "Could not find the virtual switch '$VirtualSwitch' nor create one automatically" } #VLAN Tagging is only currently supported on External switch interfaces. If a VLAN has been provied for an internal switch, throw an error if ($adapter.VirtualSwitch.SwitchType -ne 'External' -and $AccessVLANID -ne 0) { throw "VLAN tagging of interface '$InterfaceName' on non-external virtual switch '$VirtualSwitch' is not supported, either remove the AccessVlanID setting, or assign the interface to an external switch" } } if ($InterfaceName) { $adapter.InterfaceName = $InterfaceName } foreach ($item in $Ipv4Address) { $adapter.Ipv4Address.Add($item) } foreach ($item in $Ipv4DnsServers) { $adapter.Ipv4DnsServers.Add($item) } foreach ($item in $Ipv6Address) { $adapter.Ipv6Address.Add($item) } foreach ($item in $Ipv6DnsServers) { $adapter.Ipv6DnsServers.Add($item) } if ((Get-LabDefinition).DefaultVirtualizationEngine -eq 'HyperV' -and -not $MacAddress) { $macAddressPrefix = Get-LabConfigurationItem -Name MacAddressPrefix [string[]]$macAddressesInUse = (Get-LWHyperVVM | Get-VMNetworkAdapter).MacAddress $macAddressesInUse += (Get-LabMachineDefinition -All).NetworkAdapters.MacAddress if (-not $script:macIdx) { $script:macIdx = 0 } $prefixlength = 12 - $macAddressPrefix.Length while ("$macAddressPrefix{0:X$prefixLength}" -f $macIdx -in $macAddressesInUse) { $script:macIdx++ } $MacAddress = "$macAddressPrefix{0:X$prefixLength}" -f $script:macIdx++ } if ($Ipv4Gateway) { $adapter.Ipv4Gateway = $Ipv4Gateway } if ($Ipv6Gateway) { $adapter.Ipv6Gateway = $Ipv6Gateway } if ($MacAddress) { $adapter.MacAddress = $MacAddress} $adapter.ConnectionSpecificDNSSuffix = $ConnectionSpecificDNSSuffix $adapter.AppendParentSuffixes = $AppendParentSuffixes $adapter.AppendDNSSuffixes = $AppendDNSSuffixes $adapter.RegisterInDNS = $RegisterInDNS $adapter.DnsSuffixInDnsRegistration = $DnsSuffixInDnsRegistration $adapter.NetBIOSOptions = $NetBIOSOptions $adapter.UseDhcp = $UseDhcp $adapter.AccessVLANID = $AccessVLANID $adapter Write-LogFunctionExit } function Remove-LabVirtualNetworkDefinition { [CmdletBinding()] param ( [Parameter(Mandatory = $true, ValueFromPipelineByPropertyName = $true)] [string[]]$Name ) Write-LogFunctionEntry foreach ($n in $Name) { $network = $script:lab.VirtualNetworks | Where-Object Name -eq $n if (-not $network) { Write-ScreenInfo "There is no network defined with the name '$n'" -Type Warning } else { [Void]$script:lab.VirtualNetworks.Remove($network) Write-PSFMessage "Network '$n' removed. Lab has $($Script:lab.VirtualNetworks.Count) network(s) defined" } } Write-LogFunctionExit } $unattendedXmlDefaultContent2012 = @' <?xml version="1.0" encoding="utf-8"?> <unattend xmlns="urn:schemas-microsoft-com:unattend"> <settings pass="generalize"> <component name="Microsoft-Windows-Shell-Setup" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <DoNotCleanTaskBar>true</DoNotCleanTaskBar> </component> <component name="Microsoft-Windows-Security-SPP" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <SkipRearm>1</SkipRearm> </component> </settings> <settings pass="specialize"> <component name="Microsoft-Windows-UnattendedJoin" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State"> <Identification> <JoinWorkgroup >NET</JoinWorkgroup> </Identification> </component> <component name="Microsoft-Windows-Shell-Setup" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State"> <ComputerName>SERVER</ComputerName> <RegisteredOrganization>vm.net</RegisteredOrganization> <RegisteredOwner>NA</RegisteredOwner> <DoNotCleanTaskBar>true</DoNotCleanTaskBar> <TimeZone>UTC</TimeZone> </component> <component name="Microsoft-Windows-IE-InternetExplorer" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <Home_Page>about:blank</Home_Page> <DisableFirstRunWizard>true</DisableFirstRunWizard> <DisableOOBAccelerators>true</DisableOOBAccelerators> <DisableDevTools>true</DisableDevTools> <LocalIntranetSites></LocalIntranetSites> <TrustedSites></TrustedSites> </component> <component name="Microsoft-Windows-TerminalServices-RDP-WinStationExtensions" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS"> <UserAuthentication>0</UserAuthentication> </component> <component name="Microsoft-Windows-Deployment" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <RunSynchronous> <RunSynchronousCommand wcm:action="add"> <Description>EnableAdmin</Description> <Order>1</Order> <Path>cmd /c net user Administrator /active:yes</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>UnfilterAdministratorToken</Description> <Order>2</Order> <Path>cmd /c reg add HKLM\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\System /v FilterAdministratorToken /t REG_DWORD /d 0 /f</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Remove First Logon Animation</Description> <Order>3</Order> <Path>cmd /c reg add "HKLM\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Winlogon" /v EnableFirstLogonAnimation /d 0 /t REG_DWORD /f</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Do Not Open Server Manager At Logon</Description> <Order>4</Order> <Path>cmd /c reg add "HKLM\SOFTWARE\Microsoft\ServerManager" /v "DoNotOpenServerManagerAtLogon" /d 1 /t REG_DWORD /f</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Do not Open Initial Configuration Tasks At Logon</Description> <Order>5</Order> <Path>cmd /c reg add "HKLM\SOFTWARE\Microsoft\ServerManager\oobe" /v "DoNotOpenInitialConfigurationTasksAtLogon" /d 1 /t REG_DWORD /f</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Set Power Scheme to High Performance</Description> <Order>6</Order> <Path>cmd /c powercfg -setactive 8c5e7fda-e8bf-4a96-9a85-a6e23a8c635c</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Don't require password when console wakes up</Description> <Order>7</Order> <Path>cmd /c powercfg -setacvalueindex 8c5e7fda-e8bf-4a96-9a85-a6e23a8c635c fea3413e-7e05-4911-9a71-700331f1c294 0e796bdb-100d-47d6-a2d5-f7d2daa51f51 0</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Sleep timeout</Description> <Order>8</Order> <Path>cmd /c powercfg -setacvalueindex 8c5e7fda-e8bf-4a96-9a85-a6e23a8c635c 238c9fa8-0aad-41ed-83f4-97be242c8f20 29f6c1db-86da-48c5-9fdb-f2b67b1f44da 0</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>monitor timeout</Description> <Order>9</Order> <Path>cmd /c powercfg -setacvalueindex 8c5e7fda-e8bf-4a96-9a85-a6e23a8c635c 7516b95f-f776-4464-8c53-06167f40cc99 3c0bc021-c8a8-4e07-a973-6b14cbcb2b7e 0</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Enable PowerShell Remoting 1</Description> <Order>10</Order> <Path>cmd /c winrm quickconfig -quiet</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Enable PowerShell Remoting 2</Description> <Order>11</Order> <Path>cmd /c winrm quickconfig -quiet -force</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Enable PowerShell Remoting 3</Description> <Order>12</Order> <Path>cmd /c reg add HKLM\SOFTWARE\Microsoft\PowerShell\1\ShellIds\Microsoft.PowerShell /v ExecutionPolicy /t REG_SZ /d Unrestricted /f</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Disable UAC</Description> <Order>13</Order> <Path>cmd /c reg add HKLM\SOFTWARE\Microsoft\Windows\CurrentVersion\policies\system /v EnableLUA /t REG_DWORD /d 0 /f</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Configure BgInfo to start automatically</Description> <Order>14</Order> <Path>cmd /c reg add HKLM\SOFTWARE\Microsoft\Windows\CurrentVersion\Run /v BgInfo /t REG_SZ /d "C:\Windows\BgInfo.exe C:\Windows\BgInfo.bgi /Timer:0 /nolicprompt" /f</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Enable Remote Desktop firewall rules</Description> <Order>15</Order> <Path>cmd /c netsh advfirewall Firewall set rule group="Remote Desktop" new enable=yes</Path> </RunSynchronousCommand> </RunSynchronous> </component> <component name="Microsoft-Windows-International-Core" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <InputLocale>0409:00000409</InputLocale> <SystemLocale>EN-US</SystemLocale> <UILanguage>EN-US</UILanguage> <UserLocale>EN-US</UserLocale> </component> <component name="Microsoft-Windows-TapiSetup" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <TapiConfigured>0</TapiConfigured> <TapiUnattendLocation> <AreaCode>""</AreaCode> <CountryOrRegion>1</CountryOrRegion> <LongDistanceAccess>9</LongDistanceAccess> <OutsideAccess>9</OutsideAccess> <PulseOrToneDialing>1</PulseOrToneDialing> <DisableCallWaiting>""</DisableCallWaiting> <InternationalCarrierCode>""</InternationalCarrierCode> <LongDistanceCarrierCode>""</LongDistanceCarrierCode> <Name>Default</Name> </TapiUnattendLocation> </component> <component name="Microsoft-Windows-IE-ESC" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <IEHardenAdmin>false</IEHardenAdmin> <IEHardenUser>false</IEHardenUser> </component> <component name="Microsoft-Windows-TerminalServices-LocalSessionManager" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <fDenyTSConnections>false</fDenyTSConnections> </component> <component name="Networking-MPSSVC-Svc" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" /> <component name="Microsoft-Windows-TCPIP" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" /> <component name="Microsoft-Windows-DNS-Client" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" /> <component name="Microsoft-Windows-NetBT" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" /> </settings> <settings pass="oobeSystem"> <component name="Microsoft-Windows-Shell-Setup" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State"> <FirstLogonCommands> <SynchronousCommand wcm:action="add"> <CommandLine>winrm quickconfig -quiet</CommandLine> <Description>Enable Windows Remoting</Description> <Order>1</Order> </SynchronousCommand> <SynchronousCommand wcm:action="add"> <CommandLine>winrm quickconfig -quiet -force</CommandLine> <Description>Enable Windows Remoting</Description> <Order>2</Order> </SynchronousCommand> <SynchronousCommand wcm:action="add"> <CommandLine>winrm set winrm/config/service/auth @{CredSSP="true"}</CommandLine> <Description>Enable Windows Remoting CredSSP</Description> <Order>3</Order> </SynchronousCommand> <SynchronousCommand wcm:action="add"> <Description>Bring all additional disks online</Description> <Order>4</Order> <CommandLine>PowerShell -File C:\AdditionalDisksOnline.ps1</CommandLine> </SynchronousCommand> <SynchronousCommand wcm:action="add"> <Description>Disable .net Optimization</Description> <Order>5</Order> <CommandLine>PowerShell -Command "schtasks.exe /query /FO CSV | ConvertFrom-Csv | Where-Object { $_.TaskName -like '*NGEN*' } | ForEach-Object { schtasks.exe /Change /TN $_.TaskName /Disable }"</CommandLine> </SynchronousCommand> <SynchronousCommand wcm:action="add"> <Description>Configure WinRM settings</Description> <Order>6</Order> <CommandLine>PowerShell -File C:\WinRmCustomization.ps1</CommandLine> </SynchronousCommand> </FirstLogonCommands> <UserAccounts> <AdministratorPassword> <Value>Password1</Value> <PlainText>true</PlainText> </AdministratorPassword> <LocalAccounts> <LocalAccount wcm:action="add"> <Password> <Value>Password1</Value> <PlainText>true</PlainText> </Password> <Group>Administrators</Group> <DisplayName>AL</DisplayName> <Name>AL</Name> </LocalAccount> </LocalAccounts> </UserAccounts> <OOBE> <HideEULAPage>true</HideEULAPage> <ProtectYourPC>3</ProtectYourPC> <HideOnlineAccountScreens>true</HideOnlineAccountScreens> <HideLocalAccountScreen>true</HideLocalAccountScreen> <HideWirelessSetupInOOBE>true</HideWirelessSetupInOOBE> </OOBE> <RegisteredOrganization>vm.net</RegisteredOrganization> <RegisteredOwner>NA</RegisteredOwner> </component> <component name="Microsoft-Windows-International-Core" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <InputLocale>0409:00000409</InputLocale> <SystemLocale>En-US</SystemLocale> <UILanguage>EN-US</UILanguage> <UserLocale>EN-Us</UserLocale> </component> </settings> </unattend> '@ $unattendedXmlDefaultContent2008 = @' <?xml version="1.0" encoding="utf-8"?> <unattend xmlns="urn:schemas-microsoft-com:unattend"> <settings pass="generalize"> <component name="Microsoft-Windows-Shell-Setup" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <DoNotCleanTaskBar>true</DoNotCleanTaskBar> </component> <component name="Microsoft-Windows-Security-SPP" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <SkipRearm>1</SkipRearm> </component> </settings> <settings pass="specialize"> <component name="Microsoft-Windows-UnattendedJoin" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State"> <Identification> <JoinWorkgroup >NET</JoinWorkgroup> </Identification> </component> <component name="Microsoft-Windows-Shell-Setup" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State"> <ComputerName>SERVER</ComputerName> <RegisteredOrganization>vm.net</RegisteredOrganization> <RegisteredOwner>NA</RegisteredOwner> <DoNotCleanTaskBar>true</DoNotCleanTaskBar> <TimeZone>UTC</TimeZone> </component> <component name="Microsoft-Windows-IE-InternetExplorer" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <Home_Page>about:blank</Home_Page> <DisableFirstRunWizard>true</DisableFirstRunWizard> <DisableOOBAccelerators>true</DisableOOBAccelerators> <DisableDevTools>true</DisableDevTools> <LocalIntranetSites>http://*.vm.net;https://*.vm.net</LocalIntranetSites> <TrustedSites>https://*.vm.net</TrustedSites> </component> <component name="Microsoft-Windows-TerminalServices-RDP-WinStationExtensions" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS"> <UserAuthentication>0</UserAuthentication> </component> <component name="Microsoft-Windows-Deployment" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <RunSynchronous> <RunSynchronousCommand wcm:action="add"> <Description>Disable and stop Windows Firewall 1</Description> <Order>1</Order> <Path>cmd /c sc config MpsSvc start=disabled</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Disable and stop Windows Firewall 2</Description> <Order>2</Order> <Path>cmd /c sc stop MpsSvc</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>EnableAdmin</Description> <Order>3</Order> <Path>cmd /c net user Administrator /active:yes</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>UnfilterAdministratorToken</Description> <Order>4</Order> <Path>cmd /c reg add HKLM\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\System /v FilterAdministratorToken /t REG_DWORD /d 0 /f</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Remove First Logon Animation</Description> <Order>5</Order> <Path>cmd /c reg add "HKLM\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Winlogon" /v EnableFirstLogonAnimation /d 0 /t REG_DWORD /f</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Do Not Open Server Manager At Logon</Description> <Order>6</Order> <Path>cmd /c reg add "HKLM\SOFTWARE\Microsoft\ServerManager" /v "DoNotOpenServerManagerAtLogon" /d 1 /t REG_DWORD /f</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Do not Open Initial Configuration Tasks At Logon</Description> <Order>7</Order> <Path>cmd /c reg add "HKLM\SOFTWARE\Microsoft\ServerManager\oobe" /v "DoNotOpenInitialConfigurationTasksAtLogon" /d 1 /t REG_DWORD /f</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Set Power Scheme to High Performance</Description> <Order>8</Order> <Path>cmd /c powercfg -setactive 8c5e7fda-e8bf-4a96-9a85-a6e23a8c635c</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Don't require password when console wakes up</Description> <Order>9</Order> <Path>cmd /c powercfg -setacvalueindex 8c5e7fda-e8bf-4a96-9a85-a6e23a8c635c fea3413e-7e05-4911-9a71-700331f1c294 0e796bdb-100d-47d6-a2d5-f7d2daa51f51 0</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Sleep timeout</Description> <Order>10</Order> <Path>cmd /c powercfg -setacvalueindex 8c5e7fda-e8bf-4a96-9a85-a6e23a8c635c 238c9fa8-0aad-41ed-83f4-97be242c8f20 29f6c1db-86da-48c5-9fdb-f2b67b1f44da 0</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>monitor timeout</Description> <Order>11</Order> <Path>cmd /c powercfg -setacvalueindex 8c5e7fda-e8bf-4a96-9a85-a6e23a8c635c 7516b95f-f776-4464-8c53-06167f40cc99 3c0bc021-c8a8-4e07-a973-6b14cbcb2b7e 0</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Enable PowerShell Remoting 1</Description> <Order>12</Order> <Path>cmd /c winrm quickconfig -quiet</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Enable PowerShell Remoting 2</Description> <Order>13</Order> <Path>cmd /c winrm quickconfig -quiet -force</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Enable PowerShell Remoting 2</Description> <Order>14</Order> <Path>cmd /c reg add HKLM\SOFTWARE\Microsoft\PowerShell\1\ShellIds\Microsoft.PowerShell /v ExecutionPolicy /t REG_SZ /d Unrestricted /f</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Disable UAC</Description> <Order>15</Order> <Path>cmd /c reg add HKLM\SOFTWARE\Microsoft\Windows\CurrentVersion\policies\system /v EnableLUA /t REG_DWORD /d 0 /f</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Configure BgInfo to start automatically</Description> <Order>16</Order> <Path>cmd /c reg add HKLM\SOFTWARE\Microsoft\Windows\CurrentVersion\Run /v BgInfo /t REG_SZ /d "C:\Windows\BgInfo.exe C:\Windows\BgInfo.bgi /Timer:0 /nolicprompt" /f</Path> </RunSynchronousCommand> <RunSynchronousCommand wcm:action="add"> <Description>Enable Remote Desktop firewall rules</Description> <Order>17</Order> <Path>cmd /c netsh advfirewall Firewall set rule group="Remote Desktop" new enable=yes</Path> </RunSynchronousCommand> </RunSynchronous> </component> <component name="Microsoft-Windows-International-Core" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <InputLocale>0409:00000409</InputLocale> <SystemLocale>EN-US</SystemLocale> <UILanguage>EN-US</UILanguage> <UserLocale>EN-US</UserLocale> </component> <component name="Microsoft-Windows-TapiSetup" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <TapiConfigured>0</TapiConfigured> <TapiUnattendLocation> <AreaCode>""</AreaCode> <CountryOrRegion>1</CountryOrRegion> <LongDistanceAccess>9</LongDistanceAccess> <OutsideAccess>9</OutsideAccess> <PulseOrToneDialing>1</PulseOrToneDialing> <DisableCallWaiting>""</DisableCallWaiting> <InternationalCarrierCode>""</InternationalCarrierCode> <LongDistanceCarrierCode>""</LongDistanceCarrierCode> <Name>Default</Name> </TapiUnattendLocation> </component> <component name="Microsoft-Windows-IE-ESC" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <IEHardenAdmin>false</IEHardenAdmin> <IEHardenUser>false</IEHardenUser> </component> <component name="Microsoft-Windows-TerminalServices-LocalSessionManager" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <fDenyTSConnections>false</fDenyTSConnections> </component> <component name="Networking-MPSSVC-Svc" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" /> <component name="Microsoft-Windows-TCPIP" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" /> <component name="Microsoft-Windows-DNS-Client" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" /> <component name="Microsoft-Windows-NetBT" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" /> </settings> <settings pass="oobeSystem"> <component name="Microsoft-Windows-Shell-Setup" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State"> <FirstLogonCommands> <SynchronousCommand wcm:action="add"> <CommandLine>cmd /c sc config MpsSvc start=disabled</CommandLine> <Description>1</Description> <Order>1</Order> </SynchronousCommand> <SynchronousCommand wcm:action="add"> <CommandLine>cmd /c sc stop MpsSvc</CommandLine> <Description>2</Description> <Order>2</Order> </SynchronousCommand> <SynchronousCommand wcm:action="add"> <CommandLine>winrm quickconfig -quiet</CommandLine> <Description>Enable Windows Remoting</Description> <Order>3</Order> </SynchronousCommand> <SynchronousCommand wcm:action="add"> <CommandLine>winrm quickconfig -quiet -force</CommandLine> <Description>Enable Windows Remoting</Description> <Order>4</Order> </SynchronousCommand> <SynchronousCommand wcm:action="add"> <CommandLine>winrm set winrm/config/service/auth @{CredSSP="true"}</CommandLine> <Description>Enable Windows Remoting CredSSP</Description> <Order>5</Order> </SynchronousCommand> <SynchronousCommand wcm:action="add"> <Description>Bring all additional disks online</Description> <Order>6</Order> <CommandLine>PowerShell -File C:\AdditionalDisksOnline.ps1</CommandLine> </SynchronousCommand> <SynchronousCommand wcm:action="add"> <Description>Disable .net Optimization</Description> <Order>7</Order> <CommandLine>PowerShell -Command "schtasks.exe /query /FO CSV | ConvertFrom-Csv | Where-Object { $_.TaskName -like '*NGEN*' } | ForEach-Object { schtasks.exe /Change /TN $_.TaskName /Disable }"</CommandLine> </SynchronousCommand> <SynchronousCommand wcm:action="add"> <Description>Configure WinRM settings</Description> <Order>8</Order> <CommandLine>PowerShell -File C:\WinRmCustomization.ps1</CommandLine> </SynchronousCommand> </FirstLogonCommands> <UserAccounts> <AdministratorPassword> <Value>Password1</Value> <PlainText>true</PlainText> </AdministratorPassword> <LocalAccounts> <LocalAccount wcm:action="add"> <Password> <Value>Password1</Value> <PlainText>true</PlainText> </Password> <Group>Administrators</Group> <DisplayName>AL</DisplayName> <Name>AL</Name> </LocalAccount> </LocalAccounts> </UserAccounts> <OOBE> <HideEULAPage>true</HideEULAPage> <NetworkLocation>Work</NetworkLocation> <ProtectYourPC>3</ProtectYourPC> </OOBE> <RegisteredOrganization>vm.net</RegisteredOrganization> <RegisteredOwner>NA</RegisteredOwner> </component> <component name="Microsoft-Windows-International-Core" processorArchitecture="amd64" publicKeyToken="31bf3856ad364e35" language="neutral" versionScope="nonSxS" xmlns:wcm="http://schemas.microsoft.com/WMIConfig/2002/State" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <InputLocale>0409:00000409</InputLocale> <SystemLocale>En-US</SystemLocale> <UILanguage>EN-US</UILanguage> <UserLocale>EN-Us</UserLocale> </component> </settings> </unattend> '@ $kickstartContent = @" install cdrom text --non-interactive firstboot --disable reboot eula --agreed bootloader --append="biosdevname=0 net.ifnames=0" zerombr clearpart --all autopart "@ $autoyastContent = @" <?xml version="1.0"?> <!DOCTYPE profile> <profile xmlns="http://www.suse.com/1.0/yast2ns" xmlns:config="http://www.suse.com/1.0/configns"> <general> <signature-handling> <accept_unsigned_file config:type="boolean">true</accept_unsigned_file> <accept_file_without_checksum config:type="boolean">true</accept_file_without_checksum> <accept_verification_failed config:type="boolean">true</accept_verification_failed> <accept_unknown_gpg_key config:type="boolean">true</accept_unknown_gpg_key> <import_gpg_key config:type="boolean">true</import_gpg_key> <accept_non_trusted_gpg_key config:type="boolean">true</accept_non_trusted_gpg_key> </signature-handling> <self_update config:type="boolean">false</self_update> <mode> <halt config:type="boolean">false</halt> <forceboot config:type="boolean">false</forceboot> <final_reboot config:type="boolean">true</final_reboot> <final_halt config:type="boolean">false</final_halt> <confirm_base_product_license config:type="boolean">false</confirm_base_product_license> <confirm config:type="boolean">false</confirm> <second_stage config:type="boolean">true</second_stage> </mode> </general> <partitioning config:type="list"> <drive> <disklabel>gpt</disklabel> <device>/dev/sda</device> <use>free</use> <partitions config:type="list"> <partition> <filesystem config:type="symbol">vfat</filesystem> <mount>/boot</mount> <size>1G</size> </partition> <partition> <filesystem config:type="symbol">vfat</filesystem> <mount>/boot/efi</mount> <size>1G</size> </partition> <partition> <filesystem config:type="symbol">swap</filesystem> <mount>/swap</mount> <size>auto</size> </partition> <partition> <filesystem config:type="symbol">ext4</filesystem> <mount>/</mount> <size>auto</size> </partition> </partitions> </drive> </partitioning> <bootloader> <loader_type>grub2-efi</loader_type> <global> <activate config:type="boolean">true</activate> <boot_boot>true</boot_boot> </global> </bootloader> <language> <language>en_US</language> </language> <timezone> <!-- https://raw.githubusercontent.com/yast/yast-country/master/timezone/src/data/timezone_raw.ycp --> <hwclock>UTC</hwclock> <timezone>ETC/GMT</timezone> </timezone> <keyboard> <!-- https://raw.githubusercontent.com/yast/yast-country/master/keyboard/src/data/keyboard_raw.ycp --> <keymap>english-us</keymap> </keyboard> <software> <patterns config:type="list"> <pattern>base</pattern> <pattern>enhanced_base</pattern> </patterns> <install_recommended config:type="boolean">true</install_recommended> <packages config:type="list"> <package>iputils</package> <package>vim</package> <package>less</package> </packages> </software> <services-manager> <default_target>multi-user</default_target> <services> <enable config:type="list"> <service>sshd</service> </enable> </services> </services-manager> <networking> <interfaces config:type="list"> </interfaces> <net-udev config:type="list"> </net-udev> <dns> <nameservers config:type="list"> </nameservers> </dns> <routing> <routes config:type="list"> </routes> </routing> </networking> <users config:type="list"> <user> <username>root</username> <user_password>Password1</user_password> <encrypted config:type="boolean">false</encrypted> </user> </users> <firewall> <enable_firewall config:type="boolean">true</enable_firewall> <start_firewall config:type="boolean">true</start_firewall> </firewall> <scripts> <init-scripts config:type="list"> <script> <source> <![CDATA[ rpm --import https://packages.microsoft.com/keys/microsoft.asc rpm -Uvh https://packages.microsoft.com/config/sles/12/packages-microsoft-prod.rpm zypper update zypper -f -v install powershell omi openssl systemctl enable omid echo "Subsystem powershell /usr/bin/pwsh -sshs -NoLogo" >> /etc/ssh/sshd_config systemctl restart sshd ]]> </source> </script> </init-scripts> </scripts> </profile> "@ $cloudInitContent = @' version: v1 network: network: version: 2 storage: layout: name: lvm apt: primary: - arches: [amd64] uri: http://us.archive.ubuntu.com/ubuntu security: - arches: [amd64] uri: http://us.archive.ubuntu.com/ubuntu sources_list: | deb [arch=amd64] $PRIMARY $RELEASE main universe restricted multiverse deb [arch=amd64] $PRIMARY $RELEASE-updates main universe restricted multiverse deb [arch=amd64] $SECURITY $RELEASE-security main universe restricted multiverse deb [arch=amd64] $PRIMARY $RELEASE-backports main universe restricted multiverse sources: microsoft-powershell.list: source: 'deb [arch=amd64,armhf,arm64 signed-by=BC528686B50D79E339D3721CEB3E94ADBE1229CF] https://packages.microsoft.com/ubuntu/REPLACERELEASE/prod $RELEASE main' keyid: BC528686B50D79E339D3721CEB3E94ADBE1229CF # https://packages.microsoft.com/keys/microsoft.asc packages: - oddjob - oddjob-mkhomedir - sssd - adcli - krb5-workstation - realmd - samba-common - samba-common-tools - authselect-compat - sshd - powershell identity: username: {} hostname: {} password: {} late-commands: - 'echo "Subsystem powershell /usr/bin/pwsh -sshs -NoLogo" >> /etc/ssh/sshd_config' '@ Import-Module AutomatedLabCore try { $null = [AutomatedLab.Machine] } catch { $moduleroot = (Get-Module -List AutomatedLabCore)[0].ModuleBAse if ($PSEdition -eq 'Core') { Add-Type -Path $moduleroot\lib\core\AutomatedLab.dll } else { Add-Type -Path $moduleroot\lib\full\AutomatedLab.dll } } if (-not (Test-Path "alias:Get-LabPostInstallationActivity")) { New-Alias -Name Get-LabPostInstallationActivity -Value Get-LabInstallationActivity -Description "Alias so that scripts keep working" } if (-not (Test-Path "alias:Get-LabPreInstallationActivity")) { New-Alias -Name Get-LabPreInstallationActivity -Value Get-LabInstallationActivity -Description "Alias so that scripts keep working" } |