AzStackHciHostTCPMetrics/AzStackHci.HostTCPMetrics.Helpers.psm1

Import-LocalizedData -BindingVariable lnTxt -FileName AzStackHci.HostTCPMetrics.Strings.psd1
function Export-ICMPFullMeshJSONReport {
    param (
        [System.Management.Automation.Runspaces.PSSession[]]$PSSession,    # Array of PSSession
        [string]$OutputPath,                                                # Local path for JSON report
        [string]$NICName                                                    # Host NIC Name
    )

    # Initialize a report object to store the results
    $report = @{}
    $hostIPs = @{ }

    # Retrieve all Ethernet adapters with IPv4 addresses from each host
    foreach ($session in $PSSession) {
        $hostname = $session.ComputerName
        $hostIPs[$hostname] = @{ }
        Log-Info "Retrieving IP addresses for host: $hostname"
        $ipAddresses = @(
            Invoke-Command -Session $session -ScriptBlock {
                Get-NetIPAddress -AddressFamily IPv4 | Where-Object { $_.InterfaceAlias -match $using:NICName } | Select-Object ifIndex, InterfaceAlias, IPAddress
            }
        )

        if (-not $ipAddresses) {
            Log-Info "! No matching IP addresses found for $hostname"
            continue
        }
        # Filter for specific interfaces (Ethernet 3 and Ethernet 4) after retrieval
        foreach ($ip in $ipAddresses) {
            if ($ip.InterfaceAlias -and $ip.IPAddress) {
                $hostIPs[$hostname][$ip.InterfaceAlias] = $ip.IPAddress
            }
        }
    }
    Write-Progress $lnTxt.StartICMPTesting

    # Perform the full mesh ping test in a single loop
    foreach ($sourceSession in $PSSession) {
        $sourceHost = $sourceSession.ComputerName
        $sourceIPs = $hostIPs[$sourceHost]
        $report[$sourceHost] = @()

        foreach ($targetHost in $hostIPs.Keys) {
            if ($sourceHost -ne $targetHost) {  # Avoid self-pinging
                $targetIPs = $hostIPs[$targetHost]
                Log-Info "# Ping from host $sourceHost to host: $targetHost"

                foreach ($sourceIP in $sourceIPs.GetEnumerator()) {  # Loop through all source IPs
                    $sourceInterface = $sourceIP.Key
                    $sourceIPAddress = $sourceIP.Value

                    if ($targetIPs.ContainsKey($sourceInterface)) {
                        $targetIPAddress = $targetIPs[$sourceInterface]

                        try {
                            # Run ping command remotely on the source machine
                            $pingResult = Invoke-Command -Session $sourceSession -ScriptBlock {
                                param ($targetIP, $sourceIP)
                                $pingOutput = ping $targetIP -S $sourceIP -n 3 2>&1

                                # Ensure the ping output is not empty and contains "Average"
                                $pingMatch = $pingOutput | Select-String -Pattern "Average = (\d+)ms"

                                if ($pingMatch) {
                                    return [PSCustomObject]@{
                                        value = [int]$pingMatch.Matches.Groups[1].Value
                                    }
                                } else {
                                    return [PSCustomObject]@{
                                        value = -1
                                    }
                                }
                            } -ArgumentList $targetIPAddress, $sourceIPAddress -ErrorAction Stop

                            # Store only AvgPingRTT.value
                            $report[$sourceHost] += [PSCustomObject]@{
                                SourceHost      = $sourceHost
                                SourceIP        = $sourceIPAddress
                                SourceInterface = $sourceInterface
                                TargetHost      = $targetHost
                                TargetIP        = $targetIPAddress
                                TargetInterface = $targetInterface
                                AvgPingRTT      = $pingResult.value  # Extracting only the value
                            }

                            # Logging output
                            if ($pingResult.value -ne -1) {
                                Log-Info ($lnTxt.ICMPTestSuccess -f $sourceIPAddress, $targetIPAddress, $($pingResult.value))
                            } else {
                                Log-Info ($lnTxt.ICMPTestFail -f $sourceIPAddress, $targetIPAddress) -Type Warning
                            }
                        }
                        catch {
                            Log-Info "### Error pinging $targetIPAddress from $sourceIPAddress : $_" -Type Error
                        }
                    }
                }
            }
        }
    }

    # Save report as JSON
    $jsonReportPath = Join-Path -Path $OutputPath -ChildPath "ICMP_Full_Mesh_Report.json"
    if ($report -and $report.Keys.Count -gt 0) {
        $report | ConvertTo-Json -Depth 3 | Set-Content -Path $jsonReportPath
        Log-Info "ICMP Full Mesh Test Report saved to: $jsonReportPath"
    } else {
        Log-Info ($lnTxt.FailSaveICMPJSON) -Type Error
    }

    return [string]$jsonReportPath
}


