Obs/bin/ObsDep/content/Powershell/Roles/Common/MountedImage.psm1
<###################################################
# # # Copyright (c) Microsoft. All rights reserved. # # # ##################################################> Import-Module -Name "$PSScriptRoot\..\..\Common\Helpers.psm1" Import-Module -Name "$PSScriptRoot\RoleHelpers.psm1" Import-Module -Name Defender <# MountedImage Class provides a standard way of mounting and updating an image. The class supports both WIM and VHDx files and can be used to inject Windows updates, drivers and deployment artifacts. Any injection of content that is to be done on an offline image should be added to this class. #> class MountedImage : System.IDisposable { [string] $MountPath = $null [string] $ImagePath [string] $imageStatusFolder = "ImageStatus" [string] $imageStatusFileName = "ImageStatus.xml" # Constructors <# .SYNOPSIS This function mounts a WIM for offline update. .PARAMETER ImagePath Path to the WIM file to be updated. .PARAMETER ImageIndex Index for WIM file. This value should be set to 1 for most WIM files. #> MountedImage( [string] $ImagePath, [int] $ImageIndex ) { if ($ImagePath -notlike '*.wim') { throw "$ImagePath is not a WIM." } $ErrorActionPreference = [System.Management.Automation.ActionPreference]::Stop $this.Init($ImagePath) # Mount the image Trace-Execution "Mount '$ImagePath' to '$($this.MountPath)'." Mount-WindowsImageWithRetry -ImagePath $ImagePath -Path $this.mountPath -Index $ImageIndex } <# .SYNOPSIS This function mounts VHD image for offline update. .PARAMETER ImagePath Path to the VHD to be updated. #> MountedImage( [string] $ImagePath ) { if ($ImagePath -notlike '*.vhd*') { throw "$ImagePath is not a VHD." } $ErrorActionPreference = [System.Management.Automation.ActionPreference]::Stop $this.Init($ImagePath) # Mount the image Trace-Execution "Mount '$ImagePath' to '$($this.MountPath)'." Mount-WindowsImageWithRetry -ImagePath $ImagePath -Path $this.mountPath -Index 1 } # Initializes the infrastructure pieces needed to mount the image [void] Init( [string] $ImagePath ) { $ErrorActionPreference = [System.Management.Automation.ActionPreference]::Stop $this.ImagePath = $ImagePath $this.DiscardPreviousImage() $this.MountPath = Join-Path -Path "C:\Temp\" -ChildPath $(Get-Item -Path $([IO.Path]::GetTempFileName())).Name # Create the mount path if it does not exist if(-not (Test-Path $this.MountPath)) { New-Item -Path $this.MountPath -ItemType Directory } # Attempt to start defender if not already running $service = Get-Service -Name "windefend" -ErrorAction SilentlyContinue if ($service -ne $null -and $service.Status -ine "Running") { $service | Start-Service -ErrorAction SilentlyContinue | Out-Null } # set exclusion path when windefend is running if($service -ne $null -and $service.Status -eq "Running") { Trace-Execution "Exclude mount folders $($this.MountPath) from Windows Defender scan to speed up configuration." Add-MpPreference -ExclusionPath $this.MountPath -ErrorAction SilentlyContinue } } <# Gets the list of all previously mounted images and dismount them. #> [void] DiscardPreviousImage() { $mountedImages = Get-WindowsImage -Mounted foreach ($mountedImage in $mountedImages) { if ($mountedImage.ImagePath -eq $this.ImagePath) { $mountedImage | Dismount-WindowsImage -Discard } } } <# Gets the build of the mounted image from ntoskrnl.exe. #> [string] GetBuild() { $filePath = Join-Path -Path $this.MountPath -ChildPath 'Windows\System32\ntoskrnl.exe' Trace-Execution "Getting image build from '$filePath'" $fileInfo = Get-Item -Path $filePath $build = $fileInfo.VersionInfo.ProductBuildPart Trace-Execution "Image build is '$($build.ToString())'" return $build.ToString() } <# .SYNOPSIS Get version and status information from the image .DESCRIPTION If the image status file is present. The function reads contents of the file and returns it back. If the image status file is not present. It return null #> [xml] GetStatusInfoFromImage() { $content = $null $imageStatusFilePath = "$this.MountPath\$this.ImageStatusFolder\$this.ImageStatusFileName" if(Test-Path $imageStatusFilePath) { # Read the contents of the file $content = [xml](Get-Content $imageStatusFilePath) } return $content } <# .SYNOPSIS Updated the image status file with version and status. .DESCRIPTION Takes in version and the status as input. Generates a XML blob and saves it into a predefined location. .PARAMETER Version Version of deployment or update being applied to the stamp. .PARAMETER Status Status value that is to be set in the image #> UpdateImageStatusFile( [string] $Version, [string] $Status ) { $statusId = $this.GetStatusIdForStatusString($Status) Trace-Execution "Updating image status by Setting Version: $Version StatusId: $statusId Status: $Status" $imageStatusXML = "<ImageStatus Version=`"$Version`" StatusId=`"$statusId`" Status=`"$Status`" />" if( -not (Test-Path "$this.MountPath\$this.ImageStatusFolder") ) { $null = New-Item -Path $this.MountPath -Name $this.ImageStatusFolder -ItemType Directory -Force } $imageStatusFilePath = (Join-Path $this.MountPath (Join-Path $this.ImageStatusFolder $this.ImageStatusFileName)) $imageStatusXML | Out-File $imageStatusFilePath -Force } <# Returns an integer value corresponding to the status string #> [int] GetStatusIdForStatusString( [string] $Status ) { $statusToIdMap = @{} $statusToIdMap.Add("BaseBuildCompleted", 0) $statusToIdMap.Add("BuildCompleted", 1) return $statusToIdMap[$Status] } <# .SYNOPSIS Injects windows updates into the mounted image. .DESCRIPTION Looks for the windows update staging folder and updates inside it. If updates are present, it calls into Add-WindowsPackage to inject Windows Updates into the mounted image. .PARAMETER WindowsUpdateStagingFolder Folder containing all the Windows Updates to be applied. .PARAMETER ScratchDirectory Optional Scratch directory for Add-WindowsPackage #> UpdateImageWithWindowsUpdates( [string] $WindowsUpdateStagingFolder ) { $this.UpdateImageWithWindowsUpdates($WindowsUpdateStagingFolder, $null, $false) } UpdateImageWithWindowsUpdates( [string] $WindowsUpdateStagingFolder, [string] $ScratchDirectory ) { $this.UpdateImageWithWindowsUpdates($WindowsUpdateStagingFolder, $ScratchDirectory, $false) } UpdateImageWithWindowsUpdates( [string] $WindowsUpdateStagingFolder, [string] $ScratchDirectory, [bool] $IgnoreCheck ) { $ErrorActionPreference = "Stop" Trace-Execution "Performing Windows Update injection for: $($this.ImagePath). WindowsUpdateStagingFolder = '$WindowsUpdateStagingFolder', ScratchDirectory = '$ScratchDirectory'" if (Test-Path $WindowsUpdateStagingFolder) { $versionedFolder = Get-AzSVersionedPath -Path $WindowsUpdateStagingFolder -ErrorAction SilentlyContinue if ($versionedFolder -ne $null) { $WindowsUpdateStagingFolder = $versionedFolder } } Trace-Execution "Found Windows update folder: $WindowsUpdateStagingFolder" $subFolders = @("SSU", "LCU") foreach ($subFolder in $subFolders) { $UpdateFolder = Join-Path $WindowsUpdateStagingFolder $subFolder if(Test-Path $UpdateFolder) { $items = @(Get-ChildItem -Path $UpdateFolder -Include @("*.cab", "*.msu") -Recurse) if($items.Count -gt 0) { Trace-Execution "Injecting windows updates staged at: $UpdateFolder" -Verbose if ($ScratchDirectory) { Trace-Execution "Refreshing PS Drive cache." Get-PSDrive | Out-Null Trace-Execution "Installing updates using ScratchDirectory '$ScratchDirectory'" New-Item -ItemType Directory -Path $ScratchDirectory -Force try { Add-WindowsPackage -PackagePath $UpdateFolder -Path $this.MountPath -ScratchDirectory $ScratchDirectory -IgnoreCheck:$IgnoreCheck } finally { Remove-Item -Force -Recurse -Path $ScratchDirectory -ErrorAction Ignore } } else { Add-WindowsPackage -PackagePath $UpdateFolder -Path $this.MountPath -IgnoreCheck:$IgnoreCheck } } else { Trace-Warning "Windows update folder: $UpdateFolder exists but no updated were found. Skipping windows update injection." -Verbose } } else { Trace-Warning "Windows update folder: $UpdateFolder not found. Skipping windows update injection." -Verbose } } } <# .SYNOPSIS Adds drivers to the mounted image .DESCRIPTION Adds drivers to the mounted image .PARAMETER DriverPath Location where the drivers will be staged for consumption .PARAMETER SkipDriverInjection Value from the customer config that specifies if the driver injection should be skipped or not. #> AddDriversToImage( [string] $DriverPath, [bool] $SkipDriverInjection ) { Trace-Execution "Performing driver injection for $($this.ImagePath)" # Perform driver injection if needed if (-not ($this.TestSkipDriverInjection($DriverPath, $SkipDriverInjection) )) { Trace-Execution "Adding drivers to the image from: $DriverPath" Trace-Execution "Getting existing OEM drivers." $existingDrivers = Get-WindowsDriver -Path $this.MountPath Trace-Execution "Add Windows drivers from $DriverPath." $addDrivers = Add-WindowsDriver -Driver $DriverPath -Path $this.MountPath -Recurse if($existingDrivers) { $addDrivers = $addDrivers | Where-Object {$_.Driver -notin $existingDrivers.Driver} } foreach($driver in $addDrivers) { Trace-Execution "$($driver.ProviderName) $(Split-Path -Path $driver.OriginalFileName -Leaf) version $($driver.Version) added as $($driver.Driver)" } } else { Trace-Execution "Skipping driver injection." } } <# .SYNOPSIS Adds a directory to the mounted image .DESCRIPTION Adds a directory to the mounted image .PARAMETER SourcePath Location where the directory will be staged for consumption .PARAMETER Destination Destination path of the directory to be added #> AddDirectoryToImage( [string] $SourcePath, [string] $Destination ) { $ErrorActionPreference = [System.Management.Automation.ActionPreference]::Stop # only copy if the source exists if(Test-Path -Path $SourcePath) { $Directory = Split-Path -Path $SourcePath -Leaf $DestinationPath = Join-Path -Path $this.MountPath (Join-Path -Path $Destination -ChildPath $Directory) # only copy if the destination doesn't already exist if(-not (Test-Path -Path $DestinationPath)) { $TempDestinationPath = "${DestinationPath}_Temp" try { # create a temporary version of the destination directory if(-not (Test-Path -Path $TempDestinationPath)) { Trace-Execution "Creating temporary directory $TempDestinationPath" $null = New-Item -Path $TempDestinationPath -ItemType Directory } # copy each file or directory from the source to the temporary folder foreach($item in (Get-ChildItem -Path $SourcePath)) { Trace-Execution "Copying $($item.FullName) to $TempDestinationPath" Copy-Item -Path $item.FullName -Destination $TempDestinationPath -Recurse } } finally { # rename the temporary folder Trace-Execution "Renaming temporary directory $TempDestinationPath to $Directory" Rename-Item -Path $TempDestinationPath -NewName $Directory } } else { Trace-Execution "$DestinationPath already exists" } } else { Trace-Execution "$SourcePath does not exist" } } <# .SYNOPSIS Adds OEM SLP license to the mounted image .DESCRIPTION Adds OEM SLP license file to the mounted image .PARAMETER Path Source location of the OEM license file #> AddOemLicense( [string] $Path ) { $ErrorActionPreference = [System.Management.Automation.ActionPreference]::Stop if(Test-Path -Path $Path) { $DestinationPath = Join-Path -Path $this.MountPath -ChildPath "\Windows\System32\OEM" New-Item -Path $DestinationPath -ItemType Directory -ErrorAction SilentlyContinue | Out-Null Trace-Execution "Copying $Path to $DestinationPath" Copy-Item -Path $Path -Destination $DestinationPath -Force } else { Trace-Execution "$Path does not exist" } } <# .SYNOPSIS Evaluates if driver injection should be skipped or not .PARAMETER DriverPath Location where the drivers will be staged for consumption .PARAMETER SkipDriverInjection Value from the customer config that specifies if the driver injection should be skipped or not. #> [bool] TestSkipDriverInjection( [string] $DriverPath, [bool] $SkipDriverInjection ) { $ErrorActionPreference = [System.Management.Automation.ActionPreference]::Stop if ($skipDriverInjection) { Trace-Warning 'Driver injection will be skipped as instructed by $CloudBuilder.SkipDriverInjection setting.' } elseif (-not (Test-Path $DriverPath)) { Trace-Execution "Driver injection will be skipped as the driver path '$DriverPath' does not exist." $skipDriverInjection = $true } elseif (-not (Get-ChildItem $DriverPath)) { Trace-Execution "Driver injection will be skipped as the driver path '$DriverPath' does not contain any files." $skipDriverInjection = $true } elseif (-not (Get-ChildItem $DriverPath -Filter *.inf -Recurse)) { Trace-Execution "Driver injection will be skipped as the driver path '$DriverPath' does not contain any drivers (no *.inf files found)." $skipDriverInjection = $true } return $skipDriverInjection } <# .SYNOPSIS Expands the deployment artifacts to the destination location. .DESCRIPTION Expands all deployment artifacts into the destination root path. .EXAMPLE ExpandDeploymentArtifacts($Parameters.Configuration.Role.PrivateInfo.DeploymentContent) .PARAMETER DeploymentContentNode The xml node containing the deployment content to deliver to the destination. #> ExpandDeploymentArtifacts( [System.Xml.XmlElement] $DeploymentContentNode, [string] $NugetStorePath ) { Expand-DeploymentArtifacts -DeploymentContentNode $DeploymentContentNode -DestinationRootPath $this.MountPath -NugetStorePath $NugetStorePath } <# .SYNOPSIS Expands the deployment artifacts to the destination location. .DESCRIPTION Expands all deployment artifacts into the destination root path. .EXAMPLE ExpandDeploymentArtifacts($Parameters.Configuration.Role.PrivateInfo.DeploymentContent) .PARAMETER DeploymentContentNode The xml node containing the deployment content to deliver to the destination. #> ExpandDeploymentArtifacts( [System.Xml.XmlElement] $DeploymentContentNode) { $NugetStorePath = "$env:SystemDrive\CloudDeployment\NuGetStore" Expand-DeploymentArtifacts -DeploymentContentNode $DeploymentContentNode -DestinationRootPath $this.MountPath -NugetStorePath $NugetStorePath } # Dispose methods for the class # Invoke this directly if you want the VHD to be unmounted immediately, rather than when the powershell session dies. [void]Dispose() { $this.Dispose($true) } [void]Dispose( [bool]$disposing ) { # Dismount the image that was updated if ($this.MountPath -ne $null) { Trace-Execution "Dismount and save mounted image." $logFilePath = "C:\Temp\vstor.etl" $successlogFilePath = "C:\Temp\vstorGood.etl" Copy-Item $PSScriptRoot\vstortrace.wprp C:\Temp -Force $logEnabled = $false if (-not (Test-Path $logFilePath)) { Trace-Execution "Starting WPR tracing for dismount." $logEnabled = $true & Wpr.exe -start C:\Temp\vstortrace.wprp!VStorWPP -filemode } try { Dismount-WindowsImage -Path $this.mountPath -Save -Verbose:$false } finally { if ($logEnabled) { Trace-Execution "Stopping WPR tracing for dismount." & Wpr.exe -stop $logFilePath } } if ($logEnabled) { if (-not (Test-Path $successlogFilePath)) { Trace-Execution "Keeping the first successful dismount log on this host." Copy-Item $logFilePath $successlogFilePath } Trace-Execution "Removing WPR log file since dismount is successful." Remove-Item $logFilePath -Force -ErrorAction Ignore } # Attempt to start defender if not already running $service = Get-Service -Name "windefend" -ErrorAction SilentlyContinue if ($service -ne $null -and $service.Status -ine "Running") { $service | Start-Service -ErrorAction SilentlyContinue | Out-Null } if($service -ne $null -and $service.Status -eq "Running") { Trace-Execution "Remove Windows Defender exclusions for paths $($this.MountPath)." Remove-MpPreference -ExclusionPath $this.MountPath -ErrorAction SilentlyContinue } $this.MountPath = $null } } } # SIG # Begin signature block # MIIoKQYJKoZIhvcNAQcCoIIoGjCCKBYCAQExDzANBglghkgBZQMEAgEFADB5Bgor # BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCB2YOY0j8mmGx+w # Lf3bvNI1wujoUg44MT3xbQEX/bMmtKCCDXYwggX0MIID3KADAgECAhMzAAADrzBA # DkyjTQVBAAAAAAOvMA0GCSqGSIb3DQEBCwUAMH4xCzAJBgNVBAYTAlVTMRMwEQYD # VQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNy # b3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNpZ25p # bmcgUENBIDIwMTEwHhcNMjMxMTE2MTkwOTAwWhcNMjQxMTE0MTkwOTAwWjB0MQsw # CQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9u # ZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMR4wHAYDVQQDExVNaWNy # b3NvZnQgQ29ycG9yYXRpb24wggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIB # AQDOS8s1ra6f0YGtg0OhEaQa/t3Q+q1MEHhWJhqQVuO5amYXQpy8MDPNoJYk+FWA # hePP5LxwcSge5aen+f5Q6WNPd6EDxGzotvVpNi5ve0H97S3F7C/axDfKxyNh21MG # 0W8Sb0vxi/vorcLHOL9i+t2D6yvvDzLlEefUCbQV/zGCBjXGlYJcUj6RAzXyeNAN # xSpKXAGd7Fh+ocGHPPphcD9LQTOJgG7Y7aYztHqBLJiQQ4eAgZNU4ac6+8LnEGAL # go1ydC5BJEuJQjYKbNTy959HrKSu7LO3Ws0w8jw6pYdC1IMpdTkk2puTgY2PDNzB # tLM4evG7FYer3WX+8t1UMYNTAgMBAAGjggFzMIIBbzAfBgNVHSUEGDAWBgorBgEE # AYI3TAgBBggrBgEFBQcDAzAdBgNVHQ4EFgQURxxxNPIEPGSO8kqz+bgCAQWGXsEw # RQYDVR0RBD4wPKQ6MDgxHjAcBgNVBAsTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEW # MBQGA1UEBRMNMjMwMDEyKzUwMTgyNjAfBgNVHSMEGDAWgBRIbmTlUAXTgqoXNzci # tW2oynUClTBUBgNVHR8ETTBLMEmgR6BFhkNodHRwOi8vd3d3Lm1pY3Jvc29mdC5j # b20vcGtpb3BzL2NybC9NaWNDb2RTaWdQQ0EyMDExXzIwMTEtMDctMDguY3JsMGEG # CCsGAQUFBwEBBFUwUzBRBggrBgEFBQcwAoZFaHR0cDovL3d3dy5taWNyb3NvZnQu # Y29tL3BraW9wcy9jZXJ0cy9NaWNDb2RTaWdQQ0EyMDExXzIwMTEtMDctMDguY3J0 # MAwGA1UdEwEB/wQCMAAwDQYJKoZIhvcNAQELBQADggIBAISxFt/zR2frTFPB45Yd # mhZpB2nNJoOoi+qlgcTlnO4QwlYN1w/vYwbDy/oFJolD5r6FMJd0RGcgEM8q9TgQ # 2OC7gQEmhweVJ7yuKJlQBH7P7Pg5RiqgV3cSonJ+OM4kFHbP3gPLiyzssSQdRuPY # 1mIWoGg9i7Y4ZC8ST7WhpSyc0pns2XsUe1XsIjaUcGu7zd7gg97eCUiLRdVklPmp # XobH9CEAWakRUGNICYN2AgjhRTC4j3KJfqMkU04R6Toyh4/Toswm1uoDcGr5laYn # TfcX3u5WnJqJLhuPe8Uj9kGAOcyo0O1mNwDa+LhFEzB6CB32+wfJMumfr6degvLT # e8x55urQLeTjimBQgS49BSUkhFN7ois3cZyNpnrMca5AZaC7pLI72vuqSsSlLalG # OcZmPHZGYJqZ0BacN274OZ80Q8B11iNokns9Od348bMb5Z4fihxaBWebl8kWEi2O # PvQImOAeq3nt7UWJBzJYLAGEpfasaA3ZQgIcEXdD+uwo6ymMzDY6UamFOfYqYWXk # ntxDGu7ngD2ugKUuccYKJJRiiz+LAUcj90BVcSHRLQop9N8zoALr/1sJuwPrVAtx # HNEgSW+AKBqIxYWM4Ev32l6agSUAezLMbq5f3d8x9qzT031jMDT+sUAoCw0M5wVt # CUQcqINPuYjbS1WgJyZIiEkBMIIHejCCBWKgAwIBAgIKYQ6Q0gAAAAAAAzANBgkq # hkiG9w0BAQsFADCBiDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24x # EDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlv # bjEyMDAGA1UEAxMpTWljcm9zb2Z0IFJvb3QgQ2VydGlmaWNhdGUgQXV0aG9yaXR5 # IDIwMTEwHhcNMTEwNzA4MjA1OTA5WhcNMjYwNzA4MjEwOTA5WjB+MQswCQYDVQQG # EwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwG # A1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSgwJgYDVQQDEx9NaWNyb3NvZnQg # Q29kZSBTaWduaW5nIFBDQSAyMDExMIICIjANBgkqhkiG9w0BAQEFAAOCAg8AMIIC # CgKCAgEAq/D6chAcLq3YbqqCEE00uvK2WCGfQhsqa+laUKq4BjgaBEm6f8MMHt03 # a8YS2AvwOMKZBrDIOdUBFDFC04kNeWSHfpRgJGyvnkmc6Whe0t+bU7IKLMOv2akr # rnoJr9eWWcpgGgXpZnboMlImEi/nqwhQz7NEt13YxC4Ddato88tt8zpcoRb0Rrrg # OGSsbmQ1eKagYw8t00CT+OPeBw3VXHmlSSnnDb6gE3e+lD3v++MrWhAfTVYoonpy # 4BI6t0le2O3tQ5GD2Xuye4Yb2T6xjF3oiU+EGvKhL1nkkDstrjNYxbc+/jLTswM9 # sbKvkjh+0p2ALPVOVpEhNSXDOW5kf1O6nA+tGSOEy/S6A4aN91/w0FK/jJSHvMAh # dCVfGCi2zCcoOCWYOUo2z3yxkq4cI6epZuxhH2rhKEmdX4jiJV3TIUs+UsS1Vz8k # A/DRelsv1SPjcF0PUUZ3s/gA4bysAoJf28AVs70b1FVL5zmhD+kjSbwYuER8ReTB # w3J64HLnJN+/RpnF78IcV9uDjexNSTCnq47f7Fufr/zdsGbiwZeBe+3W7UvnSSmn # Eyimp31ngOaKYnhfsi+E11ecXL93KCjx7W3DKI8sj0A3T8HhhUSJxAlMxdSlQy90 # lfdu+HggWCwTXWCVmj5PM4TasIgX3p5O9JawvEagbJjS4NaIjAsCAwEAAaOCAe0w # ggHpMBAGCSsGAQQBgjcVAQQDAgEAMB0GA1UdDgQWBBRIbmTlUAXTgqoXNzcitW2o # ynUClTAZBgkrBgEEAYI3FAIEDB4KAFMAdQBiAEMAQTALBgNVHQ8EBAMCAYYwDwYD # VR0TAQH/BAUwAwEB/zAfBgNVHSMEGDAWgBRyLToCMZBDuRQFTuHqp8cx0SOJNDBa # BgNVHR8EUzBRME+gTaBLhklodHRwOi8vY3JsLm1pY3Jvc29mdC5jb20vcGtpL2Ny # bC9wcm9kdWN0cy9NaWNSb29DZXJBdXQyMDExXzIwMTFfMDNfMjIuY3JsMF4GCCsG # AQUFBwEBBFIwUDBOBggrBgEFBQcwAoZCaHR0cDovL3d3dy5taWNyb3NvZnQuY29t # L3BraS9jZXJ0cy9NaWNSb29DZXJBdXQyMDExXzIwMTFfMDNfMjIuY3J0MIGfBgNV # HSAEgZcwgZQwgZEGCSsGAQQBgjcuAzCBgzA/BggrBgEFBQcCARYzaHR0cDovL3d3 # dy5taWNyb3NvZnQuY29tL3BraW9wcy9kb2NzL3ByaW1hcnljcHMuaHRtMEAGCCsG # AQUFBwICMDQeMiAdAEwAZQBnAGEAbABfAHAAbwBsAGkAYwB5AF8AcwB0AGEAdABl # AG0AZQBuAHQALiAdMA0GCSqGSIb3DQEBCwUAA4ICAQBn8oalmOBUeRou09h0ZyKb # C5YR4WOSmUKWfdJ5DJDBZV8uLD74w3LRbYP+vj/oCso7v0epo/Np22O/IjWll11l # hJB9i0ZQVdgMknzSGksc8zxCi1LQsP1r4z4HLimb5j0bpdS1HXeUOeLpZMlEPXh6 # I/MTfaaQdION9MsmAkYqwooQu6SpBQyb7Wj6aC6VoCo/KmtYSWMfCWluWpiW5IP0 # wI/zRive/DvQvTXvbiWu5a8n7dDd8w6vmSiXmE0OPQvyCInWH8MyGOLwxS3OW560 # STkKxgrCxq2u5bLZ2xWIUUVYODJxJxp/sfQn+N4sOiBpmLJZiWhub6e3dMNABQam # ASooPoI/E01mC8CzTfXhj38cbxV9Rad25UAqZaPDXVJihsMdYzaXht/a8/jyFqGa # J+HNpZfQ7l1jQeNbB5yHPgZ3BtEGsXUfFL5hYbXw3MYbBL7fQccOKO7eZS/sl/ah # XJbYANahRr1Z85elCUtIEJmAH9AAKcWxm6U/RXceNcbSoqKfenoi+kiVH6v7RyOA # 9Z74v2u3S5fi63V4GuzqN5l5GEv/1rMjaHXmr/r8i+sLgOppO6/8MO0ETI7f33Vt # Y5E90Z1WTk+/gFcioXgRMiF670EKsT/7qMykXcGhiJtXcVZOSEXAQsmbdlsKgEhr # /Xmfwb1tbWrJUnMTDXpQzTGCGgkwghoFAgEBMIGVMH4xCzAJBgNVBAYTAlVTMRMw # EQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVN # aWNyb3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNp # Z25pbmcgUENBIDIwMTECEzMAAAOvMEAOTKNNBUEAAAAAA68wDQYJYIZIAWUDBAIB # BQCgga4wGQYJKoZIhvcNAQkDMQwGCisGAQQBgjcCAQQwHAYKKwYBBAGCNwIBCzEO # MAwGCisGAQQBgjcCARUwLwYJKoZIhvcNAQkEMSIEIFbN7YT+y3oIhivuTEAzaagY # ixjWJoAyB2IvxkC3fHt3MEIGCisGAQQBgjcCAQwxNDAyoBSAEgBNAGkAYwByAG8A # cwBvAGYAdKEagBhodHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20wDQYJKoZIhvcNAQEB # BQAEggEAy1ZjMf4s/tUOlR2sfofv8UE6J7UrVi+rNogZmZckpSaiTLMZi57QtHmS # 6RNdGa1OTNLgxfzLuMXsinUZURUTXrGFCJgwroo8UOn2BjopXvT+NZ6UbX6Jk+66 # +tgWgdUQUFidKrRdXCOxDNQWH7D1mLg9cknI5ZixoOQ0eZHINsBCYAgVLRdMydOQ # DeW/kv7uZTn20SvhUj43qNvTf1A+Eu7iQkvbYJeWQaIlWJdeLV4SxvIyGJl2vO4J # d5oNFeg7vZZC52xB7r0Vlsx/nD4V9TJQqDtzvQe5or8OKqAL4jPKrUTrsB36KLV8 # +8GJzrp6cktFGKgMLn7M1RJHJgKjg6GCF5MwghePBgorBgEEAYI3AwMBMYIXfzCC # F3sGCSqGSIb3DQEHAqCCF2wwghdoAgEDMQ8wDQYJYIZIAWUDBAIBBQAwggFSBgsq # hkiG9w0BCRABBKCCAUEEggE9MIIBOQIBAQYKKwYBBAGEWQoDATAxMA0GCWCGSAFl # AwQCAQUABCBoGquwTxuVTAlM6d7jvzLQ0KRZFLCxWrtR2SHcWDCuWgIGZkYZ0Mul # GBMyMDI0MDUxNjE4NDQyNS41MjhaMASAAgH0oIHRpIHOMIHLMQswCQYDVQQGEwJV # UzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UE # ChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSUwIwYDVQQLExxNaWNyb3NvZnQgQW1l # cmljYSBPcGVyYXRpb25zMScwJQYDVQQLEx5uU2hpZWxkIFRTUyBFU046REMwMC0w # NUUwLUQ5NDcxJTAjBgNVBAMTHE1pY3Jvc29mdCBUaW1lLVN0YW1wIFNlcnZpY2Wg # ghHpMIIHIDCCBQigAwIBAgITMwAAAehQsIDPK3KZTQABAAAB6DANBgkqhkiG9w0B # AQsFADB8MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UE # BxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSYwJAYD # VQQDEx1NaWNyb3NvZnQgVGltZS1TdGFtcCBQQ0EgMjAxMDAeFw0yMzEyMDYxODQ1 # MjJaFw0yNTAzMDUxODQ1MjJaMIHLMQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2Fz # aGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENv # cnBvcmF0aW9uMSUwIwYDVQQLExxNaWNyb3NvZnQgQW1lcmljYSBPcGVyYXRpb25z # MScwJQYDVQQLEx5uU2hpZWxkIFRTUyBFU046REMwMC0wNUUwLUQ5NDcxJTAjBgNV # BAMTHE1pY3Jvc29mdCBUaW1lLVN0YW1wIFNlcnZpY2UwggIiMA0GCSqGSIb3DQEB # AQUAA4ICDwAwggIKAoICAQDhQXdE0WzXG7wzeC9SGdH6eVwdGlF6YgpU7weOFBkp # W9yuEmJSDE1ADBx/0DTuRBaplSD8CR1QqyQmxRDD/CdvDyeZFAcZ6l2+nlMssmZy # C8TPt1GTWAUt3GXUU6g0F0tIrFNLgofCjOvm3G0j482VutKS4wZT6bNVnBVsChr2 # AjmVbGDN/6Qs/EqakL5cwpGel1te7UO13dUwaPjOy0Wi1qYNmR8i7T1luj2JdFdf # ZhMPyqyq/NDnZuONSbj8FM5xKBoar12ragC8/1CXaL1OMXBwGaRoJTYtksi9njuq # 4wDkcAwitCZ5BtQ2NqPZ0lLiQB7O10Bm9zpHWn9x1/HmdAn4koMWKUDwH5sd/zDu # 4vi887FWxm54kkWNvk8FeQ7ZZ0Q5gqGKW4g6revV2IdAxBobWdorqwvzqL70Wdsg # DU/P5c0L8vYIskUJZedCGHM2hHIsNRyw9EFoSolDM+yCedkz69787s8nIp55icLf # DoKw5hak5G6MWF6d71tcNzV9+v9RQKMa6Uwfyquredd5sqXWCXv++hek4A15WybI # c6ufT0ilazKYZvDvoaswgjP0SeLW7mvmcw0FELzF1/uWaXElLHOXIlieKF2i/YzQ # 6U50K9dbhnMaDcJSsG0hXLRTy/LQbsOD0hw7FuK0nmzotSx/5fo9g7fCzoFjk3tD # EwIDAQABo4IBSTCCAUUwHQYDVR0OBBYEFPo5W8o980kMfRVQba6T34HwelLaMB8G # A1UdIwQYMBaAFJ+nFV0AXmJdg/Tl0mWnG1M1GelyMF8GA1UdHwRYMFYwVKBSoFCG # Tmh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMvY3JsL01pY3Jvc29mdCUy # MFRpbWUtU3RhbXAlMjBQQ0ElMjAyMDEwKDEpLmNybDBsBggrBgEFBQcBAQRgMF4w # XAYIKwYBBQUHMAKGUGh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMvY2Vy # dHMvTWljcm9zb2Z0JTIwVGltZS1TdGFtcCUyMFBDQSUyMDIwMTAoMSkuY3J0MAwG # A1UdEwEB/wQCMAAwFgYDVR0lAQH/BAwwCgYIKwYBBQUHAwgwDgYDVR0PAQH/BAQD # AgeAMA0GCSqGSIb3DQEBCwUAA4ICAQCWfcJm2rwXtPi74km6PKAkni9+BWotq+Qt # DGgeT5F3ro7PsIUNKRkUytuGqI8thL3Jcrb03x6DOppYJEA+pb6o2qPjFddO1TLq # vSXrYm+OgCLL+7+3FmRmfkRu8rHvprab0O19wDbukgO8I5Oi1RegMJl8t5k/UtE0 # Wb3zAlOHnCjLGSzP/Do3ptwhXokk02IvD7SZEBbPboGbtw4LCHsT2pFakpGOBh+I # SUMXBf835CuVNfddwxmyGvNSzyEyEk5h1Vh7tpwP7z7rJ+HsiP4sdqBjj6Avopuf # 4rxUAfrEbV6aj8twFs7WVHNiIgrHNna/55kyrAG9Yt19CPvkUwxYK0uZvPl2WC39 # nfc0jOTjivC7s/IUozE4tfy3JNkyQ1cNtvZftiX3j5Dt+eLOeuGDjvhJvYMIEkpk # V68XLNH7+ZBfYa+PmfRYaoFFHCJKEoRSZ3PbDJPBiEhZ9yuxMddoMMQ19Tkyftot # 6Ez0XhSmwjYBq39DvBFWhlyDGBhrU3GteDWiVd9YGSB2WnxuFMy5fbAK6o8PWz8Q # RMiptXHK3HDBr2wWWEcrrgcTuHZIJTqepNoYlx9VRFvj/vCXaAFcmkW1nk7VE+ow # aXr5RJjryDq9ubkyDq1mdrF/geaRALXcNZbfNXIkhXzXA6a8CiamcQW/DgmLJpiV # QNriZYCHIDCCB3EwggVZoAMCAQICEzMAAAAVxedrngKbSZkAAAAAABUwDQYJKoZI # hvcNAQELBQAwgYgxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAw # DgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24x # MjAwBgNVBAMTKU1pY3Jvc29mdCBSb290IENlcnRpZmljYXRlIEF1dGhvcml0eSAy # MDEwMB4XDTIxMDkzMDE4MjIyNVoXDTMwMDkzMDE4MzIyNVowfDELMAkGA1UEBhMC # VVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNV # BAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRp # bWUtU3RhbXAgUENBIDIwMTAwggIiMA0GCSqGSIb3DQEBAQUAA4ICDwAwggIKAoIC # AQDk4aZM57RyIQt5osvXJHm9DtWC0/3unAcH0qlsTnXIyjVX9gF/bErg4r25Phdg # M/9cT8dm95VTcVrifkpa/rg2Z4VGIwy1jRPPdzLAEBjoYH1qUoNEt6aORmsHFPPF # dvWGUNzBRMhxXFExN6AKOG6N7dcP2CZTfDlhAnrEqv1yaa8dq6z2Nr41JmTamDu6 # GnszrYBbfowQHJ1S/rboYiXcag/PXfT+jlPP1uyFVk3v3byNpOORj7I5LFGc6XBp # Dco2LXCOMcg1KL3jtIckw+DJj361VI/c+gVVmG1oO5pGve2krnopN6zL64NF50Zu # yjLVwIYwXE8s4mKyzbnijYjklqwBSru+cakXW2dg3viSkR4dPf0gz3N9QZpGdc3E # XzTdEonW/aUgfX782Z5F37ZyL9t9X4C626p+Nuw2TPYrbqgSUei/BQOj0XOmTTd0 # lBw0gg/wEPK3Rxjtp+iZfD9M269ewvPV2HM9Q07BMzlMjgK8QmguEOqEUUbi0b1q # GFphAXPKZ6Je1yh2AuIzGHLXpyDwwvoSCtdjbwzJNmSLW6CmgyFdXzB0kZSU2LlQ # +QuJYfM2BjUYhEfb3BvR/bLUHMVr9lxSUV0S2yW6r1AFemzFER1y7435UsSFF5PA # PBXbGjfHCBUYP3irRbb1Hode2o+eFnJpxq57t7c+auIurQIDAQABo4IB3TCCAdkw # EgYJKwYBBAGCNxUBBAUCAwEAATAjBgkrBgEEAYI3FQIEFgQUKqdS/mTEmr6CkTxG # NSnPEP8vBO4wHQYDVR0OBBYEFJ+nFV0AXmJdg/Tl0mWnG1M1GelyMFwGA1UdIARV # MFMwUQYMKwYBBAGCN0yDfQEBMEEwPwYIKwYBBQUHAgEWM2h0dHA6Ly93d3cubWlj # cm9zb2Z0LmNvbS9wa2lvcHMvRG9jcy9SZXBvc2l0b3J5Lmh0bTATBgNVHSUEDDAK # BggrBgEFBQcDCDAZBgkrBgEEAYI3FAIEDB4KAFMAdQBiAEMAQTALBgNVHQ8EBAMC # AYYwDwYDVR0TAQH/BAUwAwEB/zAfBgNVHSMEGDAWgBTV9lbLj+iiXGJo0T2UkFvX # zpoYxDBWBgNVHR8ETzBNMEugSaBHhkVodHRwOi8vY3JsLm1pY3Jvc29mdC5jb20v # cGtpL2NybC9wcm9kdWN0cy9NaWNSb29DZXJBdXRfMjAxMC0wNi0yMy5jcmwwWgYI # KwYBBQUHAQEETjBMMEoGCCsGAQUFBzAChj5odHRwOi8vd3d3Lm1pY3Jvc29mdC5j # b20vcGtpL2NlcnRzL01pY1Jvb0NlckF1dF8yMDEwLTA2LTIzLmNydDANBgkqhkiG # 9w0BAQsFAAOCAgEAnVV9/Cqt4SwfZwExJFvhnnJL/Klv6lwUtj5OR2R4sQaTlz0x # M7U518JxNj/aZGx80HU5bbsPMeTCj/ts0aGUGCLu6WZnOlNN3Zi6th542DYunKmC # VgADsAW+iehp4LoJ7nvfam++Kctu2D9IdQHZGN5tggz1bSNU5HhTdSRXud2f8449 # xvNo32X2pFaq95W2KFUn0CS9QKC/GbYSEhFdPSfgQJY4rPf5KYnDvBewVIVCs/wM # nosZiefwC2qBwoEZQhlSdYo2wh3DYXMuLGt7bj8sCXgU6ZGyqVvfSaN0DLzskYDS # PeZKPmY7T7uG+jIa2Zb0j/aRAfbOxnT99kxybxCrdTDFNLB62FD+CljdQDzHVG2d # Y3RILLFORy3BFARxv2T5JL5zbcqOCb2zAVdJVGTZc9d/HltEAY5aGZFrDZ+kKNxn # GSgkujhLmm77IVRrakURR6nxt67I6IleT53S0Ex2tVdUCbFpAUR+fKFhbHP+Crvs # QWY9af3LwUFJfn6Tvsv4O+S3Fb+0zj6lMVGEvL8CwYKiexcdFYmNcP7ntdAoGokL # jzbaukz5m/8K6TT4JDVnK+ANuOaMmdbhIurwJ0I9JZTmdHRbatGePu1+oDEzfbzL # 6Xu/OHBE0ZDxyKs6ijoIYn/ZcGNTTY3ugm2lBRDBcQZqELQdVTNYs6FwZvKhggNM # MIICNAIBATCB+aGB0aSBzjCByzELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hp # bmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jw # b3JhdGlvbjElMCMGA1UECxMcTWljcm9zb2Z0IEFtZXJpY2EgT3BlcmF0aW9uczEn # MCUGA1UECxMeblNoaWVsZCBUU1MgRVNOOkRDMDAtMDVFMC1EOTQ3MSUwIwYDVQQD # ExxNaWNyb3NvZnQgVGltZS1TdGFtcCBTZXJ2aWNloiMKAQEwBwYFKw4DAhoDFQCM # JG4vg0juMOVn2BuKACUvP80FuqCBgzCBgKR+MHwxCzAJBgNVBAYTAlVTMRMwEQYD # VQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNy # b3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1w # IFBDQSAyMDEwMA0GCSqGSIb3DQEBCwUAAgUA6fCYTzAiGA8yMDI0MDUxNjE0MzU1 # OVoYDzIwMjQwNTE3MTQzNTU5WjBzMDkGCisGAQQBhFkKBAExKzApMAoCBQDp8JhP # AgEAMAYCAQACASAwBwIBAAICElcwCgIFAOnx6c8CAQAwNgYKKwYBBAGEWQoEAjEo # MCYwDAYKKwYBBAGEWQoDAqAKMAgCAQACAwehIKEKMAgCAQACAwGGoDANBgkqhkiG # 9w0BAQsFAAOCAQEAjk+W0YnLhKp0SORNHGi4JzUkO1Yu9ka8hF0KsneXGY/w7cVX # S7HieMN6zuxvjYkbafq4NuTYjnvv/HBW973d8eDf263+FvdNnLz8xu2aV5alXEyQ # JBQUDYlMpBkSVUJ+RQle+NnUbJGDdNv/MyVMWMnYw1UdkslSZz11w9uAyTuy116B # 0hAaV1nsg9IbBxI6e0aTaXzvupqytlF0OhEuIAPZtQsDUzP8i6y3G7IC9N29AZrD # 42CxAI7zgtKauOjrfyPDYo+D+KXip+I1MMbYGQeJ9J3ZU7xTJTJeVKTfIKuWb+sH # OrQADlVGd+SZqhcJ5LXJ/ubTEsyEFoPoODPYdzGCBA0wggQJAgEBMIGTMHwxCzAJ # BgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25k # MR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jv # c29mdCBUaW1lLVN0YW1wIFBDQSAyMDEwAhMzAAAB6FCwgM8rcplNAAEAAAHoMA0G # CWCGSAFlAwQCAQUAoIIBSjAaBgkqhkiG9w0BCQMxDQYLKoZIhvcNAQkQAQQwLwYJ # KoZIhvcNAQkEMSIEIPnqZmGVRqLA2pQa+/UnZseBN3NuNb7dTlErRE4mPGgOMIH6 # BgsqhkiG9w0BCRACLzGB6jCB5zCB5DCBvQQgKtLaxNUChCCCQdHn2k2qKB7TF8lP # YndTxbVJzwf46x0wgZgwgYCkfjB8MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2Fz # aGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENv # cnBvcmF0aW9uMSYwJAYDVQQDEx1NaWNyb3NvZnQgVGltZS1TdGFtcCBQQ0EgMjAx # MAITMwAAAehQsIDPK3KZTQABAAAB6DAiBCCn0IYn/ySGCFfhzdyA0xzr4MGrVGRb # JrdweyhslNIENjANBgkqhkiG9w0BAQsFAASCAgAmX3/gq0gguFo3Q6HtzL8B7kfK # tmZ/bto8KpG4jR6KVLttyM8wiXSwRgOb5r28JUSC0dEXncFySp5T9MoULCw2S+XV # QU8jkcfAf4OFIF5PE6br/uzVGRSPwfLjzgXO4ZTeafhmZRJNghDQG2Nfacb8toHU # rqzAWA457VXfl/H/R2G/Siv89HDXSLhX0/pymyQv9vZP1bol7C6G2xiwKioJCLam # uOzEd2wP85h/VMaDlythmLu/LnDCgCk5U5NLeFymoNN6oPw+J9oZ/MjqSwoUZqE3 # BGETiOk0t4CoGJt56EeXvoST4tLmzYzX0x+CgiGnPNoaQNOvJqp8mibo1+IjSKcf # 5qVXSB8fziqf+4izN/kmEvnfhnoCGOWAKVWZq9tx0xZcdTH12tbfGbuAz3RPy376 # 8rL4xjlrgC2oEQPSOh/Ds+BFlAnG9z47iTfdiTtTs3SR9+66CFZZ2rhnOP9Le02O # mN1Sf+qFn8NXM1BLvz9xNwC6YsLLwY29wkUhRMhQL6SgBWyUXb7tsFYWWekm22VN # o/UFbBwhA/UN55PzyBHXdF61zHFQnV7q4XaH5IMGR2qy7Y1KjUj11Fo51LQ0WbNc # OrtkClzwu49hjM7ZWXMHqA4QWlQ+9yuWr0ZsqICk7MuDSTlBwX97ZIsmEUtGmnXH # fcb/x5w9XwesznOZwA== # SIG # End signature block |