@pugi/cli 0.1.0-beta.98 → 1.0.0-alpha.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +1 -1
- package/README.md +11 -191
- package/bin/pugi +8 -0
- package/package.json +15 -71
- package/postinstall.mjs +31 -0
- package/CHANGELOG.md +0 -132
- package/THIRD_PARTY_NOTICES.md +0 -40
- package/assets/pugi-mascot.ansi +0 -16
- package/assets/pugi-prozr2-mascot.ansi +0 -9
- package/bin/run.js +0 -34
- package/dist/commands/deploy.js +0 -439
- package/dist/commands/flatten.js +0 -191
- package/dist/commands/jobs-watch.js +0 -201
- package/dist/commands/jobs.js +0 -260
- package/dist/commands/retro.js +0 -210
- package/dist/commands/smoke.js +0 -133
- package/dist/core/agent-progress/cleanup.js +0 -134
- package/dist/core/agent-progress/schema.js +0 -144
- package/dist/core/agent-progress/writer.js +0 -101
- package/dist/core/agents/adaptive-router.js +0 -330
- package/dist/core/agents/loader.js +0 -104
- package/dist/core/agents/query-decomposer.js +0 -297
- package/dist/core/agents/registry.js +0 -69
- package/dist/core/approvals/shortcut-resolver.js +0 -98
- package/dist/core/artifact-chain/dispatcher.js +0 -148
- package/dist/core/artifact-chain/exporter.js +0 -164
- package/dist/core/artifact-chain/state.js +0 -243
- package/dist/core/artifact-chain/steps.js +0 -169
- package/dist/core/ask-user/question.js +0 -92
- package/dist/core/audit/audit-trail.js +0 -275
- package/dist/core/auth/ensure-authenticated.js +0 -129
- package/dist/core/auth/env-provider.js +0 -238
- package/dist/core/auto-open-browser.js +0 -128
- package/dist/core/auto-update/channels.js +0 -122
- package/dist/core/auto-update/checker.js +0 -241
- package/dist/core/auto-update/state.js +0 -235
- package/dist/core/bare-mode/index.js +0 -107
- package/dist/core/bash/redirect.js +0 -281
- package/dist/core/bash-classifier.js +0 -1397
- package/dist/core/checkpoint/resumer.js +0 -149
- package/dist/core/checkpoint/rewinder.js +0 -291
- package/dist/core/checkpoints/shadow-git.js +0 -670
- package/dist/core/citations/parser.js +0 -109
- package/dist/core/classifier/yolo-classifier.js +0 -88
- package/dist/core/clipboard.js +0 -70
- package/dist/core/codegraph/decision-store.js +0 -248
- package/dist/core/codegraph/detect-repo.js +0 -459
- package/dist/core/codegraph/install.js +0 -134
- package/dist/core/codegraph/offer-hook.js +0 -220
- package/dist/core/compact/auto-trigger.js +0 -96
- package/dist/core/compact/buffer-rewriter.js +0 -115
- package/dist/core/compact/summarizer.js +0 -208
- package/dist/core/compact/token-counter.js +0 -108
- package/dist/core/consensus/anvil-fanout.js +0 -276
- package/dist/core/consensus/diff-capture.js +0 -491
- package/dist/core/consensus/rubric.js +0 -233
- package/dist/core/context/builder.js +0 -114
- package/dist/core/context/compaction-events.js +0 -99
- package/dist/core/context/compaction.js +0 -602
- package/dist/core/context/index.js +0 -28
- package/dist/core/context/invariants.js +0 -250
- package/dist/core/context/markdown-loader.js +0 -288
- package/dist/core/context/markdown-traverse.js +0 -255
- package/dist/core/context/pugiignore.js +0 -316
- package/dist/core/context/repo-skeleton.js +0 -533
- package/dist/core/context/tool-eviction.js +0 -55
- package/dist/core/context/watcher.js +0 -342
- package/dist/core/context/working-set.js +0 -165
- package/dist/core/coordinator/agent-tools.js +0 -77
- package/dist/core/coordinator/agent-toolset.js +0 -65
- package/dist/core/coordinator/fsm.js +0 -73
- package/dist/core/coordinator/mode-fsm.js +0 -70
- package/dist/core/cost/rate-card.js +0 -129
- package/dist/core/cost/tracker.js +0 -221
- package/dist/core/credentials.js +0 -355
- package/dist/core/cron/scheduler.js +0 -138
- package/dist/core/denial-tracking/index.js +0 -8
- package/dist/core/denial-tracking/state.js +0 -264
- package/dist/core/diagnostics/probe-runner.js +0 -93
- package/dist/core/diagnostics/probes/api.js +0 -46
- package/dist/core/diagnostics/probes/auth.js +0 -93
- package/dist/core/diagnostics/probes/bare-mode.js +0 -42
- package/dist/core/diagnostics/probes/cli-version.js +0 -127
- package/dist/core/diagnostics/probes/config.js +0 -72
- package/dist/core/diagnostics/probes/denial-tracking.js +0 -57
- package/dist/core/diagnostics/probes/disk.js +0 -81
- package/dist/core/diagnostics/probes/engine-live.js +0 -46
- package/dist/core/diagnostics/probes/git.js +0 -65
- package/dist/core/diagnostics/probes/hooks.js +0 -118
- package/dist/core/diagnostics/probes/mcp.js +0 -75
- package/dist/core/diagnostics/probes/node.js +0 -59
- package/dist/core/diagnostics/probes/pnpm.js +0 -36
- package/dist/core/diagnostics/probes/pugi-md.js +0 -89
- package/dist/core/diagnostics/probes/sandbox.js +0 -72
- package/dist/core/diagnostics/probes/session.js +0 -74
- package/dist/core/diagnostics/probes/status-snapshot.js +0 -488
- package/dist/core/diagnostics/probes/workspace.js +0 -63
- package/dist/core/diagnostics/types.js +0 -70
- package/dist/core/dispatch/cache-cleanup.js +0 -197
- package/dist/core/dispatch/cache-handoff.js +0 -295
- package/dist/core/edits/apply-patch-layer-e.js +0 -189
- package/dist/core/edits/dispatch.js +0 -511
- package/dist/core/edits/format-detector.js +0 -260
- package/dist/core/edits/format-matrix.js +0 -26
- package/dist/core/edits/fuzzy-ladder.js +0 -650
- package/dist/core/edits/index.js +0 -19
- package/dist/core/edits/journal.js +0 -199
- package/dist/core/edits/layer-a-apply.js +0 -217
- package/dist/core/edits/layer-a-fuzzy-apply.js +0 -198
- package/dist/core/edits/layer-b-apply.js +0 -211
- package/dist/core/edits/layer-c-apply.js +0 -160
- package/dist/core/edits/layer-d-ast.js +0 -572
- package/dist/core/edits/marker-parser.js +0 -401
- package/dist/core/edits/security-gate.js +0 -223
- package/dist/core/edits/verify-hook.js +0 -273
- package/dist/core/edits/worktree.js +0 -322
- package/dist/core/engine/adapter-runner.js +0 -8
- package/dist/core/engine/anvil-client.js +0 -344
- package/dist/core/engine/auto-compact.js +0 -179
- package/dist/core/engine/budgets.js +0 -192
- package/dist/core/engine/context-prefix.js +0 -155
- package/dist/core/engine/index.js +0 -12
- package/dist/core/engine/intensity.js +0 -163
- package/dist/core/engine/intent.js +0 -260
- package/dist/core/engine/native-pugi.js +0 -1616
- package/dist/core/engine/noop.js +0 -27
- package/dist/core/engine/prompts.js +0 -236
- package/dist/core/engine/strip-internal-fields.js +0 -124
- package/dist/core/engine/tool-bridge.js +0 -2173
- package/dist/core/engine/verification-patterns.js +0 -195
- package/dist/core/evaluation/golden-dataset.js +0 -293
- package/dist/core/feedback/queue.js +0 -177
- package/dist/core/feedback/submitter.js +0 -145
- package/dist/core/file-cache.js +0 -141
- package/dist/core/flatten/flatten-repo.js +0 -439
- package/dist/core/format/osc8-link.js +0 -28
- package/dist/core/hook-chains.js +0 -392
- package/dist/core/hooks/citation-verify-hook.js +0 -138
- package/dist/core/hooks/citation-verify.js +0 -112
- package/dist/core/hooks/events.js +0 -46
- package/dist/core/hooks/index.js +0 -15
- package/dist/core/hooks/registry.js +0 -216
- package/dist/core/hooks/runner.js +0 -236
- package/dist/core/hooks/v2/event-emitter.js +0 -115
- package/dist/core/hooks/v2/executor.js +0 -282
- package/dist/core/hooks/v2/index.js +0 -25
- package/dist/core/hooks/v2/lifecycle.js +0 -104
- package/dist/core/hooks/v2/loader.js +0 -216
- package/dist/core/hooks/v2/matcher.js +0 -125
- package/dist/core/hooks/v2/trust.js +0 -143
- package/dist/core/hooks/v2/types.js +0 -86
- package/dist/core/hooks/worktree-events.js +0 -158
- package/dist/core/hooks.js +0 -415
- package/dist/core/image/renderer.js +0 -71
- package/dist/core/index-store.js +0 -260
- package/dist/core/init/detector.js +0 -582
- package/dist/core/init/template-renderer.js +0 -242
- package/dist/core/jobs/registry.js +0 -462
- package/dist/core/ledger/results-tsv.js +0 -142
- package/dist/core/log-discipline/stdout-redirect.js +0 -51
- package/dist/core/lsp/cache.js +0 -105
- package/dist/core/lsp/client.js +0 -1229
- package/dist/core/lsp/language-detect.js +0 -66
- package/dist/core/lsp/post-edit-diagnostics.js +0 -171
- package/dist/core/lsp/server-detect.js +0 -173
- package/dist/core/lsp/symbol-cache.js +0 -162
- package/dist/core/lsp/symbol-tools.js +0 -664
- package/dist/core/mcp/client.js +0 -385
- package/dist/core/mcp/http-server.js +0 -553
- package/dist/core/mcp/orchestrator-config.js +0 -192
- package/dist/core/mcp/orchestrator-tools.js +0 -806
- package/dist/core/mcp/permission.js +0 -190
- package/dist/core/mcp/registry.js +0 -193
- package/dist/core/mcp/server-tools.js +0 -219
- package/dist/core/mcp/server.js +0 -397
- package/dist/core/mcp/trust.js +0 -91
- package/dist/core/memory/dual-write.js +0 -416
- package/dist/core/memory/passive-extract.js +0 -130
- package/dist/core/memory/phase1-kinds.js +0 -20
- package/dist/core/memory/secret-scanner.js +0 -304
- package/dist/core/memory-sync/queue.js +0 -170
- package/dist/core/metrics/extract.js +0 -113
- package/dist/core/modes/roo-modes.js +0 -68
- package/dist/core/onboarding/ensure-initialized.js +0 -133
- package/dist/core/onboarding/marker.js +0 -111
- package/dist/core/onboarding/telemetry-state.js +0 -108
- package/dist/core/output-style/presets.js +0 -176
- package/dist/core/output-style/state.js +0 -185
- package/dist/core/path-security.js +0 -345
- package/dist/core/permission.js +0 -369
- package/dist/core/permissions/auto-classifier.js +0 -124
- package/dist/core/permissions/bash-parser.js +0 -371
- package/dist/core/permissions/circuit-breaker.js +0 -83
- package/dist/core/permissions/constrained-edit.js +0 -91
- package/dist/core/permissions/gate.js +0 -278
- package/dist/core/permissions/index.js +0 -20
- package/dist/core/permissions/mode.js +0 -174
- package/dist/core/permissions/network-egress.js +0 -137
- package/dist/core/permissions/state.js +0 -241
- package/dist/core/permissions/tool-class.js +0 -107
- package/dist/core/plan-mode/ui-state.js +0 -51
- package/dist/core/plans/plan-artifact.js +0 -721
- package/dist/core/policy-limits/etag-store.js +0 -122
- package/dist/core/prd-check/parser.js +0 -215
- package/dist/core/prd-check/reporter.js +0 -127
- package/dist/core/prd-check/session-review.js +0 -557
- package/dist/core/prd-check/verifiers.js +0 -223
- package/dist/core/prompt-cache/client-cache.js +0 -99
- package/dist/core/prompts/assembly.js +0 -29
- package/dist/core/prompts/registry.js +0 -364
- package/dist/core/pugi-gitignore.js +0 -52
- package/dist/core/pugi-md/cc-compat-rules.js +0 -735
- package/dist/core/pugi-md/context-injector.js +0 -76
- package/dist/core/pugi-md/walk-up.js +0 -207
- package/dist/core/python/uv-installer.js +0 -270
- package/dist/core/python/uv-resolver.js +0 -83
- package/dist/core/rate-limit/narrator.js +0 -146
- package/dist/core/recipes/cli-types.js +0 -20
- package/dist/core/recipes/loader.js +0 -103
- package/dist/core/recipes/runner.js +0 -345
- package/dist/core/recipes/schema.js +0 -587
- package/dist/core/release-notes/parser.js +0 -241
- package/dist/core/release-notes/state.js +0 -116
- package/dist/core/repl/ask.js +0 -512
- package/dist/core/repl/cancellation.js +0 -98
- package/dist/core/repl/cap-warning.js +0 -91
- package/dist/core/repl/clipboard-read.js +0 -174
- package/dist/core/repl/dispatch-fsm.js +0 -220
- package/dist/core/repl/engine-bridge.js +0 -303
- package/dist/core/repl/history-search.js +0 -175
- package/dist/core/repl/history.js +0 -182
- package/dist/core/repl/kill-ring.js +0 -138
- package/dist/core/repl/model-pricing.js +0 -135
- package/dist/core/repl/privacy-banner.js +0 -71
- package/dist/core/repl/session.js +0 -4962
- package/dist/core/repl/slash-commands.js +0 -747
- package/dist/core/repl/store/index.js +0 -12
- package/dist/core/repl/store/jsonl-log.js +0 -321
- package/dist/core/repl/store/lockfile.js +0 -155
- package/dist/core/repl/store/session-store.js +0 -821
- package/dist/core/repl/store/types.js +0 -44
- package/dist/core/repl/store/uuid-v7.js +0 -68
- package/dist/core/repl/tool-route.js +0 -382
- package/dist/core/repl/workspace-context.js +0 -206
- package/dist/core/repo-map/build.js +0 -125
- package/dist/core/repo-map/cache.js +0 -185
- package/dist/core/repo-map/extractor.js +0 -254
- package/dist/core/repo-map/formatter.js +0 -145
- package/dist/core/repo-map/page-rank.js +0 -105
- package/dist/core/repo-map/scanner.js +0 -211
- package/dist/core/retro/git-collector.js +0 -251
- package/dist/core/retro/health-card.js +0 -25
- package/dist/core/retro/metrics.js +0 -342
- package/dist/core/retro/narrative.js +0 -249
- package/dist/core/retro/plane-collector.js +0 -274
- package/dist/core/retro/pr-issue-link.js +0 -65
- package/dist/core/retro/types.js +0 -16
- package/dist/core/retry-budget/budget.js +0 -284
- package/dist/core/retry-budget/index.js +0 -5
- package/dist/core/retry-budget/retry-cap.js +0 -74
- package/dist/core/routing/lead-worker.js +0 -43
- package/dist/core/routing/pre-flight-estimator.js +0 -108
- package/dist/core/runs/run-tree.js +0 -103
- package/dist/core/sandboxing/adapter.js +0 -29
- package/dist/core/sandboxing/index.js +0 -49
- package/dist/core/sandboxing/none.js +0 -19
- package/dist/core/sandboxing/seatbelt.js +0 -183
- package/dist/core/security/injection-scanner.js +0 -367
- package/dist/core/security/output-filter.js +0 -418
- package/dist/core/session/env-file.js +0 -105
- package/dist/core/session/section-budgets.js +0 -140
- package/dist/core/session.js +0 -377
- package/dist/core/settings.js +0 -400
- package/dist/core/share/formatter.js +0 -271
- package/dist/core/share/redactor.js +0 -221
- package/dist/core/share/uploader.js +0 -267
- package/dist/core/skills/defaults.js +0 -457
- package/dist/core/skills/loader.js +0 -454
- package/dist/core/skills/sources.js +0 -480
- package/dist/core/skills/trust.js +0 -172
- package/dist/core/smoke/headless-driver.js +0 -174
- package/dist/core/smoke/orchestrator.js +0 -194
- package/dist/core/smoke/runner.js +0 -238
- package/dist/core/smoke/scenario-parser.js +0 -316
- package/dist/core/statusline.js +0 -99
- package/dist/core/subagents/dispatcher-real.js +0 -600
- package/dist/core/subagents/dispatcher.js +0 -352
- package/dist/core/subagents/index.js +0 -39
- package/dist/core/subagents/isolation-matrix.js +0 -213
- package/dist/core/subagents/spawn.js +0 -101
- package/dist/core/telemetry/emitter.js +0 -229
- package/dist/core/telemetry/queue.js +0 -251
- package/dist/core/theme/context.js +0 -91
- package/dist/core/theme/presets.js +0 -228
- package/dist/core/theme/state.js +0 -181
- package/dist/core/todos/invariant.js +0 -10
- package/dist/core/todos/state.js +0 -177
- package/dist/core/tool-schema/compressor.js +0 -89
- package/dist/core/transport/version-interceptor.js +0 -166
- package/dist/core/trust.js +0 -109
- package/dist/core/tui/thinking-block.js +0 -64
- package/dist/core/vim/keymap.js +0 -288
- package/dist/core/vim/state.js +0 -92
- package/dist/core/watch-markers/marker-watcher.js +0 -133
- package/dist/core/worktree/include-parser.js +0 -249
- package/dist/core/worktree-manager/cleanup.js +0 -123
- package/dist/core/worktree-manager/manager.js +0 -303
- package/dist/index.js +0 -44
- package/dist/runtime/bootstrap.js +0 -190
- package/dist/runtime/cli.js +0 -8121
- package/dist/runtime/commands/agents.js +0 -385
- package/dist/runtime/commands/budget.js +0 -192
- package/dist/runtime/commands/cancel.js +0 -231
- package/dist/runtime/commands/chain.js +0 -489
- package/dist/runtime/commands/codegraph-status.js +0 -227
- package/dist/runtime/commands/compact.js +0 -297
- package/dist/runtime/commands/config.js +0 -595
- package/dist/runtime/commands/cost.js +0 -199
- package/dist/runtime/commands/delegate.js +0 -312
- package/dist/runtime/commands/dispatch.js +0 -126
- package/dist/runtime/commands/doctor.js +0 -579
- package/dist/runtime/commands/feedback.js +0 -184
- package/dist/runtime/commands/hooks.js +0 -187
- package/dist/runtime/commands/init.js +0 -254
- package/dist/runtime/commands/lsp.js +0 -368
- package/dist/runtime/commands/mcp.js +0 -935
- package/dist/runtime/commands/memory.js +0 -582
- package/dist/runtime/commands/model.js +0 -237
- package/dist/runtime/commands/onboarding.js +0 -275
- package/dist/runtime/commands/patch.js +0 -128
- package/dist/runtime/commands/permissions.js +0 -112
- package/dist/runtime/commands/plan.js +0 -143
- package/dist/runtime/commands/prd-check.js +0 -285
- package/dist/runtime/commands/privacy.js +0 -107
- package/dist/runtime/commands/recipe.js +0 -325
- package/dist/runtime/commands/redo-blob-store.js +0 -92
- package/dist/runtime/commands/redo.js +0 -361
- package/dist/runtime/commands/release-notes.js +0 -229
- package/dist/runtime/commands/repo-map.js +0 -95
- package/dist/runtime/commands/report.js +0 -299
- package/dist/runtime/commands/resume.js +0 -118
- package/dist/runtime/commands/review-consensus.js +0 -414
- package/dist/runtime/commands/rewind.js +0 -333
- package/dist/runtime/commands/roster.js +0 -117
- package/dist/runtime/commands/sessions.js +0 -163
- package/dist/runtime/commands/share.js +0 -316
- package/dist/runtime/commands/skills.js +0 -401
- package/dist/runtime/commands/status.js +0 -186
- package/dist/runtime/commands/stickers.js +0 -82
- package/dist/runtime/commands/style.js +0 -194
- package/dist/runtime/commands/theme.js +0 -196
- package/dist/runtime/commands/undo.js +0 -361
- package/dist/runtime/commands/update.js +0 -289
- package/dist/runtime/commands/vim.js +0 -140
- package/dist/runtime/commands/worktree.js +0 -177
- package/dist/runtime/commands/worktrees.js +0 -155
- package/dist/runtime/deprecation-warning.js +0 -69
- package/dist/runtime/engine-exit-code.js +0 -50
- package/dist/runtime/headless-repl.js +0 -195
- package/dist/runtime/headless.js +0 -548
- package/dist/runtime/load-hooks-or-exit.js +0 -71
- package/dist/runtime/plan-decompose.js +0 -531
- package/dist/runtime/sigint-guard.js +0 -272
- package/dist/runtime/stream-renderer.js +0 -195
- package/dist/runtime/update-check.js +0 -294
- package/dist/runtime/version.js +0 -65
- package/dist/runtime/worktree-bootstrap.js +0 -579
- package/dist/skills/bundled/batch.js +0 -617
- package/dist/skills/bundled/index.js +0 -45
- package/dist/skills/bundled/loop.js +0 -358
- package/dist/skills/bundled/remember.js +0 -383
- package/dist/skills/bundled/simplify.js +0 -289
- package/dist/skills/bundled/skillify.js +0 -373
- package/dist/skills/bundled/stuck.js +0 -558
- package/dist/skills/bundled/verify.js +0 -439
- package/dist/testing/vcr.js +0 -486
- package/dist/tools/agent-tool.js +0 -229
- package/dist/tools/apply-patch.js +0 -556
- package/dist/tools/ask-user-question.js +0 -337
- package/dist/tools/ask-user.js +0 -115
- package/dist/tools/bash.js +0 -1238
- package/dist/tools/brief.js +0 -224
- package/dist/tools/cron.js +0 -433
- package/dist/tools/enter-worktree.js +0 -250
- package/dist/tools/exit-worktree.js +0 -147
- package/dist/tools/file-tools.js +0 -553
- package/dist/tools/http-request.js +0 -336
- package/dist/tools/lsp-tools.js +0 -565
- package/dist/tools/mcp-tool.js +0 -260
- package/dist/tools/multi-edit.js +0 -361
- package/dist/tools/powershell.js +0 -268
- package/dist/tools/registry.js +0 -166
- package/dist/tools/server-tools.js +0 -892
- package/dist/tools/skill-tool.js +0 -96
- package/dist/tools/sleep.js +0 -99
- package/dist/tools/synthetic-output.js +0 -133
- package/dist/tools/tasks.js +0 -208
- package/dist/tools/todo-write.js +0 -184
- package/dist/tools/verify-plan-execution.js +0 -295
- package/dist/tools/web-fetch-injection-scanner.js +0 -207
- package/dist/tools/web-fetch.js +0 -720
- package/dist/tools/web-search.js +0 -458
- package/dist/tui/agent-progress-card.js +0 -111
- package/dist/tui/agent-tree-pane.js +0 -9
- package/dist/tui/agent-tree.js +0 -87
- package/dist/tui/ask-cli.js +0 -52
- package/dist/tui/ask-modal.js +0 -211
- package/dist/tui/ask-user-question-chips.js +0 -315
- package/dist/tui/ask-user-question-prompt.js +0 -203
- package/dist/tui/compact-banner.js +0 -81
- package/dist/tui/conversation-pane.js +0 -164
- package/dist/tui/cost-table.js +0 -111
- package/dist/tui/device-flow.js +0 -142
- package/dist/tui/doctor-table.js +0 -46
- package/dist/tui/feedback-prompt.js +0 -156
- package/dist/tui/input-box.js +0 -732
- package/dist/tui/login-picker.js +0 -69
- package/dist/tui/markdown-render.js +0 -266
- package/dist/tui/multi-file-diff-approval.js +0 -375
- package/dist/tui/onboarding-wizard.js +0 -240
- package/dist/tui/permissions-picker.js +0 -86
- package/dist/tui/render.js +0 -160
- package/dist/tui/repl-render.js +0 -770
- package/dist/tui/repl-splash-art.js +0 -64
- package/dist/tui/repl-splash-mascot.js +0 -154
- package/dist/tui/repl-splash.js +0 -117
- package/dist/tui/repl.js +0 -378
- package/dist/tui/slash-palette.js +0 -106
- package/dist/tui/splash-data.js +0 -61
- package/dist/tui/splash.js +0 -31
- package/dist/tui/status-bar.js +0 -209
- package/dist/tui/status-table.js +0 -7
- package/dist/tui/stickers-art.js +0 -136
- package/dist/tui/style-table.js +0 -28
- package/dist/tui/theme-table.js +0 -29
- package/dist/tui/thinking-spinner.js +0 -123
- package/dist/tui/tool-stream-pane.js +0 -140
- package/dist/tui/update-banner.js +0 -33
- package/dist/tui/vim-input.js +0 -267
- package/dist/tui/welcome-banner.js +0 -107
- package/dist/tui/welcome-data.js +0 -293
- package/dist/tui/workspace-context.js +0 -105
- package/docs/examples/codegraph.mcp.json +0 -10
- package/test/scenarios/codegen-create-file.scenario.txt +0 -13
- package/test/scenarios/compact-force.scenario.txt +0 -12
- package/test/scenarios/identity.scenario.txt +0 -11
- package/test/scenarios/persona-handoff.scenario.txt +0 -12
- package/test/scenarios/walkback.scenario.txt +0 -12
|
@@ -1,184 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* `pugi feedback` + `/feedback` slash — .
|
|
3
|
-
*
|
|
4
|
-
* In-CLI feedback collector. Parity with the upstream tool's `/feedback`
|
|
5
|
-
* built-in. The operator never has to leave the terminal to file a
|
|
6
|
-
* bug / feature / general comment / praise. The wizard collects:
|
|
7
|
-
*
|
|
8
|
-
* 1. category (bug / feature / general / praise)
|
|
9
|
-
* 2. rating (1-5)
|
|
10
|
-
* 3. comment (multi-line free text)
|
|
11
|
-
* 4. optional redacted session context (last 5 turns)
|
|
12
|
-
* 5. confirm (final y/n)
|
|
13
|
-
*
|
|
14
|
-
* # Module contract
|
|
15
|
-
*
|
|
16
|
-
* - This file owns the WIRING from the CLI surface (TTY mount,
|
|
17
|
-
* non-TTY JSON, slash dispatcher) to the queue + submitter
|
|
18
|
-
* modules. The corpus + redactor + queue persistence live in
|
|
19
|
-
* `core/feedback/{queue.ts,submitter.ts}`. The Ink prompt lives
|
|
20
|
-
* in `tui/feedback-prompt.tsx`. Both have zero coupling to the
|
|
21
|
-
* CLI dispatch surface.
|
|
22
|
-
*
|
|
23
|
-
* - `runFeedbackCommand` is the single entry point. Both the top-
|
|
24
|
-
* level `pugi feedback` handler в `runtime/cli.ts` AND the in-REPL
|
|
25
|
-
* `/feedback` slash dispatcher call it. The function returns the
|
|
26
|
-
* resolved `FeedbackRunResult` so the slash dispatcher can route
|
|
27
|
-
* the outcome message to the REPL's system pane without re-prompting.
|
|
28
|
-
*
|
|
29
|
-
* - Exit code is ALWAYS 0. Feedback is a brand surface — never a
|
|
30
|
-
* gate. Failures land as result variants; the wrapper never
|
|
31
|
-
* turns a network blip into a non-zero shell exit.
|
|
32
|
-
*
|
|
33
|
-
* - The random-source-style test seam: the run helper accepts an
|
|
34
|
-
* `interactive` flag that the spec sets to false, plus an injected
|
|
35
|
-
* `draft` so unit tests can drive the submit + queue branches
|
|
36
|
-
* without mounting Ink.
|
|
37
|
-
*/
|
|
38
|
-
import { existsSync, readFileSync } from 'node:fs';
|
|
39
|
-
import { resolve } from 'node:path';
|
|
40
|
-
import { enqueueFeedback, feedbackQueuePath, flushFeedbackQueue, } from '../../core/feedback/queue.js';
|
|
41
|
-
import { feedbackSubmitUrl, redactSessionContext, submitFeedback, } from '../../core/feedback/submitter.js';
|
|
42
|
-
/**
|
|
43
|
-
* Drive one feedback round. The function is async because of the
|
|
44
|
-
* submit round-trip, but everything else (queue write, redaction) is
|
|
45
|
-
* sync — no surprise concurrency.
|
|
46
|
-
*/
|
|
47
|
-
export async function runFeedbackCommand(ctx) {
|
|
48
|
-
if (ctx.draft == null) {
|
|
49
|
-
return { kind: 'cancelled' };
|
|
50
|
-
}
|
|
51
|
-
const envelope = {
|
|
52
|
-
category: ctx.draft.category,
|
|
53
|
-
rating: ctx.draft.rating,
|
|
54
|
-
comment: ctx.draft.comment,
|
|
55
|
-
ts: new Date().toISOString(),
|
|
56
|
-
cliVersion: ctx.cliVersion,
|
|
57
|
-
...(ctx.tier ? { tier: ctx.tier } : {}),
|
|
58
|
-
};
|
|
59
|
-
if (ctx.draft.includeSessionContext && ctx.sessionContext) {
|
|
60
|
-
const sc = ctx.sessionContext();
|
|
61
|
-
if (sc)
|
|
62
|
-
envelope.sessionContext = sc;
|
|
63
|
-
}
|
|
64
|
-
let result;
|
|
65
|
-
try {
|
|
66
|
-
result = await ctx.submit(envelope);
|
|
67
|
-
}
|
|
68
|
-
catch (err) {
|
|
69
|
-
// Defensive: a thrown submitter (should not happen — the live
|
|
70
|
-
// submitter catches everything) is treated as transient so the
|
|
71
|
-
// envelope lands in the queue.
|
|
72
|
-
const reason = err instanceof Error ? err.message : String(err);
|
|
73
|
-
result = { kind: 'transient', reason };
|
|
74
|
-
}
|
|
75
|
-
if (result.kind === 'ok') {
|
|
76
|
-
return { kind: 'submitted', envelope, httpStatus: result.httpStatus };
|
|
77
|
-
}
|
|
78
|
-
if (result.kind === 'transient') {
|
|
79
|
-
const path = enqueueFeedback(envelope, ctx.cwd);
|
|
80
|
-
return { kind: 'queued', envelope, path, reason: result.reason };
|
|
81
|
-
}
|
|
82
|
-
// permanent — log + drop
|
|
83
|
-
return {
|
|
84
|
-
kind: 'dropped',
|
|
85
|
-
envelope,
|
|
86
|
-
reason: result.reason,
|
|
87
|
-
httpStatus: result.httpStatus,
|
|
88
|
-
};
|
|
89
|
-
}
|
|
90
|
-
/**
|
|
91
|
-
* Render one human-readable toast for the operator. Centralised so the
|
|
92
|
-
* top-level `pugi feedback` shell handler + the in-REPL `/feedback`
|
|
93
|
-
* slash dispatcher agree on the copy.
|
|
94
|
-
*/
|
|
95
|
-
export function renderFeedbackToast(result) {
|
|
96
|
-
switch (result.kind) {
|
|
97
|
-
case 'submitted':
|
|
98
|
-
return 'Feedback submitted. Thank you.';
|
|
99
|
-
case 'queued':
|
|
100
|
-
return `Feedback queued locally. Will sync on next online run. (${result.path})`;
|
|
101
|
-
case 'cancelled':
|
|
102
|
-
return 'Feedback cancelled. Nothing was sent.';
|
|
103
|
-
case 'dropped':
|
|
104
|
-
return `Feedback rejected by server (${result.httpStatus}): ${result.reason}. Not queued.`;
|
|
105
|
-
case 'noop':
|
|
106
|
-
return `No feedback collected: ${result.reason}`;
|
|
107
|
-
}
|
|
108
|
-
}
|
|
109
|
-
/**
|
|
110
|
-
* Background queue flush. Invoked silently on session start so any
|
|
111
|
-
* envelopes that landed during an offline run get drained when the
|
|
112
|
-
* operator next has connectivity. The function never throws — it
|
|
113
|
-
* returns the flush stats so the caller can log them at debug level.
|
|
114
|
-
*/
|
|
115
|
-
export async function flushFeedbackQueueSilently(cwd, config) {
|
|
116
|
-
// Short-circuit when the queue file does not exist. Avoids a
|
|
117
|
-
// pointless `fs.stat` round-trip on every cold session start.
|
|
118
|
-
if (!existsSync(feedbackQueuePath(cwd))) {
|
|
119
|
-
return { attempted: 0, succeeded: 0, failed: 0 };
|
|
120
|
-
}
|
|
121
|
-
const result = await flushFeedbackQueue(cwd, async (env) => {
|
|
122
|
-
const r = await submitFeedback(env, config);
|
|
123
|
-
if (r.kind === 'ok')
|
|
124
|
-
return true;
|
|
125
|
-
if (r.kind === 'permanent') {
|
|
126
|
-
// Permanent failures are "done" from the queue's POV — they
|
|
127
|
-
// would never resolve on retry. Drop them so the queue does
|
|
128
|
-
// not grow without bound.
|
|
129
|
-
return true;
|
|
130
|
-
}
|
|
131
|
-
return false;
|
|
132
|
-
});
|
|
133
|
-
return {
|
|
134
|
-
attempted: result.attempted,
|
|
135
|
-
succeeded: result.succeeded,
|
|
136
|
-
failed: result.failed,
|
|
137
|
-
};
|
|
138
|
-
}
|
|
139
|
-
/**
|
|
140
|
-
* Re-exports — the spec imports these via the command module so the
|
|
141
|
-
* dependency graph in the test stays single-rooted.
|
|
142
|
-
*/
|
|
143
|
-
export { feedbackQueuePath, feedbackSubmitUrl, redactSessionContext, submitFeedback, };
|
|
144
|
-
/**
|
|
145
|
-
* Read the persona conversation log if present. Best-effort: returns
|
|
146
|
-
* an empty list when the file is missing or malformed. The CLI's REPL
|
|
147
|
-
* persists transcripts via the session module at a canonical relative
|
|
148
|
-
* path under `.pugi/sessions/`. The shell-level `pugi feedback` does
|
|
149
|
-
* not have access to a live session, so it tries to pick up the most
|
|
150
|
-
* recent persisted one for the `--with-context` path.
|
|
151
|
-
*
|
|
152
|
-
* Intentionally tolerant — feedback works even with no transcript.
|
|
153
|
-
*/
|
|
154
|
-
export function readMostRecentTranscript(cwd, options = {}) {
|
|
155
|
-
// The CLI may persist sessions in several places depending on the
|
|
156
|
-
// surface. We probe the conventional default; the spec drives the
|
|
157
|
-
// function via a fixture file instead of a live REPL.
|
|
158
|
-
const candidate = resolve(cwd, '.pugi', 'sessions', 'latest.jsonl');
|
|
159
|
-
if (!existsSync(candidate))
|
|
160
|
-
return [];
|
|
161
|
-
try {
|
|
162
|
-
const text = readFileSync(candidate, 'utf8');
|
|
163
|
-
const lines = text.split('\n').filter((l) => l.trim().length > 0);
|
|
164
|
-
const turns = [];
|
|
165
|
-
for (const line of lines) {
|
|
166
|
-
try {
|
|
167
|
-
const obj = JSON.parse(line);
|
|
168
|
-
if ((obj.role === 'user' || obj.role === 'assistant' || obj.role === 'system')
|
|
169
|
-
&& typeof obj.text === 'string') {
|
|
170
|
-
turns.push({ role: obj.role, text: obj.text });
|
|
171
|
-
}
|
|
172
|
-
}
|
|
173
|
-
catch {
|
|
174
|
-
// skip malformed line
|
|
175
|
-
}
|
|
176
|
-
}
|
|
177
|
-
const cap = options.maxTurns ?? 5;
|
|
178
|
-
return turns.slice(-cap);
|
|
179
|
-
}
|
|
180
|
-
catch {
|
|
181
|
-
return [];
|
|
182
|
-
}
|
|
183
|
-
}
|
|
184
|
-
//# sourceMappingURL=feedback.js.map
|
|
@@ -1,187 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* `pugi hooks` — operator surface for user-config hooks (MVP).
|
|
3
|
-
*
|
|
4
|
-
* Two subcommands ship in the MVP:
|
|
5
|
-
*
|
|
6
|
-
* pugi hooks list List configured hooks per event.
|
|
7
|
-
* pugi hooks doctor Validate the config and surface any
|
|
8
|
-
* parse / schema errors.
|
|
9
|
-
*
|
|
10
|
-
* Both accept `--json` for scripted callers. Argument grammar is
|
|
11
|
-
* intentionally narrow — no `add` / `remove` / `test` subcommands in
|
|
12
|
-
* the MVP. Operators hand-edit `~/.pugi/hooks-mvp.json` for now.
|
|
13
|
-
*
|
|
14
|
-
* Exit codes:
|
|
15
|
-
* 0 -> happy path (no hooks OR config valid).
|
|
16
|
-
* 1 -> config present but invalid (only `doctor` returns this).
|
|
17
|
-
* 2 -> unknown subcommand / argument error.
|
|
18
|
-
*
|
|
19
|
-
* Brand voice: ASCII only.
|
|
20
|
-
*/
|
|
21
|
-
import { ALL_HOOK_EVENTS_V2, defaultHooksMvpPath, loadHooksConfig, } from '../../core/hooks/index.js';
|
|
22
|
-
function parseFlags(args) {
|
|
23
|
-
const rest = [];
|
|
24
|
-
const flags = { json: false };
|
|
25
|
-
for (const arg of args) {
|
|
26
|
-
if (arg === '--json') {
|
|
27
|
-
flags.json = true;
|
|
28
|
-
continue;
|
|
29
|
-
}
|
|
30
|
-
rest.push(arg);
|
|
31
|
-
}
|
|
32
|
-
return { rest, flags };
|
|
33
|
-
}
|
|
34
|
-
/**
|
|
35
|
-
* Top-level dispatcher for `pugi hooks <subcommand>`. Returns the
|
|
36
|
-
* intended process exit code. `cli.ts` is expected to set
|
|
37
|
-
* `process.exitCode = <return value>` so error states propagate to
|
|
38
|
-
* scripted callers without throwing.
|
|
39
|
-
*/
|
|
40
|
-
export async function runHooksCommand(args, ctx) {
|
|
41
|
-
const { rest, flags } = parseFlags(args);
|
|
42
|
-
const sub = rest[0];
|
|
43
|
-
if (!sub || sub === 'help' || sub === '--help') {
|
|
44
|
-
emitUsage(ctx, flags);
|
|
45
|
-
return sub ? 0 : 2;
|
|
46
|
-
}
|
|
47
|
-
if (sub === 'list') {
|
|
48
|
-
return runList(ctx, flags);
|
|
49
|
-
}
|
|
50
|
-
if (sub === 'doctor') {
|
|
51
|
-
return runDoctor(ctx, flags);
|
|
52
|
-
}
|
|
53
|
-
ctx.writeOutput({ ok: false, error: `unknown subcommand: ${sub}` }, `pugi hooks: unknown subcommand '${sub}'. Try 'pugi hooks --help'.`);
|
|
54
|
-
return 2;
|
|
55
|
-
}
|
|
56
|
-
function emitUsage(ctx, flags) {
|
|
57
|
-
const text = [
|
|
58
|
-
'pugi hooks — user-config lifecycle hooks (MVP).',
|
|
59
|
-
'',
|
|
60
|
-
'Subcommands:',
|
|
61
|
-
' pugi hooks list Show hooks configured per event.',
|
|
62
|
-
' pugi hooks doctor Validate ~/.pugi/hooks-mvp.json.',
|
|
63
|
-
'',
|
|
64
|
-
'Flags:',
|
|
65
|
-
' --json Emit a JSON envelope instead of human text.',
|
|
66
|
-
'',
|
|
67
|
-
'Config file:',
|
|
68
|
-
' ~/.pugi/hooks-mvp.json',
|
|
69
|
-
'',
|
|
70
|
-
'Status:',
|
|
71
|
-
' MVP — 2 events out of 8. Remaining events (PostToolUse,',
|
|
72
|
-
" UserPromptSubmit, Stop, SubagentStop, PreCompact, Notification)",
|
|
73
|
-
' deferred to fast-follow PR.',
|
|
74
|
-
].join('\n');
|
|
75
|
-
ctx.writeOutput({
|
|
76
|
-
ok: true,
|
|
77
|
-
command: 'hooks',
|
|
78
|
-
usage: text,
|
|
79
|
-
}, text);
|
|
80
|
-
if (flags.json) {
|
|
81
|
-
// The structured payload is already emitted by writeOutput when
|
|
82
|
-
// --json is on; nothing extra to do.
|
|
83
|
-
}
|
|
84
|
-
}
|
|
85
|
-
function runList(ctx, flags) {
|
|
86
|
-
let config;
|
|
87
|
-
try {
|
|
88
|
-
config = loadHooksConfig(ctx.configPath);
|
|
89
|
-
}
|
|
90
|
-
catch (error) {
|
|
91
|
-
const msg = error.message;
|
|
92
|
-
ctx.writeOutput({ ok: false, error: msg }, `pugi hooks list: ${msg}\nFix the config or remove the file. Run 'pugi hooks doctor' for details.`);
|
|
93
|
-
return 1;
|
|
94
|
-
}
|
|
95
|
-
const perEvent = {
|
|
96
|
-
SessionStart: [],
|
|
97
|
-
PreToolUse: [],
|
|
98
|
-
PostToolUse: [],
|
|
99
|
-
UserPromptSubmit: [],
|
|
100
|
-
Stop: [],
|
|
101
|
-
SubagentStop: [],
|
|
102
|
-
PreCompact: [],
|
|
103
|
-
Notification: [],
|
|
104
|
-
// PUGI-487 - worktree lifecycle events.
|
|
105
|
-
WorktreeCreate: [],
|
|
106
|
-
WorktreeRemove: [],
|
|
107
|
-
};
|
|
108
|
-
for (const event of ALL_HOOK_EVENTS_V2) {
|
|
109
|
-
perEvent[event] = config.list(event).map((entry) => ({
|
|
110
|
-
matcher: entry.matcher,
|
|
111
|
-
command: entry.command,
|
|
112
|
-
timeoutMs: entry.timeoutMs,
|
|
113
|
-
blocking: entry.blocking,
|
|
114
|
-
}));
|
|
115
|
-
}
|
|
116
|
-
const total = Object.values(perEvent).reduce((acc, list) => acc + list.length, 0);
|
|
117
|
-
const payload = {
|
|
118
|
-
ok: true,
|
|
119
|
-
configPath: config.configPath(),
|
|
120
|
-
total,
|
|
121
|
-
perEvent,
|
|
122
|
-
};
|
|
123
|
-
if (flags.json) {
|
|
124
|
-
ctx.writeOutput(payload, JSON.stringify(payload, null, 2));
|
|
125
|
-
return 0;
|
|
126
|
-
}
|
|
127
|
-
const lines = [];
|
|
128
|
-
lines.push(`pugi hooks (${total} configured)`);
|
|
129
|
-
lines.push(` config: ${config.configPath()}`);
|
|
130
|
-
if (total === 0) {
|
|
131
|
-
lines.push(' no hooks configured — create the file above to add one.');
|
|
132
|
-
}
|
|
133
|
-
else {
|
|
134
|
-
for (const event of ALL_HOOK_EVENTS_V2) {
|
|
135
|
-
const list = perEvent[event];
|
|
136
|
-
if (list.length === 0)
|
|
137
|
-
continue;
|
|
138
|
-
lines.push(` ${event}:`);
|
|
139
|
-
for (const entry of list) {
|
|
140
|
-
const tags = [];
|
|
141
|
-
if (entry.matcher)
|
|
142
|
-
tags.push(`matcher=${entry.matcher}`);
|
|
143
|
-
if (entry.timeoutMs)
|
|
144
|
-
tags.push(`timeoutMs=${entry.timeoutMs}`);
|
|
145
|
-
if (entry.blocking)
|
|
146
|
-
tags.push('blocking');
|
|
147
|
-
const suffix = tags.length ? ` [${tags.join(', ')}]` : '';
|
|
148
|
-
lines.push(` - ${entry.command}${suffix}`);
|
|
149
|
-
}
|
|
150
|
-
}
|
|
151
|
-
}
|
|
152
|
-
const text = lines.join('\n');
|
|
153
|
-
ctx.writeOutput(payload, text);
|
|
154
|
-
return 0;
|
|
155
|
-
}
|
|
156
|
-
function runDoctor(ctx, flags) {
|
|
157
|
-
const path = ctx.configPath ?? defaultHooksMvpPath();
|
|
158
|
-
try {
|
|
159
|
-
const config = loadHooksConfig(ctx.configPath);
|
|
160
|
-
const total = config.flatten().length;
|
|
161
|
-
const payload = {
|
|
162
|
-
ok: true,
|
|
163
|
-
configPath: config.configPath(),
|
|
164
|
-
total,
|
|
165
|
-
issues: [],
|
|
166
|
-
};
|
|
167
|
-
const text = total
|
|
168
|
-
? `pugi hooks doctor: ${path} OK (${total} hooks).`
|
|
169
|
-
: `pugi hooks doctor: ${path} not present (no hooks configured).`;
|
|
170
|
-
ctx.writeOutput(payload, text);
|
|
171
|
-
return 0;
|
|
172
|
-
}
|
|
173
|
-
catch (error) {
|
|
174
|
-
const msg = error.message;
|
|
175
|
-
const payload = {
|
|
176
|
-
ok: false,
|
|
177
|
-
configPath: path,
|
|
178
|
-
error: msg,
|
|
179
|
-
};
|
|
180
|
-
const text = `pugi hooks doctor: ${msg}`;
|
|
181
|
-
ctx.writeOutput(payload, text);
|
|
182
|
-
return 1;
|
|
183
|
-
}
|
|
184
|
-
// flags.json is consumed by writeOutput in the host shell.
|
|
185
|
-
void flags;
|
|
186
|
-
}
|
|
187
|
-
//# sourceMappingURL=hooks.js.map
|
|
@@ -1,254 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* `pugi init` codebase scan → PUGI.md auto-gen (backlog #82).
|
|
3
|
-
*
|
|
4
|
-
* Extends the existing `init()` scaffold with a project-aware PUGI.md
|
|
5
|
-
* generator. Runs AFTER `scaffoldPugiWorkspace` so the `.pugi/` workspace
|
|
6
|
-
* exists for the `init-report.json` write.
|
|
7
|
-
*
|
|
8
|
-
* Flow:
|
|
9
|
-
*
|
|
10
|
-
* 1. Detect existing PUGI.md. If present without `--force`, refuse
|
|
11
|
-
* with `refused_existing`.
|
|
12
|
-
* 2. Run `detectWorkspace` against cwd.
|
|
13
|
-
* 3. Render PUGI.md content for the requested style.
|
|
14
|
-
* 4. Confirm with the operator unless `--yes` / `--json` / `--force`.
|
|
15
|
-
* 5. Write `<cwd>/PUGI.md` atomically (tmp + rename, fallback к
|
|
16
|
-
* direct write on EXDEV — happens когда tmp lives на a different
|
|
17
|
-
* volume).
|
|
18
|
-
* 6. Write `.pugi/init-report.json` (mode 0o600) with the raw
|
|
19
|
-
* detection payload for re-runs / auditing.
|
|
20
|
-
* 7. Emit a JSON envelope describing the outcome.
|
|
21
|
-
*/
|
|
22
|
-
import { chmodSync, existsSync, mkdirSync, readFileSync, renameSync, unlinkSync, writeFileSync, } from 'node:fs';
|
|
23
|
-
import { tmpdir } from 'node:os';
|
|
24
|
-
import { dirname, join, resolve } from 'node:path';
|
|
25
|
-
import { createInterface } from 'node:readline';
|
|
26
|
-
import { detectWorkspace } from '../../core/init/detector.js';
|
|
27
|
-
import { renderPugiMd, STYLE_WORD_CAPS, } from '../../core/init/template-renderer.js';
|
|
28
|
-
export const DEFAULT_PUGI_MD_STYLE = 'standard';
|
|
29
|
-
/** Entry point invoked by `cli.ts::init`. */
|
|
30
|
-
export async function runInitScanCommand(args, ctx) {
|
|
31
|
-
const cwd = resolve(ctx.cwd);
|
|
32
|
-
const pugiMdPath = join(cwd, 'PUGI.md');
|
|
33
|
-
const reportPath = join(cwd, '.pugi', 'init-report.json');
|
|
34
|
-
// `args` is currently consumed only via flags. Keep the parameter to
|
|
35
|
-
// mirror every other command surface so we can grow positional args
|
|
36
|
-
// (e.g. an explicit target path) without breaking the call site.
|
|
37
|
-
void args;
|
|
38
|
-
try {
|
|
39
|
-
if (existsSync(pugiMdPath) && !ctx.force) {
|
|
40
|
-
const envelope = {
|
|
41
|
-
command: 'init-scan',
|
|
42
|
-
status: 'refused_existing',
|
|
43
|
-
pugiMdPath,
|
|
44
|
-
reportPath,
|
|
45
|
-
style: ctx.style,
|
|
46
|
-
message: 'PUGI.md already exists. Re-run with --force to overwrite, or edit by hand.',
|
|
47
|
-
};
|
|
48
|
-
ctx.writeOutput(envelope, formatHuman(envelope));
|
|
49
|
-
return envelope;
|
|
50
|
-
}
|
|
51
|
-
const detection = detectWorkspace(cwd);
|
|
52
|
-
const rendered = renderPugiMd(detection, { style: ctx.style, now: ctx.now });
|
|
53
|
-
const shouldConfirm = !ctx.yes && !ctx.json && !ctx.force;
|
|
54
|
-
if (shouldConfirm) {
|
|
55
|
-
const confirm = ctx.confirm ?? defaultConfirm;
|
|
56
|
-
const ok = await confirm(`Write PUGI.md at ${pugiMdPath} (${rendered.wordCount} words, style: ${ctx.style})? [y/N] `);
|
|
57
|
-
if (!ok) {
|
|
58
|
-
const envelope = {
|
|
59
|
-
command: 'init-scan',
|
|
60
|
-
status: 'aborted',
|
|
61
|
-
pugiMdPath,
|
|
62
|
-
reportPath,
|
|
63
|
-
style: ctx.style,
|
|
64
|
-
message: 'Aborted by operator before write.',
|
|
65
|
-
};
|
|
66
|
-
ctx.writeOutput(envelope, formatHuman(envelope));
|
|
67
|
-
return envelope;
|
|
68
|
-
}
|
|
69
|
-
}
|
|
70
|
-
atomicWrite(pugiMdPath, rendered.content);
|
|
71
|
-
writeReport(reportPath, detection, rendered.wordCount, ctx.style);
|
|
72
|
-
const envelope = {
|
|
73
|
-
command: 'init-scan',
|
|
74
|
-
status: 'written',
|
|
75
|
-
pugiMdPath,
|
|
76
|
-
reportPath,
|
|
77
|
-
detection,
|
|
78
|
-
wordCount: rendered.wordCount,
|
|
79
|
-
style: ctx.style,
|
|
80
|
-
};
|
|
81
|
-
ctx.writeOutput(envelope, formatHuman(envelope));
|
|
82
|
-
return envelope;
|
|
83
|
-
}
|
|
84
|
-
catch (error) {
|
|
85
|
-
const message = error instanceof Error ? error.message : String(error);
|
|
86
|
-
const envelope = {
|
|
87
|
-
command: 'init-scan',
|
|
88
|
-
status: 'error',
|
|
89
|
-
pugiMdPath,
|
|
90
|
-
reportPath,
|
|
91
|
-
style: ctx.style,
|
|
92
|
-
message,
|
|
93
|
-
};
|
|
94
|
-
ctx.writeOutput(envelope, formatHuman(envelope));
|
|
95
|
-
return envelope;
|
|
96
|
-
}
|
|
97
|
-
}
|
|
98
|
-
/* ------------------------------------------------------------------ */
|
|
99
|
-
/* Atomic write */
|
|
100
|
-
/* ------------------------------------------------------------------ */
|
|
101
|
-
function atomicWrite(targetPath, content) {
|
|
102
|
-
const dir = dirname(targetPath);
|
|
103
|
-
mkdirSync(dir, { recursive: true });
|
|
104
|
-
const tmpName = `.pugi-md-${process.pid}-${Date.now()}.tmp`;
|
|
105
|
-
const tmpPath = join(tmpdir(), tmpName);
|
|
106
|
-
try {
|
|
107
|
-
writeFileSync(tmpPath, content, { encoding: 'utf8', mode: 0o644 });
|
|
108
|
-
renameSync(tmpPath, targetPath);
|
|
109
|
-
return;
|
|
110
|
-
}
|
|
111
|
-
catch (error) {
|
|
112
|
-
// EXDEV — cross-device link: tmpdir lives on a different volume
|
|
113
|
-
// than the workspace. Retry with the tmpfile inside the workspace.
|
|
114
|
-
cleanupTmp(tmpPath);
|
|
115
|
-
const code = error.code;
|
|
116
|
-
if (code !== 'EXDEV') {
|
|
117
|
-
// Fall through to the in-dir fallback for any rename failure so
|
|
118
|
-
// we keep the contract: PUGI.md is either fully written or not
|
|
119
|
-
// written. The direct writeFileSync at the end is the final
|
|
120
|
-
// safety net.
|
|
121
|
-
}
|
|
122
|
-
const localTmpPath = join(dir, `.pugi-md-${process.pid}-${Date.now()}.tmp`);
|
|
123
|
-
try {
|
|
124
|
-
writeFileSync(localTmpPath, content, { encoding: 'utf8', mode: 0o644 });
|
|
125
|
-
renameSync(localTmpPath, targetPath);
|
|
126
|
-
return;
|
|
127
|
-
}
|
|
128
|
-
catch {
|
|
129
|
-
cleanupTmp(localTmpPath);
|
|
130
|
-
}
|
|
131
|
-
// Final fallback — direct write. We sacrifice the atomicity guarantee
|
|
132
|
-
// here but keep behavioural correctness when both the tmp + in-dir
|
|
133
|
-
// strategies fail (rare; usually FS quirks).
|
|
134
|
-
writeFileSync(targetPath, content, { encoding: 'utf8', mode: 0o644 });
|
|
135
|
-
}
|
|
136
|
-
}
|
|
137
|
-
function cleanupTmp(path) {
|
|
138
|
-
try {
|
|
139
|
-
unlinkSync(path);
|
|
140
|
-
}
|
|
141
|
-
catch {
|
|
142
|
-
// best effort
|
|
143
|
-
}
|
|
144
|
-
}
|
|
145
|
-
function writeReport(reportPath, detection, wordCount, style) {
|
|
146
|
-
mkdirSync(dirname(reportPath), { recursive: true });
|
|
147
|
-
const payload = {
|
|
148
|
-
generatedAt: new Date().toISOString(),
|
|
149
|
-
style,
|
|
150
|
-
wordCap: STYLE_WORD_CAPS[style],
|
|
151
|
-
wordCount,
|
|
152
|
-
detection,
|
|
153
|
-
};
|
|
154
|
-
writeFileSync(reportPath, JSON.stringify(payload, null, 2), {
|
|
155
|
-
encoding: 'utf8',
|
|
156
|
-
mode: 0o600,
|
|
157
|
-
});
|
|
158
|
-
try {
|
|
159
|
-
chmodSync(reportPath, 0o600);
|
|
160
|
-
}
|
|
161
|
-
catch {
|
|
162
|
-
// Some filesystems (FAT, NTFS over CIFS) reject chmod; the mode
|
|
163
|
-
// option on writeFileSync is the primary guard. Best-effort here.
|
|
164
|
-
}
|
|
165
|
-
}
|
|
166
|
-
/* ------------------------------------------------------------------ */
|
|
167
|
-
/* Confirm prompt */
|
|
168
|
-
/* ------------------------------------------------------------------ */
|
|
169
|
-
async function defaultConfirm(question) {
|
|
170
|
-
if (process.stdin.isTTY !== true) {
|
|
171
|
-
return true;
|
|
172
|
-
}
|
|
173
|
-
const rl = createInterface({
|
|
174
|
-
input: process.stdin,
|
|
175
|
-
output: process.stderr,
|
|
176
|
-
});
|
|
177
|
-
return new Promise((resolveAnswer) => {
|
|
178
|
-
let settled = false;
|
|
179
|
-
const settle = (value) => {
|
|
180
|
-
if (settled)
|
|
181
|
-
return;
|
|
182
|
-
settled = true;
|
|
183
|
-
rl.close();
|
|
184
|
-
resolveAnswer(value);
|
|
185
|
-
};
|
|
186
|
-
rl.once('close', () => settle(false));
|
|
187
|
-
rl.question(question, (answer) => {
|
|
188
|
-
settle(/^y(es)?$/i.test(answer.trim()));
|
|
189
|
-
});
|
|
190
|
-
});
|
|
191
|
-
}
|
|
192
|
-
/* ------------------------------------------------------------------ */
|
|
193
|
-
/* Human-readable formatting */
|
|
194
|
-
/* ------------------------------------------------------------------ */
|
|
195
|
-
function formatHuman(env) {
|
|
196
|
-
switch (env.status) {
|
|
197
|
-
case 'written':
|
|
198
|
-
return [
|
|
199
|
-
`PUGI.md written to ${env.pugiMdPath}`,
|
|
200
|
-
` Style: ${env.style}`,
|
|
201
|
-
` Word count: ${env.wordCount ?? 0}`,
|
|
202
|
-
` Report: ${env.reportPath}`,
|
|
203
|
-
].join('\n');
|
|
204
|
-
case 'refused_existing':
|
|
205
|
-
return env.message ?? 'PUGI.md already exists. Use --force to overwrite.';
|
|
206
|
-
case 'aborted':
|
|
207
|
-
return env.message ?? 'Aborted.';
|
|
208
|
-
case 'error':
|
|
209
|
-
return `pugi init scan failed: ${env.message ?? 'unknown error'}`;
|
|
210
|
-
default:
|
|
211
|
-
return 'pugi init scan complete.';
|
|
212
|
-
}
|
|
213
|
-
}
|
|
214
|
-
/* ------------------------------------------------------------------ */
|
|
215
|
-
/* Helpers consumed by cli.ts */
|
|
216
|
-
/* ------------------------------------------------------------------ */
|
|
217
|
-
/** Parse `--style=<minimal|standard|detailed>` (or `--style minimal`). */
|
|
218
|
-
export function parseInitStyle(args) {
|
|
219
|
-
for (let i = 0; i < args.length; i += 1) {
|
|
220
|
-
const arg = args[i];
|
|
221
|
-
if (typeof arg !== 'string')
|
|
222
|
-
continue;
|
|
223
|
-
if (arg.startsWith('--style=')) {
|
|
224
|
-
const value = arg.slice('--style='.length);
|
|
225
|
-
const parsed = coerceStyle(value);
|
|
226
|
-
if (parsed)
|
|
227
|
-
return parsed;
|
|
228
|
-
}
|
|
229
|
-
else if (arg === '--style') {
|
|
230
|
-
const next = args[i + 1];
|
|
231
|
-
if (typeof next === 'string') {
|
|
232
|
-
const parsed = coerceStyle(next);
|
|
233
|
-
if (parsed)
|
|
234
|
-
return parsed;
|
|
235
|
-
}
|
|
236
|
-
}
|
|
237
|
-
}
|
|
238
|
-
return DEFAULT_PUGI_MD_STYLE;
|
|
239
|
-
}
|
|
240
|
-
function coerceStyle(value) {
|
|
241
|
-
if (value === 'minimal' || value === 'standard' || value === 'detailed')
|
|
242
|
-
return value;
|
|
243
|
-
return null;
|
|
244
|
-
}
|
|
245
|
-
/** Test surface: read back the on-disk report. */
|
|
246
|
-
export function readInitReport(reportPath) {
|
|
247
|
-
try {
|
|
248
|
-
return JSON.parse(readFileSync(reportPath, 'utf8'));
|
|
249
|
-
}
|
|
250
|
-
catch {
|
|
251
|
-
return null;
|
|
252
|
-
}
|
|
253
|
-
}
|
|
254
|
-
//# sourceMappingURL=init.js.map
|