Private/AzStackHci.ModuleManagement.Helpers.ps1
|
# //////////////////////////////////////////////////////////////////////////// # Function to check if a command exists # Timeout (seconds) for PSGallery version checks — prevents hangs when gallery is unreachable $script:FindModuleTimeoutSeconds = 15 Function Test-CommandExists { Param ($command) begin { # Write-Debug "Test-CommandExists: Beginning command existence check for '$command'" $oldPreference = $ErrorActionPreference $ErrorActionPreference = "Stop" [bool]$CommandExists = $false } process { try { if(Get-Command $command){ $CommandExists = $true } } Catch { $CommandExists = $false } Finally { $ErrorActionPreference=$oldPreference } } # End of process block end { # Write-Debug "Test-CommandExists: Command existence check completed" # Return the result return $CommandExists } } # End function Test-CommandExists # //////////////////////////////////////////////////////////////////////////// Function Get-AzStackHciEnvironmentCheckerModule { begin { # Write-Debug "Get-AzStackHciEnvironmentCheckerModule: Beginning Environment Checker module verification" # Check if the AzStackHci.EnvironmentChecker module is installed and running the latest version $Module = "AzStackHci.EnvironmentChecker" } process { # Check if the module is installed. # Force array cast so indexing and .Count behave consistently with 1 or many versions. # Sort descending so [0] is always the newest version. [Array]$InstalledVersions = @(Get-Module -Name $Module -ListAvailable -ErrorAction SilentlyContinue | Sort-Object -Property Version -Descending) # If only one version is installed, set the installed version if($InstalledVersions){ if($InstalledVersions.Count -eq 1) { [version]$InstalledVersion = $InstalledVersions[0].Version } elseif($InstalledVersions.Count -gt 1) { # Multiple versions installed, use the latest version (sorted descending above) [version]$InstalledVersion = $InstalledVersions[0].Version Write-Warning "There are $($InstalledVersions.Count) versions of 'AzStackHci.EnvironmentChecker' module installed" } Write-HostAzS "'AzStackHci.EnvironmentChecker' module version $($InstalledVersion.ToString()) is installed" -ForegroundColor "Green" # Do nothing, continue with the script } elseif(-not($InstalledVersions)){ # Module not found, ask user to install the module Write-HostAzS "Azure Stack HCI Environment Checker module is not installed" -ForegroundColor Red if ($script:SilentMode) { $InstallModulePrompt = "Y" } else { $InstallModulePrompt = Read-Host "Would you like to install the dependant module '$Module' now? (Y/N)" } if(([string]::IsNullOrWhiteSpace($InstallModulePrompt)) -or ($InstallModulePrompt -ne "Y")){ Throw "Error: Null or not 'Y' response. Exiting script, please install the '$Module' module and re-run the function" } elseif($InstallModulePrompt -eq "Y"){ Write-HostAzS "Installing module '$Module'...." -ForegroundColor "Green" try { # Requires -AllowClobber as some functions are included in Az.StackHCI Install-Module -Name $Module -Repository PSGallery -Force -ErrorVariable ModuleInstallError -AllowClobber -Scope AllUsers -Confirm:$false } catch { Throw "Error installing module '$Module' - $($_.Exception.Message)" } if(-not($ModuleInstallError)){ Write-HostAzS "Module '$Module' installed successfully" -ForegroundColor "Green" } else { Throw "Error installing module '$Module' - $ModuleInstallError" } } else { # All other responses Write-HostAzS "Invalid response, please enter 'Y' to install the module..." -ForegroundColor Red Throw "Exiting script, please install the '$Module' module and re-run the function" } } # Use -Global so EnvironmentChecker's exported functions land in the global session # state instead of being injected into our own module's scope. Without -Global, any # exported function from EnvironmentChecker that shares a name with one of OUR private # helpers (e.g. Get-SslCertificateChain in EnvironmentChecker 10.2509+) would shadow # the private helper at the call site inside our module, with confusing parameter # mismatch errors. (See also: rename of our internal helper to Get-AzSHciSslCertificateChain.) Import-Module -Name $Module -Force -Global # Load URLs from environment checker (Targets.json), check if multiple module version are installed, if so, use the latest version $Location = ((Get-Module -Name $Module -ListAvailable) | Sort-Object -Property Version -Descending | Select-Object -First 1).ModuleBase if(-not($Location)){ Write-HostAzS "Azure Stack HCI Environment Checker module not found, please install the module" -ForegroundColor Red Write-HostAzS "Run: 'Install-Module -Name AzStackHci.EnvironmentChecker' and troubleshoot any installation issues" -ForegroundColor Green Exit } } # End of process block end { # Write-Debug "Get-AzStackHciEnvironmentCheckerModule: Environment Checker module verification completed" # Return the module location Return $Location } } # End Function Get-AzStackHciEnvironmentCheckerModule # //////////////////////////////////////////////////////////////////////////// # Function to get the latest version of a module from PSGallery # This function will check for the latest version of a module in the PowerShell Gallery Function Get-LatestModuleVersion { [CmdletBinding()] param ( [Parameter(Mandatory=$true, Position=0)] [ValidateNotNullOrEmpty()] [string]$ModuleName ) begin { # Write-Debug "Get-LatestModuleVersion: Beginning latest module version check for '$ModuleName'" # //////////////////////////////////////////// # ///// Check for latest module version ////// # //////////////////////////////////////////// [bool]$ModuleNotInstalled = $false } process { # Get Existing Most Recent Module Version: [version]$ExistingModuleVersion = (Get-InstalledModule -Name $ModuleName -ErrorAction SilentlyContinue).Version if(-not $ExistingModuleVersion) { Write-Debug "Module '$ModuleName' is not installed via PSGallery. Skipping update check." $ModuleNotInstalled = $true return } # //////////////////////////////////////////////////////////////////////////// # Version discovery strategy: # # Step 1 (Fast path): Use Find-Module without -RequiredVersion to check for # the latest *listed* version in PSGallery. This is a single network call # and works for any module that is publicly listed. If it returns a version # higher than the installed version, use it immediately. # # Step 2 (Fallback for unlisted modules): If Step 1 does not find a newer # version, fall back to probing exact version numbers with -RequiredVersion. # This is required because Find-Module without -RequiredVersion does NOT # return unlisted packages. The only way to discover unlisted versions is # by probing exact version numbers. # Originally written for a module where all versions were "Unlisted" in # the PowerShell Gallery, this fallback has been retained for portability. # # Fallback version increment strategy: # 1. First, increment the Build (z) digit: x.y.z -> x.y.(z+1) # 2. If incrementing build finds no match, try rolling over to the next # Minor version: x.y.z -> x.(y+1).0 # 3. If that also finds no match, try rolling over to the next Major # version: x.y.z -> (x+1).0.0 # 4. Only stop if all three probes miss, meaning no further version exists. # //////////////////////////////////////////////////////////////////////////// # --- Step 1: Fast path — single call for latest listed version --- # Note: Find-Module can return multiple results when multiple repositories # are registered (e.g. PSGallery + a local repo). Sort descending and pick # the highest version across all repositories. # Uses Start-Job/Wait-Job with timeout to prevent hangs if PSGallery is unreachable. Write-Debug "Checking PowerShell Gallery for latest listed version of '$ModuleName' (timeout: $($script:FindModuleTimeoutSeconds)s)..." [version]$ListedVersion = $null [bool]$galleryReachable = $false $job = $null try { $job = Start-Job -ScriptBlock { param($Name) (Find-Module -Name $Name -ErrorAction SilentlyContinue | Sort-Object -Property Version -Descending | Select-Object -First 1).Version } -ArgumentList $ModuleName $completed = Wait-Job -Job $job -Timeout $script:FindModuleTimeoutSeconds -ErrorAction SilentlyContinue if ($completed -and $completed.State -eq 'Completed') { $ListedVersion = Receive-Job -Job $job -ErrorAction SilentlyContinue $galleryReachable = $true } else { Write-Debug "Version check timed out after $($script:FindModuleTimeoutSeconds) seconds — attempting to stop background job" if ($job) { Stop-Job -Job $job -ErrorAction SilentlyContinue } } } catch { Write-Debug "Version check failed: $($_.Exception.Message)" } finally { if ($job) { Remove-Job -Job $job -Force -ErrorAction SilentlyContinue # Verify job is actually gone — orphaned jobs can leak resources. $lingering = Get-Job -Id $job.Id -ErrorAction SilentlyContinue if ($lingering) { Write-Warning "Find-Module background job (Id $($job.Id)) did not terminate cleanly and is still present." } } } if($ListedVersion -and $ListedVersion -gt $ExistingModuleVersion) { # Latest listed version is newer than installed — use it directly Write-Debug "Latest listed version v$($ListedVersion.ToString()) is newer than installed v$($ExistingModuleVersion.ToString())" [version]$LatestModuleVersion = $ListedVersion } elseif (-not $galleryReachable) { # PSGallery was unreachable (timeout or error) — skip probing, continue with installed version Write-Debug "PSGallery unreachable; skipping unlisted version probing. Continuing with installed v$($ExistingModuleVersion.ToString())" } else { # --- Step 2: Disabled — unlisted version probing commented out --- # The module is now publicly listed on PSGallery, so probing for unlisted # versions by incrementing build/minor/major is no longer needed. # If Step 1 found no newer listed version, the installed version is current. if($ListedVersion) { Write-Debug "Listed version v$($ListedVersion.ToString()) is not newer than installed v$($ExistingModuleVersion.ToString()). No update needed." } else { Write-Debug "No listed version found in PSGallery. Continuing with installed v$($ExistingModuleVersion.ToString())." } <# --- Original Step 2: Fallback — incremental probing for unlisted versions --- Originally written for a module where all versions were "Unlisted" in PSGallery. Disabled because the module is now publicly listed and this probing adds unnecessary Find-Module -RequiredVersion calls that slow down the update check. [bool]$StopModuleVersion = $false [version]$ModuleVersion = $ExistingModuleVersion # Loop and check if a new unlisted version exists: Do { [int]$Major = $ModuleVersion.Major [int]$Minor = $ModuleVersion.Minor [int]$Build = $ModuleVersion.Build # --- Attempt 1: Increment Build (z) by 1 --- [version]$NewModuleVersion = "$Major.$Minor.$($Build + 1)" Write-Debug "Checking if module v$($NewModuleVersion) exists in PowerShell Gallery..." [version]$FoundVersion = (Find-Module -Name "$ModuleName" -RequiredVersion $NewModuleVersion -ErrorAction SilentlyContinue).Version if(-not $FoundVersion) { # --- Attempt 2: Roll over to next Minor version (x.(y+1).0) --- [version]$NewModuleVersion = "$Major.$($Minor + 1).0" Write-Debug "Build increment miss. Checking minor rollover v$($NewModuleVersion)..." [version]$FoundVersion = (Find-Module -Name "$ModuleName" -RequiredVersion $NewModuleVersion -ErrorAction SilentlyContinue).Version } if(-not $FoundVersion) { # --- Attempt 3: Roll over to next Major version ((x+1).0.0) --- [version]$NewModuleVersion = "$($Major + 1).0.0" Write-Debug "Minor rollover miss. Checking major rollover v$($NewModuleVersion)..." [version]$FoundVersion = (Find-Module -Name "$ModuleName" -RequiredVersion $NewModuleVersion -ErrorAction SilentlyContinue).Version } if($FoundVersion) { Write-Debug "Higher module version found v$($FoundVersion.ToString())" [version]$LatestModuleVersion = $FoundVersion # Continue searching from the found version [version]$ModuleVersion = $FoundVersion } else { # All three probes missed — no further version exists in PSGallery Write-Debug "No match for v$($Major).$($Minor).$($Build + 1), v$($Major).$($Minor + 1).0, or v$($Major + 1).0.0" $StopModuleVersion = $true } } While (-not($StopModuleVersion)) #> } } # End of process block end { # Write-Debug "Get-LatestModuleVersion: Latest module version check completed" # If module was not installed via PSGallery, return null (no version to report) if($ModuleNotInstalled) { return $null } # Return Function output if($LatestModuleVersion) { Return [version]$LatestModuleVersion } else { Return [version]$ExistingModuleVersion } } } # End Function Get-LatestModuleVersion # //////////////////////////////////////////////////////////////////////////// # Auto Update Module Function Function Update-ModuleVersion { <# .SYNOPSIS Checks PSGallery for a newer version of a module. Notifies by default; installs only on opt-in. .DESCRIPTION By default this function is NON-DESTRUCTIVE: it checks PSGallery for a newer version and, if found, prints a notification telling the user how to update manually. It does NOT install, uninstall or modify anything on the machine. When -InstallUpdate is specified, the function installs the newer version side-by-side via Install-Module and imports it. Older versions are left in place — PowerShell supports multiple installed versions natively (each lives under \Modules\<Name>\<Version>\) and Import-Module without -RequiredVersion loads the highest installed version automatically, so stale copies are harmless. Users who want to tidy up can run 'Uninstall-Module <Name> -AllVersions' themselves — following standard PowerShell convention (this is how Az, Microsoft.Graph, Pester etc. behave). .PARAMETER ModuleName Name of the module to check on PSGallery. .PARAMETER InstallUpdate Opt-in switch. When specified, if a newer version is available it will be installed side-by-side via Install-Module. Default is $false (notify-only). .OUTPUTS [bool] $true — a new version was INSTALLED in this call. Caller should re-run because the newly-installed code is not the code currently loaded. [bool] $false — no update available, notify-only, install skipped by the caller, or an error occurred. Caller should continue with the currently-loaded version. #> [CmdletBinding()] param ( [Parameter(Mandatory=$true, Position=0)] [ValidateNotNullOrEmpty()] [string]$ModuleName, [Parameter(Mandatory=$false)] [switch]$InstallUpdate ) begin { # Write-Debug "Update-ModuleVersion: Beginning module update process for '$ModuleName' (InstallUpdate=$($InstallUpdate.IsPresent))" [bool]$ModuleUpdated = $false } process { # Pick the highest currently-installed version of the module (a single module can have # multiple side-by-side installs under \Modules\<Name>\<Version>\). [version]$InstalledModuleVersion = (Get-Module -Name "$ModuleName" -ListAvailable | Sort-Object -Property Version -Descending | Select-Object -First 1).Version # Module not installed at all — nothing to check against. if(-not $InstalledModuleVersion) { Write-HostAzS -ForegroundColor Red "`n`tError: " -NoNewLine Write-HostAzS "Module '$ModuleName' is not installed.`n" -ForegroundColor Yellow return } Write-HostAzS -ForegroundColor Green "`n`tChecking for updates..." # Query PSGallery. Get-LatestModuleVersion is already defensive: on timeout / network error # it returns the currently-installed version, so we safely treat "no newer version" as a # no-op below. This means an offline machine will never block the caller. [version]$LatestModule = Get-LatestModuleVersion -ModuleName $ModuleName if(-not ($InstalledModuleVersion -lt $LatestModule)) { # No update available (or PSGallery unreachable and we fell back to installed version). Write-HostAzS -ForegroundColor Green "`n`tINFO: " -NoNewLine Write-HostAzS "$ModuleName module is up-to-date at v$($InstalledModuleVersion.ToString())`n" return } # //////////////////////////////////////////////////////////////////////////// # A newer version IS available. Behaviour depends on -InstallUpdate. # //////////////////////////////////////////////////////////////////////////// $galleryUrl = "https://www.powershellgallery.com/packages/$ModuleName/$($LatestModule.ToString())" if(-not $InstallUpdate) { # ----- Notify-only path (default) ----- # Print a friendly upgrade notice and RETURN WITHOUT MUTATING ANYTHING. # The caller continues its run with the currently-loaded version. Write-HostAzS "" Write-HostAzS -ForegroundColor Yellow "`tINFO: " -NoNewLine Write-HostAzS "A newer version of $ModuleName is available: " -NoNewLine Write-HostAzS -ForegroundColor Cyan "v$($InstalledModuleVersion.ToString()) -> v$($LatestModule.ToString())" Write-HostAzS -ForegroundColor Gray "`t To update, run: " -NoNewLine Write-HostAzS -ForegroundColor Cyan "Update-Module $ModuleName" Write-HostAzS -ForegroundColor Gray "`t Release notes: " -NoNewLine Write-HostAzS -ForegroundColor Cyan $galleryUrl Write-HostAzS -ForegroundColor Gray "`t To auto-install + re-run: " -NoNewLine Write-HostAzS -ForegroundColor Cyan "re-invoke with -AutoUpdate" Write-HostAzS -ForegroundColor Gray "`t To silence this check: " -NoNewLine Write-HostAzS -ForegroundColor Cyan "re-invoke with -NoAutoUpdate" Write-HostAzS "" # ModuleUpdated stays $false — caller continues. return } # ----- Install path (opt-in: -InstallUpdate) ----- Write-HostAzS -ForegroundColor Yellow "`n`tINFO: " -NoNewLine Write-HostAzS "Updating $ModuleName from v$($InstalledModuleVersion.ToString()) to v$($LatestModule.ToString())...`n" $ModuleInstallError = $null try { Install-Module -Name $ModuleName -RequiredVersion $LatestModule -ErrorAction Continue -Force -ErrorVariable ModuleInstallError } catch { Write-Warning "Error installing latest version $($LatestModule.ToString()) of module $ModuleName. $($_.Exception.Message)" } if($ModuleInstallError) { Write-Warning "Error updating module $ModuleName. $ModuleInstallError" return # $ModuleUpdated remains $false; caller should continue with existing version } # Unload the currently-loaded module from this session and import the freshly-installed version. Remove-Module -Name $ModuleName -ErrorAction SilentlyContinue try { Import-Module -Name $ModuleName -RequiredVersion $LatestModule -Force -ErrorAction Stop # Validate the imported module version matches what we installed. Get-Module may return # multiple entries when nested modules are loaded, so pick the highest version. $loadedVersion = (Get-Module -Name $ModuleName | Sort-Object -Property Version -Descending | Select-Object -First 1).Version if ($loadedVersion -ne $LatestModule) { $loadedVersionString = if ($loadedVersion) { $loadedVersion.ToString() } else { '<not loaded>' } Write-Warning "Installed v$($LatestModule.ToString()) but loaded v$loadedVersionString. Update may be incomplete." } } catch { Write-Warning "Error importing latest version $($LatestModule.ToString()) of module $ModuleName. $($_.Exception.Message)" } [bool]$ModuleUpdated = $true Write-HostAzS -ForegroundColor Green "`t`tModule updated successfully!`n" Write-HostAzS -ForegroundColor Green "`tPlease re-run the function so it executes with the freshly-installed code.`n`n" # Older versions are deliberately left in place. PowerShell supports side-by-side installs # and Import-Module loads the highest version by default, so stale copies are harmless. # Users can tidy up with 'Uninstall-Module <Name> -AllVersions' if they wish. } # End of process block end { # Write-Debug "Update-ModuleVersion: Module update process completed (ModuleUpdated=$ModuleUpdated)" # Return $true only when the module was actually reinstalled in this call, so the caller # knows to abort and re-run. Notify-only / no-update / error paths all return $false. Return $ModuleUpdated } } # End Function Update-ModuleVersion # //////////////////////////////////////////////////////////////////////////// # Function to display an ASCII art banner # This function displays an ASCII art banner in the console # Font = Slant, credit web-site: https://patorjk.com/software/taag/ Function Show-ASCIIArtBanner { begin { # Write-Debug "Show-ASCIIArtBanner: Beginning ASCII art banner display" } process { # ASCI Art Banner variable [string]$banner=@' //////////////////////////////////////////////////////////////////////////////////////// ___ __ __ / |____ __ __________ / / ____ _________ _/ / / /| /_ / / / / / ___/ _ \ / / / __ \/ ___/ __ `/ / / ___ |/ /_/ /_/ / / / __/ / /___/ /_/ / /__/ /_/ / / /_/ |_/___/\__,_/_/ \___/ /_____/\____/\___/\__,_/_/ ______ __ _ _ __ ______ __ / ____/___ ____ ____ ___ _____/ /_(_) __(_) /___ __ /_ __/__ _____/ /______ / / / __ \/ __ \/ __ \/ _ \/ ___/ __/ / | / / / __/ / / / / / / _ \/ ___/ __/ ___/ / /___/ /_/ / / / / / / / __/ /__/ /_/ /| |/ / / /_/ /_/ / / / / __(__ ) /_(__ ) \____/\____/_/ /_/_/ /_/\___/\___/\__/_/ |___/_/\__/\__, / /_/ \___/____/\__/____/ /____/ //////////////////////////////////////////////////////////////////////////////////////// '@ Write-HostAzS -ForegroundColor Green `n$banner`n } end { # Write-Debug "Show-ASCIIArtBanner: ASCII art banner display completed" } } # End Function Show-ASCIIArtBanner # SIG # Begin signature block # MIInbgYJKoZIhvcNAQcCoIInXzCCJ1sCAQExDzANBglghkgBZQMEAgEFADB5Bgor # BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCD67be0can98J9x # FYF0NMFqxbWrOl3kEla1ATbZZI0/U6CCDMkwggYEMIID7KADAgECAhMzAAACHPrN # xZvoL37EAAAAAAIcMA0GCSqGSIb3DQEBCwUAMFcxCzAJBgNVBAYTAlVTMR4wHAYD # VQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBD # b2RlIFNpZ25pbmcgUENBIDIwMjQwHhcNMjYwNDE2MTg1OTQxWhcNMjcwNDE1MTg1 # OTQxWjB0MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UE # BxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMR4wHAYD # VQQDExVNaWNyb3NvZnQgQ29ycG9yYXRpb24wggEiMA0GCSqGSIb3DQEBAQUAA4IB # DwAwggEKAoIBAQDVsZfgOKmM31HPfoWOoNEiw0SlCiIxUMC0I9NMWbucKOw/e9lP # oAoehQVu6SG65V4EPzrYsnBnFPNoi4/HoOdjhz1qkrEt4I6tEcxXU6oOeY9zGveC # /3iBeuhLYxM3M/PkcUoebF+Nednm8OkdSPoDu8imViHPQq/8CQUu0WRR4rE+dMRf # rpVqfmNi2qWCX94T4MsepijGVkwE//tJg0ryAiYdHT34LSnlG/RSBZmQRGWZ5g8j # qnKjRParSqMft1gvjuUTVgtWNZfgcLFSK5Wa0myrq8OPcgTGGsRgun+tnSS+IxDT # xVsAPH1OzvPjwomguByhUe/OcvUN0D5Wmp7xAgMBAAGjggGqMIIBpjAOBgNVHQ8B # Af8EBAMCB4AwHwYDVR0lBBgwFgYKKwYBBAGCN0wIAQYIKwYBBQUHAwMwHQYDVR0O # BBYEFNoH7a2YDjOSwpkp6DHcmUS7J+0yMFQGA1UdEQRNMEukSTBHMS0wKwYDVQQL # EyRNaWNyb3NvZnQgSXJlbGFuZCBPcGVyYXRpb25zIExpbWl0ZWQxFjAUBgNVBAUT # DTIzMDAxMis1MDc1NjkwHwYDVR0jBBgwFoAUf1k/VCHarU/vBeXmo9ctBpQSCDEw # YAYDVR0fBFkwVzBVoFOgUYZPaHR0cDovL3d3dy5taWNyb3NvZnQuY29tL3BraW9w # cy9jcmwvTWljcm9zb2Z0JTIwQ29kZSUyMFNpZ25pbmclMjBQQ0ElMjAyMDI0LmNy # bDBtBggrBgEFBQcBAQRhMF8wXQYIKwYBBQUHMAKGUWh0dHA6Ly93d3cubWljcm9z # b2Z0LmNvbS9wa2lvcHMvY2VydHMvTWljcm9zb2Z0JTIwQ29kZSUyMFNpZ25pbmcl # MjBQQ0ElMjAyMDI0LmNydDAMBgNVHRMBAf8EAjAAMA0GCSqGSIb3DQEBCwUAA4IC # AQAUnEqhaRXe0T3hIJjvdQErEkrA/7bByjn6t5IArODkkRjzkYwtKMc2yYj2quaN # rLutWw2YZcngKPy1b71YyDJQTy4NDRwaSh9Tw5thrk3NmcPrAHia5vtcBJ1CgtKK # 7mQbIcQ22d/N3813ayCDDFewu1+jsZmX+r/aTEqaOM4TVxVtRSkuCy8nAXKuChOK # Li/zA4XuH8iEYqIsj2YoNaeSxVmeGiERXpKdo3dDmYi0kO5w2D8VS4c3+9h6gElY # BaAAg/dYErBg27qT3vv0zRDJhJufvCNylA8S7/+8H5E/PV5cng6na9VV/w9OV3qu # uND6zdGa2EX38Glp50F9AIQk3p2xXmcvorDeM4XJ7UlWYBi6g80J1SSOQnInCYFE # msfUNn3+1AaTJKSJL83quKArTac2pKhu0Yzzzrzo6HrsRiQKzpnRBb1/dMa6P3hz # 75XbMRBctNsFhZC07WCmjExdLg2eHW5uV0TY8D5+6wozJf7vF3+WHkYPO85Z+BC6 # U4FkNbYNycZ9cE4j1tXRdyDCfml6c0HWPHjNVDObrv9lKt3qUqFpX38VCqVCyNOO # 1UcXfQiVjJw32U2WUKZjt/neJKHEBsm9kFsLuWzkQ53+qcaSaytmsCnk2gOglrlD # 5d3kKyvvAw+rzm0lT8K38P6PLxfZQHhu4W8dV7Av8N2ZmDCCBr0wggSloAMCAQIC # EzMAAAA5O7Y3Gb8GHWcAAAAAADkwDQYJKoZIhvcNAQEMBQAwgYgxCzAJBgNVBAYT # AlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYD # VQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xMjAwBgNVBAMTKU1pY3Jvc29mdCBS # b290IENlcnRpZmljYXRlIEF1dGhvcml0eSAyMDExMB4XDTI0MDgwODIwNTQxOFoX # DTM2MDMyMjIyMTMwNFowVzELMAkGA1UEBhMCVVMxHjAcBgNVBAoTFU1pY3Jvc29m # dCBDb3Jwb3JhdGlvbjEoMCYGA1UEAxMfTWljcm9zb2Z0IENvZGUgU2lnbmluZyBQ # Q0EgMjAyNDCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBANgBnB7jOMeq # lRYHNa265v4IY9fH8TKhemHfPINe1gpLaV3dhg324WwH06LcHbpnsBukCDNitryo # 0dtS/EW6I/yEL/bLSY8hKpbfQuWusBPr9qazYcDxCW/qnjb5JsI1s8bNOg3bVATv # QVL4tcf03aTycsz8QeCdM0l/yHRObJ9QqazM1r6VPEOJ7LL+uEEb73w6QCuhs89a # 1uv1zerOYMnsneRRwCbpyW11IcggU0cRKDDq1pjVJzIbIF6+oiXXbReOsgeI8zu1 # FyQfK0fVkaya8SmVHQ/tOf23mZ4W9k0Ri22QW9p3UgSC5OUDktKxxcCmGL6tXLfO # GSWHIIV4YrTJTT6PNty5REojHJuZHArkF9VnHTERWoTjAzfI3kP+5b4alUdhgAZ7 # ttOu1bVnXfHaqPYl2rPs20ji03LOVWsh/radgE17es5hL+t6lV0eVHrVhsssROWJ # uz2MXMCt7iw7lFPG9LXKGjsmonn2gotGdHIuEg5JnJMJVmixd5LRlkmgYRZKzhxS # CwyoGIq0PhaA7Y+VPct5pCHkijcIIDm0nlkK+0KyepolcqGm0T/GYQRMhHJlGOOm # VQop36wUVUYklUy++vDWeEgEo4s7hxN6mIbf2MSIQ/iIfMZgJxC69oukMUXCrOC3 # SkE/xIkgpfl22MM1itkZ35nNXkMolU1lAgMBAAGjggFOMIIBSjAOBgNVHQ8BAf8E # BAMCAYYwEAYJKwYBBAGCNxUBBAMCAQAwHQYDVR0OBBYEFH9ZP1Qh2q1P7wXl5qPX # LQaUEggxMBkGCSsGAQQBgjcUAgQMHgoAUwB1AGIAQwBBMA8GA1UdEwEB/wQFMAMB # Af8wHwYDVR0jBBgwFoAUci06AjGQQ7kUBU7h6qfHMdEjiTQwWgYDVR0fBFMwUTBP # oE2gS4ZJaHR0cDovL2NybC5taWNyb3NvZnQuY29tL3BraS9jcmwvcHJvZHVjdHMv # TWljUm9vQ2VyQXV0MjAxMV8yMDExXzAzXzIyLmNybDBeBggrBgEFBQcBAQRSMFAw # TgYIKwYBBQUHMAKGQmh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2kvY2VydHMv # TWljUm9vQ2VyQXV0MjAxMV8yMDExXzAzXzIyLmNydDANBgkqhkiG9w0BAQwFAAOC # AgEAFJQfOChP7onn6fLIMKrSlN1WYKwDFgAddymOUO3FrM8d7B/W/iQ6DxXsDn7D # 5W4wMwYeLystcEqfkjz4NURRgazyMu5yRzQh4LqjA4tStTcJh1opExo7nn5PuPBY # nbu0+THSuVHTe0VTTPVhily/piFrDo3axQ9P4C+Ol5yet+2gTfekICS5xS+cYfSI # vgn0JksVBVMYVI5QFu/qhnLhsEFEUzG8fvv0hjgkO+lkpV9ty6GkN4vdnd7ya6Q6 # aR9y34aiM1qmxaxBi6OUnyNl6fkuun/diTFnYDLTppOkr/mg5WSfCiDVMNCxtj4w # PKC5OmHm1DQIt/MNokbbH3UGsFP1QbzsLocuSqLCvH09Io3fDPTmscR9Y75G4qX7 # RTX8AdBPo0I6OEojf39zuFZt0qOHm65YWQE69cZM2ueE1MB05dNNgHK9gTE7zKvK # /fg8B2qjW88MT/WF5V5uvZGtqa9FSL2RazArA+rDPuf6JGYz4HpgMZHB4S6szWSK # YBv0VisCzfxgeU+dquXW9bd0auYlOB58DPcOYKdc3Se94g+xL4pcEhbB54JOgAkw # YTu/9dLeH2pDqeJZAABVDWRQCaXfO5LgyKwKCLYXpigrZYCjUSBcr+Ve8PFWMhVT # Ql0v4q8J/AUmQN5W4n101cY2L4A7GTQG1h32HHAvfQESWP0xghn7MIIZ9wIBATBu # MFcxCzAJBgNVBAYTAlVTMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24x # KDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNpZ25pbmcgUENBIDIwMjQCEzMAAAIc # +s3Fm+gvfsQAAAAAAhwwDQYJYIZIAWUDBAIBBQCgga4wGQYJKoZIhvcNAQkDMQwG # CisGAQQBgjcCAQQwHAYKKwYBBAGCNwIBCzEOMAwGCisGAQQBgjcCARUwLwYJKoZI # hvcNAQkEMSIEILCaiypjzefTjJCiwKdS8XHJLsA61te2YR1h4lax2Kj8MEIGCisG # AQQBgjcCAQwxNDAyoBSAEgBNAGkAYwByAG8AcwBvAGYAdKEagBhodHRwOi8vd3d3 # Lm1pY3Jvc29mdC5jb20wDQYJKoZIhvcNAQEBBQAEggEAWlH6QHGvrgzhOe0MbAPG # jLXRlyn36+HWYiqSaCg7qnOz70xoY7v7E1DWBo9gF44S9An008y8UTvFt9EaouCQ # YnaBtZhsu3I4SkBwrZrpS/4/+X2uYuotV9jg5skFC7y/eG5tObnxNQdEfwHYTVZ9 # UowfowY6cWP8Q/kPNjpiUSQpt8jfinxFrtt1uHNhdI+OsXOsSIS2XoHDHA650tGb # svoCO3bnhpoYqQMK9vxmMmAJzThVPLbQYriiDxSjIafzNkZPsyePRVaDdlIEGKRP # lD5MjjK0c/yNcsZC21ux+p6tE5xrR+CLXw9qyWh1TM7AoATXM9U8xHLDi+A4jDIX # LqGCF60wghepBgorBgEEAYI3AwMBMYIXmTCCF5UGCSqGSIb3DQEHAqCCF4YwgheC # AgEDMQ8wDQYJYIZIAWUDBAIBBQAwggFaBgsqhkiG9w0BCRABBKCCAUkEggFFMIIB # QQIBAQYKKwYBBAGEWQoDATAxMA0GCWCGSAFlAwQCAQUABCB7e0PVQzR7oWKpY748 # xom1/4ThQ1CMJAXNb4eFN17vMgIGaeyDroWyGBMyMDI2MDQyODIxNTI1My45NTZa # MASAAgH0oIHZpIHWMIHTMQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3Rv # bjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0 # aW9uMS0wKwYDVQQLEyRNaWNyb3NvZnQgSXJlbGFuZCBPcGVyYXRpb25zIExpbWl0 # ZWQxJzAlBgNVBAsTHm5TaGllbGQgVFNTIEVTTjo2RjFBLTA1RTAtRDk0NzElMCMG # A1UEAxMcTWljcm9zb2Z0IFRpbWUtU3RhbXAgU2VydmljZaCCEfswggcoMIIFEKAD # AgECAhMzAAACHAlVFdfDWQfRAAEAAAIcMA0GCSqGSIb3DQEBCwUAMHwxCzAJBgNV # BAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4w # HAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29m # dCBUaW1lLVN0YW1wIFBDQSAyMDEwMB4XDTI1MDgxNDE4NDgzMVoXDTI2MTExMzE4 # NDgzMVowgdMxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYD # VQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xLTAr # BgNVBAsTJE1pY3Jvc29mdCBJcmVsYW5kIE9wZXJhdGlvbnMgTGltaXRlZDEnMCUG # A1UECxMeblNoaWVsZCBUU1MgRVNOOjZGMUEtMDVFMC1EOTQ3MSUwIwYDVQQDExxN # aWNyb3NvZnQgVGltZS1TdGFtcCBTZXJ2aWNlMIICIjANBgkqhkiG9w0BAQEFAAOC # Ag8AMIICCgKCAgEAow0xEAUaFIyyLIXeFzeI8IKyBON2u0Dr02ISE5p9G5CUXfnF # u2S0E1gWCMvDWpopX6lRxjmgnqaL3BtnWlBVTo8xUNRZu23ie4YBMAJB7Ut6mnqn # HVwvDJxGO4TD3SnrCd+yg35B9QFejq3o4+OByvXjynaypZyukcQaLsKQvoxE8ElH # H7zcOXEJWmU3rnXzaW/S4SH3OPhoUbTTcy6nUgKx5pRWiQ24UEPLYzcxGJjqjkz+ # GiCWGPFHDMdW86laWvmCslouQPsN2eBk8dxJcEZmW4l6p4TthoXcfexEA9YdYaMz # 10aMhZNpdsNaDtDQUMDEC3k1D1My69MXSPlUmD9xFyDlkXiVa7BCEp3XcVtqTgzH # Gwr28JD6oE7zEPYeuZOiuCBXTZSo/wk3tbDlsESbIPV6inYqrzxiMYqlxfCdzC3C # imh9/NT/Lk9/aU+Iyyc9b3OaT0dZ8wgLaVDCGELRMrqyImdFHv0MudctzW/kPsV3 # Ja9ufpKWujEiN3CW//X8hFa9j5ImNeQzcMit3MoSaoGwnbiZJX1IyibIphlqccXF # k4oTTSOQBsAUw8U0gwOnM5UJD8mBUBd65Np6NBkx2cviJ4I34GyXFCWyy5Ft1QsB # YyVfAG3KOhCfPHQf8lQzJvLr57YW0bD/xVs4Ag4gTS6KZNyFEfX9jFdRlr0CAwEA # AaOCAUkwggFFMB0GA1UdDgQWBBRa3mOCzB8u7zpvDh8MGKVYLCk7ZDAfBgNVHSME # GDAWgBSfpxVdAF5iXYP05dJlpxtTNRnpcjBfBgNVHR8EWDBWMFSgUqBQhk5odHRw # Oi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL2NybC9NaWNyb3NvZnQlMjBUaW1l # LVN0YW1wJTIwUENBJTIwMjAxMCgxKS5jcmwwbAYIKwYBBQUHAQEEYDBeMFwGCCsG # AQUFBzAChlBodHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL2NlcnRzL01p # Y3Jvc29mdCUyMFRpbWUtU3RhbXAlMjBQQ0ElMjAyMDEwKDEpLmNydDAMBgNVHRMB # Af8EAjAAMBYGA1UdJQEB/wQMMAoGCCsGAQUFBwMIMA4GA1UdDwEB/wQEAwIHgDAN # BgkqhkiG9w0BAQsFAAOCAgEAklb6w/deaid3BujQCtWFBe0n9pkyRy+yyWEg70iD # woJ5u0e0O+4GerNzdZb1zTPsHJ8EGMyo1K7ytL21+pmdFMTl19PC8OJ5Y2p+XKUQ # y2dD+hggRMmJgDQsgbOCxHYeO+jg4t+vg61wUrovzzLkH3z0PJXXvoNuBj9Lda9C # iNMd60451Kube99ArSf6ZMj3t0p4rFbgSazDs+8TJ+8KA5GVaYjPHj9rlMuI3Wjo # hEc9apnQ6hMjMck3jlHZIwluVYeUQE0qjmApfMtTAEzbMUdY8sLTunL1GkbDSeKn # 9O7llBGnNtyM1uM9Mdv1VyWh0z/IriQKIjntqqGyoF0HvDHOFZCyUDBPLflyiu7Y # 1zQ/sPounsb96aBfQdq3h3LOn6t+m9EnNz/G6MzzWvpJk6YgTHTIqeQN/F/XpiPv # bfek3nq/PYbL3au+kBfRUHiCFXSvt6lor0HC626vUmz9ZNPOxwEWLuccomxsy3Jw # WH79vsM/7ARqoG5h6d6NahfaOuRP4XI9xtdH3Pa/NCLyQjxKXyLxzwQzjddkX2Ep # TJnlypuhPmEdea59Uz2E303LxyXSnKBvGsAnyWYAfnejr3YAiL9YrN2l2dn198Rp # A4DCm9QtZYiwC0q2fuUvui34PfPIUZByf7wHuuWu50hY9WLx1kOMI8xyo7AI6TaN # rnIwggdxMIIFWaADAgECAhMzAAAAFcXna54Cm0mZAAAAAAAVMA0GCSqGSIb3DQEB # CwUAMIGIMQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UE # BxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMTIwMAYD # VQQDEylNaWNyb3NvZnQgUm9vdCBDZXJ0aWZpY2F0ZSBBdXRob3JpdHkgMjAxMDAe # Fw0yMTA5MzAxODIyMjVaFw0zMDA5MzAxODMyMjVaMHwxCzAJBgNVBAYTAlVTMRMw # EQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVN # aWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0 # YW1wIFBDQSAyMDEwMIICIjANBgkqhkiG9w0BAQEFAAOCAg8AMIICCgKCAgEA5OGm # TOe0ciELeaLL1yR5vQ7VgtP97pwHB9KpbE51yMo1V/YBf2xK4OK9uT4XYDP/XE/H # ZveVU3Fa4n5KWv64NmeFRiMMtY0Tz3cywBAY6GB9alKDRLemjkZrBxTzxXb1hlDc # wUTIcVxRMTegCjhuje3XD9gmU3w5YQJ6xKr9cmmvHaus9ja+NSZk2pg7uhp7M62A # W36MEBydUv626GIl3GoPz130/o5Tz9bshVZN7928jaTjkY+yOSxRnOlwaQ3KNi1w # jjHINSi947SHJMPgyY9+tVSP3PoFVZhtaDuaRr3tpK56KTesy+uDRedGbsoy1cCG # MFxPLOJiss254o2I5JasAUq7vnGpF1tnYN74kpEeHT39IM9zfUGaRnXNxF803RKJ # 1v2lIH1+/NmeRd+2ci/bfV+AutuqfjbsNkz2K26oElHovwUDo9Fzpk03dJQcNIIP # 8BDyt0cY7afomXw/TNuvXsLz1dhzPUNOwTM5TI4CvEJoLhDqhFFG4tG9ahhaYQFz # ymeiXtcodgLiMxhy16cg8ML6EgrXY28MyTZki1ugpoMhXV8wdJGUlNi5UPkLiWHz # NgY1GIRH29wb0f2y1BzFa/ZcUlFdEtsluq9QBXpsxREdcu+N+VLEhReTwDwV2xo3 # xwgVGD94q0W29R6HXtqPnhZyacaue7e3PmriLq0CAwEAAaOCAd0wggHZMBIGCSsG # AQQBgjcVAQQFAgMBAAEwIwYJKwYBBAGCNxUCBBYEFCqnUv5kxJq+gpE8RjUpzxD/ # LwTuMB0GA1UdDgQWBBSfpxVdAF5iXYP05dJlpxtTNRnpcjBcBgNVHSAEVTBTMFEG # DCsGAQQBgjdMg30BATBBMD8GCCsGAQUFBwIBFjNodHRwOi8vd3d3Lm1pY3Jvc29m # dC5jb20vcGtpb3BzL0RvY3MvUmVwb3NpdG9yeS5odG0wEwYDVR0lBAwwCgYIKwYB # BQUHAwgwGQYJKwYBBAGCNxQCBAweCgBTAHUAYgBDAEEwCwYDVR0PBAQDAgGGMA8G # A1UdEwEB/wQFMAMBAf8wHwYDVR0jBBgwFoAU1fZWy4/oolxiaNE9lJBb186aGMQw # VgYDVR0fBE8wTTBLoEmgR4ZFaHR0cDovL2NybC5taWNyb3NvZnQuY29tL3BraS9j # cmwvcHJvZHVjdHMvTWljUm9vQ2VyQXV0XzIwMTAtMDYtMjMuY3JsMFoGCCsGAQUF # BwEBBE4wTDBKBggrBgEFBQcwAoY+aHR0cDovL3d3dy5taWNyb3NvZnQuY29tL3Br # aS9jZXJ0cy9NaWNSb29DZXJBdXRfMjAxMC0wNi0yMy5jcnQwDQYJKoZIhvcNAQEL # BQADggIBAJ1VffwqreEsH2cBMSRb4Z5yS/ypb+pcFLY+TkdkeLEGk5c9MTO1OdfC # cTY/2mRsfNB1OW27DzHkwo/7bNGhlBgi7ulmZzpTTd2YurYeeNg2LpypglYAA7AF # vonoaeC6Ce5732pvvinLbtg/SHUB2RjebYIM9W0jVOR4U3UkV7ndn/OOPcbzaN9l # 9qRWqveVtihVJ9AkvUCgvxm2EhIRXT0n4ECWOKz3+SmJw7wXsFSFQrP8DJ6LGYnn # 8AtqgcKBGUIZUnWKNsIdw2FzLixre24/LAl4FOmRsqlb30mjdAy87JGA0j3mSj5m # O0+7hvoyGtmW9I/2kQH2zsZ0/fZMcm8Qq3UwxTSwethQ/gpY3UA8x1RtnWN0SCyx # TkctwRQEcb9k+SS+c23Kjgm9swFXSVRk2XPXfx5bRAGOWhmRaw2fpCjcZxkoJLo4 # S5pu+yFUa2pFEUep8beuyOiJXk+d0tBMdrVXVAmxaQFEfnyhYWxz/gq77EFmPWn9 # y8FBSX5+k77L+DvktxW/tM4+pTFRhLy/AsGConsXHRWJjXD+57XQKBqJC4822rpM # +Zv/Cuk0+CQ1ZyvgDbjmjJnW4SLq8CdCPSWU5nR0W2rRnj7tfqAxM328y+l7vzhw # RNGQ8cirOoo6CGJ/2XBjU02N7oJtpQUQwXEGahC0HVUzWLOhcGbyoYIDVjCCAj4C # AQEwggEBoYHZpIHWMIHTMQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3Rv # bjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0 # aW9uMS0wKwYDVQQLEyRNaWNyb3NvZnQgSXJlbGFuZCBPcGVyYXRpb25zIExpbWl0 # ZWQxJzAlBgNVBAsTHm5TaGllbGQgVFNTIEVTTjo2RjFBLTA1RTAtRDk0NzElMCMG # A1UEAxMcTWljcm9zb2Z0IFRpbWUtU3RhbXAgU2VydmljZaIjCgEBMAcGBSsOAwIa # AxUAWmTiA01u5mxq/nVxiRJLMOskVGeggYMwgYCkfjB8MQswCQYDVQQGEwJVUzET # MBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMV # TWljcm9zb2Z0IENvcnBvcmF0aW9uMSYwJAYDVQQDEx1NaWNyb3NvZnQgVGltZS1T # dGFtcCBQQ0EgMjAxMDANBgkqhkiG9w0BAQsFAAIFAO2bnzwwIhgPMjAyNjA0Mjgy # MTAzNTZaGA8yMDI2MDQyOTIxMDM1NlowdDA6BgorBgEEAYRZCgQBMSwwKjAKAgUA # 7ZufPAIBADAHAgEAAgIPbDAHAgEAAgITrTAKAgUA7ZzwvAIBADA2BgorBgEEAYRZ # CgQCMSgwJjAMBgorBgEEAYRZCgMCoAowCAIBAAIDB6EgoQowCAIBAAIDAYagMA0G # CSqGSIb3DQEBCwUAA4IBAQBr6wa/gtdJkh0pHlNsg+3Uxb3TAaQnUhtikO130sdX # Jc5JLaKcZE+YVGfjpgDFbsV67Qj8Quns7ynbsEUJ5VEDr0xKrRMMaPtQVjwsISdA # Pw3gqfqKQ6RiO1tkF0sRLXIi0KtrF6YeF3k6eTZzHml4zqOt7BBHlmel7Q7otVIX # mARU1q8IH85whDGm4OQH2R4o9xzKh7LNcC90ymkdFjSooKLgiKzJ0Js3OVmTmvmq # M8AMmbZSTx0KkSLD7uW2NKeBhGDuRQBFrvPFtQfJ8a1FCRp9Tv8AWn6qOBjUzmbI # WQNsV/sPPtKvLIttW/iguyfkX0MfaMaQHA6XJENSu8epMYIEDTCCBAkCAQEwgZMw # fDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1Jl # ZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEmMCQGA1UEAxMd # TWljcm9zb2Z0IFRpbWUtU3RhbXAgUENBIDIwMTACEzMAAAIcCVUV18NZB9EAAQAA # AhwwDQYJYIZIAWUDBAIBBQCgggFKMBoGCSqGSIb3DQEJAzENBgsqhkiG9w0BCRAB # BDAvBgkqhkiG9w0BCQQxIgQgmey+3UZHR6WtQEJiIQeIGU719qhQ/4uqV2aXZ+kf # MVkwgfoGCyqGSIb3DQEJEAIvMYHqMIHnMIHkMIG9BCCgIGkmNhdo7+KE7dWhI+E2 # Ctx2RLWoYvvJodCIciHHaDCBmDCBgKR+MHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQI # EwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3Nv # ZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1wIFBD # QSAyMDEwAhMzAAACHAlVFdfDWQfRAAEAAAIcMCIEIDPDrXj+JvbehrLdy4OSVfjQ # HMxRQqrnhL//e2Z+eEzLMA0GCSqGSIb3DQEBCwUABIICADcxVoJqVmYafxNXFl+X # Er7t99nQp4fQk7FsCxJInHcufcZEaGwwkYEib0MAnWH6srpzrODVZjCUnBwUhjJF # rCp7UKv+XCQsxZ3e4QpL2Tgg6xO6GAsEKA/W/kGboXrUUBrKFwIPDU1VpgflN8ur # m8Dxm0iPCUwpF5KdYclB4FQLMxMDaR4X1ez5hAH0d8HNFwBK0KCI1/5qhe1QWMyt # S8xDz22gyNSctZ5UTyG4FbXYQWKIhKlKDTajVvH2BMpHUr/XVomrc1KUpcS6Ieoh # uTCTxjnuIbP7yI5JftFfiCUW/eS6Fl4ZKwDEiN6zZZKfCel0GfqM1bQYeQfROC53 # CaoI5ZArXtEoHiN0wZqAdEURjbh5QOZFIasT3kmEEd7j8LC3CNgUEXzkd/fvNBo5 # AwwT4E7Gzd5GsOKowmmVQBwNZTEMALe+hPvER0T+3JVgPZmsKTBA9ko7o398y3VO # mZ/aVRdTqRg+ukdU9iFnLd/UXQtppgkotGu2NxQTi+d2/jIB2B1tfzy3YS0uLWKO # XGVGvgTLetYj7jAmmwjYjsGX4HkAOBfoMAZF+Fl1Qi2+gwaVCc/fpKWq3QUpi/gI # Y/Xc5kYPOgOFbtmGtxtVJ8MMy+m1nnUCXNp8umw1OX1yQK8sRlqW/D01p2vuY108 # r9zO4c9F3BEI6gXKlczCRKoI # SIG # End signature block |