Public/System/Watch-ISPConnection.ps1

function Watch-ISPConnection {
    <#
    .SYNOPSIS
      Monitors connectivity to a host (IP, DNS name, or URL) and logs structured
      results.
     
    .DESCRIPTION
      Starts a background monitoring job that pings the formatted target at the
      specified interval and writes CSV rows to a per-session log file. Provides
      Start/Stop/Status.
     
    .PARAMETER Target
      IP, DNS name, or URL (e.g., https://example.com). URLs will be formatted to
      host.
     
    .PARAMETER IntervalSeconds
      Seconds between samples. Minimum 1. Default 10.
     
    .PARAMETER SessionName
      Optional name; if omitted, a random one is generated.
     
    .PARAMETER TimeoutSeconds
      Timeout for each ping attempt. Default 3.
     
    .PARAMETER Start
      Start a new monitoring session.
     
    .PARAMETER Stop
      Stop a running session by name.
     
    .PARAMETER Status
      Show status of all sessions (jobs, start time, last heartbeat, etc.).
    #>


    [CmdletBinding(DefaultParameterSetName = 'Start')]
    param(
        # --- START ---
        [Parameter(ParameterSetName = 'Start', Mandatory = $true)]
        [Alias('FirewallIP')]
        [string]$Target,

        [Parameter(ParameterSetName = 'Start')]
        [ValidateRange(1, 86400)]
        [int]$IntervalSeconds = 10,

        [Parameter(ParameterSetName = 'Start')]
        [ValidateRange(1, 60)]
        [int]$TimeoutSeconds = 3,

        [Parameter(ParameterSetName = 'Start')]
        [string]$SessionName,

        # --- STOP ---
        [Parameter(ParameterSetName = 'Stop', Mandatory = $true)]
        [switch]$Stop,

        [Parameter(ParameterSetName = 'Stop', Mandatory = $true)]
        [string]$StopSessionName,

        # --- STATUS ---
        [Parameter(ParameterSetName = 'Status', Mandatory = $true)]
        [switch]$Status
    )

    begin {
        # Load dependencies
        Initialize-TechToolboxRuntime
        
        # SAFE StrictMode-friendly init
        $existing = Get-Variable -Name ISPWatchSessions -Scope Script -ErrorAction SilentlyContinue
        if (-not $existing) {
            Set-Variable -Name ISPWatchSessions -Scope Script -Value (@{}) -Option None
        }

        if ($PSCmdlet.ParameterSetName -eq 'Start' -and $IntervalSeconds -lt 1) {
            throw "IntervalSeconds must be >= 1."
        }

        $rootLogPath = $script:cfg.settings.ispConnection.logPath
        try {
            $null = New-Item -ItemType Directory -Path $rootLogPath -Force -ErrorAction Stop
        }
        catch {
            Write-Log -Level ERROR -Message "Failed to create log directory '$rootLogPath': $($_.Exception.Message)"
            throw
        }

        function Format-TargetHost {
            param([string]$InputTarget)
            try {
                $u = [uri]$InputTarget
                if ($u -and $u.Host) { return $u.DnsSafeHost }
            }
            catch { }
            return $InputTarget
        }

        function New-SessionName {
            param([string]$Base)
            if ([string]::IsNullOrWhiteSpace($Base)) {
                return "ISPWatch_{0:yyyyMMddHHmmss}_{1}" -f (Get-Date), (Get-Random -Max 99999)
            }
            return $Base
        }
    }

    process {
        switch ($PSCmdlet.ParameterSetName) {

            'Start' {
                $hostOrIp = Format-TargetHost -InputTarget $Target
                if ([string]::IsNullOrWhiteSpace($hostOrIp)) {
                    throw "Target '$Target' could not be formatted to a hostname or IP."
                }

                $name = New-SessionName -Base $SessionName

                if ($script:ISPWatchSessions.ContainsKey($name)) {
                    throw "A session named '$name' already exists."
                }

                $sessionLogDir = Join-Path $rootLogPath $name
                $sessionCsv = Join-Path $sessionLogDir "$name.csv"

                try {
                    New-Item -ItemType Directory -Path $sessionLogDir -Force | Out-Null
                }
                catch {
                    Write-Log -Level Error -Message "Failed to ensure session log directory '$sessionLogDir': $($_.Exception.Message)"
                    throw
                }

                # Start thread job that runs the sample loop
                $job = Start-ThreadJob -Name $name -ArgumentList @($hostOrIp, $IntervalSeconds, $TimeoutSeconds, $sessionCsv) -ScriptBlock {
                    param($target, $interval, $timeout, $csvPath)

                    # Ensure CSV header
                    if (-not (Test-Path -Path $csvPath)) {
                        'Timestamp,Target,ResolvedAddress,Success,LatencyMs,ErrorType,ErrorMessage' | Out-File -FilePath $csvPath -Encoding utf8
                    }

                    while ($true) {
                        $ts = Get-Date
                        $resolved = $null
                        $success = $false
                        $latency = $null
                        $errType = ''
                        $errMsg = ''

                        try {
                            # DNS resolve (optional but useful for logging)
                            try {
                                $res = Resolve-DnsName -Name $target -Type A -ErrorAction Stop
                                $resolved = ($res | Select-Object -First 1 -ExpandProperty IPAddress)
                            }
                            catch {
                                # If Resolve-DnsName fails (non-Windows or no DNS), continue; ping may still work
                                $resolved = ''
                            }

                            # Single ping with timeout; PS7 Test-Connection returns objects
                            $pong = Test-Connection -TargetName $target -Count 1 -TimeoutSeconds $timeout -ErrorAction Stop
                            if ($pong) {
                                $success = $true
                                # On PS7+, Latency is a property; fall back if missing
                                $latency = ($pong | Select-Object -First 1 -ExpandProperty Latency -ErrorAction SilentlyContinue)
                                if (-not $latency) { $latency = '' }
                            }
                        }
                        catch {
                            $errType = $_.Exception.GetType().Name
                            $errMsg = ($_.Exception.Message -replace '[\r\n]+', ' ')
                        }

                        $line = ('{0:o},{1},{2},{3},{4},{5},"{6}"' -f $ts, $target, $resolved, $success, $latency, $errType, $errMsg)
                        try {
                            Add-Content -Path $csvPath -Value $line -Encoding utf8
                        }
                        catch {
                            # If we can't write logs, there's no point continuing
                            break
                        }

                        Start-Sleep -Seconds $interval
                    }
                }

                $script:ISPWatchSessions[$name] = [ordered]@{
                    Name            = $name
                    Target          = $hostOrIp
                    IntervalSeconds = $IntervalSeconds
                    TimeoutSeconds  = $TimeoutSeconds
                    StartTime       = (Get-Date)
                    JobId           = $job.Id
                    LogPath         = $sessionCsv
                }

                Write-Log -Level Info -Message "Session '$name' started for target '$hostOrIp' (JobId=$($job.Id))"
                [pscustomobject]$script:ISPWatchSessions[$name]
            }

            'Stop' {
                $name = $StopSessionName
                if (-not $script:ISPWatchSessions.ContainsKey($name)) {
                    Write-Log -Level Error -Message "Session '$name' not found."
                    return
                }

                $jobId = $script:ISPWatchSessions[$name].JobId
                $job = Get-Job -Id $jobId -ErrorAction SilentlyContinue
                if ($job) {
                    try {
                        Stop-Job -Job $job -ErrorAction SilentlyContinue
                        Receive-Job -Job $job -ErrorAction SilentlyContinue | Out-Null
                        Remove-Job -Job $job -Force -ErrorAction SilentlyContinue
                    }
                    catch { }
                }

                $null = $script:ISPWatchSessions.Remove($name)
                Write-Log -Level Info -Message "Session '$name' stopped and removed."
            }

            'Status' {
                if ($script:ISPWatchSessions.Count -eq 0) {
                    Write-Log -Level Info -Message "No active sessions."
                    return
                }

                $script:ISPWatchSessions.GetEnumerator() | ForEach-Object {
                    $Info = $_.Value
                    $job = Get-Job -Id $Info.JobId -ErrorAction SilentlyContinue
                    $state = if ($job) { $job.State } else { 'Unknown' }
                    [pscustomobject]@{
                        SessionName     = $Info.Name
                        Target          = $Info.Target
                        IntervalSeconds = $Info.IntervalSeconds
                        TimeoutSeconds  = $Info.TimeoutSeconds
                        Started         = $Info.StartTime
                        JobState        = $state
                        LogPath         = $Info.LogPath
                    }
                } | Format-Table -AutoSize
            }
        }
    }
}

# SIG # Begin signature block
# MIIfAgYJKoZIhvcNAQcCoIIe8zCCHu8CAQExDzANBglghkgBZQMEAgEFADB5Bgor
# BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG
# KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCBJXjSE5dAXCG2k
# LCJyguQEpMJ4ZgnszteTqIIpvGjKuqCCGEowggUMMIIC9KADAgECAhAR+U4xG7FH
# qkyqS9NIt7l5MA0GCSqGSIb3DQEBCwUAMB4xHDAaBgNVBAMME1ZBRFRFSyBDb2Rl
# IFNpZ25pbmcwHhcNMjUxMjE5MTk1NDIxWhcNMjYxMjE5MjAwNDIxWjAeMRwwGgYD
# VQQDDBNWQURURUsgQ29kZSBTaWduaW5nMIICIjANBgkqhkiG9w0BAQEFAAOCAg8A
# MIICCgKCAgEA3pzzZIUEY92GDldMWuzvbLeivHOuMupgpwbezoG5v90KeuN03S5d
# nM/eom/PcIz08+fGZF04ueuCS6b48q1qFnylwg/C/TkcVRo0WFcKoFGT8yGxdfXi
# caHtapZfbSRh73r7qR7w0CioVveNBVgfMsTgE0WKcuwxemvIe/ptmkfzwAiw/IAC
# Ib0E0BjiX4PySbwWy/QKy/qMXYY19xpRItVTKNBtXzADUtzPzUcFqJU83vM2gZFs
# Or0MhPvM7xEVkOWZFBAWAubbMCJ3rmwyVv9keVDJChhCeLSz2XR11VGDOEA2OO90
# Y30WfY9aOI2sCfQcKMeJ9ypkHl0xORdhUwZ3Wz48d3yJDXGkduPm2vl05RvnA4T6
# 29HVZTmMdvP2475/8nLxCte9IB7TobAOGl6P1NuwplAMKM8qyZh62Br23vcx1fXZ
# TJlKCxBFx1nTa6VlIJk+UbM4ZPm954peB/fIqEacm8LkZ0cPwmLE5ckW7hfK4Trs
# o+RaudU1sKeA+FvpOWgsPccVRWcEYyGkwbyTB3xrIBXA+YckbANZ0XL7fv7x29hn
# gXbZipGu3DnTISiFB43V4MhNDKZYfbWdxze0SwLe8KzIaKnwlwRgvXDMwXgk99Mi
# EbYa3DvA/5ZWikLW9PxBFD7Vdr8ZiG/tRC9I2Y6fnb+PVoZKc/2xsW0CAwEAAaNG
# MEQwDgYDVR0PAQH/BAQDAgeAMBMGA1UdJQQMMAoGCCsGAQUFBwMDMB0GA1UdDgQW
# BBRfYLVE8caSc990rnrIHUjoB7X/KjANBgkqhkiG9w0BAQsFAAOCAgEAiGB2Wmk3
# QBtd1LcynmxHzmu+X4Y5DIpMMNC2ahsqZtPUVcGqmb5IFbVuAdQphL6PSrDjaAR8
# 1S8uTfUnMa119LmIb7di7TlH2F5K3530h5x8JMj5EErl0xmZyJtSg7BTiBA/UrMz
# 6WCf8wWIG2/4NbV6aAyFwIojfAcKoO8ng44Dal/oLGzLO3FDE5AWhcda/FbqVjSJ
# 1zMfiW8odd4LgbmoyEI024KkwOkkPyJQ2Ugn6HMqlFLazAmBBpyS7wxdaAGrl18n
# 6bS7QuAwCd9hitdMMitG8YyWL6tKeRSbuTP5E+ASbu0Ga8/fxRO5ZSQhO6/5ro1j
# PGe1/Kr49Uyuf9VSCZdNIZAyjjeVAoxmV0IfxQLKz6VOG0kGDYkFGskvllIpQbQg
# WLuPLJxoskJsoJllk7MjZJwrpr08+3FQnLkRuisjDOc3l4VxFUsUe4fnJhMUONXT
# Sk7vdspgxirNbLmXU4yYWdsizz3nMUR0zebUW29A+HYme16hzrMPOeyoQjy4I5XX
# 3wXAFdworfPEr/ozDFrdXKgbLwZopymKbBwv6wtT7+1zVhJXr+jGVQ1TWr6R+8ea
# tIOFnY7HqGaxe5XB7HzOwJKdj+bpHAfXft1vUoiKr16VajLigcYCG8MdwC3sngO3
# JDyv2V+YMfsYBmItMGBwvizlQ6557NbK95EwggWNMIIEdaADAgECAhAOmxiO+dAt
# 5+/bUOIIQBhaMA0GCSqGSIb3DQEBDAUAMGUxCzAJBgNVBAYTAlVTMRUwEwYDVQQK
# EwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xJDAiBgNV
# BAMTG0RpZ2lDZXJ0IEFzc3VyZWQgSUQgUm9vdCBDQTAeFw0yMjA4MDEwMDAwMDBa
# Fw0zMTExMDkyMzU5NTlaMGIxCzAJBgNVBAYTAlVTMRUwEwYDVQQKEwxEaWdpQ2Vy
# dCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xITAfBgNVBAMTGERpZ2lD
# ZXJ0IFRydXN0ZWQgUm9vdCBHNDCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoC
# ggIBAL/mkHNo3rvkXUo8MCIwaTPswqclLskhPfKK2FnC4SmnPVirdprNrnsbhA3E
# MB/zG6Q4FutWxpdtHauyefLKEdLkX9YFPFIPUh/GnhWlfr6fqVcWWVVyr2iTcMKy
# unWZanMylNEQRBAu34LzB4TmdDttceItDBvuINXJIB1jKS3O7F5OyJP4IWGbNOsF
# xl7sWxq868nPzaw0QF+xembud8hIqGZXV59UWI4MK7dPpzDZVu7Ke13jrclPXuU1
# 5zHL2pNe3I6PgNq2kZhAkHnDeMe2scS1ahg4AxCN2NQ3pC4FfYj1gj4QkXCrVYJB
# MtfbBHMqbpEBfCFM1LyuGwN1XXhm2ToxRJozQL8I11pJpMLmqaBn3aQnvKFPObUR
# WBf3JFxGj2T3wWmIdph2PVldQnaHiZdpekjw4KISG2aadMreSx7nDmOu5tTvkpI6
# nj3cAORFJYm2mkQZK37AlLTSYW3rM9nF30sEAMx9HJXDj/chsrIRt7t/8tWMcCxB
# YKqxYxhElRp2Yn72gLD76GSmM9GJB+G9t+ZDpBi4pncB4Q+UDCEdslQpJYls5Q5S
# UUd0viastkF13nqsX40/ybzTQRESW+UQUOsxxcpyFiIJ33xMdT9j7CFfxCBRa2+x
# q4aLT8LWRV+dIPyhHsXAj6KxfgommfXkaS+YHS312amyHeUbAgMBAAGjggE6MIIB
# NjAPBgNVHRMBAf8EBTADAQH/MB0GA1UdDgQWBBTs1+OC0nFdZEzfLmc/57qYrhwP
# TzAfBgNVHSMEGDAWgBRF66Kv9JLLgjEtUYunpyGd823IDzAOBgNVHQ8BAf8EBAMC
# AYYweQYIKwYBBQUHAQEEbTBrMCQGCCsGAQUFBzABhhhodHRwOi8vb2NzcC5kaWdp
# Y2VydC5jb20wQwYIKwYBBQUHMAKGN2h0dHA6Ly9jYWNlcnRzLmRpZ2ljZXJ0LmNv
# bS9EaWdpQ2VydEFzc3VyZWRJRFJvb3RDQS5jcnQwRQYDVR0fBD4wPDA6oDigNoY0
# aHR0cDovL2NybDMuZGlnaWNlcnQuY29tL0RpZ2lDZXJ0QXNzdXJlZElEUm9vdENB
# LmNybDARBgNVHSAECjAIMAYGBFUdIAAwDQYJKoZIhvcNAQEMBQADggEBAHCgv0Nc
# Vec4X6CjdBs9thbX979XB72arKGHLOyFXqkauyL4hxppVCLtpIh3bb0aFPQTSnov
# Lbc47/T/gLn4offyct4kvFIDyE7QKt76LVbP+fT3rDB6mouyXtTP0UNEm0Mh65Zy
# oUi0mcudT6cGAxN3J0TU53/oWajwvy8LpunyNDzs9wPHh6jSTEAZNUZqaVSwuKFW
# juyk1T3osdz9HNj0d1pcVIxv76FQPfx2CWiEn2/K2yCNNWAcAgPLILCsWKAOQGPF
# mCLBsln1VWvPJ6tsds5vIy30fnFqI2si/xK4VC0nftg62fC2h5b9W9FcrBjDTZ9z
# twGpn1eqXijiuZQwgga0MIIEnKADAgECAhANx6xXBf8hmS5AQyIMOkmGMA0GCSqG
# SIb3DQEBCwUAMGIxCzAJBgNVBAYTAlVTMRUwEwYDVQQKEwxEaWdpQ2VydCBJbmMx
# GTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xITAfBgNVBAMTGERpZ2lDZXJ0IFRy
# dXN0ZWQgUm9vdCBHNDAeFw0yNTA1MDcwMDAwMDBaFw0zODAxMTQyMzU5NTlaMGkx
# CzAJBgNVBAYTAlVTMRcwFQYDVQQKEw5EaWdpQ2VydCwgSW5jLjFBMD8GA1UEAxM4
# RGlnaUNlcnQgVHJ1c3RlZCBHNCBUaW1lU3RhbXBpbmcgUlNBNDA5NiBTSEEyNTYg
# MjAyNSBDQTEwggIiMA0GCSqGSIb3DQEBAQUAA4ICDwAwggIKAoICAQC0eDHTCphB
# cr48RsAcrHXbo0ZodLRRF51NrY0NlLWZloMsVO1DahGPNRcybEKq+RuwOnPhof6p
# vF4uGjwjqNjfEvUi6wuim5bap+0lgloM2zX4kftn5B1IpYzTqpyFQ/4Bt0mAxAHe
# HYNnQxqXmRinvuNgxVBdJkf77S2uPoCj7GH8BLuxBG5AvftBdsOECS1UkxBvMgEd
# gkFiDNYiOTx4OtiFcMSkqTtF2hfQz3zQSku2Ws3IfDReb6e3mmdglTcaarps0wjU
# jsZvkgFkriK9tUKJm/s80FiocSk1VYLZlDwFt+cVFBURJg6zMUjZa/zbCclF83bR
# VFLeGkuAhHiGPMvSGmhgaTzVyhYn4p0+8y9oHRaQT/aofEnS5xLrfxnGpTXiUOeS
# LsJygoLPp66bkDX1ZlAeSpQl92QOMeRxykvq6gbylsXQskBBBnGy3tW/AMOMCZIV
# NSaz7BX8VtYGqLt9MmeOreGPRdtBx3yGOP+rx3rKWDEJlIqLXvJWnY0v5ydPpOjL
# 6s36czwzsucuoKs7Yk/ehb//Wx+5kMqIMRvUBDx6z1ev+7psNOdgJMoiwOrUG2Zd
# SoQbU2rMkpLiQ6bGRinZbI4OLu9BMIFm1UUl9VnePs6BaaeEWvjJSjNm2qA+sdFU
# eEY0qVjPKOWug/G6X5uAiynM7Bu2ayBjUwIDAQABo4IBXTCCAVkwEgYDVR0TAQH/
# BAgwBgEB/wIBADAdBgNVHQ4EFgQU729TSunkBnx6yuKQVvYv1Ensy04wHwYDVR0j
# BBgwFoAU7NfjgtJxXWRM3y5nP+e6mK4cD08wDgYDVR0PAQH/BAQDAgGGMBMGA1Ud
# JQQMMAoGCCsGAQUFBwMIMHcGCCsGAQUFBwEBBGswaTAkBggrBgEFBQcwAYYYaHR0
# cDovL29jc3AuZGlnaWNlcnQuY29tMEEGCCsGAQUFBzAChjVodHRwOi8vY2FjZXJ0
# cy5kaWdpY2VydC5jb20vRGlnaUNlcnRUcnVzdGVkUm9vdEc0LmNydDBDBgNVHR8E
# PDA6MDigNqA0hjJodHRwOi8vY3JsMy5kaWdpY2VydC5jb20vRGlnaUNlcnRUcnVz
# dGVkUm9vdEc0LmNybDAgBgNVHSAEGTAXMAgGBmeBDAEEAjALBglghkgBhv1sBwEw
# DQYJKoZIhvcNAQELBQADggIBABfO+xaAHP4HPRF2cTC9vgvItTSmf83Qh8WIGjB/
# T8ObXAZz8OjuhUxjaaFdleMM0lBryPTQM2qEJPe36zwbSI/mS83afsl3YTj+IQhQ
# E7jU/kXjjytJgnn0hvrV6hqWGd3rLAUt6vJy9lMDPjTLxLgXf9r5nWMQwr8Myb9r
# EVKChHyfpzee5kH0F8HABBgr0UdqirZ7bowe9Vj2AIMD8liyrukZ2iA/wdG2th9y
# 1IsA0QF8dTXqvcnTmpfeQh35k5zOCPmSNq1UH410ANVko43+Cdmu4y81hjajV/gx
# dEkMx1NKU4uHQcKfZxAvBAKqMVuqte69M9J6A47OvgRaPs+2ykgcGV00TYr2Lr3t
# y9qIijanrUR3anzEwlvzZiiyfTPjLbnFRsjsYg39OlV8cipDoq7+qNNjqFzeGxcy
# tL5TTLL4ZaoBdqbhOhZ3ZRDUphPvSRmMThi0vw9vODRzW6AxnJll38F0cuJG7uEB
# YTptMSbhdhGQDpOXgpIUsWTjd6xpR6oaQf/DJbg3s6KCLPAlZ66RzIg9sC+NJpud
# /v4+7RWsWCiKi9EOLLHfMR2ZyJ/+xhCx9yHbxtl5TPau1j/1MIDpMPx0LckTetiS
# uEtQvLsNz3Qbp7wGWqbIiOWCnb5WqxL3/BAPvIXKUjPSxyZsq8WhbaM2tszWkPZP
# ubdcMIIG7TCCBNWgAwIBAgIQCoDvGEuN8QWC0cR2p5V0aDANBgkqhkiG9w0BAQsF
# ADBpMQswCQYDVQQGEwJVUzEXMBUGA1UEChMORGlnaUNlcnQsIEluYy4xQTA/BgNV
# BAMTOERpZ2lDZXJ0IFRydXN0ZWQgRzQgVGltZVN0YW1waW5nIFJTQTQwOTYgU0hB
# MjU2IDIwMjUgQ0ExMB4XDTI1MDYwNDAwMDAwMFoXDTM2MDkwMzIzNTk1OVowYzEL
# MAkGA1UEBhMCVVMxFzAVBgNVBAoTDkRpZ2lDZXJ0LCBJbmMuMTswOQYDVQQDEzJE
# aWdpQ2VydCBTSEEyNTYgUlNBNDA5NiBUaW1lc3RhbXAgUmVzcG9uZGVyIDIwMjUg
# MTCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBANBGrC0Sxp7Q6q5gVrMr
# V7pvUf+GcAoB38o3zBlCMGMyqJnfFNZx+wvA69HFTBdwbHwBSOeLpvPnZ8ZN+vo8
# dE2/pPvOx/Vj8TchTySA2R4QKpVD7dvNZh6wW2R6kSu9RJt/4QhguSssp3qome7M
# rxVyfQO9sMx6ZAWjFDYOzDi8SOhPUWlLnh00Cll8pjrUcCV3K3E0zz09ldQ//nBZ
# ZREr4h/GI6Dxb2UoyrN0ijtUDVHRXdmncOOMA3CoB/iUSROUINDT98oksouTMYFO
# nHoRh6+86Ltc5zjPKHW5KqCvpSduSwhwUmotuQhcg9tw2YD3w6ySSSu+3qU8DD+n
# igNJFmt6LAHvH3KSuNLoZLc1Hf2JNMVL4Q1OpbybpMe46YceNA0LfNsnqcnpJeIt
# K/DhKbPxTTuGoX7wJNdoRORVbPR1VVnDuSeHVZlc4seAO+6d2sC26/PQPdP51ho1
# zBp+xUIZkpSFA8vWdoUoHLWnqWU3dCCyFG1roSrgHjSHlq8xymLnjCbSLZ49kPmk
# 8iyyizNDIXj//cOgrY7rlRyTlaCCfw7aSUROwnu7zER6EaJ+AliL7ojTdS5PWPsW
# eupWs7NpChUk555K096V1hE0yZIXe+giAwW00aHzrDchIc2bQhpp0IoKRR7YufAk
# prxMiXAJQ1XCmnCfgPf8+3mnAgMBAAGjggGVMIIBkTAMBgNVHRMBAf8EAjAAMB0G
# A1UdDgQWBBTkO/zyMe39/dfzkXFjGVBDz2GM6DAfBgNVHSMEGDAWgBTvb1NK6eQG
# fHrK4pBW9i/USezLTjAOBgNVHQ8BAf8EBAMCB4AwFgYDVR0lAQH/BAwwCgYIKwYB
# BQUHAwgwgZUGCCsGAQUFBwEBBIGIMIGFMCQGCCsGAQUFBzABhhhodHRwOi8vb2Nz
# cC5kaWdpY2VydC5jb20wXQYIKwYBBQUHMAKGUWh0dHA6Ly9jYWNlcnRzLmRpZ2lj
# ZXJ0LmNvbS9EaWdpQ2VydFRydXN0ZWRHNFRpbWVTdGFtcGluZ1JTQTQwOTZTSEEy
# NTYyMDI1Q0ExLmNydDBfBgNVHR8EWDBWMFSgUqBQhk5odHRwOi8vY3JsMy5kaWdp
# Y2VydC5jb20vRGlnaUNlcnRUcnVzdGVkRzRUaW1lU3RhbXBpbmdSU0E0MDk2U0hB
# MjU2MjAyNUNBMS5jcmwwIAYDVR0gBBkwFzAIBgZngQwBBAIwCwYJYIZIAYb9bAcB
# MA0GCSqGSIb3DQEBCwUAA4ICAQBlKq3xHCcEua5gQezRCESeY0ByIfjk9iJP2zWL
# pQq1b4URGnwWBdEZD9gBq9fNaNmFj6Eh8/YmRDfxT7C0k8FUFqNh+tshgb4O6Lgj
# g8K8elC4+oWCqnU/ML9lFfim8/9yJmZSe2F8AQ/UdKFOtj7YMTmqPO9mzskgiC3Q
# YIUP2S3HQvHG1FDu+WUqW4daIqToXFE/JQ/EABgfZXLWU0ziTN6R3ygQBHMUBaB5
# bdrPbF6MRYs03h4obEMnxYOX8VBRKe1uNnzQVTeLni2nHkX/QqvXnNb+YkDFkxUG
# tMTaiLR9wjxUxu2hECZpqyU1d0IbX6Wq8/gVutDojBIFeRlqAcuEVT0cKsb+zJNE
# suEB7O7/cuvTQasnM9AWcIQfVjnzrvwiCZ85EE8LUkqRhoS3Y50OHgaY7T/lwd6U
# Arb+BOVAkg2oOvol/DJgddJ35XTxfUlQ+8Hggt8l2Yv7roancJIFcbojBcxlRcGG
# 0LIhp6GvReQGgMgYxQbV1S3CrWqZzBt1R9xJgKf47CdxVRd/ndUlQ05oxYy2zRWV
# FjF7mcr4C34Mj3ocCVccAvlKV9jEnstrniLvUxxVZE/rptb7IRE2lskKPIJgbaP5
# t2nGj/ULLi49xTcBZU8atufk+EMF/cWuiC7POGT75qaL6vdCvHlshtjdNXOCIUjs
# arfNZzGCBg4wggYKAgEBMDIwHjEcMBoGA1UEAwwTVkFEVEVLIENvZGUgU2lnbmlu
# ZwIQEflOMRuxR6pMqkvTSLe5eTANBglghkgBZQMEAgEFAKCBhDAYBgorBgEEAYI3
# AgEMMQowCKACgAChAoAAMBkGCSqGSIb3DQEJAzEMBgorBgEEAYI3AgEEMBwGCisG
# AQQBgjcCAQsxDjAMBgorBgEEAYI3AgEVMC8GCSqGSIb3DQEJBDEiBCB/PP++7jRj
# Drq2bH0rRy3gpshxX1McVAw3Yw3g+aHuGzANBgkqhkiG9w0BAQEFAASCAgBI4lxx
# KBYPJeaTOKWVkkshz/jXhZeMdBUe06iDwLCyjV4pqyZW+UElStCxGz8ZB8wWcuU6
# y0b9LZmA1dAgEqk8u4PWRzIxBAFDjMmWvhLs4krELrQSvCgxvr/S0GZxP3mgX5Hj
# ZWsm3CugZMhEfaxCd2RcqHZZhkdDR0E/FJL3/MVUls1fvV65WnDYzB6zxmuRimkr
# 9V3ZafUjMYEQGmQBuoZHVrY7Ohfq3R3e54Mk/EUGyRW9pEEatfQ/QYw6ZlObw4Qm
# gQm/4Cz+pBLmUc9510mQNbUky2xjiCMqVgu8T/AHrmlzg0J+Gag0gmyXa5msFRNg
# HkbcFEBTbORPF/V651JTG7wjvERPIi0DsMQjGp8K7NRgGKkPCGtdnXizhH9ICVLF
# wrbNfLI7+h30lU4IXhIqGfHiuVrzjesu34/NUPE5Z1+KhRHiWlvbcJZ6NBM0MJ3A
# 250pCkkPDElJokrRpoKveL6O7qLDNDeFulDJX50NU7MLOamQMp8cZAZBmL8mJyed
# FrHYBSmgzKMqfYhv3T8wm+jiMVk1NzFS1csnP1SMiVJtCDoX3OvjIyx4MU9zgteg
# EfF3OyzqIEgKF9CCbW7PPrA/H1qZad/Vit9jw4d/tpr+ZeMguoS0me2/Z9gkSfxM
# W0o3y+b0SHpWqUF9zUNquSswA8uXActnsRT8faGCAyYwggMiBgkqhkiG9w0BCQYx
# ggMTMIIDDwIBATB9MGkxCzAJBgNVBAYTAlVTMRcwFQYDVQQKEw5EaWdpQ2VydCwg
# SW5jLjFBMD8GA1UEAxM4RGlnaUNlcnQgVHJ1c3RlZCBHNCBUaW1lU3RhbXBpbmcg
# UlNBNDA5NiBTSEEyNTYgMjAyNSBDQTECEAqA7xhLjfEFgtHEdqeVdGgwDQYJYIZI
# AWUDBAIBBQCgaTAYBgkqhkiG9w0BCQMxCwYJKoZIhvcNAQcBMBwGCSqGSIb3DQEJ
# BTEPFw0yNjAyMTEwMzUxMjJaMC8GCSqGSIb3DQEJBDEiBCA7gYFGKc+iqGs2gVhv
# /shE7z2Sx2gNDSvKn8aYETgZQTANBgkqhkiG9w0BAQEFAASCAgAf7bElnkpGRHr7
# lFLid8KfRICB8pdom7+aEHSjdNbWEb7U/DR8cLlACHyO59eo5TLl8G7hNlFChqeH
# kknBzFk6g7csXTJ9/Lh6AJLQtX1zDtKyqSnDSaZN5GdJeLV0/4QzO7f6T6zCi/dc
# KfDCUTJmaUisYHuDIl10x41CiXQQSQoBwbytUaShaBCz2sc3oryn9U9RWT1AHC/U
# T9/jDPOwnVAULgk/ImuCPAQztkQenGCzBE0FDji0uyJQkL2sg++W9WIXcBvEL1jk
# H0cunYzAB5pmvZUUyEnSqPg1ko8dwYPikhllJEwXqY0cWOIfsVmiVBWdzomu3zfu
# 921MIzpXyODEu+tUKi0Isr8abYfh81RFmQ9fxR2+hocl5c2mLKU7uKILX4BOt+p1
# Wz6Q3FXRCOIdXe5W3a0bF97VjaB+7ZXUW0C9gnjSK3wbpoYq/rx+f7le2WJW0wT/
# 5he7Ut5qKmtP5rtFngMTyqlkCQaO0AgM8gdv4tavKyY0ygLcpLTWE0ej+u+CJEag
# CmwwA7hMCdRn+BfqgOgxteodulXbgIEXyWwkVqrnIXA8Uh9h8yfLkc+gjUbPU/T0
# eHKnblTJ/d8U5DbiKPQ6WKwR68jjoDUz//ZBBZjHamiUGhpt2bXxfCdxRhLqcABn
# wrcTPJeJ+fsZ04N5woheBKqhmuNefQ==
# SIG # End signature block