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
# MIIoRgYJKoZIhvcNAQcCoIIoNzCCKDMCAQExDzANBglghkgBZQMEAgEFADB5Bgor
# BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG
# KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCCd8GSxomB7ArIc
# vhfO+iDwFL4sBn2r89M+c6TtvvZTSaCCDXYwggX0MIID3KADAgECAhMzAAAEhV6Z
# 7A5ZL83XAAAAAASFMA0GCSqGSIb3DQEBCwUAMH4xCzAJBgNVBAYTAlVTMRMwEQYD
# VQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNy
# b3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNpZ25p
# bmcgUENBIDIwMTEwHhcNMjUwNjE5MTgyMTM3WhcNMjYwNjE3MTgyMTM3WjB0MQsw
# CQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9u
# ZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMR4wHAYDVQQDExVNaWNy
# b3NvZnQgQ29ycG9yYXRpb24wggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIB
# AQDASkh1cpvuUqfbqxele7LCSHEamVNBfFE4uY1FkGsAdUF/vnjpE1dnAD9vMOqy
# 5ZO49ILhP4jiP/P2Pn9ao+5TDtKmcQ+pZdzbG7t43yRXJC3nXvTGQroodPi9USQi
# 9rI+0gwuXRKBII7L+k3kMkKLmFrsWUjzgXVCLYa6ZH7BCALAcJWZTwWPoiT4HpqQ
# hJcYLB7pfetAVCeBEVZD8itKQ6QA5/LQR+9X6dlSj4Vxta4JnpxvgSrkjXCz+tlJ
# 67ABZ551lw23RWU1uyfgCfEFhBfiyPR2WSjskPl9ap6qrf8fNQ1sGYun2p4JdXxe
# UAKf1hVa/3TQXjvPTiRXCnJPAgMBAAGjggFzMIIBbzAfBgNVHSUEGDAWBgorBgEE
# AYI3TAgBBggrBgEFBQcDAzAdBgNVHQ4EFgQUuCZyGiCuLYE0aU7j5TFqY05kko0w
# RQYDVR0RBD4wPKQ6MDgxHjAcBgNVBAsTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEW
# MBQGA1UEBRMNMjMwMDEyKzUwNTM1OTAfBgNVHSMEGDAWgBRIbmTlUAXTgqoXNzci
# tW2oynUClTBUBgNVHR8ETTBLMEmgR6BFhkNodHRwOi8vd3d3Lm1pY3Jvc29mdC5j
# b20vcGtpb3BzL2NybC9NaWNDb2RTaWdQQ0EyMDExXzIwMTEtMDctMDguY3JsMGEG
# CCsGAQUFBwEBBFUwUzBRBggrBgEFBQcwAoZFaHR0cDovL3d3dy5taWNyb3NvZnQu
# Y29tL3BraW9wcy9jZXJ0cy9NaWNDb2RTaWdQQ0EyMDExXzIwMTEtMDctMDguY3J0
# MAwGA1UdEwEB/wQCMAAwDQYJKoZIhvcNAQELBQADggIBACjmqAp2Ci4sTHZci+qk
# tEAKsFk5HNVGKyWR2rFGXsd7cggZ04H5U4SV0fAL6fOE9dLvt4I7HBHLhpGdE5Uj
# Ly4NxLTG2bDAkeAVmxmd2uKWVGKym1aarDxXfv3GCN4mRX+Pn4c+py3S/6Kkt5eS
# DAIIsrzKw3Kh2SW1hCwXX/k1v4b+NH1Fjl+i/xPJspXCFuZB4aC5FLT5fgbRKqns
# WeAdn8DsrYQhT3QXLt6Nv3/dMzv7G/Cdpbdcoul8FYl+t3dmXM+SIClC3l2ae0wO
# lNrQ42yQEycuPU5OoqLT85jsZ7+4CaScfFINlO7l7Y7r/xauqHbSPQ1r3oIC+e71
# 5s2G3ClZa3y99aYx2lnXYe1srcrIx8NAXTViiypXVn9ZGmEkfNcfDiqGQwkml5z9
# nm3pWiBZ69adaBBbAFEjyJG4y0a76bel/4sDCVvaZzLM3TFbxVO9BQrjZRtbJZbk
# C3XArpLqZSfx53SuYdddxPX8pvcqFuEu8wcUeD05t9xNbJ4TtdAECJlEi0vvBxlm
# M5tzFXy2qZeqPMXHSQYqPgZ9jvScZ6NwznFD0+33kbzyhOSz/WuGbAu4cHZG8gKn
# lQVT4uA2Diex9DMs2WHiokNknYlLoUeWXW1QrJLpqO82TLyKTbBM/oZHAdIc0kzo
# STro9b3+vjn2809D0+SOOCVZMIIHejCCBWKgAwIBAgIKYQ6Q0gAAAAAAAzANBgkq
# 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
# /Xmfwb1tbWrJUnMTDXpQzTGCGiYwghoiAgEBMIGVMH4xCzAJBgNVBAYTAlVTMRMw
# EQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVN
# aWNyb3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNp
# Z25pbmcgUENBIDIwMTECEzMAAASFXpnsDlkvzdcAAAAABIUwDQYJYIZIAWUDBAIB
# BQCgga4wGQYJKoZIhvcNAQkDMQwGCisGAQQBgjcCAQQwHAYKKwYBBAGCNwIBCzEO
# MAwGCisGAQQBgjcCARUwLwYJKoZIhvcNAQkEMSIEIEwpF84JKwh6G/a6JqQdWxzu
# P6lQFYbd+fFOrw4iNImUMEIGCisGAQQBgjcCAQwxNDAyoBSAEgBNAGkAYwByAG8A
# cwBvAGYAdKEagBhodHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20wDQYJKoZIhvcNAQEB
# BQAEggEAvDMEYPNuLv6DpU8aMD0FbkUHUnQWXmO8fZibd7F9+QVaEaC/XpwQxxUW
# BaFEpvGlLWLzXm9/4Q7U4iRYmHFS9dIvGatc2dirNmqF2tYJ+E+S1Sv1gd17GjYE
# Yk9rNVFlmcoVJ8+iX+p+zNCXqAceb8HRuwc/PtUK1UT6hduH7XrH4Pni46uxEFmX
# +uoUl6TahXwERmc/g3y+n09Nru8J/HBCXsp1aYQiDRTle2wOdJore5ujuainzP/d
# AmU/xnJ5ZP/ViMB+EAizlZNDuL45pJRFpBzxILB5vCl2P4TgAKMtoZbJUQJdie6o
# CqojKAINUVUfEUvBR2Udo0NQaxcfZKGCF7AwghesBgorBgEEAYI3AwMBMYIXnDCC
# F5gGCSqGSIb3DQEHAqCCF4kwgheFAgEDMQ8wDQYJYIZIAWUDBAIBBQAwggFaBgsq
# hkiG9w0BCRABBKCCAUkEggFFMIIBQQIBAQYKKwYBBAGEWQoDATAxMA0GCWCGSAFl
# AwQCAQUABCBut/QafZ/tkhWH7bAvtBabr0av/Rfp+AhhplnQWfpi/wIGaKSAU7Tj
# GBMyMDI1MDkwOTE5MjIwNC4xMTRaMASAAgH0oIHZpIHWMIHTMQswCQYDVQQGEwJV
# UzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UE
# ChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMS0wKwYDVQQLEyRNaWNyb3NvZnQgSXJl
# bGFuZCBPcGVyYXRpb25zIExpbWl0ZWQxJzAlBgNVBAsTHm5TaGllbGQgVFNTIEVT
# Tjo1MjFBLTA1RTAtRDk0NzElMCMGA1UEAxMcTWljcm9zb2Z0IFRpbWUtU3RhbXAg
# U2VydmljZaCCEf4wggcoMIIFEKADAgECAhMzAAACAAvXqn8bKhdWAAEAAAIAMA0G
# CSqGSIb3DQEBCwUAMHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9u
# MRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRp
# b24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1wIFBDQSAyMDEwMB4XDTI0
# MDcyNTE4MzEyMVoXDTI1MTAyMjE4MzEyMVowgdMxCzAJBgNVBAYTAlVTMRMwEQYD
# VQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNy
# b3NvZnQgQ29ycG9yYXRpb24xLTArBgNVBAsTJE1pY3Jvc29mdCBJcmVsYW5kIE9w
# ZXJhdGlvbnMgTGltaXRlZDEnMCUGA1UECxMeblNoaWVsZCBUU1MgRVNOOjUyMUEt
# MDVFMC1EOTQ3MSUwIwYDVQQDExxNaWNyb3NvZnQgVGltZS1TdGFtcCBTZXJ2aWNl
# MIICIjANBgkqhkiG9w0BAQEFAAOCAg8AMIICCgKCAgEAr1XaadKkP2TkunoTF573
# /tF7KJM9Doiv3ccv26mqnUhmv2DM59ikET4WnRfo5biFIHc6LqrIeqCgT9fT/Gks
# 5VKO90ZQW2avh/PMHnl0kZfX/I5zdVooXHbdUUkPiZfNXszWswmL9UlWo8mzyv9L
# p9TAtw/oXOYTAxdYSqOB5Uzz1Q3A8uCpNlumQNDJGDY6cSn0MlYukXklArChq6l+
# KYrl6r/WnOqXSknABpggSsJ33oL3onmDiN9YUApZwjnNh9M6kDaneSz78/YtD/2p
# Gpx9/LXELoazEUFxhyg4KdmoWGNYwdR7/id81geOER69l5dJv71S/mH+Lxb6L692
# n8uEmAVw6fVvE+c8wjgYZblZCNPAynCnDduRLdk1jswCqjqNc3X/WIzA7GGs4HUS
# 4YIrAUx8H2A94vDNiA8AWa7Z/HSwTCyIgeVbldXYM2BtxMKq3kneRoT27NQ7Y7n8
# ZTaAje7Blfju83spGP/QWYNZ1wYzYVGRyOpdA8Wmxq5V8f5r4HaG9zPcykOyJpRZ
# y+V3RGighFmsCJXAcMziO76HinwCIjImnCFKGJ/IbLjH6J7fJXqRPbg+H6rYLZ8X
# BpmXBFH4PTakZVYxB/P+EQbL5LNw0ZIM+eufxCljV4O+nHkM+zgSx8+07BVZPBKs
# looebsmhIcBO0779kehciYMCAwEAAaOCAUkwggFFMB0GA1UdDgQWBBSAJSTavgkj
# Kqge5xQOXn35fXd3OjAfBgNVHSMEGDAWgBSfpxVdAF5iXYP05dJlpxtTNRnpcjBf
# BgNVHR8EWDBWMFSgUqBQhk5odHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3Bz
# L2NybC9NaWNyb3NvZnQlMjBUaW1lLVN0YW1wJTIwUENBJTIwMjAxMCgxKS5jcmww
# bAYIKwYBBQUHAQEEYDBeMFwGCCsGAQUFBzAChlBodHRwOi8vd3d3Lm1pY3Jvc29m
# dC5jb20vcGtpb3BzL2NlcnRzL01pY3Jvc29mdCUyMFRpbWUtU3RhbXAlMjBQQ0El
# MjAyMDEwKDEpLmNydDAMBgNVHRMBAf8EAjAAMBYGA1UdJQEB/wQMMAoGCCsGAQUF
# BwMIMA4GA1UdDwEB/wQEAwIHgDANBgkqhkiG9w0BAQsFAAOCAgEAKPCG9njRtIqQ
# +fuECgxzWMsQOI3HvW7sV9PmEWCCOWlTuGCIzNi3ibdLZS0b2IDHg0yLrtdVuBi3
# FxVdesIXuzYyofIe/alTBdV4DhijLTXtB7NgOno7G12iO3t6jy1hPSquzGLry/2m
# EZBwIsSoS2D+H+3HCJxPDyhzMFqP+plltPACB/QNwZ7q+HGyZv3v8et+rQYg8sF3
# PTuWeDg3dR/zk1NawJ/dfFCDYlWNeCBCLvNPQBceMYXFRFKhcSUws7mFdIDDhZpx
# qyIKD2WDwFyNIGEezn+nd4kXRupeNEx+eSpJXylRD+1d45hb6PzOIF7BkcPtRtFW
# 2wXgkjLqtTWWlBkvzl2uNfYJ3CPZVaDyMDaaXgO+H6DirsJ4IG9ikId941+mWDej
# kj5aYn9QN6ROfo/HNHg1timwpFoUivqAFu6irWZFw5V+yLr8FLc7nbMa2lFSixzu
# 96zdnDsPImz0c6StbYyhKSlM3uDRi9UWydSKqnEbtJ6Mk+YuxvzprkuWQJYWfpPv
# ug+wTnioykVwc0yRVcsd4xMznnnRtZDGMSUEl9tMVnebYRshwZIyJTsBgLZmHM7q
# 2TFK/X9944SkIqyY22AcuLe0GqoNfASCIcZtzbZ/zP4lT2/N0pDbn2ffAzjZkhI+
# Qrqr983mQZWwZdr3Tk1MYElDThz2D0MwggdxMIIFWaADAgECAhMzAAAAFcXna54C
# m0mZAAAAAAAVMA0GCSqGSIb3DQEBCwUAMIGIMQswCQYDVQQGEwJVUzETMBEGA1UE
# CBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9z
# b2Z0IENvcnBvcmF0aW9uMTIwMAYDVQQDEylNaWNyb3NvZnQgUm9vdCBDZXJ0aWZp
# Y2F0ZSBBdXRob3JpdHkgMjAxMDAeFw0yMTA5MzAxODIyMjVaFw0zMDA5MzAxODMy
# MjVaMHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQH
# EwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNV
# BAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1wIFBDQSAyMDEwMIICIjANBgkqhkiG9w0B
# AQEFAAOCAg8AMIICCgKCAgEA5OGmTOe0ciELeaLL1yR5vQ7VgtP97pwHB9KpbE51
# yMo1V/YBf2xK4OK9uT4XYDP/XE/HZveVU3Fa4n5KWv64NmeFRiMMtY0Tz3cywBAY
# 6GB9alKDRLemjkZrBxTzxXb1hlDcwUTIcVxRMTegCjhuje3XD9gmU3w5YQJ6xKr9
# cmmvHaus9ja+NSZk2pg7uhp7M62AW36MEBydUv626GIl3GoPz130/o5Tz9bshVZN
# 7928jaTjkY+yOSxRnOlwaQ3KNi1wjjHINSi947SHJMPgyY9+tVSP3PoFVZhtaDua
# Rr3tpK56KTesy+uDRedGbsoy1cCGMFxPLOJiss254o2I5JasAUq7vnGpF1tnYN74
# kpEeHT39IM9zfUGaRnXNxF803RKJ1v2lIH1+/NmeRd+2ci/bfV+AutuqfjbsNkz2
# K26oElHovwUDo9Fzpk03dJQcNIIP8BDyt0cY7afomXw/TNuvXsLz1dhzPUNOwTM5
# TI4CvEJoLhDqhFFG4tG9ahhaYQFzymeiXtcodgLiMxhy16cg8ML6EgrXY28MyTZk
# i1ugpoMhXV8wdJGUlNi5UPkLiWHzNgY1GIRH29wb0f2y1BzFa/ZcUlFdEtsluq9Q
# BXpsxREdcu+N+VLEhReTwDwV2xo3xwgVGD94q0W29R6HXtqPnhZyacaue7e3Pmri
# Lq0CAwEAAaOCAd0wggHZMBIGCSsGAQQBgjcVAQQFAgMBAAEwIwYJKwYBBAGCNxUC
# BBYEFCqnUv5kxJq+gpE8RjUpzxD/LwTuMB0GA1UdDgQWBBSfpxVdAF5iXYP05dJl
# pxtTNRnpcjBcBgNVHSAEVTBTMFEGDCsGAQQBgjdMg30BATBBMD8GCCsGAQUFBwIB
# FjNodHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL0RvY3MvUmVwb3NpdG9y
# eS5odG0wEwYDVR0lBAwwCgYIKwYBBQUHAwgwGQYJKwYBBAGCNxQCBAweCgBTAHUA
# YgBDAEEwCwYDVR0PBAQDAgGGMA8GA1UdEwEB/wQFMAMBAf8wHwYDVR0jBBgwFoAU
# 1fZWy4/oolxiaNE9lJBb186aGMQwVgYDVR0fBE8wTTBLoEmgR4ZFaHR0cDovL2Ny
# bC5taWNyb3NvZnQuY29tL3BraS9jcmwvcHJvZHVjdHMvTWljUm9vQ2VyQXV0XzIw
# MTAtMDYtMjMuY3JsMFoGCCsGAQUFBwEBBE4wTDBKBggrBgEFBQcwAoY+aHR0cDov
# L3d3dy5taWNyb3NvZnQuY29tL3BraS9jZXJ0cy9NaWNSb29DZXJBdXRfMjAxMC0w
# Ni0yMy5jcnQwDQYJKoZIhvcNAQELBQADggIBAJ1VffwqreEsH2cBMSRb4Z5yS/yp
# b+pcFLY+TkdkeLEGk5c9MTO1OdfCcTY/2mRsfNB1OW27DzHkwo/7bNGhlBgi7ulm
# ZzpTTd2YurYeeNg2LpypglYAA7AFvonoaeC6Ce5732pvvinLbtg/SHUB2RjebYIM
# 9W0jVOR4U3UkV7ndn/OOPcbzaN9l9qRWqveVtihVJ9AkvUCgvxm2EhIRXT0n4ECW
# OKz3+SmJw7wXsFSFQrP8DJ6LGYnn8AtqgcKBGUIZUnWKNsIdw2FzLixre24/LAl4
# FOmRsqlb30mjdAy87JGA0j3mSj5mO0+7hvoyGtmW9I/2kQH2zsZ0/fZMcm8Qq3Uw
# xTSwethQ/gpY3UA8x1RtnWN0SCyxTkctwRQEcb9k+SS+c23Kjgm9swFXSVRk2XPX
# fx5bRAGOWhmRaw2fpCjcZxkoJLo4S5pu+yFUa2pFEUep8beuyOiJXk+d0tBMdrVX
# VAmxaQFEfnyhYWxz/gq77EFmPWn9y8FBSX5+k77L+DvktxW/tM4+pTFRhLy/AsGC
# onsXHRWJjXD+57XQKBqJC4822rpM+Zv/Cuk0+CQ1ZyvgDbjmjJnW4SLq8CdCPSWU
# 5nR0W2rRnj7tfqAxM328y+l7vzhwRNGQ8cirOoo6CGJ/2XBjU02N7oJtpQUQwXEG
# ahC0HVUzWLOhcGbyoYIDWTCCAkECAQEwggEBoYHZpIHWMIHTMQswCQYDVQQGEwJV
# UzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UE
# ChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMS0wKwYDVQQLEyRNaWNyb3NvZnQgSXJl
# bGFuZCBPcGVyYXRpb25zIExpbWl0ZWQxJzAlBgNVBAsTHm5TaGllbGQgVFNTIEVT
# Tjo1MjFBLTA1RTAtRDk0NzElMCMGA1UEAxMcTWljcm9zb2Z0IFRpbWUtU3RhbXAg
# U2VydmljZaIjCgEBMAcGBSsOAwIaAxUAjJOfLZb3ivipL3sSLlWFbLrWjmSggYMw
# gYCkfjB8MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UE
# BxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSYwJAYD
# VQQDEx1NaWNyb3NvZnQgVGltZS1TdGFtcCBQQ0EgMjAxMDANBgkqhkiG9w0BAQsF
# AAIFAOxqrF4wIhgPMjAyNTA5MDkxMzM4MzhaGA8yMDI1MDkxMDEzMzgzOFowdzA9
# BgorBgEEAYRZCgQBMS8wLTAKAgUA7GqsXgIBADAKAgEAAgIO0QIB/zAHAgEAAgIS
# 9DAKAgUA7Gv93gIBADA2BgorBgEEAYRZCgQCMSgwJjAMBgorBgEEAYRZCgMCoAow
# CAIBAAIDB6EgoQowCAIBAAIDAYagMA0GCSqGSIb3DQEBCwUAA4IBAQBMzwcyQFWz
# E3vJvRhPTZnefuwTSI9EVF33stCi0zyh84d2oS+J/7HCXE+1EJYBEO+P/AdYIM1N
# 7I3ijTUJKsFYMbYmzWqbW1y0HuH6YZPk4EfgXtKoWirMJtNBoKnNkhxYEuOa/Oig
# dNFz6z4nzeMRqCee0YljDn2g/zjnObehn0bSblkGZlMC01nuduoz13099FR8suNZ
# TFfHuFFLt4brFzkrN8gOHyg9s400OVLaF19NyJ/NivEDNXb3y14pghHTyzEciuFz
# NoR/Mn6A4eqioIX+7gzsNUTpaqGCmE4Z71/z2XSkEyYGb8CQQQbntBIAN9H7QFPZ
# k0vRBrpEDcWYMYIEDTCCBAkCAQEwgZMwfDELMAkGA1UEBhMCVVMxEzARBgNVBAgT
# Cldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29m
# dCBDb3Jwb3JhdGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUtU3RhbXAgUENB
# IDIwMTACEzMAAAIAC9eqfxsqF1YAAQAAAgAwDQYJYIZIAWUDBAIBBQCgggFKMBoG
# CSqGSIb3DQEJAzENBgsqhkiG9w0BCRABBDAvBgkqhkiG9w0BCQQxIgQgK0vbUXmO
# LCMnaDelhYe6lhFRfzACaQZjbKo2BaRjGMUwgfoGCyqGSIb3DQEJEAIvMYHqMIHn
# MIHkMIG9BCDUyO3sNZ3burBNDGUCV4NfM2gH4aWuRudIk/9KAk/ZJzCBmDCBgKR+
# MHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdS
# ZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMT
# HU1pY3Jvc29mdCBUaW1lLVN0YW1wIFBDQSAyMDEwAhMzAAACAAvXqn8bKhdWAAEA
# AAIAMCIEIPFqPodgdzJ1PgQAJk5/GOEuUv4Ge9if3bpEBY0ESbjjMA0GCSqGSIb3
# DQEBCwUABIICAGsECtILpr2twpOxzGDNFIpNWjidrK3bPtQnWuSJuBbKSe/wbolt
# fOh2uhEVhbnRGayzXE8cW4PDoF4uiK/qb6tNLwP7cycsoJ8J4/spaeAr4QwgrAAP
# 4hVjMmWb5enzY61tesHI8po9tIz1FXoYax9psGAV3HXreRqc4R+b1+0+XriXUkL0
# tXFx0dtVKM202XeprePGlMfg9/PqlGY12SeH6eoY44T2STH6h+JyJfXlDQKB4CVl
# j1pZl5I+mRTAU2cSLLjqht/nfQDne6Z9BwJxz7VQ6JnL1YbfbBSvQSWQr5GYQTzz
# GzsdqmKiAfHioAxjzf25olH740WkS6ok+y6r71QYRHavXtviDnsTYuS+qngVPQ5Q
# 5nsihtWA9B2cy2oQIuiYpG206gUaSfNaBGv+qePu82G43rbcZq78W97XXmRq2hlc
# aqYR10u2d+hJL58PplK8lnyyvwwYoy9fD00aahlOsEDauSz3XCVI+GsxIxAhCIVd
# 5cXYGf41Me1Npm6prXlLAnH1tSYDzQIZST8UV2GJhkSHFVcdbmRVAIzg8Y+rd37C
# bN227Gx+ULNug6th9DS3ALeA/VfzhO5Auv1GWIgIe9DBik8XK9s7eYWZFcTCETIq
# zsPpUZLOupbz4zyXA25v9pPRXH+S3CiT4eA5GMNq7zPSk7v0Ff101aa7
# SIG # End signature block