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