Lib/DiskImage.ps1
function GetDiskImageDriveLetter { <# .SYNOPSIS Return a disk image's associated/mounted drive letter. #> [CmdletBinding()] param ( [Parameter(Mandatory, ValueFromPipeline)] [System.Object] $DiskImage, [Parameter(Mandatory)] [ValidateSet('Basic','System','IFS')] [System.String] $PartitionType ) process { # Microsoft.Vhd.PowerShell.VirtualHardDisk $driveLetter = Get-Partition -DiskNumber $DiskImage.DiskNumber | Where-Object Type -eq $PartitionType | Where-Object DriveLetter | Select-Object -Last 1 -ExpandProperty DriveLetter; if (-not $driveLetter) { throw ($localized.CannotLocateDiskImageLetter -f $DiskImage.Path); } return $driveLetter; } } #end function GetDiskImageDriveLetter function NewDiskImageMbr { <# .SYNOPSIS Create a new MBR-formatted disk image. #> [CmdletBinding()] param ( ## Mounted VHD(X) Operating System disk image [Parameter(Mandatory)] [ValidateNotNull()] [System.Object] $Vhd # Microsoft.Vhd.PowerShell.VirtualHardDisk ) process { ## Temporarily disable Windows Explorer popup disk initialization and format notifications ## http://blogs.technet.com/b/heyscriptingguy/archive/2013/05/29/use-powershell-to-initialize-raw-disks-and-partition-and-format-volumes.aspx Stop-ShellHWDetectionService; WriteVerbose ($localized.CreatingDiskPartition -f 'Windows'); $osPartition = New-Partition -DiskNumber $Vhd.DiskNumber -UseMaximumSize -MbrType IFS -IsActive | Add-PartitionAccessPath -AssignDriveLetter -PassThru | Get-Partition; WriteVerbose ($localized.FormattingDiskPartition -f 'Windows'); $osVolume = Format-Volume -Partition $osPartition -FileSystem NTFS -Force -Confirm:$false; Start-ShellHWDetectionService; } #end proces } #end function NewDiskImageMbr function NewDiskPartFat32Partition { <# .SYNOPSIS Uses DISKPART.EXE to create a new Fat32 system partition. This permits mocking of DISKPART calls. #> [CmdletBinding()] param ( [Parameter(Mandatory)] [System.Int32] $DiskNumber, [Parameter(Mandatory)] [System.Int32] $PartitionNumber ) process { @" select disk $DiskNumber select partition $PartitionNumber format fs=fat32 label="System" "@ | & "$env:SystemRoot\System32\DiskPart.exe" | Out-Null; } } function NewDiskImageGpt { <# .SYNOPSIS Create a new GPT-formatted disk image. #> [CmdletBinding()] param ( ## Mounted VHD(X) Operating System disk image [Parameter(Mandatory)] [ValidateNotNull()] [System.Object] $Vhd # Microsoft.Vhd.PowerShell.VirtualHardDisk ) process { ## Temporarily disable Windows Explorer popup disk initialization and format notifications ## http://blogs.technet.com/b/heyscriptingguy/archive/2013/05/29/use-powershell-to-initialize-raw-disks-and-partition-and-format-volumes.aspx Stop-ShellHWDetectionService; WriteVerbose ($localized.CreatingDiskPartition -f 'EFI'); $efiPartition = New-Partition -DiskNumber $Vhd.DiskNumber -Size 250MB -GptType '{c12a7328-f81f-11d2-ba4b-00a0c93ec93b}' -AssignDriveLetter; WriteVerbose ($localized.FormattingDiskPartition -f 'EFI'); NewDiskPartFat32Partition -DiskNumber $Vhd.DiskNumber -PartitionNumber $efiPartition.PartitionNumber; WriteVerbose ($localized.CreatingDiskPartition -f 'MSR'); [ref] $null = New-Partition -DiskNumber $Vhd.DiskNumber -Size 128MB -GptType '{e3c9e316-0b5c-4db8-817d-f92df00215ae}'; WriteVerbose ($localized.CreatingDiskPartition -f 'Windows'); $osPartition = New-Partition -DiskNumber $Vhd.DiskNumber -UseMaximumSize -GptType '{ebd0a0a2-b9e5-4433-87c0-68b6b72699c7}' -AssignDriveLetter; WriteVerbose ($localized.FormattingDiskPartition -f 'Windows'); $osVolume = Format-Volume -Partition $osPartition -FileSystem NTFS -Force -Confirm:$false; Start-ShellHWDetectionService; } #end process } #end function NewDiskImageGpt function NewDiskImage { <# .SYNOPSIS Create a new formatted disk image. #> [CmdletBinding()] param ( ## VHD/x file path [Parameter(Mandatory)] [ValidateNotNullOrEmpty()] [System.String] $Path, ## Disk image partition scheme [Parameter(Mandatory)] [ValidateSet('MBR','GPT')] [System.String] $PartitionStyle, ## Disk image size in bytes [Parameter()] [System.UInt64] $Size = 127GB, ## Overwrite/recreate existing disk image [Parameter()] [System.Management.Automation.SwitchParameter] $Force, ## Do not dismount the VHD/x and return a reference [Parameter()] [System.Management.Automation.SwitchParameter] $Passthru ) begin { if ((Test-Path -Path $Path -PathType Leaf) -and (-not $Force)) { throw ($localized.ImageAlreadyExistsError -f $Path); } elseif ((Test-Path -Path $Path -PathType Leaf) -and ($Force)) { Dismount-VHD -Path $Path -ErrorAction Stop; WriteVerbose ($localized.RemovingDiskImage -f $Path); Remove-Item -Path $Path -Force -ErrorAction Stop; } } #end begin process { WriteVerbose ($localized.CreatingDiskImage -f $Path); $vhd = New-Vhd -Path $Path -Dynamic -SizeBytes $Size; WriteVerbose ($localized.MountingDiskImage -f $Path); $vhdMount = Mount-VHD -Path $Path -Passthru; WriteVerbose ($localized.InitializingDiskImage -f $Path); [ref] $null = Initialize-Disk -Number $vhdMount.DiskNumber -PartitionStyle $PartitionStyle -PassThru; switch ($PartitionStyle) { 'MBR' { NewDiskImageMbr -Vhd $vhdMount; } 'GPT' { NewDiskImageGpt -Vhd $vhdMount; } } if ($Passthru) { return $vhdMount; } else { Dismount-VHD -Path $Path; } } #end process } #end function NewDiskImage function SetDiskImageBootVolumeMbr { <# .SYNOPSIS Configure/repair MBR boot volume #> [CmdletBinding()] param ( ## Mounted VHD(X) Operating System disk image [Parameter(Mandatory)] [ValidateNotNull()] [System.Object] $Vhd # Microsoft.Vhd.PowerShell.VirtualHardDisk ) process { $bcdBootExe = 'bcdboot.exe'; $bcdEditExe = 'bcdedit.exe'; $imageName = [System.IO.Path]::GetFileNameWithoutExtension($Vhd.Path); $osPartitionDriveLetter = GetDiskImageDriveLetter -DiskImage $Vhd -PartitionType 'IFS'; WriteVerbose ($localized.RepairingBootVolume -f $osPartitionDriveLetter); $bcdBootArgs = @( ('{0}:\Windows' -f $osPartitionDriveLetter), # Path to source Windows boot files ('/s {0}:\' -f $osPartitionDriveLetter), # Volume to create the \BOOT folder on. '/v' # Enable verbose logging. '/f BIOS' # Firmware type of the target system partition ) InvokeExecutable -Path $bcdBootExe -Arguments $bcdBootArgs -LogName ('{0}-BootEdit.log' -f $imageName); $bootmgrDeviceArgs = @( ('/store {0}:\boot\bcd' -f $osPartitionDriveLetter), '/set {bootmgr} device locate' ); InvokeExecutable -Path $bcdEditExe -Arguments $bootmgrDeviceArgs -LogName ('{0}-BootmgrDevice.log' -f $imageName); $defaultDeviceArgs = @( ('/store {0}:\boot\bcd' -f $osPartitionDriveLetter), '/set {default} device locate' ); InvokeExecutable -Path $bcdEditExe -Arguments $defaultDeviceArgs -LogName ('{0}-DefaultDevice.log' -f $imageName); $defaultOsDeviceArgs = @( ('/store {0}:\boot\bcd' -f $osPartitionDriveLetter), '/set {default} osdevice locate' ); InvokeExecutable -Path $bcdEditExe -Arguments $defaultOsDeviceArgs -LogName ('{0}-DefaultOsDevice.log' -f $imageName); } #end process } #end function function SetDiskImageBootVolumeGpt { <# .SYNOPSIS Configure/repair MBR boot volume #> [CmdletBinding()] param ( ## Mounted VHD(X) Operating System disk image [Parameter(Mandatory)] [ValidateNotNull()] [System.Object] $Vhd # Microsoft.Vhd.PowerShell.VirtualHardDisk ) process { $bcdBootExe = 'bcdboot.exe'; $imageName = [System.IO.Path]::GetFileNameWithoutExtension($Vhd.Path); $systemPartitionDriveLetter = GetDiskImageDriveLetter -DiskImage $Vhd -PartitionType 'System'; $osPartitionDriveLetter = GetDiskImageDriveLetter -DiskImage $Vhd -PartitionType 'Basic'; WriteVerbose ($localized.RepairingBootVolume -f $osPartitionDriveLetter); $bcdBootArgs = @( ('{0}:\Windows' -f $osPartitionDriveLetter), # Path to source Windows boot files ('/s {0}:\' -f $systemPartitionDriveLetter), # Specifies the volume letter of the drive to create the \BOOT folder on. '/v' # Enabled verbose logging. '/f UEFI' # Specifies the firmware type of the target system partition ) InvokeExecutable -Path $bcdBootExe -Arguments $bcdBootArgs -LogName ('{0}-BootEdit.log' -f $imageName); ## Clean up and remove drive access path Remove-PSDrive -Name $osPartitionDriveLetter -PSProvider FileSystem -ErrorAction Ignore; [ref] $null = Get-PSDrive; } #end process } #end function function SetDiskImageBootVolume { <# .SYNOPSIS Sets the boot volume of a mounted disk image. #> [CmdletBinding()] param ( ## Mounted VHD(X) Operating System disk image [Parameter(Mandatory)] [ValidateNotNull()] [System.Object] $Vhd, # Microsoft.Vhd.PowerShell.VirtualHardDisk ## Disk image partition scheme [Parameter(Mandatory)] [ValidateSet('MBR','GPT')] [System.String] $PartitionStyle ) process { switch ($PartitionStyle) { 'MBR' { SetDiskImageBootVolumeMbr -Vhd $Vhd; break; } 'GPT' { SetDiskImageBootVolumeGpt -Vhd $Vhd; break; } } #end switch } #end process } #end function SetDiskImageBootVolume function AddDiskImageHotfix { <# .SYMOPSIS Adds a Windows update/hotfix package to an image. #> [CmdletBinding()] param ( [Parameter(Mandatory, ValueFromPipeline)] [ValidateNotNullOrEmpty()] [System.String] $Id, ## Mounted VHD(X) Operating System disk image [Parameter(Mandatory)] [ValidateNotNull()] [System.Object] $Vhd, # Microsoft.Vhd.PowerShell.VirtualHardDisk ## Disk image partition scheme [Parameter(Mandatory)] [ValidateSet('MBR','GPT')] [System.String] $PartitionStyle, ## Lab DSC configuration data [Parameter(ValueFromPipelineByPropertyName)] [System.Collections.Hashtable] [Microsoft.PowerShell.DesiredStateConfiguration.ArgumentToConfigurationDataTransformationAttribute()] $ConfigurationData ) process { if ($PartitionStyle -eq 'MBR') { $partitionType = 'IFS'; } elseif ($PartitionStyle -eq 'GPT') { $partitionType = 'Basic'; } $vhdDriveLetter = GetDiskImageDriveLetter -DiskImage $Vhd -PartitionType $partitionType; $resolveLabMediaParams = @{ Id = $Id; } if ($PSBoundParameters.ContainsKey('ConfigurationData')) { $resolveLabMediaParams['ConfigurationData'] = $ConfigurationData; } $media = ResolveLabMedia @resolveLabMediaParams; foreach ($hotfix in $media.Hotfixes) { $hotfixFileInfo = InvokeLabMediaHotfixDownload -Id $hotfix.Id -Uri $hotfix.Uri; $packageName = [System.IO.Path]::GetFileNameWithoutExtension($hotfixFileInfo.FullName); AddDiskImagePackage -Name $packageName -Path $hotfixFileInfo.FullName -DestinationPath $vhdDriveLetter; } } #end process } #end function AddDiskImageHotfix function AddDiskImagePackage { <# .SYNOPSIS Adds a Windows package (.cab) to an image. This is implmented primarily to support injection of packages into Nano server images. .NOTES The real difference between a hotfix and package is that a package can either be specified in the master VHD(X) image creation OR be injected into VHD(X) differencing disk. #> [CmdletBinding()] param ( ## Package name (used for logging) [Parameter(Mandatory, ValueFromPipelineByPropertyName)] [ValidateNotNullOrEmpty()] [System.String] $Name, ## File path to the package (.cab) file [Parameter(Mandatory, ValueFromPipelineByPropertyName)] [ValidateNotNullOrEmpty()] [System.String] $Path, ## Destination operating system path (mounted VHD), i.e. G:\ [Parameter(Mandatory, ValueFromPipelineByPropertyName)] [ValidateNotNullOrEmpty()] [System.String] $DestinationPath ) begin { ## We just want the drive letter if ($DestinationPath.Length -gt 1) { $DestinationPath = $DestinationPath.Substring(0,1); } } process { $logPath = '{0}:\Windows\Logs\{1}' -f $DestinationPath, $labDefaults.ModuleName; [ref] $null = NewDirectory -Path $logPath -Verbose:$false; WriteVerbose ($localized.AddingImagePackage -f $Name, $DestinationPath); $addWindowsPackageParams = @{ PackagePath = $Path; Path = '{0}:\' -f $DestinationPath; LogPath = '{0}\{1}.log' -f $logPath, $Name; LogLevel = 'Errors'; } [ref] $null = Microsoft.Dism.Powershell\Add-WindowsPackage @addWindowsPackageParams -Verbose:$false; } #end process } #end function AddDiskImagePackage # SIG # Begin signature block # MIIXtwYJKoZIhvcNAQcCoIIXqDCCF6QCAQExCzAJBgUrDgMCGgUAMGkGCisGAQQB # gjcCAQSgWzBZMDQGCisGAQQBgjcCAR4wJgIDAQAABBAfzDtgWUsITrck0sYpfvNR # AgEAAgEAAgEAAgEAAgEAMCEwCQYFKw4DAhoFAAQUGAv8L92C+QEFEkgh9APYswrr # ggqgghLqMIID7jCCA1egAwIBAgIQfpPr+3zGTlnqS5p31Ab8OzANBgkqhkiG9w0B # AQUFADCBizELMAkGA1UEBhMCWkExFTATBgNVBAgTDFdlc3Rlcm4gQ2FwZTEUMBIG # A1UEBxMLRHVyYmFudmlsbGUxDzANBgNVBAoTBlRoYXd0ZTEdMBsGA1UECxMUVGhh # d3RlIENlcnRpZmljYXRpb24xHzAdBgNVBAMTFlRoYXd0ZSBUaW1lc3RhbXBpbmcg # Q0EwHhcNMTIxMjIxMDAwMDAwWhcNMjAxMjMwMjM1OTU5WjBeMQswCQYDVQQGEwJV # UzEdMBsGA1UEChMUU3ltYW50ZWMgQ29ycG9yYXRpb24xMDAuBgNVBAMTJ1N5bWFu # dGVjIFRpbWUgU3RhbXBpbmcgU2VydmljZXMgQ0EgLSBHMjCCASIwDQYJKoZIhvcN # AQEBBQADggEPADCCAQoCggEBALGss0lUS5ccEgrYJXmRIlcqb9y4JsRDc2vCvy5Q # WvsUwnaOQwElQ7Sh4kX06Ld7w3TMIte0lAAC903tv7S3RCRrzV9FO9FEzkMScxeC # i2m0K8uZHqxyGyZNcR+xMd37UWECU6aq9UksBXhFpS+JzueZ5/6M4lc/PcaS3Er4 # ezPkeQr78HWIQZz/xQNRmarXbJ+TaYdlKYOFwmAUxMjJOxTawIHwHw103pIiq8r3 # +3R8J+b3Sht/p8OeLa6K6qbmqicWfWH3mHERvOJQoUvlXfrlDqcsn6plINPYlujI # fKVOSET/GeJEB5IL12iEgF1qeGRFzWBGflTBE3zFefHJwXECAwEAAaOB+jCB9zAd # BgNVHQ4EFgQUX5r1blzMzHSa1N197z/b7EyALt0wMgYIKwYBBQUHAQEEJjAkMCIG # CCsGAQUFBzABhhZodHRwOi8vb2NzcC50aGF3dGUuY29tMBIGA1UdEwEB/wQIMAYB # Af8CAQAwPwYDVR0fBDgwNjA0oDKgMIYuaHR0cDovL2NybC50aGF3dGUuY29tL1Ro # YXd0ZVRpbWVzdGFtcGluZ0NBLmNybDATBgNVHSUEDDAKBggrBgEFBQcDCDAOBgNV # HQ8BAf8EBAMCAQYwKAYDVR0RBCEwH6QdMBsxGTAXBgNVBAMTEFRpbWVTdGFtcC0y # MDQ4LTEwDQYJKoZIhvcNAQEFBQADgYEAAwmbj3nvf1kwqu9otfrjCR27T4IGXTdf # plKfFo3qHJIJRG71betYfDDo+WmNI3MLEm9Hqa45EfgqsZuwGsOO61mWAK3ODE2y # 0DGmCFwqevzieh1XTKhlGOl5QGIllm7HxzdqgyEIjkHq3dlXPx13SYcqFgZepjhq # IhKjURmDfrYwggSjMIIDi6ADAgECAhAOz/Q4yP6/NW4E2GqYGxpQMA0GCSqGSIb3 # DQEBBQUAMF4xCzAJBgNVBAYTAlVTMR0wGwYDVQQKExRTeW1hbnRlYyBDb3Jwb3Jh # dGlvbjEwMC4GA1UEAxMnU3ltYW50ZWMgVGltZSBTdGFtcGluZyBTZXJ2aWNlcyBD # QSAtIEcyMB4XDTEyMTAxODAwMDAwMFoXDTIwMTIyOTIzNTk1OVowYjELMAkGA1UE # BhMCVVMxHTAbBgNVBAoTFFN5bWFudGVjIENvcnBvcmF0aW9uMTQwMgYDVQQDEytT # eW1hbnRlYyBUaW1lIFN0YW1waW5nIFNlcnZpY2VzIFNpZ25lciAtIEc0MIIBIjAN # BgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAomMLOUS4uyOnREm7Dv+h8GEKU5Ow # mNutLA9KxW7/hjxTVQ8VzgQ/K/2plpbZvmF5C1vJTIZ25eBDSyKV7sIrQ8Gf2Gi0 # jkBP7oU4uRHFI/JkWPAVMm9OV6GuiKQC1yoezUvh3WPVF4kyW7BemVqonShQDhfu # ltthO0VRHc8SVguSR/yrrvZmPUescHLnkudfzRC5xINklBm9JYDh6NIipdC6Anqh # d5NbZcPuF3S8QYYq3AhMjJKMkS2ed0QfaNaodHfbDlsyi1aLM73ZY8hJnTrFxeoz # C9Lxoxv0i77Zs1eLO94Ep3oisiSuLsdwxb5OgyYI+wu9qU+ZCOEQKHKqzQIDAQAB # o4IBVzCCAVMwDAYDVR0TAQH/BAIwADAWBgNVHSUBAf8EDDAKBggrBgEFBQcDCDAO # BgNVHQ8BAf8EBAMCB4AwcwYIKwYBBQUHAQEEZzBlMCoGCCsGAQUFBzABhh5odHRw # Oi8vdHMtb2NzcC53cy5zeW1hbnRlYy5jb20wNwYIKwYBBQUHMAKGK2h0dHA6Ly90 # cy1haWEud3Muc3ltYW50ZWMuY29tL3Rzcy1jYS1nMi5jZXIwPAYDVR0fBDUwMzAx # oC+gLYYraHR0cDovL3RzLWNybC53cy5zeW1hbnRlYy5jb20vdHNzLWNhLWcyLmNy # bDAoBgNVHREEITAfpB0wGzEZMBcGA1UEAxMQVGltZVN0YW1wLTIwNDgtMjAdBgNV # HQ4EFgQURsZpow5KFB7VTNpSYxc/Xja8DeYwHwYDVR0jBBgwFoAUX5r1blzMzHSa # 1N197z/b7EyALt0wDQYJKoZIhvcNAQEFBQADggEBAHg7tJEqAEzwj2IwN3ijhCcH # bxiy3iXcoNSUA6qGTiWfmkADHN3O43nLIWgG2rYytG2/9CwmYzPkSWRtDebDZw73 # BaQ1bHyJFsbpst+y6d0gxnEPzZV03LZc3r03H0N45ni1zSgEIKOq8UvEiCmRDoDR # EfzdXHZuT14ORUZBbg2w6jiasTraCXEQ/Bx5tIB7rGn0/Zy2DBYr8X9bCT2bW+IW # yhOBbQAuOA2oKY8s4bL0WqkBrxWcLC9JG9siu8P+eJRRw4axgohd8D20UaF5Mysu # e7ncIAkTcetqGVvP6KUwVyyJST+5z3/Jvz4iaGNTmr1pdKzFHTx/kuDDvBzYBHUw # ggUZMIIEAaADAgECAhADViTO4HBjoJNSwH9//cwJMA0GCSqGSIb3DQEBCwUAMHIx # CzAJBgNVBAYTAlVTMRUwEwYDVQQKEwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3 # dy5kaWdpY2VydC5jb20xMTAvBgNVBAMTKERpZ2lDZXJ0IFNIQTIgQXNzdXJlZCBJ # RCBDb2RlIFNpZ25pbmcgQ0EwHhcNMTUwNTE5MDAwMDAwWhcNMTcwODIzMTIwMDAw # WjBgMQswCQYDVQQGEwJHQjEPMA0GA1UEBxMGT3hmb3JkMR8wHQYDVQQKExZWaXJ0 # dWFsIEVuZ2luZSBMaW1pdGVkMR8wHQYDVQQDExZWaXJ0dWFsIEVuZ2luZSBMaW1p # dGVkMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAqLQmabdimcQtYPTQ # 9RSjv3ThEmFTRJt/MzseYYtZpBTcR6BnSfj8RfkC4aGZvspFgH0cGP/SNJh1w67b # iX9oT5NFL9sUJHUsVdyPBA1LhpWcF09PP28mGGKO3oQHI4hTLD8etiIlF9qFantd # 1Pmo0jdqT4uErSmx0m4kYGUUTa5ZPAK0UZSuAiNX6iNIL+rj/BPbI3nuPJzzx438 # oHYkZGRtsx11+pLA6hIKyUzRuIDoI7JQ0nZ0MkCziVyc6xGfS54JVLaVCEteTKPz # Gc4yyvCqp6Tfe9gs8UuxJiEMdH5fvllTU4aoXbm+W8tonkE7i/19rv8S1A2VPiVV # xNLbpwIDAQABo4IBuzCCAbcwHwYDVR0jBBgwFoAUWsS5eyoKo6XqcQPAYPkt9mV1 # DlgwHQYDVR0OBBYEFP2RNOWYipdNCSRVb5jIcyRp9tUDMA4GA1UdDwEB/wQEAwIH # gDATBgNVHSUEDDAKBggrBgEFBQcDAzB3BgNVHR8EcDBuMDWgM6Axhi9odHRwOi8v # Y3JsMy5kaWdpY2VydC5jb20vc2hhMi1hc3N1cmVkLWNzLWcxLmNybDA1oDOgMYYv # aHR0cDovL2NybDQuZGlnaWNlcnQuY29tL3NoYTItYXNzdXJlZC1jcy1nMS5jcmww # QgYDVR0gBDswOTA3BglghkgBhv1sAwEwKjAoBggrBgEFBQcCARYcaHR0cHM6Ly93 # d3cuZGlnaWNlcnQuY29tL0NQUzCBhAYIKwYBBQUHAQEEeDB2MCQGCCsGAQUFBzAB # hhhodHRwOi8vb2NzcC5kaWdpY2VydC5jb20wTgYIKwYBBQUHMAKGQmh0dHA6Ly9j # YWNlcnRzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydFNIQTJBc3N1cmVkSURDb2RlU2ln # bmluZ0NBLmNydDAMBgNVHRMBAf8EAjAAMA0GCSqGSIb3DQEBCwUAA4IBAQCclXHR # DhDyJr81eiD0x+AL04ryDwdKT+PooKYgOxc7EhRn59ogxNO7jApQPSVo0I11Zfm6 # zQ6K6RPWhxDenflf2vMx7a0tIZlpHhq2F8praAMykK7THA9F3AUxIb/lWHGZCock # yD/GQvJek3LSC5NjkwQbnubWYF/XZTDzX/mJGU2DcG1OGameffR1V3xODHcUE/K3 # PWy1bzixwbQCQA96GKNCWow4/mEW31cupHHSo+XVxmjTAoC93yllE9f4Kdv6F29H # bRk0Go8Yn8WjWeLE/htxW/8ruIj0KnWkG+YwmZD+nTegYU6RvAV9HbJJYUEIfhVy # 3DeK5OlY9ima2sdtMIIFMDCCBBigAwIBAgIQBAkYG1/Vu2Z1U0O1b5VQCDANBgkq # hkiG9w0BAQsFADBlMQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5j # MRkwFwYDVQQLExB3d3cuZGlnaWNlcnQuY29tMSQwIgYDVQQDExtEaWdpQ2VydCBB # c3N1cmVkIElEIFJvb3QgQ0EwHhcNMTMxMDIyMTIwMDAwWhcNMjgxMDIyMTIwMDAw # WjByMQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5jMRkwFwYDVQQL # ExB3d3cuZGlnaWNlcnQuY29tMTEwLwYDVQQDEyhEaWdpQ2VydCBTSEEyIEFzc3Vy # ZWQgSUQgQ29kZSBTaWduaW5nIENBMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIB # CgKCAQEA+NOzHH8OEa9ndwfTCzFJGc/Q+0WZsTrbRPV/5aid2zLXcep2nQUut4/6 # kkPApfmJ1DcZ17aq8JyGpdglrA55KDp+6dFn08b7KSfH03sjlOSRI5aQd4L5oYQj # ZhJUM1B0sSgmuyRpwsJS8hRniolF1C2ho+mILCCVrhxKhwjfDPXiTWAYvqrEsq5w # MWYzcT6scKKrzn/pfMuSoeU7MRzP6vIK5Fe7SrXpdOYr/mzLfnQ5Ng2Q7+S1TqSp # 6moKq4TzrGdOtcT3jNEgJSPrCGQ+UpbB8g8S9MWOD8Gi6CxR93O8vYWxYoNzQYIH # 5DiLanMg0A9kczyen6Yzqf0Z3yWT0QIDAQABo4IBzTCCAckwEgYDVR0TAQH/BAgw # BgEB/wIBADAOBgNVHQ8BAf8EBAMCAYYwEwYDVR0lBAwwCgYIKwYBBQUHAwMweQYI # KwYBBQUHAQEEbTBrMCQGCCsGAQUFBzABhhhodHRwOi8vb2NzcC5kaWdpY2VydC5j # b20wQwYIKwYBBQUHMAKGN2h0dHA6Ly9jYWNlcnRzLmRpZ2ljZXJ0LmNvbS9EaWdp # Q2VydEFzc3VyZWRJRFJvb3RDQS5jcnQwgYEGA1UdHwR6MHgwOqA4oDaGNGh0dHA6 # Ly9jcmw0LmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydEFzc3VyZWRJRFJvb3RDQS5jcmww # OqA4oDaGNGh0dHA6Ly9jcmwzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydEFzc3VyZWRJ # RFJvb3RDQS5jcmwwTwYDVR0gBEgwRjA4BgpghkgBhv1sAAIEMCowKAYIKwYBBQUH # AgEWHGh0dHBzOi8vd3d3LmRpZ2ljZXJ0LmNvbS9DUFMwCgYIYIZIAYb9bAMwHQYD # VR0OBBYEFFrEuXsqCqOl6nEDwGD5LfZldQ5YMB8GA1UdIwQYMBaAFEXroq/0ksuC # MS1Ri6enIZ3zbcgPMA0GCSqGSIb3DQEBCwUAA4IBAQA+7A1aJLPzItEVyCx8JSl2 # qB1dHC06GsTvMGHXfgtg/cM9D8Svi/3vKt8gVTew4fbRknUPUbRupY5a4l4kgU4Q # pO4/cY5jDhNLrddfRHnzNhQGivecRk5c/5CxGwcOkRX7uq+1UcKNJK4kxscnKqEp # KBo6cSgCPC6Ro8AlEeKcFEehemhor5unXCBc2XGxDI+7qPjFEmifz0DLQESlE/Dm # ZAwlCEIysjaKJAL+L3J+HNdJRZboWR3p+nRka7LrZkPas7CM1ekN3fYBIM6ZMWM9 # CBoYs4GbT8aTEAb8B4H6i9r5gkn3Ym6hU/oSlBiFLpKR6mhsRDKyZqHnGKSaZFHv # MYIENzCCBDMCAQEwgYYwcjELMAkGA1UEBhMCVVMxFTATBgNVBAoTDERpZ2lDZXJ0 # IEluYzEZMBcGA1UECxMQd3d3LmRpZ2ljZXJ0LmNvbTExMC8GA1UEAxMoRGlnaUNl # cnQgU0hBMiBBc3N1cmVkIElEIENvZGUgU2lnbmluZyBDQQIQA1YkzuBwY6CTUsB/ # f/3MCTAJBgUrDgMCGgUAoHgwGAYKKwYBBAGCNwIBDDEKMAigAoAAoQKAADAZBgkq # hkiG9w0BCQMxDAYKKwYBBAGCNwIBBDAcBgorBgEEAYI3AgELMQ4wDAYKKwYBBAGC # NwIBFTAjBgkqhkiG9w0BCQQxFgQUXhFUc8UPYOQRt2ZwTyf6u93fKqcwDQYJKoZI # hvcNAQEBBQAEggEAPPY9k89n/onxSNBYmTURx9dD9RCZqQOIIDgAG102X3kT+14P # 90v6Xwj45s1UiF0fcPn0WBjGIM0VxmjLLvQdp4eMqISXMHHhw0h9/SIEjq8xqzL4 # DrrZ81j4PnKl71Oy1/PlHy8MSe0xUH4ON/pUsu83pjbwsic4qkkCRyuN3fCnCHHd # eNpVDz6bsJs76d14pUxrOJ94aeYREnBHkRV95fprdmv3BtxC4K2HHPgApY8QjFQu # wRzLWHsmyPi8dAMXO+hfchebg51ckc6GMIwLJs5H7+0+VTls4AVh8LFhWEHAKfZv # w+FAZun/D3+DfsZ1pIq4q0LddkOnG2l7YEyXpaGCAgswggIHBgkqhkiG9w0BCQYx # ggH4MIIB9AIBATByMF4xCzAJBgNVBAYTAlVTMR0wGwYDVQQKExRTeW1hbnRlYyBD # b3Jwb3JhdGlvbjEwMC4GA1UEAxMnU3ltYW50ZWMgVGltZSBTdGFtcGluZyBTZXJ2 # aWNlcyBDQSAtIEcyAhAOz/Q4yP6/NW4E2GqYGxpQMAkGBSsOAwIaBQCgXTAYBgkq # hkiG9w0BCQMxCwYJKoZIhvcNAQcBMBwGCSqGSIb3DQEJBTEPFw0xNzAzMTIyMTAz # MzVaMCMGCSqGSIb3DQEJBDEWBBTutXu/NXlWnoDUpvhHbXPl1E3K7jANBgkqhkiG # 9w0BAQEFAASCAQBy20ZiXyVM9RwBoxKKXa1/c6j+SmydwtNV7nEbt2rLoX8OOQXT # JC6A1V1Ms2E3GcjfMeLm83xcLOq5+zdq1y9d5tLZ6Gg+A6TMBV0SSqfoZ+dv1KTZ # dCICWi/c/HT4ResqWbx9omtVc0PUXQK3TdFy86FqcMXmRlPZI5nWAaEsJAO2fsXf # 2s96vryGEXbA8WTXwEvENc/ByRw0l42mdFkTS7m4eDs+wZg50SBO6L9ebmp9JLOx # zBe09JAf+7b6+gNxG06aoQI+ER31Uw8ekK/vi5j01mhKoRwsy2jRbIOc9ldvMta3 # tSOZqUJ29qIU30nVUlgtOuzcBKO7kNcr7pQC # SIG # End signature block |