Public/AI/Invoke-TechAgent.ps1
|
function Invoke-TechAgent { <# .SYNOPSIS Sends a prompt to the TechToolbox local agent. .DESCRIPTION This function calls the Python-based agent located in AI/Agent/. It passes the user prompt and prints the agent's response. .PARAMETER Prompt The natural-language instruction for the agent. .PARAMETER Model Optional Ollama model name (for example: llama3, mistral, qwen2.5-coder). .PARAMETER MaxIterations Maximum number of tool/reasoning iterations before the agent concludes. .PARAMETER Quiet Legacy compatibility switch. Agent traces are now suppressed by default. .PARAMETER ConfirmDestructive Explicitly authorizes destructive operations for this run. .PARAMETER SignedFilePolicy Policy to use when overwriting an existing Authenticode-signed PowerShell file. 'ignore' blocks the overwrite and 'strip' allows the overwrite while removing the signature block text. .PARAMETER AutoRetryOnRecursion Enables a single automatic retry when the Python agent hits a recursion limit. .PARAMETER DisableAutoRetryOnRecursion Disables recursion-limit auto-retry for this invocation, overriding environment defaults. .PARAMETER NoTranscript Disables the per-run console transcript log. .EXAMPLE Invoke-TechAgent "Run system diagnostics and summarize findings." #> [CmdletBinding()] param( [Parameter(Mandatory)] [ValidateNotNullOrEmpty()] [string]$Prompt, [Parameter()] [ValidateNotNullOrEmpty()] [string]$Model, [Parameter()] [ValidateRange(1, 500)] [int]$MaxIterations = 15, [Parameter()] [switch]$Quiet, [Parameter()] [switch]$ConfirmDestructive , [Parameter()] [ValidateSet('ignore', 'strip')] [string]$SignedFilePolicy, [Parameter()] [switch]$AutoRetryOnRecursion , [Parameter()] [switch]$DisableAutoRetryOnRecursion , [Parameter()] [bool]$NoTranscript = $true ) # Initialize the TechToolbox runtime and load agent configuration Initialize-TechToolboxRuntime $cfg = $script:cfg.settings.agent if ([string]::IsNullOrWhiteSpace($Model) -and $cfg -and -not [string]::IsNullOrWhiteSpace($cfg.model)) { $Model = $cfg.model } $waitTimeoutSeconds = [Math]::Max(300, ($MaxIterations * 180)) $waitPollSeconds = 15 $waitHeartbeatSeconds = 120 if ($cfg -and $cfg.wait) { $timeoutCfg = $cfg.wait.timeoutSeconds -as [int] if ($null -ne $timeoutCfg -and $timeoutCfg -gt 0) { $waitTimeoutSeconds = $timeoutCfg } $pollCfg = $cfg.wait.pollSeconds -as [int] if ($null -ne $pollCfg -and $pollCfg -gt 0) { $waitPollSeconds = $pollCfg } $heartbeatCfg = $cfg.wait.heartbeatSeconds -as [int] if ($null -ne $heartbeatCfg -and $heartbeatCfg -ge 0) { $waitHeartbeatSeconds = $heartbeatCfg } } $agentProc = $null $stdoutTask = $null $stderrTask = $null $transcriptStarted = $false $transcriptPath = $null $markdownPath = $null $markdownStatus = 'NotStarted' $markdownError = $null $capturedStdOut = '' $capturedStdErr = '' $runStartedUtc = [DateTime]::UtcNow if ($AutoRetryOnRecursion.IsPresent -and $DisableAutoRetryOnRecursion.IsPresent) { throw 'Specify only one of -AutoRetryOnRecursion or -DisableAutoRetryOnRecursion.' } $writeMarkdownLog = { param( [string]$Path, [string]$Status, [string]$PromptText, [string]$ModelName, [int]$IterationLimit, [bool]$DestructiveAuthorized, [string]$SignedFilePolicyValue, [string]$AutoRetryOnRecursionMode, [string]$StdOut, [string]$StdErr, [string]$ErrorText, [int]$ExitCode, [string]$TranscriptFile, [DateTime]$StartedUtc, [DateTime]$CompletedUtc ) if ([string]::IsNullOrWhiteSpace($Path)) { return } $dir = Split-Path -Parent $Path if (-not [string]::IsNullOrWhiteSpace($dir)) { $null = New-Item -ItemType Directory -Path $dir -Force } $renderedOutput = if ([string]::IsNullOrWhiteSpace($StdOut)) { '(none)' } else { $StdOut.TrimEnd() } $rawError = if ([string]::IsNullOrWhiteSpace($StdErr)) { '(none)' } else { $StdErr.TrimEnd() } $rawException = if ([string]::IsNullOrWhiteSpace($ErrorText)) { '(none)' } else { $ErrorText.TrimEnd() } $lines = @( '# Tech Agent Run' '' ('- Status: {0}' -f $Status) ('- StartedUtc: {0}' -f $StartedUtc.ToString('o')) ('- CompletedUtc: {0}' -f $CompletedUtc.ToString('o')) ('- Model: {0}' -f $(if ([string]::IsNullOrWhiteSpace($ModelName)) { '(default)' } else { $ModelName })) ('- MaxIterations: {0}' -f $IterationLimit) ('- ConfirmDestructive: {0}' -f $DestructiveAuthorized) ('- SignedFilePolicy: {0}' -f $(if ([string]::IsNullOrWhiteSpace($SignedFilePolicyValue)) { '(default)' } else { $SignedFilePolicyValue })) ('- AutoRetryOnRecursion: {0}' -f $AutoRetryOnRecursionMode) ('- ExitCode: {0}' -f $ExitCode) ('- TranscriptPath: {0}' -f $(if ([string]::IsNullOrWhiteSpace($TranscriptFile)) { '(none)' } else { $TranscriptFile })) '' '## Prompt' '' '```text' $PromptText '```' '' '## Output' '' $renderedOutput '' '## Error Output' '' '~~~~text' $rawError '~~~~' '' '## Exception' '' '~~~~text' $rawException '~~~~' ) Set-Content -Path $Path -Value ($lines -join [Environment]::NewLine) -Encoding utf8BOM } try { # Resolve agent path from module root (Public/AI is not the module root). $moduleRoot = Get-ModuleRoot $agentPath = Join-Path $moduleRoot 'AI\Agent\tech_agent.py' if (-not (Test-Path -LiteralPath $agentPath -PathType Leaf)) { throw "Tech agent entry script not found: $agentPath" } $pythonCommand = $null $pythonArgsPrefix = @() # Prefer repo-local virtual environment for deterministic dependencies. $venvPython = Join-Path $moduleRoot '.venv\Scripts\python.exe' if (Test-Path -LiteralPath $venvPython -PathType Leaf) { $pythonCommand = @{ Source = $venvPython } } if (-not $pythonCommand) { $pythonCommand = Get-Command -Name python -ErrorAction SilentlyContinue } if (-not $pythonCommand) { $pythonCommand = Get-Command -Name py -ErrorAction SilentlyContinue if ($pythonCommand) { $pythonArgsPrefix = @('-3') } } if (-not $pythonCommand) { throw "Python executable not found. Install Python or add it to PATH (python/py)." } if (-not [string]::IsNullOrWhiteSpace($Model)) { $ollamaCommand = Get-Command -Name ollama -ErrorAction SilentlyContinue if (-not $ollamaCommand) { throw "Ollama executable not found. Install Ollama or add it to PATH." } $ollamaListOutput = & $ollamaCommand.Source list 2>&1 if ($LASTEXITCODE -ne 0) { $ollamaError = ($ollamaListOutput | Out-String).Trim() throw ("Unable to query local Ollama models: {0}" -f $ollamaError) } $availableModels = @() foreach ($line in $ollamaListOutput) { $trimmed = "$line".Trim() if ([string]::IsNullOrWhiteSpace($trimmed)) { continue } if ($trimmed -match '^NAME\s+') { continue } $parts = $trimmed -split '\s+' if ($parts.Count -gt 0 -and -not [string]::IsNullOrWhiteSpace($parts[0])) { $availableModels += $parts[0] } } if (-not $availableModels) { throw ("No local Ollama models were found. Pull the requested model first: ollama pull {0}" -f $Model) } if ($availableModels -notcontains $Model) { $knownModels = ($availableModels | Sort-Object -Unique) -join ', ' throw ( "Ollama model '{0}' is not available locally. Run: ollama pull {0}. Available models: {1}" -f $Model, $knownModels ) } } $transcriptEnabled = $true $transcriptRoot = $null $markdownEnabled = $true $markdownRoot = $null if ($cfg -and $cfg.transcript) { if ($null -ne $cfg.transcript.enabled) { $transcriptEnabled = [bool]$cfg.transcript.enabled } if (-not [string]::IsNullOrWhiteSpace([string]$cfg.transcript.outputRoot)) { $transcriptRoot = [string]$cfg.transcript.outputRoot } $markdownEnabledProperty = $cfg.transcript.PSObject.Properties['markdownEnabled'] if ($null -ne $markdownEnabledProperty -and $null -ne $markdownEnabledProperty.Value) { $markdownEnabled = [bool]$markdownEnabledProperty.Value } $markdownOutputRootProperty = $cfg.transcript.PSObject.Properties['markdownOutputRoot'] if ($null -ne $markdownOutputRootProperty -and -not [string]::IsNullOrWhiteSpace([string]$markdownOutputRootProperty.Value)) { $markdownRoot = [string]$markdownOutputRootProperty.Value } } if ($NoTranscript) { $transcriptEnabled = $false } if ($transcriptEnabled) { if ([string]::IsNullOrWhiteSpace($transcriptRoot)) { $transcriptRoot = Join-Path $moduleRoot 'LogsAndExports\Logs\TechAgentTranscripts' } try { $null = New-Item -ItemType Directory -Path $transcriptRoot -Force $transcriptPath = Join-Path $transcriptRoot ("TechAgent_{0}_{1}.txt" -f (Get-Date -Format 'yyyyMMdd_HHmmss'), $PID) Start-Transcript -Path $transcriptPath -Force | Out-Null $transcriptStarted = $true Write-Log -Level Info -Message ("Tech agent transcript started: {0}" -f $transcriptPath) Write-Host ("Tech agent transcript: {0}" -f $transcriptPath) } catch { Write-Log -Level Warn -Message ("Tech agent transcript could not be started: {0}" -f $_.Exception.Message) } } if ($markdownEnabled) { if ([string]::IsNullOrWhiteSpace($markdownRoot)) { $markdownRoot = Join-Path $moduleRoot 'LogsAndExports\Logs\TechAgentMarkdown' } try { $null = New-Item -ItemType Directory -Path $markdownRoot -Force $markdownPath = Join-Path $markdownRoot ("TechAgent_{0}_{1}.md" -f (Get-Date -Format 'yyyyMMdd_HHmmss'), $PID) Write-Host ("Tech agent markdown log: {0}" -f $markdownPath) } catch { $markdownPath = $null Write-Log -Level Warn -Message ("Tech agent markdown log could not be initialized: {0}" -f $_.Exception.Message) } } Write-Log -Level Info -Message ("Invoking local tech agent: {0}" -f $agentPath) $pythonArgs = @() $pythonArgs += $pythonArgsPrefix $pythonArgs += @($agentPath, '--prompt', $Prompt, '--max-iterations', $MaxIterations) if (-not [string]::IsNullOrWhiteSpace($Model)) { $pythonArgs += @('--model', $Model) } # Suppress LangChain/LangGraph debug traces by default so the console only shows # the agent's final response instead of raw model metadata. $pythonArgs += '--quiet' if ($ConfirmDestructive.IsPresent) { Write-Log -Level Warn -Message 'Destructive operations explicitly authorized for this run.' $pythonArgs += '--destructive-confirmed' } if (-not [string]::IsNullOrWhiteSpace($SignedFilePolicy)) { $pythonArgs += @('--signed-file-policy', $SignedFilePolicy) } if ($AutoRetryOnRecursion.IsPresent) { $pythonArgs += '--auto-retry-on-recursion' } elseif ($DisableAutoRetryOnRecursion.IsPresent) { $pythonArgs += '--no-auto-retry-on-recursion' } $startInfo = [System.Diagnostics.ProcessStartInfo]::new() $startInfo.FileName = [string]$pythonCommand.Source $startInfo.UseShellExecute = $false $startInfo.CreateNoWindow = $true $startInfo.RedirectStandardOutput = $true $startInfo.RedirectStandardError = $true $startInfo.StandardOutputEncoding = [System.Text.UTF8Encoding]::new($false) $startInfo.StandardErrorEncoding = [System.Text.UTF8Encoding]::new($false) $startInfo.Environment['PYTHONIOENCODING'] = 'utf-8' $startInfo.Environment['PYTHONUTF8'] = '1' foreach ($arg in $pythonArgs) { [void]$startInfo.ArgumentList.Add([string]$arg) } $agentProc = [System.Diagnostics.Process]::new() $agentProc.StartInfo = $startInfo if (-not $agentProc.Start()) { throw "Failed to start Python process for tech agent." } # Use async stream readers to avoid redirected-pipe deadlocks without using # PowerShell event handlers on background threads (which require a runspace). $stdoutTask = $agentProc.StandardOutput.ReadToEndAsync() $stderrTask = $agentProc.StandardError.ReadToEndAsync() $agentDeadline = (Get-Date).AddSeconds($waitTimeoutSeconds) $agentState = [ordered]@{ TimedOut = $false } $poll = { if (-not $agentProc.HasExited) { if ((Get-Date) -ge $agentDeadline) { $agentState.TimedOut = $true try { $agentProc.Kill() } catch { } return @{ Status = 'Timeout' } } return @{ Status = 'Running' } } # Ensure process and async stream readers have fully completed. try { $null = $agentProc.WaitForExit(2000) } catch { } $stdoutText = '' $stderrText = '' try { if ($stdoutTask) { $stdoutText = [string]$stdoutTask.GetAwaiter().GetResult() } } catch { } try { if ($stderrTask) { $stderrText = [string]$stderrTask.GetAwaiter().GetResult() } } catch { } return @{ Status = 'Done' Code = [int]$agentProc.ExitCode StdOut = $stdoutText StdErr = $stderrText } } $getStatus = { param($obj) switch ($obj.Status) { 'Timeout' { return 'Timeout' } 'Done' { if ($obj.Code -eq 0) { return 'Success' } return 'Error' } default { return 'Running' } } } $terminal = @{ 'Success' = @{ Level = 'Ok'; Message = 'Tech agent completed successfully.'; Return = $true } 'Error' = @{ Level = 'Error'; Message = { param($obj, $status) "Tech agent failed with exit code $($obj.Code)." }; Return = $true } 'Timeout' = @{ Level = 'Error'; Message = "Tech agent timed out after ${waitTimeoutSeconds} seconds."; Return = $true } } $final = Wait-TerminalState ` -Target 'TechAgent' ` -PollScript $poll ` -GetStatus $getStatus ` -TerminalStates $terminal ` -TimeoutSeconds ($waitTimeoutSeconds + 60) ` -PollSeconds $waitPollSeconds ` -HeartbeatSeconds $waitHeartbeatSeconds ` -NotFoundMessage 'Tech agent process not discovered yet...' ` -WaitingMessage 'Tech agent working ' if ($agentState.TimedOut) { throw ("Tech agent timed out after {0} seconds." -f $waitTimeoutSeconds) } if ([int]$final.Code -ne 0) { $errorText = [string]$final.StdErr if ([string]::IsNullOrWhiteSpace($errorText)) { $errorText = [string]$final.StdOut } $capturedStdOut = [string]$final.StdOut $capturedStdErr = [string]$final.StdErr $errorText = $errorText.Trim() throw ("Tech agent exited with code {0}: {1}" -f $final.Code, $errorText) } $message = ([string]$final.StdOut).Trim() $capturedStdOut = [string]$final.StdOut $capturedStdErr = [string]$final.StdErr if ([string]::IsNullOrWhiteSpace($message)) { $message = 'Tech agent completed successfully with no output.' } $markdownStatus = 'Success' return $message } catch { $markdownStatus = 'Error' $markdownError = $_.Exception.Message Write-Log -Level Error -Message ("Invoke-TechAgent failed: {0}" -f $_.Exception.Message) throw } finally { if (-not [string]::IsNullOrWhiteSpace($markdownPath)) { try { $exitCode = if ($agentProc -and $agentProc.HasExited) { [int]$agentProc.ExitCode } else { -1 } & $writeMarkdownLog ` -Path $markdownPath ` -Status $markdownStatus ` -PromptText $Prompt ` -ModelName $Model ` -IterationLimit $MaxIterations ` -DestructiveAuthorized $ConfirmDestructive.IsPresent ` -SignedFilePolicyValue $SignedFilePolicy ` -AutoRetryOnRecursionMode $( if ($AutoRetryOnRecursion.IsPresent) { 'Enabled' } elseif ($DisableAutoRetryOnRecursion.IsPresent) { 'Disabled' } else { 'Default' } ) ` -StdOut $capturedStdOut ` -StdErr $capturedStdErr ` -ErrorText $markdownError ` -ExitCode $exitCode ` -TranscriptFile $transcriptPath ` -StartedUtc $runStartedUtc ` -CompletedUtc ([DateTime]::UtcNow) } catch { Write-Log -Level Warn -Message ("Tech agent markdown log could not be written: {0}" -f $_.Exception.Message) } } if ($transcriptStarted) { try { Stop-Transcript | Out-Null } catch { } } if ($agentProc -and -not $agentProc.HasExited) { try { $agentProc.Kill() } catch { } } if ($agentProc) { try { $agentProc.Dispose() } catch { } } } } # SIG # Begin signature block # MIIfAgYJKoZIhvcNAQcCoIIe8zCCHu8CAQExDzANBglghkgBZQMEAgEFADB5Bgor # BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCCqxT+MBUv3hBur # QxxYEIe9wDXXSh4kgBqR8aUMZ6g/i6CCGEowggUMMIIC9KADAgECAhAR+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 # AQQBgjcCAQsxDjAMBgorBgEEAYI3AgEVMC8GCSqGSIb3DQEJBDEiBCARGYcR0mFE # OsqcepQ583k9+SIy8h3lH//RVtpN0mpnpjANBgkqhkiG9w0BAQEFAASCAgCV13gB # mW9r+A24GN8oCq2EHJQ0btC3ggXQMHWJ7ZH2QbF3CWgDU2N0QoWP27LzIlBBXxEd # Cjjks9uaji5DtV4d8IeZedMEC0u5bztFMWGyGKToLbjW6MNkmGfvaYrWkH+MFh3W # OR9XcX0p/FXRPkvOH0bVhhCyv77PX4FBa4qMtx9ykkbAq4Gjm3vT9VABbuovWlJC # PCqCQUc1A3sf4bKkwYQLq1gC0qxH3HPgRgcCan20hqjCQSJOE9tv/zka9RIKjQbJ # 8z9LuOzpm8o07jUxRARkr2F8nclt7PmHdypizWUZ9yD5GROiKEKituUx9nUBHc9s # 1Lr1l4QSQ9RfjkJapIVZIePlC4pCLFoGAgJRiQaI3JLNQioYaos425+gSZihhj3Q # j0vQqHlPGXn+F6yYA3CqDhtKMvi/Zot2nRJxISqq+ysuy1tbz9ShTmh5kj5C7ivp # XX31w7TlWtoGxEkE7h1JIa/L9fc4PQnlKsRW878yZWk7qn5cgjujaRza+s20JAaR # T2YQFwthRePTw28H3xqW/SOmts5LNcL0xnEpX1/5/NXEFM+UQnqFiWApEyq0q9tn # KseFtA5se3ze6RE5Fb2pNbcMExba2JiXSxno3K6E7PwVo67rceBAkqo7gXMink2k # G917LPbcuYYyuSSrORKg9H75WEYZxW9XTdJqHqGCAyYwggMiBgkqhkiG9w0BCQYx # ggMTMIIDDwIBATB9MGkxCzAJBgNVBAYTAlVTMRcwFQYDVQQKEw5EaWdpQ2VydCwg # SW5jLjFBMD8GA1UEAxM4RGlnaUNlcnQgVHJ1c3RlZCBHNCBUaW1lU3RhbXBpbmcg # UlNBNDA5NiBTSEEyNTYgMjAyNSBDQTECEAqA7xhLjfEFgtHEdqeVdGgwDQYJYIZI # AWUDBAIBBQCgaTAYBgkqhkiG9w0BCQMxCwYJKoZIhvcNAQcBMBwGCSqGSIb3DQEJ # BTEPFw0yNjA2MTMwMzMzMjRaMC8GCSqGSIb3DQEJBDEiBCCF9Bh6rGGMWYlIbdCA # 0KjRvMGiNoOOxPP6TCl3zTc6IDANBgkqhkiG9w0BAQEFAASCAgBra3vEWTNntEvc # NCPPc/SWdEUlqZi1U/6uqdxgHacJJe07P4/H9dsGKlq6te51A/R6FPpjRbYObkpE # GbF4AK+mmreHC1oOM3IhMcP76BVajpk5LNfECwZp5Sq0t4Tf6JXszwYfekQ3A4rj # xMXz1pk3VXWSvtZymr6zJzZVk+WiOskw1SYi0hnm51I1j/jOaXzd0S/C/nyMUvmu # wWRcCsevuySdknhKlKbupzzGzmVP0WDpjtQ39Z8tVxNI9BY3UNYAPsBiVwpqS6Bz # C3slx8GCmXInrFtnWA+fXBJrUSD6H2/vhrfG/ceEgRX93Ot5SYLyfN9sqhagvWsl # DACPHpnQdJSkedwNEaTlOpBPxe3RmeVoRllSv2pITxgRXMKuuoTNCEt+ZoILO2U3 # rXuwF/AaHZpSvGjKQRSJ5P4TJf10m6Pz6GIx+5fMcjsk1+Hwwjj7+7UtlaxIIEb9 # m+M3SNCbmbfrElIfPfDWtisLDW44zwLjbuNlkYl9wLBS3LtrAQMksFhEL3DW1tYD # AYEKpLtL5BOs43J3VbVLlr1VnmlYkgSwVH31TrG77Hvbu45LPLomFZGbnDzf9yIG # MvJL8/AAkCb9sRIBeFu5Tky5U4CbztL6FCcZTeh5Ocp6Rvt4BCHapT76+q2Duxi6 # JknuRz1l9MGQ8LTMHMMJxEhRoT/3mg== # SIG # End signature block |