function Test-FullMeshTCP {
    param (
        [System.Management.Automation.Runspaces.PSSession[]]$PSSession,
        [int]$Port,
        [string]$localToolPath,
        [string]$OutputPath,
        [string]$NICName
    )

    if (-not $PSSession -or $PSSession.Count -eq 0) {
        throw "No PSSession(s) provided."
    }

    foreach ($Session in $PSSession) {
        if ($Session.State -ne 'Opened') {
            try {
                Connect-PSSession -Session $Session
            } catch {
                $PsSessionFail = "Failed to reconnect to $($Session.ComputerName): $($_.Exception.Message)"
                Log-Info $PsSessionFail -Type Error
                throw $PsSessionFail
            }
        }
    }

    # Output the final report in JSON format
    if (-not $OutputPath) {
        $OutputPath = Get-Location
        Log-Info "OutputPath was empty. Using current directory: $OutputPath"
    }
    # Ensure the output directory exists
    if (-not (Test-Path $OutputPath)) {
        New-Item -ItemType Directory -Path $OutputPath -Force | Out-Null
    }

    # Create Raw Log File
    $LogFilePath = Join-Path $OutputPath "FullMeshTCPTest.log"
    if (Test-Path $LogFilePath) {
        Remove-Item $LogFilePath -Force
    }

    $PingReportPath = Export-ICMPFullMeshJSONReport -PSSession $PSSession -OutputPath $OutputPath -NICName $NICName

    # Validate JSON file
    if (Test-Path -Path $PingReportPath) {
        Log-Info "Found Full mesh ping test report saved at $PingReportPath"
    } else {
        Log-Info "Missing Full mesh ping test report." -Type Error
        return
    }

    # Read the JSON file
    $pingReportContent = Get-Content -Path $PingReportPath -Raw | ConvertFrom-Json

    # Copy psping.exe to all hosts
    Copy-RemoteItem -SourcePath $localToolPath -PsSession $PSSession
    $remoteToolPath = Join-Path -Path $env:TEMP "psping.exe"
    Log-Info "remoteToolPath is $remoteToolPath"

    # Initialize report object to store results
    $report = @()

    # Accept EULA for psping on all remote machines
    foreach ($session in $PSSession) {
        Invoke-Command -Session $session -ScriptBlock {
            param ($remoteToolPath)
            & "$remoteToolPath" -accepteula
        } -ArgumentList $remoteToolPath > $null 2>&1
    }

    # Full mesh TCP test
    $latencyFlag = $true
    Write-Progress $lnTxt.StartTCPTesting
    foreach ($serverSession in $PSSession) {
        $serverHost = $serverSession.ComputerName

        $hostRecords = ($pingReportContent.PSObject.Properties | Where-Object { $_.Name -eq $serverHost }).Value
        if ($null -eq $hostRecords) {
            Log-Info "No data found for $serverHost"
            continue
        }

        $serverIPs = $hostRecords | Select-Object -ExpandProperty SourceIP -Unique
        Log-Info "Server IPs for ${serverHost}: $serverIPs"

        foreach ($serverIP in $serverIPs) {
            Write-Progress ($lnTxt.StartHostTCPServer -f $serverHost, $serverIP)
            Log-Info ($lnTxt.StartHostTCPServer -f $serverHost, $serverIP)
            Log-Info ($lnTxt.AddHostTCPFW -f $Port, $serverHost)
            $serverCmd = "$remoteToolPath -s ${serverIP}:$Port"
            Log-Info "[DEBUG] psping server command: $serverCmd"
            Invoke-Command -Session $serverSession -ScriptBlock {
                param ($remoteToolPath, $serverIP, $Port)
                $ruleName = "TmpAllowTCPPortForTesting"

                # Step 1: Add firewall rule
                try {
                    New-NetFirewallRule -DisplayName $ruleName -Direction Inbound -Protocol TCP -LocalPort $Port -Action Allow -Enabled True -ErrorAction Stop *>$null
                } catch {
                    throw "Failed to create firewall rule '$ruleName' for port ${Port}: $($_.Exception.Message)"
                }

                # Step 2: Verify the rule was created and is enabled
                $rule = Get-NetFirewallRule -DisplayName $ruleName -ErrorAction SilentlyContinue
                if (-not $rule -or $rule.Enabled -ne 'True') {
                    throw "Firewall rule '$ruleName' was not created or is not enabled."
                }

                # Step 3: Start the server tool
                Push-Location (Split-Path -Path $remoteToolPath)
                Start-Process -NoNewWindow -FilePath $remoteToolPath -ArgumentList "-s ${serverIP}:$Port -nobanner" *>$nulll
                Pop-Location
            } -ArgumentList $remoteToolPath, $serverIP, $Port

            Start-Sleep -Seconds 5

            foreach ($clientSession in $PSSession) {
                $clientHost = $clientSession.ComputerName

                if ($clientHost -ne $serverHost) {

                    # Latency Test
                    $latencyMsg1 = "## Latency Test from client $clientHost to IP $serverIP on server $serverHost"
                    Write-Progress $latencyMsg1
                    Log-Info $latencyMsg1
                    Add-Content -Path $LogFilePath -Value $latencyMsg1

                    Try {
                        # Run the latency test
                        $clientLatencyCmd = "$remoteToolPath -l 1m -n 5000 -h 5 ${serverIP}:$Port -nobanner"
                        Log-Info "[DEBUG] psping client latency command: $clientLatencyCmd"
                        $latencyOutput = Invoke-Command -Session $clientSession -ScriptBlock {
                            param ($serverAddress, $Port, $remoteToolPath)
                            Push-Location (Split-Path -Path $remoteToolPath)

                            # Run the remote tool and capture the output
                            $fullOutput = & $remoteToolPath -l 1m -n 5000 -h 5 "${serverAddress}:$Port" -nobanner
                            Pop-Location

                            # Filter the relevant latency data
                            $found = $false
                            $filteredOutput = @()
                            foreach ($line in $fullOutput) {
                                if ($line -match "TCP roundtrip latency statistics") { $found = $true }
                                if ($found) { $filteredOutput += $line }
                            }

                            if ($filteredOutput.Count -eq 0) {
                                throw $lnTxt.NetworkException
                            }

                            Write-Output $filteredOutput
                        } -ArgumentList $serverIP, $Port, $remoteToolPath -ErrorAction Stop

                        # Log and display the output
                        $latencyOutput | Out-String | Add-Content -Path $LogFilePath
                        Log-Info "## Latency test completed successfully from $clientHost to $serverIP on server $serverHost."
                    }
                    Catch {
                        # Improved error logging with detailed information
                        $errorMsg = "## Latency test failed from $clientHost to $serverIP on server $serverHost. Reason: $($_.Exception.Message)"
                        Log-Info $errorMsg -Type Warning
                    }

                    # Bandwidth Test
                    $bwMsg1 = "## Bandwidth Test from client $clientHost to IP $serverIP on server $serverHost"
                    Write-Progress $bwMsg1
                    Log-Info $bwMsg1
                    Add-Content -Path $LogFilePath -Value $bwMsg1
                    Try {
                        # Run the bandwidth test
                        $clientBandwidthCmd = "$remoteToolPath -b -l 1m -n 5000 -h 5 ${serverIP}:$Port -nobanner"
                        Log-Info "[DEBUG] psping bandwidth latency command: $clientBandwidthCmd"
                        $bandwidthOutput = Invoke-Command -Session $clientSession -ScriptBlock {
                            param ($serverAddress, $Port, $remoteToolPath)
                            Push-Location (Split-Path -Path $remoteToolPath)

                            # Execute the bandwidth test using the remote tool
                            $fullOutput = & $remoteToolPath -b -l 1m -n 5000 -h 5 "${serverAddress}:$Port" -nobanner
                            Pop-Location

                            # Process and filter the output for bandwidth statistics
                            $found = $false
                            $filteredOutput = @()
                            foreach ($line in $fullOutput) {
                                if ($line -match "TCP sender bandwidth statistics") { $found = $true }
                                if ($found) { $filteredOutput += $line }
                            }

                            # Handle missing bandwidth statistics
                            if ($filteredOutput.Count -eq 0) {
                                throw $lnTxt.NetworkException
                            }

                            Write-Output $filteredOutput
                        } -ArgumentList $serverIP, $Port, $remoteToolPath -ErrorAction Stop

                        # Log and display the output
                        $bandwidthOutput | Out-String | Add-Content -Path $LogFilePath
                        Log-Info "## Bandwidth test completed successfully from $clientHost to $serverIP on server $serverHost."
                    }
                    Catch {
                        # Enhanced error message for clear diagnosis
                        $errorMsg = "## Bandwidth test failed from $clientHost to $serverIP on server $serverHost. Reason: $($_.Exception.Message)"
                        Log-Info $errorMsg -Type Warning
                    }

                    # Reset Value
                    $LatencyMinMS = $LatencyMaxMS = $LatencyAvgMS = $null
                    $BandwidthMinGBs = $BandwidthMaxGBs = $BandwidthAvgGBs = $null

                    # Extract latency min/max/avg
                    $latencyLine = $latencyOutput | Where-Object { $_ -match "Minimum = .*Maximum = .*Average =" }
                    if ($latencyLine -match "Minimum = ([\d.]+)ms, Maximum = ([\d.]+)ms, Average = ([\d.]+)ms") {
                        $LatencyMinMs = [decimal]$matches[1]
                        $LatencyMaxMs = [decimal]$matches[2]
                        $LatencyAvgMs = [decimal]$matches[3]
                    }

                    # Validation Use Case 1: Avg Latency is greater than 1ms then warning
                    if ($null -eq $LatencyAvgMS -or $LatencyAvgMS -ge 1) {
                        $latencyFlag = $false
                        $TCPTestFailStatusMsg += "`n TCP Latency Exceeds 1ms: Host $clientHost -> Host $serverHost IP $serverIP is $LatencyAvgMs ms!"
                    }

                    # Extract bandwidth min/max/avg (GB/s only)
                    $bandwidthLine = $bandwidthOutput | Where-Object { $_ -match "Minimum = .*Maximum = .*Average =" }
                    if ($bandwidthLine -match "Minimum\s*=\s*([\d.]+)\s*GB/s,\s*Maximum\s*=\s*([\d.]+)\s*GB/s,\s*Average\s*=\s*([\d.]+)\s*GB/s") {
                        $BandwidthMinGBs = [decimal]$matches[1]
                        $BandwidthMaxGBs = [decimal]$matches[2]
                        $BandwidthAvgGBs = [decimal]$matches[3]
                    }

                    # Save simplified result with GB/s and ms units hardcoded in property names
                    $report += [PSCustomObject]@{
                        SourceHost        = $clientHost
                        TargetHost        = $serverHost
                        TargetIP          = $serverIP
                        LatencyMinMs      = $LatencyMinMs
                        LatencyMaxMs      = $LatencyMaxMs
                        LatencyAvgMs      = $LatencyAvgMs
                        BandwidthMinGBs   = $BandwidthMinGBs
                        BandwidthMaxGBs   = $BandwidthMaxGBs
                        BandwidthAvgGBs   = $BandwidthAvgGBs
                    }

                }
            }

            # Stop the TCP server
            Invoke-Command -Session $serverSession -ScriptBlock {
                Get-Process | Where-Object { $_.ProcessName -like "psping*" } | Stop-Process -Force
                # Clean up firewall rule after the test
                Remove-NetFirewallRule -DisplayName "TmpAllowTCPPortForTesting"
            }
            Log-Info "Removing firewall rule on TCP port $Port on $serverHost"
            Log-Info "- Stopping TCP server on $serverHost..."
            Log-Info ($lnTxt.RemoveHostTCPFW -f $Port, $serverHost)
            Log-Info ($lnTxt.StopHostTCPServer -f $serverHost)
        }
    }

    # TCP JSON Report
    $jsonReportPath = Join-Path -Path $OutputPath -ChildPath "TCP_Full_Mesh_Report.json"
    Try {
        # Convert report to JSON and save it
        $report | ConvertTo-Json -Depth 5 | Out-File -FilePath $jsonReportPath -Encoding utf8
        # Success message
        Log-Info "TCP Full Mesh JSON Report saved at: $jsonReportPath"
    }
    Catch {
        Log-Info "Failed to save TCP Full Mesh JSON report: $_" -Type Error
    }

    # Raw Log File
    if (Test-Path $LogFilePath) {
        Log-Info "TCP Full Mesh Raw Log saved at: $LogFilePath"
    }else{
        Log-Info "Failed to save TCP Full Mesh Raw Log file: $_" -Type Error
    }

    # Validation Use Case 1: Avg Latency is greater than 1ms then warning
    if (-not $latencyFlag) {
        Log-Info $TCPTestFailStatusMsg -Type 'WARNING'
        $TCPTestRstObject = @{
            Name               = "HCI_Node_TCP_Latency_Test"
            Title              = 'Host TCP Latency Exceeds 1ms'
            DisplayName        = 'Host TCP Latency Exceeds 1ms'
            Severity           = 'Warning'
            Description        = $TCPTestFailStatusMsg
            Tags               = @{ }
            Remediation        = $lnTxt.TCPLatencyRemidation
            TargetResourceID   = 'HostTCPLatencyExceeds1Ms'
            TargetResourceName = 'HostTCPLatencyExceeds1Ms'
            TargetResourceType = 'HostTCPLatencyExceeds1Ms'
            Timestamp          = [datetime]::UtcNow
            Status             = 'FAILURE'
            AdditionalData     = @{
                Source    = 'HostTCPLatencyValidated'
                Resource  = 'TCP_Full_Mesh_Report.json'
                Detail    = $TCPTestFailStatusMsg
                Status    = 'FAILURE'
                TimeStamp = [datetime]::UtcNow
            }
            HealthCheckSource  = $ENV:EnvChkrId
        }

        $TCPTestResults += New-AzStackHciResultObject @TCPTestRstObject
        return $TCPTestResults
    }

    $TCPTestSuccessStatusMsg = "All nodes meet the latency requirement."
    Log-Info $TCPTestSuccessStatusMsg -Type 'Info'

    $TCPTestRstObject = @{
        Name               = "HCI_Node_TCP_Latency_Test"
        Title              = 'Host TCP Latency Validation Passed'
        DisplayName        = 'Host TCP Latency Validation Passed'
        Severity           = 'INFO'
        Description        = 'All Nodes Meet The Latency Requirement'
        Tags               = @{ }
        Remediation        = ''
        TargetResourceID   = 'HostTCPLatencyValidated'
        TargetResourceName = 'HostTCPLatencyValidated'
        TargetResourceType = 'HostTCPLatencyValidated'
        Timestamp          = [datetime]::UtcNow
        Status             = 'SUCCESS'
        AdditionalData     = @{
            Source    = 'HostTCPLatencyValidated'
            Resource  = 'HostTCPLatencyValidated'
            Detail    = $TCPTestSuccessStatusMsg
            Status    = 'SUCCESS'
            TimeStamp = [datetime]::UtcNow
        }
        HealthCheckSource  = $ENV:EnvChkrId
    }
    $TCPTestResults += New-AzStackHciResultObject @TCPTestRstObject
    return $TCPTestResults
}



