Private/Tools/Test-Tooling.ps1
|
function Test-Tooling { [CmdletBinding(SupportsShouldProcess = $true)] param( [Parameter(Mandatory = $false)] [ValidateSet("PowerShell", "Git", "AzureCli", "AzureEnvVars", "AzureCliOrEnvVars", "AzureLogin", "AlzModule", "AlzModuleVersion", "YamlModule", "YamlModuleAutoInstall", "GitHubCli", "AzureDevOpsCli", "NetworkConnectivity")] [string[]]$Checks = @("PowerShell", "Git", "AzureCliOrEnvVars", "AzureLogin", "AlzModule", "AlzModuleVersion"), [Parameter(Mandatory = $false)] [switch]$destroy, [Parameter(Mandatory = $false)] [int]$HttpRequestMaxRetryCount = 0, [Parameter(Mandatory = $false)] [int]$HttpRequestRetryIntervalSeconds = 3, [Parameter(Mandatory = $false)] [int]$HttpRequestTimeoutSeconds = 10 ) $checkResults = @() $hasFailure = $false $azCliInstalledButNotLoggedIn = $false $currentScope = "CurrentUser" # Check PowerShell version if ($Checks -contains "PowerShell") { $result = Test-PowerShellVersion $checkResults += $result.Results if ($result.HasFailure) { $hasFailure = $true } } # Check Git installation if ($Checks -contains "Git") { $result = Test-GitInstallation $checkResults += $result.Results if ($result.HasFailure) { $hasFailure = $true } } # Check Azure Environment Variables only if ($Checks -contains "AzureEnvVars") { $result = Test-AzureEnvironmentVariable $checkResults += $result.Results if ($result.HasFailure) { $hasFailure = $true } } # Check Azure CLI only (used by Remove-PlatformLandingZone) if ($Checks -contains "AzureCli") { $requireLogin = $Checks -contains "AzureLogin" $result = Test-AzureCli -RequireLogin $requireLogin $checkResults += $result.Results if ($result.HasFailure) { $hasFailure = $true } if ($result.AzCliInstalledButNotLoggedIn) { $azCliInstalledButNotLoggedIn = $true } } # Check Azure CLI or Environment Variables (used by Deploy-Accelerator) # If env vars are valid, skip CLI check; otherwise check CLI if ($Checks -contains "AzureCliOrEnvVars") { $envResult = Test-AzureEnvironmentVariable $checkResults += $envResult.Results if ($envResult.HasFailure) { $hasFailure = $true } # Only check CLI if env vars are not valid if (-not $envResult.EnvVarsValid) { $requireLogin = $Checks -contains "AzureLogin" $cliResult = Test-AzureCli -RequireLogin $requireLogin $checkResults += $cliResult.Results if ($cliResult.HasFailure) { $hasFailure = $true } if ($cliResult.AzCliInstalledButNotLoggedIn) { $azCliInstalledButNotLoggedIn = $true } } } # Check ALZ Module if ($Checks -contains "AlzModule") { $checkVersion = $Checks -contains "AlzModuleVersion" $result = Test-AlzModule -CheckVersion $checkVersion -AllowContinueOnFailure:$destroy.IsPresent $checkResults += $result.Results if ($result.HasFailure) { $hasFailure = $true } if ($result.CurrentScope) { $currentScope = $result.CurrentScope } } # Check YAML Module if ($Checks -contains "YamlModule") { $autoInstall = $Checks -contains "YamlModuleAutoInstall" $result = Test-YamlModule -AutoInstall $autoInstall -Scope $currentScope $checkResults += $result.Results if ($result.HasFailure) { $hasFailure = $true } } # Check GitHub CLI if ($Checks -contains "GitHubCli") { $result = Test-GitHubCli $checkResults += $result.Results if ($result.HasFailure) { $hasFailure = $true } } # Check Azure DevOps CLI if ($Checks -contains "AzureDevOpsCli") { $result = Test-AzureDevOpsCli $checkResults += $result.Results if ($result.HasFailure) { $hasFailure = $true } } # Check Network Connectivity if ($Checks -contains "NetworkConnectivity") { $result = Test-NetworkConnectivity -HttpRequestMaxRetryCount $HttpRequestMaxRetryCount -HttpRequestRetryIntervalSeconds $HttpRequestRetryIntervalSeconds -HttpRequestTimeoutSeconds $HttpRequestTimeoutSeconds $checkResults += $result.Results if ($result.HasFailure) { $hasFailure = $true } } # Display results Write-Verbose "Showing check results" Write-Verbose $(ConvertTo-Json $checkResults -Depth 100) $checkResults | ForEach-Object {[PSCustomObject]$_} | Format-Table -Property @{ Label = "Check Result"; Expression = { switch ($_.result) { 'Success' { $color = "92"; break } 'Failure' { $color = "91"; break } 'Warning' { $color = "93"; break } default { $color = "0" } } $e = [char]27 "$e[${color}m$($_.result)${e}[0m" } }, @{ Label = "Check Details"; Expression = {$_.message} } -AutoSize -Wrap | Out-Host if($hasFailure) { Write-ToConsoleLog "Accelerator software requirements have no been met, please review and install the missing software." -IsError Write-ToConsoleLog "Cannot continue with Deployment..." -IsError throw "Accelerator software requirements have no been met, please review and install the missing software." } return @{ AzCliInstalledButNotLoggedIn = $azCliInstalledButNotLoggedIn } } # SIG # Begin signature block # MIIndAYJKoZIhvcNAQcCoIInZTCCJ2ECAQExDzANBglghkgBZQMEAgEFADB5Bgor # BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCCN2XRukFVYdgMc # CoE6kmmYc0b2v869fBV9Iun89HEGY6CCDMkwggYEMIID7KADAgECAhMzAAACHPrN # 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/AUmQN5W4n101cY2L4A7GTQG1h32HHAvfQESWP0xghoBMIIZ/QIBATBu # MFcxCzAJBgNVBAYTAlVTMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24x # KDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNpZ25pbmcgUENBIDIwMjQCEzMAAAIc # +s3Fm+gvfsQAAAAAAhwwDQYJYIZIAWUDBAIBBQCggbAwGQYJKoZIhvcNAQkDMQwG # CisGAQQBgjcCAQQwHAYKKwYBBAGCNwIBCzEOMAwGCisGAQQBgjcCARUwLwYJKoZI # hvcNAQkEMSIEINJASWYRXkXHmmIzSzqA8i/U/bU5xi6HaWlnbhji15JoMEQGCisG # AQQBgjcCAQwxNjA0oBSAEgBNAGkAYwByAG8AcwBvAGYAdKEcgBpodHRwczovL3d3 # dy5taWNyb3NvZnQuY29tIDANBgkqhkiG9w0BAQEFAASCAQA4EKr0Na43n+YE4m64 # 2xsVL0Wa5eS8mgmt8/GrdACBXbME52YiGUGPY/uJsgoCgkp8zuKb09TIZhg6pgeA # YzX2pFwiPlTb0xmHJWo+/LjL6Rm2NqzDCfXlsbjxOa05EyBTF2KjwTT5tKOzSGlg # bHkFhxpwww5h0jWD38vP+1I5tVVlggBMTZNZuIZFVt7ru1g8S4/0078oUMBX4s+d # R3KqMJyvKQbhaGXc9GawmET7+P8ZsnnRc361fn5qxajPehqXOyKWE1lxe+eyDwg+ # uQRI1SkaADsyEdvY4FHOweRVal1ZtvmDGtubapCky5ciE3nCiFFs/qEmg8sm+XwQ # ZFekoYIXsTCCF60GCisGAQQBgjcDAwExghedMIIXmQYJKoZIhvcNAQcCoIIXijCC # F4YCAQMxDzANBglghkgBZQMEAgEFADCCAVoGCyqGSIb3DQEJEAEEoIIBSQSCAUUw # ggFBAgEBBgorBgEEAYRZCgMBMDEwDQYJYIZIAWUDBAIBBQAEICpHGXeFpDagrDcN # Sg7Zoy8kco6hvoFEeG8p0ziSWIsaAgZqNX5R3lkYEzIwMjYwNjI1MDkwMjQ4LjM0 # M1owBIACAfSggdmkgdYwgdMxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5n # dG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9y # YXRpb24xLTArBgNVBAsTJE1pY3Jvc29mdCBJcmVsYW5kIE9wZXJhdGlvbnMgTGlt # aXRlZDEnMCUGA1UECxMeblNoaWVsZCBUU1MgRVNOOjQzMUEtMDVFMC1EOTQ3MSUw # IwYDVQQDExxNaWNyb3NvZnQgVGltZS1TdGFtcCBTZXJ2aWNloIIR/zCCBygwggUQ # oAMCAQICEzMAAAIdS8CShziFfjkAAQAAAh0wDQYJKoZIhvcNAQELBQAwfDELMAkG # A1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQx # HjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEmMCQGA1UEAxMdTWljcm9z # b2Z0IFRpbWUtU3RhbXAgUENBIDIwMTAwHhcNMjUwODE0MTg0ODMzWhcNMjYxMTEz # MTg0ODMzWjCB0zELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAO # BgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEt # MCsGA1UECxMkTWljcm9zb2Z0IElyZWxhbmQgT3BlcmF0aW9ucyBMaW1pdGVkMScw # JQYDVQQLEx5uU2hpZWxkIFRTUyBFU046NDMxQS0wNUUwLUQ5NDcxJTAjBgNVBAMT # HE1pY3Jvc29mdCBUaW1lLVN0YW1wIFNlcnZpY2UwggIiMA0GCSqGSIb3DQEBAQUA # A4ICDwAwggIKAoICAQCitKBoADyg6XimHnvjPDb16BQ3wMN6lEctfwUzXMc0mZcb # oqtKpQrDNwpp+im5h09MRNMK9v1ol8RK4BTSIY1QUj8PpHSS91+l7ag9f4TextNC # 8aLgk8fmp0hhRonjlX/hup7x429tbOkL5kqMfX3cN6IjVcAj3XwmhCYGGURej9Oi # fXvbWW5kmCKdyx/kuMxjeNfzhbJdRJfd2xLuH/vFUj7DXKODulr7TLej+Z7ZOy/p # QlR1JNBqnk5EZJ8KdyWc/XPciKJYhavdWjtog9ayAnOrebkbGnFQcJCTyrNSGTnT # L+4H4sYTdYgrYLvuLL2IWxJ9ItSfIwTMZENb2ZcdPg8fs7PPoIepASI2/BweqW+U # KHWkdCHU1dBICo6hUGzmaLp5qx/rLFZN97kOtHv3nTevylTpWoLZj1cxFTjAf1Bt # hdiwhRnfcmad3LbZbUsEMBvEE9AcIGWdwYNTcGB2FVRUt7zSaCAU73wV2RaGjrvD # iQ90JNGS92+Rjw+tBgT+dCMdcJrSDstwy21lvp6Mwd9D61RZe/r6dnhieSvY6RrF # yUULDhEhg0xYPboBZtCP9YR3OBrXx8q3DrovmDNc/NrqMUF88l4oTcfxAC7CmKuY # fiaz7mdSM01A6Y2ComfRTX7difsKWzAPv1g3Svd91tgEwMCkFkmk2UrursddGwID # AQABo4IBSTCCAUUwHQYDVR0OBBYEFIRZ8HE0RqZm1ebyCX3ZirzSN/FdMB8GA1Ud # IwQYMBaAFJ+nFV0AXmJdg/Tl0mWnG1M1GelyMF8GA1UdHwRYMFYwVKBSoFCGTmh0 # dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMvY3JsL01pY3Jvc29mdCUyMFRp # bWUtU3RhbXAlMjBQQ0ElMjAyMDEwKDEpLmNybDBsBggrBgEFBQcBAQRgMF4wXAYI # KwYBBQUHMAKGUGh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMvY2VydHMv # TWljcm9zb2Z0JTIwVGltZS1TdGFtcCUyMFBDQSUyMDIwMTAoMSkuY3J0MAwGA1Ud # EwEB/wQCMAAwFgYDVR0lAQH/BAwwCgYIKwYBBQUHAwgwDgYDVR0PAQH/BAQDAgeA # MA0GCSqGSIb3DQEBCwUAA4ICAQCR3B4HjLG8uyksqrQP6aLIPhDQRzFUWk1m4nGJ # HniBZGR5MMO7KY14HTcmGWwGlvBJgnm5lKAMEK/AcQPZUvyUmkWU6msnPGxdYLY1 # N8D47487kWTmPDoseHqN4EAMMR1ADHceqLtmbQnC9D3fPl/p23GSbb1ao5wdhdFd # 8BDDLWFKstfJ95uWpHrqOk//2fR8KRZTiCCxSNClDY2CPUNXT0nhjfLun013zX5e # zqpij77tEqbyqIH/k0N6KA4uOUB4WCIRchFQlb6YnKqlDD445GVqpwWNHwe7Qb7/ # tsx16Trxhf6Q+kMGTtR74j/GCJgnXFwNEGf+9zMu03vb5EiUPhSBdgu4FIKT/+kM # Q9fnPf0Kv6uRzoThjbwU+TgGGWgDK+nrbw/jF8SVBjxNzGtpRtlKHKmhwTqfL3kP # UrUGSW1masdUoLGaCWe46UzXk0oitcWVcLN2qkK0jBDjXvA0BUX9AM+/PNu6Y91O # Lp9vS0ttJxihtXrO9sGwywoQwThOPVv2ghcLx3JsmridtugRdilHCLVABulI2uf4 # /EZb25/WrrcWcwm7iCbc6HreeNb+JV/vbeq7PIetKKNYyBjQeJGIdCLQnK7SHwx2 # FFSnubFuYtByQ+I4XACUhpQ3+TvbnL9otamRFTp+qYuUQ7IflanIt3bcBjL2vy/5 # ChtrqzCCB3EwggVZoAMCAQICEzMAAAAVxedrngKbSZkAAAAAABUwDQYJKoZIhvcN # AQELBQAwgYgxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYD # VQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xMjAw # BgNVBAMTKU1pY3Jvc29mdCBSb290IENlcnRpZmljYXRlIEF1dGhvcml0eSAyMDEw # MB4XDTIxMDkzMDE4MjIyNVoXDTMwMDkzMDE4MzIyNVowfDELMAkGA1UEBhMCVVMx # EzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoT # FU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUt # U3RhbXAgUENBIDIwMTAwggIiMA0GCSqGSIb3DQEBAQUAA4ICDwAwggIKAoICAQDk # 4aZM57RyIQt5osvXJHm9DtWC0/3unAcH0qlsTnXIyjVX9gF/bErg4r25PhdgM/9c # T8dm95VTcVrifkpa/rg2Z4VGIwy1jRPPdzLAEBjoYH1qUoNEt6aORmsHFPPFdvWG # UNzBRMhxXFExN6AKOG6N7dcP2CZTfDlhAnrEqv1yaa8dq6z2Nr41JmTamDu6Gnsz # rYBbfowQHJ1S/rboYiXcag/PXfT+jlPP1uyFVk3v3byNpOORj7I5LFGc6XBpDco2 # LXCOMcg1KL3jtIckw+DJj361VI/c+gVVmG1oO5pGve2krnopN6zL64NF50ZuyjLV # wIYwXE8s4mKyzbnijYjklqwBSru+cakXW2dg3viSkR4dPf0gz3N9QZpGdc3EXzTd # EonW/aUgfX782Z5F37ZyL9t9X4C626p+Nuw2TPYrbqgSUei/BQOj0XOmTTd0lBw0 # gg/wEPK3Rxjtp+iZfD9M269ewvPV2HM9Q07BMzlMjgK8QmguEOqEUUbi0b1qGFph # AXPKZ6Je1yh2AuIzGHLXpyDwwvoSCtdjbwzJNmSLW6CmgyFdXzB0kZSU2LlQ+QuJ # YfM2BjUYhEfb3BvR/bLUHMVr9lxSUV0S2yW6r1AFemzFER1y7435UsSFF5PAPBXb # GjfHCBUYP3irRbb1Hode2o+eFnJpxq57t7c+auIurQIDAQABo4IB3TCCAdkwEgYJ # KwYBBAGCNxUBBAUCAwEAATAjBgkrBgEEAYI3FQIEFgQUKqdS/mTEmr6CkTxGNSnP # EP8vBO4wHQYDVR0OBBYEFJ+nFV0AXmJdg/Tl0mWnG1M1GelyMFwGA1UdIARVMFMw # UQYMKwYBBAGCN0yDfQEBMEEwPwYIKwYBBQUHAgEWM2h0dHA6Ly93d3cubWljcm9z # b2Z0LmNvbS9wa2lvcHMvRG9jcy9SZXBvc2l0b3J5Lmh0bTATBgNVHSUEDDAKBggr # BgEFBQcDCDAZBgkrBgEEAYI3FAIEDB4KAFMAdQBiAEMAQTALBgNVHQ8EBAMCAYYw # DwYDVR0TAQH/BAUwAwEB/zAfBgNVHSMEGDAWgBTV9lbLj+iiXGJo0T2UkFvXzpoY # xDBWBgNVHR8ETzBNMEugSaBHhkVodHRwOi8vY3JsLm1pY3Jvc29mdC5jb20vcGtp # L2NybC9wcm9kdWN0cy9NaWNSb29DZXJBdXRfMjAxMC0wNi0yMy5jcmwwWgYIKwYB # BQUHAQEETjBMMEoGCCsGAQUFBzAChj5odHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20v # cGtpL2NlcnRzL01pY1Jvb0NlckF1dF8yMDEwLTA2LTIzLmNydDANBgkqhkiG9w0B # AQsFAAOCAgEAnVV9/Cqt4SwfZwExJFvhnnJL/Klv6lwUtj5OR2R4sQaTlz0xM7U5 # 18JxNj/aZGx80HU5bbsPMeTCj/ts0aGUGCLu6WZnOlNN3Zi6th542DYunKmCVgAD # sAW+iehp4LoJ7nvfam++Kctu2D9IdQHZGN5tggz1bSNU5HhTdSRXud2f8449xvNo # 32X2pFaq95W2KFUn0CS9QKC/GbYSEhFdPSfgQJY4rPf5KYnDvBewVIVCs/wMnosZ # iefwC2qBwoEZQhlSdYo2wh3DYXMuLGt7bj8sCXgU6ZGyqVvfSaN0DLzskYDSPeZK # PmY7T7uG+jIa2Zb0j/aRAfbOxnT99kxybxCrdTDFNLB62FD+CljdQDzHVG2dY3RI # LLFORy3BFARxv2T5JL5zbcqOCb2zAVdJVGTZc9d/HltEAY5aGZFrDZ+kKNxnGSgk # ujhLmm77IVRrakURR6nxt67I6IleT53S0Ex2tVdUCbFpAUR+fKFhbHP+CrvsQWY9 # af3LwUFJfn6Tvsv4O+S3Fb+0zj6lMVGEvL8CwYKiexcdFYmNcP7ntdAoGokLjzba # ukz5m/8K6TT4JDVnK+ANuOaMmdbhIurwJ0I9JZTmdHRbatGePu1+oDEzfbzL6Xu/ # OHBE0ZDxyKs6ijoIYn/ZcGNTTY3ugm2lBRDBcQZqELQdVTNYs6FwZvKhggNaMIIC # QgIBATCCAQGhgdmkgdYwgdMxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5n # dG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9y # YXRpb24xLTArBgNVBAsTJE1pY3Jvc29mdCBJcmVsYW5kIE9wZXJhdGlvbnMgTGlt # aXRlZDEnMCUGA1UECxMeblNoaWVsZCBUU1MgRVNOOjQzMUEtMDVFMC1EOTQ3MSUw # IwYDVQQDExxNaWNyb3NvZnQgVGltZS1TdGFtcCBTZXJ2aWNloiMKAQEwBwYFKw4D # AhoDFQC6g74Ept9fOrJ+L0YsR1YeQIt5P6CBgzCBgKR+MHwxCzAJBgNVBAYTAlVT # MRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQK # ExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1l # LVN0YW1wIFBDQSAyMDEwMA0GCSqGSIb3DQEBCwUAAgUA7ec8tzAiGA8yMDI2MDYy # NTA1MzU1MVoYDzIwMjYwNjI2MDUzNTUxWjB4MD4GCisGAQQBhFkKBAExMDAuMAoC # BQDt5zy3AgEAMAsCAQACAwFEnAIB/zAHAgEAAgISDzAKAgUA7eiONwIBADA2Bgor # BgEEAYRZCgQCMSgwJjAMBgorBgEEAYRZCgMCoAowCAIBAAIDB6EgoQowCAIBAAID # AYagMA0GCSqGSIb3DQEBCwUAA4IBAQCXr/8kR2lDCcXMCqqSIl/IMDAxc/301z+5 # cXyrXuokyO28vFKMsR1xTQ6XF1aMFrmHXrgHKaN/So3jc/d3oNI1CeqRVtjGVk5U # YeBNcefNiMfCagugy1SIugD4mlKjeuA5nNKd4P91s5E7H+W0c+U0PaTrp221Yj6v # SHy0ByrEtC0FM1jWAP6KpTix2Icu/GJ1aZJdyxRqXjcP0VieqSaruIYK4vZyJ2nm # D6SV7e8i5eh/4op9iX9DhCqO1TUap1IV1G88L1P2Q7HaWBy8v5b8MgDTOCWUaOem # WgchnCM96jKOsphhxRlFmxhOgRkS2eGkPsWqIWzPbDCfhujFwFu+MYIEDTCCBAkC # AQEwgZMwfDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNV # BAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEmMCQG # A1UEAxMdTWljcm9zb2Z0IFRpbWUtU3RhbXAgUENBIDIwMTACEzMAAAIdS8CShziF # fjkAAQAAAh0wDQYJYIZIAWUDBAIBBQCgggFKMBoGCSqGSIb3DQEJAzENBgsqhkiG # 9w0BCRABBDAvBgkqhkiG9w0BCQQxIgQg30aezhe3t03xq07/HXiGmyuWcx68F4Jb # V327HL84fHIwgfoGCyqGSIb3DQEJEAIvMYHqMIHnMIHkMIG9BCCxtpXMXEiLJzrq # M77ep4rTNwrMOj6gpWN9hZvpj5QFUTCBmDCBgKR+MHwxCzAJBgNVBAYTAlVTMRMw # EQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVN # aWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0 # YW1wIFBDQSAyMDEwAhMzAAACHUvAkoc4hX45AAEAAAIdMCIEIMegHAoatNOQZs0r # B1cnEn+oYZPBjJU4bHxbvv4XZl0tMA0GCSqGSIb3DQEBCwUABIICAHQNN/7Fh78l # 4X9Lj0qU57QPn53HXRpVZgXh31Cfs1wa0ekIdZjY68JkTYBmojig4v0HJqz9/Y2o # 9CYkBdBjm5M1Kr5JO/m3mOdiIU4sXZ054RnCEKYPWFIWyq0ma2O5zlcdwWVjt1S2 # jx3dzhsDo5NPUGUYJvCHfswcFG89lV5ywasaeo3dw7jLKdxWj0cNDH9PwaAYfd9b # 5hSsQeQhFH6BgD0Rxi2tIVUdZG37fCAFWQMRNbPralY5D07How7NzG0MPnfLqKXh # MIW28pNIYPOZ6nMzx1VqqnauT7CkqiMCuizS1SgElmVTNZkgLy7hxLAQc6syqQbO # RGj2mBetK6JHMW1xHLumvhFJ6kKYRhv/XkwRrkBYlhVSR2gag90X8Mls5SarAG3M # 9HkVKgr3TOI30m6EtUJN0b/aQ6d2mAHj7h8XRAbxGIT8Zp3ZwQNjnRK4rIDIveel # 1upe3yHnXWVZad70jqiZnwwJV6gl4EKINnoj6tN6rugIgu/XEV4l+/nlN1L5fzI+ # wyURdGG6Ymmr8gtgwllj7kRnVzvJeKubnrIVruSsEwZ50jii2IhmPk3rmguuYnvL # 9ebP/cHJeuCU7EXebH7xe8IWEWSfbn5zwqaoCL4gYtIvoZyt4vm6g6ZdKgf8jFHb # W98QS70Zzz+TiLmAkyjns2FQFMGZvgT3 # SIG # End signature block |