Private/Install-AWSToolsModuleFromZip.ps1
|
<#
.Synopsis Installs AWS Tools modules from a zip file. .Description Extracts modules from zip file, copies them to target path, and generates PSGetModuleInfo.xml files for PowerShellGet compatibility. Supports both AWS.Tools and AWS.Tools.Installer. Returns a hashtable with two properties: - Version: String representing the release version (from AWS.Tools rollup, AWS.Tools.Common, or first module) - Modules: Array of hashtables, each containing Name and Version keys for each installed module .Parameter Name Specifies which module type to install. Valid values are 'AWS.Tools' or 'AWS.Tools.Installer'. Defaults to 'AWS.Tools' for backward compatibility. .Parameter ZipPath Path to the zip file (AWS.Tools.zip or AWS.Tools.Installer.zip). .Parameter ModuleNames Array of specific module names to install. If null, installs all modules. .Parameter TargetPath Target installation path for modules. #> function Install-AWSToolsModuleFromZip { [CmdletBinding()] param( [Parameter()] [ValidateSet('AWS.Tools', 'AWS.Tools.Installer')] [string] $Name = 'AWS.Tools', [Parameter(Mandatory)] [string]$ZipPath, [Parameter()] [string[]]$ModuleNames, [Parameter(Mandatory)] [string]$TargetPath ) Begin { Write-Debug ("[$($MyInvocation.MyCommand)] Begin - Name=$Name ZipPath=$ZipPath " + "ModuleNames=($($ModuleNames -join ',')) TargetPath=$TargetPath") } Process { # Get module configuration from general.psd1 $moduleConfig = $script:Config.general.Modules[$Name] if (-not $moduleConfig) { $PSCmdlet.ThrowTerminatingError( [System.Management.Automation.ErrorRecord]::new( ([System.InvalidOperationException]"Configuration not found for module type: $Name"), 'ModuleConfigNotFound', [System.Management.Automation.ErrorCategory]::InvalidOperation, $Name ) ) } $tempPath = [System.IO.Path]::GetTempPath() $randomName = [System.IO.Path]::GetRandomFileName() $tempDir = Join-Path $tempPath $randomName $extractDir = Join-Path $tempDir "AwsTools" try { # Create temporary directory New-Item -ItemType 'Directory' -Path $tempDir -Force -ErrorAction Stop | Out-Null Write-Verbose "[$($MyInvocation.MyCommand)] Expanding $($moduleConfig.ZipFileName) archive..." # Use Expand-ArchivePart when specific modules are requested for AWS.Tools # This avoids extracting the entire archive which saves significant time if ($Name -eq 'AWS.Tools' -and $ModuleNames) { Write-Verbose "[$($MyInvocation.MyCommand)] Using Expand-ArchivePart for specific modules: $($ModuleNames -join ', ')" Expand-ArchivePart -Name $ModuleNames -Path $ZipPath -DestinationPath $extractDir -Force -ErrorAction Stop } else { # Extract the entire archive for AWS.Tools.Installer or when no specific modules are requested Write-Verbose "[$($MyInvocation.MyCommand)] Extracting entire archive" # Temporarily suppress progress if PowerShell version is less than 6 if ($PSVersionTable.PSVersion.Major -lt 6) { # Must set global preference specifically for Expand-Archive $savedProgressPreference = $global:ProgressPreference $global:ProgressPreference = 'SilentlyContinue' try { Expand-Archive -Path $ZipPath -DestinationPath $extractDir -Force -ErrorAction Stop } finally { $global:ProgressPreference = $savedProgressPreference } } else { Expand-Archive -Path $ZipPath -DestinationPath $extractDir -Force -ErrorAction Stop } } $moduleDirectories = @(Get-ChildItem -Path $extractDir -Directory -ErrorAction Stop) if ($ModuleNames) { # Build mandatory modules list - only for AWS.Tools (not AWS.Tools.Installer) # AWS.Tools.Installer.zip only contains the installer module itself $mandatoryModules = @() if ($Name -eq 'AWS.Tools') { # AWS.Tools.Common is required for all AWS.Tools module installations $mandatoryModules += 'AWS.Tools.Common' # Add AWS.Tools rollup module if present in archive $rollupModulePath = Join-Path $extractDir 'AWS.Tools' if (Test-Path $rollupModulePath) { $mandatoryModules += 'AWS.Tools' Write-Verbose ("[$($MyInvocation.MyCommand)] Detected AWS.Tools rollup module " + "in archive, adding to mandatory modules") } } # Combine user-requested modules with mandatory modules (if any) $allModuleNames = @($ModuleNames) + $mandatoryModules | Select-Object -Unique # Use case-insensitive comparison for module names (PowerShell should be case-insensitive) $moduleDirectories = @($moduleDirectories | Where-Object { $moduleName = $_.Name $allModuleNames | Where-Object { $_ -eq $moduleName } -ErrorAction SilentlyContinue }) if ($moduleDirectories.Count -eq 0) { $PSCmdlet.ThrowTerminatingError( [System.Management.Automation.ErrorRecord]::new( ([System.IO.FileNotFoundException]"None of the specified modules were found in the $($moduleConfig.ZipFileName) file."), 'ModulesNotFoundInZip', [System.Management.Automation.ErrorCategory]::ObjectNotFound, $ModuleNames ) ) } # Verify all requested modules were found in archive (case-insensitive) $foundModuleNames = $moduleDirectories.Name $missingModules = @($allModuleNames | Where-Object { $requestedName = $_ -not ($foundModuleNames | Where-Object { $_ -eq $requestedName }) }) if ($missingModules.Count -gt 0) { $PSCmdlet.ThrowTerminatingError( [System.Management.Automation.ErrorRecord]::new( ([System.IO.FileNotFoundException]"Required modules not found in archive: $($missingModules -join ', ')"), 'MandatoryModulesNotFound', [System.Management.Automation.ErrorCategory]::ObjectNotFound, $missingModules ) ) } } # Track installed module versions $installedModuleInfo = @{} $totalModules = $moduleDirectories.Count $currentModule = 0 Write-Verbose "[$($MyInvocation.MyCommand)] Found $totalModules modules to install" foreach ($moduleDir in $moduleDirectories) { $currentModule++ $activityName = if ($Name -eq 'AWS.Tools.Installer') { "Installing AWS Tools Installer Module" } else { "Installing AWS Tools Modules" } $writeProgressParams = @{ Id = 2 ParentId = 1 Activity = $activityName Status = "Processing $($moduleDir.Name)" PercentComplete = (($currentModule / $totalModules) * 100) } # Write progress only if PowerShell version is greater than 6 if ($PSVersionTable.PSVersion.Major -ge 6) { Write-Progress @writeProgressParams } $psd1File = Get-ChildItem -Path $moduleDir.FullName -Filter '*.psd1' -Recurse | Select-Object -First 1 if (-not $psd1File) { $PSCmdlet.ThrowTerminatingError( [System.Management.Automation.ErrorRecord]::new( ([System.IO.FileNotFoundException]"No manifest file found for module $($moduleDir.Name). Installation cannot continue."), 'ManifestNotFound', [System.Management.Automation.ErrorCategory]::ObjectNotFound, $moduleDir.Name ) ) } $pathParts = $psd1File.FullName.Split([System.IO.Path]::DirectorySeparatorChar) # Validate directory structure before extracting version if ($pathParts.Count -lt 2) { $PSCmdlet.ThrowTerminatingError( [System.Management.Automation.ErrorRecord]::new( ([System.InvalidOperationException]"Invalid module structure for $($moduleDir.Name). Expected version subdirectory not found."), 'InvalidModuleStructure', [System.Management.Automation.ErrorCategory]::InvalidData, $moduleDir ) ) } $moduleVersion = $pathParts[-2] # Validate version format - handle both standard versions and semver with prerelease tags try { # Check if it's a semver string with prerelease tag if ($moduleVersion -match '^(\d+\.\d+\.\d+)-.+$') { # Validate the base version part [Version]::Parse($Matches[1]) | Out-Null } else { # Standard version validation [Version]::Parse($moduleVersion) | Out-Null } } catch { $PSCmdlet.ThrowTerminatingError( [System.Management.Automation.ErrorRecord]::new( ([System.ArgumentException]"Invalid version format '$moduleVersion' for module $($moduleDir.Name)."), 'InvalidVersionFormat', [System.Management.Automation.ErrorCategory]::InvalidArgument, $moduleVersion ) ) } $moduleName = $moduleDir.Name # Read the manifest to get the prerelease tag if present # Use Try-Catch to handle cases where manifest doesn't exist (e.g., in unit tests) $fullVersionString = $moduleVersion try { $manifestData = Import-PowerShellDataFile -Path $psd1File.FullName -ErrorAction Stop if ($manifestData.PrivateData.PSData.Prerelease) { $prereleaseTag = $manifestData.PrivateData.PSData.Prerelease $fullVersionString = "$moduleVersion-$prereleaseTag" } Write-Verbose ("[$($MyInvocation.MyCommand)] Installing module $moduleName " + "version $fullVersionString") } catch { # If we can't read the manifest (e.g., in unit tests with mocked structures), # just use the base version from the directory name Write-Verbose ("[$($MyInvocation.MyCommand)] Could not read manifest for $moduleName, " + "using version from directory: $moduleVersion") } # Track per-module versions (use full semver string) $installedModuleInfo[$moduleName] = $fullVersionString $moduleTargetPath = Join-Path $TargetPath $moduleName $versionTargetPath = Join-Path $moduleTargetPath $moduleVersion # Create target directory if (-not (Test-Path $versionTargetPath)) { New-Item -ItemType 'Directory' -Path $versionTargetPath -Force -ErrorAction Stop | Out-Null } # Copy module files $sourceVersionPath = $psd1File.Directory.FullName $copyItemParams = @{ Path = "$sourceVersionPath\*" Destination = $versionTargetPath Recurse = $true Force = $true ErrorAction = 'Stop' } # Temporarily suppress progress just for this operation $savedProgressPreference = $ProgressPreference $ProgressPreference = 'SilentlyContinue' try { Copy-Item @copyItemParams } finally { $ProgressPreference = $savedProgressPreference } $psd1TargetPath = Join-Path $versionTargetPath "$moduleName.psd1" $infoXmlPath = Join-Path $versionTargetPath 'PSGetModuleInfo.xml' # Generate PSGetModuleInfo.xml $newPSGetModuleInfoParams = @{ PsdPath = $psd1TargetPath NormalizedVersion = $moduleVersion } $xml = New-PSGetModuleInfo @newPSGetModuleInfoParams $writePSGetModuleInfoParams = @{ XmlContent = $xml Path = $infoXmlPath } Write-PSGetModuleInfo @writePSGetModuleInfoParams Write-Verbose ("[$($MyInvocation.MyCommand)] Successfully installed " + "$moduleName version $fullVersionString") } # Clear progress indicator $activityName = if ($Name -eq 'AWS.Tools.Installer') { "Installing AWS Tools Installer Module" } else { "Installing AWS Tools Modules" } Write-Progress -Id 2 -Activity $activityName -Completed # Build structured return with Version and Modules properties $modulesArray = @() foreach ($moduleName in $installedModuleInfo.Keys) { $modulesArray += @{ Name = $moduleName Version = $installedModuleInfo[$moduleName] } } # Determine representative version (AWS.Tools > AWS.Tools.Common > first module) if ($installedModuleInfo.ContainsKey('AWS.Tools')) { $representativeVersion = $installedModuleInfo['AWS.Tools'] } elseif ($installedModuleInfo.ContainsKey('AWS.Tools.Common')) { $representativeVersion = $installedModuleInfo['AWS.Tools.Common'] } else { # Used when installing AWS.Tools.Installer $representativeVersion = $installedModuleInfo.Values | Select-Object -First 1 } Write-Verbose ("[$($MyInvocation.MyCommand)] Representative version: $representativeVersion") # Return structured result with Version, VersionString, and Modules properties # VersionString preserves the full semver string including prerelease tags for display return @{ Version = $representativeVersion VersionString = $representativeVersion Modules = $modulesArray } } finally { if (Test-Path $tempDir) { $removeItemParams = @{ Path = $tempDir Recurse = $true Force = $true ErrorAction = 'SilentlyContinue' } # Temporarily suppress progress just for this operation $savedProgressPreference = $ProgressPreference $ProgressPreference = 'SilentlyContinue' try { Remove-Item @removeItemParams } finally { $ProgressPreference = $savedProgressPreference } } } } End { Write-Debug "[$($MyInvocation.MyCommand)] End" } } # SIG # Begin signature block # MIIutAYJKoZIhvcNAQcCoIIupTCCLqECAQExDzANBglghkgBZQMEAgEFADB5Bgor # BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCBpUBajU3x9D2Mm # 6InaSUKiNEEV6iPqdPc4UT5QSbw4oaCCE+owggXAMIIEqKADAgECAhAP0bvKeWvX # +N1MguEKmpYxMA0GCSqGSIb3DQEBCwUAMGwxCzAJBgNVBAYTAlVTMRUwEwYDVQQK # EwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xKzApBgNV # BAMTIkRpZ2lDZXJ0IEhpZ2ggQXNzdXJhbmNlIEVWIFJvb3QgQ0EwHhcNMjIwMTEz # MDAwMDAwWhcNMzExMTA5MjM1OTU5WjBiMQswCQYDVQQGEwJVUzEVMBMGA1UEChMM # RGlnaUNlcnQgSW5jMRkwFwYDVQQLExB3d3cuZGlnaWNlcnQuY29tMSEwHwYDVQQD # ExhEaWdpQ2VydCBUcnVzdGVkIFJvb3QgRzQwggIiMA0GCSqGSIb3DQEBAQUAA4IC # DwAwggIKAoICAQC/5pBzaN675F1KPDAiMGkz7MKnJS7JIT3yithZwuEppz1Yq3aa # za57G4QNxDAf8xukOBbrVsaXbR2rsnnyyhHS5F/WBTxSD1Ifxp4VpX6+n6lXFllV # cq9ok3DCsrp1mWpzMpTREEQQLt+C8weE5nQ7bXHiLQwb7iDVySAdYyktzuxeTsiT # +CFhmzTrBcZe7FsavOvJz82sNEBfsXpm7nfISKhmV1efVFiODCu3T6cw2Vbuyntd # 463JT17lNecxy9qTXtyOj4DatpGYQJB5w3jHtrHEtWoYOAMQjdjUN6QuBX2I9YI+ # EJFwq1WCQTLX2wRzKm6RAXwhTNS8rhsDdV14Ztk6MUSaM0C/CNdaSaTC5qmgZ92k # J7yhTzm1EVgX9yRcRo9k98FpiHaYdj1ZXUJ2h4mXaXpI8OCiEhtmmnTK3kse5w5j # rubU75KSOp493ADkRSWJtppEGSt+wJS00mFt6zPZxd9LBADMfRyVw4/3IbKyEbe7 # f/LVjHAsQWCqsWMYRJUadmJ+9oCw++hkpjPRiQfhvbfmQ6QYuKZ3AeEPlAwhHbJU # KSWJbOUOUlFHdL4mrLZBdd56rF+NP8m800ERElvlEFDrMcXKchYiCd98THU/Y+wh # X8QgUWtvsauGi0/C1kVfnSD8oR7FwI+isX4KJpn15GkvmB0t9dmpsh3lGwIDAQAB # o4IBZjCCAWIwDwYDVR0TAQH/BAUwAwEB/zAdBgNVHQ4EFgQU7NfjgtJxXWRM3y5n # P+e6mK4cD08wHwYDVR0jBBgwFoAUsT7DaQP4v0cB1JgmGggC72NkK8MwDgYDVR0P # AQH/BAQDAgGGMBMGA1UdJQQMMAoGCCsGAQUFBwMDMH8GCCsGAQUFBwEBBHMwcTAk # BggrBgEFBQcwAYYYaHR0cDovL29jc3AuZGlnaWNlcnQuY29tMEkGCCsGAQUFBzAC # hj1odHRwOi8vY2FjZXJ0cy5kaWdpY2VydC5jb20vRGlnaUNlcnRIaWdoQXNzdXJh # bmNlRVZSb290Q0EuY3J0MEsGA1UdHwREMEIwQKA+oDyGOmh0dHA6Ly9jcmwzLmRp # Z2ljZXJ0LmNvbS9EaWdpQ2VydEhpZ2hBc3N1cmFuY2VFVlJvb3RDQS5jcmwwHAYD # VR0gBBUwEzAHBgVngQwBAzAIBgZngQwBBAEwDQYJKoZIhvcNAQELBQADggEBAEHx # qRH0DxNHecllao3A7pgEpMbjDPKisedfYk/ak1k2zfIe4R7sD+EbP5HU5A/C5pg0 # /xkPZigfT2IxpCrhKhO61z7H0ZL+q93fqpgzRh9Onr3g7QdG64AupP2uU7SkwaT1 # IY1rzAGt9Rnu15ClMlIr28xzDxj4+87eg3Gn77tRWwR2L62t0+od/P1Tk+WMieNg # GbngLyOOLFxJy34riDkruQZhiPOuAnZ2dMFkkbiJUZflhX0901emWG4f7vtpYeJa # 3Cgh6GO6Ps9W7Zrk9wXqyvPsEt84zdp7PiuTUy9cUQBY3pBIowrHC/Q7bVUx8ALM # R3eWUaNetbxcyEMRoacwggawMIIEmKADAgECAhAIrUCyYNKcTJ9ezam9k67ZMA0G # 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 # eE4wggduMIIFVqADAgECAhADmooc3J4LAjb9FLn2g0iAMA0GCSqGSIb3DQEBCwUA # MGkxCzAJBgNVBAYTAlVTMRcwFQYDVQQKEw5EaWdpQ2VydCwgSW5jLjFBMD8GA1UE # AxM4RGlnaUNlcnQgVHJ1c3RlZCBHNCBDb2RlIFNpZ25pbmcgUlNBNDA5NiBTSEEz # ODQgMjAyMSBDQTEwHhcNMjUwNjAzMDAwMDAwWhcNMjYwNjAyMjM1OTU5WjCB9jET # MBEGCysGAQQBgjc8AgEDEwJVUzEZMBcGCysGAQQBgjc8AgECEwhEZWxhd2FyZTEd # MBsGA1UEDwwUUHJpdmF0ZSBPcmdhbml6YXRpb24xEDAOBgNVBAUTBzQxNTI5NTQx # CzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdTZWF0 # dGxlMSIwIAYDVQQKExlBbWF6b24gV2ViIFNlcnZpY2VzLCBJbmMuMRcwFQYDVQQL # Ew5TREtzIGFuZCBUb29sczEiMCAGA1UEAxMZQW1hem9uIFdlYiBTZXJ2aWNlcywg # SW5jLjCCAaIwDQYJKoZIhvcNAQEBBQADggGPADCCAYoCggGBAKc7ffRFlFB89cU3 # j76EOoK8sFXFunuUhvU6LMeAzgO2fjlMn2l3WJBPZmrHXIicBHmejbJDRgMc+FPd # oSUR6ZsuVX08txcdv8yHqC+u2BHrj+GRakq7WF148X1PR8kOhkon+qNSh+cxoIHm # m+Lxg9kZuddkr1m+Fuv79VGj3E9xp4YjbhvVEVbxLs86p812WgVz3eoqvpNXBC0m # H4eM3FUmNHcDA6ubRR/qh3OGYQHyQs4PeMOGACqcySO+XNRKxMNxIzf+fkQU8cuR # A8eYl6s0pQ+KlJudGt2psY+5nkTU9TOC99wvRmfuuG+jpG7vEMsuGy26wCOp7hry # k5kinq/EFImAxdx1WQ3pJuk+rcuSnuOcSieW+zSKEczJOCsbemDbF+9P5Iv0f0nx # kHPYrrjdgValyxosQENsHeq1HztsE46IbBq2S4I9kGzZcaal5hKT/Yxcom9tBfXq # oNs1KaL54mC4ddBGx07kAGE0kiHpY5pr4al2lGpOmBH4B5xKmwIDAQABo4ICAjCC # Af4wHwYDVR0jBBgwFoAUaDfg67Y7+F8Rhvv+YXsIiGX0TkIwHQYDVR0OBBYEFALy # CdwoelU9i0FZLXOC5NXeXmVqMD0GA1UdIAQ2MDQwMgYFZ4EMAQMwKTAnBggrBgEF # BQcCARYbaHR0cDovL3d3dy5kaWdpY2VydC5jb20vQ1BTMA4GA1UdDwEB/wQEAwIH # gDATBgNVHSUEDDAKBggrBgEFBQcDAzCBtQYDVR0fBIGtMIGqMFOgUaBPhk1odHRw # Oi8vY3JsMy5kaWdpY2VydC5jb20vRGlnaUNlcnRUcnVzdGVkRzRDb2RlU2lnbmlu # Z1JTQTQwOTZTSEEzODQyMDIxQ0ExLmNybDBToFGgT4ZNaHR0cDovL2NybDQuZGln # aWNlcnQuY29tL0RpZ2lDZXJ0VHJ1c3RlZEc0Q29kZVNpZ25pbmdSU0E0MDk2U0hB # Mzg0MjAyMUNBMS5jcmwwgZQGCCsGAQUFBwEBBIGHMIGEMCQGCCsGAQUFBzABhhho # dHRwOi8vb2NzcC5kaWdpY2VydC5jb20wXAYIKwYBBQUHMAKGUGh0dHA6Ly9jYWNl # cnRzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydFRydXN0ZWRHNENvZGVTaWduaW5nUlNB # NDA5NlNIQTM4NDIwMjFDQTEuY3J0MAkGA1UdEwQCMAAwDQYJKoZIhvcNAQELBQAD # ggIBACHYo9QgnoH4+yG1TQT8lxxsYPuy3+FiWd2sZxn/4dwBautbZSLaP5yAKlE0 # o7SFQq5IYV2FVA4c5LwJf3eoF0/8+Bh+SMmAHTteqHcAJ3VK8+VETL/qr9kFQx8j # cMA3kF2OOaLVWuB5ffEsGCRqCnS2r8r2lCstb/oEqJLyUXkqEDbyJxgkJ0vQp1XD # agCi68ujhDBwVQPoHmBux2V8TWaOuAT/5kZUJsqwUhBTxYSCKVdIYYkRppaUemJ0 # oULhtxeg1M5zumgLf432qyk6xchohECA1QSXsPflhLZ+eWk3RneSt0fAV/caEBma # mfVj4OysofDA4bCgZSWFk/16O4PEMsLZhR8FCnFr5lxFYKUgAcnVczo75KSBKp2r # aoUVAWIw9w1Z1NQljqfZbFUxWCIWLjmXmEbfy3IXeEpvlXidjPVwdqwbleyl0WR/ # ENGBEiy+I4a/ezu+HoA5zK+1b7INjLgxk8m5MnD75k6Pq+XctyTkzEh6F4Whhq7s # ck16llf7N+i28flIkqzcXuSHxeOZUZb2VNDIcCXiapaxS7cslbP4wafWGEYs2SN3 # +Q398T0fI7uJ4ATp58qdIp6w7Z9PomrVKE0LQMC1P1j9VDiCv7VqnU1cM/2uPpzA # Lop57Q8mTPr7Mh7nl0/QusIr8V/o6XWwRynumecNqn1oeMlaMYIaIDCCGhwCAQEw # fTBpMQswCQYDVQQGEwJVUzEXMBUGA1UEChMORGlnaUNlcnQsIEluYy4xQTA/BgNV # BAMTOERpZ2lDZXJ0IFRydXN0ZWQgRzQgQ29kZSBTaWduaW5nIFJTQTQwOTYgU0hB # Mzg0IDIwMjEgQ0ExAhADmooc3J4LAjb9FLn2g0iAMA0GCWCGSAFlAwQCAQUAoHww # EAYKKwYBBAGCNwIBDDECMAAwGQYJKoZIhvcNAQkDMQwGCisGAQQBgjcCAQQwHAYK # KwYBBAGCNwIBCzEOMAwGCisGAQQBgjcCARUwLwYJKoZIhvcNAQkEMSIEIMBBmwZW # dtuj1572v+kCTzRA73+SzoBxOPFtvQPP7BjjMA0GCSqGSIb3DQEBAQUABIIBgJK8 # aR1iBXnVc2dtK41BcMlepDI5qqVQcUtOnCA41eP0aEcvfyDJjSp/iv0UZjLbdma7 # 7ES4Ms0jr9QONBQBMduJmKcqLIZqQYQN+4jOG3zg808BIedc3QoS1SERBFTd2ahK # oogKnLdOAOkkmi3rRNSLpj8OPp/jl4z7e3vnBo6FVmUZscG4YiGOPC/KqGHJK83P # CCcPyXPRSI/pKoKKpETxkTc0HmUalv9gjaHWi8Q5yp6wUU2ox7kH9P0SUaq8SAM6 # e5yoMbKzv8RTlE0+NesN4FC86DSbb8D0gIJE0w64pYLil90qQ06dK35RwLZc5a4/ # 9L8eSC9V3oUYFO9LeX2LbMm4KcUDmcLwTnzONjnfIjwQE4RZdXzzDfVCKSweBbY9 # 07iw+9V24iMkocfRVC3aRx8m+nHzr44ANgeEt9/UvaAhyaVG0cWywHrYItngiSD2 # ph+fwb9jmJqTURgXox2ZEY5DgzPhoF+NrTiW8F9YEWDF+wgiPkm7IawErTS6naGC # F3YwghdyBgorBgEEAYI3AwMBMYIXYjCCF14GCSqGSIb3DQEHAqCCF08wghdLAgED # MQ8wDQYJYIZIAWUDBAIBBQAwdwYLKoZIhvcNAQkQAQSgaARmMGQCAQEGCWCGSAGG # /WwHATAxMA0GCWCGSAFlAwQCAQUABCDkYOxi+RS7j28z4jePfSUPrOSESxyxxga9 # y6Vql8WE9AIQOvGyiJRcnAnMDs8TVgtK1xgPMjAyNjAyMTQwMzEzNTFaoIITOjCC # Bu0wggTVoAMCAQICEAqA7xhLjfEFgtHEdqeVdGgwDQYJKoZIhvcNAQELBQAwaTEL # MAkGA1UEBhMCVVMxFzAVBgNVBAoTDkRpZ2lDZXJ0LCBJbmMuMUEwPwYDVQQDEzhE # aWdpQ2VydCBUcnVzdGVkIEc0IFRpbWVTdGFtcGluZyBSU0E0MDk2IFNIQTI1NiAy # MDI1IENBMTAeFw0yNTA2MDQwMDAwMDBaFw0zNjA5MDMyMzU5NTlaMGMxCzAJBgNV # BAYTAlVTMRcwFQYDVQQKEw5EaWdpQ2VydCwgSW5jLjE7MDkGA1UEAxMyRGlnaUNl # cnQgU0hBMjU2IFJTQTQwOTYgVGltZXN0YW1wIFJlc3BvbmRlciAyMDI1IDEwggIi # MA0GCSqGSIb3DQEBAQUAA4ICDwAwggIKAoICAQDQRqwtEsae0OquYFazK1e6b1H/ # hnAKAd/KN8wZQjBjMqiZ3xTWcfsLwOvRxUwXcGx8AUjni6bz52fGTfr6PHRNv6T7 # zsf1Y/E3IU8kgNkeECqVQ+3bzWYesFtkepErvUSbf+EIYLkrLKd6qJnuzK8Vcn0D # vbDMemQFoxQ2Dsw4vEjoT1FpS54dNApZfKY61HAldytxNM89PZXUP/5wWWURK+If # xiOg8W9lKMqzdIo7VA1R0V3Zp3DjjANwqAf4lEkTlCDQ0/fKJLKLkzGBTpx6EYev # vOi7XOc4zyh1uSqgr6UnbksIcFJqLbkIXIPbcNmA98Oskkkrvt6lPAw/p4oDSRZr # eiwB7x9ykrjS6GS3NR39iTTFS+ENTqW8m6THuOmHHjQNC3zbJ6nJ6SXiLSvw4Smz # 8U07hqF+8CTXaETkVWz0dVVZw7knh1WZXOLHgDvundrAtuvz0D3T+dYaNcwafsVC # GZKUhQPL1naFKBy1p6llN3QgshRta6Eq4B40h5avMcpi54wm0i2ePZD5pPIssosz # QyF4//3DoK2O65Uck5Wggn8O2klETsJ7u8xEehGifgJYi+6I03UuT1j7FnrqVrOz # aQoVJOeeStPeldYRNMmSF3voIgMFtNGh86w3ISHNm0IaadCKCkUe2LnwJKa8TIlw # CUNVwppwn4D3/Pt5pwIDAQABo4IBlTCCAZEwDAYDVR0TAQH/BAIwADAdBgNVHQ4E # FgQU5Dv88jHt/f3X85FxYxlQQ89hjOgwHwYDVR0jBBgwFoAU729TSunkBnx6yuKQ # VvYv1Ensy04wDgYDVR0PAQH/BAQDAgeAMBYGA1UdJQEB/wQMMAoGCCsGAQUFBwMI # MIGVBggrBgEFBQcBAQSBiDCBhTAkBggrBgEFBQcwAYYYaHR0cDovL29jc3AuZGln # aWNlcnQuY29tMF0GCCsGAQUFBzAChlFodHRwOi8vY2FjZXJ0cy5kaWdpY2VydC5j # b20vRGlnaUNlcnRUcnVzdGVkRzRUaW1lU3RhbXBpbmdSU0E0MDk2U0hBMjU2MjAy # NUNBMS5jcnQwXwYDVR0fBFgwVjBUoFKgUIZOaHR0cDovL2NybDMuZGlnaWNlcnQu # Y29tL0RpZ2lDZXJ0VHJ1c3RlZEc0VGltZVN0YW1waW5nUlNBNDA5NlNIQTI1NjIw # MjVDQTEuY3JsMCAGA1UdIAQZMBcwCAYGZ4EMAQQCMAsGCWCGSAGG/WwHATANBgkq # hkiG9w0BAQsFAAOCAgEAZSqt8RwnBLmuYEHs0QhEnmNAciH45PYiT9s1i6UKtW+F # ERp8FgXRGQ/YAavXzWjZhY+hIfP2JkQ38U+wtJPBVBajYfrbIYG+Dui4I4PCvHpQ # uPqFgqp1PzC/ZRX4pvP/ciZmUnthfAEP1HShTrY+2DE5qjzvZs7JIIgt0GCFD9kt # x0LxxtRQ7vllKluHWiKk6FxRPyUPxAAYH2Vy1lNM4kzekd8oEARzFAWgeW3az2xe # jEWLNN4eKGxDJ8WDl/FQUSntbjZ80FU3i54tpx5F/0Kr15zW/mJAxZMVBrTE2oi0 # fcI8VMbtoRAmaaslNXdCG1+lqvP4FbrQ6IwSBXkZagHLhFU9HCrG/syTRLLhAezu # /3Lr00GrJzPQFnCEH1Y58678IgmfORBPC1JKkYaEt2OdDh4GmO0/5cHelAK2/gTl # QJINqDr6JfwyYHXSd+V08X1JUPvB4ILfJdmL+66Gp3CSBXG6IwXMZUXBhtCyIaeh # r0XkBoDIGMUG1dUtwq1qmcwbdUfcSYCn+OwncVUXf53VJUNOaMWMts0VlRYxe5nK # +At+DI96HAlXHAL5SlfYxJ7La54i71McVWRP66bW+yERNpbJCjyCYG2j+bdpxo/1 # Cy4uPcU3AWVPGrbn5PhDBf3Froguzzhk++ami+r3Qrx5bIbY3TVzgiFI7Gq3zWcw # gga0MIIEnKADAgECAhANx6xXBf8hmS5AQyIMOkmGMA0GCSqGSIb3DQEBCwUAMGIx # CzAJBgNVBAYTAlVTMRUwEwYDVQQKEwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3 # dy5kaWdpY2VydC5jb20xITAfBgNVBAMTGERpZ2lDZXJ0IFRydXN0ZWQgUm9vdCBH # NDAeFw0yNTA1MDcwMDAwMDBaFw0zODAxMTQyMzU5NTlaMGkxCzAJBgNVBAYTAlVT # MRcwFQYDVQQKEw5EaWdpQ2VydCwgSW5jLjFBMD8GA1UEAxM4RGlnaUNlcnQgVHJ1 # c3RlZCBHNCBUaW1lU3RhbXBpbmcgUlNBNDA5NiBTSEEyNTYgMjAyNSBDQTEwggIi # MA0GCSqGSIb3DQEBAQUAA4ICDwAwggIKAoICAQC0eDHTCphBcr48RsAcrHXbo0Zo # dLRRF51NrY0NlLWZloMsVO1DahGPNRcybEKq+RuwOnPhof6pvF4uGjwjqNjfEvUi # 6wuim5bap+0lgloM2zX4kftn5B1IpYzTqpyFQ/4Bt0mAxAHeHYNnQxqXmRinvuNg # xVBdJkf77S2uPoCj7GH8BLuxBG5AvftBdsOECS1UkxBvMgEdgkFiDNYiOTx4OtiF # cMSkqTtF2hfQz3zQSku2Ws3IfDReb6e3mmdglTcaarps0wjUjsZvkgFkriK9tUKJ # m/s80FiocSk1VYLZlDwFt+cVFBURJg6zMUjZa/zbCclF83bRVFLeGkuAhHiGPMvS # GmhgaTzVyhYn4p0+8y9oHRaQT/aofEnS5xLrfxnGpTXiUOeSLsJygoLPp66bkDX1 # ZlAeSpQl92QOMeRxykvq6gbylsXQskBBBnGy3tW/AMOMCZIVNSaz7BX8VtYGqLt9 # MmeOreGPRdtBx3yGOP+rx3rKWDEJlIqLXvJWnY0v5ydPpOjL6s36czwzsucuoKs7 # Yk/ehb//Wx+5kMqIMRvUBDx6z1ev+7psNOdgJMoiwOrUG2ZdSoQbU2rMkpLiQ6bG # RinZbI4OLu9BMIFm1UUl9VnePs6BaaeEWvjJSjNm2qA+sdFUeEY0qVjPKOWug/G6 # X5uAiynM7Bu2ayBjUwIDAQABo4IBXTCCAVkwEgYDVR0TAQH/BAgwBgEB/wIBADAd # BgNVHQ4EFgQU729TSunkBnx6yuKQVvYv1Ensy04wHwYDVR0jBBgwFoAU7NfjgtJx # XWRM3y5nP+e6mK4cD08wDgYDVR0PAQH/BAQDAgGGMBMGA1UdJQQMMAoGCCsGAQUF # BwMIMHcGCCsGAQUFBwEBBGswaTAkBggrBgEFBQcwAYYYaHR0cDovL29jc3AuZGln # aWNlcnQuY29tMEEGCCsGAQUFBzAChjVodHRwOi8vY2FjZXJ0cy5kaWdpY2VydC5j # b20vRGlnaUNlcnRUcnVzdGVkUm9vdEc0LmNydDBDBgNVHR8EPDA6MDigNqA0hjJo # dHRwOi8vY3JsMy5kaWdpY2VydC5jb20vRGlnaUNlcnRUcnVzdGVkUm9vdEc0LmNy # bDAgBgNVHSAEGTAXMAgGBmeBDAEEAjALBglghkgBhv1sBwEwDQYJKoZIhvcNAQEL # BQADggIBABfO+xaAHP4HPRF2cTC9vgvItTSmf83Qh8WIGjB/T8ObXAZz8OjuhUxj # aaFdleMM0lBryPTQM2qEJPe36zwbSI/mS83afsl3YTj+IQhQE7jU/kXjjytJgnn0 # hvrV6hqWGd3rLAUt6vJy9lMDPjTLxLgXf9r5nWMQwr8Myb9rEVKChHyfpzee5kH0 # F8HABBgr0UdqirZ7bowe9Vj2AIMD8liyrukZ2iA/wdG2th9y1IsA0QF8dTXqvcnT # mpfeQh35k5zOCPmSNq1UH410ANVko43+Cdmu4y81hjajV/gxdEkMx1NKU4uHQcKf # ZxAvBAKqMVuqte69M9J6A47OvgRaPs+2ykgcGV00TYr2Lr3ty9qIijanrUR3anzE # wlvzZiiyfTPjLbnFRsjsYg39OlV8cipDoq7+qNNjqFzeGxcytL5TTLL4ZaoBdqbh # OhZ3ZRDUphPvSRmMThi0vw9vODRzW6AxnJll38F0cuJG7uEBYTptMSbhdhGQDpOX # gpIUsWTjd6xpR6oaQf/DJbg3s6KCLPAlZ66RzIg9sC+NJpud/v4+7RWsWCiKi9EO # LLHfMR2ZyJ/+xhCx9yHbxtl5TPau1j/1MIDpMPx0LckTetiSuEtQvLsNz3Qbp7wG # WqbIiOWCnb5WqxL3/BAPvIXKUjPSxyZsq8WhbaM2tszWkPZPubdcMIIFjTCCBHWg # AwIBAgIQDpsYjvnQLefv21DiCEAYWjANBgkqhkiG9w0BAQwFADBlMQswCQYDVQQG # EwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5jMRkwFwYDVQQLExB3d3cuZGlnaWNl # cnQuY29tMSQwIgYDVQQDExtEaWdpQ2VydCBBc3N1cmVkIElEIFJvb3QgQ0EwHhcN # MjIwODAxMDAwMDAwWhcNMzExMTA5MjM1OTU5WjBiMQswCQYDVQQGEwJVUzEVMBMG # A1UEChMMRGlnaUNlcnQgSW5jMRkwFwYDVQQLExB3d3cuZGlnaWNlcnQuY29tMSEw # HwYDVQQDExhEaWdpQ2VydCBUcnVzdGVkIFJvb3QgRzQwggIiMA0GCSqGSIb3DQEB # AQUAA4ICDwAwggIKAoICAQC/5pBzaN675F1KPDAiMGkz7MKnJS7JIT3yithZwuEp # pz1Yq3aaza57G4QNxDAf8xukOBbrVsaXbR2rsnnyyhHS5F/WBTxSD1Ifxp4VpX6+ # n6lXFllVcq9ok3DCsrp1mWpzMpTREEQQLt+C8weE5nQ7bXHiLQwb7iDVySAdYykt # zuxeTsiT+CFhmzTrBcZe7FsavOvJz82sNEBfsXpm7nfISKhmV1efVFiODCu3T6cw # 2Vbuyntd463JT17lNecxy9qTXtyOj4DatpGYQJB5w3jHtrHEtWoYOAMQjdjUN6Qu # BX2I9YI+EJFwq1WCQTLX2wRzKm6RAXwhTNS8rhsDdV14Ztk6MUSaM0C/CNdaSaTC # 5qmgZ92kJ7yhTzm1EVgX9yRcRo9k98FpiHaYdj1ZXUJ2h4mXaXpI8OCiEhtmmnTK # 3kse5w5jrubU75KSOp493ADkRSWJtppEGSt+wJS00mFt6zPZxd9LBADMfRyVw4/3 # IbKyEbe7f/LVjHAsQWCqsWMYRJUadmJ+9oCw++hkpjPRiQfhvbfmQ6QYuKZ3AeEP # lAwhHbJUKSWJbOUOUlFHdL4mrLZBdd56rF+NP8m800ERElvlEFDrMcXKchYiCd98 # THU/Y+whX8QgUWtvsauGi0/C1kVfnSD8oR7FwI+isX4KJpn15GkvmB0t9dmpsh3l # GwIDAQABo4IBOjCCATYwDwYDVR0TAQH/BAUwAwEB/zAdBgNVHQ4EFgQU7NfjgtJx # XWRM3y5nP+e6mK4cD08wHwYDVR0jBBgwFoAUReuir/SSy4IxLVGLp6chnfNtyA8w # DgYDVR0PAQH/BAQDAgGGMHkGCCsGAQUFBwEBBG0wazAkBggrBgEFBQcwAYYYaHR0 # cDovL29jc3AuZGlnaWNlcnQuY29tMEMGCCsGAQUFBzAChjdodHRwOi8vY2FjZXJ0 # cy5kaWdpY2VydC5jb20vRGlnaUNlcnRBc3N1cmVkSURSb290Q0EuY3J0MEUGA1Ud # HwQ+MDwwOqA4oDaGNGh0dHA6Ly9jcmwzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydEFz # c3VyZWRJRFJvb3RDQS5jcmwwEQYDVR0gBAowCDAGBgRVHSAAMA0GCSqGSIb3DQEB # DAUAA4IBAQBwoL9DXFXnOF+go3QbPbYW1/e/Vwe9mqyhhyzshV6pGrsi+IcaaVQi # 7aSId229GhT0E0p6Ly23OO/0/4C5+KH38nLeJLxSA8hO0Cre+i1Wz/n096wwepqL # sl7Uz9FDRJtDIeuWcqFItJnLnU+nBgMTdydE1Od/6Fmo8L8vC6bp8jQ87PcDx4eo # 0kxAGTVGamlUsLihVo7spNU96LHc/RzY9HdaXFSMb++hUD38dglohJ9vytsgjTVg # HAIDyyCwrFigDkBjxZgiwbJZ9VVrzyerbHbObyMt9H5xaiNrIv8SuFQtJ37YOtnw # toeW/VvRXKwYw02fc7cBqZ9Xql4o4rmUMYIDfDCCA3gCAQEwfTBpMQswCQYDVQQG # EwJVUzEXMBUGA1UEChMORGlnaUNlcnQsIEluYy4xQTA/BgNVBAMTOERpZ2lDZXJ0 # IFRydXN0ZWQgRzQgVGltZVN0YW1waW5nIFJTQTQwOTYgU0hBMjU2IDIwMjUgQ0Ex # AhAKgO8YS43xBYLRxHanlXRoMA0GCWCGSAFlAwQCAQUAoIHRMBoGCSqGSIb3DQEJ # AzENBgsqhkiG9w0BCRABBDAcBgkqhkiG9w0BCQUxDxcNMjYwMjE0MDMxMzUxWjAr # BgsqhkiG9w0BCRACDDEcMBowGDAWBBTdYjCshgotMGvaOLFoeVIwB/tBfjAvBgkq # hkiG9w0BCQQxIgQgfglL88VIn0Mvh1iVC5y83HYhh42kyLV6urXG6Hm7rnQwNwYL # KoZIhvcNAQkQAi8xKDAmMCQwIgQgSqA/oizXXITFXJOPgo5na5yuyrM/420mmqM0 # 8UYRCjMwDQYJKoZIhvcNAQEBBQAEggIAslxAhX5SNCr0z6/tl6M7zxBaZnIbwb5z # XpqbTsw4GiHf7DlPSDt9MJU6OaGFlMIkZDxzG+i1I567M65KBAHPLxFgn1SPsaVi # 1DU/l9hRsZPpIp9uiqoxSmzan2ViDyan3PSXsu+TKKitEwZ/ya22glGcSMjtqUlE # LxFQC8eF+cNJohOn021V+LpI6vA7CFKvDAvqq/qQnSlGvAwNejwNld/d08qClPPi # 5G8bregxRK6xMNZt5FoTpqBK2kLTGMXT0ksUUDMFJGkay0JEjPFqKoWVZTevofjq # WtDpzlErTti5m7MOxJFDUxCvT5j4KP8DVlfk/k6pETmHSvNGlRBozT6m7Wu+g8vv # 4J0nsZooBXZ52q48fcep8LnaExSnwY5gf+fMgkaXT7DYxAK33SN/cQW9DyIxX5GP # YwnTaoYr9J88zXMf9+nCtEjYmNGmsc4E5l0FXjwxvQkzORjFhO4RkMYWC7HTYSRc # N3EDZLIBVLVDnb+1YwAY5ein/q2uc7kS5NqoP1ECsfQN+gFMah/CBLevbvVCC1Rf # Oiw1HyKjGfvqZLj4jLZCdUtetlPxA6F4Pv01Bi+vLIXbZsOdDlS9rekeevVJ3GYZ # nJCXicp04bq1Q6AGCEBsMkxNsl+6xDdKi9Rh7NZjrvQJ/e/cP5f3y0zIu57v2X7d # CzK3fgr9b0g= # SIG # End signature block |