Private/Deploy-Accelerator-Helpers/Invoke-Terraform.ps1

function Invoke-Terraform {
    [CmdletBinding(SupportsShouldProcess = $true)]
    param (
        [Parameter(Mandatory = $false)]
        [string] $moduleFolderPath,

        [Parameter(Mandatory = $false)]
        [string] $tfvarsFileName = "",

        [Parameter(Mandatory = $false)]
        [switch] $autoApprove,

        [Parameter(Mandatory = $false)]
        [switch] $destroy,

        [Parameter(Mandatory = $false)]
        [string] $output = "",

        [Parameter(Mandatory = $false)]
        [string] $outputFilePath = "",

        [Parameter(Mandatory = $false)]
        [switch] $silent
    )

    # Resolve to absolute path for `terraform -chdir` switch
    $moduleFolderPath = (Resolve-Path $moduleFolderPath).Path

    if ($PSCmdlet.ShouldProcess("Apply Terraform", "modify")) {
        # Check and Set Subscription ID
        $removeSubscriptionId = $false
        if ($null -eq $env:ARM_SUBSCRIPTION_ID -or $env:ARM_SUBSCRIPTION_ID -eq "") {
            Write-Verbose "Setting environment variable ARM_SUBSCRIPTION_ID"
            $subscriptionId = $(az account show --query id -o tsv)
            if ($null -eq $subscriptionId -or $subscriptionId -eq "") {
                Write-Error "Subscription ID not found. Please ensure you are logged in to Azure and have selected a subscription. Use 'az account show' to check."
                return
            }
            $env:ARM_SUBSCRIPTION_ID = $subscriptionId
            $removeSubscriptionId = $true
            Write-Verbose "Environment variable ARM_SUBSCRIPTION_ID set to $subscriptionId"
        }

        terraform -chdir="$moduleFolderPath" init
        $action = "apply"
        if ($destroy) {
            $action = "destroy"
        }

        if (!$silent) {
            Write-InformationColored "Terraform init has completed, now running the $action..." -ForegroundColor Green -NewLineBefore -InformationAction Continue
        }

        $planFileName = "tfplan"

        # Start timer
        $StopWatch = New-Object -TypeName System.Diagnostics.Stopwatch
        $StopWatch.Start()

        $command = "terraform"
        $arguments = @()
        $arguments += "-chdir=$moduleFolderPath"
        $arguments += "plan"
        $arguments += "-out=$planFileName"
        $arguments += "-input=false"
        if ($tfvarsFileName -ne "") {
            $arguments += "-var-file=$tfvarsFileName"
        }

        if ($destroy) {
            $arguments += "-destroy"
        }

        if (!$silent) {
            Write-InformationColored "Running Plan Command for $action : $command $arguments" -ForegroundColor Green -NewLineBefore -InformationAction Continue
            & $command $arguments
        } else {
            & $command $arguments | Write-Verbose
        }

        $exitCode = $LASTEXITCODE

        # Stop and display timer
        $StopWatch.Stop()
        if (!$silent) {
            Write-InformationColored "Time taken to complete Terraform plan:" -ForegroundColor Green -NewLineBefore -InformationAction Continue
        }
        $StopWatch.Elapsed | Format-Table

        if ($exitCode -ne 0) {
            Write-InformationColored "Terraform plan for $action failed with exit code $exitCode. Please review the error and try again or raise an issue." -ForegroundColor Red -NewLineBefore -InformationAction Continue
            throw "Terraform plan failed with exit code $exitCode. Please review the error and try again or raise an issue."
        }

        if (!$autoApprove) {
            Write-InformationColored "Terraform plan has completed, please review the plan and confirm you wish to continue." -ForegroundColor Yellow -NewLineBefore -InformationAction Continue
            $choices = [System.Management.Automation.Host.ChoiceDescription[]] @("&Yes", "&No")
            $message = "Please confirm you wish to apply the plan."
            $title = "Confirm Terraform plan"
            $resultIndex = $host.ui.PromptForChoice($title, $message, $choices, 0)
            if ($resultIndex -eq 1) {
                Write-InformationColored "You have chosen not to apply the plan. Exiting..." -ForegroundColor Red -NewLineBefore -InformationAction Continue
                return
            }
        }

        # Start timer
        $StopWatch = New-Object -TypeName System.Diagnostics.Stopwatch
        $StopWatch.Start()

        $command = "terraform"
        $arguments = @()
        $arguments += "-chdir=$moduleFolderPath"
        $arguments += "apply"
        $arguments += "-auto-approve"
        $arguments += "-input=false"
        $arguments += "$planFileName"

        if (!$silent) {
            Write-InformationColored "Running Apply Command for $action : $command $arguments" -ForegroundColor Green -NewLineBefore -InformationAction Continue
            & $command $arguments
        } else {
            & $command $arguments | Write-Verbose
        }

        $exitCode = $LASTEXITCODE

        $currentAttempt = 0
        $maxAttempts = 5

        while ($exitCode -ne 0 -and $currentAttempt -lt $maxAttempts) {
            Write-InformationColored "Terraform $action failed with exit code $exitCode. This is likely a transient issue, so we are retrying..." -ForegroundColor Yellow -NewLineBefore -InformationAction Continue
            $currentAttempt++
            $command = "terraform"
            $arguments = @()
            $arguments += "-chdir=$moduleFolderPath"
            $arguments += "apply"
            $arguments += "-auto-approve"
            $arguments += "-input=false"
            if ($tfvarsFileName -ne "") {
                $arguments += "-var-file=$tfvarsFileName"
            }
            if ($destroy) {
                $arguments += "-destroy"
            }

            Write-InformationColored "Running Apply Command for $action : $command $arguments" -ForegroundColor Green -NewLineBefore -InformationAction Continue
            & $command $arguments
            $exitCode = $LASTEXITCODE
        }

        if ($removeSubscriptionId) {
            Write-Verbose "Removing environment variable ARM_SUBSCRIPTION_ID that was set prior to this run"
            $env:ARM_SUBSCRIPTION_ID = $null
        }

        # Stop and display timer
        $StopWatch.Stop()
        if (!$silent) {
            Write-InformationColored "Time taken to complete Terraform apply:" -ForegroundColor Green -NewLineBefore -InformationAction Continue
        }
        $StopWatch.Elapsed | Format-Table

        if ($exitCode -ne 0) {
            Write-InformationColored "Terraform $action failed with exit code $exitCode after $maxAttempts attempts. Please review the error and try again or raise an issue." -ForegroundColor Red -NewLineBefore -InformationAction Continue
            throw "Terraform $action failed with exit code $exitCode after $maxAttempts attempts. Please review the error and try again or raise an issue."
        } else {
            if ($output -ne "") {
                if ($outputFilePath -eq "") {
                    $outputFilePath = Join-Path $moduleFolderPath "output.json"
                }
                $command = "terraform"
                $arguments = @()
                $arguments += "-chdir=$moduleFolderPath"
                $arguments += "output"
                $arguments += "-json"
                $arguments += "$output"

                Write-Verbose "Outputting $output to $outputFilePath"
                Write-Verbose "Running Output Command: $command $arguments"
                & $command $arguments > $outputFilePath
            }
        }
    }
}

