Private/AI/Invoke-LocalLLM.ps1

function Invoke-LocalLLM {
    [CmdletBinding()]
    param(
        [Parameter(Mandatory)]
        [string]$Prompt,

        [string]$Model = 'qwen2.5-coder:32b',

        # Optional override for streaming
        [switch]$ForceStream,

        # Robustness knobs
        [int]$TimeoutMinutes = 15,
        [int]$RetryCount = 1,
        [int]$RetryDelaySeconds = 2,

        # Prefer returning an object (TechToolbox-friendly)
        [switch]$ThrowOnError
    )

    # -----------------------------
    # STREAMING RULES (deterministic)
    # -----------------------------
    $streamingModels = @(
        'qwen2.5-coder:14b'   # Known stable streaming model
    )

    $Stream =
    if ($ForceStream) { $true }
    elseif ($streamingModels -contains $Model) { $true }
    else { $false }

    # -----------------------------
    # Helpers
    # -----------------------------
    function New-LLMResult {
        param(
            [bool]$Success,
            [string]$Text,
            [string]$Model,
            [bool]$Stream,
            [TimeSpan]$Duration,
            [Nullable[int]]$StatusCode,
            [string]$ReasonPhrase,
            [System.Exception]$Exception,
            [System.Management.Automation.ErrorRecord]$ErrorRecord
        )

        [pscustomobject]@{
            PSTypeName   = 'TechToolbox.LocalLLM.Result'
            Success      = $Success
            Text         = $Text
            Model        = $Model
            Stream       = $Stream
            DurationMs   = [int]$Duration.TotalMilliseconds
            StatusCode   = $StatusCode
            ReasonPhrase = $ReasonPhrase
            Exception    = $Exception
            ErrorRecord  = $ErrorRecord
        }
    }

    function Format-ExceptionChain([Exception]$ex) {
        $parts = New-Object System.Collections.Generic.List[string]
        $i = 0
        while ($ex) {
            $parts.Add(("[$i] {0}: {1}" -f $ex.GetType().FullName, $ex.Message))
            $ex = $ex.InnerException
            $i++
        }
        $parts -join " | "
    }

    function Test-PortSilent {
        param(
            [string]$Host,
            [int]$Port,
            [int]$TimeoutMs = 500
        )

        try {
            $client = New-Object System.Net.Sockets.TcpClient
            $async = $client.BeginConnect($Host, $Port, $null, $null)
            $wait = $async.AsyncWaitHandle.WaitOne($TimeoutMs, $false)

            if (-not $wait) {
                $client.Close()
                return $false
            }

            $client.EndConnect($async)
            $client.Close()
            return $true
        }
        catch {
            return $false
        }
    }

    # -----------------------------
    # Build request bits
    # -----------------------------
    $baseUrl = 'http://localhost:11434/api'
    $requestUri = [Uri]"$baseUrl/generate"

    $body = @{
        model  = $Model
        prompt = $Prompt
        stream = $Stream
    } | ConvertTo-Json -Depth 5

    $handler = $null
    $client = $null
    $response = $null
    $streamObj = $null
    $reader = $null

    $sw = [System.Diagnostics.Stopwatch]::StartNew()

    try {
        Write-Log -Level Warn -Message ("Invoking local LLM at '{0}' with model '{1}' (stream={2})..." -f $requestUri, $Model, $Stream)

        # Optional preflight: fail fast if Ollama isn't listening.
        # (Fast & clean when service is down; doesn't help with GPU driver crash, but avoids confusion.)
        try {
            $isUp = Test-PortSilent -Host 'localhost' -Port 11434

            if (-not $isUp) {
                $msg = "Ollama is not listening on localhost:11434. Start Ollama and try again."
                $err = [System.Management.Automation.ErrorRecord]::new(
                    [System.Exception]::new($msg),
                    'OllamaNotListening',
                    [System.Management.Automation.ErrorCategory]::ConnectionError,
                    $requestUri
                )

                $result = New-LLMResult -Success:$false -Text $null -Model $Model -Stream:$Stream -Duration $sw.Elapsed `
                    -StatusCode $null -ReasonPhrase $null -Exception $err.Exception -ErrorRecord $err

                if ($ThrowOnError) { throw $err.Exception }
                return $result
            }
        }
        catch {
            # If preflight itself fails, ignore and proceed to the real call.
        }

        $handler = [System.Net.Http.HttpClientHandler]::new()
        $client = [System.Net.Http.HttpClient]::new($handler)
        $client.Timeout = [TimeSpan]::FromMinutes($TimeoutMinutes)

        $attempt = 0
        $lastException = $null

        while ($attempt -le $RetryCount) {
            $attempt++

            # IMPORTANT: build a fresh request each attempt (don’t reuse HttpRequestMessage/StringContent).
            $request = [System.Net.Http.HttpRequestMessage]::new([System.Net.Http.HttpMethod]::Post, $requestUri)
            $request.Content = [System.Net.Http.StringContent]::new($body, [System.Text.Encoding]::UTF8, 'application/json')

            try {
                if ($Stream) {
                    $response = $client.SendAsync(
                        $request,
                        [System.Net.Http.HttpCompletionOption]::ResponseHeadersRead
                    ).GetAwaiter().GetResult()
                }
                else {
                    $response = $client.SendAsync($request).GetAwaiter().GetResult()
                }

                # request is safe to dispose after send
                $request.Dispose()
                $request = $null

                break
            }
            catch {
                $lastException = $_.Exception
                $msg = "Send attempt $attempt/$($RetryCount+1) failed: $(Format-ExceptionChain $lastException)"
                Write-Log -Level Warn -Message $msg

                if ($request) { $request.Dispose(); $request = $null }

                if ($attempt -le $RetryCount) {
                    Start-Sleep -Seconds $RetryDelaySeconds
                    continue
                }

                # Return structured failure (or throw, if requested)
                $err = [System.Management.Automation.ErrorRecord]::new(
                    $lastException,
                    'LocalLLMSendFailed',
                    [System.Management.Automation.ErrorCategory]::ConnectionError,
                    $requestUri
                )

                $result = New-LLMResult -Success:$false -Text $null -Model $Model -Stream:$Stream -Duration $sw.Elapsed `
                    -StatusCode $null -ReasonPhrase $null -Exception $lastException -ErrorRecord $err

                if ($ThrowOnError) { throw $lastException }
                return $result
            }
        }

        # Response sanity
        if ($null -eq $response) {
            $msg = "SendAsync returned null response (unexpected). If your GPU driver reset, Ollama may have crashed mid-request."
            $ex = [System.Exception]::new($msg)
            $err = [System.Management.Automation.ErrorRecord]::new(
                $ex,
                'NullHttpResponse',
                [System.Management.Automation.ErrorCategory]::InvalidResult,
                $requestUri
            )

            $result = New-LLMResult -Success:$false -Text $null -Model $Model -Stream:$Stream -Duration $sw.Elapsed `
                -StatusCode $null -ReasonPhrase $null -Exception $ex -ErrorRecord $err

            if ($ThrowOnError) { throw $ex }
            return $result
        }

        # Non-success HTTP -> return error object with body
        if (-not $response.IsSuccessStatusCode) {
            $bodyText = $response.Content.ReadAsStringAsync().GetAwaiter().GetResult()
            $msg = "HTTP $([int]$response.StatusCode) ($($response.ReasonPhrase)): $bodyText"
            $ex = [System.Exception]::new($msg)
            $err = [System.Management.Automation.ErrorRecord]::new(
                $ex,
                'OllamaHttpError',
                [System.Management.Automation.ErrorCategory]::InvalidOperation,
                $requestUri
            )

            $result = New-LLMResult -Success:$false -Text $null -Model $Model -Stream:$Stream -Duration $sw.Elapsed `
                -StatusCode ([int]$response.StatusCode) -ReasonPhrase $response.ReasonPhrase -Exception $ex -ErrorRecord $err

            if ($ThrowOnError) { throw $ex }
            return $result
        }

        # -----------------------------
        # NON-STREAMING
        # -----------------------------
        if (-not $Stream) {
            $json = $response.Content.ReadAsStringAsync().GetAwaiter().GetResult()
            if ([string]::IsNullOrWhiteSpace($json)) {
                $msg = "Local LLM returned an empty response body."
                $ex = [System.Exception]::new($msg)
                $err = [System.Management.Automation.ErrorRecord]::new(
                    $ex,
                    'EmptyBody',
                    [System.Management.Automation.ErrorCategory]::InvalidResult,
                    $requestUri
                )

                $result = New-LLMResult -Success:$false -Text $null -Model $Model -Stream:$Stream -Duration $sw.Elapsed `
                    -StatusCode ([int]$response.StatusCode) -ReasonPhrase $response.ReasonPhrase -Exception $ex -ErrorRecord $err

                if ($ThrowOnError) { throw $ex }
                return $result
            }

            try {
                $obj = $json | ConvertFrom-Json
                $text =
                if ($obj.response) { $obj.response }
                elseif ($obj.message) { $obj.message }
                else { $json }

                Write-Log -Level OK -Message ("Local LLM call completed for model '{0}'." -f $Model)
                return (New-LLMResult -Success:$true -Text $text -Model $Model -Stream:$Stream -Duration $sw.Elapsed `
                        -StatusCode ([int]$response.StatusCode) -ReasonPhrase $response.ReasonPhrase -Exception $null -ErrorRecord $null)
            }
            catch {
                # Return raw JSON if parsing fails, but still count as success
                Write-Log -Level OK -Message ("Local LLM call completed for model '{0}' (unparsed JSON)." -f $Model)
                return (New-LLMResult -Success:$true -Text $json -Model $Model -Stream:$Stream -Duration $sw.Elapsed `
                        -StatusCode ([int]$response.StatusCode) -ReasonPhrase $response.ReasonPhrase -Exception $null -ErrorRecord $null)
            }
        }

        # -----------------------------
        # STREAMING (JSONL)
        # -----------------------------
        $sb = [System.Text.StringBuilder]::new()

        $streamObj = $response.Content.ReadAsStreamAsync().GetAwaiter().GetResult()
        $reader = [System.IO.StreamReader]::new($streamObj)

        while (-not $reader.EndOfStream) {
            $line = $reader.ReadLine()
            if ([string]::IsNullOrWhiteSpace($line)) { continue }

            try {
                $obj = $line | ConvertFrom-Json
            }
            catch {
                Write-Log -Level Warn -Message ("Malformed JSON from LLM stream: {0}" -f $line)
                continue
            }

            if ($obj.error) {
                $ex = [System.Exception]::new("Ollama stream error: $($obj.error)")
                $err = [System.Management.Automation.ErrorRecord]::new(
                    $ex,
                    'OllamaStreamError',
                    [System.Management.Automation.ErrorCategory]::InvalidOperation,
                    $requestUri
                )

                $result = New-LLMResult -Success:$false -Text $null -Model $Model -Stream:$Stream -Duration $sw.Elapsed `
                    -StatusCode ([int]$response.StatusCode) -ReasonPhrase $response.ReasonPhrase -Exception $ex -ErrorRecord $err

                if ($ThrowOnError) { throw $ex }
                return $result
            }

            if ($obj.response) { [void]$sb.Append($obj.response) }
            if ($obj.done -eq $true) { break }
        }

        $text = $sb.ToString()
        if ([string]::IsNullOrWhiteSpace($text)) {
            $msg = "Streaming mode completed but produced no text."
            $ex = [System.Exception]::new($msg)
            $err = [System.Management.Automation.ErrorRecord]::new(
                $ex,
                'EmptyStream',
                [System.Management.Automation.ErrorCategory]::InvalidResult,
                $requestUri
            )

            $result = New-LLMResult -Success:$false -Text $null -Model $Model -Stream:$Stream -Duration $sw.Elapsed `
                -StatusCode ([int]$response.StatusCode) -ReasonPhrase $response.ReasonPhrase -Exception $ex -ErrorRecord $err

            if ($ThrowOnError) { throw $ex }
            return $result
        }

        Write-Log -Level OK -Message ("Local LLM call completed for model '{0}'." -f $Model)
        return (New-LLMResult -Success:$true -Text $text -Model $Model -Stream:$Stream -Duration $sw.Elapsed `
                -StatusCode ([int]$response.StatusCode) -ReasonPhrase $response.ReasonPhrase -Exception $null -ErrorRecord $null)
    }
    catch {
        # Last-chance catch: return structured result (unless ThrowOnError)
        $ex = $_.Exception
        Write-Log -Level Error -Message ("Error invoking local LLM: {0}" -f $ex.ToString())

        if ($ThrowOnError) { throw }

        $err = [System.Management.Automation.ErrorRecord]::new(
            $ex,
            'InvokeLocalLLMFailed',
            [System.Management.Automation.ErrorCategory]::NotSpecified,
            $requestUri
        )

        return (New-LLMResult -Success:$false -Text $null -Model $Model -Stream:$Stream -Duration $sw.Elapsed `
                -StatusCode $null -ReasonPhrase $null -Exception $ex -ErrorRecord $err)
    }
    finally {
        $sw.Stop()
        if ($reader) { $reader.Dispose() }
        if ($streamObj) { $streamObj.Dispose() }
        if ($response) { $response.Dispose() }
        if ($client) { $client.Dispose() }
        if ($handler) { $handler.Dispose() }
    }
}

# SIG # Begin signature block
# MIIfAgYJKoZIhvcNAQcCoIIe8zCCHu8CAQExDzANBglghkgBZQMEAgEFADB5Bgor
# BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG
# KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCBQ0d1qXeYaCDeZ
# hgK2U0AYhRWUkrIkINVXSvKRnIlEqaCCGEowggUMMIIC9KADAgECAhAR+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
# AQQBgjcCAQsxDjAMBgorBgEEAYI3AgEVMC8GCSqGSIb3DQEJBDEiBCA1goQrNwCj
# /g0038kVBcdAzcu+e29OwT6SHHndVDIWajANBgkqhkiG9w0BAQEFAASCAgDBKKlW
# f/w1vyR01f3MjTs8NSWg9w8ee96TQn+oSPrPOcakIB1pdKJEOqxcVce2GaFHij12
# uh7abU6WvaAav45AEIR91SlFHOZgpNkh06qW5cq3yvm8xXDHTrfX5HQ8YqwxygLJ
# XEYTc6kSIoKyM8eO3aLm/F8aU30WygsR/opAFjasCzE8JSD9h+PM/od4sMyK64C2
# aOSJSOCF7zTMU7+AdZ8an/avMq0+J9FQ1EmwCB65jJFgFSQuTX9X9/6GFfbmJKY4
# FyJIh5zhEG5NIvyiGACNJzxymhAPbXQGxF0Rf9IYaMRsjNuBWY9Et6yfogk0BCY2
# 7EHNf5ndkSZfRvRhVlKalOErmnourR1iCKi7gBfxRyDQDzOjYu2s4eYKYu/0jwMT
# 225euqGtmJJsmfj14CmjsSRxXfUDKe0gEaCQhN0Z0/g4d/Y9jXdbuPnAJA7rG2CB
# O8r7a+B7cfQjnAYjGri417ioBSlRszpyEPc6N7zd3O1scpRwsKVqVu1mWymqGz7B
# jdKe+GpdODom9zPEgBj7A/ff+8By8JhIHOCJi3pN9CQwcXcHoRBUF3vP+96sLHik
# OACBtrqYwlnKO40eWWKszKdp+Ccx1mxCTOtriv4bwOWiO4NiKeKirog6Z1kwxQcw
# jpl2feUc+KUIEapbOgtrsAnx4gqQFwJOd82Cx6GCAyYwggMiBgkqhkiG9w0BCQYx
# ggMTMIIDDwIBATB9MGkxCzAJBgNVBAYTAlVTMRcwFQYDVQQKEw5EaWdpQ2VydCwg
# SW5jLjFBMD8GA1UEAxM4RGlnaUNlcnQgVHJ1c3RlZCBHNCBUaW1lU3RhbXBpbmcg
# UlNBNDA5NiBTSEEyNTYgMjAyNSBDQTECEAqA7xhLjfEFgtHEdqeVdGgwDQYJYIZI
# AWUDBAIBBQCgaTAYBgkqhkiG9w0BCQMxCwYJKoZIhvcNAQcBMBwGCSqGSIb3DQEJ
# BTEPFw0yNjAzMDcwMjA4NDJaMC8GCSqGSIb3DQEJBDEiBCAjjsmHxO9U15eI9iu1
# iqAX6I6SC4OunE6gOXjfCjD/WDANBgkqhkiG9w0BAQEFAASCAgA7JFp/4nelhif5
# diMk5IGwAsw7BKVJwM9Pm3BT2XQRlm4cpXzj3q+Y7+6757MkHpNUzHKbhnfQ7PJY
# z86nG5XlWeM4IkVHTO+7RZbK5hifMh7gv6jVHTgxfyVvsj/3BizbCDeKFJSCpBMa
# 3cnCSNtn145mlpbDEgY2/D1uXiiThBRPT6eRyuoxtUZLK3cWqtVtQJ1ttuBLeZZQ
# GddGDTqVtm1nHq1mKvipYjJEM/u4w0PcszCHgIN6bR0jofkH04KXkR9STVaKhMea
# Vo11UStKnX3StqwYnpzxX+v+FVMmfIKlTbB5qo17wYmZY4HoI8GpbFKPH+A1tD3w
# u1N4s5zY4Z+3Wns8cxpi62lzuOs8AqxcyVJspAmAQf6ikVdibooAA4bhkFtieUxk
# jQS3MBN9/0Hdb2/wNEYPC78D1riFNwN/3s6TA94jBeoursNfkKYLMu7joJbhISZ9
# jOtAveglKLsZHygursGqtC7Ht0FTVZa5AVtyo+i1KcHh5v5zb514F5RFZpEk1JN6
# KZxJD0kweRaYRfl4fORwWNODykNbkqImW2q2PhvDytTIqxbluO0/CjnLHQkIwxr2
# VbZE3wfl1XdrqsN3qqVDNRvCQm62agqwkkQY2sXUSUj7gQCOE/fIbm+VXpkM+s39
# apE3QuJJ0FW52ZVCK940LVw5d2s7HA==
# SIG # End signature block