@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,149 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Resumer — read SessionStore events for a session, apply the L8
|
|
3
|
-
* compact mask + the L9 rewind mask, and return the visible transcript
|
|
4
|
-
* the REPL bootstrap (or a programmatic consumer) should render.
|
|
5
|
-
*
|
|
6
|
-
* Separation of concerns:
|
|
7
|
-
*
|
|
8
|
-
* - This module owns the READ path: list sessions, load events,
|
|
9
|
-
* reconstruct a clean transcript stream. No writes.
|
|
10
|
-
* - The WRITE path (append a rewind-marker, undo-rewind) lives in
|
|
11
|
-
* `./rewinder.ts`.
|
|
12
|
-
* - The REPL session lifecycle (lockfile, Ink mount, dispatch FSM)
|
|
13
|
-
* stays in `core/repl/session.ts`. We do NOT spin up the REPL here.
|
|
14
|
-
*
|
|
15
|
-
* Why route resume through this module at all (vs. operators using
|
|
16
|
-
* `core/repl/store/*` directly):
|
|
17
|
-
*
|
|
18
|
-
* The store returns RAW events. Most consumers want masked events —
|
|
19
|
-
* i.e. the chronological list after compact-boundary masking AND
|
|
20
|
-
* rewind-marker masking. Doing both passes inline at every call site
|
|
21
|
-
* would scatter the mask logic; centralising it here means a future
|
|
22
|
-
* third mask (named checkpoints? selective edit?) lands in one place.
|
|
23
|
-
*/
|
|
24
|
-
import { homedir } from 'node:os';
|
|
25
|
-
import { applyCompactMask } from '../compact/buffer-rewriter.js';
|
|
26
|
-
import { SqliteSessionStore, resolveProjectStoreDir, } from '../repl/store/index.js';
|
|
27
|
-
import { applyRewindMask, findLatestActiveRewind } from './rewinder.js';
|
|
28
|
-
/**
|
|
29
|
-
* Composed mask: compact-mask first (collapses summarised slices into
|
|
30
|
-
* boundary markers + kept tail), then rewind-mask (drops everything
|
|
31
|
-
* inside an active rewind range, including any compaction markers that
|
|
32
|
-
* fell inside it).
|
|
33
|
-
*
|
|
34
|
-
* Order matters: compact-mask reads `coversUntilOffset` against the
|
|
35
|
-
* RAW event indices. Running rewind-mask first would shift indices and
|
|
36
|
-
* break the compact replay anchor. The result is the chronological
|
|
37
|
-
* stream the operator should SEE, with infra rows (rewind markers)
|
|
38
|
-
* stripped.
|
|
39
|
-
*/
|
|
40
|
-
export function applyAllMasks(events) {
|
|
41
|
-
return applyRewindMask(applyCompactMask(events));
|
|
42
|
-
}
|
|
43
|
-
/**
|
|
44
|
-
* List sessions a `pugi resume` invocation could open. Uses the
|
|
45
|
-
* READ-ONLY store view so the call never takes the lockfile — safe to
|
|
46
|
-
* run alongside a live REPL. Each row carries derived metadata
|
|
47
|
-
* (`visibleEventCount`, `hasActiveRewind`) so the renderer does not
|
|
48
|
-
* need to re-walk events.
|
|
49
|
-
*
|
|
50
|
-
* Returns an empty array when the project store does not exist (no
|
|
51
|
-
* sessions ever started for this project slug). Callers surface a
|
|
52
|
-
* "nothing to resume" message in that branch.
|
|
53
|
-
*/
|
|
54
|
-
export async function listResumableSessions(input) {
|
|
55
|
-
const dir = input.storeDir ?? resolveProjectStoreDir(input.projectSlug, input.home ?? homedir());
|
|
56
|
-
const limit = clampLimit(input.limit ?? 10, 1, 50);
|
|
57
|
-
let view;
|
|
58
|
-
try {
|
|
59
|
-
view = await SqliteSessionStore.openReadOnly(dir);
|
|
60
|
-
}
|
|
61
|
-
catch {
|
|
62
|
-
return [];
|
|
63
|
-
}
|
|
64
|
-
try {
|
|
65
|
-
const rows = await view.list({ project: input.projectSlug, limit });
|
|
66
|
-
const out = [];
|
|
67
|
-
for (const row of rows) {
|
|
68
|
-
const events = await view.events(row.id);
|
|
69
|
-
const visible = applyAllMasks(events);
|
|
70
|
-
const latest = findLatestActiveRewind(events);
|
|
71
|
-
out.push({
|
|
72
|
-
row,
|
|
73
|
-
visibleEventCount: visible.length,
|
|
74
|
-
hasActiveRewind: latest !== null,
|
|
75
|
-
updatedAt: row.updatedAt,
|
|
76
|
-
});
|
|
77
|
-
}
|
|
78
|
-
return out;
|
|
79
|
-
}
|
|
80
|
-
catch {
|
|
81
|
-
return [];
|
|
82
|
-
}
|
|
83
|
-
finally {
|
|
84
|
-
await view.close();
|
|
85
|
-
}
|
|
86
|
-
}
|
|
87
|
-
/**
|
|
88
|
-
* Load one session for replay. The caller (REPL bootstrap, tests,
|
|
89
|
-
* future programmatic exporters) gets BOTH the raw event stream and
|
|
90
|
-
* the masked view so it can choose its rendering strategy. Returns
|
|
91
|
-
* null when the session does not exist; throws when the store cannot
|
|
92
|
-
* be opened (the caller surfaces a one-line error).
|
|
93
|
-
*
|
|
94
|
-
* The PID lockfile contention is NOT relevant here — we use the
|
|
95
|
-
* read-only view. Concurrent writers from a live REPL are safe.
|
|
96
|
-
*/
|
|
97
|
-
export async function loadSessionForReplay(input) {
|
|
98
|
-
const dir = input.storeDir ?? resolveProjectStoreDir(input.projectSlug, input.home ?? homedir());
|
|
99
|
-
const view = await SqliteSessionStore.openReadOnly(dir);
|
|
100
|
-
try {
|
|
101
|
-
const row = await view.get(input.sessionId);
|
|
102
|
-
if (!row)
|
|
103
|
-
return null;
|
|
104
|
-
const rawEvents = await view.events(row.id);
|
|
105
|
-
const visibleEvents = applyAllMasks(rawEvents);
|
|
106
|
-
const latest = findLatestActiveRewind(rawEvents);
|
|
107
|
-
return {
|
|
108
|
-
row,
|
|
109
|
-
rawEvents,
|
|
110
|
-
visibleEvents,
|
|
111
|
-
hasActiveRewind: latest !== null,
|
|
112
|
-
};
|
|
113
|
-
}
|
|
114
|
-
finally {
|
|
115
|
-
await view.close();
|
|
116
|
-
}
|
|
117
|
-
}
|
|
118
|
-
/**
|
|
119
|
-
* Load raw + masked events through an already-open SessionStore.
|
|
120
|
-
*
|
|
121
|
-
* Used by the in-REPL `/rewind` slash handler — the live REPL already
|
|
122
|
-
* holds the writer lock, so we cannot open the read-only view in the
|
|
123
|
-
* same process. The store reference IS the active write handle; we
|
|
124
|
-
* just call `loadEvents` and run the masks.
|
|
125
|
-
*
|
|
126
|
-
* Same shape as `loadSessionForReplay` minus the read-only-view setup.
|
|
127
|
-
*/
|
|
128
|
-
export async function loadFromStore(store, sessionId) {
|
|
129
|
-
const row = await store.getSession(sessionId);
|
|
130
|
-
if (!row)
|
|
131
|
-
return null;
|
|
132
|
-
const rawEvents = await store.loadEvents(sessionId);
|
|
133
|
-
const visibleEvents = applyAllMasks(rawEvents);
|
|
134
|
-
const latest = findLatestActiveRewind(rawEvents);
|
|
135
|
-
return {
|
|
136
|
-
row,
|
|
137
|
-
rawEvents,
|
|
138
|
-
visibleEvents,
|
|
139
|
-
hasActiveRewind: latest !== null,
|
|
140
|
-
};
|
|
141
|
-
}
|
|
142
|
-
function clampLimit(raw, min, max) {
|
|
143
|
-
if (!Number.isFinite(raw) || raw < min)
|
|
144
|
-
return min;
|
|
145
|
-
if (raw > max)
|
|
146
|
-
return max;
|
|
147
|
-
return Math.floor(raw);
|
|
148
|
-
}
|
|
149
|
-
//# sourceMappingURL=resumer.js.map
|
|
@@ -1,291 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Type guard: discriminate a SessionEvent against the `rewind-marker`
|
|
3
|
-
* kind. The boundary check is strict — a malformed payload returns
|
|
4
|
-
* `false` so the replay layer can treat it as a regular (visible) event
|
|
5
|
-
* instead of trusting partial data.
|
|
6
|
-
*/
|
|
7
|
-
export function isRewindMarker(event) {
|
|
8
|
-
if (event.kind !== 'rewind-marker')
|
|
9
|
-
return false;
|
|
10
|
-
const p = event.payload;
|
|
11
|
-
if (p === null || typeof p !== 'object')
|
|
12
|
-
return false;
|
|
13
|
-
if (p.version !== 1)
|
|
14
|
-
return false;
|
|
15
|
-
if (p.mode !== 'rewind' && p.mode !== 'undo-rewind')
|
|
16
|
-
return false;
|
|
17
|
-
if (typeof p.toEventIndex !== 'number')
|
|
18
|
-
return false;
|
|
19
|
-
if (typeof p.fromEventIndex !== 'number')
|
|
20
|
-
return false;
|
|
21
|
-
if (typeof p.turnsRewound !== 'number')
|
|
22
|
-
return false;
|
|
23
|
-
if (p.reason !== 'manual'
|
|
24
|
-
&& p.reason !== 'to-event'
|
|
25
|
-
&& p.reason !== 'interactive'
|
|
26
|
-
&& p.reason !== 'undo') {
|
|
27
|
-
return false;
|
|
28
|
-
}
|
|
29
|
-
return true;
|
|
30
|
-
}
|
|
31
|
-
/**
|
|
32
|
-
* Append one `rewind-marker` event to the SessionStore. Returns the
|
|
33
|
-
* SessionEvent we wrote so the caller can echo it into the in-memory
|
|
34
|
-
* transcript without a re-read. Throws on store error so the caller
|
|
35
|
-
* surfaces the failure inline.
|
|
36
|
-
*/
|
|
37
|
-
export async function appendRewindMarker(input) {
|
|
38
|
-
const ts = (input.now ?? (() => Date.now()))();
|
|
39
|
-
const payload = {
|
|
40
|
-
version: 1,
|
|
41
|
-
mode: input.mode ?? 'rewind',
|
|
42
|
-
toEventIndex: input.toEventIndex,
|
|
43
|
-
fromEventIndex: input.fromEventIndex,
|
|
44
|
-
turnsRewound: input.turnsRewound,
|
|
45
|
-
reason: input.reason,
|
|
46
|
-
};
|
|
47
|
-
const event = {
|
|
48
|
-
t: ts,
|
|
49
|
-
kind: 'rewind-marker',
|
|
50
|
-
payload,
|
|
51
|
-
};
|
|
52
|
-
await input.store.appendEvent(event);
|
|
53
|
-
return event;
|
|
54
|
-
}
|
|
55
|
-
/**
|
|
56
|
-
* Apply rewind masking to a chronological event list. Pure function —
|
|
57
|
-
* returns a new array with the masked events stripped. The marker
|
|
58
|
-
* events themselves are stripped too (they are infrastructure, not
|
|
59
|
-
* conversation rows); the renderer surfaces the rewind banner via a
|
|
60
|
-
* separate informational payload, NOT by leaving the marker in the
|
|
61
|
-
* transcript.
|
|
62
|
-
*
|
|
63
|
-
* Algorithm (matched-pair cancellation):
|
|
64
|
-
*
|
|
65
|
-
* 1. Walk events newest-to-oldest. Maintain an integer `undoBalance`
|
|
66
|
-
* that counts the unmatched 'undo-rewind' markers we have seen.
|
|
67
|
-
* 2. When we hit an 'undo-rewind' marker, increment `undoBalance` —
|
|
68
|
-
* it will cancel the next older 'rewind' marker.
|
|
69
|
-
* 3. When we hit a 'rewind' marker:
|
|
70
|
-
* - If `undoBalance > 0`: decrement (the undo cancels this
|
|
71
|
-
* rewind). The marker AND the events in its masked range stay
|
|
72
|
-
* visible — the undo restored them.
|
|
73
|
-
* - Otherwise: record the marker's `[toEventIndex+1 ..
|
|
74
|
-
* fromEventIndex]` range as masked.
|
|
75
|
-
* 4. After the walk, return every event whose index is NOT inside an
|
|
76
|
-
* active masked range AND that is not itself a rewind-marker.
|
|
77
|
-
*
|
|
78
|
-
* Why not just look at the latest marker:
|
|
79
|
-
* The operator can rewind, then rewind again, then undo-rewind once.
|
|
80
|
-
* The newest rewind should still apply; only the innermost rewind is
|
|
81
|
-
* cancelled by the undo. Matched-pair walking handles every
|
|
82
|
-
* stack-depth without special-casing.
|
|
83
|
-
*/
|
|
84
|
-
export function applyRewindMask(events) {
|
|
85
|
-
// Pass 1: walk newest-to-oldest, collect active masked ranges.
|
|
86
|
-
let undoBalance = 0;
|
|
87
|
-
// Each range is half-open [start, end] inclusive on both ends so the
|
|
88
|
-
// membership check below stays a simple two-integer comparison.
|
|
89
|
-
const maskedRanges = [];
|
|
90
|
-
for (let i = events.length - 1; i >= 0; i -= 1) {
|
|
91
|
-
const ev = events[i];
|
|
92
|
-
if (!isRewindMarker(ev))
|
|
93
|
-
continue;
|
|
94
|
-
if (ev.payload.mode === 'undo-rewind') {
|
|
95
|
-
undoBalance += 1;
|
|
96
|
-
continue;
|
|
97
|
-
}
|
|
98
|
-
// mode === 'rewind'
|
|
99
|
-
if (undoBalance > 0) {
|
|
100
|
-
undoBalance -= 1;
|
|
101
|
-
continue;
|
|
102
|
-
}
|
|
103
|
-
// Active rewind — mask everything strictly AFTER toEventIndex and
|
|
104
|
-
// strictly BEFORE the marker itself. The marker (index === i) is
|
|
105
|
-
// dropped via the rewind-marker kind filter below.
|
|
106
|
-
const start = ev.payload.toEventIndex + 1;
|
|
107
|
-
const end = i - 1;
|
|
108
|
-
if (end >= start) {
|
|
109
|
-
maskedRanges.push({ start, end });
|
|
110
|
-
}
|
|
111
|
-
}
|
|
112
|
-
// Pass 2: emit only events that are NOT inside any masked range and
|
|
113
|
-
// are NOT themselves rewind-markers. Linear scan with a sorted-ranges
|
|
114
|
-
// membership check would be faster for very large logs, but the
|
|
115
|
-
// typical transcript is in the hundreds of events — O(N*M) here with
|
|
116
|
-
// M = number of rewinds ever appended stays well under a millisecond.
|
|
117
|
-
const out = [];
|
|
118
|
-
for (let i = 0; i < events.length; i += 1) {
|
|
119
|
-
const ev = events[i];
|
|
120
|
-
if (isRewindMarker(ev))
|
|
121
|
-
continue;
|
|
122
|
-
let masked = false;
|
|
123
|
-
for (const range of maskedRanges) {
|
|
124
|
-
if (i >= range.start && i <= range.end) {
|
|
125
|
-
masked = true;
|
|
126
|
-
break;
|
|
127
|
-
}
|
|
128
|
-
}
|
|
129
|
-
if (!masked)
|
|
130
|
-
out.push(ev);
|
|
131
|
-
}
|
|
132
|
-
return out;
|
|
133
|
-
}
|
|
134
|
-
/**
|
|
135
|
-
* Walk events oldest-to-newest and return the indices of the last `n`
|
|
136
|
-
* operator turns (`kind === 'user'`) — used by `/rewind N` to translate
|
|
137
|
-
* "drop the last 3 turns" into a concrete `toEventIndex`.
|
|
138
|
-
*
|
|
139
|
-
* Visibility is computed via `applyRewindMask`-aware indexing: the
|
|
140
|
-
* caller supplies the FULL event list (incl. existing markers) and we
|
|
141
|
-
* walk only the visible subset so a follow-up rewind on top of an
|
|
142
|
-
* existing rewind operates on what the operator currently SEES, not the
|
|
143
|
-
* full on-disk history.
|
|
144
|
-
*
|
|
145
|
-
* Returns the index of the event that should become the new
|
|
146
|
-
* `toEventIndex` — i.e. the event *immediately before* the Nth turn
|
|
147
|
-
* boundary, counting from the end. When `n` exceeds the visible turn
|
|
148
|
-
* count, returns `-1` (rewind everything).
|
|
149
|
-
*/
|
|
150
|
-
export function pickRewindTargetForTurns(events, turnsToDrop) {
|
|
151
|
-
if (turnsToDrop <= 0) {
|
|
152
|
-
// No-op — return the last index unchanged so the caller can detect
|
|
153
|
-
// the noop and emit a sensible message.
|
|
154
|
-
return { toEventIndex: events.length - 1, turnsRewound: 0 };
|
|
155
|
-
}
|
|
156
|
-
const visible = applyRewindMask(events);
|
|
157
|
-
// Walk visible newest-to-oldest, count user turns.
|
|
158
|
-
const userTurnVisibleIndices = [];
|
|
159
|
-
for (let i = visible.length - 1; i >= 0; i -= 1) {
|
|
160
|
-
if (visible[i].kind === 'user')
|
|
161
|
-
userTurnVisibleIndices.push(i);
|
|
162
|
-
if (userTurnVisibleIndices.length >= turnsToDrop + 1)
|
|
163
|
-
break;
|
|
164
|
-
}
|
|
165
|
-
const turnsAvailable = userTurnVisibleIndices.length;
|
|
166
|
-
if (turnsAvailable < turnsToDrop) {
|
|
167
|
-
// Fewer turns than asked — rewind everything visible.
|
|
168
|
-
return { toEventIndex: -1, turnsRewound: turnsAvailable };
|
|
169
|
-
}
|
|
170
|
-
if (turnsAvailable === turnsToDrop) {
|
|
171
|
-
// Drop ALL visible turns — anchor at -1 inside the visible list
|
|
172
|
-
// (i.e. before the first visible event). Translate back to an
|
|
173
|
-
// index in the full event list by picking the position right
|
|
174
|
-
// before the oldest visible event.
|
|
175
|
-
const oldestVisibleIdx = visible.length > 0 ? indexInFull(events, visible[0]) : -1;
|
|
176
|
-
return { toEventIndex: oldestVisibleIdx - 1, turnsRewound: turnsToDrop };
|
|
177
|
-
}
|
|
178
|
-
// `userTurnVisibleIndices` collected user turns newest-first. The
|
|
179
|
-
// OLDEST turn the operator wants to drop sits at index
|
|
180
|
-
// `turnsToDrop - 1` of that array (the Nth most recent). The new
|
|
181
|
-
// `toEventIndex` is the visible event RIGHT BEFORE that turn — that
|
|
182
|
-
// becomes the last visible row after the rewind. The N most-recent
|
|
183
|
-
// turns + everything between them get masked.
|
|
184
|
-
const cutVisibleIdx = userTurnVisibleIndices[turnsToDrop - 1];
|
|
185
|
-
// Pick the event one slot before the cut as the new toEventIndex.
|
|
186
|
-
const anchorVisibleIdx = cutVisibleIdx - 1;
|
|
187
|
-
if (anchorVisibleIdx < 0) {
|
|
188
|
-
return { toEventIndex: -1, turnsRewound: turnsToDrop };
|
|
189
|
-
}
|
|
190
|
-
const anchorEvent = visible[anchorVisibleIdx];
|
|
191
|
-
const toEventIndex = indexInFull(events, anchorEvent);
|
|
192
|
-
return { toEventIndex, turnsRewound: turnsToDrop };
|
|
193
|
-
}
|
|
194
|
-
/**
|
|
195
|
-
* Resolve a `--to <event-id>` argument into a concrete event index in
|
|
196
|
-
* the on-disk log. The L8 / L9 wire format does NOT mint an `id` field
|
|
197
|
-
* on individual events; the operator picks by the (1-based) line number
|
|
198
|
-
* we surface in `applyRewindMask`-aware listings. So `<event-id>` here
|
|
199
|
-
* is `"<n>"` where n is the 1-based visible index. We accept both
|
|
200
|
-
* 1-based (UI-facing) and 0-based (programmatic / tests) by checking
|
|
201
|
-
* for a leading `#`.
|
|
202
|
-
*
|
|
203
|
-
* Returns the matching index in the full event list, or null when the
|
|
204
|
-
* input is unparseable / out of range.
|
|
205
|
-
*/
|
|
206
|
-
export function resolveEventIdToIndex(events, eventId) {
|
|
207
|
-
const trimmed = eventId.trim();
|
|
208
|
-
if (trimmed.length === 0)
|
|
209
|
-
return null;
|
|
210
|
-
const zeroBased = trimmed.startsWith('#');
|
|
211
|
-
const raw = zeroBased ? trimmed.slice(1) : trimmed;
|
|
212
|
-
const parsed = Number.parseInt(raw, 10);
|
|
213
|
-
if (!Number.isFinite(parsed) || parsed < 0)
|
|
214
|
-
return null;
|
|
215
|
-
const visible = applyRewindMask(events);
|
|
216
|
-
const visibleIdx = zeroBased ? parsed : parsed - 1;
|
|
217
|
-
if (visibleIdx < 0 || visibleIdx >= visible.length)
|
|
218
|
-
return null;
|
|
219
|
-
return indexInFull(events, visible[visibleIdx]);
|
|
220
|
-
}
|
|
221
|
-
/**
|
|
222
|
-
* Locate the index of `target` inside `events` by identity-then-equality.
|
|
223
|
-
* Identity hits in O(1); the equality fallback walks the array and
|
|
224
|
-
* compares the (t, kind) discriminator, which is unique-enough for the
|
|
225
|
-
* append-only log where two events at the same millisecond with the
|
|
226
|
-
* same kind would also have identical payloads.
|
|
227
|
-
*/
|
|
228
|
-
function indexInFull(events, target) {
|
|
229
|
-
// Identity check first — `applyRewindMask` returns elements from the
|
|
230
|
-
// input array directly, so `===` succeeds in the common path.
|
|
231
|
-
for (let i = 0; i < events.length; i += 1) {
|
|
232
|
-
if (events[i] === target)
|
|
233
|
-
return i;
|
|
234
|
-
}
|
|
235
|
-
// Fallback: compare by t + kind + payload reference. Unlikely path.
|
|
236
|
-
for (let i = 0; i < events.length; i += 1) {
|
|
237
|
-
const ev = events[i];
|
|
238
|
-
if (ev.t === target.t && ev.kind === target.kind && ev.payload === target.payload) {
|
|
239
|
-
return i;
|
|
240
|
-
}
|
|
241
|
-
}
|
|
242
|
-
return -1;
|
|
243
|
-
}
|
|
244
|
-
export function buildRewindPickerRows(events, limit = 10) {
|
|
245
|
-
const visible = applyRewindMask(events);
|
|
246
|
-
const rows = [];
|
|
247
|
-
let turnsAgo = 0;
|
|
248
|
-
for (let i = visible.length - 1; i >= 0 && rows.length < limit; i -= 1) {
|
|
249
|
-
const ev = visible[i];
|
|
250
|
-
if (ev.kind !== 'user')
|
|
251
|
-
continue;
|
|
252
|
-
turnsAgo += 1;
|
|
253
|
-
const payload = ev.payload;
|
|
254
|
-
const preview = typeof payload?.brief === 'string'
|
|
255
|
-
? payload.brief.slice(0, 64)
|
|
256
|
-
: '(empty turn)';
|
|
257
|
-
rows.push({
|
|
258
|
-
eventIndex: indexInFull(events, ev),
|
|
259
|
-
visibleIndex: i + 1,
|
|
260
|
-
preview,
|
|
261
|
-
turnsAgo,
|
|
262
|
-
timestampEpochMs: ev.t,
|
|
263
|
-
});
|
|
264
|
-
}
|
|
265
|
-
return rows;
|
|
266
|
-
}
|
|
267
|
-
/**
|
|
268
|
-
* Find the latest active 'rewind' marker — i.e. the one that an
|
|
269
|
-
* `undo-rewind` would cancel. Walks newest-to-oldest, balancing
|
|
270
|
-
* 'undo-rewind' markers against 'rewind' markers; returns the first
|
|
271
|
-
* unmatched 'rewind' or null when every rewind has already been undone.
|
|
272
|
-
*/
|
|
273
|
-
export function findLatestActiveRewind(events) {
|
|
274
|
-
let undoBalance = 0;
|
|
275
|
-
for (let i = events.length - 1; i >= 0; i -= 1) {
|
|
276
|
-
const ev = events[i];
|
|
277
|
-
if (!isRewindMarker(ev))
|
|
278
|
-
continue;
|
|
279
|
-
if (ev.payload.mode === 'undo-rewind') {
|
|
280
|
-
undoBalance += 1;
|
|
281
|
-
continue;
|
|
282
|
-
}
|
|
283
|
-
if (undoBalance > 0) {
|
|
284
|
-
undoBalance -= 1;
|
|
285
|
-
continue;
|
|
286
|
-
}
|
|
287
|
-
return { event: ev, payload: ev.payload, index: i };
|
|
288
|
-
}
|
|
289
|
-
return null;
|
|
290
|
-
}
|
|
291
|
-
//# sourceMappingURL=rewinder.js.map
|