@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,76 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Aggregate byte cap on the full rendered block. 96 KB = 3 files at
|
|
3
|
-
* the per-file cap, which is enough for cwd + parent + homedir while
|
|
4
|
-
* leaving plenty of prompt budget for the rest of the system prompt.
|
|
5
|
-
* Anything beyond is replaced with a truncation marker.
|
|
6
|
-
*/
|
|
7
|
-
export const MAX_INJECT_BYTES = 96 * 1024;
|
|
8
|
-
/**
|
|
9
|
-
* Marker line emitted when the aggregate cap is hit. Visible to the
|
|
10
|
-
* model so it knows ambient context was clipped; visible to the
|
|
11
|
-
* operator via the doctor probe so they can decide whether to trim
|
|
12
|
-
* their `PUGI.md` hierarchy.
|
|
13
|
-
*/
|
|
14
|
-
export const TRUNCATION_MARKER = '<ambient-context-truncated reason="aggregate-cap" />';
|
|
15
|
-
/**
|
|
16
|
-
* Render a HierarchyFile array into the system-prompt block. Returns
|
|
17
|
-
* `''` when `files` is empty. Each file becomes one
|
|
18
|
-
* `<ambient-context source="..." level="...">...</ambient-context>`
|
|
19
|
-
* stanza separated by a single newline.
|
|
20
|
-
*
|
|
21
|
-
* Determinism: same input always produces byte-identical output.
|
|
22
|
-
*/
|
|
23
|
-
export function renderAmbientContext(files) {
|
|
24
|
-
if (files.length === 0)
|
|
25
|
-
return '';
|
|
26
|
-
const stanzas = [];
|
|
27
|
-
let bytes = 0;
|
|
28
|
-
let truncated = false;
|
|
29
|
-
for (const file of files) {
|
|
30
|
-
const stanza = renderStanza(file);
|
|
31
|
-
const stanzaBytes = Buffer.byteLength(stanza, 'utf8') + 1; // newline join cost
|
|
32
|
-
if (bytes + stanzaBytes > MAX_INJECT_BYTES) {
|
|
33
|
-
truncated = true;
|
|
34
|
-
break;
|
|
35
|
-
}
|
|
36
|
-
stanzas.push(stanza);
|
|
37
|
-
bytes += stanzaBytes;
|
|
38
|
-
}
|
|
39
|
-
if (truncated)
|
|
40
|
-
stanzas.push(TRUNCATION_MARKER);
|
|
41
|
-
return stanzas.join('\n');
|
|
42
|
-
}
|
|
43
|
-
/**
|
|
44
|
-
* Build a single `<ambient-context>` stanza for one HierarchyFile.
|
|
45
|
-
* The `source` attribute carries the absolute path (after realpath)
|
|
46
|
-
* so the model can cite which file a piece of guidance came from
|
|
47
|
-
* when it explains its decisions to the operator.
|
|
48
|
-
*/
|
|
49
|
-
function renderStanza(file) {
|
|
50
|
-
const sourceAttr = escapeAttr(file.path);
|
|
51
|
-
const levelAttr = String(file.level);
|
|
52
|
-
// No trailing newline inside `content` — the join adds one between
|
|
53
|
-
// stanzas. Trimming the file's trailing whitespace keeps the tag
|
|
54
|
-
// close to the content for readability when an engineer dumps the
|
|
55
|
-
// assembled prompt for debugging.
|
|
56
|
-
const trimmed = file.content.replace(/\s+$/g, '');
|
|
57
|
-
return [
|
|
58
|
-
`<ambient-context source="${sourceAttr}" level="${levelAttr}">`,
|
|
59
|
-
trimmed,
|
|
60
|
-
`</ambient-context>`,
|
|
61
|
-
].join('\n');
|
|
62
|
-
}
|
|
63
|
-
/**
|
|
64
|
-
* Escape an XML attribute value. We expect operator-controlled paths
|
|
65
|
-
* (not adversarial input) but `&`, `"` and `<` are still possible in
|
|
66
|
-
* symlinked / unicode paths so we escape them defensively. The model
|
|
67
|
-
* has been trained to read this attribute as opaque metadata.
|
|
68
|
-
*/
|
|
69
|
-
function escapeAttr(value) {
|
|
70
|
-
return value
|
|
71
|
-
.replace(/&/g, '&')
|
|
72
|
-
.replace(/"/g, '"')
|
|
73
|
-
.replace(/</g, '<')
|
|
74
|
-
.replace(/>/g, '>');
|
|
75
|
-
}
|
|
76
|
-
//# sourceMappingURL=context-injector.js.map
|
|
@@ -1,207 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* — `PUGI.md` hierarchy walk-up to `$HOME`.
|
|
3
|
-
*
|
|
4
|
-
* the upstream tool walks from `cwd` upward toward the user's homedir and
|
|
5
|
-
* concatenates every `CLAUDE.md` it finds at each intermediate level
|
|
6
|
-
* (deepest overrides shallowest). Pugi parity: same walk, looking for
|
|
7
|
-
* `PUGI.md` first at each level and accepting `CLAUDE.md` as a fallback
|
|
8
|
-
* — operators often have a leftover `~/CLAUDE.md` or a parent-dir
|
|
9
|
-
* `CLAUDE.md` from a previous the upstream tool session and we want their
|
|
10
|
-
* ambient guidance picked up automatically without a migration step.
|
|
11
|
-
*
|
|
12
|
-
* Why this is a separate module from `core/context/markdown-traverse.ts`:
|
|
13
|
-
*
|
|
14
|
-
* - `markdown-traverse.ts` is the *workspace-bounded* walk (cwd → up
|
|
15
|
-
* to but NOT including `workspaceRoot`). It guards every read by
|
|
16
|
-
* `realpathSync` containment against the workspace root and
|
|
17
|
-
* refuses to escape — by design, because the per-dir markdown is
|
|
18
|
-
* part of the project's first-party context.
|
|
19
|
-
*
|
|
20
|
-
* - This module is the *home-bounded* walk (cwd → up to `homedir()`,
|
|
21
|
-
* OR until depth limit). It picks up the operator's personal /
|
|
22
|
-
* global guidance that lives ABOVE the workspace root. The two
|
|
23
|
-
* surfaces are complementary: workspace markdown encodes project
|
|
24
|
-
* conventions; this hierarchy walk encodes operator-level taste
|
|
25
|
-
* (preferred libraries, "always run prettier", style guides).
|
|
26
|
-
*
|
|
27
|
-
* Contract:
|
|
28
|
-
*
|
|
29
|
-
* - Walks from `cwd` upward. At each directory checks `PUGI.md`
|
|
30
|
-
* (preferred); when absent falls back to `CLAUDE.md`. Only ONE
|
|
31
|
-
* file per level is loaded — preferred wins.
|
|
32
|
-
* - Stops at `homedir()` INCLUSIVE — the file at `~/PUGI.md` or
|
|
33
|
-
* `~/CLAUDE.md` IS loaded (the upstream tool parity: a `~/CLAUDE.md`
|
|
34
|
-
* applies to every project the operator opens).
|
|
35
|
-
* - Hard depth cap of `MAX_WALK_DEPTH` (20) directories regardless
|
|
36
|
-
* of how far cwd is from homedir; defense against symlinked or
|
|
37
|
-
* malicious cwd values.
|
|
38
|
-
* - Per-file byte cap `MAX_FILE_BYTES` (32 KB); over-cap files are
|
|
39
|
-
* truncated, not rejected, so a runaway `PUGI.md` does not break
|
|
40
|
-
* the prompt budget.
|
|
41
|
-
* - Returns shallow-to-deep order (cwd FIRST, homedir LAST). The
|
|
42
|
-
* caller is responsible for rendering precedence — the upstream tool's
|
|
43
|
-
* rule is "deeper overrides shallower", which means the LAST
|
|
44
|
-
* entry in the rendered system prompt wins. Our order matches
|
|
45
|
-
* that convention so the context injector can splice directly.
|
|
46
|
-
*
|
|
47
|
-
* Safety:
|
|
48
|
-
*
|
|
49
|
-
* - No `realpath` on the directories themselves: the operator's
|
|
50
|
-
* cwd may live under a workspace symlink (common with macOS
|
|
51
|
-
* `/private/var/...`) and we want to honor what the operator
|
|
52
|
-
* sees in their shell. We DO resolve the candidate file via
|
|
53
|
-
* `realpathSync` before reading, but only to defeat
|
|
54
|
-
* symlinks-pointing-outside-homedir attacks; an off-tree symlink
|
|
55
|
-
* is skipped silently.
|
|
56
|
-
* - Catch + skip every fs error per file. The walk-up surface MUST
|
|
57
|
-
* NEVER break engine boot — missing read perms on a parent dir
|
|
58
|
-
* is the common case (e.g. `/etc` on a corp laptop) and the
|
|
59
|
-
* fallback is "no ambient context", not a crash.
|
|
60
|
-
*
|
|
61
|
-
* Pure module: no logging, no network, no fs writes.
|
|
62
|
-
*/
|
|
63
|
-
import { existsSync, readFileSync, realpathSync, statSync } from 'node:fs';
|
|
64
|
-
import { dirname, resolve } from 'node:path';
|
|
65
|
-
/**
|
|
66
|
-
* Hard ceiling on parent-dir traversal depth. 20 is generous — even
|
|
67
|
-
* deep monorepo layouts rarely sit more than 8-10 levels below the
|
|
68
|
-
* homedir on a developer's laptop. The cap exists so a misconfigured
|
|
69
|
-
* cwd (e.g. cwd outside the user's home filesystem entirely) cannot
|
|
70
|
-
* cause a multi-second fs scan of unrelated directories.
|
|
71
|
-
*/
|
|
72
|
-
export const MAX_WALK_DEPTH = 20;
|
|
73
|
-
/**
|
|
74
|
-
* Per-file byte cap. 32 KB matches the per-dir markdown traverse
|
|
75
|
-
* aggregate budget — generous enough for a fully written-out
|
|
76
|
-
* project / personal `PUGI.md` (~8000 words) while keeping any one
|
|
77
|
-
* file from blowing the prompt budget on its own.
|
|
78
|
-
*/
|
|
79
|
-
export const MAX_FILE_BYTES = 32 * 1024;
|
|
80
|
-
/**
|
|
81
|
-
* Filenames consulted at each level, in lookup order. `PUGI.md` is
|
|
82
|
-
* preferred — when both files coexist in a directory the Pugi-native
|
|
83
|
-
* file wins and the the upstream tool shim is ignored. This is the same
|
|
84
|
-
* precedence used by `markdown-traverse.ts` for workspace-bounded
|
|
85
|
-
* walks; keeping the two surfaces consistent removes the "why does
|
|
86
|
-
* Pugi sometimes read CLAUDE.md and sometimes PUGI.md?" foot-gun.
|
|
87
|
-
*/
|
|
88
|
-
export const HIERARCHY_SOURCES = ['PUGI.md', 'CLAUDE.md'];
|
|
89
|
-
/**
|
|
90
|
-
* Walk from `cwd` upward, collecting ambient `PUGI.md` / `CLAUDE.md`
|
|
91
|
-
* files at each level until we reach the homedir (inclusive) or the
|
|
92
|
-
* depth cap.
|
|
93
|
-
*
|
|
94
|
-
* Returns an array ordered shallowest-first (cwd → homedir). When no
|
|
95
|
-
* files are found, returns `[]`. When `cwd` is OUTSIDE the homedir
|
|
96
|
-
* tree (e.g. the operator runs `pugi` from `/tmp`), the walk still
|
|
97
|
-
* proceeds upward but stops the moment we reach a filesystem root
|
|
98
|
-
* without ever entering the homedir — useful for ops/admin invocations
|
|
99
|
-
* where there is genuinely no personal context to load.
|
|
100
|
-
*/
|
|
101
|
-
export function walkUpPugiMd(cwd, opts = {}) {
|
|
102
|
-
const limit = clampLimit(opts.limit);
|
|
103
|
-
const home = opts.homedir;
|
|
104
|
-
let absCwd;
|
|
105
|
-
try {
|
|
106
|
-
absCwd = resolve(cwd);
|
|
107
|
-
}
|
|
108
|
-
catch {
|
|
109
|
-
return [];
|
|
110
|
-
}
|
|
111
|
-
const absHome = home ? resolve(home) : undefined;
|
|
112
|
-
const results = [];
|
|
113
|
-
let current = absCwd;
|
|
114
|
-
let level = 0;
|
|
115
|
-
const visited = new Set();
|
|
116
|
-
while (level <= limit) {
|
|
117
|
-
if (visited.has(current))
|
|
118
|
-
break; // pathological symlink loop guard
|
|
119
|
-
visited.add(current);
|
|
120
|
-
const found = tryLoadDirectory(current, level);
|
|
121
|
-
if (found)
|
|
122
|
-
results.push(found);
|
|
123
|
-
// Inclusive home boundary: load home if we are here, then stop.
|
|
124
|
-
if (absHome && current === absHome)
|
|
125
|
-
break;
|
|
126
|
-
const parent = dirname(current);
|
|
127
|
-
if (parent === current)
|
|
128
|
-
break; // hit filesystem root before homedir
|
|
129
|
-
current = parent;
|
|
130
|
-
level += 1;
|
|
131
|
-
}
|
|
132
|
-
return results;
|
|
133
|
-
}
|
|
134
|
-
/**
|
|
135
|
-
* Pick the first matching file in `dir`, read + cap it, and produce
|
|
136
|
-
* a HierarchyFile row. Returns `undefined` when no file in
|
|
137
|
-
* `HIERARCHY_SOURCES` exists or all reads error out (perms, symlink
|
|
138
|
-
* escape, etc.). NEVER throws — fs errors degrade to "no file at
|
|
139
|
-
* this level".
|
|
140
|
-
*/
|
|
141
|
-
function tryLoadDirectory(dir, level) {
|
|
142
|
-
for (const source of HIERARCHY_SOURCES) {
|
|
143
|
-
const candidate = resolve(dir, source);
|
|
144
|
-
if (!existsSync(candidate))
|
|
145
|
-
continue;
|
|
146
|
-
// Realpath the FILE to defeat symlink-points-elsewhere attacks.
|
|
147
|
-
// We do not realpath the directory itself — operators often run
|
|
148
|
-
// pugi from inside a workspace symlink and the walk should honor
|
|
149
|
-
// the path they see.
|
|
150
|
-
let realPath;
|
|
151
|
-
try {
|
|
152
|
-
realPath = realpathSync(candidate);
|
|
153
|
-
}
|
|
154
|
-
catch {
|
|
155
|
-
// Broken symlink or perms issue on the link itself. Skip this
|
|
156
|
-
// file and try the next source in the same directory.
|
|
157
|
-
continue;
|
|
158
|
-
}
|
|
159
|
-
let rawBytes;
|
|
160
|
-
try {
|
|
161
|
-
rawBytes = statSync(realPath).size;
|
|
162
|
-
}
|
|
163
|
-
catch {
|
|
164
|
-
continue;
|
|
165
|
-
}
|
|
166
|
-
let content;
|
|
167
|
-
try {
|
|
168
|
-
content = readFileSync(realPath, 'utf8');
|
|
169
|
-
}
|
|
170
|
-
catch {
|
|
171
|
-
continue;
|
|
172
|
-
}
|
|
173
|
-
let truncated = false;
|
|
174
|
-
if (Buffer.byteLength(content, 'utf8') > MAX_FILE_BYTES) {
|
|
175
|
-
// Trim by character index sized to the byte cap. Mild over-trim
|
|
176
|
-
// on multi-byte boundaries is acceptable — we never under-trim
|
|
177
|
-
// (we'd exceed the cap) and the truncation is operator-visible
|
|
178
|
-
// via the `truncated` flag.
|
|
179
|
-
content = content.slice(0, MAX_FILE_BYTES);
|
|
180
|
-
truncated = true;
|
|
181
|
-
}
|
|
182
|
-
return {
|
|
183
|
-
path: realPath,
|
|
184
|
-
content,
|
|
185
|
-
level,
|
|
186
|
-
source,
|
|
187
|
-
truncated,
|
|
188
|
-
rawBytes,
|
|
189
|
-
};
|
|
190
|
-
}
|
|
191
|
-
return undefined;
|
|
192
|
-
}
|
|
193
|
-
/**
|
|
194
|
-
* Bound the limit to `[0, MAX_WALK_DEPTH]`. A negative or zero value
|
|
195
|
-
* still permits the cwd-level file to load (level 0 is always
|
|
196
|
-
* considered) — passing `limit: 0` means "current directory only".
|
|
197
|
-
*/
|
|
198
|
-
function clampLimit(limit) {
|
|
199
|
-
if (typeof limit !== 'number' || !Number.isFinite(limit))
|
|
200
|
-
return MAX_WALK_DEPTH;
|
|
201
|
-
if (limit < 0)
|
|
202
|
-
return 0;
|
|
203
|
-
if (limit > MAX_WALK_DEPTH)
|
|
204
|
-
return MAX_WALK_DEPTH;
|
|
205
|
-
return Math.floor(limit);
|
|
206
|
-
}
|
|
207
|
-
//# sourceMappingURL=walk-up.js.map
|
|
@@ -1,270 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* uv binary downloader + cache для Python skills runtime (#175 follow-up).
|
|
3
|
-
*
|
|
4
|
-
* Extends `uv-resolver.ts`: that module locates an existing uv on disk
|
|
5
|
-
* across 4 tiers; this module populates the second tier
|
|
6
|
-
* (`~/.pugi/bin/uv`) by fetching the official Astral release asset,
|
|
7
|
-
* verifying SHA-256, atomically installing к the bundled location, и
|
|
8
|
-
* leaving a `.uv-version` marker so later runs can tell which version
|
|
9
|
-
* landed without spawning the binary.
|
|
10
|
-
*
|
|
11
|
-
* Design choices:
|
|
12
|
-
* - Pin a single version (`DEFAULT_UV_VERSION`) с frozen SHA-256
|
|
13
|
-
* table. Multi-version cache + auto-upgrade is out of scope here.
|
|
14
|
-
* - tarball-only support (.tar.gz). Linux + macOS only. Windows ships
|
|
15
|
-
* `detectUvAsset` mapping but `downloadUv` throws unsupported when
|
|
16
|
-
* handed the .zip — Windows install today goes through
|
|
17
|
-
* `PUGI_UV_PATH` or Astral's PowerShell installer (uvInstallHint).
|
|
18
|
-
* A real .zip extraction primitive lands в a follow-up.
|
|
19
|
-
* - Atomic install: download к `uv.tmp.<rand>`, fsync directory,
|
|
20
|
-
* rename к `uv`. Checksum mismatch deletes the tmp и throws —
|
|
21
|
-
* final path is never partially written.
|
|
22
|
-
* - All network I/O goes through the `opts.fetch` seam so the spec
|
|
23
|
-
* never hits GitHub. Tests inject a stub Response с canned bytes.
|
|
24
|
-
*
|
|
25
|
-
* Refreshing pinned checksums:
|
|
26
|
-
* gh release download <ver> --repo astral-sh/uv \
|
|
27
|
-
* -p 'uv-aarch64-apple-darwin.tar.gz.sha256' \
|
|
28
|
-
* -p 'uv-x86_64-apple-darwin.tar.gz.sha256' \
|
|
29
|
-
* -p 'uv-x86_64-unknown-linux-gnu.tar.gz.sha256' \
|
|
30
|
-
* -p 'uv-aarch64-unknown-linux-gnu.tar.gz.sha256' \
|
|
31
|
-
* -p 'uv-x86_64-pc-windows-msvc.zip.sha256' -D /tmp/uv-sums
|
|
32
|
-
* for f in /tmp/uv-sums/*.sha256; do head -1 "$f"; done
|
|
33
|
-
* Then paste the hex digests into `PINNED_CHECKSUMS` below.
|
|
34
|
-
*/
|
|
35
|
-
import { createHash, randomBytes } from 'node:crypto';
|
|
36
|
-
import { chmod, mkdir, open, rename, rm, stat, unlink, writeFile, readFile, } from 'node:fs/promises';
|
|
37
|
-
import { Readable } from 'node:stream';
|
|
38
|
-
import { pipeline } from 'node:stream/promises';
|
|
39
|
-
import { createWriteStream } from 'node:fs';
|
|
40
|
-
import { homedir } from 'node:os';
|
|
41
|
-
import path from 'node:path';
|
|
42
|
-
import { x as tarExtract } from 'tar';
|
|
43
|
-
export const DEFAULT_UV_VERSION = '0.5.11';
|
|
44
|
-
/**
|
|
45
|
-
* SHA-256 digests pinned per supported asset для `DEFAULT_UV_VERSION`.
|
|
46
|
-
*
|
|
47
|
-
* Source-of-truth: <https://github.com/astral-sh/uv/releases/download/
|
|
48
|
-
* 0.5.11/{asset}.sha256>. Verified. If you bump
|
|
49
|
-
* DEFAULT_UV_VERSION, refresh these via the gh-release-download recipe
|
|
50
|
-
* в the file header.
|
|
51
|
-
*/
|
|
52
|
-
const PINNED_CHECKSUMS = Object.freeze({
|
|
53
|
-
'0.5.11': Object.freeze({
|
|
54
|
-
'darwin-arm64': '695f3640d5b1a4e28de7e36e3a2e14072852dcc6c70bf9e4deec6ada00d516b4',
|
|
55
|
-
'darwin-x64': '7e23d1d892c23f9e74245c4fd3d3e246438ce9b34460f85eee61f784de137b0b',
|
|
56
|
-
'linux-x64': '14411de26cdea5f5139fafaf2b675b1c633e744dd49c6d6a9fc8817ec065158b',
|
|
57
|
-
'linux-arm64': '055c329c38a93c01d378349d51cb4d521d1998c8a79355ddc00f863ce451942f',
|
|
58
|
-
'win32-x64': '3e8203e6434b45427f20824419f8d8d53f970a76d94ccdcad07f8498fa01a9d0',
|
|
59
|
-
}),
|
|
60
|
-
});
|
|
61
|
-
const ASSET_TEMPLATES = Object.freeze({
|
|
62
|
-
'darwin-arm64': {
|
|
63
|
-
name: 'uv-aarch64-apple-darwin.tar.gz',
|
|
64
|
-
archive: 'tar.gz',
|
|
65
|
-
innerDir: 'uv-aarch64-apple-darwin',
|
|
66
|
-
exeName: 'uv',
|
|
67
|
-
},
|
|
68
|
-
'darwin-x64': {
|
|
69
|
-
name: 'uv-x86_64-apple-darwin.tar.gz',
|
|
70
|
-
archive: 'tar.gz',
|
|
71
|
-
innerDir: 'uv-x86_64-apple-darwin',
|
|
72
|
-
exeName: 'uv',
|
|
73
|
-
},
|
|
74
|
-
'linux-x64': {
|
|
75
|
-
name: 'uv-x86_64-unknown-linux-gnu.tar.gz',
|
|
76
|
-
archive: 'tar.gz',
|
|
77
|
-
innerDir: 'uv-x86_64-unknown-linux-gnu',
|
|
78
|
-
exeName: 'uv',
|
|
79
|
-
},
|
|
80
|
-
'linux-arm64': {
|
|
81
|
-
name: 'uv-aarch64-unknown-linux-gnu.tar.gz',
|
|
82
|
-
archive: 'tar.gz',
|
|
83
|
-
innerDir: 'uv-aarch64-unknown-linux-gnu',
|
|
84
|
-
exeName: 'uv',
|
|
85
|
-
},
|
|
86
|
-
'win32-x64': {
|
|
87
|
-
name: 'uv-x86_64-pc-windows-msvc.zip',
|
|
88
|
-
archive: 'zip',
|
|
89
|
-
innerDir: '',
|
|
90
|
-
exeName: 'uv.exe',
|
|
91
|
-
},
|
|
92
|
-
});
|
|
93
|
-
/**
|
|
94
|
-
* Translate a (platform, arch) pair к the canonical uv release asset.
|
|
95
|
-
*
|
|
96
|
-
* Throws when the host is unsupported (e.g. linux-arm 32-bit) — the
|
|
97
|
-
* caller should surface `uvInstallHint()` instead of attempting a
|
|
98
|
-
* download.
|
|
99
|
-
*/
|
|
100
|
-
export function detectUvAsset(platform, arch, version = DEFAULT_UV_VERSION) {
|
|
101
|
-
const key = mapPlatformArch(platform, arch);
|
|
102
|
-
if (!key) {
|
|
103
|
-
throw new Error(`uv installer: unsupported host platform=${platform} arch=${arch}. ` +
|
|
104
|
-
`Set PUGI_UV_PATH к an existing uv binary or install via the official script.`);
|
|
105
|
-
}
|
|
106
|
-
const tmpl = ASSET_TEMPLATES[key];
|
|
107
|
-
const sumTable = PINNED_CHECKSUMS[version];
|
|
108
|
-
if (!sumTable) {
|
|
109
|
-
throw new Error(`uv installer: no pinned checksums for version ${version}. ` +
|
|
110
|
-
`Either pass a version с entries в PINNED_CHECKSUMS or refresh the table.`);
|
|
111
|
-
}
|
|
112
|
-
const sha256 = sumTable[key];
|
|
113
|
-
if (!sha256) {
|
|
114
|
-
throw new Error(`uv installer: missing pinned sha256 for ${version}/${key}.`);
|
|
115
|
-
}
|
|
116
|
-
return {
|
|
117
|
-
assetName: tmpl.name,
|
|
118
|
-
url: `https://github.com/astral-sh/uv/releases/download/${version}/${tmpl.name}`,
|
|
119
|
-
sha256,
|
|
120
|
-
exeName: tmpl.exeName,
|
|
121
|
-
archive: tmpl.archive,
|
|
122
|
-
innerDir: tmpl.innerDir,
|
|
123
|
-
};
|
|
124
|
-
}
|
|
125
|
-
/**
|
|
126
|
-
* Download + verify + install uv к `<homeDir>/.pugi/bin/uv`.
|
|
127
|
-
* Atomic: nothing visible to the resolver lands until checksum passes.
|
|
128
|
-
*/
|
|
129
|
-
export async function downloadUv(opts = {}) {
|
|
130
|
-
const version = opts.version ?? DEFAULT_UV_VERSION;
|
|
131
|
-
const home = opts.homeDir ?? homedir();
|
|
132
|
-
const plat = opts.platform ?? process.platform;
|
|
133
|
-
const arch = opts.arch ?? process.arch;
|
|
134
|
-
const httpFetch = opts.fetch ?? fetch;
|
|
135
|
-
const asset = detectUvAsset(plat, arch, version);
|
|
136
|
-
if (asset.archive !== 'tar.gz') {
|
|
137
|
-
// Windows .zip extraction lands в a follow-up. Today, Windows
|
|
138
|
-
// users go through PUGI_UV_PATH or Astral's PowerShell installer.
|
|
139
|
-
throw new Error(`uv installer: archive=${asset.archive} not supported yet. ` +
|
|
140
|
-
`On Windows, run PowerShell: irm https://astral.sh/uv/install.ps1 | iex, ` +
|
|
141
|
-
`then point PUGI_UV_PATH at the resulting uv.exe.`);
|
|
142
|
-
}
|
|
143
|
-
const binDir = path.join(home, '.pugi', 'bin');
|
|
144
|
-
await mkdir(binDir, { recursive: true });
|
|
145
|
-
// Download к a randomly-named tmp в the same dir so rename() is atomic.
|
|
146
|
-
const tmpArchive = path.join(binDir, `.uv-archive.${randomBytes(8).toString('hex')}.tar.gz`);
|
|
147
|
-
const tmpExtractDir = path.join(binDir, `.uv-extract.${randomBytes(8).toString('hex')}`);
|
|
148
|
-
const finalBinary = path.join(binDir, asset.exeName);
|
|
149
|
-
const tmpBinary = `${finalBinary}.tmp.${randomBytes(8).toString('hex')}`;
|
|
150
|
-
const versionMarker = path.join(binDir, '.uv-version');
|
|
151
|
-
let downloadedHash = '';
|
|
152
|
-
try {
|
|
153
|
-
// 1. Fetch + stream к tmp file while hashing on the fly.
|
|
154
|
-
const response = await httpFetch(asset.url);
|
|
155
|
-
if (!response.ok) {
|
|
156
|
-
throw new Error(`uv installer: download failed ${response.status} ${response.statusText} for ${asset.url}`);
|
|
157
|
-
}
|
|
158
|
-
if (!response.body) {
|
|
159
|
-
throw new Error(`uv installer: fetch returned empty body for ${asset.url}`);
|
|
160
|
-
}
|
|
161
|
-
downloadedHash = await streamToFileWithHash(response.body, tmpArchive);
|
|
162
|
-
// 2. Verify SHA-256 BEFORE touching extraction.
|
|
163
|
-
const expectedSha = (opts.expectedSha256 ?? asset.sha256).toLowerCase();
|
|
164
|
-
if (downloadedHash.toLowerCase() !== expectedSha) {
|
|
165
|
-
throw new Error(`uv installer: SHA-256 mismatch for ${asset.assetName}. ` +
|
|
166
|
-
`Expected ${expectedSha}, got ${downloadedHash}. ` +
|
|
167
|
-
`Refusing к install — checksum table may be stale or download was tampered with.`);
|
|
168
|
-
}
|
|
169
|
-
// 3. Extract just the uv binary к tmpExtractDir.
|
|
170
|
-
await mkdir(tmpExtractDir, { recursive: true });
|
|
171
|
-
const innerPath = asset.innerDir
|
|
172
|
-
? `${asset.innerDir}/${asset.exeName}`
|
|
173
|
-
: asset.exeName;
|
|
174
|
-
await tarExtract({
|
|
175
|
-
file: tmpArchive,
|
|
176
|
-
cwd: tmpExtractDir,
|
|
177
|
-
filter: (entryPath) => normalizeEntryPath(entryPath) === innerPath,
|
|
178
|
-
});
|
|
179
|
-
const extractedBinary = path.join(tmpExtractDir, innerPath);
|
|
180
|
-
const extractedStat = await stat(extractedBinary).catch(() => null);
|
|
181
|
-
if (!extractedStat || !extractedStat.isFile()) {
|
|
182
|
-
throw new Error(`uv installer: archive did not contain expected entry ${innerPath}`);
|
|
183
|
-
}
|
|
184
|
-
// 4. Move binary к its final tmp slot в binDir и chmod.
|
|
185
|
-
await rename(extractedBinary, tmpBinary);
|
|
186
|
-
await chmod(tmpBinary, 0o755);
|
|
187
|
-
// 5. Atomic swap — rename() over any existing uv binary.
|
|
188
|
-
await rename(tmpBinary, finalBinary);
|
|
189
|
-
// 6. Persist version marker AFTER binary lands so a partial install
|
|
190
|
-
// never reports a stale version.
|
|
191
|
-
await writeFile(versionMarker, `${version}\n`, 'utf8');
|
|
192
|
-
return { installedAt: finalBinary, version };
|
|
193
|
-
}
|
|
194
|
-
catch (err) {
|
|
195
|
-
// Best-effort cleanup of any tmp artefacts. Final binary is left
|
|
196
|
-
// alone — either it was atomically swapped (success path) or it
|
|
197
|
-
// was never touched.
|
|
198
|
-
await unlink(tmpBinary).catch(() => undefined);
|
|
199
|
-
throw err;
|
|
200
|
-
}
|
|
201
|
-
finally {
|
|
202
|
-
await unlink(tmpArchive).catch(() => undefined);
|
|
203
|
-
await rm(tmpExtractDir, { recursive: true, force: true }).catch(() => undefined);
|
|
204
|
-
}
|
|
205
|
-
}
|
|
206
|
-
/**
|
|
207
|
-
* Read the `.uv-version` marker. Returns null when the marker is
|
|
208
|
-
* absent OR unreadable — callers should treat a null as "not installed
|
|
209
|
-
* via Pugi" и fall back к the resolver.
|
|
210
|
-
*/
|
|
211
|
-
export async function getInstalledVersion(home) {
|
|
212
|
-
const dir = home ?? homedir();
|
|
213
|
-
const marker = path.join(dir, '.pugi', 'bin', '.uv-version');
|
|
214
|
-
try {
|
|
215
|
-
const raw = await readFile(marker, 'utf8');
|
|
216
|
-
const trimmed = raw.trim();
|
|
217
|
-
return trimmed.length > 0 ? trimmed : null;
|
|
218
|
-
}
|
|
219
|
-
catch {
|
|
220
|
-
return null;
|
|
221
|
-
}
|
|
222
|
-
}
|
|
223
|
-
// --- internals --------------------------------------------------------------
|
|
224
|
-
function mapPlatformArch(platform, arch) {
|
|
225
|
-
if (platform === 'darwin' && arch === 'arm64')
|
|
226
|
-
return 'darwin-arm64';
|
|
227
|
-
if (platform === 'darwin' && arch === 'x64')
|
|
228
|
-
return 'darwin-x64';
|
|
229
|
-
if (platform === 'linux' && arch === 'x64')
|
|
230
|
-
return 'linux-x64';
|
|
231
|
-
if (platform === 'linux' && arch === 'arm64')
|
|
232
|
-
return 'linux-arm64';
|
|
233
|
-
if (platform === 'win32' && arch === 'x64')
|
|
234
|
-
return 'win32-x64';
|
|
235
|
-
return null;
|
|
236
|
-
}
|
|
237
|
-
/**
|
|
238
|
-
* Stream a ReadableStream к disk while computing SHA-256. Returns
|
|
239
|
-
* the hex digest. Throws on any I/O error after best-effort cleanup
|
|
240
|
-
* of the partial file.
|
|
241
|
-
*/
|
|
242
|
-
async function streamToFileWithHash(body, dest) {
|
|
243
|
-
const hash = createHash('sha256');
|
|
244
|
-
const fileHandle = await open(dest, 'w');
|
|
245
|
-
const fileStream = createWriteStream(dest, { fd: fileHandle.fd, autoClose: false });
|
|
246
|
-
try {
|
|
247
|
-
// Node's `Readable.fromWeb` adapts a fetch body к a Node stream.
|
|
248
|
-
const nodeStream = Readable.fromWeb(body);
|
|
249
|
-
nodeStream.on('data', (chunk) => hash.update(chunk));
|
|
250
|
-
await pipeline(nodeStream, fileStream);
|
|
251
|
-
await fileHandle.sync().catch(() => undefined);
|
|
252
|
-
return hash.digest('hex');
|
|
253
|
-
}
|
|
254
|
-
finally {
|
|
255
|
-
await fileHandle.close().catch(() => undefined);
|
|
256
|
-
}
|
|
257
|
-
}
|
|
258
|
-
/**
|
|
259
|
-
* Normalize archive entry paths. tar entries may carry a leading `./`
|
|
260
|
-
* or trailing slash; the filter compares against `<innerDir>/<exe>`.
|
|
261
|
-
*/
|
|
262
|
-
function normalizeEntryPath(entryPath) {
|
|
263
|
-
let p = entryPath;
|
|
264
|
-
if (p.startsWith('./'))
|
|
265
|
-
p = p.slice(2);
|
|
266
|
-
if (p.endsWith('/'))
|
|
267
|
-
p = p.slice(0, -1);
|
|
268
|
-
return p;
|
|
269
|
-
}
|
|
270
|
-
//# sourceMappingURL=uv-installer.js.map
|
|
@@ -1,83 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* uv binary path resolver для Python skills runtime .
|
|
3
|
-
*
|
|
4
|
-
* Pugi skills that author Python (data plots, light scripting,
|
|
5
|
-
* notebook generation) shell out to Astral's `uv` — the de-facto
|
|
6
|
-
* Python package + interpreter manager since 2024. We do NOT bundle
|
|
7
|
-
* uv ourselves в the npm tarball; binary distribution is the
|
|
8
|
-
* follow-up. This module does the cheap part: locate uv on disk и
|
|
9
|
-
* give callers a single answer about whether Python skills can run.
|
|
10
|
-
*
|
|
11
|
-
* Resolution order (first hit wins):
|
|
12
|
-
* 1. `PUGI_UV_PATH` env var — explicit override, no validation beyond exists()
|
|
13
|
-
* 2. `~/.pugi/bin/uv` — future bundled location (download follow-up)
|
|
14
|
-
* 3. `~/.local/bin/uv` — uv's own default install location
|
|
15
|
-
* 4. PATH lookup via `which` (isaacs, MIT, ~70M weekly DLs — npm itself uses it)
|
|
16
|
-
*
|
|
17
|
-
* Pure-ish: reads fs + env, but never spawns processes и never
|
|
18
|
-
* touches the network. Result is safe к cache for the session.
|
|
19
|
-
*/
|
|
20
|
-
import { stat } from 'node:fs/promises';
|
|
21
|
-
import { homedir, platform } from 'node:os';
|
|
22
|
-
import path from 'node:path';
|
|
23
|
-
import which from 'which';
|
|
24
|
-
/**
|
|
25
|
-
* Find uv on disk. Returns the first path that exists и appears
|
|
26
|
-
* к be a regular file. Never throws; not-found resolves к
|
|
27
|
-
* `{ path: null, source: 'not-found' }`.
|
|
28
|
-
*/
|
|
29
|
-
export async function resolveUvPath(deps = {}) {
|
|
30
|
-
const env = deps.env ?? process.env;
|
|
31
|
-
const home = deps.homeDir ?? homedir();
|
|
32
|
-
const plat = deps.platform ?? platform();
|
|
33
|
-
const exeName = plat === 'win32' ? 'uv.exe' : 'uv';
|
|
34
|
-
// 1. Env override
|
|
35
|
-
const envOverride = env.PUGI_UV_PATH;
|
|
36
|
-
if (envOverride && await fileExists(envOverride)) {
|
|
37
|
-
return { path: envOverride, source: 'env-override' };
|
|
38
|
-
}
|
|
39
|
-
// 2. Pugi bundled location (follow-up download will populate this)
|
|
40
|
-
const bundled = path.join(home, '.pugi', 'bin', exeName);
|
|
41
|
-
if (await fileExists(bundled)) {
|
|
42
|
-
return { path: bundled, source: 'pugi-bundled' };
|
|
43
|
-
}
|
|
44
|
-
// 3. uv's default install location
|
|
45
|
-
const localBin = path.join(home, '.local', 'bin', exeName);
|
|
46
|
-
if (await fileExists(localBin)) {
|
|
47
|
-
return { path: localBin, source: 'local-bin' };
|
|
48
|
-
}
|
|
49
|
-
// 4. PATH lookup — honour overridden env.PATH so tests can isolate
|
|
50
|
-
try {
|
|
51
|
-
const whichOptions = {};
|
|
52
|
-
if (env.PATH !== undefined) {
|
|
53
|
-
whichOptions.path = env.PATH;
|
|
54
|
-
}
|
|
55
|
-
const fromPath = await which(exeName, whichOptions);
|
|
56
|
-
if (fromPath)
|
|
57
|
-
return { path: fromPath, source: 'path-lookup' };
|
|
58
|
-
}
|
|
59
|
-
catch {
|
|
60
|
-
// which throws when not found — fall through к not-found
|
|
61
|
-
}
|
|
62
|
-
return { path: null, source: 'not-found' };
|
|
63
|
-
}
|
|
64
|
-
/** Human-readable install hint, ready для surfacing в pugi doctor. */
|
|
65
|
-
export function uvInstallHint() {
|
|
66
|
-
const lines = [
|
|
67
|
-
'uv not found. Install via:',
|
|
68
|
-
' macOS / Linux: curl -LsSf https://astral.sh/uv/install.sh | sh',
|
|
69
|
-
' Windows: powershell -c "irm https://astral.sh/uv/install.ps1 | iex"',
|
|
70
|
-
'Or set PUGI_UV_PATH к an existing binary.',
|
|
71
|
-
];
|
|
72
|
-
return lines.join('\n');
|
|
73
|
-
}
|
|
74
|
-
async function fileExists(filePath) {
|
|
75
|
-
try {
|
|
76
|
-
const stats = await stat(filePath);
|
|
77
|
-
return stats.isFile();
|
|
78
|
-
}
|
|
79
|
-
catch {
|
|
80
|
-
return false;
|
|
81
|
-
}
|
|
82
|
-
}
|
|
83
|
-
//# sourceMappingURL=uv-resolver.js.map
|