@pugi/cli 0.1.0-beta.9 → 0.1.0-beta.91
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 +1792 -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/cron.js +433 -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 +99 -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/multi-file-diff-approval.js +375 -0
- 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
|
@@ -3,10 +3,10 @@
|
|
|
3
3
|
* compaction result is allowed to replace the original transcript.
|
|
4
4
|
*
|
|
5
5
|
* Per pattern card §2:
|
|
6
|
-
*
|
|
7
|
-
*
|
|
8
|
-
*
|
|
9
|
-
*
|
|
6
|
+
* - principle 4: never summarize secrets into durable memory
|
|
7
|
+
* - principle 5: do not erase open decisions
|
|
8
|
+
* - principle 6: cache stable prompt parts (static hash must survive
|
|
9
|
+
* compaction unchanged)
|
|
10
10
|
*
|
|
11
11
|
* Plus our own physical-integrity invariant: artifact refs emitted by
|
|
12
12
|
* the compaction must point to files that exist on disk and match the
|
|
@@ -26,12 +26,12 @@ import { existsSync, readFileSync } from 'node:fs';
|
|
|
26
26
|
*
|
|
27
27
|
* Coverage spans two shapes:
|
|
28
28
|
*
|
|
29
|
-
*
|
|
30
|
-
*
|
|
31
|
-
*
|
|
32
|
-
*
|
|
33
|
-
*
|
|
34
|
-
*
|
|
29
|
+
* 1. Cooperative `keyword = value` / `keyword: value` pairs covering
|
|
30
|
+
* api_key, access_token, password, client_secret, private_key,
|
|
31
|
+
* .env.* etc.
|
|
32
|
+
* 2. Provider-specific bare tokens that travel without a keyword:
|
|
33
|
+
* Bearer JWTs, AWS access keys, GitHub PATs, Slack tokens, Stripe
|
|
34
|
+
* keys, Anthropic/OpenAI keys, and PEM private key blocks.
|
|
35
35
|
*
|
|
36
36
|
* High-entropy base64-shaped blobs after a `:` or `=` are also caught
|
|
37
37
|
* as a defence in depth — operators who exfiltrate keys via raw JSON
|
|
@@ -44,12 +44,12 @@ const SECRET_PATTERNS = [
|
|
|
44
44
|
// 1. Keyword=value / keyword: value — the cooperative shape.
|
|
45
45
|
/(api[_-]?key|access[_-]?token|id[_-]?token|password|passwd|client[_-]?secret|private[_-]?key|\.env\.[A-Z_]+|(?<![a-z])token|(?<![a-z])secret)\s*[:=]\s*\S+/gi,
|
|
46
46
|
// 2. Authorization: Bearer <jwt-or-opaque-token>. JWTs are eyJ... but
|
|
47
|
-
//
|
|
48
|
-
//
|
|
47
|
+
// we accept any non-whitespace token after Bearer so opaque bearer
|
|
48
|
+
// tokens are also caught.
|
|
49
49
|
/Authorization\s*:\s*Bearer\s+\S+/gi,
|
|
50
50
|
/\bBearer\s+(?:eyJ[A-Za-z0-9_\-.]{16,}|[A-Za-z0-9_\-.]{20,})/g,
|
|
51
51
|
// 3. AWS access keys. AKIA prefix is the long-lived IAM key shape;
|
|
52
|
-
//
|
|
52
|
+
// aws_access_key_id is the typical .aws/credentials shape.
|
|
53
53
|
/\bAKIA[0-9A-Z]{16}\b/g,
|
|
54
54
|
/aws_access_key_id\s*[:=]\s*\S+/gi,
|
|
55
55
|
/aws_secret_access_key\s*[:=]\s*\S+/gi,
|
|
@@ -65,24 +65,24 @@ const SECRET_PATTERNS = [
|
|
|
65
65
|
// 6. Stripe live/test secret keys.
|
|
66
66
|
/\bsk_(?:live|test)_[A-Za-z0-9]{24,}\b/g,
|
|
67
67
|
// 7. Anthropic and OpenAI API keys. Anthropic uses sk-ant-<32+chars>;
|
|
68
|
-
//
|
|
69
|
-
//
|
|
70
|
-
//
|
|
68
|
+
// OpenAI legacy keys are sk-<40+chars>. Both share the sk- prefix
|
|
69
|
+
// so we keep them in their own patterns to avoid catching every
|
|
70
|
+
// Stripe sk_ as well (Stripe uses an underscore, not a dash).
|
|
71
71
|
/\bsk-ant-[A-Za-z0-9_-]{32,}\b/g,
|
|
72
72
|
/\bsk-(?!ant-)[A-Za-z0-9]{40,}\b/g,
|
|
73
73
|
// 8. PEM-encoded private key blocks. Matches RSA / EC / DSA / OPENSSH
|
|
74
|
-
//
|
|
75
|
-
//
|
|
76
|
-
//
|
|
77
|
-
//
|
|
74
|
+
// and the bare PRIVATE KEY variant. The body may contain real
|
|
75
|
+
// newlines (when scanning raw transcript content) OR literal `\n`
|
|
76
|
+
// sequences (when scanning JSON.stringify'd summaries) — both are
|
|
77
|
+
// covered by the broad character class.
|
|
78
78
|
/-----BEGIN (?:RSA |EC |DSA |OPENSSH |ENCRYPTED |)PRIVATE KEY-----[\s\S\\n]*?-----END (?:RSA |EC |DSA |OPENSSH |ENCRYPTED |)PRIVATE KEY-----/g,
|
|
79
79
|
// 8b. Lone PEM begin/end markers — survive even when the body is too
|
|
80
|
-
//
|
|
80
|
+
// long to capture or escaped beyond recognition.
|
|
81
81
|
/-----BEGIN (?:RSA |EC |DSA |OPENSSH |ENCRYPTED |)PRIVATE KEY-----/g,
|
|
82
82
|
// 9. High-entropy base64-shaped blobs after a colon or equals. 40+ chars
|
|
83
|
-
//
|
|
84
|
-
//
|
|
85
|
-
//
|
|
83
|
+
// of the base64url alphabet is well above the ~6 bytes/char threshold
|
|
84
|
+
// where false positives become rare. Word-boundary anchored so prose
|
|
85
|
+
// is not swept up.
|
|
86
86
|
/[:=]\s*"?([A-Za-z0-9_\-+/]{40,}={0,2})"?(?=\s|,|\}|$)/g,
|
|
87
87
|
];
|
|
88
88
|
/**
|
|
@@ -97,23 +97,23 @@ const DECISION_RX = /^\s*(?:DECISION|OPEN|BLOCKED|REJECTED):/;
|
|
|
97
97
|
* violation must cause the engine to drop the compaction result.
|
|
98
98
|
*
|
|
99
99
|
* Inputs:
|
|
100
|
-
*
|
|
101
|
-
*
|
|
102
|
-
*
|
|
103
|
-
*
|
|
104
|
-
*
|
|
105
|
-
*
|
|
106
|
-
*
|
|
107
|
-
*
|
|
100
|
+
* - `before`: the compaction input snapshot the caller computed
|
|
101
|
+
* - `after`: the compaction result the tier function produced
|
|
102
|
+
* - `summaryText`: the concrete prose / structured summary the
|
|
103
|
+
* compaction wrote into the dynamic block (or empty string for
|
|
104
|
+
* microcompact tiers that only reshape existing content)
|
|
105
|
+
* - `staticHashBefore` / `staticHashAfter`: instructions+toolSchema
|
|
106
|
+
* hash from the context builder, captured before and after the
|
|
107
|
+
* compaction (compaction must never touch static blocks)
|
|
108
108
|
*/
|
|
109
109
|
export function checkInvariants(args) {
|
|
110
110
|
const { before, after, summaryText } = args;
|
|
111
111
|
const violations = [];
|
|
112
112
|
// 1. secrets-never-summarize — sweep the post-compaction summary text.
|
|
113
|
-
//
|
|
114
|
-
//
|
|
115
|
-
//
|
|
116
|
-
//
|
|
113
|
+
// `summaryText` is what gets written to .pugi/session.db / replaces
|
|
114
|
+
// the transcript turns. We grep there, not the input, because the
|
|
115
|
+
// pre-compaction transcript is the operator's own data; we only
|
|
116
|
+
// police what we are about to make durable.
|
|
117
117
|
if (summaryText.length > 0) {
|
|
118
118
|
const firstMatch = findFirstSecret(summaryText);
|
|
119
119
|
if (firstMatch !== null) {
|
|
@@ -124,10 +124,10 @@ export function checkInvariants(args) {
|
|
|
124
124
|
}
|
|
125
125
|
}
|
|
126
126
|
// 2. open-decisions-preserved — every DECISION/OPEN/BLOCKED/REJECTED
|
|
127
|
-
//
|
|
128
|
-
//
|
|
129
|
-
//
|
|
130
|
-
//
|
|
127
|
+
// line in the pre-compaction transcript must appear verbatim in
|
|
128
|
+
// the post-compaction summary OR remain in the after-state's
|
|
129
|
+
// `decisionsPreserved`. The compaction result surfaces the latter
|
|
130
|
+
// so we cross-check both.
|
|
131
131
|
const beforeDecisions = [];
|
|
132
132
|
for (const turn of before.transcript) {
|
|
133
133
|
for (const line of turn.content.split('\n')) {
|
|
@@ -147,17 +147,17 @@ export function checkInvariants(args) {
|
|
|
147
147
|
}
|
|
148
148
|
}
|
|
149
149
|
// 3. artifact-refs-resolvable — every artifact ref must point to a
|
|
150
|
-
//
|
|
151
|
-
//
|
|
152
|
-
//
|
|
153
|
-
//
|
|
150
|
+
// file under .pugi/artifacts/ that exists and SHA256-matches the
|
|
151
|
+
// ref. We compute the hash physically rather than trusting the
|
|
152
|
+
// bookkeeping; if the disk write was corrupted, we want to know
|
|
153
|
+
// before we promote the compaction.
|
|
154
154
|
for (const ref of after.artifactsCreated) {
|
|
155
155
|
const violation = verifyArtifact(ref);
|
|
156
156
|
if (violation)
|
|
157
157
|
violations.push(violation);
|
|
158
158
|
}
|
|
159
159
|
// 4. static-hash-unchanged — instructions and tool schema hashes
|
|
160
|
-
//
|
|
160
|
+
// must be byte-identical before and after compaction.
|
|
161
161
|
if (args.staticHashBefore.instructionsHash !== args.staticHashAfter.instructionsHash) {
|
|
162
162
|
violations.push({
|
|
163
163
|
invariant: 'static-hash-unchanged',
|
|
@@ -219,16 +219,16 @@ function findFirstSecret(summaryText) {
|
|
|
219
219
|
function redact(input) {
|
|
220
220
|
// Two shapes to handle:
|
|
221
221
|
//
|
|
222
|
-
//
|
|
223
|
-
//
|
|
224
|
-
//
|
|
225
|
-
//
|
|
226
|
-
//
|
|
222
|
+
// 1. Keyword=value / keyword: value — keep the keyword visible so
|
|
223
|
+
// the operator can see which secret leaked, but mask the value.
|
|
224
|
+
// 2. Bare token (Bearer ..., AKIA..., PEM block, etc.) — keep the
|
|
225
|
+
// first 2 and last 2 chars of the token; mask the middle. PEM
|
|
226
|
+
// blocks are dropped entirely except for the BEGIN line.
|
|
227
227
|
if (input.startsWith('-----BEGIN')) {
|
|
228
228
|
// PEM blocks may arrive with real newlines, with escaped `\n`
|
|
229
229
|
// (JSON-stringified payloads), or with `\r\n` (Windows). Cut on
|
|
230
230
|
// the BEGIN header end so the body never leaks.
|
|
231
|
-
// Codex P1 retro
|
|
231
|
+
// Codex P1 retro: matching on `\n` only let the
|
|
232
232
|
// escaped-newline case dump the full PEM into invariant evidence.
|
|
233
233
|
const headerEnd = input.search(/-----(\r?\n|\\r?\\n|$)/);
|
|
234
234
|
const firstLine = headerEnd > 0 ? input.slice(0, headerEnd + 5) : '-----BEGIN PRIVATE KEY-----';
|
|
@@ -6,11 +6,11 @@
|
|
|
6
6
|
* context block. They are loaded once per session, deterministically, with
|
|
7
7
|
* the following safety budget:
|
|
8
8
|
*
|
|
9
|
-
*
|
|
10
|
-
*
|
|
11
|
-
*
|
|
12
|
-
*
|
|
13
|
-
*
|
|
9
|
+
* - max import depth: 3 (deeper chains are skipped, not fatal)
|
|
10
|
+
* - max total loaded bytes: 64 KB across PUGI.md + AGENTS.md + all imports
|
|
11
|
+
* - HTML comment stripping (`<!-- ... -->`) — comments often carry stale
|
|
12
|
+
* annotations that bias the model long after they go out of date
|
|
13
|
+
* - workspace containment — `@import ../../../etc/passwd` is rejected
|
|
14
14
|
*
|
|
15
15
|
* Missing files are not errors. If `PUGI.md` is absent we simply skip it
|
|
16
16
|
* and report nothing; the engine builds its context from instructions +
|
|
@@ -26,22 +26,40 @@ export const MAX_TOTAL_BYTES = 64 * 1024;
|
|
|
26
26
|
/**
|
|
27
27
|
* Source filenames we look for at the workspace root. Order matters:
|
|
28
28
|
* PUGI.md is the canonical Pugi-native file; AGENTS.md is the
|
|
29
|
-
* cross-CLI compatibility shim used by other agentic CLIs.
|
|
29
|
+
* cross-CLI compatibility shim used by other agentic CLIs; CLAUDE.md
|
|
30
|
+
* is the the upstream tool drop-in compat shim —
|
|
31
|
+
* operators migrating from CC routinely keep a workspace-root
|
|
32
|
+
* `CLAUDE.md` documenting project conventions, and we pick it up so
|
|
33
|
+
* Pugi sees the same ambient guidance without a manual rename.
|
|
34
|
+
*
|
|
35
|
+
* All three files are loaded when present (no "first one wins"
|
|
36
|
+
* dropdown). Each entry is HTML-stripped + @import-expanded + capped
|
|
37
|
+
* against the shared 64 KB budget. Pugi's precedence convention is
|
|
38
|
+
* "shown last = highest specificity"; PUGI.md / AGENTS.md / CLAUDE.md
|
|
39
|
+
* each have a stable position in this list so the context builder's
|
|
40
|
+
* render order is deterministic.
|
|
30
41
|
*/
|
|
31
|
-
export const MARKDOWN_SOURCES = ['PUGI.md', 'AGENTS.md'];
|
|
42
|
+
export const MARKDOWN_SOURCES = ['PUGI.md', 'AGENTS.md', 'CLAUDE.md'];
|
|
32
43
|
/**
|
|
33
44
|
* Load PUGI.md + AGENTS.md from `workspaceRoot`. Either or both may be
|
|
34
45
|
* absent. Returns the combined load result with per-file detail plus a
|
|
35
46
|
* flat list of warnings (best-effort: a missing file is a warning, not
|
|
36
47
|
* an error).
|
|
37
48
|
*/
|
|
38
|
-
export async function loadMarkdownContext(workspaceRoot) {
|
|
49
|
+
export async function loadMarkdownContext(workspaceRoot, env = process.env) {
|
|
39
50
|
const warnings = [];
|
|
40
51
|
const loaded = [];
|
|
41
52
|
let budgetRemaining = MAX_TOTAL_BYTES;
|
|
42
53
|
const visited = new Set();
|
|
43
54
|
const absRoot = resolve(workspaceRoot);
|
|
44
|
-
|
|
55
|
+
// #20 : allow operators to opt OUT of CLAUDE.md
|
|
56
|
+
// ingest via `PUGI_CC_COMPAT_DISABLE=1`. PUGI.md / AGENTS.md remain
|
|
57
|
+
// loaded — they are Pugi-native surfaces, not shims.
|
|
58
|
+
const ccCompatDisabled = env.PUGI_CC_COMPAT_DISABLE === '1';
|
|
59
|
+
const activeSources = ccCompatDisabled
|
|
60
|
+
? MARKDOWN_SOURCES.filter((s) => s !== 'CLAUDE.md')
|
|
61
|
+
: MARKDOWN_SOURCES;
|
|
62
|
+
for (const source of activeSources) {
|
|
45
63
|
const candidate = resolve(absRoot, source);
|
|
46
64
|
if (!existsSync(candidate)) {
|
|
47
65
|
warnings.push({
|
|
@@ -200,7 +218,7 @@ function expandFile(input) {
|
|
|
200
218
|
// readFileSync then follows the symlink and inlines arbitrary
|
|
201
219
|
// contents into the static context. Realpath the target AND the
|
|
202
220
|
// workspace root so the comparison runs against the physical paths.
|
|
203
|
-
// Mirrors the apps/pugi-cli/src/core/trust.ts pattern from PR
|
|
221
|
+
// Mirrors the apps/pugi-cli/src/core/trust.ts pattern from PR.
|
|
204
222
|
let realTarget;
|
|
205
223
|
let realRoot;
|
|
206
224
|
try {
|
|
@@ -0,0 +1,255 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Per-directory PUGI.md / AGENTS.md / CLAUDE.md / GEMINI.md traverse-up
|
|
3
|
+
* loader — β5a R4+P5.
|
|
4
|
+
*
|
|
5
|
+
* the upstream tool, peer CLI, and Gemini CLI all support a "walk up from
|
|
6
|
+
* cwd to the workspace root, pick up agent-context markdown at every
|
|
7
|
+
* level" pattern. Without this, a `pugi explain` invoked from
|
|
8
|
+
* `apps/admin-api/` cannot see project-local conventions encoded in
|
|
9
|
+
* `apps/admin-api/PUGI.md` (or the cross-CLI shim files) — the
|
|
10
|
+
* existing `loadMarkdownContext` only reads files at `workspaceRoot`.
|
|
11
|
+
*
|
|
12
|
+
* The β5a quality gate (≥80% win-rate vs the upstream tool per CEO)
|
|
13
|
+
* surfaces this gap repeatedly: monorepo-local conventions (NestJS
|
|
14
|
+
* controller style, Prisma migration name format, cabinet brand voice
|
|
15
|
+
* gates) live in per-app context files, and Pugi was blind to them
|
|
16
|
+
* pre-β5a.
|
|
17
|
+
*
|
|
18
|
+
* Contract:
|
|
19
|
+
*
|
|
20
|
+
* - Walk from `cwd` upward until we reach `workspaceRoot` OR cross
|
|
21
|
+
* the filesystem boundary. The workspace root file itself is
|
|
22
|
+
* loaded by the legacy `loadMarkdownContext` so we do NOT include
|
|
23
|
+
* it here (no double-load, no double-budget-charge).
|
|
24
|
+
*
|
|
25
|
+
* - At each intermediate directory, look for the four canonical
|
|
26
|
+
* filenames: `PUGI.md` (native), `AGENTS.md` (cross-CLI shim),
|
|
27
|
+
* `CLAUDE.md` (the upstream tool compat), `GEMINI.md` (Gemini CLI compat).
|
|
28
|
+
*
|
|
29
|
+
* - HTML comments stripped, identical to the legacy loader.
|
|
30
|
+
*
|
|
31
|
+
* - `@import` expansion is intentionally NOT performed here — the
|
|
32
|
+
* per-dir surface is "drop a small file with the local
|
|
33
|
+
* conventions"; deep @import chains belong at workspace root.
|
|
34
|
+
* Keeping this surface flat means the per-dir budget cannot be
|
|
35
|
+
* blown out by a runaway @import in an unrelated subtree.
|
|
36
|
+
*
|
|
37
|
+
* - Aggregate budget: `MAX_TRAVERSE_BYTES` across ALL files found
|
|
38
|
+
* in the walk. When exhausted, remaining files are skipped with
|
|
39
|
+
* a `budget_exhausted` warning. Default 32 KB — half the
|
|
40
|
+
* workspace-root budget, because per-dir files are meant to be
|
|
41
|
+
* terse delta conventions, not full project briefs.
|
|
42
|
+
*
|
|
43
|
+
* - The walk is bounded: `MAX_TRAVERSE_DEPTH` levels above
|
|
44
|
+
* workspaceRoot are NEVER traversed (defense against being
|
|
45
|
+
* invoked from a malicious cwd outside the workspace; in
|
|
46
|
+
* practice cwd is always inside workspaceRoot but the symlink
|
|
47
|
+
* case demands belt + suspenders).
|
|
48
|
+
*
|
|
49
|
+
* - Order returned: shallowest-first (workspace root would be
|
|
50
|
+
* first if included, then each level closer to cwd). Closest-
|
|
51
|
+
* to-cwd files are the most specific and the context builder
|
|
52
|
+
* emits them LAST so the model treats them as the highest-
|
|
53
|
+
* priority conventions.
|
|
54
|
+
*
|
|
55
|
+
* This module is pure: no logging, no network, no fs writes. Filesystem
|
|
56
|
+
* reads only.
|
|
57
|
+
*/
|
|
58
|
+
import { existsSync, readFileSync, realpathSync, statSync } from 'node:fs';
|
|
59
|
+
import { dirname, isAbsolute, relative, resolve, sep } from 'node:path';
|
|
60
|
+
import { stripHtmlComments } from './markdown-loader.js';
|
|
61
|
+
/**
|
|
62
|
+
* Per-traverse total byte cap. Half the workspace-root budget — these
|
|
63
|
+
* files are meant to be terse "in this subdir, do X". 32 KB still fits
|
|
64
|
+
* ~8000 words of guidance.
|
|
65
|
+
*/
|
|
66
|
+
export const MAX_TRAVERSE_BYTES = 32 * 1024;
|
|
67
|
+
/**
|
|
68
|
+
* Per-file byte cap. A single per-dir file should never dominate; the
|
|
69
|
+
* 8 KB cap keeps any one level honest. Files larger than this are
|
|
70
|
+
* loaded up to the cap and flagged truncated.
|
|
71
|
+
*/
|
|
72
|
+
export const MAX_TRAVERSE_PER_FILE_BYTES = 8 * 1024;
|
|
73
|
+
/**
|
|
74
|
+
* Maximum number of parent directories above workspaceRoot we will
|
|
75
|
+
* traverse. Zero in normal operation — cwd is always inside the
|
|
76
|
+
* workspace; the cap exists so a misconfigured invocation never
|
|
77
|
+
* walks the whole filesystem looking for AGENTS.md.
|
|
78
|
+
*/
|
|
79
|
+
export const MAX_TRAVERSE_DEPTH = 0;
|
|
80
|
+
/**
|
|
81
|
+
* Filenames we look for at every level of the walk. The order here
|
|
82
|
+
* also defines the per-directory load order: PUGI.md first (highest
|
|
83
|
+
* trust), then cross-CLI compat shims. When the same directory has
|
|
84
|
+
* multiple files (e.g. both PUGI.md AND CLAUDE.md), all are loaded
|
|
85
|
+
* — operators sometimes keep both during a tool migration.
|
|
86
|
+
*/
|
|
87
|
+
export const TRAVERSE_SOURCES = ['PUGI.md', 'AGENTS.md', 'CLAUDE.md', 'GEMINI.md'];
|
|
88
|
+
/**
|
|
89
|
+
* Walk from `opts.cwd` upward toward `opts.workspaceRoot`, loading
|
|
90
|
+
* every PUGI.md / AGENTS.md / CLAUDE.md / GEMINI.md we encounter at
|
|
91
|
+
* intermediate levels. The workspace root itself is NOT loaded here
|
|
92
|
+
* — `loadMarkdownContext(workspaceRoot)` owns that file.
|
|
93
|
+
*
|
|
94
|
+
* Returned `loaded` is sorted shallowest-first (closest-to-root) so
|
|
95
|
+
* the caller can emit per-dir context in increasing specificity.
|
|
96
|
+
*
|
|
97
|
+
* Safety properties (proven by spec):
|
|
98
|
+
*
|
|
99
|
+
* - Never reads outside the workspace tree (symlinks resolved via
|
|
100
|
+
* realpathSync; off-tree symlinks rejected as
|
|
101
|
+
* `import_escapes_workspace`).
|
|
102
|
+
* - Never reads more than `MAX_TRAVERSE_BYTES` total or more than
|
|
103
|
+
* `MAX_TRAVERSE_PER_FILE_BYTES` per file.
|
|
104
|
+
* - Never walks above the workspace root.
|
|
105
|
+
* - Never visits the workspace root itself (single source of truth
|
|
106
|
+
* for workspace-level docs stays with `loadMarkdownContext`).
|
|
107
|
+
*/
|
|
108
|
+
export async function loadTraversedMarkdown(opts) {
|
|
109
|
+
const warnings = [];
|
|
110
|
+
const loaded = [];
|
|
111
|
+
let budgetRemaining = MAX_TRAVERSE_BYTES;
|
|
112
|
+
let absRoot;
|
|
113
|
+
let absCwd;
|
|
114
|
+
try {
|
|
115
|
+
absRoot = realpathSync(resolve(opts.workspaceRoot));
|
|
116
|
+
absCwd = realpathSync(resolve(opts.cwd));
|
|
117
|
+
}
|
|
118
|
+
catch (error) {
|
|
119
|
+
warnings.push({
|
|
120
|
+
kind: 'read_error',
|
|
121
|
+
message: `realpath failed for traverse anchor: ${error.message}`,
|
|
122
|
+
});
|
|
123
|
+
return { loaded, warnings, totalBytes: 0 };
|
|
124
|
+
}
|
|
125
|
+
// Containment guard: if cwd is not inside workspaceRoot, refuse
|
|
126
|
+
// to walk. Returning a clean empty result keeps the engine happy
|
|
127
|
+
// and surfaces nothing about the off-tree cwd to the model.
|
|
128
|
+
const relCwd = relative(absRoot, absCwd);
|
|
129
|
+
if (relCwd.startsWith('..') || isAbsolute(relCwd)) {
|
|
130
|
+
warnings.push({
|
|
131
|
+
kind: 'import_escapes_workspace',
|
|
132
|
+
message: `cwd is outside workspaceRoot; per-dir traverse skipped (cwd=${absCwd}, root=${absRoot})`,
|
|
133
|
+
path: absCwd,
|
|
134
|
+
});
|
|
135
|
+
return { loaded, warnings, totalBytes: 0 };
|
|
136
|
+
}
|
|
137
|
+
// Collect the walk: every directory from cwd UP TO (but not
|
|
138
|
+
// including) workspaceRoot.
|
|
139
|
+
const dirsToVisit = [];
|
|
140
|
+
let current = absCwd;
|
|
141
|
+
while (current !== absRoot) {
|
|
142
|
+
dirsToVisit.push(current);
|
|
143
|
+
const parent = dirname(current);
|
|
144
|
+
if (parent === current)
|
|
145
|
+
break; // hit filesystem root before workspaceRoot — defensive
|
|
146
|
+
current = parent;
|
|
147
|
+
if (dirsToVisit.length > 64)
|
|
148
|
+
break; // pathological depth, refuse
|
|
149
|
+
}
|
|
150
|
+
// Walk shallowest-first so we charge the budget in the order
|
|
151
|
+
// that matches what we return.
|
|
152
|
+
dirsToVisit.reverse();
|
|
153
|
+
for (const dir of dirsToVisit) {
|
|
154
|
+
if (budgetRemaining <= 0) {
|
|
155
|
+
warnings.push({
|
|
156
|
+
kind: 'budget_exhausted',
|
|
157
|
+
message: `per-dir traverse budget exhausted before reaching ${dir}`,
|
|
158
|
+
path: dir,
|
|
159
|
+
});
|
|
160
|
+
break;
|
|
161
|
+
}
|
|
162
|
+
for (const source of TRAVERSE_SOURCES) {
|
|
163
|
+
const candidate = resolve(dir, source);
|
|
164
|
+
if (!existsSync(candidate))
|
|
165
|
+
continue;
|
|
166
|
+
// Symlink guard: same realpath check as the workspace-root
|
|
167
|
+
// loader. A symlink inside the workspace that points outside
|
|
168
|
+
// the workspace must NOT be inlined.
|
|
169
|
+
let realCandidate;
|
|
170
|
+
try {
|
|
171
|
+
realCandidate = realpathSync(candidate);
|
|
172
|
+
}
|
|
173
|
+
catch (error) {
|
|
174
|
+
warnings.push({
|
|
175
|
+
kind: 'read_error',
|
|
176
|
+
message: `realpath failed for ${candidate}: ${error.message}`,
|
|
177
|
+
path: candidate,
|
|
178
|
+
});
|
|
179
|
+
continue;
|
|
180
|
+
}
|
|
181
|
+
const realRel = relative(absRoot, realCandidate);
|
|
182
|
+
if (realRel.startsWith('..') || isAbsolute(realRel)) {
|
|
183
|
+
warnings.push({
|
|
184
|
+
kind: 'import_escapes_workspace',
|
|
185
|
+
message: `traverse file escapes workspace via symlink: ${candidate} -> ${realCandidate}`,
|
|
186
|
+
path: candidate,
|
|
187
|
+
});
|
|
188
|
+
continue;
|
|
189
|
+
}
|
|
190
|
+
let raw;
|
|
191
|
+
let rawBytes;
|
|
192
|
+
try {
|
|
193
|
+
rawBytes = statSync(candidate).size;
|
|
194
|
+
raw = readFileSync(candidate, 'utf8');
|
|
195
|
+
}
|
|
196
|
+
catch (error) {
|
|
197
|
+
warnings.push({
|
|
198
|
+
kind: 'read_error',
|
|
199
|
+
message: `could not read ${candidate}: ${error.message}`,
|
|
200
|
+
path: candidate,
|
|
201
|
+
});
|
|
202
|
+
continue;
|
|
203
|
+
}
|
|
204
|
+
const stripped = stripHtmlComments(raw);
|
|
205
|
+
// Per-file cap first, then global budget.
|
|
206
|
+
const perFileCap = Math.min(MAX_TRAVERSE_PER_FILE_BYTES, budgetRemaining);
|
|
207
|
+
let content = stripped;
|
|
208
|
+
let truncated = false;
|
|
209
|
+
let contentBytes = Buffer.byteLength(content, 'utf8');
|
|
210
|
+
if (contentBytes > perFileCap) {
|
|
211
|
+
// Codepoint-safe slice: convert byte cap to char cap by
|
|
212
|
+
// taking min(byte cap, char-length-up-to-cap). We accept
|
|
213
|
+
// mild over-trim for safety.
|
|
214
|
+
content = content.slice(0, perFileCap);
|
|
215
|
+
truncated = true;
|
|
216
|
+
contentBytes = Buffer.byteLength(content, 'utf8');
|
|
217
|
+
}
|
|
218
|
+
const distance = distanceSegments(absCwd, dir);
|
|
219
|
+
loaded.push({
|
|
220
|
+
source,
|
|
221
|
+
resolvedPath: candidate,
|
|
222
|
+
dir,
|
|
223
|
+
distanceFromCwd: distance,
|
|
224
|
+
rawBytes,
|
|
225
|
+
loadedBytes: contentBytes,
|
|
226
|
+
truncated,
|
|
227
|
+
content,
|
|
228
|
+
});
|
|
229
|
+
budgetRemaining -= contentBytes;
|
|
230
|
+
if (budgetRemaining <= 0)
|
|
231
|
+
break;
|
|
232
|
+
}
|
|
233
|
+
}
|
|
234
|
+
return {
|
|
235
|
+
loaded,
|
|
236
|
+
warnings,
|
|
237
|
+
totalBytes: MAX_TRAVERSE_BYTES - budgetRemaining,
|
|
238
|
+
};
|
|
239
|
+
}
|
|
240
|
+
/**
|
|
241
|
+
* How many path segments separate `from` and `to`. Both must be
|
|
242
|
+
* absolute. `to` is assumed to be an ancestor of (or equal to)
|
|
243
|
+
* `from`; if not, returns -1 so callers can ignore the file.
|
|
244
|
+
*/
|
|
245
|
+
function distanceSegments(from, to) {
|
|
246
|
+
if (from === to)
|
|
247
|
+
return 0;
|
|
248
|
+
const rel = relative(to, from);
|
|
249
|
+
if (rel.startsWith('..') || isAbsolute(rel))
|
|
250
|
+
return -1;
|
|
251
|
+
if (rel.length === 0)
|
|
252
|
+
return 0;
|
|
253
|
+
return rel.split(sep).length;
|
|
254
|
+
}
|
|
255
|
+
//# sourceMappingURL=markdown-traverse.js.map
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* `.pugiignore` parser -
|
|
2
|
+
* `.pugiignore` parser - Phase 1 (three-tier context).
|
|
3
3
|
*
|
|
4
4
|
* The CLI walks the workspace to build a repo skeleton and to feed the
|
|
5
5
|
* chokidar watcher. Both passes need a single source of truth for which
|
|
@@ -9,32 +9,32 @@
|
|
|
9
9
|
*
|
|
10
10
|
* Design choices:
|
|
11
11
|
*
|
|
12
|
-
*
|
|
13
|
-
*
|
|
14
|
-
*
|
|
15
|
-
*
|
|
16
|
-
*
|
|
17
|
-
*
|
|
18
|
-
*
|
|
12
|
+
* 1. We layer four sources, last-write-wins per the .gitignore spec:
|
|
13
|
+
* (a) Built-in DEFAULT_IGNORE_PATTERNS - secret / binary / build
|
|
14
|
+
* paths every workspace should drop.
|
|
15
|
+
* (b) `~/.pugi/global.pugiignore` - operator-global overrides.
|
|
16
|
+
* (c) Repo `.gitignore` (when present) - so we don't re-walk
|
|
17
|
+
* node_modules / dist / etc.
|
|
18
|
+
* (d) Repo `.pugiignore` - workspace-local extensions.
|
|
19
19
|
*
|
|
20
|
-
*
|
|
21
|
-
*
|
|
22
|
-
*
|
|
23
|
-
*
|
|
24
|
-
*
|
|
25
|
-
*
|
|
20
|
+
* 2. The `ignore` npm package handles negation (`!path`) and nested
|
|
21
|
+
* directories correctly. Rolling a minimal glob matcher is doable
|
|
22
|
+
* but the test surface for gitignore semantics (esp. negation +
|
|
23
|
+
* ancestor matching) is wide enough that pulling the audited
|
|
24
|
+
* library is the safer call. It is already in the workspace via
|
|
25
|
+
* transitive deps; we add it as an explicit pugi-cli dep.
|
|
26
26
|
*
|
|
27
|
-
*
|
|
28
|
-
*
|
|
29
|
-
*
|
|
30
|
-
*
|
|
31
|
-
*
|
|
32
|
-
*
|
|
27
|
+
* 3. Privacy is defense-in-depth: even if the operator deletes the
|
|
28
|
+
* built-in defaults from their `.pugiignore`, we re-add the
|
|
29
|
+
* secret patterns at the END of the chain so a typo in user
|
|
30
|
+
* config cannot expose `.env` / `*.pem` / `*.key`. The user can
|
|
31
|
+
* explicitly negate (`!.env.example`) but cannot wipe the
|
|
32
|
+
* defense.
|
|
33
33
|
*
|
|
34
|
-
*
|
|
35
|
-
*
|
|
36
|
-
*
|
|
37
|
-
*
|
|
34
|
+
* 4. Paths are normalised to forward slashes (the `ignore` package
|
|
35
|
+
* requires POSIX-style separators even on Windows) and stripped
|
|
36
|
+
* of any leading workspace root so `isIgnored` answers in
|
|
37
|
+
* repo-relative terms.
|
|
38
38
|
*
|
|
39
39
|
* The API surface is minimal: `loadPugiIgnore(cwd)` returns a
|
|
40
40
|
* `PugiIgnore` matcher with `isIgnored(absPath): boolean`. The matcher
|
|
@@ -59,19 +59,19 @@ const ignoreFactory = ignoreModule.default
|
|
|
59
59
|
* Built-in patterns shipped with every Pugi workspace. The list is a
|
|
60
60
|
* conservative union of three categories:
|
|
61
61
|
*
|
|
62
|
-
*
|
|
63
|
-
*
|
|
64
|
-
*
|
|
65
|
-
*
|
|
62
|
+
* - **Secrets** (defense in depth): `.env*`, `*.pem`, `*.key`,
|
|
63
|
+
* `secrets/**`, `.netrc`, `id_rsa*`, `*.secret`, `credentials*`.
|
|
64
|
+
* These are re-applied at the END of the chain so user config
|
|
65
|
+
* cannot accidentally expose them.
|
|
66
66
|
*
|
|
67
|
-
*
|
|
68
|
-
*
|
|
69
|
-
*
|
|
67
|
+
* - **Build outputs / caches**: `node_modules/`, `dist/`, `build/`,
|
|
68
|
+
* `.next/`, `.nuxt/`, `.svelte-kit/`, `coverage/`, `.nx/`, `.cache/`,
|
|
69
|
+
* `.turbo/`, `.parcel-cache/`, `out/`, `.output/`, `*.tsbuildinfo`.
|
|
70
70
|
*
|
|
71
|
-
*
|
|
72
|
-
*
|
|
73
|
-
*
|
|
74
|
-
*
|
|
71
|
+
* - **Large binaries / data**: lockfiles (`*.lock`, `*-lock.json`,
|
|
72
|
+
* `*-lock.yaml`), images (`*.png`, `*.jpg`, `*.jpeg`, `*.gif`,
|
|
73
|
+
* `*.ico`), PDFs, videos, archives (`*.zip`, `*.tar`, `*.gz`),
|
|
74
|
+
* DB dumps (`*.sql`, `*.dump`), logs.
|
|
75
75
|
*
|
|
76
76
|
* The split between baseline patterns (applied first) and secret
|
|
77
77
|
* patterns (applied last) matters: a user `.pugiignore` can `!`-negate
|
|
@@ -173,7 +173,7 @@ export const SECRET_IGNORE_PATTERNS = Object.freeze([
|
|
|
173
173
|
// used by Java + .NET tooling. Both can carry private keys when
|
|
174
174
|
// bundled (PKCS#7 / PKCS#12) and the file extension alone does not
|
|
175
175
|
// tell us whether the bundle includes a key. Treat as secrets by
|
|
176
|
-
// default. triple-review P1 (PR
|
|
176
|
+
// default. triple-review P1 (PR).
|
|
177
177
|
'*.cer',
|
|
178
178
|
'*.der',
|
|
179
179
|
'*.p12',
|
|
@@ -216,11 +216,11 @@ export function workspaceGitIgnorePath(cwd) {
|
|
|
216
216
|
* is built in this exact order so later layers override earlier ones,
|
|
217
217
|
* with the trailing secret block as the inviolable backstop:
|
|
218
218
|
*
|
|
219
|
-
*
|
|
220
|
-
*
|
|
221
|
-
*
|
|
222
|
-
*
|
|
223
|
-
*
|
|
219
|
+
* 1. BASELINE_IGNORE_PATTERNS (built-in defaults)
|
|
220
|
+
* 2. ~/.pugi/global.pugiignore (operator global, optional)
|
|
221
|
+
* 3. <cwd>/.gitignore (workspace, optional)
|
|
222
|
+
* 4. <cwd>/.pugiignore (workspace, optional)
|
|
223
|
+
* 5. SECRET_IGNORE_PATTERNS (defense-in-depth backstop)
|
|
224
224
|
*
|
|
225
225
|
* Any FS error reading optional files is swallowed - workspace
|
|
226
226
|
* context is best-effort and must never block the REPL bootstrap.
|