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 |