@pugi/cli 0.1.0-beta.7 → 0.1.0-beta.87
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 +96 -0
- package/THIRD_PARTY_NOTICES.md +40 -0
- 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 +2 -2
- 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 +12 -12
- 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 +293 -7
- package/dist/core/edits/format-matrix.js +26 -0
- package/dist/core/edits/fuzzy-ladder.js +650 -0
- package/dist/core/edits/index.js +3 -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 +322 -0
- package/dist/core/engine/anvil-client.js +140 -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 +134 -16
- package/dist/core/engine/strip-internal-fields.js +124 -0
- package/dist/core/engine/tool-bridge.js +1295 -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 +44 -0
- package/dist/core/hooks/index.js +15 -0
- package/dist/core/hooks/registry.js +213 -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/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 +776 -0
- package/dist/core/lsp/language-detect.js +66 -0
- package/dist/core/lsp/post-edit-diagnostics.js +171 -0
- package/dist/core/lsp/symbol-tools.js +372 -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 +2157 -214
- package/dist/core/repl/slash-commands.js +533 -40
- 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 +286 -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 +457 -0
- 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-manager/cleanup.js +123 -0
- package/dist/core/worktree-manager/manager.js +303 -0
- package/dist/index.js +28 -0
- package/dist/runtime/bootstrap.js +190 -0
- package/dist/runtime/cli.js +4162 -488
- package/dist/runtime/commands/agents.js +30 -30
- 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 +32 -32
- package/dist/runtime/commands/cost.js +199 -0
- package/dist/runtime/commands/delegate.js +244 -13
- 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 +184 -0
- package/dist/runtime/commands/init.js +254 -0
- package/dist/runtime/commands/lsp.js +368 -0
- 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 +128 -0
- 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 +177 -0
- 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 +531 -0
- package/dist/runtime/update-check.js +28 -28
- package/dist/runtime/version.js +65 -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 +556 -0
- package/dist/tools/ask-user-question.js +222 -0
- package/dist/tools/ask-user.js +115 -0
- package/dist/tools/bash.js +623 -45
- package/dist/tools/brief.js +224 -0
- package/dist/tools/enter-worktree.js +250 -0
- package/dist/tools/exit-worktree.js +147 -0
- package/dist/tools/file-tools.js +161 -44
- package/dist/tools/lsp-tools.js +189 -0
- 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 +85 -0
- 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-prompt.js +203 -0
- package/dist/tui/compact-banner.js +81 -0
- package/dist/tui/conversation-pane.js +85 -11
- package/dist/tui/cost-table.js +111 -0
- package/dist/tui/device-flow.js +2 -2
- package/dist/tui/doctor-table.js +46 -0
- package/dist/tui/feedback-prompt.js +156 -0
- package/dist/tui/input-box.js +247 -32
- package/dist/tui/login-picker.js +3 -3
- package/dist/tui/markdown-render.js +6 -6
- package/dist/tui/onboarding-wizard.js +240 -0
- package/dist/tui/permissions-picker.js +86 -0
- package/dist/tui/render.js +35 -0
- package/dist/tui/repl-render.js +332 -54
- 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 +124 -44
- 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/docs/examples/codegraph.mcp.json +10 -0
- package/package.json +23 -6
- package/test/scenarios/codegen-create-file.scenario.txt +13 -0
- package/test/scenarios/compact-force.scenario.txt +11 -0
- package/test/scenarios/identity.scenario.txt +11 -0
- package/test/scenarios/persona-handoff.scenario.txt +11 -0
- package/test/scenarios/walkback.scenario.txt +12 -0
- package/dist/core/engine/compaction-hook.js +0 -154
package/dist/core/session.js
CHANGED
|
@@ -18,6 +18,98 @@ export function openSession(root) {
|
|
|
18
18
|
enabled,
|
|
19
19
|
};
|
|
20
20
|
}
|
|
21
|
+
/**
|
|
22
|
+
* MVP — fire the `SessionStart` lifecycle event for all hooks
|
|
23
|
+
* declared in `~/.pugi/hooks-mvp.json`. Single-call surface; the REPL
|
|
24
|
+
* boot path invokes this once after `openSession`. Best-effort: any
|
|
25
|
+
* failure (missing config, hook spawn error) is swallowed so a
|
|
26
|
+
* misconfigured hook can never crash the REPL.
|
|
27
|
+
*
|
|
28
|
+
* Returns the number of hooks that fired (0 when no config / no
|
|
29
|
+
* matching hooks). Tests assert on the return value as the
|
|
30
|
+
* single-call invariant.
|
|
31
|
+
*/
|
|
32
|
+
export async function fireSessionStartMvp(session) {
|
|
33
|
+
try {
|
|
34
|
+
const { loadHooksConfig, fireHooks } = await import('./hooks/index.js');
|
|
35
|
+
// Defense-in-depth: `loadHooksConfig` is contractually non-null
|
|
36
|
+
// (returns `HooksConfig.empty(path)` when the file is absent), but
|
|
37
|
+
// the dynamic import boundary above can in principle return an
|
|
38
|
+
// unexpected shape if the module is mis-resolved at runtime. Guard
|
|
39
|
+
// the optional-chained `isEmpty()` call so a malformed loader can
|
|
40
|
+
// never raise `TypeError: Cannot read properties of undefined` and
|
|
41
|
+
// crash the REPL boot path. Belt-and-suspenders with the
|
|
42
|
+
// surrounding try/catch — the catch still swallows everything else.
|
|
43
|
+
const config = loadHooksConfig();
|
|
44
|
+
if (!config || config.isEmpty())
|
|
45
|
+
return 0;
|
|
46
|
+
const outcome = await fireHooks({
|
|
47
|
+
config,
|
|
48
|
+
event: 'SessionStart',
|
|
49
|
+
payload: {
|
|
50
|
+
event: 'SessionStart',
|
|
51
|
+
sessionId: session.id,
|
|
52
|
+
workspaceRoot: session.root,
|
|
53
|
+
startedAt: new Date().toISOString(),
|
|
54
|
+
},
|
|
55
|
+
workspaceRoot: session.root,
|
|
56
|
+
});
|
|
57
|
+
return outcome.results.length;
|
|
58
|
+
}
|
|
59
|
+
catch {
|
|
60
|
+
// SessionStart is never blocking — log nothing, return 0. A
|
|
61
|
+
// broken `hooks-mvp.json` is surfaced via `pugi hooks doctor`.
|
|
62
|
+
return 0;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* P1 — fire the v2 `SessionStart` event from `~/.pugi/hooks.json`
|
|
67
|
+
* (global) + `<workspaceRoot>/.pugi/hooks.json` (project). Companion to
|
|
68
|
+
* `fireSessionStartMvp`; both surfaces run because they read different
|
|
69
|
+
* config files.
|
|
70
|
+
*
|
|
71
|
+
* Headless by default (no trust prompt) — the v2 trust ledger gates
|
|
72
|
+
* first-run executions. Operators with no prior trust decision will see
|
|
73
|
+
* the SessionStart hook skipped with a `denied by trust ledger` stderr
|
|
74
|
+
* note; running `pugi hooks trust allow <command>` enrolls it.
|
|
75
|
+
*
|
|
76
|
+
* Returns the number of hooks that ran (excluding trust-denied skips).
|
|
77
|
+
* Never throws.
|
|
78
|
+
*/
|
|
79
|
+
export async function fireSessionStartV2(session) {
|
|
80
|
+
try {
|
|
81
|
+
const { fireSessionStart } = await import('./hooks/v2/index.js');
|
|
82
|
+
const outcome = await fireSessionStart({
|
|
83
|
+
sessionId: session.id,
|
|
84
|
+
workspaceRoot: session.root,
|
|
85
|
+
transcriptPath: session.eventsPath,
|
|
86
|
+
permissionMode: 'ask',
|
|
87
|
+
});
|
|
88
|
+
return outcome.results.filter((r) => r.exitCode !== -1).length;
|
|
89
|
+
}
|
|
90
|
+
catch {
|
|
91
|
+
return 0;
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
/**
|
|
95
|
+
* P1 — fire the v2 `SessionEnd` event. Called by the REPL
|
|
96
|
+
* teardown path. Companion to `fireSessionStartV2`.
|
|
97
|
+
*/
|
|
98
|
+
export async function fireSessionEndV2(session) {
|
|
99
|
+
try {
|
|
100
|
+
const { fireSessionEnd } = await import('./hooks/v2/index.js');
|
|
101
|
+
const outcome = await fireSessionEnd({
|
|
102
|
+
sessionId: session.id,
|
|
103
|
+
workspaceRoot: session.root,
|
|
104
|
+
transcriptPath: session.eventsPath,
|
|
105
|
+
permissionMode: 'ask',
|
|
106
|
+
});
|
|
107
|
+
return outcome.results.filter((r) => r.exitCode !== -1).length;
|
|
108
|
+
}
|
|
109
|
+
catch {
|
|
110
|
+
return 0;
|
|
111
|
+
}
|
|
112
|
+
}
|
|
21
113
|
export function recordCommandStarted(session, command) {
|
|
22
114
|
if (!session.enabled)
|
|
23
115
|
return;
|
package/dist/core/settings.js
CHANGED
|
@@ -20,6 +20,22 @@ const pugiSettingsSchema = z.object({
|
|
|
20
20
|
allow: z.array(z.string()).default([]),
|
|
21
21
|
deny: z.array(z.string()).default([]),
|
|
22
22
|
notAutomatic: z.array(z.string()).default([]),
|
|
23
|
+
// task — operator-declared read-only paths. Every
|
|
24
|
+
// edit / write tool call whose target matches one of these
|
|
25
|
+
// patterns is denied with source `readonly_paths`, regardless
|
|
26
|
+
// of the active permission mode. Match grammar mirrors
|
|
27
|
+
// allow/deny: exact path OR tail-* glob. Targets are
|
|
28
|
+
// workspace-relative, no leading slash.
|
|
29
|
+
//
|
|
30
|
+
// Why a dedicated field вместо leaning on a generic deny rule
|
|
31
|
+
// like `deny: ["edit:vendor/x"]`?
|
|
32
|
+
// - Explicit intent surfaces в `pugi doctor` and the REPL
|
|
33
|
+
// ("3 paths read-only") без parsing rule strings.
|
|
34
|
+
// - Covers BOTH the `edit` tool AND the `write` tool with
|
|
35
|
+
// one entry; a generic deny needs two rules.
|
|
36
|
+
// - Survives a future migration of the `permissions` schema
|
|
37
|
+
// because the field carries its own contract.
|
|
38
|
+
readonlyPaths: z.array(z.string()).default([]),
|
|
23
39
|
})
|
|
24
40
|
.default({}),
|
|
25
41
|
privacy: z
|
|
@@ -28,6 +44,17 @@ const pugiSettingsSchema = z.object({
|
|
|
28
44
|
telemetry: z.enum(['off', 'anonymous', 'community']).default('off'),
|
|
29
45
|
})
|
|
30
46
|
.default({}),
|
|
47
|
+
// beta.13 P1 fix: ui.cyberZoo gates the cyber-zoo splash +
|
|
48
|
+
// ambient art in the REPL. Schema must declare the key explicitly
|
|
49
|
+
// because Zod's strip pass swallows unknown keys, which is how the
|
|
50
|
+
// initial `pugi init` write (which serialises `ui.cyberZoo`) was
|
|
51
|
+
// bypassed by the runtime reader — the value never made it past the
|
|
52
|
+
// schema gate so admin-api always saw the historical 'on' default.
|
|
53
|
+
ui: z
|
|
54
|
+
.object({
|
|
55
|
+
cyberZoo: z.enum(['on', 'off']).default('on'),
|
|
56
|
+
})
|
|
57
|
+
.default({}),
|
|
31
58
|
artifacts: z
|
|
32
59
|
.object({
|
|
33
60
|
defaultPath: z.string().default('.pugi/artifacts'),
|
|
@@ -38,6 +65,12 @@ const pugiSettingsSchema = z.object({
|
|
|
38
65
|
// fetcher. Default-off matches the spec posture; the schema must
|
|
39
66
|
// declare it explicitly because Zod's strict-pass strips unknown
|
|
40
67
|
// keys and would silently swallow the operator's intent.
|
|
68
|
+
//
|
|
69
|
+
// β1b T4 : added `web.search.enabled` to gate the
|
|
70
|
+
// Brave-Search-backed `web_search` tool. Distinct from `web.fetch`
|
|
71
|
+
// because search queries themselves are an egress event that can
|
|
72
|
+
// leak operator intent — an operator may want fetch without
|
|
73
|
+
// implicitly enabling search-as-egress.
|
|
41
74
|
web: z
|
|
42
75
|
.object({
|
|
43
76
|
fetch: z
|
|
@@ -45,15 +78,263 @@ const pugiSettingsSchema = z.object({
|
|
|
45
78
|
enabled: z.boolean().optional(),
|
|
46
79
|
})
|
|
47
80
|
.optional(),
|
|
81
|
+
search: z
|
|
82
|
+
.object({
|
|
83
|
+
enabled: z.boolean().optional(),
|
|
84
|
+
})
|
|
85
|
+
.optional(),
|
|
86
|
+
})
|
|
87
|
+
.optional(),
|
|
88
|
+
// β7 L9 — per-language LSP toggle. When omitted, every supported
|
|
89
|
+
// server is available subject to binary detection on PATH. When
|
|
90
|
+
// present, only languages set to `true` are launched (false silently
|
|
91
|
+
// skips that language even if the binary is installed). Use this in
|
|
92
|
+
// workspaces where a heavyweight server (rust-analyzer indexing a
|
|
93
|
+
// monorepo, pyright on a fresh venv) wastes resources for the
|
|
94
|
+
// current task. The `pugi lsp servers` subcommand surfaces the
|
|
95
|
+
// current toggle state per server.
|
|
96
|
+
//
|
|
97
|
+
// Schema is intentionally permissive (`optional()` on the section AND
|
|
98
|
+
// on every per-language flag) so a partial config keeps the
|
|
99
|
+
// backwards-compatible "every language enabled" default.
|
|
100
|
+
lsp: z
|
|
101
|
+
.object({
|
|
102
|
+
typescript: z.boolean().optional(),
|
|
103
|
+
javascript: z.boolean().optional(),
|
|
104
|
+
python: z.boolean().optional(),
|
|
105
|
+
go: z.boolean().optional(),
|
|
106
|
+
rust: z.boolean().optional(),
|
|
107
|
+
// post-edit auto-diagnostics. When `true`,
|
|
108
|
+
// a successful `edit`/`write`/`multi_edit` triggers a diagnostic
|
|
109
|
+
// pull on the touched file(s) and the result is appended to the
|
|
110
|
+
// tool envelope so the model can self-correct in the same turn.
|
|
111
|
+
// Off by default — the cold-start of `typescript-language-server`
|
|
112
|
+
// is heavy enough that we opt in explicitly until dogfood proves
|
|
113
|
+
// the throughput trade is worth it. Also enabled via env var
|
|
114
|
+
// `PUGI_LSP_POST_EDIT=1` for CI / one-off operator probes.
|
|
115
|
+
postEditDiagnostics: z.boolean().optional(),
|
|
48
116
|
})
|
|
49
117
|
.optional(),
|
|
118
|
+
// β1 Pl9 — per-command budget overrides. Optional. Partial
|
|
119
|
+
// overrides merge against the β1 defaults in
|
|
120
|
+
// `core/engine/budgets.ts::beta1DefaultBudgets`. The schema is
|
|
121
|
+
// intentionally loose at the leaf (positive integers) so a typo lands
|
|
122
|
+
// a deterministic `BudgetConfigError` at `resolveBudget()` instead of
|
|
123
|
+
// a Zod parse error two layers up.
|
|
124
|
+
// task — operator-customizable status line.
|
|
125
|
+
// When `command` is set, Pugi spawns it on each turn boundary with
|
|
126
|
+
// a single JSON document on stdin (see `statusline.ts` for the
|
|
127
|
+
// schema). The command's stdout (first non-empty line, trimmed) is
|
|
128
|
+
// displayed in the Ink Footer. Failures are non-fatal — they emit
|
|
129
|
+
// a structured log line and the footer falls back to the default.
|
|
130
|
+
// Mirrors CC's `statusLine` config so cross-tool muscle memory
|
|
131
|
+
// carries over для operators who lived in CC first.
|
|
132
|
+
statusLine: z
|
|
133
|
+
.object({
|
|
134
|
+
command: z.string().min(1),
|
|
135
|
+
timeoutMs: z.number().int().positive().max(5000).default(500),
|
|
136
|
+
})
|
|
137
|
+
.optional(),
|
|
138
|
+
budgets: z
|
|
139
|
+
.object({
|
|
140
|
+
code: z
|
|
141
|
+
.object({ maxTokens: z.number().int().positive().optional(), maxToolCalls: z.number().int().positive().optional() })
|
|
142
|
+
.optional(),
|
|
143
|
+
fix: z
|
|
144
|
+
.object({ maxTokens: z.number().int().positive().optional(), maxToolCalls: z.number().int().positive().optional() })
|
|
145
|
+
.optional(),
|
|
146
|
+
build: z
|
|
147
|
+
.object({ maxTokens: z.number().int().positive().optional(), maxToolCalls: z.number().int().positive().optional() })
|
|
148
|
+
.optional(),
|
|
149
|
+
plan: z
|
|
150
|
+
.object({ maxTokens: z.number().int().positive().optional(), maxToolCalls: z.number().int().positive().optional() })
|
|
151
|
+
.optional(),
|
|
152
|
+
explain: z
|
|
153
|
+
.object({ maxTokens: z.number().int().positive().optional(), maxToolCalls: z.number().int().positive().optional() })
|
|
154
|
+
.optional(),
|
|
155
|
+
review_triple: z
|
|
156
|
+
.object({ maxTokens: z.number().int().positive().optional(), maxToolCalls: z.number().int().positive().optional() })
|
|
157
|
+
.optional(),
|
|
158
|
+
})
|
|
159
|
+
.optional(),
|
|
160
|
+
// #24 (CEO P1) — hook chains. First-class config
|
|
161
|
+
// for `PostToolUseFailure` + `TaskCompleted` event chains. The shape
|
|
162
|
+
// is intentionally loose (`z.any()` at the leaf) because the
|
|
163
|
+
// canonical reader lives in `core/hook-chains.ts` where the
|
|
164
|
+
// matcher/run/timeoutMs grammar is validated. Declaring the key here
|
|
165
|
+
// keeps Zod's strip-pass from swallowing it before the chain reader
|
|
166
|
+
// sees it. See `hook-chains.ts` for the full schema.
|
|
167
|
+
hooks: z.any().optional(),
|
|
50
168
|
});
|
|
51
|
-
|
|
169
|
+
/**
|
|
170
|
+
* #20 — the upstream tool drop-in compat ingest.
|
|
171
|
+
*
|
|
172
|
+
* Operators migrating from the upstream tool typically keep a `.claude/`
|
|
173
|
+
* directory at workspace root with settings.json, slash commands,
|
|
174
|
+
* and ambient guidance files. We honour the existence of that
|
|
175
|
+
* directory and mirror the subset of keys that map cleanly onto
|
|
176
|
+
* Pugi's own settings surface — Pugi values ALWAYS win on conflict
|
|
177
|
+
* (the operator opted into Pugi as their primary), CC fills gaps.
|
|
178
|
+
*
|
|
179
|
+
* Opt-out: `PUGI_CC_COMPAT_DISABLE=1` short-circuits the merger and
|
|
180
|
+
* loads only `.pugi/settings.json` (or the empty default).
|
|
181
|
+
*
|
|
182
|
+
* Key mirror table:
|
|
183
|
+
* - `permissions.defaultMode` → `permissions.mode`
|
|
184
|
+
* (CC values map: `acceptEdits|plan|bypassPermissions|default` →
|
|
185
|
+
* `acceptEdits|plan|bypassPermissions|ask`).
|
|
186
|
+
* - `permissions.allow` → `permissions.allow` (concatenated, deduped).
|
|
187
|
+
* - `permissions.deny` → `permissions.deny` (concatenated, deduped).
|
|
188
|
+
* - `enabledPlugins` → ignored (CC-only concept; Pugi has its own
|
|
189
|
+
* plugin surface and we do not want to silently activate them).
|
|
190
|
+
* - `hooks` → currently ignored. Pugi's hook system is
|
|
191
|
+
* managed via `apps/pugi-cli/src/core/hooks/`; future work can
|
|
192
|
+
* wire CC hook entries through that bridge.
|
|
193
|
+
*
|
|
194
|
+
* Unknown CC keys are dropped on the floor by Zod's strip semantics
|
|
195
|
+
* just like the existing PUGI settings path — we never warn on
|
|
196
|
+
* unrecognised CC keys, because the CC surface is wider and we want
|
|
197
|
+
* fallthrough to be silent (operator does not need a stream of "we
|
|
198
|
+
* skipped this CC concept" warnings on every CLI invocation).
|
|
199
|
+
*/
|
|
200
|
+
const ccPermissionsSchema = z
|
|
201
|
+
.object({
|
|
202
|
+
defaultMode: z.string().optional(),
|
|
203
|
+
allow: z.array(z.string()).optional(),
|
|
204
|
+
deny: z.array(z.string()).optional(),
|
|
205
|
+
})
|
|
206
|
+
.passthrough()
|
|
207
|
+
.optional();
|
|
208
|
+
const ccSettingsSchema = z
|
|
209
|
+
.object({
|
|
210
|
+
permissions: ccPermissionsSchema,
|
|
211
|
+
enabledPlugins: z.unknown().optional(),
|
|
212
|
+
hooks: z.unknown().optional(),
|
|
213
|
+
})
|
|
214
|
+
.passthrough();
|
|
215
|
+
/**
|
|
216
|
+
* Env var that disables ingest entirely. Useful for CI
|
|
217
|
+
* sandboxes where a stray `.claude/` from a parent checkout could
|
|
218
|
+
* otherwise leak permissions into Pugi.
|
|
219
|
+
*/
|
|
220
|
+
export const CC_COMPAT_DISABLE_ENV = 'PUGI_CC_COMPAT_DISABLE';
|
|
221
|
+
/**
|
|
222
|
+
* Map a CC `permissions.defaultMode` to the closest Pugi permission
|
|
223
|
+
* mode. Unknown / missing values map to `undefined` so the caller
|
|
224
|
+
* keeps Pugi's own default.
|
|
225
|
+
*
|
|
226
|
+
* CC's `default` mode = "ask the user for each tool" which is Pugi's
|
|
227
|
+
* `ask` mode. `acceptEdits` / `plan` / `bypassPermissions` map 1:1.
|
|
228
|
+
*/
|
|
229
|
+
export function mapCcPermissionMode(mode) {
|
|
230
|
+
if (typeof mode !== 'string')
|
|
231
|
+
return undefined;
|
|
232
|
+
switch (mode) {
|
|
233
|
+
case 'acceptEdits':
|
|
234
|
+
return 'acceptEdits';
|
|
235
|
+
case 'plan':
|
|
236
|
+
return 'plan';
|
|
237
|
+
case 'bypassPermissions':
|
|
238
|
+
return 'bypassPermissions';
|
|
239
|
+
case 'default':
|
|
240
|
+
return 'ask';
|
|
241
|
+
default:
|
|
242
|
+
return undefined;
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
/**
|
|
246
|
+
* Merge a parsed CC settings object into a Pugi settings object.
|
|
247
|
+
* Pugi ALWAYS wins on conflict; CC values fill gaps only.
|
|
248
|
+
*/
|
|
249
|
+
export function mergeCcIntoPugi(pugi, cc, opts) {
|
|
250
|
+
const merged = {
|
|
251
|
+
...pugi,
|
|
252
|
+
permissions: { ...pugi.permissions },
|
|
253
|
+
};
|
|
254
|
+
const pugiWroteMode = pugiPermissionKeyPresent(opts.pugiRawJson, 'mode');
|
|
255
|
+
if (!pugiWroteMode) {
|
|
256
|
+
const ccMode = mapCcPermissionMode(cc.permissions?.defaultMode);
|
|
257
|
+
if (ccMode)
|
|
258
|
+
merged.permissions.mode = ccMode;
|
|
259
|
+
}
|
|
260
|
+
if (Array.isArray(cc.permissions?.allow)) {
|
|
261
|
+
merged.permissions.allow = dedupeKeepFirst([
|
|
262
|
+
...pugi.permissions.allow,
|
|
263
|
+
...cc.permissions.allow,
|
|
264
|
+
]);
|
|
265
|
+
}
|
|
266
|
+
if (Array.isArray(cc.permissions?.deny)) {
|
|
267
|
+
merged.permissions.deny = dedupeKeepFirst([
|
|
268
|
+
...pugi.permissions.deny,
|
|
269
|
+
...cc.permissions.deny,
|
|
270
|
+
]);
|
|
271
|
+
}
|
|
272
|
+
// `enabledPlugins` and `hooks` are intentionally NOT mirrored. See
|
|
273
|
+
// the doc-block above for rationale.
|
|
274
|
+
void opts.pugiSettingsExisted;
|
|
275
|
+
return merged;
|
|
276
|
+
}
|
|
277
|
+
function pugiPermissionKeyPresent(raw, key) {
|
|
278
|
+
if (!raw || typeof raw !== 'object')
|
|
279
|
+
return false;
|
|
280
|
+
const permissions = raw.permissions;
|
|
281
|
+
if (!permissions || typeof permissions !== 'object')
|
|
282
|
+
return false;
|
|
283
|
+
return Object.prototype.hasOwnProperty.call(permissions, key);
|
|
284
|
+
}
|
|
285
|
+
function dedupeKeepFirst(items) {
|
|
286
|
+
const seen = new Set();
|
|
287
|
+
const out = [];
|
|
288
|
+
for (const item of items) {
|
|
289
|
+
if (seen.has(item))
|
|
290
|
+
continue;
|
|
291
|
+
seen.add(item);
|
|
292
|
+
out.push(item);
|
|
293
|
+
}
|
|
294
|
+
return out;
|
|
295
|
+
}
|
|
296
|
+
/**
|
|
297
|
+
* Read + parse `.claude/settings.json` at `root`. Returns `undefined`
|
|
298
|
+
* when the file is absent, malformed, or the operator has opted out
|
|
299
|
+
* via `PUGI_CC_COMPAT_DISABLE=1`. Never throws — a broken CC settings
|
|
300
|
+
* file degrades to "no ingest", not a Pugi boot crash.
|
|
301
|
+
*/
|
|
302
|
+
export function loadCcSettings(root, env = process.env) {
|
|
303
|
+
if (env[CC_COMPAT_DISABLE_ENV] === '1')
|
|
304
|
+
return undefined;
|
|
305
|
+
const ccPath = resolve(root, '.claude/settings.json');
|
|
306
|
+
if (!existsSync(ccPath))
|
|
307
|
+
return undefined;
|
|
308
|
+
let parsed;
|
|
309
|
+
try {
|
|
310
|
+
parsed = JSON.parse(readFileSync(ccPath, 'utf8'));
|
|
311
|
+
}
|
|
312
|
+
catch {
|
|
313
|
+
return undefined;
|
|
314
|
+
}
|
|
315
|
+
const result = ccSettingsSchema.safeParse(parsed);
|
|
316
|
+
if (!result.success)
|
|
317
|
+
return undefined;
|
|
318
|
+
return result.data;
|
|
319
|
+
}
|
|
320
|
+
export function loadSettings(root, env = process.env) {
|
|
52
321
|
const settingsPath = resolve(root, '.pugi/settings.json');
|
|
53
|
-
|
|
54
|
-
|
|
322
|
+
const pugiExists = existsSync(settingsPath);
|
|
323
|
+
let pugiRawJson = undefined;
|
|
324
|
+
let pugi;
|
|
325
|
+
if (pugiExists) {
|
|
326
|
+
pugiRawJson = JSON.parse(readFileSync(settingsPath, 'utf8'));
|
|
327
|
+
pugi = pugiSettingsSchema.parse(pugiRawJson);
|
|
328
|
+
}
|
|
329
|
+
else {
|
|
330
|
+
pugi = pugiSettingsSchema.parse({});
|
|
55
331
|
}
|
|
56
|
-
const
|
|
57
|
-
|
|
332
|
+
const cc = loadCcSettings(root, env);
|
|
333
|
+
if (!cc)
|
|
334
|
+
return pugi;
|
|
335
|
+
return mergeCcIntoPugi(pugi, cc, {
|
|
336
|
+
pugiSettingsExisted: pugiExists,
|
|
337
|
+
pugiRawJson,
|
|
338
|
+
});
|
|
58
339
|
}
|
|
59
340
|
//# sourceMappingURL=settings.js.map
|
|
@@ -0,0 +1,271 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Markdown transcript formatter used by `pugi share` ().
|
|
3
|
+
*
|
|
4
|
+
* Walks the session's `.pugi/events.jsonl` audit log and reconstructs a
|
|
5
|
+
* Markdown document the operator (and downstream Gist / pugi.io readers)
|
|
6
|
+
* can read top-to-bottom. The format is intentionally human-first — turn
|
|
7
|
+
* headers, code-block-wrapped tool I/O, and a small front-matter block
|
|
8
|
+
* with session metadata. Machine-readability is a non-goal here; the
|
|
9
|
+
* JSONL log remains the source of truth for tooling.
|
|
10
|
+
*
|
|
11
|
+
* Why we reconstruct from JSONL instead of from the live REPL state:
|
|
12
|
+
*
|
|
13
|
+
* - The CLI top-level `pugi share` command runs from a fresh shell and
|
|
14
|
+
* has no in-memory REPL state to read; only the event log is
|
|
15
|
+
* persisted.
|
|
16
|
+
* - The in-REPL `/share` slash uses the same handler so behaviour is
|
|
17
|
+
* identical regardless of entry point. Operators sharing a session
|
|
18
|
+
* from inside the REPL get the exact same output they would get from
|
|
19
|
+
* a follow-up shell command.
|
|
20
|
+
* - The JSONL log is append-only and survives REPL crashes, so a
|
|
21
|
+
* `--share` after a crash is the most useful debug surface.
|
|
22
|
+
*
|
|
23
|
+
* Event vocabulary the formatter knows about (see
|
|
24
|
+
* `packages/pugi-sdk/src/audit-trace.ts` for the schema):
|
|
25
|
+
*
|
|
26
|
+
* session.created Session boundary; emits front matter.
|
|
27
|
+
* session.command_started One-line "running pugi <cmd>" header.
|
|
28
|
+
* session.command_completed Status line ("success" / "error").
|
|
29
|
+
* tool_call Markdown turn header + inputSummary
|
|
30
|
+
* rendered as a fenced block.
|
|
31
|
+
* tool_result "Result (status):" + outputSummary
|
|
32
|
+
* rendered as a fenced block.
|
|
33
|
+
* file_mutation Inline `path` + operation summary.
|
|
34
|
+
* subagent.* Indented "[subagent <role>] ..." line.
|
|
35
|
+
* hook.* Quiet "[hook <event>] ..." line.
|
|
36
|
+
* compaction.* "[compaction <tier>] ..." line.
|
|
37
|
+
*
|
|
38
|
+
* Unknown event types are surfaced as a single italic line ("[event
|
|
39
|
+
* type=...]") so a future event added to the SDK does not silently
|
|
40
|
+
* vanish from the transcript.
|
|
41
|
+
*
|
|
42
|
+
* Performance: the formatter is O(n) over the events file, runs entirely
|
|
43
|
+
* in memory, and is bounded by the session log size (currently capped at
|
|
44
|
+
* a few MB per session). No streaming I/O is needed for typical sessions
|
|
45
|
+
* — the operator does not run /share against multi-GB logs.
|
|
46
|
+
*/
|
|
47
|
+
/**
|
|
48
|
+
* Format a session's event log as Markdown. Pure — no I/O. The caller
|
|
49
|
+
* reads `.pugi/events.jsonl` and hands the contents in.
|
|
50
|
+
*/
|
|
51
|
+
export function formatTranscript(input) {
|
|
52
|
+
const now = input.now ? input.now() : new Date();
|
|
53
|
+
const events = parseEvents(input.eventsJsonl);
|
|
54
|
+
const filteredSessionId = pickSessionId(input.sessionId, events);
|
|
55
|
+
const sessionEvents = events.filter((e) => typeof e.raw.sessionId !== 'string' || e.raw.sessionId === filteredSessionId);
|
|
56
|
+
const lines = [];
|
|
57
|
+
// Front matter — a fenced block at the top so downstream readers can
|
|
58
|
+
// grok the context before any turn content. Not YAML front matter
|
|
59
|
+
// (`---`) because Gist + pugi.io render Markdown directly; the fenced
|
|
60
|
+
// approach renders predictably without a parser.
|
|
61
|
+
lines.push('# Pugi session transcript');
|
|
62
|
+
lines.push('');
|
|
63
|
+
lines.push('```');
|
|
64
|
+
lines.push(`session_id: ${filteredSessionId}`);
|
|
65
|
+
lines.push(`workspace: ${input.workspaceRoot}`);
|
|
66
|
+
lines.push(`cli_version: ${input.cliVersion}`);
|
|
67
|
+
lines.push(`exported_at: ${now.toISOString()}`);
|
|
68
|
+
lines.push(`event_count: ${sessionEvents.length}`);
|
|
69
|
+
lines.push('```');
|
|
70
|
+
lines.push('');
|
|
71
|
+
if (sessionEvents.length === 0) {
|
|
72
|
+
lines.push('_No events recorded for this session._');
|
|
73
|
+
return {
|
|
74
|
+
markdown: `${lines.join('\n')}\n`,
|
|
75
|
+
turnCount: 0,
|
|
76
|
+
eventCount: 0,
|
|
77
|
+
};
|
|
78
|
+
}
|
|
79
|
+
let turnCount = 0;
|
|
80
|
+
for (const event of sessionEvents) {
|
|
81
|
+
const rendered = renderEvent(event);
|
|
82
|
+
if (rendered === null)
|
|
83
|
+
continue;
|
|
84
|
+
lines.push(...rendered.lines);
|
|
85
|
+
lines.push('');
|
|
86
|
+
if (rendered.isTurn)
|
|
87
|
+
turnCount += 1;
|
|
88
|
+
}
|
|
89
|
+
return {
|
|
90
|
+
markdown: `${lines.join('\n').replace(/\n{3,}/g, '\n\n')}\n`,
|
|
91
|
+
turnCount,
|
|
92
|
+
eventCount: sessionEvents.length,
|
|
93
|
+
};
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Parse the JSONL log. Malformed lines are skipped silently — the file
|
|
97
|
+
* is append-only and may have partial-write tail rows. Returning the
|
|
98
|
+
* stable typed shape lets the formatter walk without re-checking every
|
|
99
|
+
* field.
|
|
100
|
+
*/
|
|
101
|
+
function parseEvents(raw) {
|
|
102
|
+
const out = [];
|
|
103
|
+
for (const line of raw.split('\n')) {
|
|
104
|
+
const trimmed = line.trim();
|
|
105
|
+
if (trimmed.length === 0)
|
|
106
|
+
continue;
|
|
107
|
+
try {
|
|
108
|
+
const parsed = JSON.parse(trimmed);
|
|
109
|
+
const type = typeof parsed.type === 'string' ? parsed.type : '';
|
|
110
|
+
const timestamp = typeof parsed.timestamp === 'string' ? parsed.timestamp : '';
|
|
111
|
+
if (type.length === 0)
|
|
112
|
+
continue;
|
|
113
|
+
out.push({ raw: parsed, type, timestamp });
|
|
114
|
+
}
|
|
115
|
+
catch {
|
|
116
|
+
// partial-write or corrupt row; skip without affecting the rest
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
return out;
|
|
120
|
+
}
|
|
121
|
+
/**
|
|
122
|
+
* Resolve the effective session id. When the operator passes a non-empty
|
|
123
|
+
* value we honour it. When they pass an empty string / placeholder
|
|
124
|
+
* (`'no-session'`), we fall back to the newest `session.created` event
|
|
125
|
+
* id in the file. Last-resort fallback is the literal placeholder so the
|
|
126
|
+
* transcript still renders something meaningful in the front matter.
|
|
127
|
+
*/
|
|
128
|
+
function pickSessionId(provided, events) {
|
|
129
|
+
if (provided && provided !== 'no-session')
|
|
130
|
+
return provided;
|
|
131
|
+
for (let i = events.length - 1; i >= 0; i -= 1) {
|
|
132
|
+
const e = events[i];
|
|
133
|
+
if (!e)
|
|
134
|
+
continue;
|
|
135
|
+
if (e.type === 'session' && e.raw.name === 'created' && typeof e.raw.sessionId === 'string') {
|
|
136
|
+
return e.raw.sessionId;
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
return provided || 'unknown-session';
|
|
140
|
+
}
|
|
141
|
+
/**
|
|
142
|
+
* Format one event as a Markdown block. Returns `null` to suppress the
|
|
143
|
+
* event entirely (e.g. session.created is captured in front matter so we
|
|
144
|
+
* skip it here).
|
|
145
|
+
*/
|
|
146
|
+
function renderEvent(event) {
|
|
147
|
+
const ts = event.timestamp || '';
|
|
148
|
+
switch (event.type) {
|
|
149
|
+
case 'session': {
|
|
150
|
+
const name = String(event.raw.name ?? '');
|
|
151
|
+
if (name === 'created')
|
|
152
|
+
return null; // captured by front matter
|
|
153
|
+
if (name === 'command_started') {
|
|
154
|
+
const command = String(event.raw.command ?? '');
|
|
155
|
+
return {
|
|
156
|
+
lines: [`## ${ts} — command \`${escapeInline(command)}\``],
|
|
157
|
+
isTurn: false,
|
|
158
|
+
};
|
|
159
|
+
}
|
|
160
|
+
if (name === 'command_completed') {
|
|
161
|
+
const command = String(event.raw.command ?? '');
|
|
162
|
+
const status = String(event.raw.status ?? 'unknown');
|
|
163
|
+
return {
|
|
164
|
+
lines: [`_command \`${escapeInline(command)}\` finished: ${status}_`],
|
|
165
|
+
isTurn: false,
|
|
166
|
+
};
|
|
167
|
+
}
|
|
168
|
+
return {
|
|
169
|
+
lines: [`_session ${name}_`],
|
|
170
|
+
isTurn: false,
|
|
171
|
+
};
|
|
172
|
+
}
|
|
173
|
+
case 'tool_call': {
|
|
174
|
+
const tool = String(event.raw.tool ?? 'unknown');
|
|
175
|
+
const summary = String(event.raw.inputSummary ?? '');
|
|
176
|
+
const out = [`### ${ts} — tool \`${escapeInline(tool)}\``];
|
|
177
|
+
if (summary.length > 0) {
|
|
178
|
+
out.push('');
|
|
179
|
+
out.push('Input:');
|
|
180
|
+
out.push(fenced(summary));
|
|
181
|
+
}
|
|
182
|
+
return { lines: out, isTurn: true };
|
|
183
|
+
}
|
|
184
|
+
case 'tool_result': {
|
|
185
|
+
const status = String(event.raw.status ?? 'unknown');
|
|
186
|
+
const summary = String(event.raw.outputSummary ?? '');
|
|
187
|
+
const out = [`Result (${status}):`];
|
|
188
|
+
if (summary.length > 0) {
|
|
189
|
+
out.push(fenced(summary));
|
|
190
|
+
}
|
|
191
|
+
return { lines: out, isTurn: false };
|
|
192
|
+
}
|
|
193
|
+
case 'file_mutation': {
|
|
194
|
+
const path = String(event.raw.path ?? '');
|
|
195
|
+
const op = String(event.raw.operation ?? '');
|
|
196
|
+
return {
|
|
197
|
+
lines: [`- file ${op}: \`${escapeInline(path)}\``],
|
|
198
|
+
isTurn: false,
|
|
199
|
+
};
|
|
200
|
+
}
|
|
201
|
+
case 'subagent.spawned':
|
|
202
|
+
case 'subagent.tool_call':
|
|
203
|
+
case 'subagent.completed':
|
|
204
|
+
case 'subagent.blocked':
|
|
205
|
+
case 'subagent.failed': {
|
|
206
|
+
const role = String(event.raw.role ?? '');
|
|
207
|
+
const persona = String(event.raw.personaSlug ?? '');
|
|
208
|
+
const detail = String(event.raw.detail ?? event.raw.error ?? event.raw.toolName ?? '');
|
|
209
|
+
const tail = detail.length > 0 ? ` ${detail}` : '';
|
|
210
|
+
return {
|
|
211
|
+
lines: [`_[subagent ${role} / ${persona}] ${event.type}${tail}_`],
|
|
212
|
+
isTurn: false,
|
|
213
|
+
};
|
|
214
|
+
}
|
|
215
|
+
case 'hook.invoked':
|
|
216
|
+
case 'hook.result':
|
|
217
|
+
case 'hook.skipped': {
|
|
218
|
+
const ev = String(event.raw.event ?? '');
|
|
219
|
+
const reason = String(event.raw.reason ?? event.raw.runSummary ?? event.raw.matchSummary ?? '');
|
|
220
|
+
const tail = reason.length > 0 ? ` ${reason}` : '';
|
|
221
|
+
return {
|
|
222
|
+
lines: [`_[hook ${ev}] ${event.type.replace('hook.', '')}${tail}_`],
|
|
223
|
+
isTurn: false,
|
|
224
|
+
};
|
|
225
|
+
}
|
|
226
|
+
case 'compaction.started':
|
|
227
|
+
case 'compaction.completed':
|
|
228
|
+
case 'compaction.skipped':
|
|
229
|
+
case 'compaction.invariant_violated': {
|
|
230
|
+
const tier = String(event.raw.tier ?? '');
|
|
231
|
+
return {
|
|
232
|
+
lines: [`_[compaction ${tier}] ${event.type.replace('compaction.', '')}_`],
|
|
233
|
+
isTurn: false,
|
|
234
|
+
};
|
|
235
|
+
}
|
|
236
|
+
default: {
|
|
237
|
+
return {
|
|
238
|
+
lines: [`_[event type=${event.type}]_`],
|
|
239
|
+
isTurn: false,
|
|
240
|
+
};
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
/**
|
|
245
|
+
* Wrap a string in a fenced code block. Pick a fence length that does
|
|
246
|
+
* not collide with backtick runs inside the content. Markdown 0.30 allows
|
|
247
|
+
* variable-length fences; we pick the shortest that is longer than the
|
|
248
|
+
* longest run inside the content (min 3, max 7).
|
|
249
|
+
*/
|
|
250
|
+
function fenced(content) {
|
|
251
|
+
const longestRun = (content.match(/`+/g) ?? [])
|
|
252
|
+
.map((s) => s.length)
|
|
253
|
+
.reduce((max, n) => (n > max ? n : max), 0);
|
|
254
|
+
const fenceLen = Math.min(7, Math.max(3, longestRun + 1));
|
|
255
|
+
const fence = '`'.repeat(fenceLen);
|
|
256
|
+
// Trim trailing whitespace inside content so the closing fence sits
|
|
257
|
+
// tight against the body; preserve leading whitespace (matters for code).
|
|
258
|
+
return `${fence}\n${content.replace(/\s+$/u, '')}\n${fence}`;
|
|
259
|
+
}
|
|
260
|
+
/**
|
|
261
|
+
* Escape inline-Markdown specials (backtick, pipe) inside a span that we
|
|
262
|
+
* are wrapping in inline code. The closing backtick rule says a `<code>`
|
|
263
|
+
* span can contain backticks as long as the fence length differs — for
|
|
264
|
+
* simplicity we replace bare backticks with a Unicode look-alike when
|
|
265
|
+
* they appear in identifier-like positions (e.g. paths or tool names).
|
|
266
|
+
* Backticks in real content go through `fenced()` instead.
|
|
267
|
+
*/
|
|
268
|
+
function escapeInline(text) {
|
|
269
|
+
return text.replace(/`/g, 'ˋ');
|
|
270
|
+
}
|
|
271
|
+
//# sourceMappingURL=formatter.js.map
|