# SIG # Begin signature block
# MIIoQgYJKoZIhvcNAQcCoIIoMzCCKC8CAQExDzANBglghkgBZQMEAgEFADB5Bgor
# BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG
# KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCCd8GSxomB7ArIc
# vhfO+iDwFL4sBn2r89M+c6TtvvZTSaCCDXYwggX0MIID3KADAgECAhMzAAAEBGx0
# Bv9XKydyAAAAAAQEMA0GCSqGSIb3DQEBCwUAMH4xCzAJBgNVBAYTAlVTMRMwEQYD
# VQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNy
# b3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNpZ25p
# bmcgUENBIDIwMTEwHhcNMjQwOTEyMjAxMTE0WhcNMjUwOTExMjAxMTE0WjB0MQsw
# CQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9u
# ZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMR4wHAYDVQQDExVNaWNy
# b3NvZnQgQ29ycG9yYXRpb24wggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIB
# AQC0KDfaY50MDqsEGdlIzDHBd6CqIMRQWW9Af1LHDDTuFjfDsvna0nEuDSYJmNyz
# NB10jpbg0lhvkT1AzfX2TLITSXwS8D+mBzGCWMM/wTpciWBV/pbjSazbzoKvRrNo
# DV/u9omOM2Eawyo5JJJdNkM2d8qzkQ0bRuRd4HarmGunSouyb9NY7egWN5E5lUc3
# a2AROzAdHdYpObpCOdeAY2P5XqtJkk79aROpzw16wCjdSn8qMzCBzR7rvH2WVkvF
# HLIxZQET1yhPb6lRmpgBQNnzidHV2Ocxjc8wNiIDzgbDkmlx54QPfw7RwQi8p1fy
# 4byhBrTjv568x8NGv3gwb0RbAgMBAAGjggFzMIIBbzAfBgNVHSUEGDAWBgorBgEE
# AYI3TAgBBggrBgEFBQcDAzAdBgNVHQ4EFgQU8huhNbETDU+ZWllL4DNMPCijEU4w
# RQYDVR0RBD4wPKQ6MDgxHjAcBgNVBAsTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEW
# MBQGA1UEBRMNMjMwMDEyKzUwMjkyMzAfBgNVHSMEGDAWgBRIbmTlUAXTgqoXNzci
# tW2oynUClTBUBgNVHR8ETTBLMEmgR6BFhkNodHRwOi8vd3d3Lm1pY3Jvc29mdC5j
# b20vcGtpb3BzL2NybC9NaWNDb2RTaWdQQ0EyMDExXzIwMTEtMDctMDguY3JsMGEG
# CCsGAQUFBwEBBFUwUzBRBggrBgEFBQcwAoZFaHR0cDovL3d3dy5taWNyb3NvZnQu
# Y29tL3BraW9wcy9jZXJ0cy9NaWNDb2RTaWdQQ0EyMDExXzIwMTEtMDctMDguY3J0
# MAwGA1UdEwEB/wQCMAAwDQYJKoZIhvcNAQELBQADggIBAIjmD9IpQVvfB1QehvpC
# Ge7QeTQkKQ7j3bmDMjwSqFL4ri6ae9IFTdpywn5smmtSIyKYDn3/nHtaEn0X1NBj
# L5oP0BjAy1sqxD+uy35B+V8wv5GrxhMDJP8l2QjLtH/UglSTIhLqyt8bUAqVfyfp
# h4COMRvwwjTvChtCnUXXACuCXYHWalOoc0OU2oGN+mPJIJJxaNQc1sjBsMbGIWv3
# cmgSHkCEmrMv7yaidpePt6V+yPMik+eXw3IfZ5eNOiNgL1rZzgSJfTnvUqiaEQ0X
# dG1HbkDv9fv6CTq6m4Ty3IzLiwGSXYxRIXTxT4TYs5VxHy2uFjFXWVSL0J2ARTYL
# E4Oyl1wXDF1PX4bxg1yDMfKPHcE1Ijic5lx1KdK1SkaEJdto4hd++05J9Bf9TAmi
# u6EK6C9Oe5vRadroJCK26uCUI4zIjL/qG7mswW+qT0CW0gnR9JHkXCWNbo8ccMk1
# sJatmRoSAifbgzaYbUz8+lv+IXy5GFuAmLnNbGjacB3IMGpa+lbFgih57/fIhamq
# 5VhxgaEmn/UjWyr+cPiAFWuTVIpfsOjbEAww75wURNM1Imp9NJKye1O24EspEHmb
# DmqCUcq7NqkOKIG4PVm3hDDED/WQpzJDkvu4FrIbvyTGVU01vKsg4UfcdiZ0fQ+/
# V0hf8yrtq9CkB8iIuk5bBxuPMIIHejCCBWKgAwIBAgIKYQ6Q0gAAAAAAAzANBgkq
# hkiG9w0BAQsFADCBiDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24x
# EDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlv
# bjEyMDAGA1UEAxMpTWljcm9zb2Z0IFJvb3QgQ2VydGlmaWNhdGUgQXV0aG9yaXR5
# IDIwMTEwHhcNMTEwNzA4MjA1OTA5WhcNMjYwNzA4MjEwOTA5WjB+MQswCQYDVQQG
# EwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwG
# A1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSgwJgYDVQQDEx9NaWNyb3NvZnQg
# Q29kZSBTaWduaW5nIFBDQSAyMDExMIICIjANBgkqhkiG9w0BAQEFAAOCAg8AMIIC
# CgKCAgEAq/D6chAcLq3YbqqCEE00uvK2WCGfQhsqa+laUKq4BjgaBEm6f8MMHt03
# a8YS2AvwOMKZBrDIOdUBFDFC04kNeWSHfpRgJGyvnkmc6Whe0t+bU7IKLMOv2akr
# rnoJr9eWWcpgGgXpZnboMlImEi/nqwhQz7NEt13YxC4Ddato88tt8zpcoRb0Rrrg
# OGSsbmQ1eKagYw8t00CT+OPeBw3VXHmlSSnnDb6gE3e+lD3v++MrWhAfTVYoonpy
# 4BI6t0le2O3tQ5GD2Xuye4Yb2T6xjF3oiU+EGvKhL1nkkDstrjNYxbc+/jLTswM9
# sbKvkjh+0p2ALPVOVpEhNSXDOW5kf1O6nA+tGSOEy/S6A4aN91/w0FK/jJSHvMAh
# dCVfGCi2zCcoOCWYOUo2z3yxkq4cI6epZuxhH2rhKEmdX4jiJV3TIUs+UsS1Vz8k
# A/DRelsv1SPjcF0PUUZ3s/gA4bysAoJf28AVs70b1FVL5zmhD+kjSbwYuER8ReTB
# w3J64HLnJN+/RpnF78IcV9uDjexNSTCnq47f7Fufr/zdsGbiwZeBe+3W7UvnSSmn
# Eyimp31ngOaKYnhfsi+E11ecXL93KCjx7W3DKI8sj0A3T8HhhUSJxAlMxdSlQy90
# lfdu+HggWCwTXWCVmj5PM4TasIgX3p5O9JawvEagbJjS4NaIjAsCAwEAAaOCAe0w
# ggHpMBAGCSsGAQQBgjcVAQQDAgEAMB0GA1UdDgQWBBRIbmTlUAXTgqoXNzcitW2o
# ynUClTAZBgkrBgEEAYI3FAIEDB4KAFMAdQBiAEMAQTALBgNVHQ8EBAMCAYYwDwYD
# VR0TAQH/BAUwAwEB/zAfBgNVHSMEGDAWgBRyLToCMZBDuRQFTuHqp8cx0SOJNDBa
# BgNVHR8EUzBRME+gTaBLhklodHRwOi8vY3JsLm1pY3Jvc29mdC5jb20vcGtpL2Ny
# bC9wcm9kdWN0cy9NaWNSb29DZXJBdXQyMDExXzIwMTFfMDNfMjIuY3JsMF4GCCsG
# AQUFBwEBBFIwUDBOBggrBgEFBQcwAoZCaHR0cDovL3d3dy5taWNyb3NvZnQuY29t
# L3BraS9jZXJ0cy9NaWNSb29DZXJBdXQyMDExXzIwMTFfMDNfMjIuY3J0MIGfBgNV
# HSAEgZcwgZQwgZEGCSsGAQQBgjcuAzCBgzA/BggrBgEFBQcCARYzaHR0cDovL3d3
# dy5taWNyb3NvZnQuY29tL3BraW9wcy9kb2NzL3ByaW1hcnljcHMuaHRtMEAGCCsG
# AQUFBwICMDQeMiAdAEwAZQBnAGEAbABfAHAAbwBsAGkAYwB5AF8AcwB0AGEAdABl
# AG0AZQBuAHQALiAdMA0GCSqGSIb3DQEBCwUAA4ICAQBn8oalmOBUeRou09h0ZyKb
# C5YR4WOSmUKWfdJ5DJDBZV8uLD74w3LRbYP+vj/oCso7v0epo/Np22O/IjWll11l
# hJB9i0ZQVdgMknzSGksc8zxCi1LQsP1r4z4HLimb5j0bpdS1HXeUOeLpZMlEPXh6
# I/MTfaaQdION9MsmAkYqwooQu6SpBQyb7Wj6aC6VoCo/KmtYSWMfCWluWpiW5IP0
# wI/zRive/DvQvTXvbiWu5a8n7dDd8w6vmSiXmE0OPQvyCInWH8MyGOLwxS3OW560
# STkKxgrCxq2u5bLZ2xWIUUVYODJxJxp/sfQn+N4sOiBpmLJZiWhub6e3dMNABQam
# ASooPoI/E01mC8CzTfXhj38cbxV9Rad25UAqZaPDXVJihsMdYzaXht/a8/jyFqGa
# J+HNpZfQ7l1jQeNbB5yHPgZ3BtEGsXUfFL5hYbXw3MYbBL7fQccOKO7eZS/sl/ah
# XJbYANahRr1Z85elCUtIEJmAH9AAKcWxm6U/RXceNcbSoqKfenoi+kiVH6v7RyOA
# 9Z74v2u3S5fi63V4GuzqN5l5GEv/1rMjaHXmr/r8i+sLgOppO6/8MO0ETI7f33Vt
# Y5E90Z1WTk+/gFcioXgRMiF670EKsT/7qMykXcGhiJtXcVZOSEXAQsmbdlsKgEhr
# /Xmfwb1tbWrJUnMTDXpQzTGCGiIwghoeAgEBMIGVMH4xCzAJBgNVBAYTAlVTMRMw
# EQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVN
# aWNyb3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNp
# Z25pbmcgUENBIDIwMTECEzMAAAQEbHQG/1crJ3IAAAAABAQwDQYJYIZIAWUDBAIB
# BQCgga4wGQYJKoZIhvcNAQkDMQwGCisGAQQBgjcCAQQwHAYKKwYBBAGCNwIBCzEO
# MAwGCisGAQQBgjcCARUwLwYJKoZIhvcNAQkEMSIEIEwpF84JKwh6G/a6JqQdWxzu
# P6lQFYbd+fFOrw4iNImUMEIGCisGAQQBgjcCAQwxNDAyoBSAEgBNAGkAYwByAG8A
# cwBvAGYAdKEagBhodHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20wDQYJKoZIhvcNAQEB
# BQAEggEAFN6mpK5yG4OgqPEldEeCBbccG+ugIJAPUwrdT6fafGk2hSN7NuNOiWWP
# VqCX1C5gztEDpqM+zuUxI5i1pT26LY2E22ma/3vYndJMk+I9X8D+SHNvSEQcHaQg
# dzdtqYpUThFvPkh3Cj1QF6C+0bagvwbwCvhxAbIWA6fFBAbDZw8i+kBmYsl7pN4j
# dVENFD1SmiltyWeD3H6sEsvZ+3v++NS/1/glRT5LOHJTNVqZbdqm3gx6iiXX1HfC
# k5X7R8PEu5d1vgv86IM/0QDI39dlHlD8wGuzFOtIDOzBbRYWqjMiVvBf3X66WyKi
# eUHXBnas07Ig+lLM8aLSEfbopDcZVKGCF6wwgheoBgorBgEEAYI3AwMBMYIXmDCC
# F5QGCSqGSIb3DQEHAqCCF4UwgheBAgEDMQ8wDQYJYIZIAWUDBAIBBQAwggFZBgsq
# hkiG9w0BCRABBKCCAUgEggFEMIIBQAIBAQYKKwYBBAGEWQoDATAxMA0GCWCGSAFl
# AwQCAQUABCBo3PDGaFcgstZrZoNkTGDkQi0rPPmzp3+Y6X+nQ3H9XQIGaC3V5HNL
# GBIyMDI1MDYxMDE1NDgzMi4zMlowBIACAfSggdmkgdYwgdMxCzAJBgNVBAYTAlVT
# MRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQK
# ExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xLTArBgNVBAsTJE1pY3Jvc29mdCBJcmVs
# YW5kIE9wZXJhdGlvbnMgTGltaXRlZDEnMCUGA1UECxMeblNoaWVsZCBUU1MgRVNO
# OjMyMUEtMDVFMC1EOTQ3MSUwIwYDVQQDExxNaWNyb3NvZnQgVGltZS1TdGFtcCBT
# ZXJ2aWNloIIR+zCCBygwggUQoAMCAQICEzMAAAH4o6EmDAxASP4AAQAAAfgwDQYJ
# KoZIhvcNAQELBQAwfDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24x
# EDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlv
# bjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUtU3RhbXAgUENBIDIwMTAwHhcNMjQw
# NzI1MTgzMTA4WhcNMjUxMDIyMTgzMTA4WjCB0zELMAkGA1UEBhMCVVMxEzARBgNV
# BAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jv
# c29mdCBDb3Jwb3JhdGlvbjEtMCsGA1UECxMkTWljcm9zb2Z0IElyZWxhbmQgT3Bl
# cmF0aW9ucyBMaW1pdGVkMScwJQYDVQQLEx5uU2hpZWxkIFRTUyBFU046MzIxQS0w
# NUUwLUQ5NDcxJTAjBgNVBAMTHE1pY3Jvc29mdCBUaW1lLVN0YW1wIFNlcnZpY2Uw
# ggIiMA0GCSqGSIb3DQEBAQUAA4ICDwAwggIKAoICAQDFHbeldicPYG44N15ezYK7
# 9PmQoj5sDDxxu03nQKb8UCuNfIvhFOox7qVpD8Kp4xPGByS9mvUmtbQyLgXXmvH9
# W94aEoGahvjkOY5xXnHLHuH1OTn00CXk80wBYoAhZ/bvRJYABbFBulUiGE9YKdVX
# ei1W9qERp3ykyahJetPlns2TVGcHvQDZur0eTzAh4Le8G7ERfYTxfnQiAAezJpH2
# ugWrcSvNQQeVLxidKrfe6Lm4FysU5wU4Jkgu5UVVOASpKtfhSJfR62qLuNS0rKmA
# h+VplxXlwjlcj94LFjzAM2YGmuFgw2VjF2ZD1otENxMpa111amcm3KXl7eAe5iiP
# zG4NDRdk3LsRJHAkgrTf6tNmp9pjIzhdIrWzRpr6Y7r2+j82YnhH9/X4q5wE8njJ
# R1uolYzfEy8HAtjJy+KAj9YriSA+iDRQE1zNpDANVelxT5Mxw69Y/wcFaZYlAiZN
# kicAWK9epRoFujfAB881uxCm800a7/XamDQXw78J1F+A8d86EhZDQPwAsJj4uyLB
# vNx6NutWXg31+fbA6DawNrxF82gPrXgjSkWPL+WrU2wGj1XgZkGKTNftmNYJGB3U
# UIFcal+kOKQeNDTlg6QBqR1YNPZsZJpRkkZVi16kik9MCzWB3+9SiBx2IvnWjuyG
# 4ciUHpBJSJDbhdiFFttAIQIDAQABo4IBSTCCAUUwHQYDVR0OBBYEFL3OxnPPntCV
# Pmeu3+iK0u/U5Du2MB8GA1UdIwQYMBaAFJ+nFV0AXmJdg/Tl0mWnG1M1GelyMF8G
# A1UdHwRYMFYwVKBSoFCGTmh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMv
# Y3JsL01pY3Jvc29mdCUyMFRpbWUtU3RhbXAlMjBQQ0ElMjAyMDEwKDEpLmNybDBs
# BggrBgEFBQcBAQRgMF4wXAYIKwYBBQUHMAKGUGh0dHA6Ly93d3cubWljcm9zb2Z0
# LmNvbS9wa2lvcHMvY2VydHMvTWljcm9zb2Z0JTIwVGltZS1TdGFtcCUyMFBDQSUy
# MDIwMTAoMSkuY3J0MAwGA1UdEwEB/wQCMAAwFgYDVR0lAQH/BAwwCgYIKwYBBQUH
# AwgwDgYDVR0PAQH/BAQDAgeAMA0GCSqGSIb3DQEBCwUAA4ICAQBh+TwbPOkRWcaX
# vLqhejK0JvjYfHpM4DT52RoEjfp+0MT20u5tRr/ExscHmtw2JGEUdn3dF590+lzj
# 4UXQMCXmU/zEoA77b3dFY8oMU4UjGC1ljTy3wP1xJCmAZTPLDeURNl5s0sQDXsD8
# JOkDYX26HyPzgrKB4RuP5uJ1YOIR9rKgfYDn/nLAknEi4vMVUdpy9bFIIqgX2GVK
# tlIbl9dZLedqZ/i23r3RRPoAbJYsVZ7z3lygU/Gb+bRQgyOOn1VEUfudvc2DZDiA
# 9L0TllMxnqcCWZSJwOPQ1cCzbBC5CudidtEAn8NBbfmoujsNrD0Cwi2qMWFsxwbr
# yANziPvgvYph7/aCgEcvDNKflQN+1LUdkjRlGyqY0cjRNm+9RZf1qObpJ8sFMS2h
# OjqAs5fRQP/2uuEaN2SILDhLBTmiwKWCqCI0wrmd2TaDEWUNccLIunmoHoGg+lzz
# ZGE7TILOg/2C/vO/YShwBYSyoTn7Raa7m5quZ+9zOIt9TVJjbjQ5lbyV3ixLx+fJ
# uf+MMyYUCFrNXXMfRARFYSx8tKnCQ5doiZY0UnmWZyd/VVObpyZ9qxJxi0SWmOpn
# 0aigKaTVcUCk5E+z887jchwWY9HBqC3TSJBLD6sF4gfTQpCr4UlP/rZIHvSD2D9H
# xNLqTpv/C3ZRaGqtb5DyXDpfOB7H9jCCB3EwggVZoAMCAQICEzMAAAAVxedrngKb
# SZkAAAAAABUwDQYJKoZIhvcNAQELBQAwgYgxCzAJBgNVBAYTAlVTMRMwEQYDVQQI
# EwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3Nv
# ZnQgQ29ycG9yYXRpb24xMjAwBgNVBAMTKU1pY3Jvc29mdCBSb290IENlcnRpZmlj
# YXRlIEF1dGhvcml0eSAyMDEwMB4XDTIxMDkzMDE4MjIyNVoXDTMwMDkzMDE4MzIy
# NVowfDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcT
# B1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEmMCQGA1UE
# AxMdTWljcm9zb2Z0IFRpbWUtU3RhbXAgUENBIDIwMTAwggIiMA0GCSqGSIb3DQEB
# AQUAA4ICDwAwggIKAoICAQDk4aZM57RyIQt5osvXJHm9DtWC0/3unAcH0qlsTnXI
# yjVX9gF/bErg4r25PhdgM/9cT8dm95VTcVrifkpa/rg2Z4VGIwy1jRPPdzLAEBjo
# YH1qUoNEt6aORmsHFPPFdvWGUNzBRMhxXFExN6AKOG6N7dcP2CZTfDlhAnrEqv1y
# aa8dq6z2Nr41JmTamDu6GnszrYBbfowQHJ1S/rboYiXcag/PXfT+jlPP1uyFVk3v
# 3byNpOORj7I5LFGc6XBpDco2LXCOMcg1KL3jtIckw+DJj361VI/c+gVVmG1oO5pG
# ve2krnopN6zL64NF50ZuyjLVwIYwXE8s4mKyzbnijYjklqwBSru+cakXW2dg3viS
# kR4dPf0gz3N9QZpGdc3EXzTdEonW/aUgfX782Z5F37ZyL9t9X4C626p+Nuw2TPYr
# bqgSUei/BQOj0XOmTTd0lBw0gg/wEPK3Rxjtp+iZfD9M269ewvPV2HM9Q07BMzlM
# jgK8QmguEOqEUUbi0b1qGFphAXPKZ6Je1yh2AuIzGHLXpyDwwvoSCtdjbwzJNmSL
# W6CmgyFdXzB0kZSU2LlQ+QuJYfM2BjUYhEfb3BvR/bLUHMVr9lxSUV0S2yW6r1AF
# emzFER1y7435UsSFF5PAPBXbGjfHCBUYP3irRbb1Hode2o+eFnJpxq57t7c+auIu
# rQIDAQABo4IB3TCCAdkwEgYJKwYBBAGCNxUBBAUCAwEAATAjBgkrBgEEAYI3FQIE
# FgQUKqdS/mTEmr6CkTxGNSnPEP8vBO4wHQYDVR0OBBYEFJ+nFV0AXmJdg/Tl0mWn
# G1M1GelyMFwGA1UdIARVMFMwUQYMKwYBBAGCN0yDfQEBMEEwPwYIKwYBBQUHAgEW
# M2h0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMvRG9jcy9SZXBvc2l0b3J5
# Lmh0bTATBgNVHSUEDDAKBggrBgEFBQcDCDAZBgkrBgEEAYI3FAIEDB4KAFMAdQBi
# AEMAQTALBgNVHQ8EBAMCAYYwDwYDVR0TAQH/BAUwAwEB/zAfBgNVHSMEGDAWgBTV
# 9lbLj+iiXGJo0T2UkFvXzpoYxDBWBgNVHR8ETzBNMEugSaBHhkVodHRwOi8vY3Js
# Lm1pY3Jvc29mdC5jb20vcGtpL2NybC9wcm9kdWN0cy9NaWNSb29DZXJBdXRfMjAx
# MC0wNi0yMy5jcmwwWgYIKwYBBQUHAQEETjBMMEoGCCsGAQUFBzAChj5odHRwOi8v
# d3d3Lm1pY3Jvc29mdC5jb20vcGtpL2NlcnRzL01pY1Jvb0NlckF1dF8yMDEwLTA2
# LTIzLmNydDANBgkqhkiG9w0BAQsFAAOCAgEAnVV9/Cqt4SwfZwExJFvhnnJL/Klv
# 6lwUtj5OR2R4sQaTlz0xM7U518JxNj/aZGx80HU5bbsPMeTCj/ts0aGUGCLu6WZn
# OlNN3Zi6th542DYunKmCVgADsAW+iehp4LoJ7nvfam++Kctu2D9IdQHZGN5tggz1
# bSNU5HhTdSRXud2f8449xvNo32X2pFaq95W2KFUn0CS9QKC/GbYSEhFdPSfgQJY4
# rPf5KYnDvBewVIVCs/wMnosZiefwC2qBwoEZQhlSdYo2wh3DYXMuLGt7bj8sCXgU
# 6ZGyqVvfSaN0DLzskYDSPeZKPmY7T7uG+jIa2Zb0j/aRAfbOxnT99kxybxCrdTDF
# NLB62FD+CljdQDzHVG2dY3RILLFORy3BFARxv2T5JL5zbcqOCb2zAVdJVGTZc9d/
# HltEAY5aGZFrDZ+kKNxnGSgkujhLmm77IVRrakURR6nxt67I6IleT53S0Ex2tVdU
# CbFpAUR+fKFhbHP+CrvsQWY9af3LwUFJfn6Tvsv4O+S3Fb+0zj6lMVGEvL8CwYKi
# excdFYmNcP7ntdAoGokLjzbaukz5m/8K6TT4JDVnK+ANuOaMmdbhIurwJ0I9JZTm
# dHRbatGePu1+oDEzfbzL6Xu/OHBE0ZDxyKs6ijoIYn/ZcGNTTY3ugm2lBRDBcQZq
# ELQdVTNYs6FwZvKhggNWMIICPgIBATCCAQGhgdmkgdYwgdMxCzAJBgNVBAYTAlVT
# MRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQK
# ExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xLTArBgNVBAsTJE1pY3Jvc29mdCBJcmVs
# YW5kIE9wZXJhdGlvbnMgTGltaXRlZDEnMCUGA1UECxMeblNoaWVsZCBUU1MgRVNO
# OjMyMUEtMDVFMC1EOTQ3MSUwIwYDVQQDExxNaWNyb3NvZnQgVGltZS1TdGFtcCBT
# ZXJ2aWNloiMKAQEwBwYFKw4DAhoDFQC2RC395tZJDkOcb5opHM8QsIUT0aCBgzCB
# gKR+MHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQH
# EwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNV
# BAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1wIFBDQSAyMDEwMA0GCSqGSIb3DQEBCwUA
# AgUA6/KwgTAiGA8yMDI1MDYxMDEzMjQxN1oYDzIwMjUwNjExMTMyNDE3WjB0MDoG
# CisGAQQBhFkKBAExLDAqMAoCBQDr8rCBAgEAMAcCAQACAgGDMAcCAQACAhsaMAoC
# BQDr9AIBAgEAMDYGCisGAQQBhFkKBAIxKDAmMAwGCisGAQQBhFkKAwKgCjAIAgEA
# AgMHoSChCjAIAgEAAgMBhqAwDQYJKoZIhvcNAQELBQADggEBAGHs1JCg88sikYIt
# Lwbit2kttXJgcSo1DWkO0JPGVWUZxcL9eDij00cUPopJujQ8nGHl8k7s3rasESl+
# kTMJLGXtYyYWH7UgVS2wXBq/eKQGWp2yIpbTImRNxWz9z7RmW3n0cGajbST9huim
# lvFVUJL/J5FX0pKBp8AT4VWVAp36mTXbKlD57mqmk2wjspazGVLonDy1NUhc+YBz
# 7RNcwTuK5KcXu35isVyfqp/am9a8aJ4WX1k6sfuAp3QiyeqZeZWUf4UyABBvxcM7
# vhocjuSalEB7EFi6jxLHH04Q+LUQoRFm220xNgnEZnwObo8IBis9DhxJkCJcnBFK
# 8YR51dYxggQNMIIECQIBATCBkzB8MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2Fz
# aGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENv
# cnBvcmF0aW9uMSYwJAYDVQQDEx1NaWNyb3NvZnQgVGltZS1TdGFtcCBQQ0EgMjAx
# MAITMwAAAfijoSYMDEBI/gABAAAB+DANBglghkgBZQMEAgEFAKCCAUowGgYJKoZI
# hvcNAQkDMQ0GCyqGSIb3DQEJEAEEMC8GCSqGSIb3DQEJBDEiBCD/etn/qRIntWpX
# iLthSboZM8g8mq07HMhVm/TbRku6tzCB+gYLKoZIhvcNAQkQAi8xgeowgecwgeQw
# gb0EIO/MM/JfDVSQBQVi3xtHhR2Mz3RC/nGdVqIoPcjRnPdaMIGYMIGApH4wfDEL
# MAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1v
# bmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEmMCQGA1UEAxMdTWlj
# cm9zb2Z0IFRpbWUtU3RhbXAgUENBIDIwMTACEzMAAAH4o6EmDAxASP4AAQAAAfgw
# IgQgaLBzyYXNwIdr2AA6Rfg8RDPGjX7hWdVFTFl4CBzsFEkwDQYJKoZIhvcNAQEL
# BQAEggIAXlB0YpLrpWlvQ57Nza8y4hnwXKC0pFO6NEoGKEhh0DG0ihgM1Ey/ULzt
# NTLbSFvNAjWdfl46i7PvpvFjGSNV3vWsXMmQu5LYSih+FHHRdnC3xTCs1AThDyPr
# JmQpQ7blefHNcHsCSkoYDLBFBVnnP+g3F4bkkWxEnwkrt1bHD3jjh+FPE1E5GiQN
# paFzRImqDq3qMeckoP+WIR/1vH01KIp0D9ZDibMIklCHHUQk1J1guEyekL+Hfzz6
# CtxhC2BBpweLVDRRtJDhjzOufoye608m0/GmOsWQma+Do8GGBOY31qIYTXx9PBPc
# uAm60goXqd76EKWZSIa946umjMuX9eCvOSAo4raoZjvwqs+p1XViilK5zcpnHy46
# 8KDNMJGh20c3khC7dlSH0AmVuqDx78WHQJlzSf05zlDfBFYgy4K4nhL3OdXsm3G9
# 0gYcSKrTafXwYFgvgHxsP0E+rCCK/5WnahwbHSWK4ERD4p/sDY1wPLwVqc/nystk
# w2J3K20EtIy44pmM3kXGMppTTsp4jkRiZW3cz/fcQtFwI1EWbkEkXH4gVU2eekPD
# VUhxUKVAKfp+rTcNuucBxkCfi05xJi4l73XXRFYWWP/T/Pedmyqf+RxNS8bMyZtS
# CWlzDiXQVkwF+klQtP7Jw0q6Mqs2qXhmfE36ATAcC7kw5xqpEiI=
# SIG # End signature block