@pugi/cli 0.1.0-beta.99 → 1.0.0-alpha.2
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 -195
- 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,12 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Public re-exports for the SessionStore module — Sprint .
|
|
3
|
-
*
|
|
4
|
-
* Consumers (`repl/session.ts`, `runtime/cli.ts`, the `/resume`
|
|
5
|
-
* dispatcher) import from `./store/index.js` so the internal split
|
|
6
|
-
* (types / lockfile / jsonl-log / session-store / uuid-v7) can change
|
|
7
|
-
* without touching the call sites.
|
|
8
|
-
*/
|
|
9
|
-
export { SessionLockBusyError, } from './types.js';
|
|
10
|
-
export { SqliteSessionStore, SqliteSessionStoreReadOnlyView, FtsSyntaxError, resolveProjectStoreDir, } from './session-store.js';
|
|
11
|
-
export { uuidV7, uuidV7Timestamp } from './uuid-v7.js';
|
|
12
|
-
//# sourceMappingURL=index.js.map
|
|
@@ -1,321 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Append-only JSONL event log — Sprint .
|
|
3
|
-
*
|
|
4
|
-
* Durable truth for the session. The SQLite index is rebuildable from
|
|
5
|
-
* the JSONL; if the index is lost or corrupt, `pugi sessions --rebuild`
|
|
6
|
-
* walks every line and reconstructs the table. The JSONL is therefore
|
|
7
|
-
* the source of truth and the SQLite is the cache.
|
|
8
|
-
*
|
|
9
|
-
* File layout per session directory:
|
|
10
|
-
*
|
|
11
|
-
* <sessionDir>/events.0.jsonl active rotation file
|
|
12
|
-
* <sessionDir>/events.1.jsonl previous rotation, oldest event first
|
|
13
|
-
* <sessionDir>/events.2.jsonl older still
|
|
14
|
-
* ...
|
|
15
|
-
*
|
|
16
|
-
* Rotation threshold is 50 MB per spec §4.2 must-ship #2. When the
|
|
17
|
-
* active file exceeds the threshold AFTER a write, we close it, rename
|
|
18
|
-
* it to the next numbered slot, and resume appending to a fresh
|
|
19
|
-
* `events.0.jsonl`. The reader stitches across files in REVERSE numeric
|
|
20
|
-
* order so the on-disk order is preserved.
|
|
21
|
-
*
|
|
22
|
-
* Crash safety:
|
|
23
|
-
*
|
|
24
|
-
* - Each write is a single `appendFileSync` of `JSON.stringify(event)\n`.
|
|
25
|
-
* On Linux/macOS, append-mode writes of bytes < PIPE_BUF (4096) are
|
|
26
|
-
* atomic with respect to other appends. Our events are well under
|
|
27
|
-
* 4 KB so the OS guarantees per-line atomicity even if two
|
|
28
|
-
* processes hold append fds (the lockfile already prevents that,
|
|
29
|
-
* but defence-in-depth).
|
|
30
|
-
* - After each write we call `fsync(fd)` so the bytes are durable
|
|
31
|
-
* against power loss / kernel panic. Slow but correct — the
|
|
32
|
-
* conversation-flow path is throughput-bound by the LLM, not by
|
|
33
|
-
* the disk, so a few hundred extra microseconds per event is
|
|
34
|
-
* invisible to the operator.
|
|
35
|
-
* - On reopen we walk every line and drop any that fail JSON parse.
|
|
36
|
-
* A crash mid-write leaves a truncated tail; we treat it as
|
|
37
|
-
* "event was never written" and continue from there. The next
|
|
38
|
-
* write may overlap with the truncated bytes; that is acceptable
|
|
39
|
-
* because the truncated line is invalid JSON either way and the
|
|
40
|
-
* reader is already designed to skip it.
|
|
41
|
-
*/
|
|
42
|
-
import { closeSync, existsSync, fsyncSync, mkdirSync, openSync, readdirSync, readFileSync, renameSync, statSync, writeSync, } from 'node:fs';
|
|
43
|
-
import { resolve } from 'node:path';
|
|
44
|
-
/**
|
|
45
|
-
* Rotate the active JSONL file when its size exceeds this threshold.
|
|
46
|
-
* Picked per spec §4.2. Operators can override via the
|
|
47
|
-
* `PUGI_SESSION_LOG_ROTATE_BYTES` env so heavy-traffic dogfooding can
|
|
48
|
-
* keep more events in the active file before rotation.
|
|
49
|
-
*/
|
|
50
|
-
const DEFAULT_ROTATE_BYTES = 50 * 1024 * 1024;
|
|
51
|
-
/**
|
|
52
|
-
* Active JSONL filename. The store opens fd against this path; the
|
|
53
|
-
* reader stitches across `events.<n>.jsonl` for n > 0.
|
|
54
|
-
*/
|
|
55
|
-
const ACTIVE_FILE = 'events.0.jsonl';
|
|
56
|
-
/**
|
|
57
|
-
* Append-only writer + reader for one session's JSONL log. The writer
|
|
58
|
-
* keeps a fd open for the lifetime of the session so each append
|
|
59
|
-
* avoids a fresh syscall pair; the fd is reopened transparently on
|
|
60
|
-
* rotation. `close()` releases the fd and is idempotent.
|
|
61
|
-
*/
|
|
62
|
-
export class JsonlEventLog {
|
|
63
|
-
sessionDir;
|
|
64
|
-
rotateBytes;
|
|
65
|
-
activeFd = null;
|
|
66
|
-
activeBytes = 0;
|
|
67
|
-
constructor(opts) {
|
|
68
|
-
this.sessionDir = opts.sessionDir;
|
|
69
|
-
this.rotateBytes =
|
|
70
|
-
opts.rotateBytes
|
|
71
|
-
?? readEnvInt('PUGI_SESSION_LOG_ROTATE_BYTES')
|
|
72
|
-
?? DEFAULT_ROTATE_BYTES;
|
|
73
|
-
mkdirSync(this.sessionDir, { recursive: true, mode: 0o700 });
|
|
74
|
-
}
|
|
75
|
-
/**
|
|
76
|
-
* Append one event. Returns the on-disk byte length written so the
|
|
77
|
-
* caller can budget without re-stat()-ing the file. The write is
|
|
78
|
-
* fsynced before this call returns.
|
|
79
|
-
*
|
|
80
|
-
* Rotation runs AFTER the event is durable. A rotation failure does
|
|
81
|
-
* NOT roll back the write — the event is already on disk. The error
|
|
82
|
-
* is surfaced to the caller so the operator knows the rotation
|
|
83
|
-
* threshold was tripped but the underlying filesystem refused the
|
|
84
|
-
* rename (EPERM / EXDEV / ENOSPC during rename). The next append
|
|
85
|
-
* will attempt rotation again from a fresh ensureFd().
|
|
86
|
-
*/
|
|
87
|
-
append(event) {
|
|
88
|
-
const line = `${JSON.stringify(event)}\n`;
|
|
89
|
-
const fd = this.ensureFd();
|
|
90
|
-
const bytes = Buffer.byteLength(line, 'utf8');
|
|
91
|
-
writeSync(fd, line, null, 'utf8');
|
|
92
|
-
fsyncSync(fd);
|
|
93
|
-
this.activeBytes += bytes;
|
|
94
|
-
if (this.activeBytes >= this.rotateBytes) {
|
|
95
|
-
this.rotate();
|
|
96
|
-
}
|
|
97
|
-
return bytes;
|
|
98
|
-
}
|
|
99
|
-
/**
|
|
100
|
-
* Load events in insertion order. Stitches across rotation files in
|
|
101
|
-
* REVERSE numeric order (events.N.jsonl ... events.0.jsonl) so the
|
|
102
|
-
* returned stream is oldest-first.
|
|
103
|
-
*
|
|
104
|
-
* `fromOffset` skips the first N events from the stitched stream;
|
|
105
|
-
* `limit` caps the returned count. Defaults: return all events.
|
|
106
|
-
*/
|
|
107
|
-
read(opts) {
|
|
108
|
-
const files = this.discoverRotationFiles();
|
|
109
|
-
// Reverse numeric order so events.N.jsonl (oldest) is read first.
|
|
110
|
-
const ordered = files.slice().sort((a, b) => b.index - a.index);
|
|
111
|
-
const fromOffset = Math.max(0, opts?.fromOffset ?? 0);
|
|
112
|
-
const limit = clampLimit(opts?.limit ?? Number.MAX_SAFE_INTEGER);
|
|
113
|
-
const out = [];
|
|
114
|
-
let skipped = 0;
|
|
115
|
-
for (const file of ordered) {
|
|
116
|
-
const raw = safeReadText(file.path);
|
|
117
|
-
if (raw.length === 0)
|
|
118
|
-
continue;
|
|
119
|
-
for (const line of raw.split('\n')) {
|
|
120
|
-
const trimmed = line.trim();
|
|
121
|
-
if (trimmed.length === 0)
|
|
122
|
-
continue;
|
|
123
|
-
const parsed = safeParseEvent(trimmed);
|
|
124
|
-
if (!parsed)
|
|
125
|
-
continue;
|
|
126
|
-
if (skipped < fromOffset) {
|
|
127
|
-
skipped += 1;
|
|
128
|
-
continue;
|
|
129
|
-
}
|
|
130
|
-
out.push(parsed);
|
|
131
|
-
if (out.length >= limit)
|
|
132
|
-
return out;
|
|
133
|
-
}
|
|
134
|
-
}
|
|
135
|
-
return out;
|
|
136
|
-
}
|
|
137
|
-
/**
|
|
138
|
-
* Count the total events on disk (across all rotation files). Used
|
|
139
|
-
* by the SessionStore to reconcile `event_count` on the SQLite row
|
|
140
|
-
* after a crash recovery walk.
|
|
141
|
-
*/
|
|
142
|
-
countEvents() {
|
|
143
|
-
const files = this.discoverRotationFiles();
|
|
144
|
-
let total = 0;
|
|
145
|
-
for (const file of files) {
|
|
146
|
-
const raw = safeReadText(file.path);
|
|
147
|
-
if (raw.length === 0)
|
|
148
|
-
continue;
|
|
149
|
-
for (const line of raw.split('\n')) {
|
|
150
|
-
const trimmed = line.trim();
|
|
151
|
-
if (trimmed.length === 0)
|
|
152
|
-
continue;
|
|
153
|
-
if (safeParseEvent(trimmed))
|
|
154
|
-
total += 1;
|
|
155
|
-
}
|
|
156
|
-
}
|
|
157
|
-
return total;
|
|
158
|
-
}
|
|
159
|
-
/** Release the file descriptor. Idempotent. */
|
|
160
|
-
close() {
|
|
161
|
-
if (this.activeFd !== null) {
|
|
162
|
-
try {
|
|
163
|
-
fsyncSync(this.activeFd);
|
|
164
|
-
}
|
|
165
|
-
catch {
|
|
166
|
-
/* ignore — fd may have been closed elsewhere */
|
|
167
|
-
}
|
|
168
|
-
try {
|
|
169
|
-
closeSync(this.activeFd);
|
|
170
|
-
}
|
|
171
|
-
catch {
|
|
172
|
-
/* ignore */
|
|
173
|
-
}
|
|
174
|
-
this.activeFd = null;
|
|
175
|
-
}
|
|
176
|
-
}
|
|
177
|
-
ensureFd() {
|
|
178
|
-
if (this.activeFd !== null)
|
|
179
|
-
return this.activeFd;
|
|
180
|
-
const path = resolve(this.sessionDir, ACTIVE_FILE);
|
|
181
|
-
// Read the existing size BEFORE opening with 'a' — `openSync(path,
|
|
182
|
-
// 'a', ...)` creates the file when missing, which makes a post-open
|
|
183
|
-
// `existsSync` always true. If a previous rotate() failed midway
|
|
184
|
-
// and left stale bytes at this path, we want `activeBytes` to
|
|
185
|
-
// inherit the real pre-open size so the next append accounting is
|
|
186
|
-
// consistent. Without this ordering the rotate-threshold check
|
|
187
|
-
// tripped on a fresh 0-byte file holding pre-rotation bytes,
|
|
188
|
-
// causing an immediate re-rotation loop.
|
|
189
|
-
let preOpenBytes = 0;
|
|
190
|
-
try {
|
|
191
|
-
preOpenBytes = existsSync(path) ? statSync(path).size : 0;
|
|
192
|
-
}
|
|
193
|
-
catch {
|
|
194
|
-
preOpenBytes = 0;
|
|
195
|
-
}
|
|
196
|
-
// O_APPEND so concurrent appends from a future feature (e.g. an
|
|
197
|
-
// out-of-band tool process) cannot overwrite each other. The
|
|
198
|
-
// lockfile already serializes us; this is defence-in-depth.
|
|
199
|
-
this.activeFd = openSync(path, 'a', 0o600);
|
|
200
|
-
this.activeBytes = preOpenBytes;
|
|
201
|
-
return this.activeFd;
|
|
202
|
-
}
|
|
203
|
-
/**
|
|
204
|
-
* Close the active file, shift every existing rotation file's number
|
|
205
|
-
* up by one (events.N -> events.N+1), and reopen a fresh
|
|
206
|
-
* events.0.jsonl. Synchronous and rare (50 MB / few-KB events =
|
|
207
|
-
* thousands of events between rotations).
|
|
208
|
-
*
|
|
209
|
-
* A rename failure is NOT swallowed — we rethrow so the caller knows
|
|
210
|
-
* the active file is over the threshold but the data inside it is
|
|
211
|
-
* still durable (we already fsynced before invoking rotate). Without
|
|
212
|
-
* the rethrow, ensureFd() inherited the pre-rotation size on the
|
|
213
|
-
* next append and tripped an infinite re-rotation loop.
|
|
214
|
-
*/
|
|
215
|
-
rotate() {
|
|
216
|
-
this.close();
|
|
217
|
-
const files = this.discoverRotationFiles();
|
|
218
|
-
// Rename in REVERSE order so we never clobber an existing slot.
|
|
219
|
-
const ordered = files.slice().sort((a, b) => b.index - a.index);
|
|
220
|
-
const renameErrors = [];
|
|
221
|
-
for (const file of ordered) {
|
|
222
|
-
const next = resolve(this.sessionDir, `events.${file.index + 1}.jsonl`);
|
|
223
|
-
try {
|
|
224
|
-
renameSync(file.path, next);
|
|
225
|
-
}
|
|
226
|
-
catch (err) {
|
|
227
|
-
// Collect every failed rename so the caller sees the full
|
|
228
|
-
// picture instead of just the first failure. We still
|
|
229
|
-
// continue the loop: any rename that succeeds shifts at
|
|
230
|
-
// least some history out of the way.
|
|
231
|
-
renameErrors.push(err instanceof Error ? err : new Error(String(err)));
|
|
232
|
-
}
|
|
233
|
-
}
|
|
234
|
-
this.activeBytes = 0;
|
|
235
|
-
if (renameErrors.length > 0) {
|
|
236
|
-
const msg = renameErrors.map((e) => e.message).join('; ');
|
|
237
|
-
const wrapped = new Error(`JSONL rotation failed in ${this.sessionDir}: ${msg}. ` +
|
|
238
|
-
'The data already on disk is safe; the active log may exceed ' +
|
|
239
|
-
'the configured threshold until the underlying issue is fixed.');
|
|
240
|
-
// Preserve the original errors on .cause for upstream
|
|
241
|
-
// observability. Node 16.9+ supports the standard cause field.
|
|
242
|
-
wrapped.cause = renameErrors;
|
|
243
|
-
throw wrapped;
|
|
244
|
-
}
|
|
245
|
-
// Lazily reopen on next append.
|
|
246
|
-
}
|
|
247
|
-
/**
|
|
248
|
-
* Walk the session directory and return every `events.<n>.jsonl`
|
|
249
|
-
* file with its numeric index. Sort order is left to the caller.
|
|
250
|
-
*/
|
|
251
|
-
discoverRotationFiles() {
|
|
252
|
-
if (!existsSync(this.sessionDir))
|
|
253
|
-
return [];
|
|
254
|
-
let entries;
|
|
255
|
-
try {
|
|
256
|
-
entries = readdirSync(this.sessionDir);
|
|
257
|
-
}
|
|
258
|
-
catch {
|
|
259
|
-
return [];
|
|
260
|
-
}
|
|
261
|
-
const out = [];
|
|
262
|
-
for (const name of entries) {
|
|
263
|
-
const match = /^events\.(\d+)\.jsonl$/.exec(name);
|
|
264
|
-
if (!match)
|
|
265
|
-
continue;
|
|
266
|
-
const index = Number.parseInt(match[1] ?? '', 10);
|
|
267
|
-
if (!Number.isFinite(index))
|
|
268
|
-
continue;
|
|
269
|
-
out.push({ index, path: resolve(this.sessionDir, name) });
|
|
270
|
-
}
|
|
271
|
-
return out;
|
|
272
|
-
}
|
|
273
|
-
}
|
|
274
|
-
function safeReadText(path) {
|
|
275
|
-
try {
|
|
276
|
-
return readFileSync(path, 'utf8');
|
|
277
|
-
}
|
|
278
|
-
catch {
|
|
279
|
-
return '';
|
|
280
|
-
}
|
|
281
|
-
}
|
|
282
|
-
function safeParseEvent(line) {
|
|
283
|
-
try {
|
|
284
|
-
const parsed = JSON.parse(line);
|
|
285
|
-
if (typeof parsed === 'object'
|
|
286
|
-
&& parsed !== null
|
|
287
|
-
&& typeof parsed.t === 'number'
|
|
288
|
-
&& typeof parsed.kind === 'string') {
|
|
289
|
-
// `payload` may be undefined on the wire (older clients); coerce
|
|
290
|
-
// to null so the consumer never has to type-narrow `unknown |
|
|
291
|
-
// undefined`.
|
|
292
|
-
const payload = parsed.payload ?? null;
|
|
293
|
-
return {
|
|
294
|
-
t: parsed.t,
|
|
295
|
-
kind: parsed.kind,
|
|
296
|
-
payload,
|
|
297
|
-
};
|
|
298
|
-
}
|
|
299
|
-
}
|
|
300
|
-
catch {
|
|
301
|
-
/* fall through — truncated tail, JSON parse error */
|
|
302
|
-
}
|
|
303
|
-
return null;
|
|
304
|
-
}
|
|
305
|
-
function clampLimit(raw) {
|
|
306
|
-
if (!Number.isFinite(raw) || raw <= 0)
|
|
307
|
-
return 1;
|
|
308
|
-
if (raw > 100000)
|
|
309
|
-
return 100000;
|
|
310
|
-
return Math.floor(raw);
|
|
311
|
-
}
|
|
312
|
-
function readEnvInt(key) {
|
|
313
|
-
const raw = process.env[key];
|
|
314
|
-
if (!raw)
|
|
315
|
-
return undefined;
|
|
316
|
-
const n = Number.parseInt(raw, 10);
|
|
317
|
-
if (!Number.isFinite(n) || n <= 0)
|
|
318
|
-
return undefined;
|
|
319
|
-
return n;
|
|
320
|
-
}
|
|
321
|
-
//# sourceMappingURL=jsonl-log.js.map
|
|
@@ -1,155 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* PID lockfile — Sprint .
|
|
3
|
-
*
|
|
4
|
-
* Prevents two REPL processes in the same project directory from
|
|
5
|
-
* writing to the same `session.db` concurrently. SQLite's WAL handles
|
|
6
|
-
* that case correctly at the page level, but the JSONL event log lives
|
|
7
|
-
* outside the SQL transaction — concurrent appends from two processes
|
|
8
|
-
* would interleave events. The lockfile is the higher-level mutex.
|
|
9
|
-
*
|
|
10
|
-
* Algorithm:
|
|
11
|
-
*
|
|
12
|
-
* 1. Try `O_CREAT | O_EXCL` open of `<dir>/session.lock`. Write the
|
|
13
|
-
* caller's PID + process start time as the body. If the open
|
|
14
|
-
* succeeds, we hold the lock; return.
|
|
15
|
-
* 2. If `EEXIST`, read the existing PID. Probe with `kill(pid, 0)`
|
|
16
|
-
* (signal 0 = "check liveness, do not deliver"). If the probe
|
|
17
|
-
* returns ESRCH, the holder is dead — unlink the stale lock and
|
|
18
|
-
* retry the exclusive create.
|
|
19
|
-
* 3. If the probe says the PID is alive, throw `SessionLockBusyError`
|
|
20
|
-
* so the caller surfaces a clean message to the operator.
|
|
21
|
-
*
|
|
22
|
-
* Why not `proper-lockfile`: pulling a 14kB dependency for a 60-line
|
|
23
|
-
* PID file is poor cost/benefit. The stale-detection loop is the only
|
|
24
|
-
* tricky part and we test it explicitly. The dependency would also
|
|
25
|
-
* pull `signal-exit` + `retry` transitively, raising the install
|
|
26
|
-
* footprint of the CLI bundle.
|
|
27
|
-
*
|
|
28
|
-
* Brand voice / privacy: the lockfile body is `{"pid":N,"createdAt":"…"}`
|
|
29
|
-
* — operator-readable if they `cat` the file, no secrets. Mode 0600 so
|
|
30
|
-
* other users on the box cannot read which PID is editing the store.
|
|
31
|
-
*/
|
|
32
|
-
import { closeSync, openSync, readFileSync, unlinkSync, writeSync, } from 'node:fs';
|
|
33
|
-
import { SessionLockBusyError } from './types.js';
|
|
34
|
-
/**
|
|
35
|
-
* Take an exclusive lock on `lockPath`. Throws `SessionLockBusyError`
|
|
36
|
-
* if a live process already holds the lock. `kill` is injectable for
|
|
37
|
-
* tests so the stale-detection branch is exercisable without spawning
|
|
38
|
-
* a real child process.
|
|
39
|
-
*/
|
|
40
|
-
export function takeLock(lockPath, options) {
|
|
41
|
-
const kill = options?.kill ?? ((pid, signal) => process.kill(pid, signal));
|
|
42
|
-
const now = options?.now ?? (() => new Date());
|
|
43
|
-
const body = { pid: process.pid, createdAt: now().toISOString() };
|
|
44
|
-
const serialized = JSON.stringify(body);
|
|
45
|
-
// First attempt: exclusive create. Retry once if the existing lock
|
|
46
|
-
// turns out to be stale (dead PID).
|
|
47
|
-
for (let attempt = 0; attempt < 2; attempt += 1) {
|
|
48
|
-
try {
|
|
49
|
-
const fd = openSync(lockPath, 'wx', 0o600);
|
|
50
|
-
try {
|
|
51
|
-
writeSync(fd, serialized);
|
|
52
|
-
}
|
|
53
|
-
finally {
|
|
54
|
-
closeSync(fd);
|
|
55
|
-
}
|
|
56
|
-
return { path: lockPath, release: () => releaseLock(lockPath, process.pid) };
|
|
57
|
-
}
|
|
58
|
-
catch (error) {
|
|
59
|
-
// EEXIST is the only branch that triggers stale-detection. Any
|
|
60
|
-
// other error (EACCES, ENOENT on parent, ENOSPC) propagates so
|
|
61
|
-
// the caller sees the real failure.
|
|
62
|
-
if (!isEexist(error)) {
|
|
63
|
-
throw error;
|
|
64
|
-
}
|
|
65
|
-
const holder = readLockBody(lockPath);
|
|
66
|
-
if (!holder) {
|
|
67
|
-
// Corrupt body — treat as stale and unlink.
|
|
68
|
-
tryUnlink(lockPath);
|
|
69
|
-
continue;
|
|
70
|
-
}
|
|
71
|
-
if (holder.pid === process.pid) {
|
|
72
|
-
// We already hold the lock (reentrant open from same process).
|
|
73
|
-
// Return a handle that is a no-op on release; the original
|
|
74
|
-
// owner's release will unlink.
|
|
75
|
-
return { path: lockPath, release: () => undefined };
|
|
76
|
-
}
|
|
77
|
-
if (isAlive(holder.pid, kill)) {
|
|
78
|
-
throw new SessionLockBusyError(holder.pid, lockPath);
|
|
79
|
-
}
|
|
80
|
-
// Stale lock — unlink and retry the exclusive create.
|
|
81
|
-
tryUnlink(lockPath);
|
|
82
|
-
}
|
|
83
|
-
}
|
|
84
|
-
// Two retries failed — the lock is contested by another process
|
|
85
|
-
// racing us. Surface as busy rather than spinning.
|
|
86
|
-
const holder = readLockBody(lockPath);
|
|
87
|
-
throw new SessionLockBusyError(holder?.pid ?? 0, lockPath);
|
|
88
|
-
}
|
|
89
|
-
/**
|
|
90
|
-
* Read a lockfile body. Returns null when the file is missing,
|
|
91
|
-
* unreadable, or malformed. The store never throws from this path —
|
|
92
|
-
* the caller treats null as "stale, unlink and retry".
|
|
93
|
-
*/
|
|
94
|
-
export function readLockBody(lockPath) {
|
|
95
|
-
let raw;
|
|
96
|
-
try {
|
|
97
|
-
raw = readFileSync(lockPath, 'utf8');
|
|
98
|
-
}
|
|
99
|
-
catch {
|
|
100
|
-
return null;
|
|
101
|
-
}
|
|
102
|
-
try {
|
|
103
|
-
const parsed = JSON.parse(raw);
|
|
104
|
-
if (typeof parsed === 'object'
|
|
105
|
-
&& parsed !== null
|
|
106
|
-
&& typeof parsed.pid === 'number'
|
|
107
|
-
&& typeof parsed.createdAt === 'string') {
|
|
108
|
-
return parsed;
|
|
109
|
-
}
|
|
110
|
-
}
|
|
111
|
-
catch {
|
|
112
|
-
/* fall through */
|
|
113
|
-
}
|
|
114
|
-
return null;
|
|
115
|
-
}
|
|
116
|
-
function isAlive(pid, kill) {
|
|
117
|
-
if (!Number.isFinite(pid) || pid <= 0)
|
|
118
|
-
return false;
|
|
119
|
-
try {
|
|
120
|
-
kill(pid, 0);
|
|
121
|
-
return true;
|
|
122
|
-
}
|
|
123
|
-
catch (error) {
|
|
124
|
-
// ESRCH = no such process. Any other error (EPERM = process exists
|
|
125
|
-
// but we lack permission) means the process IS alive.
|
|
126
|
-
const code = error?.code;
|
|
127
|
-
if (code === 'ESRCH')
|
|
128
|
-
return false;
|
|
129
|
-
return true;
|
|
130
|
-
}
|
|
131
|
-
}
|
|
132
|
-
function releaseLock(lockPath, expectedPid) {
|
|
133
|
-
const holder = readLockBody(lockPath);
|
|
134
|
-
if (!holder)
|
|
135
|
-
return;
|
|
136
|
-
// Defence-in-depth: only unlink if we still own the lock. A future
|
|
137
|
-
// session may have taken over after our process died and was
|
|
138
|
-
// replaced — better to leave the new owner's file in place than to
|
|
139
|
-
// race-condition delete it.
|
|
140
|
-
if (holder.pid !== expectedPid)
|
|
141
|
-
return;
|
|
142
|
-
tryUnlink(lockPath);
|
|
143
|
-
}
|
|
144
|
-
function tryUnlink(path) {
|
|
145
|
-
try {
|
|
146
|
-
unlinkSync(path);
|
|
147
|
-
}
|
|
148
|
-
catch {
|
|
149
|
-
/* ignore — file may be gone already */
|
|
150
|
-
}
|
|
151
|
-
}
|
|
152
|
-
function isEexist(error) {
|
|
153
|
-
return error?.code === 'EEXIST';
|
|
154
|
-
}
|
|
155
|
-
//# sourceMappingURL=lockfile.js.map
|