# SIG # Begin signature block
# MIIoPgYJKoZIhvcNAQcCoIIoLzCCKCsCAQExDzANBglghkgBZQMEAgEFADB5Bgor
# BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG
# KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCAYWL+NEXLl0SuX
# 26H0IfFU/JncYienaosoIDOxgIt+dqCCDYUwggYDMIID66ADAgECAhMzAAAEhJji
# 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/Xmfwb1tbWrJUnMTDXpQzTGCGg8wghoLAgEBMIGVMH4x
# CzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRt
# b25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01p
# Y3Jvc29mdCBDb2RlIFNpZ25pbmcgUENBIDIwMTECEzMAAASEmOIS4HijMV0AAAAA
# BIQwDQYJYIZIAWUDBAIBBQCggbAwGQYJKoZIhvcNAQkDMQwGCisGAQQBgjcCAQQw
# HAYKKwYBBAGCNwIBCzEOMAwGCisGAQQBgjcCARUwLwYJKoZIhvcNAQkEMSIEII5l
# AGYxaR3+X8yMEhSQKpWJW4Q58yCg0CqdbuUan45/MEQGCisGAQQBgjcCAQwxNjA0
# oBSAEgBNAGkAYwByAG8AcwBvAGYAdKEcgBpodHRwczovL3d3dy5taWNyb3NvZnQu
# Y29tIDANBgkqhkiG9w0BAQEFAASCAQAnjPs5gzRhxq74eW99A+2Bm5Z7hWOsT62E
# MEL+ZIR5alO2ZImAXn3HgfXvinXG6iaEQytZxmKs828o55cTZq29swFTqmzb+ZlX
# sWORJ5G8iXMViGrDMFpZdevTMuU1ZZQkUqz07s3Boclbf/Qs/5yJXWNMJ+ewfc7y
# HgulpcTNMWJ7Zyq/5mKHGMOoqWe1l0bqq1Mge6Gd2pEjQdLg5qNCBkeQUEnhEOEN
# Xzbsw9B4ZyAayEoT7XWwrO+jsqpJVJmdTL4a/HSms9iSzFUd7X02a4q48FrQCAE4
# bA88g9ytRKAATcsq3hcjVxM6hGj7Cg29vn4GW9mwD61+JnVpXdi8oYIXlzCCF5MG
# CisGAQQBgjcDAwExgheDMIIXfwYJKoZIhvcNAQcCoIIXcDCCF2wCAQMxDzANBglg
# hkgBZQMEAgEFADCCAVIGCyqGSIb3DQEJEAEEoIIBQQSCAT0wggE5AgEBBgorBgEE
# AYRZCgMBMDEwDQYJYIZIAWUDBAIBBQAEIDx4nx6xp4+Dx007PKMIULN+rRfODYsI
# 2nvb2gpWMBuYAgZoo6PcoZAYEzIwMjUwOTE2MTMzMTIzLjI0OVowBIACAfSggdGk
# gc4wgcsxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQH
# EwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJTAjBgNV
# BAsTHE1pY3Jvc29mdCBBbWVyaWNhIE9wZXJhdGlvbnMxJzAlBgNVBAsTHm5TaGll
# bGQgVFNTIEVTTjozNzAzLTA1RTAtRDk0NzElMCMGA1UEAxMcTWljcm9zb2Z0IFRp
# bWUtU3RhbXAgU2VydmljZaCCEe0wggcgMIIFCKADAgECAhMzAAACCkeyFNnusrMP
# AAEAAAIKMA0GCSqGSIb3DQEBCwUAMHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpX
# YXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQg
# Q29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1wIFBDQSAy
# MDEwMB4XDTI1MDEzMDE5NDI1N1oXDTI2MDQyMjE5NDI1N1owgcsxCzAJBgNVBAYT
# AlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYD
# VQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJTAjBgNVBAsTHE1pY3Jvc29mdCBB
# bWVyaWNhIE9wZXJhdGlvbnMxJzAlBgNVBAsTHm5TaGllbGQgVFNTIEVTTjozNzAz
# LTA1RTAtRDk0NzElMCMGA1UEAxMcTWljcm9zb2Z0IFRpbWUtU3RhbXAgU2Vydmlj
# ZTCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBALLs3PASlvsGnACT0snT
# RerfRNxONzA1nzRPEC0BuFJonaTCckDUC78Oj398QGMAe/oya23d+Y8N4gmdDtUF
# 4irxFx+NyOts8iDfADm/kxB+U81IE069xdE/59mcDLQQsPN+ckecKRk2xBRnsYXs
# vFQtMo5hjZgnDhOuZweiGRjoOMJnLGGqYZDDB1uOg9ZFti7+jMV6b/J8k/KNUGqX
# XTrxtxWHnwDxzkIPpNY38ve743L7s7z4O96vqoFPjgTLul89kxnUeLvK8Iu/ksbN
# IHqjY4PkYhnLvPiSHxRgd3OOf1bH5BnbbfdIlAq1mGkju4d/egxODTNqdB/PuaC5
# 15+YYGtASDWjHef7dOLOHQP3NWY1A/2wWOw9C00Zn0gP0fwD6QUifHiatLEDZLIY
# w5BzGUYzfSS0gozeKoK4URT0IdUyk33j/V+XhPo+icn7+hYmoWVq1i4wq/Zny6Km
# ucVwDEKk6wMhUE70rW8d4AyJBBSVwT0IPnfojVgueY7aodqA8+ZeF04asdScJS2i
# nbV6W6QeHvmr/nMf46c16Snm52mNA1kK+JgBl0ewueRTQ19QCvqvVSoNxKrXJ/lG
# LCKYHxKOx2YeWXiA+OapWLT+uaczWgARCbc/JZxNBCJtguK4o3tjiKjlslNXCb69
# FFuknlQv8PfcL//uqcdt6hYFAgMBAAGjggFJMIIBRTAdBgNVHQ4EFgQUAq8XQQSP
# gDI99jxb+quwC9+1nCQwHwYDVR0jBBgwFoAUn6cVXQBeYl2D9OXSZacbUzUZ6XIw
# XwYDVR0fBFgwVjBUoFKgUIZOaHR0cDovL3d3dy5taWNyb3NvZnQuY29tL3BraW9w
# cy9jcmwvTWljcm9zb2Z0JTIwVGltZS1TdGFtcCUyMFBDQSUyMDIwMTAoMSkuY3Js
# MGwGCCsGAQUFBwEBBGAwXjBcBggrBgEFBQcwAoZQaHR0cDovL3d3dy5taWNyb3Nv
# ZnQuY29tL3BraW9wcy9jZXJ0cy9NaWNyb3NvZnQlMjBUaW1lLVN0YW1wJTIwUENB
# JTIwMjAxMCgxKS5jcnQwDAYDVR0TAQH/BAIwADAWBgNVHSUBAf8EDDAKBggrBgEF
# BQcDCDAOBgNVHQ8BAf8EBAMCB4AwDQYJKoZIhvcNAQELBQADggIBALCTc9bQLKM5
# qpcIw66xN17t6nYLA//FXUnej0oCIpJ2rxrZRHlEIdiGcmx56bB4xQfaQ15+mcCH
# YGfwfaikPO0MmSPw5hF38pDKwTXY3Bpjco7ABbxGDw51e/l9Rkiw5aCmuQJAzRWg
# Ud1dgTGQk3cTMFtQJTDfJviAAU8rnpP7pw+kjwwkskJBcc0yC2LZBmFf0qR9EB98
# VgwVymxsj6jOILkGuuTffrVwkUeGAQlHQnjaeplMZKBSkMYdJkBEao1wBXmH45AM
# t4XandRHH8/oxxSDWcnaAw9gGwP5vB30HRz9chn0y3e6OUQs+mSKOMJ1JnGHO7Yz
# dJlYmvnu5j4UL4uym8g4fU6mh9qeHckYOiw1wAS73JQxYqwQEyeAPpVZBJBhnpf0
# XzTMKh5nBOrSL0C6OdjPPHlbZ8SBl6NG7orUYIBKbO02VgmUAKOpp9ZGh9OqQmFX
# 8na/3tfeu4O9+m465ChS1UDBesOwbY04G69Wyjkp1bniEFKzP/j45EHiouerj8Y2
# 1qNQeispEocY6AWjwMppcb5Q0A3CEY3EdsgtJrn0/ETEyAKMFE/fzbzIYqyq5ITP
# Ms1afPlp/1mZ4fjzT1/Ai20jjUmS6Jj1fqGYyYti/w5jfi+84r7PLGRL70FQtUA/
# qC37JodoGWuWfwaLj90GvbLpKuv/nqDQMIIHcTCCBVmgAwIBAgITMwAAABXF52ue
# 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
# BmoQtB1VM1izoXBm8qGCA1AwggI4AgEBMIH5oYHRpIHOMIHLMQswCQYDVQQGEwJV
# UzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UE
# ChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSUwIwYDVQQLExxNaWNyb3NvZnQgQW1l
# cmljYSBPcGVyYXRpb25zMScwJQYDVQQLEx5uU2hpZWxkIFRTUyBFU046MzcwMy0w
# NUUwLUQ5NDcxJTAjBgNVBAMTHE1pY3Jvc29mdCBUaW1lLVN0YW1wIFNlcnZpY2Wi
# IwoBATAHBgUrDgMCGgMVANEAxUmUDpsqr3dWe7dSQmCbkeVhoIGDMIGApH4wfDEL
# MAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1v
# bmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEmMCQGA1UEAxMdTWlj
# cm9zb2Z0IFRpbWUtU3RhbXAgUENBIDIwMTAwDQYJKoZIhvcNAQELBQACBQDsc7Ii
# MCIYDzIwMjUwOTE2MDk1MzM4WhgPMjAyNTA5MTcwOTUzMzhaMHcwPQYKKwYBBAGE
# WQoEATEvMC0wCgIFAOxzsiICAQAwCgIBAAICL2QCAf8wBwIBAAICEvUwCgIFAOx1
# A6ICAQAwNgYKKwYBBAGEWQoEAjEoMCYwDAYKKwYBBAGEWQoDAqAKMAgCAQACAweh
# IKEKMAgCAQACAwGGoDANBgkqhkiG9w0BAQsFAAOCAQEAHE+JYGj+kqmWKuJxPciM
# fNtCx7+4qtRl664RITekMkW8tbYvzJmSoaTWLrhUy6Zr2ufu+lUnwhH4u9pFas+6
# PhClUedgmqWhTY1D3QZDx4Ux3kk0V+MJwHzDjB33WUgWW6CB8vZ9S8yxNGBh/ARQ
# +s31L6mTpvH6ReA+xJ38wAd8zfTao8/c7LK3sTdC4sGkj9y37mVSuO/B4DDff5t2
# jvjsTbQHuXdy9P2+0PZeR1pPWJpj1ysMwLkYNJ9pdTdu1GUCc7bNvV5wMybLNNBK
# IXMrVPO7ZuVpX5bhucm9incrG7q7+5PCdJnoKZg1GBhqhLTw3O+0tNyVK7jpsXxg
# mDGCBA0wggQJAgEBMIGTMHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5n
# dG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9y
# YXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1wIFBDQSAyMDEwAhMz
# AAACCkeyFNnusrMPAAEAAAIKMA0GCWCGSAFlAwQCAQUAoIIBSjAaBgkqhkiG9w0B
# CQMxDQYLKoZIhvcNAQkQAQQwLwYJKoZIhvcNAQkEMSIEIEEYBYdskYfb4v2pygB4
# SUiAmvWaSabN39tY7vQt57AEMIH6BgsqhkiG9w0BCRACLzGB6jCB5zCB5DCBvQQg
# TZrL/LR6kr9fdnTcpNyWioTy70hdG107sx7HSHD35JowgZgwgYCkfjB8MQswCQYD
# VQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEe
# MBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSYwJAYDVQQDEx1NaWNyb3Nv
# ZnQgVGltZS1TdGFtcCBQQ0EgMjAxMAITMwAAAgpHshTZ7rKzDwABAAACCjAiBCB/
# jQuWM9NSA1MSlgx+rTaBqa7N6nE34f7dZ/fqYDKmczANBgkqhkiG9w0BAQsFAASC
# AgApaRIWP3wafKfDL+Tir7Wx81IuQ+6O2kka7skU/xgkpR9ydKIzYJYJPqS0C4VS
# 3PQGTwIX/AraTJASrWK3ckGeMOfgDOyoBXLoKX+sAMrZ5dXWGl9C5gLPKYWb6ys1
# ZfvU6wCB4cb2KPhbAGRFmNeEEoMhZ2PoaarG0NlCnca8GFpD/v89MRIxFfUuOydV
# vgin+0ZMuj3onWWsYKujkxmjez6Ni0HpeKGvOBYO1sBkc3OQ35BQo4HAW6djp38n
# 2cUU0jNhDMNQuT2kxmdFJEpjj3Jy1E4JMlGG5cib6dbPY7lF529q5XA/xW5rjyZt
# tGratPma47/H9u/BH6Jx5V9zTZ7NGdXm71e/TimElMi0vWBTyaUXq0ahUwrfW58x
# W8sRSG9v0/xGsFvCgNMbsZLKk6OrOHoIN6/YO+8n42Cd05Rf7Mb2j0a5+/RYeLtP
# 5WhP+bx/nrJ/PZ54hAac8OUnbYhQq2Eg99fUU7uUx3xKoPwgJVu464yUqYNSOAQ4
# lRNHNoxZRTQuX8H377hfWUd46f928a7Mzyw2Cxyp9w8k71vkiFUGUoo0lL0YaUqo
# 2yEMtYv/cNR+CksSvmEmSU+fZOsWorYOx3dMMxwyjQFG6lthp849ec5I6KbR+zN/
# LPPpJZaCRmMC6n8IkbkcgxGPXL1qkwktq/sYw3LukKxe/Q==
# SIG # End signature block