scripts/commit-bug-review.ps1
|
#!/usr/bin/env pwsh param( [int]$MaxIterations = 148, [ValidateSet('claude', 'copilot')] [string]$Type = 'copilot', [string]$Model = 'gpt-5.4', [ValidateSet('low', 'medium', 'high', 'max', 'xhigh')] [string]$Effort = 'max', [switch]$DryRun ) $ErrorActionPreference = 'Stop' $RepoRoot = Split-Path $PSScriptRoot -Parent $TrackerPath = Join-Path $RepoRoot "docs\trackers\features\commit-bug-review-TRACKER.md" $PromptPath = Join-Path $RepoRoot "scripts\commit-bug-review-prompt.md" # Resolve engine binary if ($Type -eq 'copilot') { # Try common Copilot CLI locations on Windows $candidatePaths = @( "copilot", "gh copilot", "$env:LOCALAPPDATA\Programs\copilot\copilot.exe", "/home/vscode/.local/bin/copilot" ) $EngineBin = "copilot" foreach ($candidate in $candidatePaths) { if (Get-Command $candidate -ErrorAction SilentlyContinue) { $EngineBin = $candidate break } } Write-Host "Using Copilot CLI: $EngineBin" -ForegroundColor DarkGray } else { $EngineBin = "claude" } function Get-PendingCount { $tracker = Get-Content $TrackerPath -Raw return ([regex]::Matches($tracker, '\| PENDING \|')).Count } function Get-CompletedCount { $tracker = Get-Content $TrackerPath -Raw $done = ([regex]::Matches($tracker, '\| DONE \|')).Count $needsWork = ([regex]::Matches($tracker, '\| NEEDS_WORK \|')).Count $blocked = ([regex]::Matches($tracker, '\| BLOCKED \|')).Count return $done + $needsWork + $blocked } function Invoke-Iteration { param([int]$Iteration) Write-Host "" Write-Host ("=" * 50) -ForegroundColor Yellow Write-Host " Iteration $Iteration of $MaxIterations ($Type / $Model)" -ForegroundColor Yellow Write-Host ("=" * 50) -ForegroundColor Yellow $prompt = Get-Content $PromptPath -Raw if ($Type -eq 'copilot') { $copilotEffort = if ($Effort -eq 'max') { 'xhigh' } else { $Effort } $engineArgs = @( '-p', $prompt '--model', $Model '--allow-all' '--reasoning-effort', $copilotEffort ) if ($DryRun) { Write-Host " [DRY RUN] Would run: $EngineBin $($engineArgs -join ' ')" -ForegroundColor DarkGray return $true } & $EngineBin @engineArgs 2>&1 | ForEach-Object { $line = "$PSItem" if ($line -and $line -notmatch '^\s*$') { Write-Host " $line" -ForegroundColor White } } } else { $sessionId = [guid]::NewGuid().ToString() $engineArgs = @( '--dangerously-skip-permissions' '--session-id', $sessionId '--no-session-persistence' '--model', $Model '--effort', $Effort '--verbose' '--output-format', 'stream-json' '-p', $prompt ) if ($DryRun) { Write-Host " [DRY RUN] Would run: claude $($engineArgs -join ' ')" -ForegroundColor DarkGray return $true } # Stream and parse JSON - show tool use with file details & $EngineBin @engineArgs 2>&1 | ForEach-Object { $line = $PSItem try { $obj = $line | ConvertFrom-Json -ErrorAction Stop switch ($obj.type) { 'assistant' { if ($obj.message.content) { foreach ($content in $obj.message.content) { switch ($content.type) { 'tool_use' { $toolName = $content.name $detail = "" if ($content.input) { switch ($toolName) { 'Read' { $detail = Split-Path $content.input.file_path -Leaf } 'Write' { $detail = Split-Path $content.input.file_path -Leaf } 'Edit' { $detail = Split-Path $content.input.file_path -Leaf } 'Glob' { $detail = $content.input.pattern -replace '.*/',''} 'Grep' { $detail = $content.input.pattern.Substring(0, [Math]::Min(30, $content.input.pattern.Length)) } 'Bash' { $detail = ($content.input.command -split '\n')[0].Substring(0, [Math]::Min(40, ($content.input.command -split '\n')[0].Length)) } } } if ($detail) { Write-Host " 🔧 $toolName " -ForegroundColor DarkCyan -NoNewline Write-Host $detail -ForegroundColor DarkGray } else { Write-Host " 🔧 $toolName" -ForegroundColor DarkCyan } } 'text' { if ($content.text) { Write-Host $content.text -ForegroundColor White } } } } } } 'result' { $duration = [math]::Round($obj.duration_ms / 1000, 1) $cost = [math]::Round($obj.total_cost_usd, 4) Write-Host "" Write-Host "✓ Completed in ${duration}s (`$$cost)" -ForegroundColor Green } } } catch { if ($line -and $line -notmatch '^\s*$') { Write-Host $line -ForegroundColor DarkGray } } } } return $LASTEXITCODE -eq 0 } # Main loop $total = 135 # 138 commits minus 3 pre-marked version bumps Write-Host "Ralph Wiggum: commit-bug-review ($Type, $Model, effort=$Effort)" -ForegroundColor Cyan Write-Host "Tracker: $TrackerPath" -ForegroundColor DarkGray Write-Host "Prompt: $PromptPath" -ForegroundColor DarkGray Write-Host "" $iteration = 0 $stalledCount = 0 $maxStalled = 3 while ($iteration -lt $MaxIterations) { $iteration++ $pending = Get-PendingCount $completed = Get-CompletedCount $completedBefore = $completed $percent = [math]::Min(100, [math]::Round(($completed / $total) * 100)) $barFilled = [math]::Min(20, [math]::Floor($percent / 5)) $barEmpty = [math]::Max(0, 20 - $barFilled) Write-Host "[$(('=' * $barFilled))$(('-' * $barEmpty))] $percent% ($completed/$total)" -ForegroundColor Cyan if ($pending -eq 0) { Write-Host "All items completed!" -ForegroundColor Green break } if (-not (Invoke-Iteration -Iteration $iteration)) { Write-Host "Iteration failed. Stopping." -ForegroundColor Red exit 1 } # Zero-trust: verify the iteration actually progressed $completedAfter = Get-CompletedCount if ($completedAfter -le $completedBefore) { $stalledCount++ Write-Host "⚠️ No progress detected (stalled $stalledCount/$maxStalled)" -ForegroundColor Yellow Write-Host " Before: $completedBefore completed, $pending pending" -ForegroundColor Yellow Write-Host " After: $completedAfter completed" -ForegroundColor Yellow if ($stalledCount -ge $maxStalled) { Write-Host "✗ Stalled $maxStalled times in a row. AI is claiming done without finishing." -ForegroundColor Red Write-Host " Check the tracker for items marked DONE that shouldn't be." -ForegroundColor Red exit 1 } } else { $stalledCount = 0 Write-Host "✓ Progress: $completedBefore → $completedAfter completed" -ForegroundColor Green } Start-Sleep -Seconds 2 } Write-Host "Done. Completed: $(Get-CompletedCount) / $total" -ForegroundColor Green # SIG # Begin signature block # MIItZQYJKoZIhvcNAQcCoIItVjCCLVICAQMxDTALBglghkgBZQMEAgEwewYKKwYB # BAGCNwIBBKBtBGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCAHUA7yk0De+ID4 # ycYISia/bWt+yBX1ULSWavKQq39r2aCCFWUwggaTMIIEe6ADAgECAhMzAAA5v7gU # 41QXmFPaAAAAADm/MA0GCSqGSIb3DQEBDAUAMFoxCzAJBgNVBAYTAlVTMR4wHAYD # VQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xKzApBgNVBAMTIk1pY3Jvc29mdCBJ # RCBWZXJpZmllZCBDUyBBT0MgQ0EgMDQwHhcNMjYwNDE4MDMwNDIxWhcNMjYwNDIx # MDMwNDIxWjBXMQswCQYDVQQGEwJVUzERMA8GA1UECBMIVmlyZ2luaWExDzANBgNV # BAcTBlZpZW5uYTERMA8GA1UEChMIZGJhdG9vbHMxETAPBgNVBAMTCGRiYXRvb2xz # MIIBojANBgkqhkiG9w0BAQEFAAOCAY8AMIIBigKCAYEA2N5wFYucbTV9+nWJOZ8S # ptO6fRVKykqZV0UTa9Xyq+l3ckQT4HToVnGIoBabZ8dvmwZC5jASzMNDkGZSrFgj # BYWW7nEQ721zTnpVKaaxDx5+yHDzBNEW1T3DtB09jnQV1dXgfC/O0iqPYTQRW8sW # w8LQxDeNZk6mRZft/H8RJhh5eg1hsB0uFtMGanT+XouO4Gq53r7ngJyDedgK7TLi # /9cMfdjebpyVFS6oV42/WRNOaxrX6+akw1wEL6OmPXic+Jdb/qgXxvciUDMCUVbJ # DbMA+8bkznOz+M8D24wUDuOIGuLrgQ2C5Um5iq8Nl9SZrbSfD6lfXiIFfIGe4/hX # qXVdZnDuPOf+yz6IVzmckMaLCiWjURybB19Jl4n2oWXZgXKCaP7da43woOXCShkU # bm+MdJo0cT7zEkvZiDe0nAydyZbp1i79tFltF+/6/khhNHydnn/n1DXm8bsjWgfv # hE4t/DVfPDDugIVF6MG+rAPlyoGi/gMMRJZtn7UUJ7OFAgMBAAGjggHTMIIBzzAM # BgNVHRMBAf8EAjAAMA4GA1UdDwEB/wQEAwIHgDA6BgNVHSUEMzAxBgorBgEEAYI3 # YQEABggrBgEFBQcDAwYZKwYBBAGCN2H5+cEspPS4DoOuxLIcm56wGDAdBgNVHQ4E # FgQUZHQdNODk8b9RZgV608aFwQOeI8owHwYDVR0jBBgwFoAUayVB3vtrfP0YgAot # f492XapzPbgwZwYDVR0fBGAwXjBcoFqgWIZWaHR0cDovL3d3dy5taWNyb3NvZnQu # Y29tL3BraW9wcy9jcmwvTWljcm9zb2Z0JTIwSUQlMjBWZXJpZmllZCUyMENTJTIw # QU9DJTIwQ0ElMjAwNC5jcmwwdAYIKwYBBQUHAQEEaDBmMGQGCCsGAQUFBzAChlho # dHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL2NlcnRzL01pY3Jvc29mdCUy # MElEJTIwVmVyaWZpZWQlMjBDUyUyMEFPQyUyMENBJTIwMDQuY3J0MFQGA1UdIARN # MEswSQYEVR0gADBBMD8GCCsGAQUFBwIBFjNodHRwOi8vd3d3Lm1pY3Jvc29mdC5j # b20vcGtpb3BzL0RvY3MvUmVwb3NpdG9yeS5odG0wDQYJKoZIhvcNAQEMBQADggIB # ADNc4iklA54XgZxveEenpAwmuPiP5w2aQv2HnZwjN2Nw4k0+SwiYXandPfMcbvSO # Ko/bwo7mUouswI9Rx02P9GNqGtxxY3t8KUEiXmPjcHI2eRWIHtFCFMsQ9a4yDgg6 # zxJiWzXmQYr6gAasuJfWhGuWdSzVrybdLvRDMe0zGwPpGbvDBn/OMqQhxfuMOyKh # UdF0jOCHl7RinkpgQeOin+R/JJt9j4OsGikBvrBytlyKIlkFgfW8eGWMBuSEVF1i # rwxXDoIR8dGQ3sBFixwegVD9Kxs6S230KkBAUiy6+FRI1a4KEgJbsE0FxQg3Fl7O # 8kOtHBgtUFVUhNKCRKPSY9Fpc21lYW2JmjQeRw4o9eEHDBcc6gmQpup1sS6m4D7y # Lwkp8IoDutKup9AZBJBm0ksuNqfiuf7YMpkytLbrbx+MEfWIKw+8lVjt2A3mpc09 # YakLuP5tniUN7Ziir/JAsQkqHuEnrruqlynlcIz4q4QKY0wrXBUgos+Ia8furZ4W # k3SBqyt4DDHL4hqiWvq4gVYFkDtvPH2v8G5+yyRTExMqks3Rv1fTJM6k9WimqJio # 9hnlBEcVTiNhAYs5lT0XYBmYwcZqPMWgF6DD7EPnXSATm4Hlmc+kyHFGKusWUzgb # /swMYmnkUrN+S7oAWTfq8LxnGzXu0V3yAY3/D5S7tAe2MIIHKDCCBRCgAwIBAgIT # MwAAABYxko2SAmV7mgAAAAAAFjANBgkqhkiG9w0BAQwFADBjMQswCQYDVQQGEwJV # UzEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMTQwMgYDVQQDEytNaWNy # b3NvZnQgSUQgVmVyaWZpZWQgQ29kZSBTaWduaW5nIFBDQSAyMDIxMB4XDTI2MDMy # NjE4MTEyOVoXDTMxMDMyNjE4MTEyOVowWjELMAkGA1UEBhMCVVMxHjAcBgNVBAoT # FU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjErMCkGA1UEAxMiTWljcm9zb2Z0IElEIFZl # cmlmaWVkIENTIEFPQyBDQSAwNDCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoC # ggIBAMpV+sjb6Akwz/RtDk5Uo1284BLMttRQy9e5/5WXtga6h89pkdWjeAwcXmKd # P4YxKkhx8hn2q1dTVQbryvLy81tC04vg2bfSJ9emojX4HKIBYs7VhPRafMbtc866 # hN55aw1m/kWaPEKxF4Fm/LPLMLJdlu7URB8nFZMfh5tTC0CJb2uox/14OP/BiGIR # 8214lXdkV6JsPbO0Iev0mEV133tducIeBChMipzTZfnGVEq1QYFr7460cEGOIn+7 # AGfNOSq7gWOlmNB4m2uZ1r66vUJPaN+VYgH/Kmfu7tX229b3Alsli38fYS0nQY41 # bElntMS7yNY+Kd047eXM8/tS3NL+ZUNX3ge5xZqW2aytrrNIbwGgQnzsgzxBJvu9 # +b87jUWFiRS/z1YiPkRLY7iTHKIxJ973kIyK8K5itE/aEq/Ht6A8ytaAMMGTEwuC # spk72FE1Qyby+TfDlv1KiAc7IlWHHIWbxoVd8jGCoMXhLSDhuFuGfOWOZKUIuW6Y # xlxcUYOOkXa02gC7dUYUZi0e1NI1Uq9mmAHdvjKdqgORs9/5aDjnbeO3hWd5qwGB # mELWytkjY0KcIEF+CMurTimoQoBYSiHJbYrb0pKSQe+3lVoTVpzdi36jKI7MxLnu # 1fBAMksa8uD85cU7RU29zMOd18h9dgTEH9pvY+4xLB3lUgE7AgMBAAGjggHcMIIB # 2DAOBgNVHQ8BAf8EBAMCAYYwEAYJKwYBBAGCNxUBBAMCAQAwHQYDVR0OBBYEFGsl # Qd77a3z9GIAKLX+Pdl2qcz24MFQGA1UdIARNMEswSQYEVR0gADBBMD8GCCsGAQUF # BwIBFjNodHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL0RvY3MvUmVwb3Np # dG9yeS5odG0wGQYJKwYBBAGCNxQCBAweCgBTAHUAYgBDAEEwEgYDVR0TAQH/BAgw # BgEB/wIBADAfBgNVHSMEGDAWgBTZQSmwDw9jbO9p1/XNKZ6kSGow5jBwBgNVHR8E # aTBnMGWgY6Bhhl9odHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL2NybC9N # aWNyb3NvZnQlMjBJRCUyMFZlcmlmaWVkJTIwQ29kZSUyMFNpZ25pbmclMjBQQ0El # MjAyMDIxLmNybDB9BggrBgEFBQcBAQRxMG8wbQYIKwYBBQUHMAKGYWh0dHA6Ly93 # d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMvY2VydHMvTWljcm9zb2Z0JTIwSUQlMjBW # ZXJpZmllZCUyMENvZGUlMjBTaWduaW5nJTIwUENBJTIwMjAyMS5jcnQwDQYJKoZI # hvcNAQEMBQADggIBAAbVUF5UdNVEGWOVxkPciIzHA/IyNDIs1oSdW7mY4ObaFEEl # 8fwawEsBOascBzwXjuOaTkKelQ+IiC4JvDpn8pWPgOyiKrbbw1Iswt7AV8YyuLu7 # A0YshILlxyny9R0AMMQixLZ0T5Aj07CQOm/de5QPt36ECBwtVww6XUCx8Rm2xl6e # Ea9JhSub8W4urQGoQYv0Zczlz4ej2ryj5Wf9L4ZZA3bL6CRE7XmzSQjTmdSmr904 # PiuL2uBWzq2KkR3RhmoaAP4Jk2JplasNM5Bs0+dx3YX2o6xRrbSaJiu6hPk/AkBo # j5BCJMTZkl4wk6Q6nOFNSCpUxnBmJ0RRkoKq51p5ADTxbCeRAx8rIfNpTyPjxQtx # TiFTC8yy/t9K6s570YB1FAI+8XxZxIrmgMd7xVUkzi2/oooKb3UeovH0lqYGBEjp # HZJ9jee7boQbOe7SsKD/vr3PzFabg9VwLZlovpWUpWoWnU2w3xiozJ/m35tsZVvT # 2egUpwkb9TDIaXFGZAGV94FRDHn/K2XNnOwSeGskX19MB+N7yPc51fmUSd49MVAt # GW/NkUGpRech5Aq/d8dCML2bZ+j9nTUMgj+qnd3wuEZVRbQEIbTh9HAck0fyKqoI # b+qh8y/6TKbYDEEOcfM2BGMvrQk4WIGCK6u8uFhA2EH3kpaiMUDqyFCeCCQxMIIH # njCCBYagAwIBAgITMwAAAAeHozSje6WOHAAAAAAABzANBgkqhkiG9w0BAQwFADB3 # MQswCQYDVQQGEwJVUzEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMUgw # RgYDVQQDEz9NaWNyb3NvZnQgSWRlbnRpdHkgVmVyaWZpY2F0aW9uIFJvb3QgQ2Vy # dGlmaWNhdGUgQXV0aG9yaXR5IDIwMjAwHhcNMjEwNDAxMjAwNTIwWhcNMzYwNDAx # MjAxNTIwWjBjMQswCQYDVQQGEwJVUzEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBv # cmF0aW9uMTQwMgYDVQQDEytNaWNyb3NvZnQgSUQgVmVyaWZpZWQgQ29kZSBTaWdu # aW5nIFBDQSAyMDIxMIICIjANBgkqhkiG9w0BAQEFAAOCAg8AMIICCgKCAgEAsvDA # rxmIKOLdVHpMSWxpCFUJtFL/ekr4weslKPdnF3cpTeuV8veqtmKVgok2rO0D05Bp # yvUDCg1wdsoEtuxACEGcgHfjPF/nZsOkg7c0mV8hpMT/GvB4uhDvWXMIeQPsDgCz # UGzTvoi76YDpxDOxhgf8JuXWJzBDoLrmtThX01CE1TCCvH2sZD/+Hz3RDwl2MsvD # SdX5rJDYVuR3bjaj2QfzZFmwfccTKqMAHlrz4B7ac8g9zyxlTpkTuJGtFnLBGaso # Onn5NyYlf0xF9/bjVRo4Gzg2Yc7KR7yhTVNiuTGH5h4eB9ajm1OCShIyhrKqgOkc # 4smz6obxO+HxKeJ9bYmPf6KLXVNLz8UaeARo0BatvJ82sLr2gqlFBdj1sYfqOf00 # Qm/3B4XGFPDK/H04kteZEZsBRc3VT2d/iVd7OTLpSH9yCORV3oIZQB/Qr4nD4YT/ # lWkhVtw2v2s0TnRJubL/hFMIQa86rcaGMhNsJrhysLNNMeBhiMezU1s5zpusf54q # lYu2v5sZ5zL0KvBDLHtL8F9gn6jOy3v7Jm0bbBHjrW5yQW7S36ALAt03QDpwW1JG # 1Hxu/FUXJbBO2AwwVG4Fre+ZQ5Od8ouwt59FpBxVOBGfN4vN2m3fZx1gqn52Gvai # Bz6ozorgIEjn+PhUXILhAV5Q/ZgCJ0u2+ldFGjcCAwEAAaOCAjUwggIxMA4GA1Ud # DwEB/wQEAwIBhjAQBgkrBgEEAYI3FQEEAwIBADAdBgNVHQ4EFgQU2UEpsA8PY2zv # adf1zSmepEhqMOYwVAYDVR0gBE0wSzBJBgRVHSAAMEEwPwYIKwYBBQUHAgEWM2h0 # dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMvRG9jcy9SZXBvc2l0b3J5Lmh0 # bTAZBgkrBgEEAYI3FAIEDB4KAFMAdQBiAEMAQTAPBgNVHRMBAf8EBTADAQH/MB8G # A1UdIwQYMBaAFMh+0mqFKhvKGZgEByfPUBBPaKiiMIGEBgNVHR8EfTB7MHmgd6B1 # hnNodHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL2NybC9NaWNyb3NvZnQl # MjBJZGVudGl0eSUyMFZlcmlmaWNhdGlvbiUyMFJvb3QlMjBDZXJ0aWZpY2F0ZSUy # MEF1dGhvcml0eSUyMDIwMjAuY3JsMIHDBggrBgEFBQcBAQSBtjCBszCBgQYIKwYB # BQUHMAKGdWh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMvY2VydHMvTWlj # cm9zb2Z0JTIwSWRlbnRpdHklMjBWZXJpZmljYXRpb24lMjBSb290JTIwQ2VydGlm # aWNhdGUlMjBBdXRob3JpdHklMjAyMDIwLmNydDAtBggrBgEFBQcwAYYhaHR0cDov # L29uZW9jc3AubWljcm9zb2Z0LmNvbS9vY3NwMA0GCSqGSIb3DQEBDAUAA4ICAQB/ # JSqe/tSr6t1mCttXI0y6XmyQ41uGWzl9xw+WYhvOL47BV09Dgfnm/tU4ieeZ7NAR # 5bguorTCNr58HOcA1tcsHQqt0wJsdClsu8bpQD9e/al+lUgTUJEV80Xhco7xdgRr # ehbyhUf4pkeAhBEjABvIUpD2LKPho5Z4DPCT5/0TlK02nlPwUbv9URREhVYCtsDM # +31OFU3fDV8BmQXv5hT2RurVsJHZgP4y26dJDVF+3pcbtvh7R6NEDuYHYihfmE2H # dQRq5jRvLE1Eb59PYwISFCX2DaLZ+zpU4bX0I16ntKq4poGOFaaKtjIA1vRElIta # OKcwtc04CBrXSfyL2Op6mvNIxTk4OaswIkTXbFL81ZKGD+24uMCwo/pLNhn7VHLf # nxlMVzHQVL+bHa9KhTyzwdG/L6uderJQn0cGpLQMStUuNDArxW2wF16QGZ1NtBWg # KA8Kqv48M8HfFqNifN6+zt6J0GwzvU8g0rYGgTZR8zDEIJfeZxwWDHpSxB5FJ1VV # U1LIAtB7o9PXbjXzGifaIMYTzU4YKt4vMNwwBmetQDHhdAtTPplOXrnI9SI6HeTt # jDD3iUN/7ygbahmYOHk7VB7fwT4ze+ErCbMh6gHV1UuXPiLciloNxH6K4aMfZN1o # LVk6YFeIJEokuPgNPa6EnTiOL60cPqfny+Fq8UiuZzGCF1YwghdSAgEBMHEwWjEL # MAkGA1UEBhMCVVMxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjErMCkG # A1UEAxMiTWljcm9zb2Z0IElEIFZlcmlmaWVkIENTIEFPQyBDQSAwNAITMwAAOb+4 # FONUF5hT2gAAAAA5vzALBglghkgBZQMEAgGgfDAQBgorBgEEAYI3AgEMMQIwADAZ # BgkqhkiG9w0BCQMxDAYKKwYBBAGCNwIBBDAcBgorBgEEAYI3AgELMQ4wDAYKKwYB # BAGCNwIBFTAvBgkqhkiG9w0BCQQxIgQgyXMfsEJS+90T/pNYWvbxXXCcgwksNWaR # bEWio510jHUwCwYJKoZIhvcNAQEBBIIBgJrZ196IppRB29rFs3bvhYwi2RwPwSks # ib2fyddfvXkPWf7nvWuVUHSThfBKtyQcB7yiXxhy7MMiCEV1VGOh/P35sSC9ssIb # 3qB/9l3F9cLWCoUOwi/L5p4SaXTKMAHp3pZMUWm0fg+MXP7Rh3d5UhbxLCk9+RfC # KsSqksdJKj/11ZiVbqqN17l8+clIWtiX6i2JQovosAxZOh2tyuriNjj91NeQXwCm # kgGE8B58F8JRVpNjftMmVDEm3BDR4XB2i1WYanOTQNxmxLCU6vb3QkRWsrn1Gp9O # 55FFmsPVA0d3QzzIOQyS37+X3CeSdqQ5I9Dn1BiXTxv86+DmRzhYW5f1vou9glGB # 3gFxAPJOpdPYBYGGGJiVNinkZe10z0CTx74UoGv9iwqgJZUR8kjwRRRY99ZtnPBo # sks9y16KRmfDJ9CYgFzKxaja3JEXQUypxqDAKoHoZyCbhF41MnOQ/6L25v7y27kI # a917Bp2AGzuSfoMN2nfIrrscvfi0w/SCsaGCFLwwghS4BgorBgEEAYI3AwMBMYIU # qDCCFKQGCSqGSIb3DQEHAqCCFJUwghSRAgEDMQ8wDQYJYIZIAWUDBAIBBQAwggF0 # BgsqhkiG9w0BCRABBKCCAWMEggFfMIIBWwIBAQYKKwYBBAGEWQoDATAxMA0GCWCG # SAFlAwQCAQUABCAvn3TIdivTxCuGDQe32Aard81QMjjLBxwdFf/DncTrsQIGacZo # FVOBGBMyMDI2MDQxODA0NTkyNi40OTdaMASAAgH0AghNjaZhvq5z2KCB6aSB5jCB # 4zELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1Jl # ZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEtMCsGA1UECxMk # TWljcm9zb2Z0IElyZWxhbmQgT3BlcmF0aW9ucyBMaW1pdGVkMScwJQYDVQQLEx5u # U2hpZWxkIFRTUyBFU046N0IxQS0wNUUwLUQ5NDcxNTAzBgNVBAMTLE1pY3Jvc29m # dCBQdWJsaWMgUlNBIFRpbWUgU3RhbXBpbmcgQXV0aG9yaXR5oIIPKTCCB4IwggVq # oAMCAQICEzMAAAAF5c8P/2YuyYcAAAAAAAUwDQYJKoZIhvcNAQEMBQAwdzELMAkG # A1UEBhMCVVMxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjFIMEYGA1UE # AxM/TWljcm9zb2Z0IElkZW50aXR5IFZlcmlmaWNhdGlvbiBSb290IENlcnRpZmlj # YXRlIEF1dGhvcml0eSAyMDIwMB4XDTIwMTExOTIwMzIzMVoXDTM1MTExOTIwNDIz # MVowYTELMAkGA1UEBhMCVVMxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlv # bjEyMDAGA1UEAxMpTWljcm9zb2Z0IFB1YmxpYyBSU0EgVGltZXN0YW1waW5nIENB # IDIwMjAwggIiMA0GCSqGSIb3DQEBAQUAA4ICDwAwggIKAoICAQCefOdSY/3gxZ8F # fWO1BiKjHB7X55cz0RMFvWVGR3eRwV1wb3+yq0OXDEqhUhxqoNv6iYWKjkMcLhEF # xvJAeNcLAyT+XdM5i2CgGPGcb95WJLiw7HzLiBKrxmDj1EQB/mG5eEiRBEp7dDGz # xKCnTYocDOcRr9KxqHydajmEkzXHOeRGwU+7qt8Md5l4bVZrXAhK+WSk5CihNQsW # bzT1nRliVDwunuLkX1hyIWXIArCfrKM3+RHh+Sq5RZ8aYyik2r8HxT+l2hmRllBv # E2Wok6IEaAJanHr24qoqFM9WLeBUSudz+qL51HwDYyIDPSQ3SeHtKog0ZubDk4hE # LQSxnfVYXdTGncaBnB60QrEuazvcob9n4yR65pUNBCF5qeA4QwYnilBkfnmeAjRN # 3LVuLr0g0FXkqfYdUmj1fFFhH8k8YBozrEaXnsSL3kdTD01X+4LfIWOuFzTzuosl # BrBILfHNj8RfOxPgjuwNvE6YzauXi4orp4Sm6tF245DaFOSYbWFK5ZgG6cUY2/bU # q3g3bQAqZt65KcaewEJ3ZyNEobv35Nf6xN6FrA6jF9447+NHvCjeWLCQZ3M8lgeC # cnnhTFtyQX3XgCoc6IRXvFOcPVrr3D9RPHCMS6Ckg8wggTrtIVnY8yjbvGOUsAdZ # beXUIQAWMs0d3cRDv09SvwVRd61evQIDAQABo4ICGzCCAhcwDgYDVR0PAQH/BAQD # AgGGMBAGCSsGAQQBgjcVAQQDAgEAMB0GA1UdDgQWBBRraSg6NS9IY0DPe9ivSek+ # 2T3bITBUBgNVHSAETTBLMEkGBFUdIAAwQTA/BggrBgEFBQcCARYzaHR0cDovL3d3 # dy5taWNyb3NvZnQuY29tL3BraW9wcy9Eb2NzL1JlcG9zaXRvcnkuaHRtMBMGA1Ud # JQQMMAoGCCsGAQUFBwMIMBkGCSsGAQQBgjcUAgQMHgoAUwB1AGIAQwBBMA8GA1Ud # EwEB/wQFMAMBAf8wHwYDVR0jBBgwFoAUyH7SaoUqG8oZmAQHJ89QEE9oqKIwgYQG # A1UdHwR9MHsweaB3oHWGc2h0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMv # Y3JsL01pY3Jvc29mdCUyMElkZW50aXR5JTIwVmVyaWZpY2F0aW9uJTIwUm9vdCUy # MENlcnRpZmljYXRlJTIwQXV0aG9yaXR5JTIwMjAyMC5jcmwwgZQGCCsGAQUFBwEB # BIGHMIGEMIGBBggrBgEFBQcwAoZ1aHR0cDovL3d3dy5taWNyb3NvZnQuY29tL3Br # aW9wcy9jZXJ0cy9NaWNyb3NvZnQlMjBJZGVudGl0eSUyMFZlcmlmaWNhdGlvbiUy # MFJvb3QlMjBDZXJ0aWZpY2F0ZSUyMEF1dGhvcml0eSUyMDIwMjAuY3J0MA0GCSqG # SIb3DQEBDAUAA4ICAQBfiHbHfm21WhV150x4aPpO4dhEmSUVpbixNDmv6TvuIHv1 # xIs174bNGO/ilWMm+Jx5boAXrJxagRhHQtiFprSjMktTliL4sKZyt2i+SXncM23g # RezzsoOiBhv14YSd1Klnlkzvgs29XNjT+c8hIfPRe9rvVCMPiH7zPZcw5nNjthDQ # +zD563I1nUJ6y59TbXWsuyUsqw7wXZoGzZwijWT5oc6GvD3HDokJY401uhnj3ubB # hbkR83RbfMvmzdp3he2bvIUztSOuFzRqrLfEvsPkVHYnvH1wtYyrt5vShiKheGpX # a2AWpsod4OJyT4/y0dggWi8g/tgbhmQlZqDUf3UqUQsZaLdIu/XSjgoZqDjamzCP # JtOLi2hBwL+KsCh0Nbwc21f5xvPSwym0Ukr4o5sCcMUcSy6TEP7uMV8RX0eH/4JL # EpGyae6Ki8JYg5v4fsNGif1OXHJ2IWG+7zyjTDfkmQ1snFOTgyEX8qBpefQbF0fx # 6URrYiarjmBprwP6ZObwtZXJ23jK3Fg/9uqM3j0P01nzVygTppBabzxPAh/hHhhl # s6kwo3QLJ6No803jUsZcd4JQxiYHHc+Q/wAMcPUnYKv/q2O444LO1+n6j01z5mgg # CSlRwD9faBIySAcA9S8h22hIAcRQqIGEjolCK9F6nK9ZyX4lhthsGHumaABdWzCC # B58wggWHoAMCAQICEzMAAABZfNpx6Y1e9cAAAAAAAFkwDQYJKoZIhvcNAQEMBQAw # YTELMAkGA1UEBhMCVVMxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEy # MDAGA1UEAxMpTWljcm9zb2Z0IFB1YmxpYyBSU0EgVGltZXN0YW1waW5nIENBIDIw # MjAwHhcNMjYwMTA4MTg1OTAxWhcNMjcwMTA3MTg1OTAxWjCB4zELMAkGA1UEBhMC # VVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNV # BAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEtMCsGA1UECxMkTWljcm9zb2Z0IEly # ZWxhbmQgT3BlcmF0aW9ucyBMaW1pdGVkMScwJQYDVQQLEx5uU2hpZWxkIFRTUyBF # U046N0IxQS0wNUUwLUQ5NDcxNTAzBgNVBAMTLE1pY3Jvc29mdCBQdWJsaWMgUlNB # IFRpbWUgU3RhbXBpbmcgQXV0aG9yaXR5MIICIjANBgkqhkiG9w0BAQEFAAOCAg8A # MIICCgKCAgEApi7n/jR4Rf6FP7mMVrXjpuJ3d0Dguddie2+Q/cIbDgsYBEWGe8sx # In9W2y5vMNp4WebvJvYpIzvRaYdCmt0JWqm9QfuXS4HiM3Du6sugBH2QocsTRWUy # UNK0NLFQEjerCx2uO92a9XST73+eO4MSJKXMlN/sOjB3Urds7ht5HJoWFH5jy3KY # QI6qU4B99isbbBl1UznX5BIFJ748TKSvwLo6eepKKm4Xx9m8Jvr+G6TRmbpnCxqL # FIcgBPYgQa9LtzrcibyNXdrxHQrqLbKLDQ02WdNKnDL9l+/uLCwsHCF9uMFOf5c6 # XqY/MNdBDdX5JE/3FdsYo6wFPHMaJ3tooAfDejgCGX1QZYsf2Q0/dVSiQToliSOV # +m5QZnqBDKoN7B/EPhWhgiWim3gdnTFC+pqO4nH5yvwtH8hnnqAsubDIzN17n6+2 # MGiKWvL+BJnBUbCCS+QiCko8FAcaxIHTLezOvtjvARvq/TJlqXVdS9aefPeKdNpJ # awIWss9XBWZfLedxjn93blWk6SG36br3sZY1u+w06EA4dFp+0T2P+GGSgzpGzIl8 # EueGoxwD/Bxq9/miPk17JFB0Zl4spyWz1ywpLewFTM+J3HHaKI0f0I9oOr9sskQ5 # dqxiiFcydGKe759STzWzxU7sNBIXp9+fmipIehXyV/UDHq76R1KFvwECAwEAAaOC # AcswggHHMB0GA1UdDgQWBBSPlucLXwPS+WdJGEI6I4MYa5c2djAfBgNVHSMEGDAW # gBRraSg6NS9IY0DPe9ivSek+2T3bITBsBgNVHR8EZTBjMGGgX6BdhltodHRwOi8v # d3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL2NybC9NaWNyb3NvZnQlMjBQdWJsaWMl # MjBSU0ElMjBUaW1lc3RhbXBpbmclMjBDQSUyMDIwMjAuY3JsMHkGCCsGAQUFBwEB # BG0wazBpBggrBgEFBQcwAoZdaHR0cDovL3d3dy5taWNyb3NvZnQuY29tL3BraW9w # cy9jZXJ0cy9NaWNyb3NvZnQlMjBQdWJsaWMlMjBSU0ElMjBUaW1lc3RhbXBpbmcl # MjBDQSUyMDIwMjAuY3J0MAwGA1UdEwEB/wQCMAAwFgYDVR0lAQH/BAwwCgYIKwYB # BQUHAwgwDgYDVR0PAQH/BAQDAgeAMGYGA1UdIARfMF0wUQYMKwYBBAGCN0yDfQEB # MEEwPwYIKwYBBQUHAgEWM2h0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMv # RG9jcy9SZXBvc2l0b3J5Lmh0bTAIBgZngQwBBAIwDQYJKoZIhvcNAQEMBQADggIB # AEQyHML9lyOkb/NRETvPZinm+tTFSOwTldre3/ZEa8RLC9uayTdsseFIUBqOCjZd # DzGgYM/exSEoIs94gJJuRW/pXyxKnx8nwmizrKtrB/ZhWKzy1xX447tT1LokY9DO # mY+dx/NBzid7V+pnj4eIppsgMSgV111BPYSCST1/PEJj8RnnZq8nbt1J+sj8gObK # f1XXQ7eJCmDYX4L281OuTudWCgOgdr35DelivQgcadKa3mKuAZPjfrOjladc/wXE # yGzAfVqLKJJmJXOGivXQL0LnZw/cLz88SgwpRey0uOg87RMyR5b/UWvXzopcPjlB # GAGjTEiaC1ZN3NYWeUP6nv6IMEi5Ks1xFcNFi6r9phloOFZIjRJh08hZWHle9e5Y # DfVryhDRI76g9rc2TzSTTzrjCKLInUqxtKdpp5+D2+yl6CGuljyWMLDF8JS3HkdB # E2AAk4jwgoAIrh3TXgyTSGVE7SVBVEd6CoiDb0rpeqRzfrRbZkkZRyy2UqiSRWPo # oAeMbKstE/N6lh+JlQmriBfLFum8pvVcnYU3brzZpSg1ej1HJErVyHR/rqTr7KPz # yTO+4Lv68la00Oz9SkgzoeGbSRTrglQnswwRBwBzuSl+3sabth5Er6gqCpSBhcJ+ # bNtfyclclmc9rwANZWwiqy40DEIjQJleRQuh+qjkF7rPMYID1DCCA9ACAQEweDBh # MQswCQYDVQQGEwJVUzEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMTIw # MAYDVQQDEylNaWNyb3NvZnQgUHVibGljIFJTQSBUaW1lc3RhbXBpbmcgQ0EgMjAy # MAITMwAAAFl82nHpjV71wAAAAAAAWTANBglghkgBZQMEAgEFAKCCAS0wGgYJKoZI # hvcNAQkDMQ0GCyqGSIb3DQEJEAEEMC8GCSqGSIb3DQEJBDEiBCDaNnKkzydelmuI # Rv80DiotX0hnA7rkPeolmMl0deNBNDCB3QYLKoZIhvcNAQkQAi8xgc0wgcowgccw # gaAEIMtFurHbhumxxcQn4eOP3GtxRXDtHw7LIx9IHZgYrf68MHwwZaRjMGExCzAJ # BgNVBAYTAlVTMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xMjAwBgNV # BAMTKU1pY3Jvc29mdCBQdWJsaWMgUlNBIFRpbWVzdGFtcGluZyBDQSAyMDIwAhMz # AAAAWXzacemNXvXAAAAAAABZMCIEICxWpzaGb/nGFr7HB6YjwlDz2c+BfP2MRo4C # vosyKbbjMA0GCSqGSIb3DQEBCwUABIICAG3Ki0RtZaFHogXPz97HC9vBpaYohZLG # YxkWIHHGQFF70OpMcKcaGPah8Lwbd15BT+mhwH8s77v/ME6h9S5ncB3xtSe+Gt7D # 6xlwD1tA3l2Rpglzkcb91riecrSF3DOkrFdByFsD2WZFRTPSaZroSaW465dbCebO # 9cxX7MpIbpLdBQcsjJrvnc7y5RlZQRZdLfbC1895cAHXomVT6oVFhdgrbc7/mcZT # R5JLV52oLGJ+tgKJI1Jq6vlvcQNQfHiVABF1pKSdpnHTW6vqJIDkHqYSnL5IGRnJ # bMKoCnOVq7SAaF+Of0lrNddiMi4T4Ax8Q6zS7uYBFdIBD/ghN7m9AznkUVJAUGXG # inUEsQbkmK10Co+0UJ7dAo8wBLFToFpjz3fMJW7CN/8aBg0TTp6ov7VuS7hSyVI5 # SuzRKpOyDJgzvw7NXhtuem2R+oSufOgWVgCZoZuBqyF5WKzbz5RiLINAh31D/geW # miTB0vVyKOuvVr1DNisvrBFuc3XCgER1LhtFjZKjWAgLMuMZdD+80yanW1xOnufL # upKhA17+1+9T6jjaPKR3ieMXyvTa39i8SyZWtL5YUv9gxHeUtO2aMkX5Mdwubmv/ # SmNSALiAt8VK+woH3BF0UG+2NiSzvT2qzXwmmh7RgCxyYHaPDH629w13yeINUbWo # c/9Bv9Jvz6go # SIG # End signature block |