Src/Public/Public.ps1
<#
.SYNOPSIS Clears cached Virtual Engine Evergreen package files. .DESCRIPTION Removes all Evergreen package installer and metadata files from the '%ALLUSERSPROFILE%\VirtualEngine\Evergreen' directory. .EXAMPLE Clear-EvergreenPackageCache Removes all cached Evergreen packages and package metadata. #> function Clear-EvergreenPackageCache { [CmdletBinding(SupportsShouldProcess)] param ( ## Force cached file removal. [Parameter(ValueFromPipelineByPropertyName)] [System.Management.Automation.SwitchParameter] $Force ) process { Remove-Item -Path "$cachedMetadataPath\*" -Exclude 'AppDeployToolkit*' -Force:$Force -Recurse } } <# .SYNOPSIS Searches for available Virtual Engine Evergreen packages. .DESCRIPTION Queries the Virtual Engine Evergreen API for one or more available packages. .EXAMPLE Find-EvergreenPackage -Name 7-Zip-x64 -SubscriptionKey abcdef1234567890 Name : 7-Zip-x64 Version : 19.0 Platform : Intune Uri : https://../7-Zip-x64/7-Zip-x64.19.0.intunewin?sv=2019-02-02 MetadataUri : https://../7-Zip-x64/7-Zip-x64.19.0.intunewin.metadata?sv=2019-02-02 Name : 7-Zip-x64 Version : 19.0 Platform : Zip Uri : https://../7-Zip-x64/7-Zip-x64.19.0.zip?sv=2019-02-02 MetadataUri : Searches for the latest 7-Zip-x64 package. .EXAMPLE Find-EvergreenPackage -Name 7-Zip-x64 -Platform Zip Name : 7-Zip-x64 Version : 19.0 Platform : Zip Uri : https://../7-Zip-x64/7-Zip-x64.19.0.zip?sv=2019-02-02 MetadataUri : Searches for the latest 7-Zip-x64 archive package. .EXAMPLE Find-EvergreenPackage -Name Google-Chrome-Enterprise-x64 -Platform Intune -All Name : Google-Chrome-Enterprise-x64 Version : 81.0.4044.129 Platform : Intune Uri : https://../Google-Chrome-Enterprise-x64/Google-Chrome-Enterprise-x64.81.0.4044.129.intunewin?sv=2019-02-02 MetadataUri : https://../Google-Chrome-Enterprise-x64/Google-Chrome-Enterprise-x64.81.0.4044.129.intunewin.metadata?sv=2019-02-02 Name : Google-Chrome-Enterprise-x64 Version : 81.0.4044.138 Platform : Intune Uri : https://../Google-Chrome-Enterprise-x64/Google-Chrome-Enterprise-x64.81.0.4044.138.intunewin?sv=2019-02-02 MetadataUri : https://../Google-Chrome-Enterprise-x64/Google-Chrome-Enterprise-x64.81.0.4044.138.intunewin.metadata?sv=2019-02-02 Name : Google-Chrome-Enterprise-x64 Version : 81.0.4044.92 Platform : Intune Uri : https://../Google-Chrome-Enterprise-x64/Google-Chrome-Enterprise-x64.81.0.4044.92.intunewin?sv=2019-02-02 MetadataUri : https://../Google-Chrome-Enterprise-x64/Google-Chrome-Enterprise-x64.81.0.4044.92.intunewin.metadata?sv=2019-02-02 Name : Google-Chrome-Enterprise-x64 Version : 83.0.4103.61 Platform : Intune Uri : https://../Google-Chrome-Enterprise-x64/Google-Chrome-Enterprise-x64.83.0.4103.61.intunewin?sv=2019-02-02 MetadataUri : https://../Google-Chrome-Enterprise-x64/Google-Chrome-Enterprise-x64.83.0.4103.61.intunewin.metadata?sv=2019-02-02 Searches for all available Google-Chrome-Enterprise-x64 Intune packages. #> function Find-EvergreenPackage { [CmdletBinding()] [OutputType([System.Management.Automation.PSObject])] param ( ## Evergreen package(s) to search for. [Parameter(ValueFromPipelineByPropertyName)] [System.String[]] $Name, ## Virtual Engine Evergreen Api subscription key. [Parameter(Mandatory, ValueFromPipelineByPropertyName)] [Alias('Key')] [System.String] $SubscriptionKey, ## Enumerate customer package repository. [Parameter(ValueFromPipelineByPropertyName)] [System.String] $CustomerCode, ## Evergreen package format. [Parameter(ValueFromPipelineByPropertyName)] [ValidateSet('Exe', 'Zip', 'Intune')] [System.String] $Platform, ## Return all available package versions. [Parameter(ValueFromPipelineByPropertyName)] [System.Management.Automation.SwitchParameter] $All, ## Include hidden (test) packages. [Parameter(ValueFromPipelineByPropertyName)] [System.Management.Automation.SwitchParameter] $Hidden, ## Web proxy credential required to access Virtual Engine Evergreen Api. [Parameter(ValueFromPipelineByPropertyName)] [System.Management.Automation.PSCredential] [System.Management.Automation.CredentialAttribute()] $Credential ) process { $invokeEvergreenPackageApiParams = @{ SubscriptionKey = $SubscriptionKey Hidden = $Hidden All = $All } if ($PSBoundParameters.ContainsKey('CustomerCode')) { $invokeEvergreenPackageApiParams['CustomerCode'] = $CustomerCode } if ($PSBoundParameters.ContainsKey('Credential')) { $invokeEvergreenPackageApiParams['Credential'] = $Credential } if ($PSBoundParameters.ContainsKey('Platform')) { $invokeEvergreenPackageApiParams['Platform'] = $Platform } if ($PSBoundParameters.ContainsKey('Name')) { foreach ($packageName in $Name) { $invokeEvergreenPackageApiParams['Name'] = $packageName $packages = Invoke-EvergreenPackageApi @invokeEvergreenPackageApiParams ## Evergreen API returns an empty string if there are no matching packages if ($packages -is [System.String]) { Write-Error ($localized.CannotFindPackageError -f $packageName) } elseif (($packages -is [System.Array]) -and ($packages.Count -eq 0)) { Write-Error ($localized.CannotFindPackageError -f $packageName) } else { $packages | ForEach-Object { $package = [PSCustomObject] @{ Name = $_.Name Description = $_.Description Version = $_.Version -as [System.Version] Revision = $_.Revision -as [System.Int32] Platform = $_.Platform PublishDate = $_.Date Uri = $_.Uri MetadataUri = $_.MetadataUri IsCustom = $PSBoundParameters.ContainsKey('CustomerCode') } $package.PSObject.TypeNames.Insert(0, 'VirtualEngine.Evergreen.Package') Write-Output -InputObject $package } } } } else { $packages = Invoke-EvergreenPackageApi @invokeEvergreenPackageApiParams if (($packages -is [System.Object[]]) -and ($packages[0] -isnot [System.Management.Automation.PSObject])) { ## Deserialisation sometimes results in an [Object[]] with a nested [PSObject]s $packages[0] | ForEach-Object { $package = [PSCustomObject] @{ Name = $_.Name Description = $_.Description Version = $_.Version -as [System.Version] Revision = $_.Revision -as [System.Int32] Platform = $_.Platform PublishDate = $_.Date Uri = $_.Uri MetadataUri = $_.MetadataUri IsCustom = $PSBoundParameters.ContainsKey('CustomerCode') } $package.PSObject.TypeNames.Insert(0, 'VirtualEngine.Evergreen.Package') Write-Output -InputObject $package } } else { $packages | ForEach-Object { $package = [PSCustomObject] @{ Name = $_.Name Description = $_.Description Version = $_.Version -as [System.Version] Revision = $_.Revision -as [System.Int32] Platform = $_.Platform PublishDate = $_.Date Uri = $_.Uri MetadataUri = $_.MetadataUri IsCustom = $PSBoundParameters.ContainsKey('CustomerCode') } $package.PSObject.TypeNames.Insert(0, 'VirtualEngine.Evergreen.Package') Write-Output -InputObject $package } } } } } <# .SYNOPSIS Returns Virtual Engine Evergreen package metadata. .DESCRIPTION Returns the cached Virtual Engine Evergreen package metadata. Downloads package metadata to the local file system if necessary. #> function Get-EvergreenPackageCacheMetadata { [CmdletBinding()] [OutputType([System.Collections.Hashtable])] param ( ## Evergreen package name. [Parameter(Mandatory, ValueFromPipelineByPropertyName)] [System.String] $Name, ## Evergreen Api subscription key. [Parameter(Mandatory, ValueFromPipelineByPropertyName)] [System.String] $SubscriptionKey, ## Enumerate customer package repository. [Parameter(ValueFromPipelineByPropertyName)] [System.String] $CustomerCode, ## Web proxy credential required to access Virtual Engine Evergreen Api. [Parameter(ValueFromPipelineByPropertyName)] [System.Management.Automation.PSCredential] [System.Management.Automation.CredentialAttribute()] $Credential, ## Force refresh of cached package metadata. [Parameter(ValueFromPipelineByPropertyName)] [System.Management.Automation.SwitchParameter] $Force ) process { Write-Verbose -Message ($localized.RetrievingPackageMetadata -f $Name) try { $destinationFilename = '{0}.psd1' -f $Name $destinationPath = Join-Path -Path $cachedMetadataPath -ChildPath $destinationFilename $metadataPath = (Get-Item -Path $destinationPath -ErrorAction SilentlyContinue).FullName if ($Force -or ($null -eq $metadataPath)) { $invokeEvergreenMetadataDownloadParams = @{ Name = $Name SubscriptionKey = $SubscriptionKey Force = $Force } if ($PSBoundParameters.ContainsKey('CustomerCode')) { $invokeEvergreenMetadataDownloadParams['CustomerCode'] = $CustomerCode } if ($PSBoundParameters.ContainsKey('Credential')) { $invokeEvergreenMetadataDownloadParams['Credential'] = $Credential } $metadataPath = (Invoke-EvergreenMetadataDownload @invokeEvergreenMetadataDownloadParams).FullName } else { Write-Verbose -Message ($localized.UsingCachedPackage -f $metadataPath) } return Import-PowerShellDataFile -Path $metadataPath } catch { Write-Error $_ } } } <# .SYNOPSIS Installs one or more Virtual Engine Evergreen package(s). .DESCRIPTION Downloads Virtual Engine Evergreen packages to the local file system and intitiates the install. .EXAMPLE Install-EvergreenPackage -Name 7-Zip-x64 -SubscriptionKey abcdef1234567890 Downloads and installs the latest '7-Zip-x64' Evergreen package. .EXAMPLE Install-EvergreenPackage -Name 7-Zip-x64 -SubscriptionKey abcdef1234567890 -Verify Downloads, installs the latest '7-Zip-x64' Evergreen package and tests the installation is detected successfully. Throws an error when no Intune detection rule has been defined or defined Intune detection rule(s) fail. .EXAMPLE Install-EvergreenPackage -Name 7-Zip-x64 -SubscriptionKey abcdef1234567890 -Verify -IgnoreEmptyDetectionRule Downloads, installs the latest '7-Zip-x64' Evergreen package and tests the installation is detected successfully. Does not throw an error when no Intune detection rule has been defined. #> function Install-EvergreenPackage { [CmdletBinding(SupportsShouldProcess, DefaultParameterSetName = 'Default')] param ( ## Evergreen package(s) to install. [Parameter(Mandatory, ValueFromPipelineByPropertyName)] [System.String[]] $Name, ## Evergreen Api subscription key. [Parameter(Mandatory, ValueFromPipelineByPropertyName)] [System.String] $SubscriptionKey, ## Run the application's detection rules after installation to verify the application is detected as installed. [Parameter(Mandatory, ValueFromPipelineByPropertyName, ParameterSetName = 'Verify')] [System.Management.Automation.SwitchParameter] $Verify, ## Enumerate customer package repository. [Parameter(ValueFromPipelineByPropertyName)] [System.String] $CustomerCode, ## Custom package installation arguments. [Parameter(ValueFromPipelineByPropertyName)] [System.String] $Arguments, ## Web proxy credential required to access Virtual Engine Evergreen Api. [Parameter(ValueFromPipelineByPropertyName)] [System.Management.Automation.PSCredential] [System.Management.Automation.CredentialAttribute()] $Credential, ## Force refresh of cached package metadata. [Parameter(ValueFromPipelineByPropertyName)] [System.Management.Automation.SwitchParameter] $Force, ## Does not fail application verification if no detection rules are defined. [Parameter(ValueFromPipelineByPropertyName, ParameterSetName = 'Verify')] [System.Management.Automation.SwitchParameter] $IgnoreEmptyDetectionRule, ## Download retry attempts. [Parameter(ValueFromPipelineByPropertyName)] [System.Uint32] $Retry = 3 ) process { foreach ($packageName in $Name) { try { ## Cache metadata file to ensure any future Test-EvergreenPackage uses the ## detection rules used when initially installing the package $assertEvergreenPackageCacheMetadataParams = @{ Name = $packageName SubscriptionKey = $SubscriptionKey Force = $Force.ToBool() } if ($PSBoundParameters.ContainsKey('CustomerCode')) { $assertEvergreenPackageCacheMetadataParams['CustomerCode'] = $CustomerCode } if ($PSBoundParameters.ContainsKey('Credential')) { $assertEvergreenPackageCacheMetadataParams['Credential'] = $Credential } Assert-EvergreenPackageCacheMetadata @assertEvergreenPackageCacheMetadataParams ## Test to see if the package is already installed (assuming detection rules are defined) $isEvergreenPackageInstalled = Test-EvergreenPackage @assertEvergreenPackageCacheMetadataParams -IgnoreEmptyDetectionRule if ($null -eq $isEvergreenPackageInstalled) { ## Unable to determine installation state if ($Verify -and (-not $IgnoreEmptyDetectionRule)) { ## We have a package with no detection rules and we are not ignoring them throw ($localized.CannotFindPackageDetectionRuleError -f $packageName) } } elseif ($isEvergreenPackageInstalled -eq $true) { ## Already installed/present return } ## Ensure the binaries are cached $assertEvergreenPackageDownloadParams = @{ Name = $packageName SubscriptionKey = $SubscriptionKey Force = $Force.ToBool() Retry = $Retry ErrorAction = 'Stop' } if ($PSBoundParameters.ContainsKey('CustomerCode')) { $assertEvergreenPackageDownloadParams['CustomerCode'] = $CustomerCode } if ($PSBoundParameters.ContainsKey('Credential')) { $assertEvergreenPackageDownloadParams['Credential'] = $Credential } $package = Assert-EvergreenPackageDownload @assertEvergreenPackageDownloadParams ## Start the installation $customArguments = @() if ($null -eq $package.TempPath) { ## We have a self-extracting .EXE $customArguments += '-y' } $customArguments += '-DeploymentType','Install' if ($PSBoundParameters.ContainsKey('Arguments')) { $customArguments += $Arguments } $startWaitProcessParams = @{ FilePath = $package.Path ArgumentList = $customArguments } Write-Verbose -Message ($localized.InstallingPackage -f $packageName) $null = Start-WaitProcess @startWaitProcessParams if ($Verify) { $assertEvergreenPackageCacheMetadataParams['IgnoreEmptyDetectionRule'] = $IgnoreEmptyDetectionRule.ToBool() $isEvergreenPackageInstalled = Test-EvergreenPackage @assertEvergreenPackageCacheMetadataParams if ($isEvergreenPackageInstalled -eq $false) { throw ($localized.PackageIsNotInstalled -f $packageName) } } } catch { Write-Error $_ } finally { ## Clear up temp files, folders and downloads if ($null -ne $package) { if (($null -ne $package.TempPath) -and (Test-Path -Path $package.TempPath)) { Write-Verbose -Message ($localized.RemoveTemporaryDirectory -f $package.TempPath) Remove-Item -Path $package.TempPath -Force -Recurse -WhatIf:$false } } } } } } <# .SYNOPSIS Saves Virtual Engine Evergreen packages. .DESCRIPTION Downloads Virtual Engine evergreen packages to the local file system. .EXAMPLE Save-EvergreenPackage -Name 7-Zip-x64 -SubscriptionKey abcdef1234567890 Directory: C:\Users\User\Documents Mode LastWriteTime Length Name ---- ------------- ------ ---- -a---- 20/05/2020 12:56 2095648 7-Zip-x64.19.0.intunewin -a---- 20/05/2020 12:56 17519 7-Zip-x64.19.0.intunewin.metadata -a---- 20/05/2020 12:56 2056278 7-Zip-x64.19.0.zip Saves all available formats of the latest 7-Zip-x64 packages to the current path. .EXAMPLE Save-EvergreenPackage -Name 7-Zip-x64,KeePass -SubscriptionKey abcdef1234567890 -Platform Zip -Path ~\Downloads Directory: C:\Users\User\Downloads Mode LastWriteTime Length Name ---- ------------- ------ ---- -a---- 20/05/2020 13:41 2056278 7-Zip-x64.19.0.zip -a---- 20/05/2020 13:41 3533503 KeePass.2.45.zip Saves the latest 7-Zip-x64 and KeePass archive packages to the user's Downloads directory. #> function Save-EvergreenPackage { [CmdletBinding()] [OutputType([System.IO.FileInfo])] param ( ## Evergreen package name. [Parameter(Mandatory, ValueFromPipelineByPropertyName)] [System.String[]] $Name, ## Evergreen Api subscription key. [Parameter(Mandatory, ValueFromPipelineByPropertyName)] [System.String] $SubscriptionKey, ## Enumerate customer package repository. [Parameter(ValueFromPipelineByPropertyName)] [System.String] $CustomerCode, ## Evergreen package format. [Parameter(ValueFromPipelineByPropertyName)] [ValidateSet('Exe', 'Zip', 'Intune')] [System.String] $Platform, ## Return all available package versions. [Parameter(ValueFromPipelineByPropertyName)] [System.Management.Automation.SwitchParameter] $All, ## File path to save the packages. [Parameter(ValueFromPipelineByPropertyName)] [System.String] $Path = (Get-Location -PSProvider Filesystem).Path, ## Web proxy credential required to access Virtual Engine Evergreen Api. [Parameter(ValueFromPipelineByPropertyName)] [System.Management.Automation.PSCredential] [System.Management.Automation.CredentialAttribute()] $Credential, ## Overwrite any existing file(s). [Parameter(ValueFromPipelineByPropertyName)] [System.Management.Automation.SwitchParameter] $Force, ## Download retry attempts. [Parameter(ValueFromPipelineByPropertyName)] [System.Uint32] $Retry = 3, ## Save package metadata. [Parameter(ValueFromPipelineByPropertyName)] [System.Management.Automation.SwitchParameter] $Metadata ) begin { if (-not (Test-Path -Path $Path -PathType Container)) { throw ($localized.InvalidDestinationPathError -f $Path) } } process { $null = $PSBoundParameters.Remove('Path') $null = $PSBoundParameters.Remove('Force') $null = $PSBoundParameters.Remove('Metadata') $null = $PSBoundParameters.Remove('Retry') $availablePackages = Find-EvergreenPackage @PSBoundParameters foreach ($package in $availablePackages) { if ($Metadata) { $destinationPath = Join-Path -Path $Path -ChildPath ($package.MetadataUri -as [System.Uri]).AbsolutePath.Split('/')[-1] } else { $destinationPath = Join-Path -Path $Path -ChildPath ($package.Uri -as [System.Uri]).AbsolutePath.Split('/')[-1] } if (-not $Force -and (Test-Path -Path $destinationPath)) { Write-Verbose -Message ($localized.UsingCachedPackage -f $destinationPath) Get-Item -Path $destinationPath } else { if (-not $Metadata) { ## Download primary package $uri = $package.Uri -as [System.Uri] $filename = $uri.AbsolutePath.Split('/')[-1] Write-Verbose -Message ($localized.DownloadingActivity -f $filename) $invokeWebClientDownloadParams = @{ Uri = $package.Uri DestinationPath = $destinationPath Retry = $Retry } if ($PSBoundParameters.ContainsKey('Credential')) { $invokeWebClientDownloadParams['Credential'] = $Credential } Invoke-WebClientDownload @invokeWebClientDownloadParams } if ($Metadata -or ($Platform -eq 'Intune')) { $uri = $package.MetadataUri -as [System.Uri] $filename = $uri.AbsolutePath.Split('/')[-1] $destinationPath = Join-Path -Path $Path -ChildPath $filename if (-not $Force -and (Test-Path -Path $destinationPath)) { Write-Verbose -Message ($localized.UsingCachedPackage -f $destinationPath) Get-Item -Path $destinationPath } else { Write-Verbose -Message ($localized.DownloadingActivity -f $filename) $invokeWebClientDownloadParams = @{ Uri = $package.MetadataUri DestinationPath = $destinationPath Retry = $Retry } if ($PSBoundParameters.ContainsKey('Credential')) { $invokeWebClientDownloadParams['Credential'] = $Credential } Invoke-WebClientDownload @invokeWebClientDownloadParams } } } } } } <# .SYNOPSIS Tests whether one or more Virtual Engine Evergreen packages are installed. .DESCRIPTION Downloads Virtual Engine Evergreen package metadata to the local file system to test whether the package is installed (or not). .EXAMPLE Test-EvergreenPackage -Name 7-Zip-x64 -SubscriptionKey abcdef1234567890 False Downloads and latest '7-Zip-x64' Evergreen package metadata and tests wheter the package is installed. #> function Test-EvergreenPackage { [CmdletBinding(DefaultParameterSetName = 'Default')] [OutputType([System.Boolean])] param ( ## Evergreen package name. [Parameter(Mandatory, ValueFromPipelineByPropertyName, ParameterSetName = 'Default')] [System.String[]] $Name, ## Evergreen Api subscription key. [Parameter(Mandatory, ValueFromPipelineByPropertyName, ParameterSetName = 'Default')] [System.String] $SubscriptionKey, ## Enumerate customer package repository. [Parameter(ValueFromPipelineByPropertyName, ParameterSetName = 'Default')] [System.String] $CustomerCode, ## Evergreen package metadata file path. [Parameter(Mandatory, ValueFromPipeline, ParameterSetName = 'Metadata')] [Microsoft.PowerShell.DesiredStateConfiguration.ArgumentToConfigurationDataTransformationAttribute()] [System.Collections.Hashtable] $Metadata, ## Test all applications in the local cache. [Parameter(Mandatory, ValueFromPipeline, ParameterSetName = 'Cached')] [System.Management.Automation.SwitchParameter] $Cached, ## Web proxy credential required to access Virtual Engine Evergreen Api. [Parameter(ValueFromPipelineByPropertyName, ParameterSetName = 'Default')] [System.Management.Automation.PSCredential] [System.Management.Automation.CredentialAttribute()] $Credential, ## Does not fail application test if no detection rules are defined. [Parameter(ValueFromPipelineByPropertyName)] [System.Management.Automation.SwitchParameter] $IgnoreEmptyDetectionRule, ## Force refresh of cached package metadata. [Parameter(ValueFromPipelineByPropertyName, ParameterSetName = 'Default')] [System.Management.Automation.SwitchParameter] $Force ) process { $metadataCollection = @() if ($PSCmdlet.ParameterSetName -eq 'Cached') { $cachedMetadataItems = Get-ChildItem -Path $cachedMetadataPath -Filter '*.psd1' foreach ($cachedMetadataItem in $cachedMetadataItems) { $metadataCollection += Import-PowerShellDataFile -Path $cachedMetadataItem.FullName } } elseif ($PSCmdlet.ParameterSetName -eq 'Default') { foreach ($packageName in $Name) { $getEvergreenPackageCacheMetadataParams = @{ Name = $packageName SubscriptionKey = $SubscriptionKey Force = $Force.ToBool() } if ($PSBoundParameters.ContainsKey('CustomerCode')) { $getEvergreenPackageCacheMetadataParams['CustomerCode'] = $CustomerCode } if ($PSBoundParameters.ContainsKey('Credential')) { $getEvergreenPackageCacheMetadataParams['Credential'] = $Credential } $metadataCollection += Get-EvergreenPackageCacheMetadata @getEvergreenPackageCacheMetadataParams } } else { $metadataCollection += $Metadata } $isCompliant = $true foreach ($metadataItem in $metadataCollection) { $packageName = Get-PackageMetadataPackageIdentifier -Metadata $metadataItem Write-Verbose -Message ($localized.TestingPackage -f $packageName) $testPackageDetectionRuleParams = @{ Metadata = $metadataItem PackageName = $packageName IgnoreEmptyDetectionRule = $IgnoreEmptyDetectionRule.ToBool() } $isInstalled = Test-PackageDetectionRule @testPackageDetectionRuleParams if (($null -eq $isInstalled) -and ($IgnoreEmptyDetectionRule -eq $true)) { Write-Warning -Message ($localized.UnableToDetermineInstallStateWarning -f $packageName) $isCompliant = $null } elseif ($isInstalled -eq $true) { Write-Verbose -Message ($localized.PackageIsInstalled -f $packageName) } else { Write-Verbose -Message ($localized.PackageIsNotInstalled -f $packageName) $isCompliant = $false } } return $isCompliant } } <# .SYNOPSIS Uninstalls one or more Virtual Engine Evergreen package(s). .DESCRIPTION Downloads Virtual Engine Evergreen packages to the local file system and intitiates the uninstall. .EXAMPLE Uninstall-EvergreenPackage -Name 7-Zip-x64 -SubscriptionKey abcdef1234567890 Downloads and uninstalls the '7-Zip-x64' package. .EXAMPLE Uninstall-EvergreenPackage -Name 7-Zip-x64 -SubscriptionKey abcdef1234567890 -Verify Uninstalls the cached '7-Zip-x64' Evergreen package and tests the installation is no longer detected successfully. Throws an error when no Intune detection rule has been defined or defined Intune detection rule(s) pass. .EXAMPLE Uninstall-EvergreenPackage -Name 7-Zip-x64 -SubscriptionKey abcdef1234567890 -Verify -IgnoreEmptyDetectionRule Uninstalls the cached '7-Zip-x64' Evergreen package and tests the installation is no longer detected successfully. Does not throw an error when no Intune detection rule has been defined. #> function Uninstall-EvergreenPackage { [CmdletBinding(SupportsShouldProcess, DefaultParameterSetName = 'Default')] param ( ## Evergreen package(s) to uninstall. [Parameter(Mandatory, ValueFromPipelineByPropertyName)] [System.String[]] $Name, ## Evergreen Api subscription key. [Parameter(Mandatory, ValueFromPipelineByPropertyName)] [System.String] $SubscriptionKey, ## Run the application's detection rules after installation to verify the application is detected as installed. [Parameter(Mandatory, ValueFromPipelineByPropertyName, ParameterSetName = 'Verify')] [System.Management.Automation.SwitchParameter] $Verify, ## Enumerate customer package repository. [Parameter(ValueFromPipelineByPropertyName)] [System.String] $CustomerCode, ## Custom package installation arguments. [Parameter(ValueFromPipelineByPropertyName)] [System.String] $Arguments, ## Web proxy credential required to access Virtual Engine Evergreen Api. [Parameter(ValueFromPipelineByPropertyName)] [System.Management.Automation.PSCredential] [System.Management.Automation.CredentialAttribute()] $Credential, ## Force refresh of cached package metadata. [Parameter(ValueFromPipelineByPropertyName)] [System.Management.Automation.SwitchParameter] $Force, ## Does not fail application verification if no detection rules are defined. [Parameter(ValueFromPipelineByPropertyName, ParameterSetName = 'Verify')] [System.Management.Automation.SwitchParameter] $IgnoreEmptyDetectionRule, ## Download retry attempts. [Parameter(ValueFromPipelineByPropertyName)] [System.Uint32] $Retry = 3, ## Cached files are removed after uninstall if the process exit code matches/is successful. [Parameter(ValueFromPipelineByPropertyName)] [System.UInt32[]] $ValidExitCodes = @(0, 3010) ) process { foreach ($packageName in $Name) { try { ## Cache metadata file to ensure any future Test-EvergreenPackage uses the ## detection rules used when initially installing the package $assertEvergreenPackageCacheMetadataParams = @{ Name = $packageName SubscriptionKey = $SubscriptionKey Force = $Force.ToBool() } if ($PSBoundParameters.ContainsKey('CustomerCode')) { $assertEvergreenPackageCacheMetadataParams['CustomerCode'] = $CustomerCode } if ($PSBoundParameters.ContainsKey('Credential')) { $assertEvergreenPackageCacheMetadataParams['Credential'] = $Credential } Assert-EvergreenPackageCacheMetadata @assertEvergreenPackageCacheMetadataParams ## Test to see if the package is already installed (assuming detection rules are defined) $isEvergreenPackageInstalled = Test-EvergreenPackage @assertEvergreenPackageCacheMetadataParams -IgnoreEmptyDetectionRule if ($null -eq $isEvergreenPackageInstalled) { ## Unable to determine installation state if ($Verify -and (-not $IgnoreEmptyDetectionRule)) { ## We have a package with no detection rules and we are not ignoring them throw ($localized.CannotFindPackageDetectionRuleError -f $packageName) } } elseif ($isEvergreenPackageInstalled -eq $false) { ## Already uninstalled/not present return } ## Ensure the binaries are cached $assertEvergreenPackageDownloadParams = @{ Name = $packageName SubscriptionKey = $SubscriptionKey Force = $Force.ToBool() Retry = $Retry ErrorAction = 'Stop' } if ($PSBoundParameters.ContainsKey('CustomerCode')) { $assertEvergreenPackageDownloadParams['CustomerCode'] = $CustomerCode } if ($PSBoundParameters.ContainsKey('Credential')) { $assertEvergreenPackageDownloadParams['Credential'] = $Credential } $package = Assert-EvergreenPackageDownload @assertEvergreenPackageDownloadParams ## Start the uninstallation $customArguments = @() if ($null -eq $package.TempPath) { ## We have a self-extracting .EXE $customArguments += '-y' } $customArguments += '-DeploymentType','Uninstall' if ($PSBoundParameters.ContainsKey('Arguments')) { $customArguments += $Arguments } $startWaitProcessParams = @{ FilePath = $package.Path ArgumentList = $customArguments } Write-Verbose -Message ($localized.UninstallingPackage -f $package.Name) $exitCode = Start-WaitProcess @startWaitProcessParams if ($Verify) { $assertEvergreenPackageCacheMetadataParams['IgnoreEmptyDetectionRule'] = $IgnoreEmptyDetectionRule.ToBool() $isEvergreenPackageInstalled = Test-EvergreenPackage @assertEvergreenPackageCacheMetadataParams if ($isEvergreenPackageInstalled -eq $true) { throw ($localized.PackageIsInstalled -f $packageName) } } if ($exitCode -in $ValidExitCodes) { ## Remove any cached metadata files to ensure any future Test-EvergreenPackage ## detection rules are downloaded again $assertEvergreenPackageCacheMetadataParams = @{ Name = $packageName SubscriptionKey = $SubscriptionKey Force = $true NotPresent = $true } Assert-EvergreenPackageCacheMetadata @assertEvergreenPackageCacheMetadataParams } } catch { Write-Error $_ } finally { ## Clear up temp files, folders and downloads if ($null -ne $package) { if (($null -ne $package.TempPath) -and (Test-Path -Path $package.TempPath)) { Write-Verbose -Message ($localized.RemoveTemporaryDirectory -f $package.TempPath) Remove-Item -Path $package.TempPath -Force -Recurse -WhatIf:$false } } } } } } # SIG # Begin signature block # MIIuwAYJKoZIhvcNAQcCoIIusTCCLq0CAQExDzANBglghkgBZQMEAgEFADB5Bgor # BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCC//DhQqqjgDqPu # ns6d7La7Lq+TCzpCVGkNQqbBeRM2j6CCE6QwggWQMIIDeKADAgECAhAFmxtXno4h # MuI5B72nd3VcMA0GCSqGSIb3DQEBDAUAMGIxCzAJBgNVBAYTAlVTMRUwEwYDVQQK # EwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xITAfBgNV # BAMTGERpZ2lDZXJ0IFRydXN0ZWQgUm9vdCBHNDAeFw0xMzA4MDExMjAwMDBaFw0z # ODAxMTUxMjAwMDBaMGIxCzAJBgNVBAYTAlVTMRUwEwYDVQQKEwxEaWdpQ2VydCBJ # bmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xITAfBgNVBAMTGERpZ2lDZXJ0 # IFRydXN0ZWQgUm9vdCBHNDCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIB # AL/mkHNo3rvkXUo8MCIwaTPswqclLskhPfKK2FnC4SmnPVirdprNrnsbhA3EMB/z # G6Q4FutWxpdtHauyefLKEdLkX9YFPFIPUh/GnhWlfr6fqVcWWVVyr2iTcMKyunWZ # anMylNEQRBAu34LzB4TmdDttceItDBvuINXJIB1jKS3O7F5OyJP4IWGbNOsFxl7s # Wxq868nPzaw0QF+xembud8hIqGZXV59UWI4MK7dPpzDZVu7Ke13jrclPXuU15zHL # 2pNe3I6PgNq2kZhAkHnDeMe2scS1ahg4AxCN2NQ3pC4FfYj1gj4QkXCrVYJBMtfb # BHMqbpEBfCFM1LyuGwN1XXhm2ToxRJozQL8I11pJpMLmqaBn3aQnvKFPObURWBf3 # JFxGj2T3wWmIdph2PVldQnaHiZdpekjw4KISG2aadMreSx7nDmOu5tTvkpI6nj3c # AORFJYm2mkQZK37AlLTSYW3rM9nF30sEAMx9HJXDj/chsrIRt7t/8tWMcCxBYKqx # YxhElRp2Yn72gLD76GSmM9GJB+G9t+ZDpBi4pncB4Q+UDCEdslQpJYls5Q5SUUd0 # viastkF13nqsX40/ybzTQRESW+UQUOsxxcpyFiIJ33xMdT9j7CFfxCBRa2+xq4aL # T8LWRV+dIPyhHsXAj6KxfgommfXkaS+YHS312amyHeUbAgMBAAGjQjBAMA8GA1Ud # EwEB/wQFMAMBAf8wDgYDVR0PAQH/BAQDAgGGMB0GA1UdDgQWBBTs1+OC0nFdZEzf # Lmc/57qYrhwPTzANBgkqhkiG9w0BAQwFAAOCAgEAu2HZfalsvhfEkRvDoaIAjeNk # aA9Wz3eucPn9mkqZucl4XAwMX+TmFClWCzZJXURj4K2clhhmGyMNPXnpbWvWVPjS # PMFDQK4dUPVS/JA7u5iZaWvHwaeoaKQn3J35J64whbn2Z006Po9ZOSJTROvIXQPK # 7VB6fWIhCoDIc2bRoAVgX+iltKevqPdtNZx8WorWojiZ83iL9E3SIAveBO6Mm0eB # cg3AFDLvMFkuruBx8lbkapdvklBtlo1oepqyNhR6BvIkuQkRUNcIsbiJeoQjYUIp # 5aPNoiBB19GcZNnqJqGLFNdMGbJQQXE9P01wI4YMStyB0swylIQNCAmXHE/A7msg # dDDS4Dk0EIUhFQEI6FUy3nFJ2SgXUE3mvk3RdazQyvtBuEOlqtPDBURPLDab4vri # RbgjU2wGb2dVf0a1TD9uKFp5JtKkqGKX0h7i7UqLvBv9R0oN32dmfrJbQdA75PQ7 # 9ARj6e/CVABRoIoqyc54zNXqhwQYs86vSYiv85KZtrPmYQ/ShQDnUBrkG5WdGaG5 # nLGbsQAe79APT0JsyQq87kP6OnGlyE0mpTX9iV28hWIdMtKgK1TtmlfB2/oQzxm3 # i0objwG2J5VT6LaJbVu8aNQj6ItRolb58KaAoNYes7wPD1N1KarqE3fk3oyBIa0H # EEcRrYc9B9F1vM/zZn4wggawMIIEmKADAgECAhAIrUCyYNKcTJ9ezam9k67ZMA0G # CSqGSIb3DQEBDAUAMGIxCzAJBgNVBAYTAlVTMRUwEwYDVQQKEwxEaWdpQ2VydCBJ # bmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xITAfBgNVBAMTGERpZ2lDZXJ0 # IFRydXN0ZWQgUm9vdCBHNDAeFw0yMTA0MjkwMDAwMDBaFw0zNjA0MjgyMzU5NTla # MGkxCzAJBgNVBAYTAlVTMRcwFQYDVQQKEw5EaWdpQ2VydCwgSW5jLjFBMD8GA1UE # AxM4RGlnaUNlcnQgVHJ1c3RlZCBHNCBDb2RlIFNpZ25pbmcgUlNBNDA5NiBTSEEz # ODQgMjAyMSBDQTEwggIiMA0GCSqGSIb3DQEBAQUAA4ICDwAwggIKAoICAQDVtC9C # 0CiteLdd1TlZG7GIQvUzjOs9gZdwxbvEhSYwn6SOaNhc9es0JAfhS0/TeEP0F9ce # 2vnS1WcaUk8OoVf8iJnBkcyBAz5NcCRks43iCH00fUyAVxJrQ5qZ8sU7H/Lvy0da # E6ZMswEgJfMQ04uy+wjwiuCdCcBlp/qYgEk1hz1RGeiQIXhFLqGfLOEYwhrMxe6T # SXBCMo/7xuoc82VokaJNTIIRSFJo3hC9FFdd6BgTZcV/sk+FLEikVoQ11vkunKoA # FdE3/hoGlMJ8yOobMubKwvSnowMOdKWvObarYBLj6Na59zHh3K3kGKDYwSNHR7Oh # D26jq22YBoMbt2pnLdK9RBqSEIGPsDsJ18ebMlrC/2pgVItJwZPt4bRc4G/rJvmM # 1bL5OBDm6s6R9b7T+2+TYTRcvJNFKIM2KmYoX7BzzosmJQayg9Rc9hUZTO1i4F4z # 8ujo7AqnsAMrkbI2eb73rQgedaZlzLvjSFDzd5Ea/ttQokbIYViY9XwCFjyDKK05 # huzUtw1T0PhH5nUwjewwk3YUpltLXXRhTT8SkXbev1jLchApQfDVxW0mdmgRQRNY # mtwmKwH0iU1Z23jPgUo+QEdfyYFQc4UQIyFZYIpkVMHMIRroOBl8ZhzNeDhFMJlP # /2NPTLuqDQhTQXxYPUez+rbsjDIJAsxsPAxWEQIDAQABo4IBWTCCAVUwEgYDVR0T # AQH/BAgwBgEB/wIBADAdBgNVHQ4EFgQUaDfg67Y7+F8Rhvv+YXsIiGX0TkIwHwYD # VR0jBBgwFoAU7NfjgtJxXWRM3y5nP+e6mK4cD08wDgYDVR0PAQH/BAQDAgGGMBMG # A1UdJQQMMAoGCCsGAQUFBwMDMHcGCCsGAQUFBwEBBGswaTAkBggrBgEFBQcwAYYY # aHR0cDovL29jc3AuZGlnaWNlcnQuY29tMEEGCCsGAQUFBzAChjVodHRwOi8vY2Fj # ZXJ0cy5kaWdpY2VydC5jb20vRGlnaUNlcnRUcnVzdGVkUm9vdEc0LmNydDBDBgNV # HR8EPDA6MDigNqA0hjJodHRwOi8vY3JsMy5kaWdpY2VydC5jb20vRGlnaUNlcnRU # cnVzdGVkUm9vdEc0LmNybDAcBgNVHSAEFTATMAcGBWeBDAEDMAgGBmeBDAEEATAN # BgkqhkiG9w0BAQwFAAOCAgEAOiNEPY0Idu6PvDqZ01bgAhql+Eg08yy25nRm95Ry # sQDKr2wwJxMSnpBEn0v9nqN8JtU3vDpdSG2V1T9J9Ce7FoFFUP2cvbaF4HZ+N3HL # IvdaqpDP9ZNq4+sg0dVQeYiaiorBtr2hSBh+3NiAGhEZGM1hmYFW9snjdufE5Btf # Q/g+lP92OT2e1JnPSt0o618moZVYSNUa/tcnP/2Q0XaG3RywYFzzDaju4ImhvTnh # OE7abrs2nfvlIVNaw8rpavGiPttDuDPITzgUkpn13c5UbdldAhQfQDN8A+KVssIh # dXNSy0bYxDQcoqVLjc1vdjcshT8azibpGL6QB7BDf5WIIIJw8MzK7/0pNVwfiThV # 9zeKiwmhywvpMRr/LhlcOXHhvpynCgbWJme3kuZOX956rEnPLqR0kq3bPKSchh/j # wVYbKyP/j7XqiHtwa+aguv06P0WmxOgWkVKLQcBIhEuWTatEQOON8BUozu3xGFYH # Ki8QxAwIZDwzj64ojDzLj4gLDb879M4ee47vtevLt/B3E+bnKD+sEq6lLyJsQfmC # XBVmzGwOysWGw/YmMwwHS6DTBwJqakAwSEs0qFEgu60bhQjiWQ1tygVQK+pKHJ6l # /aCnHwZ05/LWUpD9r4VIIflXO7ScA+2GRfS0YW6/aOImYIbqyK+p/pQd52MbOoZW # eE4wggdYMIIFQKADAgECAhAIfHT3o/FeY5ksO94AUhTmMA0GCSqGSIb3DQEBCwUA # MGkxCzAJBgNVBAYTAlVTMRcwFQYDVQQKEw5EaWdpQ2VydCwgSW5jLjFBMD8GA1UE # AxM4RGlnaUNlcnQgVHJ1c3RlZCBHNCBDb2RlIFNpZ25pbmcgUlNBNDA5NiBTSEEz # ODQgMjAyMSBDQTEwHhcNMjMxMDE4MDAwMDAwWhcNMjYxMjE2MjM1OTU5WjBgMQsw # CQYDVQQGEwJHQjEPMA0GA1UEBxMGTG9uZG9uMR8wHQYDVQQKExZWaXJ0dWFsIEVu # Z2luZSBMaW1pdGVkMR8wHQYDVQQDExZWaXJ0dWFsIEVuZ2luZSBMaW1pdGVkMIIC # IjANBgkqhkiG9w0BAQEFAAOCAg8AMIICCgKCAgEAtyhrsCMi6pgLcX5sWY7I09dO # WKweRHfDwW5AN6ffgLCYO9dqWWxvqu95FqnNVRyt1VNzEl3TevKVhRE0GGdirei3 # VqnFFjLDwD2jHhGY8qoSYyfffj/WYq2DkvNI62C3gUwSeP3FeqKRalc2c3V2v4jh # yEYhrgG3nfnWQ/Oq2xzuiCqHy1E4U+IKKDtrXls4JX2Z4J/uAHZIAyKfrcTRQOhZ # R4ZS1cQkeSBU9Urx578rOmxL0si0GAoaYQC49W7OimRelbahxZw/R+f5ch+C1ycU # CpeXLg+bFhpa0+EXnkGidlILZbJiZJn7qvMQTZgipQKZ8nhX3rtJLqTeodPWzcCk # tXQUy0q5fxhR3e6Ls7XQesq/G2yMcCMTCd6eltze0OgabvL6Xkhir5lATHaJtnmw # FlcKzRr1YXK1k1D84hWfKSAdUg8T1O72ztimbqFLg6WoC8M2qqsHcm2DOc9hM3i2 # CWRpegikitRvZ9i1wkA9arGh7+a7UD+nLh2hnGmO06wONLNqABOEn4JOVnNrQ1gY # eDeH9FDx7IYuAvMsfXG9Bo+I97TR2VfwDAx+ccR+UQLON3aQyFZ3BefYnvUu0gUR # ikEAnAS4Jnc3BHizgb0voz0iWRDjFoTTmCmrInCVDGc+5KMy0xyoUwdQvYvRGAWB # 61OCWnXBXbAEPniTZ80CAwEAAaOCAgMwggH/MB8GA1UdIwQYMBaAFGg34Ou2O/hf # EYb7/mF7CIhl9E5CMB0GA1UdDgQWBBRuAv58K4EDYLmb7WNcxt5+r4NfnzA+BgNV # HSAENzA1MDMGBmeBDAEEATApMCcGCCsGAQUFBwIBFhtodHRwOi8vd3d3LmRpZ2lj # ZXJ0LmNvbS9DUFMwDgYDVR0PAQH/BAQDAgeAMBMGA1UdJQQMMAoGCCsGAQUFBwMD # MIG1BgNVHR8Ega0wgaowU6BRoE+GTWh0dHA6Ly9jcmwzLmRpZ2ljZXJ0LmNvbS9E # aWdpQ2VydFRydXN0ZWRHNENvZGVTaWduaW5nUlNBNDA5NlNIQTM4NDIwMjFDQTEu # Y3JsMFOgUaBPhk1odHRwOi8vY3JsNC5kaWdpY2VydC5jb20vRGlnaUNlcnRUcnVz # dGVkRzRDb2RlU2lnbmluZ1JTQTQwOTZTSEEzODQyMDIxQ0ExLmNybDCBlAYIKwYB # BQUHAQEEgYcwgYQwJAYIKwYBBQUHMAGGGGh0dHA6Ly9vY3NwLmRpZ2ljZXJ0LmNv # bTBcBggrBgEFBQcwAoZQaHR0cDovL2NhY2VydHMuZGlnaWNlcnQuY29tL0RpZ2lD # ZXJ0VHJ1c3RlZEc0Q29kZVNpZ25pbmdSU0E0MDk2U0hBMzg0MjAyMUNBMS5jcnQw # CQYDVR0TBAIwADANBgkqhkiG9w0BAQsFAAOCAgEAnXMg6efkBrwLIvd1Xmuh0dam # 9FhUtDEj+P5SIqdP/U4veOv66NEQhBHLbW2Dvrdm6ec0HMj9b4e8pt4ylKFzHIPj # fpuRffHVR9JQSx8qpryN6pP49DfCkAYeZGqjY3pGRzd/xQ0cfwcuYbUF+vwVk7tj # q8c93VHCM0rb5M4N2hD1Ze1pvZxtaf9QnFKFzgXZjr02K6bswQc2+n5jFCp7zV1f # KTstyb68rhSJBWKK1tMeFk6a6HXr5buTD3skluC0oyPmD7yAd97r2owjDMEveEso # kADP/z7XQk7wqbwbpi4W6Uju2qHK/9UUsVRF5KTVEAIzVw2V1Aq/Jh3JuSV7b7C1 # 4CghNekltBb+w7YVp8/IFcj7axqnpNQ/+f7RVc3A5hyjV+MkoSwn8Sg7a7hn6SzX # jec/TfRVvWCmG94MQHko+6206uIXrZnmQ6UQYFyOHRlyKDEozzkZhIcVlsZloUjL # 3FZ5V/l8TIIzbc3bkEnu4iByksNvRxI6c5264OLauYlWv50ZUPwXmZ9gX8bs3BqZ # avbGUrOW2PIjtWhtvs4zHhMBCoU1Z0OMvXcF9rUDqefmVCZK46xz3DGKVkDQnxY6 # UWQ3GL60/lEzju4YS99LJVQks2UGmP6LAzlEZ1dnGqi1aQ51OidCYEs39B75PsvO # By2iAR8pBVi/byWBypExghpyMIIabgIBATB9MGkxCzAJBgNVBAYTAlVTMRcwFQYD # VQQKEw5EaWdpQ2VydCwgSW5jLjFBMD8GA1UEAxM4RGlnaUNlcnQgVHJ1c3RlZCBH # NCBDb2RlIFNpZ25pbmcgUlNBNDA5NiBTSEEzODQgMjAyMSBDQTECEAh8dPej8V5j # mSw73gBSFOYwDQYJYIZIAWUDBAIBBQCggYQwGAYKKwYBBAGCNwIBDDEKMAigAoAA # oQKAADAZBgkqhkiG9w0BCQMxDAYKKwYBBAGCNwIBBDAcBgorBgEEAYI3AgELMQ4w # DAYKKwYBBAGCNwIBFTAvBgkqhkiG9w0BCQQxIgQgdw3vBPJr2DL3Jk73eQ8flQx7 # GhnTFFtvnhqIp0mzyA8wDQYJKoZIhvcNAQEBBQAEggIAor3mBNcmKlzPJuBxNamk # jvNbEMM6fYraetL0ivCovOOSnwvVZBL/izM30/8U7yQcBINvlgaaPBRCXtQ9FpTb # JvlixhCH6pvpEIsz/SlTz2qbQMAfQGZLsAsz9cZtzW4Cw5QUE7fC1UdAlXYtNyR7 # RVYxvhXGRlwvHgURUlFtOl8IKnyWp17lOWvtlFCNk+ChMu/2EiLL2MwB3KWVeHkK # V66uy10vA+jF1dlYiEEgVArD7Sa0M4wR1DW5eBV6+m65iXiglTrKYOnK5aoOlJAM # bfRJlD3muUMYNc6RLQty1VIuni1d5vwdLFYzDlaYM7MrfHxGHrerPukL1JqJk6HL # WJ2duNc/GPuTvOV9zbfAPULBANwCRwLCkzaGK7fcZdeZT3+FW/IXwZWg5JozDJ7D # gwBU1eJUCw4ij2931JrIE8XH6PbvUBRwQacT5UoyQOmEimZGXS1MrFYGRZwi6JND # IT0CXMREwAit+kathfie/BZPPAV2HwFkrNihGEC+NRxG9VFZnf2NbfZj9MNj5Vvg # t7hdFsACquFB9qwhl1waef2KfQbsDlXGK8tYN80YY1P8AyOCW1dQWQPCvLmh9Sk0 # vI/dIcCAoFcJHGredUJ52MzHO7CjBNJi6gNF0620YuvFfU2yrncleUhwfiKQQE35 # DegxeIDpUqeWFGamCss/PUChghc/MIIXOwYKKwYBBAGCNwMDATGCFyswghcnBgkq # hkiG9w0BBwKgghcYMIIXFAIBAzEPMA0GCWCGSAFlAwQCAQUAMHcGCyqGSIb3DQEJ # EAEEoGgEZjBkAgEBBglghkgBhv1sBwEwMTANBglghkgBZQMEAgEFAAQgfya4Ipyw # WDPkXjy1zdal9RDp4QexQtM3AnrANVAOhisCECDcPWuLSdN6k1R4d9qIq5cYDzIw # MjMxMjA4MTcxMDU4WqCCEwkwggbCMIIEqqADAgECAhAFRK/zlJ0IOaa/2z9f5WEW # MA0GCSqGSIb3DQEBCwUAMGMxCzAJBgNVBAYTAlVTMRcwFQYDVQQKEw5EaWdpQ2Vy # dCwgSW5jLjE7MDkGA1UEAxMyRGlnaUNlcnQgVHJ1c3RlZCBHNCBSU0E0MDk2IFNI # QTI1NiBUaW1lU3RhbXBpbmcgQ0EwHhcNMjMwNzE0MDAwMDAwWhcNMzQxMDEzMjM1 # OTU5WjBIMQswCQYDVQQGEwJVUzEXMBUGA1UEChMORGlnaUNlcnQsIEluYy4xIDAe # BgNVBAMTF0RpZ2lDZXJ0IFRpbWVzdGFtcCAyMDIzMIICIjANBgkqhkiG9w0BAQEF # AAOCAg8AMIICCgKCAgEAo1NFhx2DjlusPlSzI+DPn9fl0uddoQ4J3C9Io5d6Oyqc # Z9xiFVjBqZMRp82qsmrdECmKHmJjadNYnDVxvzqX65RQjxwg6seaOy+WZuNp52n+ # W8PWKyAcwZeUtKVQgfLPywemMGjKg0La/H8JJJSkghraarrYO8pd3hkYhftF6g1h # bJ3+cV7EBpo88MUueQ8bZlLjyNY+X9pD04T10Mf2SC1eRXWWdf7dEKEbg8G45lKV # tUfXeCk5a+B4WZfjRCtK1ZXO7wgX6oJkTf8j48qG7rSkIWRw69XloNpjsy7pBe6q # 9iT1HbybHLK3X9/w7nZ9MZllR1WdSiQvrCuXvp/k/XtzPjLuUjT71Lvr1KAsNJvj # 3m5kGQc3AZEPHLVRzapMZoOIaGK7vEEbeBlt5NkP4FhB+9ixLOFRr7StFQYU6mII # E9NpHnxkTZ0P387RXoyqq1AVybPKvNfEO2hEo6U7Qv1zfe7dCv95NBB+plwKWEwA # PoVpdceDZNZ1zY8SdlalJPrXxGshuugfNJgvOuprAbD3+yqG7HtSOKmYCaFxsmxx # rz64b5bV4RAT/mFHCoz+8LbH1cfebCTwv0KCyqBxPZySkwS0aXAnDU+3tTbRyV8I # pHCj7ArxES5k4MsiK8rxKBMhSVF+BmbTO77665E42FEHypS34lCh8zrTioPLQHsC # AwEAAaOCAYswggGHMA4GA1UdDwEB/wQEAwIHgDAMBgNVHRMBAf8EAjAAMBYGA1Ud # JQEB/wQMMAoGCCsGAQUFBwMIMCAGA1UdIAQZMBcwCAYGZ4EMAQQCMAsGCWCGSAGG # /WwHATAfBgNVHSMEGDAWgBS6FtltTYUvcyl2mi91jGogj57IbzAdBgNVHQ4EFgQU # pbbvE+fvzdBkodVWqWUxo97V40kwWgYDVR0fBFMwUTBPoE2gS4ZJaHR0cDovL2Ny # bDMuZGlnaWNlcnQuY29tL0RpZ2lDZXJ0VHJ1c3RlZEc0UlNBNDA5NlNIQTI1NlRp # bWVTdGFtcGluZ0NBLmNybDCBkAYIKwYBBQUHAQEEgYMwgYAwJAYIKwYBBQUHMAGG # GGh0dHA6Ly9vY3NwLmRpZ2ljZXJ0LmNvbTBYBggrBgEFBQcwAoZMaHR0cDovL2Nh # Y2VydHMuZGlnaWNlcnQuY29tL0RpZ2lDZXJ0VHJ1c3RlZEc0UlNBNDA5NlNIQTI1 # NlRpbWVTdGFtcGluZ0NBLmNydDANBgkqhkiG9w0BAQsFAAOCAgEAgRrW3qCptZgX # vHCNT4o8aJzYJf/LLOTN6l0ikuyMIgKpuM+AqNnn48XtJoKKcS8Y3U623mzX4WCc # K+3tPUiOuGu6fF29wmE3aEl3o+uQqhLXJ4Xzjh6S2sJAOJ9dyKAuJXglnSoFeoQp # mLZXeY/bJlYrsPOnvTcM2Jh2T1a5UsK2nTipgedtQVyMadG5K8TGe8+c+njikxp2 # oml101DkRBK+IA2eqUTQ+OVJdwhaIcW0z5iVGlS6ubzBaRm6zxbygzc0brBBJt3e # WpdPM43UjXd9dUWhpVgmagNF3tlQtVCMr1a9TMXhRsUo063nQwBw3syYnhmJA+rU # kTfvTVLzyWAhxFZH7doRS4wyw4jmWOK22z75X7BC1o/jF5HRqsBV44a/rCcsQdCa # M0qoNtS5cpZ+l3k4SF/Kwtw9Mt911jZnWon49qfH5U81PAC9vpwqbHkB3NpE5jre # ODsHXjlY9HxzMVWggBHLFAx+rrz+pOt5Zapo1iLKO+uagjVXKBbLafIymrLS2Dq4 # sUaGa7oX/cR3bBVsrquvczroSUa31X/MtjjA2Owc9bahuEMs305MfR5ocMB3CtQC # 4Fxguyj/OOVSWtasFyIjTvTs0xf7UGv/B3cfcZdEQcm4RtNsMnxYL2dHZeUbc7aZ # +WssBkbvQR7w8F/g29mtkIBEr4AQQYowggauMIIElqADAgECAhAHNje3JFR82Ees # /ShmKl5bMA0GCSqGSIb3DQEBCwUAMGIxCzAJBgNVBAYTAlVTMRUwEwYDVQQKEwxE # aWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xITAfBgNVBAMT # GERpZ2lDZXJ0IFRydXN0ZWQgUm9vdCBHNDAeFw0yMjAzMjMwMDAwMDBaFw0zNzAz # MjIyMzU5NTlaMGMxCzAJBgNVBAYTAlVTMRcwFQYDVQQKEw5EaWdpQ2VydCwgSW5j # LjE7MDkGA1UEAxMyRGlnaUNlcnQgVHJ1c3RlZCBHNCBSU0E0MDk2IFNIQTI1NiBU # aW1lU3RhbXBpbmcgQ0EwggIiMA0GCSqGSIb3DQEBAQUAA4ICDwAwggIKAoICAQDG # hjUGSbPBPXJJUVXHJQPE8pE3qZdRodbSg9GeTKJtoLDMg/la9hGhRBVCX6SI82j6 # ffOciQt/nR+eDzMfUBMLJnOWbfhXqAJ9/UO0hNoR8XOxs+4rgISKIhjf69o9xBd/ # qxkrPkLcZ47qUT3w1lbU5ygt69OxtXXnHwZljZQp09nsad/ZkIdGAHvbREGJ3Hxq # V3rwN3mfXazL6IRktFLydkf3YYMZ3V+0VAshaG43IbtArF+y3kp9zvU5EmfvDqVj # bOSmxR3NNg1c1eYbqMFkdECnwHLFuk4fsbVYTXn+149zk6wsOeKlSNbwsDETqVcp # licu9Yemj052FVUmcJgmf6AaRyBD40NjgHt1biclkJg6OBGz9vae5jtb7IHeIhTZ # girHkr+g3uM+onP65x9abJTyUpURK1h0QCirc0PO30qhHGs4xSnzyqqWc0Jon7ZG # s506o9UD4L/wojzKQtwYSH8UNM/STKvvmz3+DrhkKvp1KCRB7UK/BZxmSVJQ9FHz # NklNiyDSLFc1eSuo80VgvCONWPfcYd6T/jnA+bIwpUzX6ZhKWD7TA4j+s4/TXkt2 # ElGTyYwMO1uKIqjBJgj5FBASA31fI7tk42PgpuE+9sJ0sj8eCXbsq11GdeJgo1gJ # ASgADoRU7s7pXcheMBK9Rp6103a50g5rmQzSM7TNsQIDAQABo4IBXTCCAVkwEgYD # VR0TAQH/BAgwBgEB/wIBADAdBgNVHQ4EFgQUuhbZbU2FL3MpdpovdYxqII+eyG8w # HwYDVR0jBBgwFoAU7NfjgtJxXWRM3y5nP+e6mK4cD08wDgYDVR0PAQH/BAQDAgGG # MBMGA1UdJQQMMAoGCCsGAQUFBwMIMHcGCCsGAQUFBwEBBGswaTAkBggrBgEFBQcw # AYYYaHR0cDovL29jc3AuZGlnaWNlcnQuY29tMEEGCCsGAQUFBzAChjVodHRwOi8v # Y2FjZXJ0cy5kaWdpY2VydC5jb20vRGlnaUNlcnRUcnVzdGVkUm9vdEc0LmNydDBD # BgNVHR8EPDA6MDigNqA0hjJodHRwOi8vY3JsMy5kaWdpY2VydC5jb20vRGlnaUNl # cnRUcnVzdGVkUm9vdEc0LmNybDAgBgNVHSAEGTAXMAgGBmeBDAEEAjALBglghkgB # hv1sBwEwDQYJKoZIhvcNAQELBQADggIBAH1ZjsCTtm+YqUQiAX5m1tghQuGwGC4Q # TRPPMFPOvxj7x1Bd4ksp+3CKDaopafxpwc8dB+k+YMjYC+VcW9dth/qEICU0MWfN # thKWb8RQTGIdDAiCqBa9qVbPFXONASIlzpVpP0d3+3J0FNf/q0+KLHqrhc1DX+1g # tqpPkWaeLJ7giqzl/Yy8ZCaHbJK9nXzQcAp876i8dU+6WvepELJd6f8oVInw1Ypx # dmXazPByoyP6wCeCRK6ZJxurJB4mwbfeKuv2nrF5mYGjVoarCkXJ38SNoOeY+/um # nXKvxMfBwWpx2cYTgAnEtp/Nh4cku0+jSbl3ZpHxcpzpSwJSpzd+k1OsOx0ISQ+U # zTl63f8lY5knLD0/a6fxZsNBzU+2QJshIUDQtxMkzdwdeDrknq3lNHGS1yZr5Dhz # q6YBT70/O3itTK37xJV77QpfMzmHQXh6OOmc4d0j/R0o08f56PGYX/sr2H7yRp11 # LB4nLCbbbxV7HhmLNriT1ObyF5lZynDwN7+YAN8gFk8n+2BnFqFmut1VwDophrCY # oCvtlUG3OtUVmDG0YgkPCr2B2RP+v6TR81fZvAT6gt4y3wSJ8ADNXcL50CN/AAvk # dgIm2fBldkKmKYcJRyvmfxqkhQ/8mJb2VVQrH4D6wPIOK+XW+6kvRBVK5xMOHds3 # OBqhK/bt1nz8MIIFjTCCBHWgAwIBAgIQDpsYjvnQLefv21DiCEAYWjANBgkqhkiG # 9w0BAQwFADBlMQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5jMRkw # FwYDVQQLExB3d3cuZGlnaWNlcnQuY29tMSQwIgYDVQQDExtEaWdpQ2VydCBBc3N1 # cmVkIElEIFJvb3QgQ0EwHhcNMjIwODAxMDAwMDAwWhcNMzExMTA5MjM1OTU5WjBi # MQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5jMRkwFwYDVQQLExB3 # d3cuZGlnaWNlcnQuY29tMSEwHwYDVQQDExhEaWdpQ2VydCBUcnVzdGVkIFJvb3Qg # RzQwggIiMA0GCSqGSIb3DQEBAQUAA4ICDwAwggIKAoICAQC/5pBzaN675F1KPDAi # MGkz7MKnJS7JIT3yithZwuEppz1Yq3aaza57G4QNxDAf8xukOBbrVsaXbR2rsnny # yhHS5F/WBTxSD1Ifxp4VpX6+n6lXFllVcq9ok3DCsrp1mWpzMpTREEQQLt+C8weE # 5nQ7bXHiLQwb7iDVySAdYyktzuxeTsiT+CFhmzTrBcZe7FsavOvJz82sNEBfsXpm # 7nfISKhmV1efVFiODCu3T6cw2Vbuyntd463JT17lNecxy9qTXtyOj4DatpGYQJB5 # w3jHtrHEtWoYOAMQjdjUN6QuBX2I9YI+EJFwq1WCQTLX2wRzKm6RAXwhTNS8rhsD # dV14Ztk6MUSaM0C/CNdaSaTC5qmgZ92kJ7yhTzm1EVgX9yRcRo9k98FpiHaYdj1Z # XUJ2h4mXaXpI8OCiEhtmmnTK3kse5w5jrubU75KSOp493ADkRSWJtppEGSt+wJS0 # 0mFt6zPZxd9LBADMfRyVw4/3IbKyEbe7f/LVjHAsQWCqsWMYRJUadmJ+9oCw++hk # pjPRiQfhvbfmQ6QYuKZ3AeEPlAwhHbJUKSWJbOUOUlFHdL4mrLZBdd56rF+NP8m8 # 00ERElvlEFDrMcXKchYiCd98THU/Y+whX8QgUWtvsauGi0/C1kVfnSD8oR7FwI+i # sX4KJpn15GkvmB0t9dmpsh3lGwIDAQABo4IBOjCCATYwDwYDVR0TAQH/BAUwAwEB # /zAdBgNVHQ4EFgQU7NfjgtJxXWRM3y5nP+e6mK4cD08wHwYDVR0jBBgwFoAUReui # r/SSy4IxLVGLp6chnfNtyA8wDgYDVR0PAQH/BAQDAgGGMHkGCCsGAQUFBwEBBG0w # azAkBggrBgEFBQcwAYYYaHR0cDovL29jc3AuZGlnaWNlcnQuY29tMEMGCCsGAQUF # BzAChjdodHRwOi8vY2FjZXJ0cy5kaWdpY2VydC5jb20vRGlnaUNlcnRBc3N1cmVk # SURSb290Q0EuY3J0MEUGA1UdHwQ+MDwwOqA4oDaGNGh0dHA6Ly9jcmwzLmRpZ2lj # ZXJ0LmNvbS9EaWdpQ2VydEFzc3VyZWRJRFJvb3RDQS5jcmwwEQYDVR0gBAowCDAG # BgRVHSAAMA0GCSqGSIb3DQEBDAUAA4IBAQBwoL9DXFXnOF+go3QbPbYW1/e/Vwe9 # mqyhhyzshV6pGrsi+IcaaVQi7aSId229GhT0E0p6Ly23OO/0/4C5+KH38nLeJLxS # A8hO0Cre+i1Wz/n096wwepqLsl7Uz9FDRJtDIeuWcqFItJnLnU+nBgMTdydE1Od/ # 6Fmo8L8vC6bp8jQ87PcDx4eo0kxAGTVGamlUsLihVo7spNU96LHc/RzY9HdaXFSM # b++hUD38dglohJ9vytsgjTVgHAIDyyCwrFigDkBjxZgiwbJZ9VVrzyerbHbObyMt # 9H5xaiNrIv8SuFQtJ37YOtnwtoeW/VvRXKwYw02fc7cBqZ9Xql4o4rmUMYIDdjCC # A3ICAQEwdzBjMQswCQYDVQQGEwJVUzEXMBUGA1UEChMORGlnaUNlcnQsIEluYy4x # OzA5BgNVBAMTMkRpZ2lDZXJ0IFRydXN0ZWQgRzQgUlNBNDA5NiBTSEEyNTYgVGlt # ZVN0YW1waW5nIENBAhAFRK/zlJ0IOaa/2z9f5WEWMA0GCWCGSAFlAwQCAQUAoIHR # MBoGCSqGSIb3DQEJAzENBgsqhkiG9w0BCRABBDAcBgkqhkiG9w0BCQUxDxcNMjMx # MjA4MTcxMDU4WjArBgsqhkiG9w0BCRACDDEcMBowGDAWBBRm8CsywsLJD4JdzqqK # ycZPGZzPQDAvBgkqhkiG9w0BCQQxIgQgDOZ7ebJwZZkG6XzrNXJIU6mUMhjy/DMo # znbCFl3XQt0wNwYLKoZIhvcNAQkQAi8xKDAmMCQwIgQg0vbkbe10IszR1EBXaEE2 # b4KK2lWarjMWr00amtQMeCgwDQYJKoZIhvcNAQEBBQAEggIAa3pg8L4WYohgrriJ # W6z9iwmwfG8LiesGHXEwOVCoHD/09tyhriU/7QW1uE72fdzx676IuVdXbNEhcK00 # I+HJZYXR3RI1uOq7ZqgglzY4seYgbTxOmXea1k9oGQh5CcaQTq8PQgXOs/MVaEfR # NYDkJ3ZWGcDFallWnPRxUHn9yOZ+ZjtUunXR6xeu5DR5Br/r1+jW1+zdt8yihhat # qUeoVZKlZOLQsBQRXFJ3KEdTa3o2bp7Ri+9gNtRcVVrFUhq/b5dsUSrnlL7zuPuK # g0vgHAWuwBb90kkLRnAQTD3LA7yUenWgDEeUAXSQ9TZPOTErLt2sMGwDv9AJW823 # eLlOPArtXvzTv5I1Yxi2G0reU9/3X4uYHkUg5yKNQb2ebZ25JNrY1o2fIYXAMwSX # agX6YSKUBfhxpo22VLf6cw/crNb91s8vJC5rvyqiKTzwmfi/3JAPbu9M30PDIu+j # SNJP25OuP7Pwc+LDSjJOs+UdxFvocbcpoVKwNDZNUXKQNKdfBOoYq6fGBZDLQkON # XedZV7R8NlqBHVoZzsuPHPNdRl7NwVZWIDSlaySo/VTRkw6EjwFbpf2+3REDD3ZJ # TXMoNF4leO8k6ie9DkE9bWHiKMpKmi28DwGwiXzHz/IeEYe8lp5YNYIptVvTj4gN # FDKLZDwxc4m6HOPFypW6epM5xoQ= # SIG # End signature block |