@pugi/cli 0.1.0-beta.9 → 0.1.0-beta.90
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +132 -0
- package/LICENSE +1 -1
- package/assets/pugi-prozr2-mascot.ansi +9 -0
- package/bin/run.js +33 -1
- package/dist/commands/deploy.js +40 -40
- package/dist/commands/flatten.js +191 -0
- package/dist/commands/jobs-watch.js +201 -0
- package/dist/commands/jobs.js +42 -27
- package/dist/commands/smoke.js +133 -0
- package/dist/core/agent-progress/cleanup.js +134 -0
- package/dist/core/agent-progress/schema.js +144 -0
- package/dist/core/agent-progress/writer.js +101 -0
- package/dist/core/agents/adaptive-router.js +330 -0
- package/dist/core/agents/query-decomposer.js +297 -0
- package/dist/core/agents/registry.js +3 -3
- package/dist/core/approvals/shortcut-resolver.js +98 -0
- package/dist/core/artifact-chain/dispatcher.js +148 -0
- package/dist/core/artifact-chain/exporter.js +164 -0
- package/dist/core/artifact-chain/state.js +243 -0
- package/dist/core/artifact-chain/steps.js +169 -0
- package/dist/core/ask-user/question.js +92 -0
- package/dist/core/audit/audit-trail.js +275 -0
- package/dist/core/auth/ensure-authenticated.js +129 -0
- package/dist/core/auth/env-provider.js +238 -0
- package/dist/core/auto-open-browser.js +4 -4
- package/dist/core/auto-update/channels.js +122 -0
- package/dist/core/auto-update/checker.js +241 -0
- package/dist/core/auto-update/state.js +235 -0
- package/dist/core/bare-mode/index.js +107 -0
- package/dist/core/bash/redirect.js +281 -0
- package/dist/core/bash-classifier.js +436 -40
- package/dist/core/checkpoint/resumer.js +149 -0
- package/dist/core/checkpoint/rewinder.js +291 -0
- package/dist/core/checkpoints/shadow-git.js +670 -0
- package/dist/core/citations/parser.js +109 -0
- package/dist/core/classifier/yolo-classifier.js +88 -0
- package/dist/core/codegraph/decision-store.js +248 -0
- package/dist/core/codegraph/detect-repo.js +459 -0
- package/dist/core/codegraph/install.js +134 -0
- package/dist/core/codegraph/offer-hook.js +220 -0
- package/dist/core/compact/auto-trigger.js +96 -0
- package/dist/core/compact/buffer-rewriter.js +115 -0
- package/dist/core/compact/summarizer.js +208 -0
- package/dist/core/compact/token-counter.js +108 -0
- package/dist/core/consensus/anvil-fanout.js +25 -25
- package/dist/core/consensus/diff-capture.js +121 -12
- package/dist/core/consensus/rubric.js +21 -21
- package/dist/core/context/builder.js +6 -6
- package/dist/core/context/compaction-events.js +8 -8
- package/dist/core/context/compaction.js +31 -31
- package/dist/core/context/index.js +15 -8
- package/dist/core/context/invariants.js +51 -51
- package/dist/core/context/markdown-loader.js +28 -10
- package/dist/core/context/markdown-traverse.js +255 -0
- package/dist/core/context/pugiignore.js +41 -41
- package/dist/core/context/repo-skeleton.js +37 -37
- package/dist/core/context/tool-eviction.js +55 -0
- package/dist/core/context/watcher.js +32 -32
- package/dist/core/context/working-set.js +23 -23
- package/dist/core/coordinator/agent-tools.js +77 -0
- package/dist/core/coordinator/agent-toolset.js +65 -0
- package/dist/core/coordinator/fsm.js +73 -0
- package/dist/core/coordinator/mode-fsm.js +70 -0
- package/dist/core/cost/rate-card.js +129 -0
- package/dist/core/cost/tracker.js +221 -0
- package/dist/core/credentials.js +13 -13
- package/dist/core/cron/scheduler.js +138 -0
- package/dist/core/denial-tracking/index.js +8 -0
- package/dist/core/denial-tracking/state.js +264 -0
- package/dist/core/diagnostics/probe-runner.js +93 -0
- package/dist/core/diagnostics/probes/api.js +46 -0
- package/dist/core/diagnostics/probes/auth.js +93 -0
- package/dist/core/diagnostics/probes/bare-mode.js +42 -0
- package/dist/core/diagnostics/probes/cli-version.js +127 -0
- package/dist/core/diagnostics/probes/config.js +72 -0
- package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
- package/dist/core/diagnostics/probes/disk.js +81 -0
- package/dist/core/diagnostics/probes/engine-live.js +46 -0
- package/dist/core/diagnostics/probes/git.js +65 -0
- package/dist/core/diagnostics/probes/hooks.js +118 -0
- package/dist/core/diagnostics/probes/mcp.js +75 -0
- package/dist/core/diagnostics/probes/node.js +59 -0
- package/dist/core/diagnostics/probes/pnpm.js +36 -0
- package/dist/core/diagnostics/probes/pugi-md.js +89 -0
- package/dist/core/diagnostics/probes/sandbox.js +40 -0
- package/dist/core/diagnostics/probes/session.js +74 -0
- package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
- package/dist/core/diagnostics/probes/workspace.js +63 -0
- package/dist/core/diagnostics/types.js +70 -0
- package/dist/core/dispatch/cache-cleanup.js +197 -0
- package/dist/core/dispatch/cache-handoff.js +295 -0
- package/dist/core/edits/apply-patch-layer-e.js +189 -0
- package/dist/core/edits/dispatch.js +333 -7
- package/dist/core/edits/format-detector.js +260 -0
- package/dist/core/edits/format-matrix.js +26 -0
- package/dist/core/edits/fuzzy-ladder.js +650 -0
- package/dist/core/edits/index.js +5 -1
- package/dist/core/edits/journal.js +199 -0
- package/dist/core/edits/layer-a-apply.js +15 -15
- package/dist/core/edits/layer-a-fuzzy-apply.js +198 -0
- package/dist/core/edits/layer-b-apply.js +9 -9
- package/dist/core/edits/layer-c-apply.js +6 -6
- package/dist/core/edits/layer-d-ast.js +557 -14
- package/dist/core/edits/marker-parser.js +12 -12
- package/dist/core/edits/security-gate.js +27 -27
- package/dist/core/edits/verify-hook.js +273 -0
- package/dist/core/edits/worktree.js +29 -29
- package/dist/core/engine/anvil-client.js +214 -26
- package/dist/core/engine/auto-compact.js +179 -0
- package/dist/core/engine/budgets.js +186 -0
- package/dist/core/engine/context-prefix.js +155 -0
- package/dist/core/engine/index.js +1 -1
- package/dist/core/engine/intensity.js +158 -0
- package/dist/core/engine/intent.js +260 -0
- package/dist/core/engine/native-pugi.js +1295 -227
- package/dist/core/engine/prompts.js +129 -19
- package/dist/core/engine/strip-internal-fields.js +124 -0
- package/dist/core/engine/tool-bridge.js +1731 -59
- package/dist/core/evaluation/golden-dataset.js +293 -0
- package/dist/core/feedback/queue.js +177 -0
- package/dist/core/feedback/submitter.js +145 -0
- package/dist/core/file-cache.js +113 -1
- package/dist/core/flatten/flatten-repo.js +439 -0
- package/dist/core/format/osc8-link.js +28 -0
- package/dist/core/hook-chains.js +392 -0
- package/dist/core/hooks/citation-verify-hook.js +138 -0
- package/dist/core/hooks/citation-verify.js +112 -0
- package/dist/core/hooks/events.js +46 -0
- package/dist/core/hooks/index.js +15 -0
- package/dist/core/hooks/registry.js +216 -0
- package/dist/core/hooks/runner.js +236 -0
- package/dist/core/hooks/v2/event-emitter.js +115 -0
- package/dist/core/hooks/v2/executor.js +282 -0
- package/dist/core/hooks/v2/index.js +25 -0
- package/dist/core/hooks/v2/lifecycle.js +104 -0
- package/dist/core/hooks/v2/loader.js +216 -0
- package/dist/core/hooks/v2/matcher.js +125 -0
- package/dist/core/hooks/v2/trust.js +143 -0
- package/dist/core/hooks/v2/types.js +86 -0
- package/dist/core/hooks/worktree-events.js +158 -0
- package/dist/core/image/renderer.js +71 -0
- package/dist/core/init/detector.js +582 -0
- package/dist/core/init/template-renderer.js +242 -0
- package/dist/core/jobs/registry.js +18 -18
- package/dist/core/ledger/results-tsv.js +142 -0
- package/dist/core/log-discipline/stdout-redirect.js +51 -0
- package/dist/core/lsp/cache.js +105 -0
- package/dist/core/lsp/client.js +551 -41
- package/dist/core/lsp/language-detect.js +66 -0
- package/dist/core/lsp/post-edit-diagnostics.js +171 -0
- package/dist/core/lsp/server-detect.js +173 -0
- package/dist/core/lsp/symbol-cache.js +162 -0
- package/dist/core/lsp/symbol-tools.js +664 -0
- package/dist/core/mcp/client.js +97 -28
- package/dist/core/mcp/http-server.js +553 -0
- package/dist/core/mcp/orchestrator-tools.js +662 -0
- package/dist/core/mcp/permission.js +190 -0
- package/dist/core/mcp/registry.js +39 -17
- package/dist/core/mcp/server-tools.js +219 -0
- package/dist/core/mcp/server.js +397 -0
- package/dist/core/mcp/trust.js +10 -10
- package/dist/core/memory/dual-write.js +416 -0
- package/dist/core/memory/passive-extract.js +130 -0
- package/dist/core/memory/phase1-kinds.js +20 -0
- package/dist/core/memory/secret-scanner.js +304 -0
- package/dist/core/memory-sync/queue.js +170 -0
- package/dist/core/metrics/extract.js +113 -0
- package/dist/core/modes/roo-modes.js +68 -0
- package/dist/core/onboarding/ensure-initialized.js +133 -0
- package/dist/core/onboarding/marker.js +111 -0
- package/dist/core/onboarding/telemetry-state.js +108 -0
- package/dist/core/output-style/presets.js +176 -0
- package/dist/core/output-style/state.js +185 -0
- package/dist/core/path-security.js +287 -5
- package/dist/core/permission.js +82 -22
- package/dist/core/permissions/auto-classifier.js +124 -0
- package/dist/core/permissions/bash-parser.js +371 -0
- package/dist/core/permissions/circuit-breaker.js +83 -0
- package/dist/core/permissions/constrained-edit.js +91 -0
- package/dist/core/permissions/gate.js +278 -0
- package/dist/core/permissions/index.js +20 -0
- package/dist/core/permissions/mode.js +174 -0
- package/dist/core/permissions/network-egress.js +137 -0
- package/dist/core/permissions/state.js +241 -0
- package/dist/core/permissions/tool-class.js +93 -0
- package/dist/core/plan-mode/ui-state.js +51 -0
- package/dist/core/plans/plan-artifact.js +721 -0
- package/dist/core/policy-limits/etag-store.js +122 -0
- package/dist/core/prd-check/parser.js +215 -0
- package/dist/core/prd-check/reporter.js +127 -0
- package/dist/core/prd-check/session-review.js +557 -0
- package/dist/core/prd-check/verifiers.js +223 -0
- package/dist/core/prompt-cache/client-cache.js +99 -0
- package/dist/core/prompts/assembly.js +29 -0
- package/dist/core/prompts/registry.js +364 -0
- package/dist/core/pugi-md/cc-compat-rules.js +735 -0
- package/dist/core/pugi-md/context-injector.js +76 -0
- package/dist/core/pugi-md/walk-up.js +207 -0
- package/dist/core/python/uv-installer.js +270 -0
- package/dist/core/python/uv-resolver.js +83 -0
- package/dist/core/rate-limit/narrator.js +146 -0
- package/dist/core/recipes/cli-types.js +20 -0
- package/dist/core/recipes/loader.js +103 -0
- package/dist/core/recipes/runner.js +345 -0
- package/dist/core/recipes/schema.js +587 -0
- package/dist/core/release-notes/parser.js +241 -0
- package/dist/core/release-notes/state.js +116 -0
- package/dist/core/repl/ask.js +37 -37
- package/dist/core/repl/cancellation.js +26 -26
- package/dist/core/repl/cap-warning.js +4 -4
- package/dist/core/repl/clipboard-read.js +11 -11
- package/dist/core/repl/dispatch-fsm.js +12 -12
- package/dist/core/repl/history-search.js +15 -15
- package/dist/core/repl/history.js +28 -18
- package/dist/core/repl/kill-ring.js +5 -5
- package/dist/core/repl/model-pricing.js +135 -0
- package/dist/core/repl/privacy-banner.js +22 -22
- package/dist/core/repl/session.js +2148 -217
- package/dist/core/repl/slash-commands.js +501 -41
- package/dist/core/repl/store/index.js +1 -1
- package/dist/core/repl/store/jsonl-log.js +22 -22
- package/dist/core/repl/store/lockfile.js +10 -10
- package/dist/core/repl/store/session-store.js +136 -107
- package/dist/core/repl/store/types.js +15 -15
- package/dist/core/repl/store/uuid-v7.js +12 -12
- package/dist/core/repl/workspace-context.js +43 -21
- package/dist/core/repo-map/build.js +125 -0
- package/dist/core/repo-map/cache.js +185 -0
- package/dist/core/repo-map/extractor.js +254 -0
- package/dist/core/repo-map/formatter.js +145 -0
- package/dist/core/repo-map/page-rank.js +105 -0
- package/dist/core/repo-map/scanner.js +211 -0
- package/dist/core/retry-budget/budget.js +284 -0
- package/dist/core/retry-budget/index.js +5 -0
- package/dist/core/retry-budget/retry-cap.js +74 -0
- package/dist/core/routing/lead-worker.js +43 -0
- package/dist/core/routing/pre-flight-estimator.js +108 -0
- package/dist/core/runs/run-tree.js +103 -0
- package/dist/core/security/injection-scanner.js +367 -0
- package/dist/core/security/output-filter.js +418 -0
- package/dist/core/session/env-file.js +105 -0
- package/dist/core/session/section-budgets.js +140 -0
- package/dist/core/session.js +92 -0
- package/dist/core/settings.js +324 -5
- package/dist/core/share/formatter.js +271 -0
- package/dist/core/share/redactor.js +221 -0
- package/dist/core/share/uploader.js +267 -0
- package/dist/core/skills/defaults.js +30 -30
- package/dist/core/skills/loader.js +22 -22
- package/dist/core/skills/sources.js +27 -27
- package/dist/core/smoke/headless-driver.js +174 -0
- package/dist/core/smoke/orchestrator.js +194 -0
- package/dist/core/smoke/runner.js +238 -0
- package/dist/core/smoke/scenario-parser.js +316 -0
- package/dist/core/statusline.js +99 -0
- package/dist/core/subagents/dispatcher-real.js +600 -0
- package/dist/core/subagents/dispatcher.js +132 -43
- package/dist/core/subagents/index.js +19 -6
- package/dist/core/subagents/isolation-matrix.js +213 -0
- package/dist/core/subagents/spawn.js +19 -4
- package/dist/core/telemetry/emitter.js +229 -0
- package/dist/core/telemetry/queue.js +251 -0
- package/dist/core/theme/context.js +91 -0
- package/dist/core/theme/presets.js +228 -0
- package/dist/core/theme/state.js +181 -0
- package/dist/core/todos/invariant.js +10 -0
- package/dist/core/todos/state.js +177 -0
- package/dist/core/tool-schema/compressor.js +89 -0
- package/dist/core/transport/version-interceptor.js +166 -0
- package/dist/core/trust.js +2 -2
- package/dist/core/tui/thinking-block.js +64 -0
- package/dist/core/vim/keymap.js +288 -0
- package/dist/core/vim/state.js +92 -0
- package/dist/core/watch-markers/marker-watcher.js +133 -0
- package/dist/core/worktree/include-parser.js +249 -0
- package/dist/core/worktree-manager/cleanup.js +123 -0
- package/dist/core/worktree-manager/manager.js +303 -0
- package/dist/index.js +36 -0
- package/dist/runtime/bootstrap.js +190 -0
- package/dist/runtime/cli.js +4185 -549
- package/dist/runtime/commands/agents.js +31 -31
- package/dist/runtime/commands/budget.js +5 -5
- package/dist/runtime/commands/cancel.js +231 -0
- package/dist/runtime/commands/chain.js +489 -0
- package/dist/runtime/commands/codegraph-status.js +227 -0
- package/dist/runtime/commands/compact.js +297 -0
- package/dist/runtime/commands/config.js +73 -39
- package/dist/runtime/commands/cost.js +199 -0
- package/dist/runtime/commands/delegate.js +27 -4
- package/dist/runtime/commands/dispatch.js +126 -0
- package/dist/runtime/commands/doctor.js +579 -0
- package/dist/runtime/commands/feedback.js +184 -0
- package/dist/runtime/commands/hooks.js +187 -0
- package/dist/runtime/commands/init.js +254 -0
- package/dist/runtime/commands/lsp.js +200 -38
- package/dist/runtime/commands/mcp.js +879 -0
- package/dist/runtime/commands/memory.js +582 -0
- package/dist/runtime/commands/model.js +237 -0
- package/dist/runtime/commands/onboarding.js +275 -0
- package/dist/runtime/commands/patch.js +12 -12
- package/dist/runtime/commands/permissions.js +112 -0
- package/dist/runtime/commands/plan.js +143 -0
- package/dist/runtime/commands/prd-check.js +285 -0
- package/dist/runtime/commands/privacy.js +17 -17
- package/dist/runtime/commands/recipe.js +325 -0
- package/dist/runtime/commands/redo-blob-store.js +92 -0
- package/dist/runtime/commands/redo.js +361 -0
- package/dist/runtime/commands/release-notes.js +229 -0
- package/dist/runtime/commands/repo-map.js +95 -0
- package/dist/runtime/commands/report.js +299 -0
- package/dist/runtime/commands/resume.js +118 -0
- package/dist/runtime/commands/review-consensus.js +68 -53
- package/dist/runtime/commands/rewind.js +333 -0
- package/dist/runtime/commands/roster.js +14 -14
- package/dist/runtime/commands/sessions.js +163 -0
- package/dist/runtime/commands/share.js +316 -0
- package/dist/runtime/commands/skills.js +31 -31
- package/dist/runtime/commands/status.js +186 -0
- package/dist/runtime/commands/stickers.js +82 -0
- package/dist/runtime/commands/style.js +194 -0
- package/dist/runtime/commands/theme.js +196 -0
- package/dist/runtime/commands/undo.js +54 -22
- package/dist/runtime/commands/update.js +289 -0
- package/dist/runtime/commands/vim.js +140 -0
- package/dist/runtime/commands/worktree.js +8 -8
- package/dist/runtime/commands/worktrees.js +155 -0
- package/dist/runtime/headless-repl.js +195 -0
- package/dist/runtime/headless.js +543 -0
- package/dist/runtime/load-hooks-or-exit.js +71 -0
- package/dist/runtime/plan-decompose.js +22 -22
- package/dist/runtime/sigint-guard.js +272 -0
- package/dist/runtime/update-check.js +28 -28
- package/dist/runtime/version.js +65 -0
- package/dist/runtime/worktree-bootstrap.js +579 -0
- package/dist/skills/bundled/batch.js +617 -0
- package/dist/skills/bundled/index.js +45 -0
- package/dist/skills/bundled/loop.js +358 -0
- package/dist/skills/bundled/remember.js +383 -0
- package/dist/skills/bundled/simplify.js +289 -0
- package/dist/skills/bundled/skillify.js +373 -0
- package/dist/skills/bundled/stuck.js +558 -0
- package/dist/skills/bundled/verify.js +439 -0
- package/dist/testing/vcr.js +486 -0
- package/dist/tools/agent-tool.js +229 -0
- package/dist/tools/apply-patch.js +89 -28
- package/dist/tools/ask-user-question.js +337 -0
- package/dist/tools/ask-user.js +115 -0
- package/dist/tools/bash.js +624 -46
- package/dist/tools/brief.js +224 -0
- package/dist/tools/enter-worktree.js +250 -0
- package/dist/tools/exit-worktree.js +147 -0
- package/dist/tools/file-tools.js +161 -44
- package/dist/tools/lsp-tools.js +377 -1
- package/dist/tools/mcp-tool.js +260 -0
- package/dist/tools/multi-edit.js +361 -0
- package/dist/tools/powershell.js +268 -0
- package/dist/tools/registry.js +86 -4
- package/dist/tools/skill-tool.js +96 -0
- package/dist/tools/sleep.js +99 -0
- package/dist/tools/synthetic-output.js +133 -0
- package/dist/tools/tasks.js +208 -0
- package/dist/tools/todo-write.js +184 -0
- package/dist/tools/verify-plan-execution.js +295 -0
- package/dist/tools/web-fetch-injection-scanner.js +207 -0
- package/dist/tools/web-fetch.js +195 -10
- package/dist/tools/web-search.js +458 -0
- package/dist/tui/agent-progress-card.js +111 -0
- package/dist/tui/agent-tree.js +11 -1
- package/dist/tui/ask-modal.js +14 -14
- package/dist/tui/ask-user-question-chips.js +315 -0
- package/dist/tui/ask-user-question-prompt.js +203 -0
- package/dist/tui/compact-banner.js +81 -0
- package/dist/tui/conversation-pane.js +85 -11
- package/dist/tui/cost-table.js +111 -0
- package/dist/tui/device-flow.js +2 -2
- package/dist/tui/doctor-table.js +46 -0
- package/dist/tui/feedback-prompt.js +156 -0
- package/dist/tui/input-box.js +247 -32
- package/dist/tui/login-picker.js +3 -3
- package/dist/tui/markdown-render.js +6 -6
- package/dist/tui/onboarding-wizard.js +240 -0
- package/dist/tui/permissions-picker.js +86 -0
- package/dist/tui/render.js +36 -1
- package/dist/tui/repl-render.js +176 -25
- package/dist/tui/repl-splash-art.js +16 -16
- package/dist/tui/repl-splash-mascot.js +48 -24
- package/dist/tui/repl-splash.js +22 -22
- package/dist/tui/repl.js +125 -45
- package/dist/tui/slash-palette.js +6 -6
- package/dist/tui/splash.js +2 -2
- package/dist/tui/status-bar.js +109 -31
- package/dist/tui/status-table.js +7 -0
- package/dist/tui/stickers-art.js +136 -0
- package/dist/tui/style-table.js +28 -0
- package/dist/tui/theme-table.js +29 -0
- package/dist/tui/thinking-spinner.js +123 -0
- package/dist/tui/tool-stream-pane.js +53 -4
- package/dist/tui/update-banner.js +27 -2
- package/dist/tui/vim-input.js +267 -0
- package/dist/tui/welcome-banner.js +107 -0
- package/dist/tui/welcome-data.js +293 -0
- package/dist/tui/workspace-context.js +2 -2
- package/package.json +31 -16
- package/test/scenarios/codegen-create-file.scenario.txt +13 -0
- package/test/scenarios/compact-force.scenario.txt +12 -0
- package/test/scenarios/identity.scenario.txt +12 -0
- package/test/scenarios/persona-handoff.scenario.txt +12 -0
- package/test/scenarios/walkback.scenario.txt +12 -0
- package/dist/core/engine/compaction-hook.js +0 -154
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Approximate token counter for the `/compact` auto-trigger gate.
|
|
3
|
+
*
|
|
4
|
+
* Pugi does not bundle tiktoken — adding a 3 MB native module for a
|
|
5
|
+
* single heuristic check after every turn is the wrong trade. We instead
|
|
6
|
+
* use the OpenAI rule-of-thumb that 1 token ≈ 4 characters of English
|
|
7
|
+
* (≈ ¾ of a word). For mixed-language conversations the approximation
|
|
8
|
+
* skews high by 10-20% which is the safe direction — the auto-compact
|
|
9
|
+
* threshold trips slightly EARLY, never LATE. A late trigger would
|
|
10
|
+
* exceed the model's actual context window and crash the next turn.
|
|
11
|
+
*
|
|
12
|
+
* Per-model context windows are exported from `MODEL_CONTEXT_WINDOW` so
|
|
13
|
+
* the caller can resolve the budget from the active model slug without
|
|
14
|
+
* round-tripping to the server. New models added in Anvil should grow
|
|
15
|
+
* this table in lockstep; the fall-back is the conservative 32k window.
|
|
16
|
+
*
|
|
17
|
+
* Override hook: when `PUGI_TOKEN_COUNTER_OVERRIDE` is set the function
|
|
18
|
+
* parses it as a JSON object `{"text": <n>}` (number of tokens to claim
|
|
19
|
+
* per char). Used only by the integration spec — never by production.
|
|
20
|
+
*/
|
|
21
|
+
/** Default chars-per-token ratio for the OpenAI rule-of-thumb. */
|
|
22
|
+
const DEFAULT_CHARS_PER_TOKEN = 4;
|
|
23
|
+
/**
|
|
24
|
+
* Conservative fallback window when the model slug is unknown. 32k is
|
|
25
|
+
* the smallest window any current Anvil-served model exposes; using it
|
|
26
|
+
* as a fall-back guarantees we never overestimate budget and skip a
|
|
27
|
+
* compaction that should have fired.
|
|
28
|
+
*/
|
|
29
|
+
const FALLBACK_WINDOW = 32_000;
|
|
30
|
+
/**
|
|
31
|
+
* Known context windows for models Anvil exposes today. Keep in sync
|
|
32
|
+
* with `apps/admin-api/src/pugi/model-registry.ts` — when a new model
|
|
33
|
+
* lands there, add it here. The table is intentionally narrow: only
|
|
34
|
+
* models we have actually validated.
|
|
35
|
+
*/
|
|
36
|
+
export const MODEL_CONTEXT_WINDOW = Object.freeze({
|
|
37
|
+
'sonnet-4.6': 200_000,
|
|
38
|
+
'sonnet-4.5': 200_000,
|
|
39
|
+
'opus-4.7': 1_000_000,
|
|
40
|
+
'opus-4.6': 200_000,
|
|
41
|
+
'haiku-4.5': 200_000,
|
|
42
|
+
'deepseek-chat-v3.1': 128_000,
|
|
43
|
+
'gpt-5': 200_000,
|
|
44
|
+
'gemini-2.5-pro': 1_000_000,
|
|
45
|
+
});
|
|
46
|
+
/**
|
|
47
|
+
* Estimate the token count of a single string. Returns a positive
|
|
48
|
+
* integer for non-empty input and zero for empty input. The function is
|
|
49
|
+
* pure — same input always yields the same output.
|
|
50
|
+
*
|
|
51
|
+
* The approximation is `ceil(byteLength / chars-per-token)`. We use
|
|
52
|
+
* `Buffer.byteLength` so multi-byte UTF-8 sequences count proportionally
|
|
53
|
+
* to their on-the-wire size, not their char count — this matches the
|
|
54
|
+
* tokenizer's behaviour on CJK / cyrillic / emoji where one char often
|
|
55
|
+
* eats 3-4 tokens.
|
|
56
|
+
*/
|
|
57
|
+
export function estimateTokens(text) {
|
|
58
|
+
if (text.length === 0)
|
|
59
|
+
return 0;
|
|
60
|
+
const ratio = readCharsPerTokenOverride() ?? DEFAULT_CHARS_PER_TOKEN;
|
|
61
|
+
const bytes = Buffer.byteLength(text, 'utf8');
|
|
62
|
+
return Math.max(1, Math.ceil(bytes / ratio));
|
|
63
|
+
}
|
|
64
|
+
/**
|
|
65
|
+
* Resolve the context window in tokens for the given model slug. Falls
|
|
66
|
+
* back to the conservative 32k window when the slug is unknown.
|
|
67
|
+
*/
|
|
68
|
+
export function contextWindowForModel(model) {
|
|
69
|
+
if (!model)
|
|
70
|
+
return FALLBACK_WINDOW;
|
|
71
|
+
const known = MODEL_CONTEXT_WINDOW[model];
|
|
72
|
+
return known ?? FALLBACK_WINDOW;
|
|
73
|
+
}
|
|
74
|
+
/**
|
|
75
|
+
* Sum the token estimates of an arbitrary list of strings. Convenience
|
|
76
|
+
* for the auto-trigger which has to count across N transcript turns.
|
|
77
|
+
*/
|
|
78
|
+
export function estimateTokensInMany(parts) {
|
|
79
|
+
let total = 0;
|
|
80
|
+
for (const part of parts) {
|
|
81
|
+
total += estimateTokens(part);
|
|
82
|
+
}
|
|
83
|
+
return total;
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* Read the test override env. Returns the parsed chars-per-token ratio
|
|
87
|
+
* when set, or undefined when absent / malformed. Never throws.
|
|
88
|
+
*/
|
|
89
|
+
function readCharsPerTokenOverride() {
|
|
90
|
+
const raw = process.env['PUGI_TOKEN_COUNTER_OVERRIDE'];
|
|
91
|
+
if (!raw)
|
|
92
|
+
return undefined;
|
|
93
|
+
try {
|
|
94
|
+
const parsed = JSON.parse(raw);
|
|
95
|
+
if (typeof parsed === 'object'
|
|
96
|
+
&& parsed !== null
|
|
97
|
+
&& typeof parsed.charsPerToken === 'number') {
|
|
98
|
+
const ratio = parsed.charsPerToken;
|
|
99
|
+
if (Number.isFinite(ratio) && ratio > 0)
|
|
100
|
+
return ratio;
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
catch {
|
|
104
|
+
/* malformed env, fall through */
|
|
105
|
+
}
|
|
106
|
+
return undefined;
|
|
107
|
+
}
|
|
108
|
+
//# sourceMappingURL=token-counter.js.map
|
|
@@ -1,21 +1,21 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* Anvil fan-out — `pugi review --consensus`
|
|
2
|
+
* Anvil fan-out — `pugi review --consensus` .
|
|
3
3
|
*
|
|
4
4
|
* Posts the captured diff to Anvil's consensus endpoint and consumes the
|
|
5
5
|
* SSE stream that interleaves per-reviewer events (`type:"verdict"`) and
|
|
6
6
|
* the final consensus event (`type:"consensus"`).
|
|
7
7
|
*
|
|
8
|
-
* Endpoint contract (admin-api side, ships as
|
|
8
|
+
* Endpoint contract (admin-api side, ships as follow-up):
|
|
9
9
|
*
|
|
10
|
-
*
|
|
11
|
-
*
|
|
12
|
-
*
|
|
13
|
-
*
|
|
14
|
-
*
|
|
15
|
-
*
|
|
16
|
-
*
|
|
17
|
-
*
|
|
18
|
-
*
|
|
10
|
+
* POST {apiUrl}/api/pugi/review-consensus
|
|
11
|
+
* Authorization: Bearer {apiKey}
|
|
12
|
+
* Content-Type: application/json
|
|
13
|
+
* Body: { diff, context: { branch, commit, title } }
|
|
14
|
+
* Response: text/event-stream
|
|
15
|
+
* data: { reviewer: "codex"|"claude"|"deepseek", type:"started" }
|
|
16
|
+
* data: { reviewer, type:"verdict", severity:"P0|P1|P2|P3|CLEAN",
|
|
17
|
+
* rawContent:"<reviewer text>", latencyMs, error? }
|
|
18
|
+
* data: { type:"consensus", rubric_verdict, reasoning }
|
|
19
19
|
*
|
|
20
20
|
* The CLI side does NOT trust the server's `rubric_verdict` — we recompute
|
|
21
21
|
* it locally from the per-reviewer verdicts so a malformed / forged server
|
|
@@ -24,15 +24,15 @@
|
|
|
24
24
|
*
|
|
25
25
|
* Graceful degradation:
|
|
26
26
|
*
|
|
27
|
-
*
|
|
28
|
-
*
|
|
29
|
-
*
|
|
30
|
-
*
|
|
31
|
-
*
|
|
32
|
-
*
|
|
33
|
-
*
|
|
27
|
+
* - 404 from runtime → "endpoint_missing" (admin-api endpoint pending,
|
|
28
|
+
* ships CLI-only). Caller falls back to the
|
|
29
|
+
* legacy `pugi review --triple --remote` flow OR
|
|
30
|
+
* prints a "backend not deployed" notice depending
|
|
31
|
+
* on the operator's invocation.
|
|
32
|
+
* - 401/403 / 429 → matching status with an actionable message.
|
|
33
|
+
* - 5xx / timeout → "failed" with the truncated body.
|
|
34
34
|
*
|
|
35
|
-
* Local-first contract
|
|
35
|
+
* Local-first contract : this module never touches the disk,
|
|
36
36
|
* never logs the diff payload, and never retries on transient errors.
|
|
37
37
|
*/
|
|
38
38
|
/**
|
|
@@ -75,7 +75,7 @@ export async function dispatchConsensus(config, request, sink) {
|
|
|
75
75
|
return {
|
|
76
76
|
status: 'endpoint_missing',
|
|
77
77
|
code,
|
|
78
|
-
message: 'POST /api/pugi/review-consensus not deployed on this runtime
|
|
78
|
+
message: 'POST /api/pugi/review-consensus not deployed on this runtime .',
|
|
79
79
|
};
|
|
80
80
|
}
|
|
81
81
|
if (code === 401 || code === 403) {
|
|
@@ -143,11 +143,11 @@ export async function dispatchConsensus(config, request, sink) {
|
|
|
143
143
|
* Async-iterable SSE parser. Spec'd against the
|
|
144
144
|
* [HTML SSE spec](https://html.spec.whatwg.org/multipage/server-sent-events.html):
|
|
145
145
|
*
|
|
146
|
-
*
|
|
147
|
-
*
|
|
148
|
-
*
|
|
149
|
-
*
|
|
150
|
-
*
|
|
146
|
+
* - Events are delimited by a blank line.
|
|
147
|
+
* - Each line is `field:value` (whitespace after `:` stripped).
|
|
148
|
+
* - Multiple `data:` lines in one event concatenate with `\n`.
|
|
149
|
+
* - We only care about `data:` payloads carrying JSON; everything
|
|
150
|
+
* else (event:, id:, retry:) is ignored.
|
|
151
151
|
*
|
|
152
152
|
* The parser tolerates JSON-parse failures by dropping the malformed
|
|
153
153
|
* event and continuing; a single bad event must not block the consensus
|
|
@@ -1,15 +1,19 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* Diff capture — `pugi review --consensus`
|
|
2
|
+
* Diff capture — `pugi review --consensus` .
|
|
3
3
|
*
|
|
4
4
|
* Captures the diff that the consensus fan-out will send to Anvil. Four
|
|
5
5
|
* supported source kinds (in order of precedence):
|
|
6
6
|
*
|
|
7
|
-
*
|
|
8
|
-
*
|
|
9
|
-
*
|
|
10
|
-
*
|
|
11
|
-
*
|
|
12
|
-
*
|
|
7
|
+
* 1. `--pr <number>` — uses `gh pr diff <num>` (gh CLI required).
|
|
8
|
+
* 2. `--commit <sha>` — diff of that commit vs its first parent.
|
|
9
|
+
* When `--base <ref>` is ALSO provided, the
|
|
10
|
+
* diff is the range `<base>..<commit>` instead
|
|
11
|
+
* (mirrors `git diff base..commit` — covers the
|
|
12
|
+
* full PR-style payload, not just the tip).
|
|
13
|
+
* 3. `--branch <name>` — diff of HEAD vs `origin/<name>` merge-base.
|
|
14
|
+
* 4. (default) — diff of HEAD vs `origin/main` merge-base
|
|
15
|
+
* covering BOTH committed-since-base AND
|
|
16
|
+
* uncommitted (staged + working tree) edits.
|
|
13
17
|
*
|
|
14
18
|
* The shape mirrors the existing `performRemoteTripleReview` flow:
|
|
15
19
|
* uncommitted edits are deliberately included by computing the diff
|
|
@@ -95,6 +99,16 @@ export function captureDiff(spec) {
|
|
|
95
99
|
return captureFromPr(cwd, spec.pr);
|
|
96
100
|
}
|
|
97
101
|
if (typeof spec.commit === 'string' && spec.commit.length > 0) {
|
|
102
|
+
// When `--base` is supplied alongside `--commit`, callers want the
|
|
103
|
+
// full PR-style range diff (`base..commit`), not just the tip
|
|
104
|
+
// commit's parent diff. This matches the convention used by
|
|
105
|
+
// `git diff <base>..<commit>` everywhere else in the toolchain and
|
|
106
|
+
// is the verified-correct mode for reviewing a PR head ref. Without
|
|
107
|
+
// this branch, `--base` was silently ignored when `--commit` was
|
|
108
|
+
// present — see feedback_pugi_review_use_range_diff_not_worktree.
|
|
109
|
+
if (typeof spec.baseRef === 'string' && spec.baseRef.length > 0) {
|
|
110
|
+
return captureFromRange(cwd, spec.baseRef, spec.commit);
|
|
111
|
+
}
|
|
98
112
|
return captureFromCommit(cwd, spec.commit);
|
|
99
113
|
}
|
|
100
114
|
if (typeof spec.branch === 'string' && spec.branch.length > 0) {
|
|
@@ -114,7 +128,18 @@ function captureFromPr(cwd, pr) {
|
|
|
114
128
|
// Fetch the PR head into a private ref so we have local objects to
|
|
115
129
|
// diff against. `pull/<num>/head` is GitHub's special refspec exposed
|
|
116
130
|
// to anyone with read access on the repo.
|
|
117
|
-
|
|
131
|
+
//
|
|
132
|
+
// The leading `+` (force-update) is REQUIRED: if a prior invocation
|
|
133
|
+
// died before reaching the `finally` cleanup (SIGKILL, host crash,
|
|
134
|
+
// operator Ctrl-C inside a `try` block in a wrapping caller, hung
|
|
135
|
+
// gh CLI), the tempRef survives on disk. Without `+`, the next
|
|
136
|
+
// `fetch <pr>/head:<tempRef>` aborts with
|
|
137
|
+
// `! [rejected] pull/N/head -> refs/pugi/consensus-pr-N (non-fast-forward)`
|
|
138
|
+
// and the entire consensus run fails for an operator who did nothing
|
|
139
|
+
// wrong. `+` semantics: replace the ref unconditionally — exactly
|
|
140
|
+
// the recovery behavior we want for a sandboxed `refs/pugi/...` slot
|
|
141
|
+
// that no human ever reads.
|
|
142
|
+
safeExec(cwd, 'git', ['fetch', 'origin', `+pull/${pr}/head:${tempRef}`]);
|
|
118
143
|
try {
|
|
119
144
|
// Resolve the base ref to diff against. Prefer the PR's declared
|
|
120
145
|
// base; fall back to `origin/main`. We compute the merge-base so a
|
|
@@ -149,8 +174,10 @@ function captureFromPr(cwd, pr) {
|
|
|
149
174
|
safeExec(cwd, 'git', ['update-ref', '-d', tempRef]);
|
|
150
175
|
}
|
|
151
176
|
catch {
|
|
152
|
-
// Swallow: a leftover ref under refs/pugi/ is harmless. The
|
|
153
|
-
// run
|
|
177
|
+
// Swallow: a leftover ref under refs/pugi/ is harmless. The
|
|
178
|
+
// next run force-overwrites it via `fetch +pull/<n>/head:<ref>`
|
|
179
|
+
// (note the leading `+` in the refspec above), so a stale tempRef
|
|
180
|
+
// never blocks a future invocation even if cleanup never runs.
|
|
154
181
|
}
|
|
155
182
|
}
|
|
156
183
|
}
|
|
@@ -192,6 +219,88 @@ function captureFromCommit(cwd, commit) {
|
|
|
192
219
|
},
|
|
193
220
|
};
|
|
194
221
|
}
|
|
222
|
+
/**
|
|
223
|
+
* Range capture for `--commit <X> --base <Y>` — diff equivalent to
|
|
224
|
+
* `git diff <base>..<commit>`. Used when the operator names BOTH endpoints
|
|
225
|
+
* (typical PR review against a remote head SHA).
|
|
226
|
+
*
|
|
227
|
+
* Critical: this MUST be a pure read-only range diff against named refs.
|
|
228
|
+
* The previous behavior fell through to `captureFromCommit` which only
|
|
229
|
+
* showed the tip commit (`commit~1..commit`) — fine for single-commit
|
|
230
|
+
* review, wrong for multi-commit PRs. Worse, a stale fallback path was
|
|
231
|
+
* sending the working tree diff (`git diff` with no args), which caused
|
|
232
|
+
* every review on to surface identical noise from uncommitted
|
|
233
|
+
* `.gitignore` edits instead of the actual PR contents.
|
|
234
|
+
*
|
|
235
|
+
* Working tree integrity: only `git diff <ref>..<ref>` and metadata
|
|
236
|
+
* `log` / `rev-parse` / `name-rev` are used — none of these touch the
|
|
237
|
+
* index, working tree, or HEAD.
|
|
238
|
+
*/
|
|
239
|
+
function captureFromRange(cwd, baseRef, commit) {
|
|
240
|
+
// Resolve both endpoints up front so an unknown ref errors with a
|
|
241
|
+
// clear message before the diff invocation. `rev-parse` is read-only.
|
|
242
|
+
const fullCommit = safeExec(cwd, 'git', ['rev-parse', commit]).trim();
|
|
243
|
+
if (!fullCommit)
|
|
244
|
+
throw new Error(`Unknown commit ref: ${commit}`);
|
|
245
|
+
// Task #63 — refresh the base ref via `git fetch` BEFORE
|
|
246
|
+
// resolving. Previously a stale local `main` (last fetched days ago)
|
|
247
|
+
// would silently produce a diff containing every commit that landed
|
|
248
|
+
// upstream after the fetch, swamping the model's review с unrelated
|
|
249
|
+
// changes. The fetch is opportunistic: failures (offline, auth) are
|
|
250
|
+
// swallowed so the existing rev-parse path stays the safety net и
|
|
251
|
+
// returns the clear "Unknown base ref" error if the stale local copy
|
|
252
|
+
// is also missing.
|
|
253
|
+
//
|
|
254
|
+
// We fetch ONLY the base ref (not all refs) so the overhead is
|
|
255
|
+
// bounded — typical PR-style review payload, base = main, one ref.
|
|
256
|
+
// `--no-tags --quiet` keeps the network footprint minimal.
|
|
257
|
+
const remoteCandidate = baseRef.includes('/') ? baseRef : `origin/${baseRef}`;
|
|
258
|
+
if (remoteCandidate.startsWith('origin/')) {
|
|
259
|
+
const bareRef = remoteCandidate.slice('origin/'.length);
|
|
260
|
+
safeExecOptional(cwd, 'git', [
|
|
261
|
+
'fetch',
|
|
262
|
+
'--no-tags',
|
|
263
|
+
'--quiet',
|
|
264
|
+
'origin',
|
|
265
|
+
bareRef,
|
|
266
|
+
]);
|
|
267
|
+
}
|
|
268
|
+
// Resolve the base: accept already-qualified refs (`origin/main`,
|
|
269
|
+
// `refs/heads/foo`) and bare branch names. If the bare name isn't
|
|
270
|
+
// locally resolvable, retry against `origin/<name>` — the common
|
|
271
|
+
// CI shape where local main is absent but the remote tracking ref is.
|
|
272
|
+
let resolvedBase = safeExecOptional(cwd, 'git', ['rev-parse', baseRef]).trim();
|
|
273
|
+
let effectiveBase = baseRef;
|
|
274
|
+
if (!resolvedBase && !baseRef.includes('/')) {
|
|
275
|
+
const remoteBase = `origin/${baseRef}`;
|
|
276
|
+
resolvedBase = safeExecOptional(cwd, 'git', ['rev-parse', remoteBase]).trim();
|
|
277
|
+
if (resolvedBase)
|
|
278
|
+
effectiveBase = remoteBase;
|
|
279
|
+
}
|
|
280
|
+
if (!resolvedBase)
|
|
281
|
+
throw new Error(`Unknown base ref: ${baseRef}`);
|
|
282
|
+
const diff = safeExec(cwd, 'git', [
|
|
283
|
+
'diff',
|
|
284
|
+
`${resolvedBase}..${fullCommit}`,
|
|
285
|
+
'--',
|
|
286
|
+
'.',
|
|
287
|
+
...PROTECTED_PATHSPEC_EXCLUDES,
|
|
288
|
+
]);
|
|
289
|
+
const cappedDiff = capDiff(diff);
|
|
290
|
+
const subject = safeExec(cwd, 'git', ['log', '-1', '--pretty=%s', fullCommit]).trim();
|
|
291
|
+
const branch = safeExec(cwd, 'git', ['name-rev', '--name-only', fullCommit]).trim() || 'detached';
|
|
292
|
+
const stats = computeStats(cappedDiff);
|
|
293
|
+
return {
|
|
294
|
+
diff: cappedDiff,
|
|
295
|
+
context: {
|
|
296
|
+
branch,
|
|
297
|
+
commit: shortSha(fullCommit),
|
|
298
|
+
title: subject || `commit ${shortSha(fullCommit)}`,
|
|
299
|
+
ref: `range:${effectiveBase}..${shortSha(fullCommit)}`,
|
|
300
|
+
stats,
|
|
301
|
+
},
|
|
302
|
+
};
|
|
303
|
+
}
|
|
195
304
|
function captureFromBranch(cwd, branch, baseRef) {
|
|
196
305
|
const remoteRef = branch.includes('/') ? branch : `origin/${branch}`;
|
|
197
306
|
const mergeBase = safeExec(cwd, 'git', ['merge-base', baseRef, remoteRef]).trim();
|
|
@@ -227,8 +336,8 @@ function captureFromBase(cwd, baseRef) {
|
|
|
227
336
|
const mergeBase = safeExecOptional(cwd, 'git', ['merge-base', baseRef, 'HEAD']).trim();
|
|
228
337
|
if (mergeBase) {
|
|
229
338
|
// Two parts (non-overlapping):
|
|
230
|
-
//
|
|
231
|
-
//
|
|
339
|
+
// 1. Committed since base: `<base>..HEAD`
|
|
340
|
+
// 2. Uncommitted (staged + working tree as a single union): `git diff HEAD`
|
|
232
341
|
// `git diff HEAD` already reports BOTH staged AND working-tree
|
|
233
342
|
// changes relative to HEAD, so we MUST NOT add a separate
|
|
234
343
|
// `--cached` invocation: doing so emits the same staged hunks
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* Consensus rubric — `pugi review --consensus`
|
|
2
|
+
* Consensus rubric — `pugi review --consensus` .
|
|
3
3
|
*
|
|
4
4
|
* Three independent reviewers (Codex / Claude / DeepSeek) produce findings
|
|
5
5
|
* tagged `[P0]` / `[P1]` / `[P2]` / `[P3]`. The rubric translates the per-
|
|
@@ -7,16 +7,16 @@
|
|
|
7
7
|
*
|
|
8
8
|
* Rubric (verbatim from /triple-review skill + admin-api OES MCP triple_review):
|
|
9
9
|
*
|
|
10
|
-
*
|
|
11
|
-
*
|
|
12
|
-
*
|
|
13
|
-
*
|
|
14
|
-
*
|
|
10
|
+
* any reviewer reports [P0] -> BLOCK
|
|
11
|
+
* two or more reviewers report [P1] -> BLOCK (consensus)
|
|
12
|
+
* exactly one reviewer reports [P1] -> WARN (asymmetric)
|
|
13
|
+
* no reviewer reports [P0] or [P1] -> PASS (P2/P3 only)
|
|
14
|
+
* every reviewer errored -> BLOCK (no signal)
|
|
15
15
|
*
|
|
16
16
|
* The rubric never reads model text beyond the severity markers; the
|
|
17
17
|
* reviewer-side narrative is shown to the operator unchanged. Keeping the
|
|
18
18
|
* verdict deterministic + LLM-free is the entire point of the gate (CEO
|
|
19
|
-
* directive
|
|
19
|
+
* directive): a model that disagrees with the rubric can be
|
|
20
20
|
* audited, a model that produces the verdict cannot.
|
|
21
21
|
*/
|
|
22
22
|
/**
|
|
@@ -35,13 +35,13 @@ const SEVERITY_TOKEN = /\[\s*[Pp]([0-3])\s*\]/g;
|
|
|
35
35
|
* Heuristics (intentionally permissive — different models format very
|
|
36
36
|
* differently and a strict parser would drop signal):
|
|
37
37
|
*
|
|
38
|
-
*
|
|
39
|
-
*
|
|
40
|
-
*
|
|
41
|
-
*
|
|
42
|
-
*
|
|
43
|
-
*
|
|
44
|
-
*
|
|
38
|
+
* 1. Split the text on `[Px]` tokens, preserving the marker.
|
|
39
|
+
* 2. Each marker starts a new finding. The summary is the rest of the
|
|
40
|
+
* same line (and the next line if the first is `:` or empty after
|
|
41
|
+
* stripping whitespace).
|
|
42
|
+
* 3. Empty / whitespace-only summaries are dropped — a bare `[P1]`
|
|
43
|
+
* with no context cannot be acted on, and treating it as a finding
|
|
44
|
+
* would falsely trigger consensus.
|
|
45
45
|
*/
|
|
46
46
|
export function parseFindings(raw) {
|
|
47
47
|
if (typeof raw !== 'string' || raw.length === 0)
|
|
@@ -88,8 +88,8 @@ function extractSummary(slice) {
|
|
|
88
88
|
/**
|
|
89
89
|
* Compute the highest BLOCKING severity from a finding list. Returns
|
|
90
90
|
* `null` when the reviewer is clean for gating purposes, i.e. either:
|
|
91
|
-
*
|
|
92
|
-
*
|
|
91
|
+
* - no findings at all, OR
|
|
92
|
+
* - only P2 / P3 findings (informational, non-blocking by rubric).
|
|
93
93
|
*
|
|
94
94
|
* `null` is the right contract for downstream tooling that gates on
|
|
95
95
|
* "did this reviewer flag anything that should block ship?" - the
|
|
@@ -143,7 +143,7 @@ export function aggregate(verdicts) {
|
|
|
143
143
|
const erroredReviewers = verdicts.filter((v) => v.errored).length;
|
|
144
144
|
// Zero reviewers = zero signal. Falling through to the no-P0/no-P1
|
|
145
145
|
// branch would emit a false PASS — exactly the regression flagged by
|
|
146
|
-
// Codex during PR
|
|
146
|
+
// Codex during PR review. Treat empty input as BLOCK so the gate
|
|
147
147
|
// fails closed when the backend returns no events at all (5xx that
|
|
148
148
|
// somehow drained the SSE, server-side bug, dispatcher misconfig).
|
|
149
149
|
if (totalReviewers === 0) {
|
|
@@ -214,11 +214,11 @@ export function aggregate(verdicts) {
|
|
|
214
214
|
}
|
|
215
215
|
/**
|
|
216
216
|
* Map a rubric verdict to the conventional exit code Pugi CLI uses for
|
|
217
|
-
* gates (spec
|
|
217
|
+
* gates (spec ):
|
|
218
218
|
*
|
|
219
|
-
*
|
|
220
|
-
*
|
|
221
|
-
*
|
|
219
|
+
* PASS -> 0
|
|
220
|
+
* WARN -> 1
|
|
221
|
+
* BLOCK -> 2
|
|
222
222
|
*
|
|
223
223
|
* The non-zero codes are distinct so a shell script can branch on the
|
|
224
224
|
* exact outcome without re-parsing stdout.
|
|
@@ -5,11 +5,11 @@
|
|
|
5
5
|
* Per `docs/research/pugi-cli-corpus/patterns/context-compaction.md` §6,
|
|
6
6
|
* static and dynamic context live in different sections:
|
|
7
7
|
*
|
|
8
|
-
*
|
|
9
|
-
*
|
|
10
|
-
*
|
|
11
|
-
*
|
|
12
|
-
*
|
|
8
|
+
* STATIC DYNAMIC
|
|
9
|
+
* - system instructions - transcript turns (recent)
|
|
10
|
+
* - tool schemas (sorted) - session memory ref
|
|
11
|
+
* - safety rules - open task graph snapshot
|
|
12
|
+
* - PUGI.md + AGENTS.md
|
|
13
13
|
*
|
|
14
14
|
* Static blocks hash deterministically; identical session starts emit
|
|
15
15
|
* identical static prefixes, hitting the model provider's prompt cache.
|
|
@@ -50,7 +50,7 @@ export async function buildContext(input) {
|
|
|
50
50
|
bytes: Buffer.byteLength(instructionsContent, 'utf8'),
|
|
51
51
|
});
|
|
52
52
|
// 3. Safety rules — optional, hashed into the instructions block via
|
|
53
|
-
//
|
|
53
|
+
// concatenation when present so consumers can read both as one.
|
|
54
54
|
if (input.safetyRules) {
|
|
55
55
|
const safety = normalizeNewlines(input.safetyRules);
|
|
56
56
|
blocks.push({
|
|
@@ -3,11 +3,11 @@
|
|
|
3
3
|
* session's events.jsonl using the shared AuditEvent discriminated union
|
|
4
4
|
* in `@pugi/sdk/audit-trace.ts`.
|
|
5
5
|
*
|
|
6
|
-
* Until PR
|
|
6
|
+
* Until PR these events were written with the ad-hoc shape
|
|
7
7
|
* `{ id, ts, sessionId, ... }` and the comment in this header claimed
|
|
8
8
|
* "we keep `session.ts` untouched and write our events through a
|
|
9
9
|
* dedicated helper that bypasses the AuditEvent schema". The downside,
|
|
10
|
-
* caught by Claude review on PR
|
|
10
|
+
* caught by Claude review on PR, is that every consumer that
|
|
11
11
|
* calls `auditEventSchema.parse(line)` (cabinet UI, replay tooling,
|
|
12
12
|
* `core/index-store.ts`) would throw on every compaction event, and
|
|
13
13
|
* `safeParse` consumers would silently drop them.
|
|
@@ -19,12 +19,12 @@
|
|
|
19
19
|
*
|
|
20
20
|
* Events emitted:
|
|
21
21
|
*
|
|
22
|
-
*
|
|
23
|
-
*
|
|
24
|
-
*
|
|
25
|
-
*
|
|
26
|
-
*
|
|
27
|
-
*
|
|
22
|
+
* compaction.started { sessionId, tier, trigger }
|
|
23
|
+
* compaction.completed { sessionId, tier, bytesReclaimed,
|
|
24
|
+
* newContextSize, artifactsCreated }
|
|
25
|
+
* compaction.skipped { sessionId, tier, reason }
|
|
26
|
+
* compaction.invariant_violated { sessionId, invariant, evidence,
|
|
27
|
+
* artifactRef? }
|
|
28
28
|
*
|
|
29
29
|
* All four are appended in mode 0o600 and respect the
|
|
30
30
|
* `session.enabled` flag (no-op when `.pugi/` is absent).
|
|
@@ -2,42 +2,42 @@
|
|
|
2
2
|
* Six-tier context compaction engine for the Pugi CLI agent loop.
|
|
3
3
|
*
|
|
4
4
|
* Spec: `docs/research/pugi-cli-corpus/patterns/context-compaction.md`,
|
|
5
|
-
* sprint slot:
|
|
5
|
+
* sprint slot: §.
|
|
6
6
|
*
|
|
7
7
|
* Tiers and triggers (selectTier rules):
|
|
8
8
|
*
|
|
9
|
-
*
|
|
10
|
-
*
|
|
11
|
-
*
|
|
12
|
-
*
|
|
13
|
-
*
|
|
14
|
-
*
|
|
15
|
-
*
|
|
9
|
+
* pressure | tier
|
|
10
|
+
* --------------+----------------------------------------------------
|
|
11
|
+
* < 0.5 | microcompact (only if redundant tool outputs)
|
|
12
|
+
* 0.5 .. 0.7 | cached_microcompact
|
|
13
|
+
* 0.7 .. 0.85 | reactive_summary
|
|
14
|
+
* 0.85 .. 0.95 | full_compaction
|
|
15
|
+
* > 0.95 | reset (with checkpoint)
|
|
16
16
|
*
|
|
17
17
|
* Tier behaviours (per pattern card §3):
|
|
18
18
|
*
|
|
19
|
-
*
|
|
20
|
-
*
|
|
21
|
-
*
|
|
22
|
-
*
|
|
23
|
-
*
|
|
24
|
-
*
|
|
25
|
-
*
|
|
26
|
-
*
|
|
27
|
-
*
|
|
28
|
-
*
|
|
29
|
-
*
|
|
30
|
-
*
|
|
31
|
-
*
|
|
32
|
-
*
|
|
33
|
-
*
|
|
34
|
-
*
|
|
35
|
-
*
|
|
36
|
-
*
|
|
37
|
-
*
|
|
38
|
-
*
|
|
39
|
-
*
|
|
40
|
-
*
|
|
19
|
+
* 1. microcompact — sync; strip redundant token deltas,
|
|
20
|
+
* collapse repeated status lines, dedupe
|
|
21
|
+
* identical tool argument echoes; keep
|
|
22
|
+
* tool RESULTS verbatim; target 10-20%
|
|
23
|
+
* 2. cached_microcompact — sync; replace inline tool output with
|
|
24
|
+
* { artifactRef, size } when an artifact
|
|
25
|
+
* with matching sha256 already exists;
|
|
26
|
+
* target 30-50% on repetitive sessions
|
|
27
|
+
* 3. reactive_summary — async-shaped; summarize the last N=10
|
|
28
|
+
* turns into a structured turn-summary
|
|
29
|
+
* artifact; replace those turns with a
|
|
30
|
+
* single turn_summary event
|
|
31
|
+
* 4. session_memory — async-shaped; distill long-running build
|
|
32
|
+
* state into .pugi/session.db (or jsonl
|
|
33
|
+
* fallback if SQLite not yet present)
|
|
34
|
+
* 5. full_compaction — sync, slow; rebuild from event log +
|
|
35
|
+
* artifacts + session_memory + PUGI.md;
|
|
36
|
+
* keep open decisions, FSM state, active
|
|
37
|
+
* tool calls, last 3 turns verbatim
|
|
38
|
+
* 6. reset — manual or >0.95; save full state to
|
|
39
|
+
* .pugi/checkpoints/<name>/ and start
|
|
40
|
+
* fresh with only PUGI.md + session_memory
|
|
41
41
|
*
|
|
42
42
|
* The compaction NEVER touches static blocks. Invariants enforce that
|
|
43
43
|
* (static-hash-unchanged) plus secrets-never-summarize and
|
|
@@ -316,7 +316,7 @@ function tierSessionMemory(input) {
|
|
|
316
316
|
const summaryText = JSON.stringify(memory, null, 2);
|
|
317
317
|
const artifactsCreated = [];
|
|
318
318
|
if (input.workspaceRoot) {
|
|
319
|
-
// SQLite migration arrives in
|
|
319
|
+
// SQLite migration arrives in (per spec). Until then we append
|
|
320
320
|
// a JSONL line to .pugi/session-memory.jsonl, which the next session
|
|
321
321
|
// bootstraps into context.
|
|
322
322
|
const path = resolve(input.workspaceRoot, '.pugi', 'session-memory.jsonl');
|
|
@@ -1,16 +1,16 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* Three-tier context model -
|
|
2
|
+
* Three-tier context model - Phase 1 barrel.
|
|
3
3
|
*
|
|
4
4
|
* Bundles the four primitives so the REPL bootstrap can import from a
|
|
5
5
|
* single path:
|
|
6
6
|
*
|
|
7
|
-
*
|
|
8
|
-
*
|
|
9
|
-
*
|
|
10
|
-
*
|
|
11
|
-
*
|
|
12
|
-
*
|
|
13
|
-
*
|
|
7
|
+
* import {
|
|
8
|
+
* loadPugiIgnore,
|
|
9
|
+
* buildRepoSkeleton,
|
|
10
|
+
* renderSkeleton,
|
|
11
|
+
* WorkingSet,
|
|
12
|
+
* PugiWatcher,
|
|
13
|
+
* } from '../context/index.js';
|
|
14
14
|
*
|
|
15
15
|
* No new logic lives here - just re-exports.
|
|
16
16
|
*/
|
|
@@ -18,4 +18,11 @@ export { BASELINE_IGNORE_PATTERNS, SECRET_IGNORE_PATTERNS, globalPugiIgnorePath,
|
|
|
18
18
|
export { COLLAPSE_DIR_ENTRIES, MAX_README_LINES, MAX_SKELETON_BYTES, MAX_TREE_DEPTH, MAX_WALK_NODES, TOP_LANGUAGES, buildRepoSkeleton, detectPackageManager, languageForExtension, readGitBranch, readPackageJson, readReadme, renderSkeleton, topLanguages, } from './repo-skeleton.js';
|
|
19
19
|
export { DEFAULT_WORKING_SET_CAPACITY, WorkingSet, } from './working-set.js';
|
|
20
20
|
export { MAX_WATCHED_PATHS, PugiWatcher, THROTTLE_WINDOW_MS, } from './watcher.js';
|
|
21
|
+
/**
|
|
22
|
+
* β5a R4+P5 — per-directory PUGI.md / AGENTS.md / CLAUDE.md / GEMINI.md
|
|
23
|
+
* traverse-up. Loads agent-context markdown at every directory between
|
|
24
|
+
* `cwd` and `workspaceRoot` (workspace root file is owned by
|
|
25
|
+
* `loadMarkdownContext` in `markdown-loader.ts` — no double-load).
|
|
26
|
+
*/
|
|
27
|
+
export { MAX_TRAVERSE_BYTES, MAX_TRAVERSE_PER_FILE_BYTES, MAX_TRAVERSE_DEPTH, TRAVERSE_SOURCES, loadTraversedMarkdown, } from './markdown-traverse.js';
|
|
21
28
|
//# sourceMappingURL=index.js.map
|