@pugi/cli 0.1.0-beta.98 → 1.0.0-alpha.1
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/LICENSE +1 -1
- package/README.md +11 -191
- package/bin/pugi +8 -0
- package/package.json +15 -71
- package/postinstall.mjs +31 -0
- package/CHANGELOG.md +0 -132
- package/THIRD_PARTY_NOTICES.md +0 -40
- package/assets/pugi-mascot.ansi +0 -16
- package/assets/pugi-prozr2-mascot.ansi +0 -9
- package/bin/run.js +0 -34
- package/dist/commands/deploy.js +0 -439
- package/dist/commands/flatten.js +0 -191
- package/dist/commands/jobs-watch.js +0 -201
- package/dist/commands/jobs.js +0 -260
- package/dist/commands/retro.js +0 -210
- package/dist/commands/smoke.js +0 -133
- package/dist/core/agent-progress/cleanup.js +0 -134
- package/dist/core/agent-progress/schema.js +0 -144
- package/dist/core/agent-progress/writer.js +0 -101
- package/dist/core/agents/adaptive-router.js +0 -330
- package/dist/core/agents/loader.js +0 -104
- package/dist/core/agents/query-decomposer.js +0 -297
- package/dist/core/agents/registry.js +0 -69
- package/dist/core/approvals/shortcut-resolver.js +0 -98
- package/dist/core/artifact-chain/dispatcher.js +0 -148
- package/dist/core/artifact-chain/exporter.js +0 -164
- package/dist/core/artifact-chain/state.js +0 -243
- package/dist/core/artifact-chain/steps.js +0 -169
- package/dist/core/ask-user/question.js +0 -92
- package/dist/core/audit/audit-trail.js +0 -275
- package/dist/core/auth/ensure-authenticated.js +0 -129
- package/dist/core/auth/env-provider.js +0 -238
- package/dist/core/auto-open-browser.js +0 -128
- package/dist/core/auto-update/channels.js +0 -122
- package/dist/core/auto-update/checker.js +0 -241
- package/dist/core/auto-update/state.js +0 -235
- package/dist/core/bare-mode/index.js +0 -107
- package/dist/core/bash/redirect.js +0 -281
- package/dist/core/bash-classifier.js +0 -1397
- package/dist/core/checkpoint/resumer.js +0 -149
- package/dist/core/checkpoint/rewinder.js +0 -291
- package/dist/core/checkpoints/shadow-git.js +0 -670
- package/dist/core/citations/parser.js +0 -109
- package/dist/core/classifier/yolo-classifier.js +0 -88
- package/dist/core/clipboard.js +0 -70
- package/dist/core/codegraph/decision-store.js +0 -248
- package/dist/core/codegraph/detect-repo.js +0 -459
- package/dist/core/codegraph/install.js +0 -134
- package/dist/core/codegraph/offer-hook.js +0 -220
- package/dist/core/compact/auto-trigger.js +0 -96
- package/dist/core/compact/buffer-rewriter.js +0 -115
- package/dist/core/compact/summarizer.js +0 -208
- package/dist/core/compact/token-counter.js +0 -108
- package/dist/core/consensus/anvil-fanout.js +0 -276
- package/dist/core/consensus/diff-capture.js +0 -491
- package/dist/core/consensus/rubric.js +0 -233
- package/dist/core/context/builder.js +0 -114
- package/dist/core/context/compaction-events.js +0 -99
- package/dist/core/context/compaction.js +0 -602
- package/dist/core/context/index.js +0 -28
- package/dist/core/context/invariants.js +0 -250
- package/dist/core/context/markdown-loader.js +0 -288
- package/dist/core/context/markdown-traverse.js +0 -255
- package/dist/core/context/pugiignore.js +0 -316
- package/dist/core/context/repo-skeleton.js +0 -533
- package/dist/core/context/tool-eviction.js +0 -55
- package/dist/core/context/watcher.js +0 -342
- package/dist/core/context/working-set.js +0 -165
- package/dist/core/coordinator/agent-tools.js +0 -77
- package/dist/core/coordinator/agent-toolset.js +0 -65
- package/dist/core/coordinator/fsm.js +0 -73
- package/dist/core/coordinator/mode-fsm.js +0 -70
- package/dist/core/cost/rate-card.js +0 -129
- package/dist/core/cost/tracker.js +0 -221
- package/dist/core/credentials.js +0 -355
- package/dist/core/cron/scheduler.js +0 -138
- package/dist/core/denial-tracking/index.js +0 -8
- package/dist/core/denial-tracking/state.js +0 -264
- package/dist/core/diagnostics/probe-runner.js +0 -93
- package/dist/core/diagnostics/probes/api.js +0 -46
- package/dist/core/diagnostics/probes/auth.js +0 -93
- package/dist/core/diagnostics/probes/bare-mode.js +0 -42
- package/dist/core/diagnostics/probes/cli-version.js +0 -127
- package/dist/core/diagnostics/probes/config.js +0 -72
- package/dist/core/diagnostics/probes/denial-tracking.js +0 -57
- package/dist/core/diagnostics/probes/disk.js +0 -81
- package/dist/core/diagnostics/probes/engine-live.js +0 -46
- package/dist/core/diagnostics/probes/git.js +0 -65
- package/dist/core/diagnostics/probes/hooks.js +0 -118
- package/dist/core/diagnostics/probes/mcp.js +0 -75
- package/dist/core/diagnostics/probes/node.js +0 -59
- package/dist/core/diagnostics/probes/pnpm.js +0 -36
- package/dist/core/diagnostics/probes/pugi-md.js +0 -89
- package/dist/core/diagnostics/probes/sandbox.js +0 -72
- package/dist/core/diagnostics/probes/session.js +0 -74
- package/dist/core/diagnostics/probes/status-snapshot.js +0 -488
- package/dist/core/diagnostics/probes/workspace.js +0 -63
- package/dist/core/diagnostics/types.js +0 -70
- package/dist/core/dispatch/cache-cleanup.js +0 -197
- package/dist/core/dispatch/cache-handoff.js +0 -295
- package/dist/core/edits/apply-patch-layer-e.js +0 -189
- package/dist/core/edits/dispatch.js +0 -511
- package/dist/core/edits/format-detector.js +0 -260
- package/dist/core/edits/format-matrix.js +0 -26
- package/dist/core/edits/fuzzy-ladder.js +0 -650
- package/dist/core/edits/index.js +0 -19
- package/dist/core/edits/journal.js +0 -199
- package/dist/core/edits/layer-a-apply.js +0 -217
- package/dist/core/edits/layer-a-fuzzy-apply.js +0 -198
- package/dist/core/edits/layer-b-apply.js +0 -211
- package/dist/core/edits/layer-c-apply.js +0 -160
- package/dist/core/edits/layer-d-ast.js +0 -572
- package/dist/core/edits/marker-parser.js +0 -401
- package/dist/core/edits/security-gate.js +0 -223
- package/dist/core/edits/verify-hook.js +0 -273
- package/dist/core/edits/worktree.js +0 -322
- package/dist/core/engine/adapter-runner.js +0 -8
- package/dist/core/engine/anvil-client.js +0 -344
- package/dist/core/engine/auto-compact.js +0 -179
- package/dist/core/engine/budgets.js +0 -192
- package/dist/core/engine/context-prefix.js +0 -155
- package/dist/core/engine/index.js +0 -12
- package/dist/core/engine/intensity.js +0 -163
- package/dist/core/engine/intent.js +0 -260
- package/dist/core/engine/native-pugi.js +0 -1616
- package/dist/core/engine/noop.js +0 -27
- package/dist/core/engine/prompts.js +0 -236
- package/dist/core/engine/strip-internal-fields.js +0 -124
- package/dist/core/engine/tool-bridge.js +0 -2173
- package/dist/core/engine/verification-patterns.js +0 -195
- package/dist/core/evaluation/golden-dataset.js +0 -293
- package/dist/core/feedback/queue.js +0 -177
- package/dist/core/feedback/submitter.js +0 -145
- package/dist/core/file-cache.js +0 -141
- package/dist/core/flatten/flatten-repo.js +0 -439
- package/dist/core/format/osc8-link.js +0 -28
- package/dist/core/hook-chains.js +0 -392
- package/dist/core/hooks/citation-verify-hook.js +0 -138
- package/dist/core/hooks/citation-verify.js +0 -112
- package/dist/core/hooks/events.js +0 -46
- package/dist/core/hooks/index.js +0 -15
- package/dist/core/hooks/registry.js +0 -216
- package/dist/core/hooks/runner.js +0 -236
- package/dist/core/hooks/v2/event-emitter.js +0 -115
- package/dist/core/hooks/v2/executor.js +0 -282
- package/dist/core/hooks/v2/index.js +0 -25
- package/dist/core/hooks/v2/lifecycle.js +0 -104
- package/dist/core/hooks/v2/loader.js +0 -216
- package/dist/core/hooks/v2/matcher.js +0 -125
- package/dist/core/hooks/v2/trust.js +0 -143
- package/dist/core/hooks/v2/types.js +0 -86
- package/dist/core/hooks/worktree-events.js +0 -158
- package/dist/core/hooks.js +0 -415
- package/dist/core/image/renderer.js +0 -71
- package/dist/core/index-store.js +0 -260
- package/dist/core/init/detector.js +0 -582
- package/dist/core/init/template-renderer.js +0 -242
- package/dist/core/jobs/registry.js +0 -462
- package/dist/core/ledger/results-tsv.js +0 -142
- package/dist/core/log-discipline/stdout-redirect.js +0 -51
- package/dist/core/lsp/cache.js +0 -105
- package/dist/core/lsp/client.js +0 -1229
- package/dist/core/lsp/language-detect.js +0 -66
- package/dist/core/lsp/post-edit-diagnostics.js +0 -171
- package/dist/core/lsp/server-detect.js +0 -173
- package/dist/core/lsp/symbol-cache.js +0 -162
- package/dist/core/lsp/symbol-tools.js +0 -664
- package/dist/core/mcp/client.js +0 -385
- package/dist/core/mcp/http-server.js +0 -553
- package/dist/core/mcp/orchestrator-config.js +0 -192
- package/dist/core/mcp/orchestrator-tools.js +0 -806
- package/dist/core/mcp/permission.js +0 -190
- package/dist/core/mcp/registry.js +0 -193
- package/dist/core/mcp/server-tools.js +0 -219
- package/dist/core/mcp/server.js +0 -397
- package/dist/core/mcp/trust.js +0 -91
- package/dist/core/memory/dual-write.js +0 -416
- package/dist/core/memory/passive-extract.js +0 -130
- package/dist/core/memory/phase1-kinds.js +0 -20
- package/dist/core/memory/secret-scanner.js +0 -304
- package/dist/core/memory-sync/queue.js +0 -170
- package/dist/core/metrics/extract.js +0 -113
- package/dist/core/modes/roo-modes.js +0 -68
- package/dist/core/onboarding/ensure-initialized.js +0 -133
- package/dist/core/onboarding/marker.js +0 -111
- package/dist/core/onboarding/telemetry-state.js +0 -108
- package/dist/core/output-style/presets.js +0 -176
- package/dist/core/output-style/state.js +0 -185
- package/dist/core/path-security.js +0 -345
- package/dist/core/permission.js +0 -369
- package/dist/core/permissions/auto-classifier.js +0 -124
- package/dist/core/permissions/bash-parser.js +0 -371
- package/dist/core/permissions/circuit-breaker.js +0 -83
- package/dist/core/permissions/constrained-edit.js +0 -91
- package/dist/core/permissions/gate.js +0 -278
- package/dist/core/permissions/index.js +0 -20
- package/dist/core/permissions/mode.js +0 -174
- package/dist/core/permissions/network-egress.js +0 -137
- package/dist/core/permissions/state.js +0 -241
- package/dist/core/permissions/tool-class.js +0 -107
- package/dist/core/plan-mode/ui-state.js +0 -51
- package/dist/core/plans/plan-artifact.js +0 -721
- package/dist/core/policy-limits/etag-store.js +0 -122
- package/dist/core/prd-check/parser.js +0 -215
- package/dist/core/prd-check/reporter.js +0 -127
- package/dist/core/prd-check/session-review.js +0 -557
- package/dist/core/prd-check/verifiers.js +0 -223
- package/dist/core/prompt-cache/client-cache.js +0 -99
- package/dist/core/prompts/assembly.js +0 -29
- package/dist/core/prompts/registry.js +0 -364
- package/dist/core/pugi-gitignore.js +0 -52
- package/dist/core/pugi-md/cc-compat-rules.js +0 -735
- package/dist/core/pugi-md/context-injector.js +0 -76
- package/dist/core/pugi-md/walk-up.js +0 -207
- package/dist/core/python/uv-installer.js +0 -270
- package/dist/core/python/uv-resolver.js +0 -83
- package/dist/core/rate-limit/narrator.js +0 -146
- package/dist/core/recipes/cli-types.js +0 -20
- package/dist/core/recipes/loader.js +0 -103
- package/dist/core/recipes/runner.js +0 -345
- package/dist/core/recipes/schema.js +0 -587
- package/dist/core/release-notes/parser.js +0 -241
- package/dist/core/release-notes/state.js +0 -116
- package/dist/core/repl/ask.js +0 -512
- package/dist/core/repl/cancellation.js +0 -98
- package/dist/core/repl/cap-warning.js +0 -91
- package/dist/core/repl/clipboard-read.js +0 -174
- package/dist/core/repl/dispatch-fsm.js +0 -220
- package/dist/core/repl/engine-bridge.js +0 -303
- package/dist/core/repl/history-search.js +0 -175
- package/dist/core/repl/history.js +0 -182
- package/dist/core/repl/kill-ring.js +0 -138
- package/dist/core/repl/model-pricing.js +0 -135
- package/dist/core/repl/privacy-banner.js +0 -71
- package/dist/core/repl/session.js +0 -4962
- package/dist/core/repl/slash-commands.js +0 -747
- package/dist/core/repl/store/index.js +0 -12
- package/dist/core/repl/store/jsonl-log.js +0 -321
- package/dist/core/repl/store/lockfile.js +0 -155
- package/dist/core/repl/store/session-store.js +0 -821
- package/dist/core/repl/store/types.js +0 -44
- package/dist/core/repl/store/uuid-v7.js +0 -68
- package/dist/core/repl/tool-route.js +0 -382
- package/dist/core/repl/workspace-context.js +0 -206
- package/dist/core/repo-map/build.js +0 -125
- package/dist/core/repo-map/cache.js +0 -185
- package/dist/core/repo-map/extractor.js +0 -254
- package/dist/core/repo-map/formatter.js +0 -145
- package/dist/core/repo-map/page-rank.js +0 -105
- package/dist/core/repo-map/scanner.js +0 -211
- package/dist/core/retro/git-collector.js +0 -251
- package/dist/core/retro/health-card.js +0 -25
- package/dist/core/retro/metrics.js +0 -342
- package/dist/core/retro/narrative.js +0 -249
- package/dist/core/retro/plane-collector.js +0 -274
- package/dist/core/retro/pr-issue-link.js +0 -65
- package/dist/core/retro/types.js +0 -16
- package/dist/core/retry-budget/budget.js +0 -284
- package/dist/core/retry-budget/index.js +0 -5
- package/dist/core/retry-budget/retry-cap.js +0 -74
- package/dist/core/routing/lead-worker.js +0 -43
- package/dist/core/routing/pre-flight-estimator.js +0 -108
- package/dist/core/runs/run-tree.js +0 -103
- package/dist/core/sandboxing/adapter.js +0 -29
- package/dist/core/sandboxing/index.js +0 -49
- package/dist/core/sandboxing/none.js +0 -19
- package/dist/core/sandboxing/seatbelt.js +0 -183
- package/dist/core/security/injection-scanner.js +0 -367
- package/dist/core/security/output-filter.js +0 -418
- package/dist/core/session/env-file.js +0 -105
- package/dist/core/session/section-budgets.js +0 -140
- package/dist/core/session.js +0 -377
- package/dist/core/settings.js +0 -400
- package/dist/core/share/formatter.js +0 -271
- package/dist/core/share/redactor.js +0 -221
- package/dist/core/share/uploader.js +0 -267
- package/dist/core/skills/defaults.js +0 -457
- package/dist/core/skills/loader.js +0 -454
- package/dist/core/skills/sources.js +0 -480
- package/dist/core/skills/trust.js +0 -172
- package/dist/core/smoke/headless-driver.js +0 -174
- package/dist/core/smoke/orchestrator.js +0 -194
- package/dist/core/smoke/runner.js +0 -238
- package/dist/core/smoke/scenario-parser.js +0 -316
- package/dist/core/statusline.js +0 -99
- package/dist/core/subagents/dispatcher-real.js +0 -600
- package/dist/core/subagents/dispatcher.js +0 -352
- package/dist/core/subagents/index.js +0 -39
- package/dist/core/subagents/isolation-matrix.js +0 -213
- package/dist/core/subagents/spawn.js +0 -101
- package/dist/core/telemetry/emitter.js +0 -229
- package/dist/core/telemetry/queue.js +0 -251
- package/dist/core/theme/context.js +0 -91
- package/dist/core/theme/presets.js +0 -228
- package/dist/core/theme/state.js +0 -181
- package/dist/core/todos/invariant.js +0 -10
- package/dist/core/todos/state.js +0 -177
- package/dist/core/tool-schema/compressor.js +0 -89
- package/dist/core/transport/version-interceptor.js +0 -166
- package/dist/core/trust.js +0 -109
- package/dist/core/tui/thinking-block.js +0 -64
- package/dist/core/vim/keymap.js +0 -288
- package/dist/core/vim/state.js +0 -92
- package/dist/core/watch-markers/marker-watcher.js +0 -133
- package/dist/core/worktree/include-parser.js +0 -249
- package/dist/core/worktree-manager/cleanup.js +0 -123
- package/dist/core/worktree-manager/manager.js +0 -303
- package/dist/index.js +0 -44
- package/dist/runtime/bootstrap.js +0 -190
- package/dist/runtime/cli.js +0 -8121
- package/dist/runtime/commands/agents.js +0 -385
- package/dist/runtime/commands/budget.js +0 -192
- package/dist/runtime/commands/cancel.js +0 -231
- package/dist/runtime/commands/chain.js +0 -489
- package/dist/runtime/commands/codegraph-status.js +0 -227
- package/dist/runtime/commands/compact.js +0 -297
- package/dist/runtime/commands/config.js +0 -595
- package/dist/runtime/commands/cost.js +0 -199
- package/dist/runtime/commands/delegate.js +0 -312
- package/dist/runtime/commands/dispatch.js +0 -126
- package/dist/runtime/commands/doctor.js +0 -579
- package/dist/runtime/commands/feedback.js +0 -184
- package/dist/runtime/commands/hooks.js +0 -187
- package/dist/runtime/commands/init.js +0 -254
- package/dist/runtime/commands/lsp.js +0 -368
- package/dist/runtime/commands/mcp.js +0 -935
- package/dist/runtime/commands/memory.js +0 -582
- package/dist/runtime/commands/model.js +0 -237
- package/dist/runtime/commands/onboarding.js +0 -275
- package/dist/runtime/commands/patch.js +0 -128
- package/dist/runtime/commands/permissions.js +0 -112
- package/dist/runtime/commands/plan.js +0 -143
- package/dist/runtime/commands/prd-check.js +0 -285
- package/dist/runtime/commands/privacy.js +0 -107
- package/dist/runtime/commands/recipe.js +0 -325
- package/dist/runtime/commands/redo-blob-store.js +0 -92
- package/dist/runtime/commands/redo.js +0 -361
- package/dist/runtime/commands/release-notes.js +0 -229
- package/dist/runtime/commands/repo-map.js +0 -95
- package/dist/runtime/commands/report.js +0 -299
- package/dist/runtime/commands/resume.js +0 -118
- package/dist/runtime/commands/review-consensus.js +0 -414
- package/dist/runtime/commands/rewind.js +0 -333
- package/dist/runtime/commands/roster.js +0 -117
- package/dist/runtime/commands/sessions.js +0 -163
- package/dist/runtime/commands/share.js +0 -316
- package/dist/runtime/commands/skills.js +0 -401
- package/dist/runtime/commands/status.js +0 -186
- package/dist/runtime/commands/stickers.js +0 -82
- package/dist/runtime/commands/style.js +0 -194
- package/dist/runtime/commands/theme.js +0 -196
- package/dist/runtime/commands/undo.js +0 -361
- package/dist/runtime/commands/update.js +0 -289
- package/dist/runtime/commands/vim.js +0 -140
- package/dist/runtime/commands/worktree.js +0 -177
- package/dist/runtime/commands/worktrees.js +0 -155
- package/dist/runtime/deprecation-warning.js +0 -69
- package/dist/runtime/engine-exit-code.js +0 -50
- package/dist/runtime/headless-repl.js +0 -195
- package/dist/runtime/headless.js +0 -548
- package/dist/runtime/load-hooks-or-exit.js +0 -71
- package/dist/runtime/plan-decompose.js +0 -531
- package/dist/runtime/sigint-guard.js +0 -272
- package/dist/runtime/stream-renderer.js +0 -195
- package/dist/runtime/update-check.js +0 -294
- package/dist/runtime/version.js +0 -65
- package/dist/runtime/worktree-bootstrap.js +0 -579
- package/dist/skills/bundled/batch.js +0 -617
- package/dist/skills/bundled/index.js +0 -45
- package/dist/skills/bundled/loop.js +0 -358
- package/dist/skills/bundled/remember.js +0 -383
- package/dist/skills/bundled/simplify.js +0 -289
- package/dist/skills/bundled/skillify.js +0 -373
- package/dist/skills/bundled/stuck.js +0 -558
- package/dist/skills/bundled/verify.js +0 -439
- package/dist/testing/vcr.js +0 -486
- package/dist/tools/agent-tool.js +0 -229
- package/dist/tools/apply-patch.js +0 -556
- package/dist/tools/ask-user-question.js +0 -337
- package/dist/tools/ask-user.js +0 -115
- package/dist/tools/bash.js +0 -1238
- package/dist/tools/brief.js +0 -224
- package/dist/tools/cron.js +0 -433
- package/dist/tools/enter-worktree.js +0 -250
- package/dist/tools/exit-worktree.js +0 -147
- package/dist/tools/file-tools.js +0 -553
- package/dist/tools/http-request.js +0 -336
- package/dist/tools/lsp-tools.js +0 -565
- package/dist/tools/mcp-tool.js +0 -260
- package/dist/tools/multi-edit.js +0 -361
- package/dist/tools/powershell.js +0 -268
- package/dist/tools/registry.js +0 -166
- package/dist/tools/server-tools.js +0 -892
- package/dist/tools/skill-tool.js +0 -96
- package/dist/tools/sleep.js +0 -99
- package/dist/tools/synthetic-output.js +0 -133
- package/dist/tools/tasks.js +0 -208
- package/dist/tools/todo-write.js +0 -184
- package/dist/tools/verify-plan-execution.js +0 -295
- package/dist/tools/web-fetch-injection-scanner.js +0 -207
- package/dist/tools/web-fetch.js +0 -720
- package/dist/tools/web-search.js +0 -458
- package/dist/tui/agent-progress-card.js +0 -111
- package/dist/tui/agent-tree-pane.js +0 -9
- package/dist/tui/agent-tree.js +0 -87
- package/dist/tui/ask-cli.js +0 -52
- package/dist/tui/ask-modal.js +0 -211
- package/dist/tui/ask-user-question-chips.js +0 -315
- package/dist/tui/ask-user-question-prompt.js +0 -203
- package/dist/tui/compact-banner.js +0 -81
- package/dist/tui/conversation-pane.js +0 -164
- package/dist/tui/cost-table.js +0 -111
- package/dist/tui/device-flow.js +0 -142
- package/dist/tui/doctor-table.js +0 -46
- package/dist/tui/feedback-prompt.js +0 -156
- package/dist/tui/input-box.js +0 -732
- package/dist/tui/login-picker.js +0 -69
- package/dist/tui/markdown-render.js +0 -266
- package/dist/tui/multi-file-diff-approval.js +0 -375
- package/dist/tui/onboarding-wizard.js +0 -240
- package/dist/tui/permissions-picker.js +0 -86
- package/dist/tui/render.js +0 -160
- package/dist/tui/repl-render.js +0 -770
- package/dist/tui/repl-splash-art.js +0 -64
- package/dist/tui/repl-splash-mascot.js +0 -154
- package/dist/tui/repl-splash.js +0 -117
- package/dist/tui/repl.js +0 -378
- package/dist/tui/slash-palette.js +0 -106
- package/dist/tui/splash-data.js +0 -61
- package/dist/tui/splash.js +0 -31
- package/dist/tui/status-bar.js +0 -209
- package/dist/tui/status-table.js +0 -7
- package/dist/tui/stickers-art.js +0 -136
- package/dist/tui/style-table.js +0 -28
- package/dist/tui/theme-table.js +0 -29
- package/dist/tui/thinking-spinner.js +0 -123
- package/dist/tui/tool-stream-pane.js +0 -140
- package/dist/tui/update-banner.js +0 -33
- package/dist/tui/vim-input.js +0 -267
- package/dist/tui/welcome-banner.js +0 -107
- package/dist/tui/welcome-data.js +0 -293
- package/dist/tui/workspace-context.js +0 -105
- package/docs/examples/codegraph.mcp.json +0 -10
- package/test/scenarios/codegen-create-file.scenario.txt +0 -13
- package/test/scenarios/compact-force.scenario.txt +0 -12
- package/test/scenarios/identity.scenario.txt +0 -11
- package/test/scenarios/persona-handoff.scenario.txt +0 -12
- package/test/scenarios/walkback.scenario.txt +0 -12
|
@@ -1,342 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* chokidar-based filewatch - Phase 1 (three-tier context).
|
|
3
|
-
*
|
|
4
|
-
* The REPL needs a live signal when files change so the operator's
|
|
5
|
-
* agent can re-read a stale file, and so the status bar can surface a
|
|
6
|
-
* "file changed" badge. chokidar wraps fsevents / inotify / kqueue so
|
|
7
|
-
* the cross-platform surface stays sane.
|
|
8
|
-
*
|
|
9
|
-
* Design choices:
|
|
10
|
-
*
|
|
11
|
-
* 1. **Ignore-aware**: every watched path is filtered through the
|
|
12
|
-
* same `PugiIgnore` matcher the skeleton walker uses. We pass the
|
|
13
|
-
* matcher into chokidar's `ignored` option so `node_modules/`,
|
|
14
|
-
* `dist/`, `.git/`, and secret files are never watched at all -
|
|
15
|
-
* this matters because chokidar's resource bound is the number
|
|
16
|
-
* of watched paths, not the number of total files in the repo.
|
|
17
|
-
*
|
|
18
|
-
* 2. **Throttle**: chokidar can fire dozens of `change` events when
|
|
19
|
-
* a code formatter rewrites a directory. We batch events in a
|
|
20
|
-
* `THROTTLE_WINDOW_MS` window and emit a single aggregated event
|
|
21
|
-
* so the REPL system line / status bar do not flicker.
|
|
22
|
-
*
|
|
23
|
-
* 3. **Watch cap**: per spec, the watcher caps watched paths at
|
|
24
|
-
* `MAX_WATCHED_PATHS`. Once the count crosses the cap we close
|
|
25
|
-
* the watcher and fall back to "no live updates this session" -
|
|
26
|
-
* better to silently lose the badge than to consume thousands of
|
|
27
|
-
* file descriptors. The fallback path emits one warning event so
|
|
28
|
-
* the operator knows live updates are off.
|
|
29
|
-
*
|
|
30
|
-
* 4. **Lifecycle**: `start()` is async (chokidar's `ready` event)
|
|
31
|
-
* so the caller can `await` the initial scan. `close()` is also
|
|
32
|
-
* async (chokidar's close is async). The watcher is reusable
|
|
33
|
-
* across multiple `start` / `close` pairs, but Phase 1 expects
|
|
34
|
-
* one-watcher-per-REPL-session.
|
|
35
|
-
*
|
|
36
|
-
* 5. **Testability**: the chokidar handle is injectable via the
|
|
37
|
-
* `watchFactory` option. Production wires `chokidar.watch`;
|
|
38
|
-
* tests pass a fake emitter that lets the spec drive `add` /
|
|
39
|
-
* `change` / `unlink` events synchronously.
|
|
40
|
-
*/
|
|
41
|
-
import { EventEmitter } from 'node:events';
|
|
42
|
-
import chokidar from 'chokidar';
|
|
43
|
-
/** Per-spec watch-path cap. */
|
|
44
|
-
export const MAX_WATCHED_PATHS = 10_000;
|
|
45
|
-
/** Batching window for the change throttle. */
|
|
46
|
-
export const THROTTLE_WINDOW_MS = 250;
|
|
47
|
-
/**
|
|
48
|
-
* Number of `add` events between cap-check re-evaluations. The ready
|
|
49
|
-
* handler triggers the FIRST cap check, but a `git checkout` of a
|
|
50
|
-
* giant subtree post-ready can blow past the cap without ever
|
|
51
|
-
* re-evaluating. Sample every N adds so the cap fires within a
|
|
52
|
-
* bounded window even on post-ready growth. Cheap: one Map
|
|
53
|
-
* iteration per N adds. triple-review P2 (PR).
|
|
54
|
-
*/
|
|
55
|
-
export const CAP_CHECK_ADD_INTERVAL = 1_000;
|
|
56
|
-
/**
|
|
57
|
-
* Concrete watcher. Construct, then `await start()`. Subscribe via
|
|
58
|
-
* `.on('batch', cb)`. `await close()` to tear down.
|
|
59
|
-
*/
|
|
60
|
-
export class PugiWatcher extends EventEmitter {
|
|
61
|
-
cwd;
|
|
62
|
-
ignore;
|
|
63
|
-
maxWatchedPaths;
|
|
64
|
-
throttleWindowMs;
|
|
65
|
-
now;
|
|
66
|
-
watchFactory;
|
|
67
|
-
setTimeoutImpl;
|
|
68
|
-
clearTimeoutImpl;
|
|
69
|
-
handle = null;
|
|
70
|
-
buffer = [];
|
|
71
|
-
windowStart = null;
|
|
72
|
-
timer = null;
|
|
73
|
-
closed = false;
|
|
74
|
-
capExceededEmitted = false;
|
|
75
|
-
capCheckTimer = null;
|
|
76
|
-
/**
|
|
77
|
-
* Counter of `add` events received since the last cap-check sample.
|
|
78
|
-
* When it crosses CAP_CHECK_ADD_INTERVAL we schedule another check.
|
|
79
|
-
* Resets on every cap-check run. triple-review P2 (PR).
|
|
80
|
-
*/
|
|
81
|
-
addsSinceLastCapCheck = 0;
|
|
82
|
-
constructor(options) {
|
|
83
|
-
super();
|
|
84
|
-
this.cwd = options.cwd;
|
|
85
|
-
this.ignore = options.ignore;
|
|
86
|
-
this.maxWatchedPaths = options.maxWatchedPaths ?? MAX_WATCHED_PATHS;
|
|
87
|
-
this.throttleWindowMs = options.throttleWindowMs ?? THROTTLE_WINDOW_MS;
|
|
88
|
-
this.now = options.now ?? (() => Date.now());
|
|
89
|
-
this.watchFactory = options.watchFactory ?? defaultWatchFactory;
|
|
90
|
-
this.setTimeoutImpl = options.setTimeoutImpl ?? ((fn, ms) => setTimeout(fn, ms));
|
|
91
|
-
this.clearTimeoutImpl = options.clearTimeoutImpl ?? ((h) => clearTimeout(h));
|
|
92
|
-
// Default no-op error listener - Node's EventEmitter.emit('error', ...)
|
|
93
|
-
// THROWS synchronously when no listener is registered. Production
|
|
94
|
-
// bootstrap awaits start() before constructing ReplSession, so there
|
|
95
|
-
// is no listener on the watcher at the moment chokidar may surface
|
|
96
|
-
// an error during the initial scan. The default listener swallows
|
|
97
|
-
// the event so emit() does not crash the CLI; any real consumer
|
|
98
|
-
// that subscribes later receives the events as normal because
|
|
99
|
-
// EventEmitter delivers to every registered listener, not just one.
|
|
100
|
-
// Codex P1 (PR).
|
|
101
|
-
this.on('error', () => {
|
|
102
|
-
/* defensive no-op so emit('error') never throws */
|
|
103
|
-
});
|
|
104
|
-
}
|
|
105
|
-
/**
|
|
106
|
-
* Open the chokidar handle and wait for the `ready` event. Resolves
|
|
107
|
-
* once the initial scan completes. Rejects when the underlying
|
|
108
|
-
* watcher errors out before `ready`.
|
|
109
|
-
*/
|
|
110
|
-
async start() {
|
|
111
|
-
if (this.handle)
|
|
112
|
-
return;
|
|
113
|
-
if (this.closed) {
|
|
114
|
-
throw new Error('PugiWatcher: cannot start a closed watcher');
|
|
115
|
-
}
|
|
116
|
-
const opts = {
|
|
117
|
-
// chokidar does not tell the predicate whether the path is a
|
|
118
|
-
// file or a directory, so we ask the matcher both ways. If
|
|
119
|
-
// either form matches, the path is excluded. This catches
|
|
120
|
-
// gitignore-style dir patterns (`node_modules/`) that would
|
|
121
|
-
// otherwise only match descendants, leaving the dir itself
|
|
122
|
-
// visible to chokidar's recursion.
|
|
123
|
-
ignored: (path) => this.ignore.isIgnored(path) || this.ignore.isIgnored(path, true),
|
|
124
|
-
ignoreInitial: true,
|
|
125
|
-
persistent: true,
|
|
126
|
-
// Disable polling - we want fsevents / inotify / kqueue speed.
|
|
127
|
-
usePolling: false,
|
|
128
|
-
// Aggregate events so we get one `change` per save, not three.
|
|
129
|
-
awaitWriteFinish: {
|
|
130
|
-
stabilityThreshold: 50,
|
|
131
|
-
pollInterval: 25,
|
|
132
|
-
},
|
|
133
|
-
// Keep the depth shallow at the chokidar layer too - even after
|
|
134
|
-
// ignore filtering, a multi-million file dir could exhaust file
|
|
135
|
-
// descriptors. The skeleton walker has its own depth cap; this
|
|
136
|
-
// one is a safety net.
|
|
137
|
-
depth: 12,
|
|
138
|
-
};
|
|
139
|
-
const watcher = this.watchFactory(this.cwd, opts);
|
|
140
|
-
this.handle = watcher;
|
|
141
|
-
return new Promise((resolveReady, rejectReady) => {
|
|
142
|
-
let settled = false;
|
|
143
|
-
watcher.on('ready', () => {
|
|
144
|
-
if (settled)
|
|
145
|
-
return;
|
|
146
|
-
settled = true;
|
|
147
|
-
this.emit('ready');
|
|
148
|
-
this.scheduleCapCheck();
|
|
149
|
-
resolveReady();
|
|
150
|
-
});
|
|
151
|
-
watcher.on('error', (err) => {
|
|
152
|
-
const error = err instanceof Error ? err : new Error(String(err));
|
|
153
|
-
this.emit('error', error);
|
|
154
|
-
if (!settled) {
|
|
155
|
-
settled = true;
|
|
156
|
-
rejectReady(error);
|
|
157
|
-
}
|
|
158
|
-
});
|
|
159
|
-
watcher.on('add', (path) => {
|
|
160
|
-
this.queue('add', path);
|
|
161
|
-
// Re-evaluate the cap every CAP_CHECK_ADD_INTERVAL adds so a
|
|
162
|
-
// post-ready burst (e.g. `git checkout` of a giant subtree)
|
|
163
|
-
// does not silently blow past the watched-paths cap. The
|
|
164
|
-
// ready-time check is preserved; this is the missing follow-up
|
|
165
|
-
// sampler the comment promised. triple-review P2 (PR).
|
|
166
|
-
this.addsSinceLastCapCheck += 1;
|
|
167
|
-
if (this.addsSinceLastCapCheck >= CAP_CHECK_ADD_INTERVAL) {
|
|
168
|
-
this.addsSinceLastCapCheck = 0;
|
|
169
|
-
this.scheduleCapCheck();
|
|
170
|
-
}
|
|
171
|
-
});
|
|
172
|
-
watcher.on('change', (path) => this.queue('change', path));
|
|
173
|
-
watcher.on('unlink', (path) => this.queue('unlink', path));
|
|
174
|
-
});
|
|
175
|
-
}
|
|
176
|
-
/** Close the underlying watcher and clear the throttle timer. */
|
|
177
|
-
async close() {
|
|
178
|
-
this.closed = true;
|
|
179
|
-
if (this.timer !== null) {
|
|
180
|
-
this.clearTimeoutImpl(this.timer);
|
|
181
|
-
this.timer = null;
|
|
182
|
-
}
|
|
183
|
-
if (this.capCheckTimer !== null) {
|
|
184
|
-
this.clearTimeoutImpl(this.capCheckTimer);
|
|
185
|
-
this.capCheckTimer = null;
|
|
186
|
-
}
|
|
187
|
-
if (this.handle) {
|
|
188
|
-
try {
|
|
189
|
-
await this.handle.close();
|
|
190
|
-
}
|
|
191
|
-
catch {
|
|
192
|
-
/* idempotent - chokidar may already be closed */
|
|
193
|
-
}
|
|
194
|
-
this.handle = null;
|
|
195
|
-
}
|
|
196
|
-
this.buffer = [];
|
|
197
|
-
this.windowStart = null;
|
|
198
|
-
}
|
|
199
|
-
/**
|
|
200
|
-
* Force-flush any pending events. Exposed primarily for tests that
|
|
201
|
-
* want to assert the batched payload without waiting for the
|
|
202
|
-
* timeout. Production code can also use this on REPL teardown so
|
|
203
|
-
* pending events are surfaced before close.
|
|
204
|
-
*/
|
|
205
|
-
flush() {
|
|
206
|
-
if (this.buffer.length === 0)
|
|
207
|
-
return;
|
|
208
|
-
if (this.timer !== null) {
|
|
209
|
-
this.clearTimeoutImpl(this.timer);
|
|
210
|
-
this.timer = null;
|
|
211
|
-
}
|
|
212
|
-
const events = this.buffer;
|
|
213
|
-
this.buffer = [];
|
|
214
|
-
const start = this.windowStart ?? this.now();
|
|
215
|
-
const end = this.now();
|
|
216
|
-
this.windowStart = null;
|
|
217
|
-
this.emit('batch', {
|
|
218
|
-
events: dedupeBatch(events),
|
|
219
|
-
windowStartEpochMs: start,
|
|
220
|
-
windowEndEpochMs: end,
|
|
221
|
-
});
|
|
222
|
-
}
|
|
223
|
-
/** Test surface - returns the current pending buffer. */
|
|
224
|
-
pendingCount() {
|
|
225
|
-
return this.buffer.length;
|
|
226
|
-
}
|
|
227
|
-
/* ------------------------------------------------------------ */
|
|
228
|
-
/* Internals */
|
|
229
|
-
/* ------------------------------------------------------------ */
|
|
230
|
-
queue(kind, path) {
|
|
231
|
-
if (this.closed)
|
|
232
|
-
return;
|
|
233
|
-
// Defensive double-check: even if chokidar's `ignored` predicate
|
|
234
|
-
// accepted the path, the matcher gets the last word. Same dir-aware
|
|
235
|
-
// double-call as the predicate above.
|
|
236
|
-
if (this.ignore.isIgnored(path) || this.ignore.isIgnored(path, true))
|
|
237
|
-
return;
|
|
238
|
-
const atEpochMs = this.now();
|
|
239
|
-
const event = {
|
|
240
|
-
path: relPathPosix(this.cwd, path),
|
|
241
|
-
absPath: path,
|
|
242
|
-
kind,
|
|
243
|
-
atEpochMs,
|
|
244
|
-
};
|
|
245
|
-
if (this.windowStart === null)
|
|
246
|
-
this.windowStart = atEpochMs;
|
|
247
|
-
this.buffer.push(event);
|
|
248
|
-
if (this.timer === null) {
|
|
249
|
-
this.timer = this.setTimeoutImpl(() => this.flush(), this.throttleWindowMs);
|
|
250
|
-
}
|
|
251
|
-
}
|
|
252
|
-
/**
|
|
253
|
-
* Lazy cap check - chokidar reports its watched paths via
|
|
254
|
-
* `getWatched()`. We sample once on ready, and again every
|
|
255
|
-
* CAP_CHECK_ADD_INTERVAL `add` events thereafter (e.g. a `git
|
|
256
|
-
* checkout` of a giant subtree post-ready). The cost is one Map
|
|
257
|
-
* iteration; the payoff is silent fallback when an operator targets
|
|
258
|
-
* a giant repo. Guarded against double-scheduling so the add-handler
|
|
259
|
-
* cannot stack multiple timers.
|
|
260
|
-
*/
|
|
261
|
-
scheduleCapCheck() {
|
|
262
|
-
if (this.closed || this.capExceededEmitted)
|
|
263
|
-
return;
|
|
264
|
-
if (this.capCheckTimer !== null)
|
|
265
|
-
return;
|
|
266
|
-
this.capCheckTimer = this.setTimeoutImpl(() => this.runCapCheck(), 100);
|
|
267
|
-
}
|
|
268
|
-
runCapCheck() {
|
|
269
|
-
this.capCheckTimer = null;
|
|
270
|
-
if (this.closed || this.capExceededEmitted || !this.handle)
|
|
271
|
-
return;
|
|
272
|
-
try {
|
|
273
|
-
const watched = this.handle.getWatched();
|
|
274
|
-
let count = 0;
|
|
275
|
-
for (const entries of Object.values(watched)) {
|
|
276
|
-
count += entries.length;
|
|
277
|
-
}
|
|
278
|
-
if (count > this.maxWatchedPaths) {
|
|
279
|
-
this.capExceededEmitted = true;
|
|
280
|
-
this.emit('capExceeded', {
|
|
281
|
-
watchedCount: count,
|
|
282
|
-
cap: this.maxWatchedPaths,
|
|
283
|
-
});
|
|
284
|
-
// Stop watching so we do not consume more file descriptors.
|
|
285
|
-
void this.close();
|
|
286
|
-
}
|
|
287
|
-
}
|
|
288
|
-
catch {
|
|
289
|
-
/* chokidar threw on getWatched - safe to ignore, cap check is best-effort */
|
|
290
|
-
}
|
|
291
|
-
}
|
|
292
|
-
}
|
|
293
|
-
/* ------------------------------------------------------------------ */
|
|
294
|
-
/* Helpers */
|
|
295
|
-
/* ------------------------------------------------------------------ */
|
|
296
|
-
/**
|
|
297
|
-
* Convert an absolute path to a repo-relative POSIX path. chokidar
|
|
298
|
-
* already uses forward slashes on Windows because it normalises
|
|
299
|
-
* internally, but we run the same fix unconditionally so the contract
|
|
300
|
-
* is platform-agnostic.
|
|
301
|
-
*/
|
|
302
|
-
function relPathPosix(cwd, abs) {
|
|
303
|
-
// chokidar may report paths already relative to cwd. Normalise both
|
|
304
|
-
// sides to forward slashes for the prefix compare.
|
|
305
|
-
const cwdPosix = cwd.replace(/\\/g, '/');
|
|
306
|
-
const absPosix = abs.replace(/\\/g, '/');
|
|
307
|
-
if (absPosix.startsWith(`${cwdPosix}/`)) {
|
|
308
|
-
return absPosix.slice(cwdPosix.length + 1);
|
|
309
|
-
}
|
|
310
|
-
if (absPosix === cwdPosix)
|
|
311
|
-
return '';
|
|
312
|
-
return absPosix;
|
|
313
|
-
}
|
|
314
|
-
/**
|
|
315
|
-
* Collapse consecutive (path, kind) duplicates inside a batch. We
|
|
316
|
-
* keep the FIRST occurrence so the order is "first-touch first".
|
|
317
|
-
* Different kinds for the same path are preserved (`add` then
|
|
318
|
-
* `change`) - the model wants both signals.
|
|
319
|
-
*/
|
|
320
|
-
function dedupeBatch(events) {
|
|
321
|
-
const seen = new Set();
|
|
322
|
-
const out = [];
|
|
323
|
-
for (const event of events) {
|
|
324
|
-
const key = `${event.kind}:${event.path}`;
|
|
325
|
-
if (seen.has(key))
|
|
326
|
-
continue;
|
|
327
|
-
seen.add(key);
|
|
328
|
-
out.push(event);
|
|
329
|
-
}
|
|
330
|
-
return out;
|
|
331
|
-
}
|
|
332
|
-
/**
|
|
333
|
-
* Production chokidar factory. Returns the real `FSWatcher` cast to
|
|
334
|
-
* the minimal interface so the watcher class never directly imports
|
|
335
|
-
* chokidar types beyond `WatchOptions`. The cast is safe - chokidar's
|
|
336
|
-
* `FSWatcher` is a superset of `ChokidarLike`.
|
|
337
|
-
*/
|
|
338
|
-
function defaultWatchFactory(cwd, opts) {
|
|
339
|
-
const watcher = chokidar.watch(cwd, opts);
|
|
340
|
-
return watcher;
|
|
341
|
-
}
|
|
342
|
-
//# sourceMappingURL=watcher.js.map
|
|
@@ -1,165 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Tier 1 working-set tracker - Phase 1 (three-tier context).
|
|
3
|
-
*
|
|
4
|
-
* The model context window is finite. Tier 0 (repo skeleton, ~5KB) is
|
|
5
|
-
* always loaded; Tier 2 (RAG, Anvil-side, deferred to ) answers
|
|
6
|
-
* queries on demand. Tier 1 - the working set - lives between them: the
|
|
7
|
-
* subset of files the agent has touched THIS session, bounded so a
|
|
8
|
-
* long REPL session does not unbounded-grow the system prompt.
|
|
9
|
-
*
|
|
10
|
-
* Design choices:
|
|
11
|
-
*
|
|
12
|
-
* 1. **LRU semantics**: the most-recently-touched file stays in the
|
|
13
|
-
* window; the oldest is evicted when the cap is hit. We use a
|
|
14
|
-
* `Map` so insertion order doubles as recency order (delete +
|
|
15
|
-
* re-set bubbles the entry to the tail).
|
|
16
|
-
*
|
|
17
|
-
* 2. **Source tracking**: every `track()` call records the source -
|
|
18
|
-
* `read` (agent fetched the file), `edit` (agent modified it),
|
|
19
|
-
* `write` (agent created it). The source seeds future heuristics
|
|
20
|
-
* (e.g. always pin edited files near the cap, never evict files
|
|
21
|
-
* written this session) but Phase 1 keeps the policy simple:
|
|
22
|
-
* pure LRU.
|
|
23
|
-
*
|
|
24
|
-
* 3. **No persistence**: working set is in-memory only. A REPL
|
|
25
|
-
* restart rebuilds from scratch (and from `/resume` SessionStore
|
|
26
|
-
* replay if/when we wire that). Per spec: "node:sqlite NOT
|
|
27
|
-
* needed for this sprint."
|
|
28
|
-
*
|
|
29
|
-
* 4. **Path normalisation**: we resolve every path to an absolute
|
|
30
|
-
* form on insert so two `track()` calls with `./foo.ts` and
|
|
31
|
-
* `/abs/cwd/foo.ts` collapse to one entry.
|
|
32
|
-
*
|
|
33
|
-
* 5. **Defensive cap**: `MAX_WORKING_SET = 50` per spec. We hold the
|
|
34
|
-
* cap as a configurable option so tests can dial it down to
|
|
35
|
-
* exercise eviction without 51 fixture files.
|
|
36
|
-
*/
|
|
37
|
-
import { resolve } from 'node:path';
|
|
38
|
-
/** Spec-mandated default cap. Mirrors the doc string in the sprint plan. */
|
|
39
|
-
export const DEFAULT_WORKING_SET_CAPACITY = 50;
|
|
40
|
-
export class WorkingSet {
|
|
41
|
-
entries = new Map();
|
|
42
|
-
capacity;
|
|
43
|
-
now;
|
|
44
|
-
constructor(options = {}) {
|
|
45
|
-
this.capacity = options.capacity ?? DEFAULT_WORKING_SET_CAPACITY;
|
|
46
|
-
if (!Number.isInteger(this.capacity) || this.capacity <= 0) {
|
|
47
|
-
throw new Error(`WorkingSet capacity must be a positive integer, got ${options.capacity}`);
|
|
48
|
-
}
|
|
49
|
-
this.now = options.now ?? (() => Date.now());
|
|
50
|
-
}
|
|
51
|
-
/**
|
|
52
|
-
* Record that the agent touched `path`. The path is normalised to an
|
|
53
|
-
* absolute form so duplicate calls with different relative paths
|
|
54
|
-
* collapse. Returns the entry as stored.
|
|
55
|
-
*
|
|
56
|
-
* LRU bump: an existing entry is deleted and re-inserted so insertion
|
|
57
|
-
* order tracks recency. The oldest entry is evicted when the
|
|
58
|
-
* insertion would breach `capacity`.
|
|
59
|
-
*/
|
|
60
|
-
track(path, source, sizeBytes = 0) {
|
|
61
|
-
const absPath = resolve(path);
|
|
62
|
-
const previous = this.entries.get(absPath);
|
|
63
|
-
const touchCount = previous ? previous.touchCount + 1 : 1;
|
|
64
|
-
// Prefer the new size when the caller supplies one; otherwise keep
|
|
65
|
-
// the previous reading so we do not "forget" the file's footprint
|
|
66
|
-
// just because the second track() call did not stat the file.
|
|
67
|
-
const nextSize = sizeBytes > 0
|
|
68
|
-
? sizeBytes
|
|
69
|
-
: (previous?.sizeBytes ?? 0);
|
|
70
|
-
const entry = Object.freeze({
|
|
71
|
-
absPath,
|
|
72
|
-
source,
|
|
73
|
-
touchedAtEpochMs: this.now(),
|
|
74
|
-
touchCount,
|
|
75
|
-
sizeBytes: nextSize,
|
|
76
|
-
});
|
|
77
|
-
// Delete first so re-insert lands at the tail (most-recent).
|
|
78
|
-
if (previous)
|
|
79
|
-
this.entries.delete(absPath);
|
|
80
|
-
this.entries.set(absPath, entry);
|
|
81
|
-
this.evictExcess();
|
|
82
|
-
return entry;
|
|
83
|
-
}
|
|
84
|
-
/**
|
|
85
|
-
* Drop a single path from the set. Returns `true` when the entry
|
|
86
|
-
* existed. Used by the file watcher's `unlink` event - a removed
|
|
87
|
-
* file is no longer in the working set.
|
|
88
|
-
*/
|
|
89
|
-
forget(path) {
|
|
90
|
-
const absPath = resolve(path);
|
|
91
|
-
return this.entries.delete(absPath);
|
|
92
|
-
}
|
|
93
|
-
/** Clear the entire set - used by `/clear` slash + REPL teardown. */
|
|
94
|
-
clear() {
|
|
95
|
-
this.entries.clear();
|
|
96
|
-
}
|
|
97
|
-
/** True when the path is currently retained. */
|
|
98
|
-
has(path) {
|
|
99
|
-
return this.entries.has(resolve(path));
|
|
100
|
-
}
|
|
101
|
-
/** Live count - O(1). */
|
|
102
|
-
size() {
|
|
103
|
-
return this.entries.size;
|
|
104
|
-
}
|
|
105
|
-
/** Configured cap - surfaced so `/context` can render `12/50`. */
|
|
106
|
-
capacityLimit() {
|
|
107
|
-
return this.capacity;
|
|
108
|
-
}
|
|
109
|
-
/**
|
|
110
|
-
* Snapshot of the set ordered MOST RECENT first. Suitable for
|
|
111
|
-
* direct injection into the agent's system prompt; the oldest
|
|
112
|
-
* entries are the least relevant and trail the list so the model's
|
|
113
|
-
* recency bias works in our favour.
|
|
114
|
-
*/
|
|
115
|
-
forSerialization() {
|
|
116
|
-
// `Map` iterates in insertion order = LRU order with oldest first.
|
|
117
|
-
// We want recency-first so reverse.
|
|
118
|
-
const all = Array.from(this.entries.values());
|
|
119
|
-
all.reverse();
|
|
120
|
-
return all;
|
|
121
|
-
}
|
|
122
|
-
/** Aggregate summary used by `/context` slash + status bar. */
|
|
123
|
-
summary() {
|
|
124
|
-
let totalSize = 0;
|
|
125
|
-
let oldest = null;
|
|
126
|
-
for (const entry of this.entries.values()) {
|
|
127
|
-
totalSize += entry.sizeBytes;
|
|
128
|
-
if (oldest === null || entry.touchedAtEpochMs < oldest) {
|
|
129
|
-
oldest = entry.touchedAtEpochMs;
|
|
130
|
-
}
|
|
131
|
-
}
|
|
132
|
-
return {
|
|
133
|
-
count: this.entries.size,
|
|
134
|
-
capacity: this.capacity,
|
|
135
|
-
totalSizeBytes: totalSize,
|
|
136
|
-
oldestTouchedAtEpochMs: oldest,
|
|
137
|
-
};
|
|
138
|
-
}
|
|
139
|
-
/**
|
|
140
|
-
* Public LRU eviction trigger - called automatically on every
|
|
141
|
-
* `track()` but exposed so callers (tests, future memory-pressure
|
|
142
|
-
* hook) can force a sweep without inserting a sentinel entry.
|
|
143
|
-
* Returns the count of evicted entries.
|
|
144
|
-
*/
|
|
145
|
-
evict() {
|
|
146
|
-
return this.evictExcess();
|
|
147
|
-
}
|
|
148
|
-
/**
|
|
149
|
-
* Walk the head of the insertion-ordered map (oldest first) and
|
|
150
|
-
* delete entries until size <= capacity. Returns the number of
|
|
151
|
-
* entries dropped.
|
|
152
|
-
*/
|
|
153
|
-
evictExcess() {
|
|
154
|
-
let removed = 0;
|
|
155
|
-
while (this.entries.size > this.capacity) {
|
|
156
|
-
const oldestKey = this.entries.keys().next().value;
|
|
157
|
-
if (oldestKey === undefined)
|
|
158
|
-
break;
|
|
159
|
-
this.entries.delete(oldestKey);
|
|
160
|
-
removed += 1;
|
|
161
|
-
}
|
|
162
|
-
return removed;
|
|
163
|
-
}
|
|
164
|
-
}
|
|
165
|
-
//# sourceMappingURL=working-set.js.map
|
|
@@ -1,77 +0,0 @@
|
|
|
1
|
-
function snapshot(task) {
|
|
2
|
-
return { ...task };
|
|
3
|
-
}
|
|
4
|
-
function isTerminal(status) {
|
|
5
|
-
return status === 'completed' || status === 'cancelled' || status === 'failed';
|
|
6
|
-
}
|
|
7
|
-
export function createAgentCoordinator(options) {
|
|
8
|
-
const tasks = new Map();
|
|
9
|
-
return {
|
|
10
|
-
async createTask(agentType, prompt) {
|
|
11
|
-
const { taskId } = await options.spawn(agentType, prompt);
|
|
12
|
-
const task = {
|
|
13
|
-
id: taskId,
|
|
14
|
-
agentType,
|
|
15
|
-
prompt,
|
|
16
|
-
status: 'queued',
|
|
17
|
-
createdAt: Date.now(),
|
|
18
|
-
};
|
|
19
|
-
tasks.set(task.id, task);
|
|
20
|
-
task.status = 'running';
|
|
21
|
-
task.startedAt = Date.now();
|
|
22
|
-
return snapshot(task);
|
|
23
|
-
},
|
|
24
|
-
async sendMessage(taskId, message) {
|
|
25
|
-
const task = tasks.get(taskId);
|
|
26
|
-
if (!task) {
|
|
27
|
-
throw new Error(`agent task ${taskId} not found`);
|
|
28
|
-
}
|
|
29
|
-
if (task.status !== 'running') {
|
|
30
|
-
throw new Error(`agent task ${taskId} is not running (status=${task.status})`);
|
|
31
|
-
}
|
|
32
|
-
await options.send(taskId, message);
|
|
33
|
-
},
|
|
34
|
-
async stopTask(taskId) {
|
|
35
|
-
const task = tasks.get(taskId);
|
|
36
|
-
if (!task) {
|
|
37
|
-
throw new Error(`agent task ${taskId} not found`);
|
|
38
|
-
}
|
|
39
|
-
if (isTerminal(task.status)) {
|
|
40
|
-
return;
|
|
41
|
-
}
|
|
42
|
-
await options.stop(taskId);
|
|
43
|
-
task.status = 'cancelled';
|
|
44
|
-
task.completedAt = Date.now();
|
|
45
|
-
},
|
|
46
|
-
listTasks() {
|
|
47
|
-
return Array.from(tasks.values())
|
|
48
|
-
.sort((a, b) => b.createdAt - a.createdAt)
|
|
49
|
-
.map(snapshot);
|
|
50
|
-
},
|
|
51
|
-
getTask(taskId) {
|
|
52
|
-
const task = tasks.get(taskId);
|
|
53
|
-
return task ? snapshot(task) : null;
|
|
54
|
-
},
|
|
55
|
-
markCompleted(taskId, output) {
|
|
56
|
-
const task = tasks.get(taskId);
|
|
57
|
-
if (!task) {
|
|
58
|
-
return;
|
|
59
|
-
}
|
|
60
|
-
task.status = 'completed';
|
|
61
|
-
task.completedAt = Date.now();
|
|
62
|
-
task.output = output;
|
|
63
|
-
delete task.error;
|
|
64
|
-
},
|
|
65
|
-
markFailed(taskId, error) {
|
|
66
|
-
const task = tasks.get(taskId);
|
|
67
|
-
if (!task) {
|
|
68
|
-
return;
|
|
69
|
-
}
|
|
70
|
-
task.status = 'failed';
|
|
71
|
-
task.completedAt = Date.now();
|
|
72
|
-
task.error = error;
|
|
73
|
-
delete task.output;
|
|
74
|
-
},
|
|
75
|
-
};
|
|
76
|
-
}
|
|
77
|
-
//# sourceMappingURL=agent-tools.js.map
|
|
@@ -1,65 +0,0 @@
|
|
|
1
|
-
// apps/pugi-cli/src/core/coordinator/agent-toolset.ts
|
|
2
|
-
import { randomUUID } from 'node:crypto';
|
|
3
|
-
export class AgentCoordinator {
|
|
4
|
-
agents = new Map();
|
|
5
|
-
spawnAgent(subagentType, prompt) {
|
|
6
|
-
const handle = {
|
|
7
|
-
id: randomUUID(),
|
|
8
|
-
subagentType,
|
|
9
|
-
prompt,
|
|
10
|
-
status: 'spawned',
|
|
11
|
-
spawnedAt: new Date().toISOString(),
|
|
12
|
-
};
|
|
13
|
-
this.agents.set(handle.id, handle);
|
|
14
|
-
return handle;
|
|
15
|
-
}
|
|
16
|
-
sendMessage(agentId, _message) {
|
|
17
|
-
const agent = this.agents.get(agentId);
|
|
18
|
-
if (!agent) {
|
|
19
|
-
return { ok: false, reason: `agent ${agentId} not found` };
|
|
20
|
-
}
|
|
21
|
-
if (agent.status !== 'spawned' && agent.status !== 'running') {
|
|
22
|
-
return { ok: false, reason: `agent ${agentId} not accepting messages (status=${agent.status})` };
|
|
23
|
-
}
|
|
24
|
-
return { ok: true };
|
|
25
|
-
}
|
|
26
|
-
taskStop(agentId) {
|
|
27
|
-
const agent = this.agents.get(agentId);
|
|
28
|
-
if (!agent) {
|
|
29
|
-
return { ok: false, reason: `agent ${agentId} not found` };
|
|
30
|
-
}
|
|
31
|
-
agent.status = 'stopped';
|
|
32
|
-
agent.completedAt = new Date().toISOString();
|
|
33
|
-
return { ok: true };
|
|
34
|
-
}
|
|
35
|
-
markCompleted(agentId, result) {
|
|
36
|
-
const agent = this.agents.get(agentId);
|
|
37
|
-
if (!agent) {
|
|
38
|
-
return;
|
|
39
|
-
}
|
|
40
|
-
agent.status = 'completed';
|
|
41
|
-
agent.completedAt = new Date().toISOString();
|
|
42
|
-
agent.result = result;
|
|
43
|
-
}
|
|
44
|
-
markFailed(agentId, reason) {
|
|
45
|
-
const agent = this.agents.get(agentId);
|
|
46
|
-
if (!agent) {
|
|
47
|
-
return;
|
|
48
|
-
}
|
|
49
|
-
agent.status = 'failed';
|
|
50
|
-
agent.completedAt = new Date().toISOString();
|
|
51
|
-
agent.result = reason;
|
|
52
|
-
}
|
|
53
|
-
listAgents(filter) {
|
|
54
|
-
const all = Array.from(this.agents.values());
|
|
55
|
-
if (!filter?.status) {
|
|
56
|
-
return all;
|
|
57
|
-
}
|
|
58
|
-
const wanted = filter.status;
|
|
59
|
-
return all.filter((a) => a.status === wanted);
|
|
60
|
-
}
|
|
61
|
-
getAgent(id) {
|
|
62
|
-
return this.agents.get(id);
|
|
63
|
-
}
|
|
64
|
-
}
|
|
65
|
-
//# sourceMappingURL=agent-toolset.js.map
|