Private/Tools/Test-Tooling.ps1
function Test-Tooling { [CmdletBinding(SupportsShouldProcess = $true)] param( [Parameter(Mandatory = $false)] [switch]$skipAlzModuleVersionCheck ) $checkResults = @() $hasFailure = $false # Check if PowerShell is the correct version Write-Verbose "Checking PowerShell version" $powerShellVersionTable = $PSVersionTable $powerShellVersion = $powerShellVersionTable.PSVersion.ToString() if ($powerShellVersionTable.PSVersion.Major -lt 7) { $checkResults += @{ message = "PowerShell version $powerShellVersion is not supported. Please upgrade to PowerShell 7.4 or higher. Either switch to the `pwsh` prompt or follow the instructions here: https://aka.ms/install-powershell" result = "Failure" } $hasFailure = $true } elseif ($powerShellVersionTable.PSVersion.Major -eq 7 -and $powerShellVersionTable.PSVersion.Minor -lt 4) { $checkResults += @{ message = "PowerShell version $powerShellVersion is not supported. Please upgrade to PowerShell 7.4 or higher. Either switch to the `pwsh` prompt or follow the instructions here: https://aka.ms/install-powershell" result = "Failure" } $hasFailure = $true } else { $checkResults += @{ message = "PowerShell version $powerShellVersion is supported." result = "Success" } } # Check if Git is installed Write-Verbose "Checking Git installation" $gitPath = Get-Command git -ErrorAction SilentlyContinue if ($gitPath) { $checkResults += @{ message = "Git is installed." result = "Success" } } else { $checkResults += @{ message = "Git is not installed. Follow the instructions here: https://git-scm.com/downloads" result = "Failure" } $hasFailure = $true } # Check if using Service Principal Auth Write-Verbose "Checking Azure environment variables" $nonAzCliEnvVars = @( "ARM_CLIENT_ID", "ARM_SUBSCRIPTION_ID", "ARM_TENANT_ID" ) $envVarsSet = $true $envVarValid = $true $envVarUnique = $true $envVarAtLeastOneSet = $false $envVarsWithValue = @() $checkedEnvVars = @() foreach($envVar in $nonAzCliEnvVars) { $envVarValue = [System.Environment]::GetEnvironmentVariable($envVar) if($envVarValue -eq $null -or $envVarValue -eq "" ) { $envVarsSet = $false continue } $envVarAtLeastOneSet = $true $envVarsWithValue += $envVar if($envVarValue -notmatch("^(\{){0,1}[0-9a-fA-F]{8}\-[0-9a-fA-F]{4}\-[0-9a-fA-F]{4}\-[0-9a-fA-F]{4}\-[0-9a-fA-F]{12}(\}){0,1}$")) { $envVarValid = $false continue } if($checkedEnvVars -contains $envVarValue) { $envVarUnique = $false continue } $checkedEnvVars += $envVarValue } if($envVarsSet) { Write-Verbose "Using Service Principal Authentication, skipping Azure CLI checks" if($envVarValid -and $envVarUnique) { $checkResults += @{ message = "Azure environment variables are set and are valid unique GUIDs." result = "Success" } } if(-not $envVarValid) { $checkResults += @{ message = "Azure environment variables are set, but are not all valid GUIDs." result = "Failure" } $hasFailure = $true } if (-not $envVarUnique) { $envVarValidationOutput = "" foreach($envVar in $nonAzCliEnvVars) { $envVarValue = [System.Environment]::GetEnvironmentVariable($envVar) $envVarValidationOutput += " $envVar ($envVarValue)" } $checkResults += @{ message = "Azure environment variables are set, but are not unique GUIDs. There is at least one duplicate:$envVarValidationOutput." result = "Failure" } $hasFailure = $true } } else { if($envVarAtLeastOneSet) { $envVarValidationOutput = "" foreach($envVar in $envVarsWithValue) { $envVarValue = [System.Environment]::GetEnvironmentVariable($envVar) $envVarValidationOutput += " $envVar ($envVarValue)" } $checkResults += @{ message = "At least one environment variable is set, but the other expected environment variables are not set. This could cause Terraform to fail in unexpected ways. Set environment variables:$envVarValidationOutput." result = "Warning" } } # Check if Azure CLI is installed Write-Verbose "Checking Azure CLI installation" $azCliPath = Get-Command az -ErrorAction SilentlyContinue if ($azCliPath) { $checkResults += @{ message = "Azure CLI is installed." result = "Success" } } else { $checkResults += @{ message = "Azure CLI is not installed. Follow the instructions here: https://learn.microsoft.com/en-us/cli/azure/install-azure-cli" result = "Failure" } $hasFailure = $true } # Check if Azure CLI is logged in Write-Verbose "Checking Azure CLI login status" $azCliAccount = $(az account show -o json) | ConvertFrom-Json if ($azCliAccount) { $checkResults += @{ message = "Azure CLI is logged in. Tenant ID: $($azCliAccount.tenantId), Subscription: $($azCliAccount.name) ($($azCliAccount.id))" result = "Success" } } else { $checkResults += @{ message = "Azure CLI is not logged in. Please login to Azure CLI using 'az login -t `"00000000-0000-0000-0000-000000000000}`"', replacing the empty GUID with your tenant ID." result = "Failure" } $hasFailure = $true } } if($skipAlzModuleVersionCheck.IsPresent) { Write-Verbose "Skipping ALZ module version check" } else { # Check if latest ALZ module is installed Write-Verbose "Checking ALZ module version" $alzModuleCurrentVersion = Get-InstalledModule -Name ALZ -ErrorAction SilentlyContinue if($null -eq $alzModuleCurrentVersion) { $checkResults += @{ message = "ALZ module is not correctly installed. Please install the latest version using 'Install-Module ALZ'." result = "Failure" } $hasFailure = $true } $alzModuleLatestVersion = Find-Module -Name ALZ if ($null -ne $alzModuleCurrentVersion) { if ($alzModuleCurrentVersion.Version -lt $alzModuleLatestVersion.Version) { $checkResults += @{ message = "ALZ module is not the latest version. Your version: $($alzModuleCurrentVersion.Version), Latest version: $($alzModuleLatestVersion.Version). Please update to the latest version using 'Update-Module ALZ'." result = "Failure" } $hasFailure = $true } else { $checkResults += @{ message = "ALZ module is the latest version ($($alzModuleCurrentVersion.Version))." result = "Success" } } } } 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 if($hasFailure) { Write-InformationColored "Accelerator software requirements have no been met, please review and install the missing software." -ForegroundColor Red -InformationAction Continue Write-InformationColored "Cannot continue with Deployment..." -ForegroundColor Red -InformationAction Continue throw "Accelerator software requirements have no been met, please review and install the missing software." } } # SIG # Begin signature block # MIIoOwYJKoZIhvcNAQcCoIIoLDCCKCgCAQExDzANBglghkgBZQMEAgEFADB5Bgor # BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCB/8UjrsOzfs837 # 6BXerMr1ab/OuLmN9baEMzkvHLTiqaCCDYUwggYDMIID66ADAgECAhMzAAAEhJji # EuB4ozFdAAAAAASEMA0GCSqGSIb3DQEBCwUAMH4xCzAJBgNVBAYTAlVTMRMwEQYD # VQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNy # b3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNpZ25p # bmcgUENBIDIwMTEwHhcNMjUwNjE5MTgyMTM1WhcNMjYwNjE3MTgyMTM1WjB0MQsw # CQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9u # ZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMR4wHAYDVQQDExVNaWNy # b3NvZnQgQ29ycG9yYXRpb24wggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIB # AQDtekqMKDnzfsyc1T1QpHfFtr+rkir8ldzLPKmMXbRDouVXAsvBfd6E82tPj4Yz # aSluGDQoX3NpMKooKeVFjjNRq37yyT/h1QTLMB8dpmsZ/70UM+U/sYxvt1PWWxLj # MNIXqzB8PjG6i7H2YFgk4YOhfGSekvnzW13dLAtfjD0wiwREPvCNlilRz7XoFde5 # KO01eFiWeteh48qUOqUaAkIznC4XB3sFd1LWUmupXHK05QfJSmnei9qZJBYTt8Zh # ArGDh7nQn+Y1jOA3oBiCUJ4n1CMaWdDhrgdMuu026oWAbfC3prqkUn8LWp28H+2S # LetNG5KQZZwvy3Zcn7+PQGl5AgMBAAGjggGCMIIBfjAfBgNVHSUEGDAWBgorBgEE # AYI3TAgBBggrBgEFBQcDAzAdBgNVHQ4EFgQUBN/0b6Fh6nMdE4FAxYG9kWCpbYUw # VAYDVR0RBE0wS6RJMEcxLTArBgNVBAsTJE1pY3Jvc29mdCBJcmVsYW5kIE9wZXJh # dGlvbnMgTGltaXRlZDEWMBQGA1UEBRMNMjMwMDEyKzUwNTM2MjAfBgNVHSMEGDAW # gBRIbmTlUAXTgqoXNzcitW2oynUClTBUBgNVHR8ETTBLMEmgR6BFhkNodHRwOi8v # d3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL2NybC9NaWNDb2RTaWdQQ0EyMDExXzIw # MTEtMDctMDguY3JsMGEGCCsGAQUFBwEBBFUwUzBRBggrBgEFBQcwAoZFaHR0cDov # L3d3dy5taWNyb3NvZnQuY29tL3BraW9wcy9jZXJ0cy9NaWNDb2RTaWdQQ0EyMDEx # XzIwMTEtMDctMDguY3J0MAwGA1UdEwEB/wQCMAAwDQYJKoZIhvcNAQELBQADggIB # AGLQps1XU4RTcoDIDLP6QG3NnRE3p/WSMp61Cs8Z+JUv3xJWGtBzYmCINmHVFv6i # 8pYF/e79FNK6P1oKjduxqHSicBdg8Mj0k8kDFA/0eU26bPBRQUIaiWrhsDOrXWdL # m7Zmu516oQoUWcINs4jBfjDEVV4bmgQYfe+4/MUJwQJ9h6mfE+kcCP4HlP4ChIQB # UHoSymakcTBvZw+Qst7sbdt5KnQKkSEN01CzPG1awClCI6zLKf/vKIwnqHw/+Wvc # Ar7gwKlWNmLwTNi807r9rWsXQep1Q8YMkIuGmZ0a1qCd3GuOkSRznz2/0ojeZVYh # ZyohCQi1Bs+xfRkv/fy0HfV3mNyO22dFUvHzBZgqE5FbGjmUnrSr1x8lCrK+s4A+ # bOGp2IejOphWoZEPGOco/HEznZ5Lk6w6W+E2Jy3PHoFE0Y8TtkSE4/80Y2lBJhLj # 27d8ueJ8IdQhSpL/WzTjjnuYH7Dx5o9pWdIGSaFNYuSqOYxrVW7N4AEQVRDZeqDc # fqPG3O6r5SNsxXbd71DCIQURtUKss53ON+vrlV0rjiKBIdwvMNLQ9zK0jy77owDy # XXoYkQxakN2uFIBO1UNAvCYXjs4rw3SRmBX9qiZ5ENxcn/pLMkiyb68QdwHUXz+1 # fI6ea3/jjpNPz6Dlc/RMcXIWeMMkhup/XEbwu73U+uz/MIIHejCCBWKgAwIBAgIK # YQ6Q0gAAAAAAAzANBgkqhkiG9w0BAQsFADCBiDELMAkGA1UEBhMCVVMxEzARBgNV # BAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jv # c29mdCBDb3Jwb3JhdGlvbjEyMDAGA1UEAxMpTWljcm9zb2Z0IFJvb3QgQ2VydGlm # aWNhdGUgQXV0aG9yaXR5IDIwMTEwHhcNMTEwNzA4MjA1OTA5WhcNMjYwNzA4MjEw # OTA5WjB+MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UE # BxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSgwJgYD # VQQDEx9NaWNyb3NvZnQgQ29kZSBTaWduaW5nIFBDQSAyMDExMIICIjANBgkqhkiG # 9w0BAQEFAAOCAg8AMIICCgKCAgEAq/D6chAcLq3YbqqCEE00uvK2WCGfQhsqa+la # UKq4BjgaBEm6f8MMHt03a8YS2AvwOMKZBrDIOdUBFDFC04kNeWSHfpRgJGyvnkmc # 6Whe0t+bU7IKLMOv2akrrnoJr9eWWcpgGgXpZnboMlImEi/nqwhQz7NEt13YxC4D # dato88tt8zpcoRb0RrrgOGSsbmQ1eKagYw8t00CT+OPeBw3VXHmlSSnnDb6gE3e+ # lD3v++MrWhAfTVYoonpy4BI6t0le2O3tQ5GD2Xuye4Yb2T6xjF3oiU+EGvKhL1nk # kDstrjNYxbc+/jLTswM9sbKvkjh+0p2ALPVOVpEhNSXDOW5kf1O6nA+tGSOEy/S6 # A4aN91/w0FK/jJSHvMAhdCVfGCi2zCcoOCWYOUo2z3yxkq4cI6epZuxhH2rhKEmd # X4jiJV3TIUs+UsS1Vz8kA/DRelsv1SPjcF0PUUZ3s/gA4bysAoJf28AVs70b1FVL # 5zmhD+kjSbwYuER8ReTBw3J64HLnJN+/RpnF78IcV9uDjexNSTCnq47f7Fufr/zd # sGbiwZeBe+3W7UvnSSmnEyimp31ngOaKYnhfsi+E11ecXL93KCjx7W3DKI8sj0A3 # T8HhhUSJxAlMxdSlQy90lfdu+HggWCwTXWCVmj5PM4TasIgX3p5O9JawvEagbJjS # 4NaIjAsCAwEAAaOCAe0wggHpMBAGCSsGAQQBgjcVAQQDAgEAMB0GA1UdDgQWBBRI # bmTlUAXTgqoXNzcitW2oynUClTAZBgkrBgEEAYI3FAIEDB4KAFMAdQBiAEMAQTAL # BgNVHQ8EBAMCAYYwDwYDVR0TAQH/BAUwAwEB/zAfBgNVHSMEGDAWgBRyLToCMZBD # uRQFTuHqp8cx0SOJNDBaBgNVHR8EUzBRME+gTaBLhklodHRwOi8vY3JsLm1pY3Jv # c29mdC5jb20vcGtpL2NybC9wcm9kdWN0cy9NaWNSb29DZXJBdXQyMDExXzIwMTFf # MDNfMjIuY3JsMF4GCCsGAQUFBwEBBFIwUDBOBggrBgEFBQcwAoZCaHR0cDovL3d3 # dy5taWNyb3NvZnQuY29tL3BraS9jZXJ0cy9NaWNSb29DZXJBdXQyMDExXzIwMTFf # MDNfMjIuY3J0MIGfBgNVHSAEgZcwgZQwgZEGCSsGAQQBgjcuAzCBgzA/BggrBgEF # BQcCARYzaHR0cDovL3d3dy5taWNyb3NvZnQuY29tL3BraW9wcy9kb2NzL3ByaW1h # cnljcHMuaHRtMEAGCCsGAQUFBwICMDQeMiAdAEwAZQBnAGEAbABfAHAAbwBsAGkA # YwB5AF8AcwB0AGEAdABlAG0AZQBuAHQALiAdMA0GCSqGSIb3DQEBCwUAA4ICAQBn # 8oalmOBUeRou09h0ZyKbC5YR4WOSmUKWfdJ5DJDBZV8uLD74w3LRbYP+vj/oCso7 # v0epo/Np22O/IjWll11lhJB9i0ZQVdgMknzSGksc8zxCi1LQsP1r4z4HLimb5j0b # pdS1HXeUOeLpZMlEPXh6I/MTfaaQdION9MsmAkYqwooQu6SpBQyb7Wj6aC6VoCo/ # KmtYSWMfCWluWpiW5IP0wI/zRive/DvQvTXvbiWu5a8n7dDd8w6vmSiXmE0OPQvy # CInWH8MyGOLwxS3OW560STkKxgrCxq2u5bLZ2xWIUUVYODJxJxp/sfQn+N4sOiBp # mLJZiWhub6e3dMNABQamASooPoI/E01mC8CzTfXhj38cbxV9Rad25UAqZaPDXVJi # hsMdYzaXht/a8/jyFqGaJ+HNpZfQ7l1jQeNbB5yHPgZ3BtEGsXUfFL5hYbXw3MYb # BL7fQccOKO7eZS/sl/ahXJbYANahRr1Z85elCUtIEJmAH9AAKcWxm6U/RXceNcbS # oqKfenoi+kiVH6v7RyOA9Z74v2u3S5fi63V4GuzqN5l5GEv/1rMjaHXmr/r8i+sL # gOppO6/8MO0ETI7f33VtY5E90Z1WTk+/gFcioXgRMiF670EKsT/7qMykXcGhiJtX # cVZOSEXAQsmbdlsKgEhr/Xmfwb1tbWrJUnMTDXpQzTGCGgwwghoIAgEBMIGVMH4x # CzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRt # b25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01p # Y3Jvc29mdCBDb2RlIFNpZ25pbmcgUENBIDIwMTECEzMAAASEmOIS4HijMV0AAAAA # BIQwDQYJYIZIAWUDBAIBBQCggbAwGQYJKoZIhvcNAQkDMQwGCisGAQQBgjcCAQQw # HAYKKwYBBAGCNwIBCzEOMAwGCisGAQQBgjcCARUwLwYJKoZIhvcNAQkEMSIEIHPP # YyiU2nuKqDNnOOPBrJUGeCCmYIc/p/R6gUGpvvClMEQGCisGAQQBgjcCAQwxNjA0 # oBSAEgBNAGkAYwByAG8AcwBvAGYAdKEcgBpodHRwczovL3d3dy5taWNyb3NvZnQu # Y29tIDANBgkqhkiG9w0BAQEFAASCAQBZjVMOcx7C1JaSjwHcrQXShNQ/PChCFHnf # EuMuBF5CwoPfJa1IIalV9TMVIYR+iJCPhldtB5wBFHll6qVoc+koNnr49DmKFQmD # fQs9sM4aH16Z7vnkKpAfX/4BhlcPMRvbLefMXWbjvXl9804Qrr4c8KCIfJeEuNPF # Edu0F/ctL838zcMQeMcJBaPGsnkqDLSy/Ht3hkiUMO8Epg0mt+8pxaTxbOWn0oX/ # K7CVMnyFH39Y7YI7TfYa8m4nt6oXPJbpBU6zLGPLCstB9IZPc48qYIdzYbSNjaT6 # juWow0BCRMG914JG4fB7xZWkbhbaN/xQJxXJlSthJXBoMIFWOK2qoYIXlDCCF5AG # CisGAQQBgjcDAwExgheAMIIXfAYJKoZIhvcNAQcCoIIXbTCCF2kCAQMxDzANBglg # hkgBZQMEAgEFADCCAVIGCyqGSIb3DQEJEAEEoIIBQQSCAT0wggE5AgEBBgorBgEE # AYRZCgMBMDEwDQYJYIZIAWUDBAIBBQAEINxS6p3CSHq79/rCbyCWiW18IvMlMGKS # BoQSERypNCVDAgZoo6YiWQ4YEzIwMjUwOTE1MTcxNTM4LjI5N1owBIACAfSggdGk # gc4wgcsxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQH # EwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJTAjBgNV # BAsTHE1pY3Jvc29mdCBBbWVyaWNhIE9wZXJhdGlvbnMxJzAlBgNVBAsTHm5TaGll # bGQgVFNTIEVTTjpGMDAyLTA1RTAtRDk0NzElMCMGA1UEAxMcTWljcm9zb2Z0IFRp # bWUtU3RhbXAgU2VydmljZaCCEeowggcgMIIFCKADAgECAhMzAAACBTx1bIJEh83+ # AAEAAAIFMA0GCSqGSIb3DQEBCwUAMHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpX # YXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQg # Q29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1wIFBDQSAy # MDEwMB4XDTI1MDEzMDE5NDI0OVoXDTI2MDQyMjE5NDI0OVowgcsxCzAJBgNVBAYT # AlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYD # VQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJTAjBgNVBAsTHE1pY3Jvc29mdCBB # bWVyaWNhIE9wZXJhdGlvbnMxJzAlBgNVBAsTHm5TaGllbGQgVFNTIEVTTjpGMDAy # LTA1RTAtRDk0NzElMCMGA1UEAxMcTWljcm9zb2Z0IFRpbWUtU3RhbXAgU2Vydmlj # ZTCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBAJKS8t93uAXWvbAZ3Lzk # IVhcdQLzLATIPgtEu/RZgRd55nS7li4runZxdXNCZk84dM3xNaobZI+VRv7s+V3M # UMMCVHe9TymI6OaG72sdbjczZ1uiv2OX2CW6HPBR3ZJyJUZrt/23ru0zcoUpFIxc # W0coXcrAEHtpWj5vWrLmn0NaWjY3kUasGocwRWU3LOt4digMsv6bx9Kyoy7+JhSr # HMrkXhLshjk16YAHwH4DdCDBUiLrgokh94plR6JYoJN/ih1SA/cBCKXGHjw/rPsB # PggDR0wS0qFgsWzhb8o0MyxivsqlA8pUJwLkTy4Md0p7C5vLN6eRPHLh9/U3eDzK # Gjk+L0F+NHRXK2uSakN96nwk/BxvgE04hc6jWl90dnwS+dHskVkVCKqkxkWU7kIC # 5Ngfy6Nzk9QeVowAnw0Rr1MUlM5IGsHs9GB6H/o0nbG73LE+H+RU30Eayz3cwLpS # OmjF4zjHvRvBvCIrxI0cg7wPxyqXtVJ69RhuM3g2iAUXCEEKWGh0T/N4Y+rrLqLE # PjPrkgdjfPAVBsFVf/D4v9Uc2f8EwazY8YeeVGM78qTw0ik3iyGQVCoDV8zTx+us # NI0Rj1UoO2mxSkAXnVjWhDq0mFYzV3ed4JeHpv/o35d4c5ELCdjzcr6kUwyGDyKx # dBvopGXrmSDxdgF3gnCRsqhVAgMBAAGjggFJMIIBRTAdBgNVHQ4EFgQUdHpauzbt # vr7IndsRn2jk10vVsvAwHwYDVR0jBBgwFoAUn6cVXQBeYl2D9OXSZacbUzUZ6XIw # XwYDVR0fBFgwVjBUoFKgUIZOaHR0cDovL3d3dy5taWNyb3NvZnQuY29tL3BraW9w # cy9jcmwvTWljcm9zb2Z0JTIwVGltZS1TdGFtcCUyMFBDQSUyMDIwMTAoMSkuY3Js # MGwGCCsGAQUFBwEBBGAwXjBcBggrBgEFBQcwAoZQaHR0cDovL3d3dy5taWNyb3Nv # ZnQuY29tL3BraW9wcy9jZXJ0cy9NaWNyb3NvZnQlMjBUaW1lLVN0YW1wJTIwUENB # JTIwMjAxMCgxKS5jcnQwDAYDVR0TAQH/BAIwADAWBgNVHSUBAf8EDDAKBggrBgEF # BQcDCDAOBgNVHQ8BAf8EBAMCB4AwDQYJKoZIhvcNAQELBQADggIBAA67GTjtvWLv # lXzVPHrGjXTE0ivjNnFhV+QXlMWraSd08eDNXIueyzD8cqQRlEKBlWmQoQnpjiO8 # bm26AyL5aO3uFQxKKmT5GkHmAVC+HXGYAQvZ+V6DNNYSyePTCsRKmUjlne+B/Z4Z # cCv9FyoaNKmi4dsPYdj1jcXQ6XoVEMJX8cQYpEfOfYzmtCkUZKNpxPgOSpViZ6b8 # Cs59K9WiOcoQhb3XhTEa26ElKv2M6jlGpNfsYipu283vOFaV36LdfF2F01x+VDP1 # iGgWpB7EF6eghGAA8C3AfrzFzOv0swLeX0AsSmey87RGIo9cXiXbb859wV99qmGe # X9MPCSIl/E7IAx9QXfEj37eLNPVZfYIWzZFo4Crd1yiHInD7FEbQTzCQIeqRXnsF # tQETMtfwv2UYnUOjFg2mgNfuJDMn1B2TKmLl+/vvcTcKHwD62jF4WnoWJ9BnIJzw # hwgGUfJqToxtPphNVA2BD+HwUX5Lk6o/sIQIW8gfYq3Y/QaU670LQF9qyGdOOh2a # 1kVmym1S+KuT/Yc4suMGIyMPAmxAAUDMm7Phm1PKiuu8RHXQaX+ZuZWIeqG80AJ9 # PM+It+MODK+n2zv9se78JqUqZ6IlQsaOH+XACPnfX2mCCwYmpuEngVsVz6hTnN99 # ub+sqNVnyTE0PeKbXRCnjWfa3+qnI/oRMIIHcTCCBVmgAwIBAgITMwAAABXF52ue # AptJmQAAAAAAFTANBgkqhkiG9w0BAQsFADCBiDELMAkGA1UEBhMCVVMxEzARBgNV # BAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jv # c29mdCBDb3Jwb3JhdGlvbjEyMDAGA1UEAxMpTWljcm9zb2Z0IFJvb3QgQ2VydGlm # aWNhdGUgQXV0aG9yaXR5IDIwMTAwHhcNMjEwOTMwMTgyMjI1WhcNMzAwOTMwMTgz # MjI1WjB8MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UE # BxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSYwJAYD # VQQDEx1NaWNyb3NvZnQgVGltZS1TdGFtcCBQQ0EgMjAxMDCCAiIwDQYJKoZIhvcN # AQEBBQADggIPADCCAgoCggIBAOThpkzntHIhC3miy9ckeb0O1YLT/e6cBwfSqWxO # dcjKNVf2AX9sSuDivbk+F2Az/1xPx2b3lVNxWuJ+Slr+uDZnhUYjDLWNE893MsAQ # GOhgfWpSg0S3po5GawcU88V29YZQ3MFEyHFcUTE3oAo4bo3t1w/YJlN8OWECesSq # /XJprx2rrPY2vjUmZNqYO7oaezOtgFt+jBAcnVL+tuhiJdxqD89d9P6OU8/W7IVW # Te/dvI2k45GPsjksUZzpcGkNyjYtcI4xyDUoveO0hyTD4MmPfrVUj9z6BVWYbWg7 # mka97aSueik3rMvrg0XnRm7KMtXAhjBcTyziYrLNueKNiOSWrAFKu75xqRdbZ2De # +JKRHh09/SDPc31BmkZ1zcRfNN0Sidb9pSB9fvzZnkXftnIv231fgLrbqn427DZM # 9ituqBJR6L8FA6PRc6ZNN3SUHDSCD/AQ8rdHGO2n6Jl8P0zbr17C89XYcz1DTsEz # OUyOArxCaC4Q6oRRRuLRvWoYWmEBc8pnol7XKHYC4jMYctenIPDC+hIK12NvDMk2 # ZItboKaDIV1fMHSRlJTYuVD5C4lh8zYGNRiER9vcG9H9stQcxWv2XFJRXRLbJbqv # UAV6bMURHXLvjflSxIUXk8A8FdsaN8cIFRg/eKtFtvUeh17aj54WcmnGrnu3tz5q # 4i6tAgMBAAGjggHdMIIB2TASBgkrBgEEAYI3FQEEBQIDAQABMCMGCSsGAQQBgjcV # AgQWBBQqp1L+ZMSavoKRPEY1Kc8Q/y8E7jAdBgNVHQ4EFgQUn6cVXQBeYl2D9OXS # ZacbUzUZ6XIwXAYDVR0gBFUwUzBRBgwrBgEEAYI3TIN9AQEwQTA/BggrBgEFBQcC # ARYzaHR0cDovL3d3dy5taWNyb3NvZnQuY29tL3BraW9wcy9Eb2NzL1JlcG9zaXRv # cnkuaHRtMBMGA1UdJQQMMAoGCCsGAQUFBwMIMBkGCSsGAQQBgjcUAgQMHgoAUwB1 # AGIAQwBBMAsGA1UdDwQEAwIBhjAPBgNVHRMBAf8EBTADAQH/MB8GA1UdIwQYMBaA # FNX2VsuP6KJcYmjRPZSQW9fOmhjEMFYGA1UdHwRPME0wS6BJoEeGRWh0dHA6Ly9j # cmwubWljcm9zb2Z0LmNvbS9wa2kvY3JsL3Byb2R1Y3RzL01pY1Jvb0NlckF1dF8y # MDEwLTA2LTIzLmNybDBaBggrBgEFBQcBAQROMEwwSgYIKwYBBQUHMAKGPmh0dHA6 # Ly93d3cubWljcm9zb2Z0LmNvbS9wa2kvY2VydHMvTWljUm9vQ2VyQXV0XzIwMTAt # MDYtMjMuY3J0MA0GCSqGSIb3DQEBCwUAA4ICAQCdVX38Kq3hLB9nATEkW+Geckv8 # qW/qXBS2Pk5HZHixBpOXPTEztTnXwnE2P9pkbHzQdTltuw8x5MKP+2zRoZQYIu7p # Zmc6U03dmLq2HnjYNi6cqYJWAAOwBb6J6Gngugnue99qb74py27YP0h1AdkY3m2C # DPVtI1TkeFN1JFe53Z/zjj3G82jfZfakVqr3lbYoVSfQJL1AoL8ZthISEV09J+BA # ljis9/kpicO8F7BUhUKz/AyeixmJ5/ALaoHCgRlCGVJ1ijbCHcNhcy4sa3tuPywJ # eBTpkbKpW99Jo3QMvOyRgNI95ko+ZjtPu4b6MhrZlvSP9pEB9s7GdP32THJvEKt1 # MMU0sHrYUP4KWN1APMdUbZ1jdEgssU5HLcEUBHG/ZPkkvnNtyo4JvbMBV0lUZNlz # 138eW0QBjloZkWsNn6Qo3GcZKCS6OEuabvshVGtqRRFHqfG3rsjoiV5PndLQTHa1 # V1QJsWkBRH58oWFsc/4Ku+xBZj1p/cvBQUl+fpO+y/g75LcVv7TOPqUxUYS8vwLB # gqJ7Fx0ViY1w/ue10CgaiQuPNtq6TPmb/wrpNPgkNWcr4A245oyZ1uEi6vAnQj0l # lOZ0dFtq0Z4+7X6gMTN9vMvpe784cETRkPHIqzqKOghif9lwY1NNje6CbaUFEMFx # BmoQtB1VM1izoXBm8qGCA00wggI1AgEBMIH5oYHRpIHOMIHLMQswCQYDVQQGEwJV # UzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UE # ChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSUwIwYDVQQLExxNaWNyb3NvZnQgQW1l # cmljYSBPcGVyYXRpb25zMScwJQYDVQQLEx5uU2hpZWxkIFRTUyBFU046RjAwMi0w # NUUwLUQ5NDcxJTAjBgNVBAMTHE1pY3Jvc29mdCBUaW1lLVN0YW1wIFNlcnZpY2Wi # IwoBATAHBgUrDgMCGgMVANWwf2nW0mf6SvAIy+o6FW9ett60oIGDMIGApH4wfDEL # MAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1v # bmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEmMCQGA1UEAxMdTWlj # cm9zb2Z0IFRpbWUtU3RhbXAgUENBIDIwMTAwDQYJKoZIhvcNAQELBQACBQDscmMF # MCIYDzIwMjUwOTE1MTAwMzQ5WhgPMjAyNTA5MTYxMDAzNDlaMHQwOgYKKwYBBAGE # WQoEATEsMCowCgIFAOxyYwUCAQAwBwIBAAICHugwBwIBAAICEnQwCgIFAOxztIUC # AQAwNgYKKwYBBAGEWQoEAjEoMCYwDAYKKwYBBAGEWQoDAqAKMAgCAQACAwehIKEK # MAgCAQACAwGGoDANBgkqhkiG9w0BAQsFAAOCAQEASgU2+9wlq8DUO3tLXTH8wo1Y # 5ynSA5p5LkBs1yTVUYrGUh3+QdKlRBq5OFAeuLsj6YlU8dVMfAoap8e7QxeMsjFo # DLVXXPbgiP8vDKpp53ew8P+2B43itggCWuvIg+3NLYLC6vxXVrMVuttIL7DDywJR # f0lDq845MQMb5CdsX31TC13sKFaO+wmZUhpwZFlJwombbvSaQW34595NsVkrL1Bd # i6r/EmqiXpJu5aiJvVrccBhxqlTCtRe2vmeM/bnbS7JD70l42B0JrRtqhCI622A/ # k5zFHu477NGCUhrEXiUMPVW9lX1MKolc1EheEJnvg5wz2lb46hPiU4ih+DQFAjGC # BA0wggQJAgEBMIGTMHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9u # MRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRp # b24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1wIFBDQSAyMDEwAhMzAAAC # BTx1bIJEh83+AAEAAAIFMA0GCWCGSAFlAwQCAQUAoIIBSjAaBgkqhkiG9w0BCQMx # DQYLKoZIhvcNAQkQAQQwLwYJKoZIhvcNAQkEMSIEICX2V3ok7eBDEpFNDpUlArG9 # 7lfun030T/EoPw1pY6fwMIH6BgsqhkiG9w0BCRACLzGB6jCB5zCB5DCBvQQggA0D # Pdx5jS6aF1YtHawmmrQ4+q0kNMBhGaMdWTARb0gwgZgwgYCkfjB8MQswCQYDVQQG # EwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwG # A1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSYwJAYDVQQDEx1NaWNyb3NvZnQg # VGltZS1TdGFtcCBQQ0EgMjAxMAITMwAAAgU8dWyCRIfN/gABAAACBTAiBCD+d91D # 3s5NzcClxQDTM5R+JZZLatKnDvedCKR5TBLtUTANBgkqhkiG9w0BAQsFAASCAgBT # jj91F4LXPSE+4LxmPY9ckn2vdqJeZtmaaB7yxIOt/IbyJBs9TxuFTeEVsqyEPOzr # gAaaJIICl/EzKUqAougnN+HKdGx4q67ZAgty1KQxkXUVFvglf8gtLwVRiAunR5YQ # zr9TbYRFQsHeZLyC7awqpDyxPNVBM/RkcrbY5MHcflRGb4vqNqevgRdgVI8IYf0A # YNvXn/st5b96PJ+crxKi5Cye3w5wZ/ZEkkfAcpPNLZe/sIgncRZwJo/+4YAGZ1cK # chG4aDnNHyRNVylV5YssZyTYodCubnUJp9gk2tqExZ3V6+wpUu+uUVTUL9vGJ37C # Fe7hVsg2v4jOgcwH+RM24FRGVuvtfn6tIFLCsS+4dQ+i6AYnc8dmmYg1p+r3/fUM # jsoz8QVxzbW0g4BWmaf7nyfv6QCliclJ9U0WpWnlHlhSFqesmwaLefnPAe5HE9vK # P/JZJ+9NEzQasSeTVabuS7CtAenOwI84hczqUfFUC1JX/5kBwI83wZVn4R8SRJZs # fEY4TADITqqd0o6I5kjlunGUVr+BbdHz4yoqW1wi2JcCqfydAs5BGWXiqyWossHR # xtFtwO7bFbeGGlbhaY5vvyGOqTLjZca0yuA6X6UGG+xFd0BFYa3/0p9XhYlJqmuc # uFBbiViFcL2BscT+0aD2rh9epfwx4/KaM62uioY+sA== # SIG # End signature block |