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
    )

    $TCPTestResults = @()

    if (-not $PSSession -or $PSSession.Count -eq 0) {
        $errorMsg = "No PSSession(s) provided to test."
        Log-Info $errorMsg -Type 'WARNING'
        $TCPTestRstObject = @{
            Name               = "HCI_Node_TCP_Connection_Test"
            Title              = 'Host TCP Connection Pre-check Failed'
            DisplayName        = 'Host TCP Connection Pre-check Failed'
            Severity           = 'Warning'
            Description        = $errorMsg
            Tags               = @{ }
            Remediation        = "Please provide valid PSSessions to the function."
            TargetResourceID   = 'HostTCPConnectionFailed'
            TargetResourceName = 'HostTCPConnectionFailed'
            TargetResourceType = 'HostTCPConnectionFailed'
            Timestamp          = [datetime]::UtcNow
            Status             = 'FAILURE'
            AdditionalData     = @{
                Source    = 'HostTCPConnectionValidation'
                Resource  = 'PSSession'
                Detail    = $errorMsg
                Status    = 'FAILURE'
                TimeStamp = [datetime]::UtcNow
            }
            HealthCheckSource  = $ENV:EnvChkrId
        }
        $TCPTestResults += New-AzStackHciResultObject @TCPTestRstObject
        return $TCPTestResults
    }

    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 'WARNING'
                $TCPTestRstObject = @{
                    Name               = "HCI_Node_TCP_Connection_Test"
                    Title              = "Failed to reconnect PSSession to $($Session.ComputerName)"
                    DisplayName        = "Failed to reconnect PSSession to $($Session.ComputerName)"
                    Severity           = 'Warning'
                    Description        = $PsSessionFail
                    Tags               = @{ }
                    Remediation        = "Ensure the host $($Session.ComputerName) is reachable and WinRM is configured correctly."
                    TargetResourceID   = $Session.ComputerName
                    TargetResourceName = $Session.ComputerName
                    TargetResourceType = 'Node'
                    Timestamp          = [datetime]::UtcNow
                    Status             = 'FAILURE'
                    AdditionalData     = @{
                        Source    = 'HostTCPConnectionValidation'
                        Resource  = $Session.ComputerName
                        Detail    = $PsSessionFail
                        Status    = 'FAILURE'
                        TimeStamp = [datetime]::UtcNow
                    }
                    HealthCheckSource  = $ENV:EnvChkrId
                }
                $TCPTestResults += New-AzStackHciResultObject @TCPTestRstObject
                return $TCPTestResults
            }
        }
    }

    # 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

    $remoteToolPaths = @{}

    foreach ($session in $PSSession) {

    # Copy psping.exe to all hosts
        try {
            # Explicitly specify the destination to ensure consistency
            Copy-RemoteItem -SourcePath $localToolPath -PsSession $PSSession
            $tempPath = Invoke-Command -Session $session -ScriptBlock { $env:TEMP }

            $remoteToolPath = Join-Path -Path $tempPath (Split-Path -Leaf $localToolPath)
            $remoteToolPaths[$session.ComputerName] = $remoteToolPath
    
            # Verify the file exists on each remote machine
            $exists = Invoke-Command -Session $session -ScriptBlock {
                param($p) Test-Path -Path $p
            } -ArgumentList $remoteToolPath

            if (-not $exists) {
                $errorMsg = "psping.exe not found at $remoteToolPath on $($session.ComputerName) after copy attempt."
                Log-Info $errorMsg -Type 'WARNING'
                $TCPTestRstObject = @{
                    Name               = "HCI_Node_Tool_Deployment_Test"
                    Title              = "Failed to deploy psping.exe to $($session.ComputerName)"
                    DisplayName        = "Failed to deploy psping.exe to $($session.ComputerName)"
                    Severity           = 'Warning'
                    Description        = $errorMsg
                    Tags               = @{ }
                    Remediation        = "Ensure WinRM is configured correctly and there are no permission issues for copying files to the temp directory on $($session.ComputerName)."
                    TargetResourceID   = $session.ComputerName
                    TargetResourceName = $session.ComputerName
                    TargetResourceType = 'Node'
                    Timestamp          = [datetime]::UtcNow
                    Status             = 'FAILURE'
                    AdditionalData     = @{
                        Source    = 'ToolDeployment'
                        Resource  = $session.ComputerName
                        Detail    = $errorMsg
                        Status    = 'FAILURE'
                        TimeStamp = [datetime]::UtcNow
                    }
                    HealthCheckSource  = $ENV:EnvChkrId
                }
                $TCPTestResults += New-AzStackHciResultObject @TCPTestRstObject
                return $TCPTestResults
            }
            Log-Info "Verified psping.exe on $($session.ComputerName) at $remoteToolPath"
        } 
        catch {
            $errorMsg = "Failed to copy psping.exe to $($session.ComputerName): $($_.Exception.Message)"
            Log-Info $errorMsg -Type 'WARNING'
            $TCPTestRstObject = @{
                Name               = "HCI_Node_Tool_Deployment_Test"
                Title              = "Failed to deploy psping.exe to $($session.ComputerName)"
                DisplayName        = "Failed to deploy psping.exe to $($session.ComputerName)"
                Severity           = 'Warning'
                Description        = $errorMsg
                Tags               = @{ }
                Remediation        = "Ensure WinRM is configured correctly, check network connectivity, and verify permissions for copying files to the temp directory on $($session.ComputerName)."
                TargetResourceID   = $session.ComputerName
                TargetResourceName = $session.ComputerName
                TargetResourceType = 'Node'
                Timestamp          = [datetime]::UtcNow
                Status             = 'FAILURE'
                AdditionalData     = @{
                    Source    = 'ToolDeployment'
                    Resource  = $session.ComputerName
                    Detail    = $errorMsg
                    Status    = 'FAILURE'
                    TimeStamp = [datetime]::UtcNow
                }
                HealthCheckSource  = $ENV:EnvChkrId
            }
            $TCPTestResults += New-AzStackHciResultObject @TCPTestRstObject
            return $TCPTestResults
        }
    }

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

    # Accept EULA for psping on all remote machines
    foreach ($session in $PSSession) {
        Invoke-Command -Session $session -ScriptBlock {
            $localToolPath = Join-Path -Path $env:TEMP "psping.exe"
            if (Test-Path -Path $localToolPath) {
                & "$localToolPath" -accepteula
            }
        } > $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) {
            try {
                Write-Progress ($lnTxt.StartHostTCPServer -f $serverHost, $serverIP)
                Log-Info ($lnTxt.StartHostTCPServer -f $serverHost, $serverIP)
                Log-Info ($lnTxt.AddHostTCPFW -f $Port, $serverHost)
                Log-Info "[DEBUG] Starting psping server on ${serverHost}:${serverIP}:${Port}"
                Invoke-Command -Session $serverSession -ScriptBlock {
                    param ($serverIP, $Port)
                    $ruleName = "TmpAllowTCPPortForTesting"
    
                    # Construct the tool path locally on this remote machine
                    $localToolPath = Join-Path -Path $env:TEMP "psping.exe"
    
                    # Verify the tool exists
                    if (-not (Test-Path -Path $localToolPath)) {
                        throw "psping.exe not found at $localToolPath on $env:COMPUTERNAME"
                    }

                    # 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 $localToolPath)
                    Start-Process -NoNewWindow -FilePath $localToolPath -ArgumentList "-s ${serverIP}:$Port -nobanner" *>$null
                    Pop-Location
                } -ArgumentList $serverIP, $Port -ErrorAction Stop
            }
            catch {
                $errorMsg = "Failed to start psping server on $serverHost ($serverIP): $($_.Exception.Message)"
                Log-Info $errorMsg -Type 'Warning'

                # Attempt to clean up resources on the server host in case of partial failure
                Invoke-Command -Session $serverSession -ScriptBlock {
                    Get-Process | Where-Object { $_.ProcessName -like "psping*" } | Stop-Process -Force -ErrorAction SilentlyContinue
                    Remove-NetFirewallRule -DisplayName "TmpAllowTCPPortForTesting" -ErrorAction SilentlyContinue
                } -ErrorAction SilentlyContinue

                # Skip to the next server IP as tests cannot be performed
                continue
            }

            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
                        Log-Info "[DEBUG] Running psping client latency test from $clientHost to ${serverIP}:${Port}"
                        $latencyOutput = Invoke-Command -Session $clientSession -ScriptBlock {
                            param ($serverAddress, $Port)
    
                            # Construct the tool path locally on this remote machine
                            $localToolPath = Join-Path -Path $env:TEMP "psping.exe"
    
                            # Verify the tool exists
                            if (-not (Test-Path -Path $localToolPath)) {
                                throw "psping.exe not found at $localToolPath on $env:COMPUTERNAME"
                            }
    
                            Push-Location (Split-Path -Path $localToolPath)

                            # Run the remote tool and capture the output
                            $fullOutput = & $localToolPath -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 -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
                        Log-Info "[DEBUG] Running psping bandwidth test from $clientHost to ${serverIP}:${Port}"
                        $bandwidthOutput = Invoke-Command -Session $clientSession -ScriptBlock {
                            param ($serverAddress, $Port)
    
                            # Construct the tool path locally on this remote machine
                            $localToolPath = Join-Path -Path $env:TEMP "psping.exe"
    
                            # Verify the tool exists
                            if (-not (Test-Path -Path $localToolPath)) {
                                throw "psping.exe not found at $localToolPath on $env:COMPUTERNAME"
                            }
    
                            Push-Location (Split-Path -Path $localToolPath)

                            # Execute the bandwidth test using the remote tool
                            $fullOutput = & $localToolPath -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 -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
# MIIncQYJKoZIhvcNAQcCoIInYjCCJ14CAQExDzANBglghkgBZQMEAgEFADB5Bgor
# BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG
# KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCBUFQCB96nktAgK
# 1QV/TbwNxiq/KXs/VW8uXU8udyufQaCCDMkwggYEMIID7KADAgECAhMzAAACHPrN
# xZvoL37EAAAAAAIcMA0GCSqGSIb3DQEBCwUAMFcxCzAJBgNVBAYTAlVTMR4wHAYD
# VQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBD
# b2RlIFNpZ25pbmcgUENBIDIwMjQwHhcNMjYwNDE2MTg1OTQxWhcNMjcwNDE1MTg1
# OTQxWjB0MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UE
# BxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMR4wHAYD
# VQQDExVNaWNyb3NvZnQgQ29ycG9yYXRpb24wggEiMA0GCSqGSIb3DQEBAQUAA4IB
# DwAwggEKAoIBAQDVsZfgOKmM31HPfoWOoNEiw0SlCiIxUMC0I9NMWbucKOw/e9lP
# oAoehQVu6SG65V4EPzrYsnBnFPNoi4/HoOdjhz1qkrEt4I6tEcxXU6oOeY9zGveC
# /3iBeuhLYxM3M/PkcUoebF+Nednm8OkdSPoDu8imViHPQq/8CQUu0WRR4rE+dMRf
# rpVqfmNi2qWCX94T4MsepijGVkwE//tJg0ryAiYdHT34LSnlG/RSBZmQRGWZ5g8j
# qnKjRParSqMft1gvjuUTVgtWNZfgcLFSK5Wa0myrq8OPcgTGGsRgun+tnSS+IxDT
# xVsAPH1OzvPjwomguByhUe/OcvUN0D5Wmp7xAgMBAAGjggGqMIIBpjAOBgNVHQ8B
# Af8EBAMCB4AwHwYDVR0lBBgwFgYKKwYBBAGCN0wIAQYIKwYBBQUHAwMwHQYDVR0O
# BBYEFNoH7a2YDjOSwpkp6DHcmUS7J+0yMFQGA1UdEQRNMEukSTBHMS0wKwYDVQQL
# EyRNaWNyb3NvZnQgSXJlbGFuZCBPcGVyYXRpb25zIExpbWl0ZWQxFjAUBgNVBAUT
# DTIzMDAxMis1MDc1NjkwHwYDVR0jBBgwFoAUf1k/VCHarU/vBeXmo9ctBpQSCDEw
# YAYDVR0fBFkwVzBVoFOgUYZPaHR0cDovL3d3dy5taWNyb3NvZnQuY29tL3BraW9w
# cy9jcmwvTWljcm9zb2Z0JTIwQ29kZSUyMFNpZ25pbmclMjBQQ0ElMjAyMDI0LmNy
# bDBtBggrBgEFBQcBAQRhMF8wXQYIKwYBBQUHMAKGUWh0dHA6Ly93d3cubWljcm9z
# b2Z0LmNvbS9wa2lvcHMvY2VydHMvTWljcm9zb2Z0JTIwQ29kZSUyMFNpZ25pbmcl
# MjBQQ0ElMjAyMDI0LmNydDAMBgNVHRMBAf8EAjAAMA0GCSqGSIb3DQEBCwUAA4IC
# AQAUnEqhaRXe0T3hIJjvdQErEkrA/7bByjn6t5IArODkkRjzkYwtKMc2yYj2quaN
# rLutWw2YZcngKPy1b71YyDJQTy4NDRwaSh9Tw5thrk3NmcPrAHia5vtcBJ1CgtKK
# 7mQbIcQ22d/N3813ayCDDFewu1+jsZmX+r/aTEqaOM4TVxVtRSkuCy8nAXKuChOK
# Li/zA4XuH8iEYqIsj2YoNaeSxVmeGiERXpKdo3dDmYi0kO5w2D8VS4c3+9h6gElY
# BaAAg/dYErBg27qT3vv0zRDJhJufvCNylA8S7/+8H5E/PV5cng6na9VV/w9OV3qu
# uND6zdGa2EX38Glp50F9AIQk3p2xXmcvorDeM4XJ7UlWYBi6g80J1SSOQnInCYFE
# msfUNn3+1AaTJKSJL83quKArTac2pKhu0Yzzzrzo6HrsRiQKzpnRBb1/dMa6P3hz
# 75XbMRBctNsFhZC07WCmjExdLg2eHW5uV0TY8D5+6wozJf7vF3+WHkYPO85Z+BC6
# U4FkNbYNycZ9cE4j1tXRdyDCfml6c0HWPHjNVDObrv9lKt3qUqFpX38VCqVCyNOO
# 1UcXfQiVjJw32U2WUKZjt/neJKHEBsm9kFsLuWzkQ53+qcaSaytmsCnk2gOglrlD
# 5d3kKyvvAw+rzm0lT8K38P6PLxfZQHhu4W8dV7Av8N2ZmDCCBr0wggSloAMCAQIC
# EzMAAAA5O7Y3Gb8GHWcAAAAAADkwDQYJKoZIhvcNAQEMBQAwgYgxCzAJBgNVBAYT
# AlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYD
# VQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xMjAwBgNVBAMTKU1pY3Jvc29mdCBS
# b290IENlcnRpZmljYXRlIEF1dGhvcml0eSAyMDExMB4XDTI0MDgwODIwNTQxOFoX
# DTM2MDMyMjIyMTMwNFowVzELMAkGA1UEBhMCVVMxHjAcBgNVBAoTFU1pY3Jvc29m
# dCBDb3Jwb3JhdGlvbjEoMCYGA1UEAxMfTWljcm9zb2Z0IENvZGUgU2lnbmluZyBQ
# Q0EgMjAyNDCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBANgBnB7jOMeq
# lRYHNa265v4IY9fH8TKhemHfPINe1gpLaV3dhg324WwH06LcHbpnsBukCDNitryo
# 0dtS/EW6I/yEL/bLSY8hKpbfQuWusBPr9qazYcDxCW/qnjb5JsI1s8bNOg3bVATv
# QVL4tcf03aTycsz8QeCdM0l/yHRObJ9QqazM1r6VPEOJ7LL+uEEb73w6QCuhs89a
# 1uv1zerOYMnsneRRwCbpyW11IcggU0cRKDDq1pjVJzIbIF6+oiXXbReOsgeI8zu1
# FyQfK0fVkaya8SmVHQ/tOf23mZ4W9k0Ri22QW9p3UgSC5OUDktKxxcCmGL6tXLfO
# GSWHIIV4YrTJTT6PNty5REojHJuZHArkF9VnHTERWoTjAzfI3kP+5b4alUdhgAZ7
# ttOu1bVnXfHaqPYl2rPs20ji03LOVWsh/radgE17es5hL+t6lV0eVHrVhsssROWJ
# uz2MXMCt7iw7lFPG9LXKGjsmonn2gotGdHIuEg5JnJMJVmixd5LRlkmgYRZKzhxS
# CwyoGIq0PhaA7Y+VPct5pCHkijcIIDm0nlkK+0KyepolcqGm0T/GYQRMhHJlGOOm
# VQop36wUVUYklUy++vDWeEgEo4s7hxN6mIbf2MSIQ/iIfMZgJxC69oukMUXCrOC3
# SkE/xIkgpfl22MM1itkZ35nNXkMolU1lAgMBAAGjggFOMIIBSjAOBgNVHQ8BAf8E
# BAMCAYYwEAYJKwYBBAGCNxUBBAMCAQAwHQYDVR0OBBYEFH9ZP1Qh2q1P7wXl5qPX
# LQaUEggxMBkGCSsGAQQBgjcUAgQMHgoAUwB1AGIAQwBBMA8GA1UdEwEB/wQFMAMB
# Af8wHwYDVR0jBBgwFoAUci06AjGQQ7kUBU7h6qfHMdEjiTQwWgYDVR0fBFMwUTBP
# oE2gS4ZJaHR0cDovL2NybC5taWNyb3NvZnQuY29tL3BraS9jcmwvcHJvZHVjdHMv
# TWljUm9vQ2VyQXV0MjAxMV8yMDExXzAzXzIyLmNybDBeBggrBgEFBQcBAQRSMFAw
# TgYIKwYBBQUHMAKGQmh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2kvY2VydHMv
# TWljUm9vQ2VyQXV0MjAxMV8yMDExXzAzXzIyLmNydDANBgkqhkiG9w0BAQwFAAOC
# AgEAFJQfOChP7onn6fLIMKrSlN1WYKwDFgAddymOUO3FrM8d7B/W/iQ6DxXsDn7D
# 5W4wMwYeLystcEqfkjz4NURRgazyMu5yRzQh4LqjA4tStTcJh1opExo7nn5PuPBY
# nbu0+THSuVHTe0VTTPVhily/piFrDo3axQ9P4C+Ol5yet+2gTfekICS5xS+cYfSI
# vgn0JksVBVMYVI5QFu/qhnLhsEFEUzG8fvv0hjgkO+lkpV9ty6GkN4vdnd7ya6Q6
# aR9y34aiM1qmxaxBi6OUnyNl6fkuun/diTFnYDLTppOkr/mg5WSfCiDVMNCxtj4w
# PKC5OmHm1DQIt/MNokbbH3UGsFP1QbzsLocuSqLCvH09Io3fDPTmscR9Y75G4qX7
# RTX8AdBPo0I6OEojf39zuFZt0qOHm65YWQE69cZM2ueE1MB05dNNgHK9gTE7zKvK
# /fg8B2qjW88MT/WF5V5uvZGtqa9FSL2RazArA+rDPuf6JGYz4HpgMZHB4S6szWSK
# YBv0VisCzfxgeU+dquXW9bd0auYlOB58DPcOYKdc3Se94g+xL4pcEhbB54JOgAkw
# YTu/9dLeH2pDqeJZAABVDWRQCaXfO5LgyKwKCLYXpigrZYCjUSBcr+Ve8PFWMhVT
# Ql0v4q8J/AUmQN5W4n101cY2L4A7GTQG1h32HHAvfQESWP0xghn+MIIZ+gIBATBu
# MFcxCzAJBgNVBAYTAlVTMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24x
# KDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNpZ25pbmcgUENBIDIwMjQCEzMAAAIc
# +s3Fm+gvfsQAAAAAAhwwDQYJYIZIAWUDBAIBBQCgga4wGQYJKoZIhvcNAQkDMQwG
# CisGAQQBgjcCAQQwHAYKKwYBBAGCNwIBCzEOMAwGCisGAQQBgjcCARUwLwYJKoZI
# hvcNAQkEMSIEIHBfP8S90JPgfgMOmlO0aWjv/GHUmaL5Cl8AC9iNOVWMMEIGCisG
# AQQBgjcCAQwxNDAyoBSAEgBNAGkAYwByAG8AcwBvAGYAdKEagBhodHRwOi8vd3d3
# Lm1pY3Jvc29mdC5jb20wDQYJKoZIhvcNAQEBBQAEggEAjaSy7vfVm9KgUZrZ7D30
# ID5ZGil4K6QX0qckhSe6mn/rzdPmXV1NPxFobZrQ2boasSrOJjOy/QhSfTxLN5L2
# lPqwEmSqBk/gJQPo42NYpDdZe0PqPCwc8KmBrVCWzo/O4rs86K97jco4KQDGp1nT
# jG1R9ea3HsEboLbRgg87i4bGK1AYHa5BV8+A3gC7QoC4HdX9AKJowKDOVQGsRF+c
# KQfG7LQK1D0BZY5JNmSFQ1yvOWLSb+oOIN8h9/WuH/lFSBAhzaaPh479a3+xglD4
# C53fUUvFFX+R6mKHG59xkpw+SC52wwBl6qvG8G9G1F5Wc8oF8u6NcBMGvqI4L8M2
# JKGCF7AwghesBgorBgEEAYI3AwMBMYIXnDCCF5gGCSqGSIb3DQEHAqCCF4kwgheF
# AgEDMQ8wDQYJYIZIAWUDBAIBBQAwggFaBgsqhkiG9w0BCRABBKCCAUkEggFFMIIB
# QQIBAQYKKwYBBAGEWQoDATAxMA0GCWCGSAFlAwQCAQUABCAA7UIDrQOVYXJvbaXZ
# rzYxK+XFQNeu7GNVz9ZgZqaurAIGaeuJ1G3MGBMyMDI2MDUwMzE0MzExMS41MTVa
# MASAAgH0oIHZpIHWMIHTMQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3Rv
# bjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0
# aW9uMS0wKwYDVQQLEyRNaWNyb3NvZnQgSXJlbGFuZCBPcGVyYXRpb25zIExpbWl0
# ZWQxJzAlBgNVBAsTHm5TaGllbGQgVFNTIEVTTjoyRDFBLTA1RTAtRDk0NzElMCMG
# A1UEAxMcTWljcm9zb2Z0IFRpbWUtU3RhbXAgU2VydmljZaCCEf4wggcoMIIFEKAD
# AgECAhMzAAACEtEIBjzKGE+qAAEAAAISMA0GCSqGSIb3DQEBCwUAMHwxCzAJBgNV
# BAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4w
# HAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29m
# dCBUaW1lLVN0YW1wIFBDQSAyMDEwMB4XDTI1MDgxNDE4NDgxNVoXDTI2MTExMzE4
# NDgxNVowgdMxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYD
# VQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xLTAr
# BgNVBAsTJE1pY3Jvc29mdCBJcmVsYW5kIE9wZXJhdGlvbnMgTGltaXRlZDEnMCUG
# A1UECxMeblNoaWVsZCBUU1MgRVNOOjJEMUEtMDVFMC1EOTQ3MSUwIwYDVQQDExxN
# aWNyb3NvZnQgVGltZS1TdGFtcCBTZXJ2aWNlMIICIjANBgkqhkiG9w0BAQEFAAOC
# Ag8AMIICCgKCAgEAr0zToDkpWQtsZekS0cV0quDdKSTGkovvBaZH0OAIEi0O3CcO
# 77JiX8c4Epq9uibHVZZ1W/LoufE172vkRXO+QYNtWWorECJ2AcZQ10bpAltkhZNi
# XlVJ8L3QzhKgrXrmMkm2J+/g81U23JPcO4wXHEftonT3wpd//936rjmwxMm7Nkbs
# ygbJf+4AVBMNr4aMPQhBd76od0KMB6WrvyEGOOU0893OFufS5EDey4n44WgaxJE0
# Vnv3/OOvuOw5Kp1KPqjjYJ+L9ywLuBMtcDfLpNQO/h1eFEoMrbiEM67TOfNlXfxb
# Dz4MlsYvLioxgd2Xzey1QxrV1+i+JyVDJMiSe9gKOuzpiQQFE19DUPgsidyjLTzX
# EhSVLBlRor0eCVf7gC6Rfk8NY3rO2sggOL79vU5FuDKTh/sIOtcUHeHC42jBGB+t
# fdKC1KOBR+UlN9aOzg8mpUNI2FgqQvirVP9ppbeMUfvp2wA9voyTiRWvDgzCxo8x
# lJ1nscYTHIQrmkF9j/Ca0IDmt8fvOn64nnlJOGUYZYHMC1l0xtgkYTE1ESUqqkaw
# Kk7iqbxdnLyycS+dR+zaxPudMDLrQFz8lgfy9obk0D8HC2dzhWpYNn5hdkoPEzgC
# qQUOp8v3Qj/sd4anyupe5KoCkjABOP3yhSQ4W9Z+DrJnhM/rbsXC7oTv26cCAwEA
# AaOCAUkwggFFMB0GA1UdDgQWBBRSBblSxb5cYKYOwvd/VfoXOfu33jAfBgNVHSME
# GDAWgBSfpxVdAF5iXYP05dJlpxtTNRnpcjBfBgNVHR8EWDBWMFSgUqBQhk5odHRw
# Oi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL2NybC9NaWNyb3NvZnQlMjBUaW1l
# LVN0YW1wJTIwUENBJTIwMjAxMCgxKS5jcmwwbAYIKwYBBQUHAQEEYDBeMFwGCCsG
# AQUFBzAChlBodHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL2NlcnRzL01p
# Y3Jvc29mdCUyMFRpbWUtU3RhbXAlMjBQQ0ElMjAyMDEwKDEpLmNydDAMBgNVHRMB
# Af8EAjAAMBYGA1UdJQEB/wQMMAoGCCsGAQUFBwMIMA4GA1UdDwEB/wQEAwIHgDAN
# BgkqhkiG9w0BAQsFAAOCAgEAXnSAkmX79Rc7lxS1wOozXJ7V0ou5DntVcOJplIkD
# jvEN8BIQph4U+gSOLZuVReP/z9YdUiUkcPwL1PM245/kEX1EegpxNc8HDA6hKCHg
# 0ALNEcuxnGOlgKLokXfUer1D5hiW8PABM9R+neiteTgPaaRlJFvGTYvotc0uqGiE
# S5hMQhL8RNFhpS9RcIWHtnQGEnrdOUvCAhs4FeViawcmLTKv+1870c/MeTHi0QDd
# eR+7/Wg4qhkJ2k1iEHJdmYf8rIV0NRBZcdRTTdHee35SXP5neNCfAkjDIuZycRud
# 6jzPLCNLiNYzGXBswzJygj4EeSORT7wMvaFuKeRAXoXC3wwYvgIsI1zn3DGY625Y
# +yZSi8UNSNHuri36Zv9a+Q4vJwDpYK36S0TB2pf7xLiiH32nk7YK73Rg98W6fZ2I
# NuzYzZ7Ghgvfffkj4EUXg1E0EffY1pEqkbpDTP7h/DBqtzoPXsyw2MUh+7yvWcq2
# BGZSuca6CY6X4ioMuc5PWpsmvOOli7ARNA7Ab8kKdCc2gNDLacglsweZEc9/VQB6
# hls/b6Kk32nkwuHExKlaeoSVrKB5U9xlp1+c8J/7GJj4Rw7AiQ8tcp+WmfyD8KxX
# 2QlKbDi4SUjnglv4617R8+a/cDWJyaMt8279Wn7f2yMedN7kfGIQ5SZj66RdhdlZ
# Oq8wggdxMIIFWaADAgECAhMzAAAAFcXna54Cm0mZAAAAAAAVMA0GCSqGSIb3DQEB
# CwUAMIGIMQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UE
# BxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMTIwMAYD
# VQQDEylNaWNyb3NvZnQgUm9vdCBDZXJ0aWZpY2F0ZSBBdXRob3JpdHkgMjAxMDAe
# Fw0yMTA5MzAxODIyMjVaFw0zMDA5MzAxODMyMjVaMHwxCzAJBgNVBAYTAlVTMRMw
# EQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVN
# aWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0
# YW1wIFBDQSAyMDEwMIICIjANBgkqhkiG9w0BAQEFAAOCAg8AMIICCgKCAgEA5OGm
# TOe0ciELeaLL1yR5vQ7VgtP97pwHB9KpbE51yMo1V/YBf2xK4OK9uT4XYDP/XE/H
# ZveVU3Fa4n5KWv64NmeFRiMMtY0Tz3cywBAY6GB9alKDRLemjkZrBxTzxXb1hlDc
# wUTIcVxRMTegCjhuje3XD9gmU3w5YQJ6xKr9cmmvHaus9ja+NSZk2pg7uhp7M62A
# W36MEBydUv626GIl3GoPz130/o5Tz9bshVZN7928jaTjkY+yOSxRnOlwaQ3KNi1w
# jjHINSi947SHJMPgyY9+tVSP3PoFVZhtaDuaRr3tpK56KTesy+uDRedGbsoy1cCG
# MFxPLOJiss254o2I5JasAUq7vnGpF1tnYN74kpEeHT39IM9zfUGaRnXNxF803RKJ
# 1v2lIH1+/NmeRd+2ci/bfV+AutuqfjbsNkz2K26oElHovwUDo9Fzpk03dJQcNIIP
# 8BDyt0cY7afomXw/TNuvXsLz1dhzPUNOwTM5TI4CvEJoLhDqhFFG4tG9ahhaYQFz
# ymeiXtcodgLiMxhy16cg8ML6EgrXY28MyTZki1ugpoMhXV8wdJGUlNi5UPkLiWHz
# NgY1GIRH29wb0f2y1BzFa/ZcUlFdEtsluq9QBXpsxREdcu+N+VLEhReTwDwV2xo3
# xwgVGD94q0W29R6HXtqPnhZyacaue7e3PmriLq0CAwEAAaOCAd0wggHZMBIGCSsG
# AQQBgjcVAQQFAgMBAAEwIwYJKwYBBAGCNxUCBBYEFCqnUv5kxJq+gpE8RjUpzxD/
# LwTuMB0GA1UdDgQWBBSfpxVdAF5iXYP05dJlpxtTNRnpcjBcBgNVHSAEVTBTMFEG
# DCsGAQQBgjdMg30BATBBMD8GCCsGAQUFBwIBFjNodHRwOi8vd3d3Lm1pY3Jvc29m
# dC5jb20vcGtpb3BzL0RvY3MvUmVwb3NpdG9yeS5odG0wEwYDVR0lBAwwCgYIKwYB
# BQUHAwgwGQYJKwYBBAGCNxQCBAweCgBTAHUAYgBDAEEwCwYDVR0PBAQDAgGGMA8G
# A1UdEwEB/wQFMAMBAf8wHwYDVR0jBBgwFoAU1fZWy4/oolxiaNE9lJBb186aGMQw
# VgYDVR0fBE8wTTBLoEmgR4ZFaHR0cDovL2NybC5taWNyb3NvZnQuY29tL3BraS9j
# cmwvcHJvZHVjdHMvTWljUm9vQ2VyQXV0XzIwMTAtMDYtMjMuY3JsMFoGCCsGAQUF
# BwEBBE4wTDBKBggrBgEFBQcwAoY+aHR0cDovL3d3dy5taWNyb3NvZnQuY29tL3Br
# aS9jZXJ0cy9NaWNSb29DZXJBdXRfMjAxMC0wNi0yMy5jcnQwDQYJKoZIhvcNAQEL
# BQADggIBAJ1VffwqreEsH2cBMSRb4Z5yS/ypb+pcFLY+TkdkeLEGk5c9MTO1OdfC
# cTY/2mRsfNB1OW27DzHkwo/7bNGhlBgi7ulmZzpTTd2YurYeeNg2LpypglYAA7AF
# vonoaeC6Ce5732pvvinLbtg/SHUB2RjebYIM9W0jVOR4U3UkV7ndn/OOPcbzaN9l
# 9qRWqveVtihVJ9AkvUCgvxm2EhIRXT0n4ECWOKz3+SmJw7wXsFSFQrP8DJ6LGYnn
# 8AtqgcKBGUIZUnWKNsIdw2FzLixre24/LAl4FOmRsqlb30mjdAy87JGA0j3mSj5m
# O0+7hvoyGtmW9I/2kQH2zsZ0/fZMcm8Qq3UwxTSwethQ/gpY3UA8x1RtnWN0SCyx
# TkctwRQEcb9k+SS+c23Kjgm9swFXSVRk2XPXfx5bRAGOWhmRaw2fpCjcZxkoJLo4
# S5pu+yFUa2pFEUep8beuyOiJXk+d0tBMdrVXVAmxaQFEfnyhYWxz/gq77EFmPWn9
# y8FBSX5+k77L+DvktxW/tM4+pTFRhLy/AsGConsXHRWJjXD+57XQKBqJC4822rpM
# +Zv/Cuk0+CQ1ZyvgDbjmjJnW4SLq8CdCPSWU5nR0W2rRnj7tfqAxM328y+l7vzhw
# RNGQ8cirOoo6CGJ/2XBjU02N7oJtpQUQwXEGahC0HVUzWLOhcGbyoYIDWTCCAkEC
# AQEwggEBoYHZpIHWMIHTMQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3Rv
# bjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0
# aW9uMS0wKwYDVQQLEyRNaWNyb3NvZnQgSXJlbGFuZCBPcGVyYXRpb25zIExpbWl0
# ZWQxJzAlBgNVBAsTHm5TaGllbGQgVFNTIEVTTjoyRDFBLTA1RTAtRDk0NzElMCMG
# A1UEAxMcTWljcm9zb2Z0IFRpbWUtU3RhbXAgU2VydmljZaIjCgEBMAcGBSsOAwIa
# AxUA5VHBr4h00EN7jUdQ33SE+qbk/8CggYMwgYCkfjB8MQswCQYDVQQGEwJVUzET
# MBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMV
# TWljcm9zb2Z0IENvcnBvcmF0aW9uMSYwJAYDVQQDEx1NaWNyb3NvZnQgVGltZS1T
# dGFtcCBQQ0EgMjAxMDANBgkqhkiG9w0BAQsFAAIFAO2hPHgwIhgPMjAyNjA1MDMw
# MzE2MDhaGA8yMDI2MDUwNDAzMTYwOFowdzA9BgorBgEEAYRZCgQBMS8wLTAKAgUA
# 7aE8eAIBADAKAgEAAgIqUwIB/zAHAgEAAgITfTAKAgUA7aKN+AIBADA2BgorBgEE
# AYRZCgQCMSgwJjAMBgorBgEEAYRZCgMCoAowCAIBAAIDB6EgoQowCAIBAAIDAYag
# MA0GCSqGSIb3DQEBCwUAA4IBAQAob1ciAs9EnklvfxsjRuxn1EXgx4NOdLcKwO7r
# /m10QuGw3cDnK5jwhUNWdMeurIe4d2zoVwlJgrsFVTWv+nvZyWMlnmDEU98vaSfz
# 3JV2V0NV5kkFXpPHpgAfHt+9sECdSs0S5NBD1s6J01gk2n5kuYpvZxwhdnWuPgd2
# EU+7p6GemsEK6nFR0BnCQZuMemNqEoeAA9i+Vq9eMf7zaz7JPTPSDm+IhzYSvVNN
# KSqZoNx6r90e81uqbjDqpfPH8XKaD9ETB1R/lHk9ba4h9LiMKlS0pcn/hCfSXvlM
# DTtRk7xFgU16wGTho3HULYVtQGFdC6H06IGmCtpThd1CtjCeMYIEDTCCBAkCAQEw
# gZMwfDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcT
# B1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEmMCQGA1UE
# AxMdTWljcm9zb2Z0IFRpbWUtU3RhbXAgUENBIDIwMTACEzMAAAIS0QgGPMoYT6oA
# AQAAAhIwDQYJYIZIAWUDBAIBBQCgggFKMBoGCSqGSIb3DQEJAzENBgsqhkiG9w0B
# CRABBDAvBgkqhkiG9w0BCQQxIgQgibqpdlvhw/l4FpIIbnpvH3QnJz3vdKFlClNE
# HXD7hVAwgfoGCyqGSIb3DQEJEAIvMYHqMIHnMIHkMIG9BCBz+X5GvO7WngknH4BZ
# eYU+BzBL1Jy5oJ8wVlTNIxfYgzCBmDCBgKR+MHwxCzAJBgNVBAYTAlVTMRMwEQYD
# VQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNy
# b3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1w
# IFBDQSAyMDEwAhMzAAACEtEIBjzKGE+qAAEAAAISMCIEIH62C3OysJpL+INwkKSZ
# fhuOdJ2Sj9gOYyp366y/mz5vMA0GCSqGSIb3DQEBCwUABIICAIAYGiyz2lTvJLtN
# C1v0v6hnRPQYIoN80cBU0+G0RO+CxwIq6CVxhQcojhGwZ5RJf7DhMvuUIIXm+793
# KnsU8cBrB4pM6GS2OcP+7VMSDYtiys96L3lxk4ZioiFNjlq7Zr9XB4PWAj/5HpPG
# wtQoDDFGvp3Y49zYXgY93/N0A6Rt7hnh3wkKPx9u6ahyIzgNCW2oIBA64Ipj2gO0
# XL2wBRl0Xse4OVFnLRWJF7Rzq5GNcCx/DHPv7FRSi7gksdBkXcPmxVqPgOxWe4w4
# dn99xg2RFmhOVfNfhwSxjvMBTQnHTtKdBrO6bkiz1++MdI4/II6aquIJT5Q5D7CP
# 2rCm8AH/teHZcdy8OzKomgTCcCALjFbQMWx2VFYkaL6JALURB6yQuUlXziCs0d+j
# My4CZYhLut90694lWO6e5sGExvdyLy1OH2QBUlMtkmFNSg6HC5hgRx0rfPAMRN1g
# Dx2vD2doTZ9ZO6OywhiBj9TQlqQXdvyIEqeK5puGQ/L8AhPXYtSxoMVZm1pxSirN
# wXo+wIYsYwHX7Ii8Wg9Fg5CGZe7LsxSAyW26GTTpTyCsbzRB3JdyV6/+2/neaQO9
# PkP+zAuv8pZk9JsOEmizf0N2JgCtpkFzjqpj+vCsloGGItoaXx4Z64HOM8llkYOk
# nNnBkSvKEZ1AFu82bWJD4k53xz2z
# SIG # End signature block