Public/AI/Invoke-TechAgent.ps1
|
function Invoke-TechAgent { <# .SYNOPSIS Sends a prompt to the TechToolbox local agent. .DESCRIPTION This function calls the TechToolbox.Agent C# runtime 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 C# agent hits an iteration 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 } } $transcriptStarted = $false $transcriptPath = $null $markdownPath = $null $markdownStatus = 'NotStarted' $markdownError = $null $capturedStdOut = '' $capturedStdErr = '' $runStartedUtc = [DateTime]::UtcNow $agentProc = $null $stdoutTask = $null $stderrTask = $null $requestPath = $null $resolveExpectedOutputPath = { param( [string]$PromptText ) if ([string]::IsNullOrWhiteSpace($PromptText)) { return $null } $fileNameMatch = [regex]::Match( $PromptText, '(?is)\bname\s+it\s+["'']?(?<name>[^"''`\r\n]+?\.help\.txt)\b') if (-not $fileNameMatch.Success) { return $null } $fileName = $fileNameMatch.Groups['name'].Value.Trim() if ([string]::IsNullOrWhiteSpace($fileName)) { return $null } $pathMatches = [regex]::Matches($PromptText, '(?i)[A-Za-z]:\\[^\s"''`\r\n]+') if ($pathMatches.Count -eq 0) { return $null } $candidateDirs = @() foreach ($match in $pathMatches) { $candidatePath = [string]$match.Value if ([string]::IsNullOrWhiteSpace($candidatePath)) { continue } $candidatePath = $candidatePath.Trim().TrimEnd('.', ',', ';') if ($candidatePath -match '(?i)\.[A-Za-z0-9]{1,5}$') { continue } $candidateDirs += $candidatePath } if ($candidateDirs.Count -eq 0) { return $null } $targetDirectory = $candidateDirs | Where-Object { $_ -match '(?i)\\en-US$' } | Select-Object -Last 1 if ([string]::IsNullOrWhiteSpace($targetDirectory)) { $targetDirectory = $candidateDirs | Select-Object -Last 1 } if ([string]::IsNullOrWhiteSpace($targetDirectory)) { return $null } return (Join-Path -Path $targetDirectory -ChildPath $fileName) } $expectedOutputPath = & $resolveExpectedOutputPath -PromptText $Prompt $effectivePrompt = $Prompt if (-not [string]::IsNullOrWhiteSpace($expectedOutputPath)) { $effectivePrompt = @" $Prompt Hard requirement: - Create the output file at this exact path: $expectedOutputPath - Use WRITE-FILE to create/update the file. - Do not return a final answer until WRITE-FILE has succeeded. "@ } 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 { $moduleRoot = Get-ModuleRoot $assemblyCandidates = @( (Join-Path $moduleRoot 'AgentRuntime\TechToolbox.Agent\TechToolbox.Agent.dll'), (Join-Path $moduleRoot 'src\TechToolbox.Agent\bin\Release\net8.0\publish\TechToolbox.Agent.dll'), (Join-Path $moduleRoot 'src\TechToolbox.Agent\bin\Release\net8.0\TechToolbox.Agent.dll'), (Join-Path $moduleRoot 'src\TechToolbox.Agent\bin\Debug\net8.0\TechToolbox.Agent.dll') ) $agentAssemblyPath = $assemblyCandidates | Where-Object { Test-Path -LiteralPath $_ -PathType Leaf } | Select-Object -First 1 if ([string]::IsNullOrWhiteSpace($agentAssemblyPath)) { throw "TechToolbox.Agent assembly not found. Install the packaged agent runtime or build/publish src\TechToolbox.Agent." } 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) } } if ($ConfirmDestructive.IsPresent) { Write-Log -Level Warn -Message 'Destructive operations explicitly authorized for this run.' } $autoRetryOnIterationLimit = $false if ($AutoRetryOnRecursion.IsPresent) { $autoRetryOnIterationLimit = $true } elseif ($DisableAutoRetryOnRecursion.IsPresent) { $autoRetryOnIterationLimit = $false } $memoryPath = $null $memoryPathProperty = if ($cfg) { $cfg.PSObject.Properties['memoryPath'] } else { $null } if ($null -ne $memoryPathProperty -and -not [string]::IsNullOrWhiteSpace([string]$memoryPathProperty.Value)) { $memoryPath = [string]$memoryPathProperty.Value } if (-not [string]::IsNullOrWhiteSpace($memoryPath)) { try { $memoryDirectory = Split-Path -Path $memoryPath -Parent if (-not [string]::IsNullOrWhiteSpace($memoryDirectory)) { $null = New-Item -ItemType Directory -Path $memoryDirectory -Force } if (-not (Test-Path -LiteralPath $memoryPath -PathType Leaf)) { $memorySeed = @{ Preferences = @{} Facts = @{} History = @() } | ConvertTo-Json -Depth 4 Set-Content -LiteralPath $memoryPath -Value $memorySeed -Encoding utf8 Write-Log -Level Info -Message ("Initialized missing agent memory file: {0}" -f $memoryPath) } $memoryHistoryPath = Join-Path $memoryDirectory (([System.IO.Path]::GetFileNameWithoutExtension($memoryPath)) + '.history.json') if (-not (Test-Path -LiteralPath $memoryHistoryPath -PathType Leaf)) { Set-Content -LiteralPath $memoryHistoryPath -Value '[]' -Encoding utf8 Write-Log -Level Info -Message ("Initialized missing agent memory history file: {0}" -f $memoryHistoryPath) } } catch { throw ("Failed to initialize agent memory files at '{0}': {1}" -f $memoryPath, $_.Exception.Message) } } $diagnosticTracePath = $null $diagnosticTracePathProperty = if ($cfg) { $cfg.PSObject.Properties['diagnosticTracePath'] } else { $null } if ($null -ne $diagnosticTracePathProperty -and -not [string]::IsNullOrWhiteSpace([string]$diagnosticTracePathProperty.Value)) { $diagnosticTracePath = [string]$diagnosticTracePathProperty.Value } Write-Log -Level Info -Message ("Invoking local tech agent via C# assembly: {0}" -f $agentAssemblyPath) $request = [ordered]@{ Prompt = $effectivePrompt Model = $(if ([string]::IsNullOrWhiteSpace($Model)) { 'llama3' } else { $Model }) Verbose = $false MaxIterations = $MaxIterations ConfirmDestructive = $ConfirmDestructive.IsPresent MemoryPath = $memoryPath AutoRetryOnRecursion = $autoRetryOnIterationLimit ReturnMetadata = $false SignedFilePolicy = $(if ([string]::IsNullOrWhiteSpace($SignedFilePolicy)) { 'ignore' } else { $SignedFilePolicy }) DiagnosticTracePath = $diagnosticTracePath ExpectedOutputPath = $expectedOutputPath } $requestPath = Join-Path ([System.IO.Path]::GetTempPath()) ("techtoolbox-agent-request-{0}.json" -f ([guid]::NewGuid().ToString('N'))) $request | ConvertTo-Json -Depth 6 | Set-Content -LiteralPath $requestPath -Encoding utf8 $childPwsh = Join-Path $PSHOME 'pwsh.exe' if (-not (Test-Path -LiteralPath $childPwsh -PathType Leaf)) { $childPwsh = (Get-Process -Id $PID).Path } $childScript = @' $ErrorActionPreference = 'Stop' [Console]::OutputEncoding = [System.Text.UTF8Encoding]::new($false) $request = Get-Content -LiteralPath $env:TT_AGENT_REQUEST_PATH -Raw | ConvertFrom-Json Add-Type -Path $env:TT_AGENT_ASSEMBLY_PATH -ErrorAction Stop $result = [TechToolbox.Agent.Agent.AgentCore]::RunAgent( [string]$request.Prompt, [string]$request.Model, [bool]$request.Verbose, [int]$request.MaxIterations, [bool]$request.ConfirmDestructive, [string]$request.MemoryPath, [bool]$request.AutoRetryOnRecursion, [bool]$request.ReturnMetadata, [string]$request.SignedFilePolicy, [string]$request.DiagnosticTracePath, [string]$request.ExpectedOutputPath) [Console]::Write($result) '@ $encodedChildScript = [Convert]::ToBase64String([System.Text.Encoding]::Unicode.GetBytes($childScript)) $startInfo = [System.Diagnostics.ProcessStartInfo]::new() $startInfo.FileName = $childPwsh $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['TT_AGENT_ASSEMBLY_PATH'] = $agentAssemblyPath $startInfo.Environment['TT_AGENT_REQUEST_PATH'] = $requestPath [void]$startInfo.ArgumentList.Add('-NoProfile') [void]$startInfo.ArgumentList.Add('-NonInteractive') [void]$startInfo.ArgumentList.Add('-EncodedCommand') [void]$startInfo.ArgumentList.Add($encodedChildScript) try { $agentProc = [System.Diagnostics.Process]::new() $agentProc.StartInfo = $startInfo if (-not $agentProc.Start()) { throw 'Failed to start child PowerShell process for TechToolbox.Agent.' } $stdoutTask = $agentProc.StandardOutput.ReadToEndAsync() $stderrTask = $agentProc.StandardError.ReadToEndAsync() if (-not $agentProc.WaitForExit($waitTimeoutSeconds * 1000)) { try { $agentProc.Kill() } catch { } throw ("Tech agent timed out after {0} seconds." -f $waitTimeoutSeconds) } try { $null = $agentProc.WaitForExit(2000) } catch { } $capturedStdOut = if ($stdoutTask) { [string]$stdoutTask.GetAwaiter().GetResult() } else { '' } $capturedStdErr = if ($stderrTask) { [string]$stderrTask.GetAwaiter().GetResult() } else { '' } if ($agentProc.ExitCode -ne 0) { $errorText = if ([string]::IsNullOrWhiteSpace($capturedStdErr)) { $capturedStdOut } else { $capturedStdErr } throw ("Tech agent exited with code {0}: {1}" -f $agentProc.ExitCode, $errorText.Trim()) } $message = $capturedStdOut } catch { throw ("Tech agent failed: {0}" -f $_.Exception.Message) } $message = ([string]$message).Trim() $capturedStdOut = $message if ([string]::IsNullOrWhiteSpace($message)) { $message = 'Tech agent completed successfully with no output.' } # Surface orchestrator-level failures as real failures so markdown status # and caller behavior do not report false positives. $knownFailurePrefixes = @( 'Agent returned invalid JSON twice.', 'LLM request repeatedly failed', 'Iteration limit reached.' ) foreach ($failurePrefix in $knownFailurePrefixes) { if ($message.StartsWith($failurePrefix, [System.StringComparison]::OrdinalIgnoreCase)) { throw ("Tech agent failed: {0}" -f $message) } } if (-not [string]::IsNullOrWhiteSpace($expectedOutputPath)) { if (-not (Test-Path -LiteralPath $expectedOutputPath -PathType Leaf)) { throw ("Tech agent failed: expected output file was not created: {0}" -f $expectedOutputPath) } Write-Log -Level Info -Message ("Tech agent created expected output file: {0}" -f $expectedOutputPath) } $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 ($markdownStatus -eq 'Success') { 0 } 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 (-not [string]::IsNullOrWhiteSpace($requestPath) -and (Test-Path -LiteralPath $requestPath -PathType Leaf)) { try { Remove-Item -LiteralPath $requestPath -Force } catch { } } if ($agentProc) { try { $agentProc.Dispose() } catch { } } } } # SIG # Begin signature block # MIIfAgYJKoZIhvcNAQcCoIIe8zCCHu8CAQExDzANBglghkgBZQMEAgEFADB5Bgor # BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCA1jWWWM1Uea7En # CgEgmt9ECDc1WOM7n+F14pyIUzCsbKCCGEowggUMMIIC9KADAgECAhAR+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 # AQQBgjcCAQsxDjAMBgorBgEEAYI3AgEVMC8GCSqGSIb3DQEJBDEiBCB5K3tpqLNH # FHiBK4XdItdrDc2HVwVibVyNwSgnFJJy+TANBgkqhkiG9w0BAQEFAASCAgCTh2aI # HI3CYvV6tlwvNzKpO8OBwlpaX91UFHeZ856064jwWtU8hOQl8dsaLxGLSNHIwYY4 # afPnq4ztKsruePs6ftF8Gh5WDV7UX7yiRSJzQATwlX8WTehI3mAPJofnVmna/GPb # /aFdpU5Y82cZH/+eDLJLtZxYi5L1E9ykMsZuBDDeDU2ykztUwiAs21JpoEHzEE6J # Dyzcl2B5Ln6DOR8YvXf+rtI8ysGfuYh48vOMmMwQToklWeECLlytxyW+32zzlUn8 # dsyWmSob6/W2WDF5jJn63ESH/RCVABcdCcHSg5rcaR1c4Ra6GYYtbg/8B4nyJH1i # dYLedPihfRUzwXLC09V6cYu9uDfh9CmhosO42r6+cB081tCK2TSTrB/6bSXdsQvH # 78nFpRV2bvWO7lOY1YwafxSFtLBs8QLrGureZoWCw4UK6TVSSRokQrjqTVmzLzoL # Jsr9QUXvC+2P9X6gfAwpdNDLs/iGHy/oyb62SVY5e48fexz6h08ye5Uicectnj8t # Jty1Q1RaJFjRtWABKXATBgpHtyDjkziJ0wMOXul9ytE6K4FcZQjSQHkP1lhjBHmh # HWijSLZy4YgGCBy1JpX+4P5dI2uRXE+zetcA+YKDHcwZztSPb3Ydrl/u7EDp2eY2 # nr+ml6LdtTaKK8lCIN1IV7X5QH31BxgrTiUC7aGCAyYwggMiBgkqhkiG9w0BCQYx # ggMTMIIDDwIBATB9MGkxCzAJBgNVBAYTAlVTMRcwFQYDVQQKEw5EaWdpQ2VydCwg # SW5jLjFBMD8GA1UEAxM4RGlnaUNlcnQgVHJ1c3RlZCBHNCBUaW1lU3RhbXBpbmcg # UlNBNDA5NiBTSEEyNTYgMjAyNSBDQTECEAqA7xhLjfEFgtHEdqeVdGgwDQYJYIZI # AWUDBAIBBQCgaTAYBgkqhkiG9w0BCQMxCwYJKoZIhvcNAQcBMBwGCSqGSIb3DQEJ # BTEPFw0yNjA2MjEwNDEwNDhaMC8GCSqGSIb3DQEJBDEiBCA8ICKd9tBL4550oi0T # yK1YaYHp4SKth7hevAsAxXMWBjANBgkqhkiG9w0BAQEFAASCAgDJCgCvKNRUjwb6 # Wk03mnVxDHVrvgleMuwAod2lBPaDFayy3DjtiCUNwszmxIVoZkNG3VI9chZZmG36 # d2y3JXB/PDAaFtDJ43tYlg73JFtxC9z83mOULPIaSC3jxpzWXvSSnm6dnUkMdcaP # mP5PPT2BBmEbqURpN4m4XYTwb3HuFkNzbtUmgAD9OAEhvLYCyM6/zi/zt3jRL5TM # FTweyPlSb37jFvEmEf+75bHkG8GdvTO+wb7oppuenmr1MstmnMqbmU/LFpAiNhkj # H77npIjot2RiOvtsMjrLOygWVngeFojUqAt87kML9MSYcenpvW+7S4Dg8GiXTKF1 # Ay8rq8cTO1eRex2Dw/W/nTwNo2fXmoRylogZaY+BssqsYtJ8GeneXDbEyJ4i+yTK # xeDTbTj82kQNjURmsOjCiSb3Ogbi4/ouFYes55z1u4doYnRc0pkwiA0gjDl8pkBR # yUlP15SiJre76WnAb4fc3Knu5119tUDXGkGMybcaea/Uy8xhgYN++Sr85PRskwfi # ebRx2hDIlhdi6YQA0D+nrYK75qZzfj57lDZf1urV/ysFAPt3DusnsQWDVTIZEjG4 # p0NnhLuWxyvQSS0OROe0h2QnaFoYQ59Fd8koHgcVR7e6oCa0JzjgHl40hDjjmo5b # CUlvj2sAqNt86N7NHgCvmapNiQessQ== # SIG # End signature block |