@pugi/cli 0.1.0-beta.3 → 0.1.0-beta.31

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.
Files changed (219) hide show
  1. package/THIRD_PARTY_NOTICES.md +40 -0
  2. package/assets/pugi-mascot.ansi +15 -40
  3. package/bin/run.js +33 -1
  4. package/dist/commands/jobs-watch.js +201 -0
  5. package/dist/commands/jobs.js +15 -0
  6. package/dist/core/agent-progress/cleanup.js +134 -0
  7. package/dist/core/agent-progress/schema.js +144 -0
  8. package/dist/core/agent-progress/writer.js +101 -0
  9. package/dist/core/artifact-chain/dispatcher.js +148 -0
  10. package/dist/core/artifact-chain/exporter.js +164 -0
  11. package/dist/core/artifact-chain/state.js +243 -0
  12. package/dist/core/artifact-chain/steps.js +169 -0
  13. package/dist/core/auth/env-provider.js +238 -0
  14. package/dist/core/auto-update/channels.js +122 -0
  15. package/dist/core/auto-update/checker.js +241 -0
  16. package/dist/core/auto-update/state.js +235 -0
  17. package/dist/core/bare-mode/index.js +107 -0
  18. package/dist/core/checkpoint/resumer.js +149 -0
  19. package/dist/core/checkpoint/rewinder.js +291 -0
  20. package/dist/core/compact/auto-trigger.js +96 -0
  21. package/dist/core/compact/buffer-rewriter.js +115 -0
  22. package/dist/core/compact/summarizer.js +208 -0
  23. package/dist/core/compact/token-counter.js +108 -0
  24. package/dist/core/consensus/diff-capture.js +73 -0
  25. package/dist/core/context/index.js +7 -0
  26. package/dist/core/context/markdown-traverse.js +255 -0
  27. package/dist/core/cost/rate-card.js +129 -0
  28. package/dist/core/cost/tracker.js +221 -0
  29. package/dist/core/denial-tracking/index.js +8 -0
  30. package/dist/core/denial-tracking/state.js +264 -0
  31. package/dist/core/diagnostics/probe-runner.js +93 -0
  32. package/dist/core/diagnostics/probes/api.js +46 -0
  33. package/dist/core/diagnostics/probes/auth.js +86 -0
  34. package/dist/core/diagnostics/probes/bare-mode.js +42 -0
  35. package/dist/core/diagnostics/probes/cli-version.js +127 -0
  36. package/dist/core/diagnostics/probes/config.js +72 -0
  37. package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
  38. package/dist/core/diagnostics/probes/disk.js +81 -0
  39. package/dist/core/diagnostics/probes/git.js +65 -0
  40. package/dist/core/diagnostics/probes/mcp.js +75 -0
  41. package/dist/core/diagnostics/probes/node.js +59 -0
  42. package/dist/core/diagnostics/probes/pnpm.js +36 -0
  43. package/dist/core/diagnostics/probes/pugi-md.js +89 -0
  44. package/dist/core/diagnostics/probes/session.js +74 -0
  45. package/dist/core/diagnostics/probes/status-snapshot.js +442 -0
  46. package/dist/core/diagnostics/probes/workspace.js +63 -0
  47. package/dist/core/diagnostics/types.js +70 -0
  48. package/dist/core/dispatch/cache-cleanup.js +197 -0
  49. package/dist/core/dispatch/cache-handoff.js +295 -0
  50. package/dist/core/edits/dispatch.js +218 -2
  51. package/dist/core/edits/journal.js +199 -0
  52. package/dist/core/edits/layer-d-ast.js +557 -14
  53. package/dist/core/edits/verify-hook.js +273 -0
  54. package/dist/core/edits/worktree.js +111 -18
  55. package/dist/core/engine/anvil-client.js +115 -5
  56. package/dist/core/engine/budgets.js +89 -0
  57. package/dist/core/engine/context-prefix.js +155 -0
  58. package/dist/core/engine/intent.js +260 -0
  59. package/dist/core/engine/native-pugi.js +852 -210
  60. package/dist/core/engine/prompts.js +89 -6
  61. package/dist/core/engine/strip-internal-fields.js +124 -0
  62. package/dist/core/engine/tool-bridge.js +972 -33
  63. package/dist/core/feedback/queue.js +177 -0
  64. package/dist/core/feedback/submitter.js +145 -0
  65. package/dist/core/file-cache.js +113 -1
  66. package/dist/core/hooks/events.js +44 -0
  67. package/dist/core/hooks/index.js +15 -0
  68. package/dist/core/hooks/registry.js +213 -0
  69. package/dist/core/hooks/runner.js +236 -0
  70. package/dist/core/init/scaffold.js +195 -0
  71. package/dist/core/lsp/cache.js +105 -0
  72. package/dist/core/lsp/client.js +174 -29
  73. package/dist/core/lsp/language-detect.js +66 -0
  74. package/dist/core/lsp/post-edit-diagnostics.js +171 -0
  75. package/dist/core/mcp/client.js +75 -6
  76. package/dist/core/mcp/http-server.js +553 -0
  77. package/dist/core/mcp/permission.js +190 -0
  78. package/dist/core/mcp/registry.js +24 -2
  79. package/dist/core/mcp/server-tools.js +219 -0
  80. package/dist/core/mcp/server.js +397 -0
  81. package/dist/core/memory/dual-write.js +416 -0
  82. package/dist/core/memory/dual-write.spec.js +297 -0
  83. package/dist/core/memory/phase1-kinds.js +20 -0
  84. package/dist/core/memory-sync/queue.js +158 -0
  85. package/dist/core/memory-sync/queue.spec.js +105 -0
  86. package/dist/core/onboarding/marker.js +111 -0
  87. package/dist/core/onboarding/telemetry-state.js +108 -0
  88. package/dist/core/output-style/presets.js +176 -0
  89. package/dist/core/output-style/state.js +185 -0
  90. package/dist/core/permissions/gate.js +187 -0
  91. package/dist/core/permissions/index.js +18 -0
  92. package/dist/core/permissions/mode.js +102 -0
  93. package/dist/core/permissions/state.js +215 -0
  94. package/dist/core/permissions/tool-class.js +93 -0
  95. package/dist/core/prd-check/parser.js +215 -0
  96. package/dist/core/prd-check/reporter.js +127 -0
  97. package/dist/core/prd-check/session-review.js +557 -0
  98. package/dist/core/prd-check/verifiers.js +223 -0
  99. package/dist/core/pugi-md/context-injector.js +76 -0
  100. package/dist/core/pugi-md/walk-up.js +207 -0
  101. package/dist/core/release-notes/parser.js +241 -0
  102. package/dist/core/release-notes/state.js +116 -0
  103. package/dist/core/repl/codebase-survey.js +308 -0
  104. package/dist/core/repl/history.js +11 -1
  105. package/dist/core/repl/init-interview.js +457 -0
  106. package/dist/core/repl/model-pricing.js +135 -0
  107. package/dist/core/repl/onboarding-state.js +297 -0
  108. package/dist/core/repl/session.js +1529 -30
  109. package/dist/core/repl/slash-commands.js +361 -13
  110. package/dist/core/repl/store/session-store.js +31 -2
  111. package/dist/core/repl/workspace-context.js +22 -0
  112. package/dist/core/repo-map/build.js +125 -0
  113. package/dist/core/repo-map/cache.js +185 -0
  114. package/dist/core/repo-map/extractor.js +254 -0
  115. package/dist/core/repo-map/formatter.js +145 -0
  116. package/dist/core/repo-map/scanner.js +211 -0
  117. package/dist/core/retry-budget/budget.js +284 -0
  118. package/dist/core/retry-budget/index.js +5 -0
  119. package/dist/core/session.js +44 -0
  120. package/dist/core/settings.js +80 -0
  121. package/dist/core/share/formatter.js +271 -0
  122. package/dist/core/share/redactor.js +221 -0
  123. package/dist/core/share/uploader.js +267 -0
  124. package/dist/core/skills/defaults.js +457 -0
  125. package/dist/core/subagents/dispatcher-real.js +600 -0
  126. package/dist/core/subagents/dispatcher.js +113 -24
  127. package/dist/core/subagents/index.js +18 -5
  128. package/dist/core/subagents/isolation-matrix.js +213 -0
  129. package/dist/core/subagents/spawn.js +19 -4
  130. package/dist/core/telemetry/emitter.js +229 -0
  131. package/dist/core/telemetry/queue.js +251 -0
  132. package/dist/core/theme/context.js +91 -0
  133. package/dist/core/theme/presets.js +228 -0
  134. package/dist/core/theme/state.js +181 -0
  135. package/dist/core/todos/invariant.js +10 -0
  136. package/dist/core/todos/state.js +177 -0
  137. package/dist/core/transport/version-interceptor.js +166 -0
  138. package/dist/core/vim/keymap.js +288 -0
  139. package/dist/core/vim/state.js +92 -0
  140. package/dist/index.js +28 -0
  141. package/dist/runtime/bootstrap.js +190 -0
  142. package/dist/runtime/cli.js +2603 -278
  143. package/dist/runtime/commands/chain.js +489 -0
  144. package/dist/runtime/commands/compact.js +297 -0
  145. package/dist/runtime/commands/cost.js +199 -0
  146. package/dist/runtime/commands/delegate.js +312 -0
  147. package/dist/runtime/commands/dispatch.js +126 -0
  148. package/dist/runtime/commands/doctor.js +390 -0
  149. package/dist/runtime/commands/feedback.js +184 -0
  150. package/dist/runtime/commands/hooks.js +184 -0
  151. package/dist/runtime/commands/lsp.js +212 -28
  152. package/dist/runtime/commands/mcp.js +824 -0
  153. package/dist/runtime/commands/memory.js +508 -0
  154. package/dist/runtime/commands/memory.spec.js +174 -0
  155. package/dist/runtime/commands/model.js +237 -0
  156. package/dist/runtime/commands/onboarding.js +275 -0
  157. package/dist/runtime/commands/patch.js +17 -0
  158. package/dist/runtime/commands/permissions.js +87 -0
  159. package/dist/runtime/commands/plan.js +143 -0
  160. package/dist/runtime/commands/prd-check.js +285 -0
  161. package/dist/runtime/commands/release-notes.js +229 -0
  162. package/dist/runtime/commands/repo-map.js +95 -0
  163. package/dist/runtime/commands/report.js +299 -0
  164. package/dist/runtime/commands/resume.js +118 -0
  165. package/dist/runtime/commands/review-consensus.js +17 -2
  166. package/dist/runtime/commands/rewind.js +333 -0
  167. package/dist/runtime/commands/roster.js +117 -0
  168. package/dist/runtime/commands/sessions.js +163 -0
  169. package/dist/runtime/commands/share.js +316 -0
  170. package/dist/runtime/commands/status.js +178 -0
  171. package/dist/runtime/commands/stickers.js +82 -0
  172. package/dist/runtime/commands/style.js +194 -0
  173. package/dist/runtime/commands/theme.js +196 -0
  174. package/dist/runtime/commands/update.js +289 -0
  175. package/dist/runtime/commands/vim.js +140 -0
  176. package/dist/runtime/commands/worktree.js +50 -6
  177. package/dist/runtime/headless.js +543 -0
  178. package/dist/runtime/load-hooks-or-exit.js +71 -0
  179. package/dist/runtime/plan-decompose.js +531 -0
  180. package/dist/runtime/version.js +65 -0
  181. package/dist/tools/agent-tool.js +229 -0
  182. package/dist/tools/apply-patch.js +281 -39
  183. package/dist/tools/ask-user-question.js +213 -0
  184. package/dist/tools/ask-user.js +115 -0
  185. package/dist/tools/file-tools.js +85 -14
  186. package/dist/tools/mcp-tool.js +260 -0
  187. package/dist/tools/multi-edit.js +361 -0
  188. package/dist/tools/registry.js +30 -2
  189. package/dist/tools/skill-tool.js +96 -0
  190. package/dist/tools/tasks.js +208 -0
  191. package/dist/tools/todo-write.js +184 -0
  192. package/dist/tools/web-fetch.js +147 -2
  193. package/dist/tools/web-search.js +458 -0
  194. package/dist/tui/agent-progress-card.js +111 -0
  195. package/dist/tui/agent-tree.js +10 -0
  196. package/dist/tui/ask-modal.js +2 -2
  197. package/dist/tui/ask-user-question-prompt.js +192 -0
  198. package/dist/tui/compact-banner.js +81 -0
  199. package/dist/tui/conversation-pane.js +82 -8
  200. package/dist/tui/cost-table.js +111 -0
  201. package/dist/tui/doctor-table.js +46 -0
  202. package/dist/tui/feedback-prompt.js +156 -0
  203. package/dist/tui/input-box.js +46 -2
  204. package/dist/tui/markdown-render.js +4 -4
  205. package/dist/tui/onboarding-wizard.js +240 -0
  206. package/dist/tui/repl-render.js +293 -35
  207. package/dist/tui/repl-splash.js +2 -2
  208. package/dist/tui/repl.js +45 -13
  209. package/dist/tui/splash.js +1 -1
  210. package/dist/tui/status-bar.js +94 -16
  211. package/dist/tui/status-table.js +7 -0
  212. package/dist/tui/stickers-art.js +136 -0
  213. package/dist/tui/style-table.js +28 -0
  214. package/dist/tui/theme-table.js +29 -0
  215. package/dist/tui/tool-stream-pane.js +7 -0
  216. package/dist/tui/update-banner.js +20 -2
  217. package/dist/tui/vim-input.js +267 -0
  218. package/docs/examples/codegraph.mcp.json +10 -0
  219. package/package.json +9 -6
@@ -0,0 +1,107 @@
1
+ /**
2
+ * Leak L22 (2026-05-27) — `--bare` mode predicate.
3
+ *
4
+ * Mirror of Claude Code's `--bare` flag: when active the CLI behaves
5
+ * like a plain LLM frontend with NO project auto-discovery. Useful for:
6
+ *
7
+ * - headless scripting where the operator wants deterministic, repo-
8
+ * independent behavior (`pugi --bare --print "..."`),
9
+ * - dropping into a workspace without auto-creating `.pugi/`,
10
+ * - REPL sessions that should NOT inject ambient `PUGI.md` / `CLAUDE.md`
11
+ * into the model prompt,
12
+ * - support / triage flows where the engineer needs the CLI to act
13
+ * like a fresh install regardless of where it's invoked.
14
+ *
15
+ * Discovery surfaces gated by `isBareMode()`:
16
+ *
17
+ * 1. `PUGI.md` / `AGENTS.md` / `CLAUDE.md` / `GEMINI.md` parent-dir
18
+ * walk-up (`loadTraversedMarkdown` in `core/context/markdown-traverse.ts`).
19
+ * 2. Workspace-root markdown context (`loadMarkdownContext` consumers).
20
+ * 3. Auto-init `.pugi/` scaffold on REPL boot in untouched dirs.
21
+ * 4. Persona / skill auto-load from `.pugi/skills/`.
22
+ * 5. Workspace summary (`readPugiSummary`) read on REPL session start.
23
+ *
24
+ * Activation precedence — the bare bit is "sticky" once set so any
25
+ * subprocess the CLI spawns inherits it without re-passing the flag:
26
+ *
27
+ * 1. Top-level `--bare` arg parsed by `parseArgs` in `runtime/cli.ts`.
28
+ * The parser sets `process.env.PUGI_BARE='1'` BEFORE the dispatch
29
+ * flows so callsites checking the env see the activated state.
30
+ * 2. `PUGI_BARE=1` env var (any value matching `/^(1|true|yes|on)$/i`).
31
+ * 3. Default: bare mode OFF — full auto-discovery as before.
32
+ *
33
+ * This mirrors the existing `PUGI_SKIP_SPLASH` / `PUGI_NO_AUTO_INIT`
34
+ * env-flag pattern so the bare module fits the rest of the runtime
35
+ * configuration grammar without inventing a new wire.
36
+ *
37
+ * Test surface: `apps/pugi-cli/test/bare-mode.spec.ts` exercises the
38
+ * env precedence, value parsing, and the explicit-set / clear helpers.
39
+ */
40
+ /**
41
+ * Env var consulted by `isBareMode()`. Kept as an export so the spec
42
+ * + the runtime CLI can use the same constant — no string-typing of
43
+ * the wire name across modules.
44
+ */
45
+ export const PUGI_BARE_ENV = 'PUGI_BARE';
46
+ /**
47
+ * Truthy values recognised on the `PUGI_BARE` env. Anything else
48
+ * (empty string, `0`, `false`, `no`, `off`, `disabled`, undefined) is
49
+ * treated as bare-mode OFF. The list is intentionally short — the
50
+ * value is set by the CLI parser and is not customer-typed prose, so
51
+ * we do not need a permissive boolean coercion.
52
+ */
53
+ const TRUTHY = new Set(['1', 'true', 'yes', 'on']);
54
+ /**
55
+ * Return true when bare mode is active for the current process. Reads
56
+ * `process.env[PUGI_BARE_ENV]` and applies the truthy-value match.
57
+ *
58
+ * Safe to call from any module (no FS, no side-effects). The runtime
59
+ * cost is a single env-var lookup + lower-case + set membership, so
60
+ * gating hot-path callsites with `if (isBareMode()) return ...` adds
61
+ * effectively zero overhead in the default (non-bare) case.
62
+ */
63
+ export function isBareMode(env = process.env) {
64
+ const raw = env[PUGI_BARE_ENV];
65
+ if (typeof raw !== 'string' || raw.length === 0)
66
+ return false;
67
+ return TRUTHY.has(raw.toLowerCase());
68
+ }
69
+ /**
70
+ * Explicitly activate bare mode for the current process. Called by
71
+ * `parseArgs` in `runtime/cli.ts` when `--bare` is seen on the command
72
+ * line so downstream modules (engine, REPL bootstrap, doctor probe)
73
+ * see a consistent activated state via `isBareMode()` regardless of
74
+ * whether the operator set the env var manually or used the flag.
75
+ *
76
+ * Subprocess inheritance is the reason we mutate `process.env` rather
77
+ * than threading a `bare: boolean` field through every call signature
78
+ * — every Node child_process spawn inherits `process.env` by default,
79
+ * so the bare bit propagates to MCP servers / hook scripts / git
80
+ * subprocesses without ceremony.
81
+ */
82
+ export function setBareMode(env = process.env) {
83
+ env[PUGI_BARE_ENV] = '1';
84
+ }
85
+ /**
86
+ * Clear bare mode for the current process. Provided primarily for the
87
+ * spec so adjacent tests do not leak state between cases. Production
88
+ * code does NOT call this — bare mode is a one-shot per process.
89
+ */
90
+ export function clearBareMode(env = process.env) {
91
+ delete env[PUGI_BARE_ENV];
92
+ }
93
+ /**
94
+ * Human-readable one-line banner printed by the dispatcher when bare
95
+ * mode is active and the invocation is NOT JSON-only. Kept as a single
96
+ * constant so the spec can assert the exact wording and downstream
97
+ * tools (status bars, doctor row, REPL header) stay in lockstep.
98
+ */
99
+ export const BARE_MODE_BANNER = 'Pugi --bare mode: project auto-discovery disabled.';
100
+ /**
101
+ * Short label rendered inside the `pugi doctor` table when bare mode
102
+ * is active. The doctor probe surfaces `BARE MODE` as a separate row
103
+ * so operators triaging "why is Pugi ignoring my PUGI.md" see the
104
+ * cause without grep'ing the env.
105
+ */
106
+ export const BARE_MODE_DOCTOR_LABEL = 'BARE MODE';
107
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1,149 @@
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
@@ -0,0 +1,291 @@
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
@@ -0,0 +1,96 @@
1
+ /**
2
+ * Auto-compact threshold gate.
3
+ *
4
+ * Decides whether the conversation buffer has crossed the threshold
5
+ * percent of the active model's context window. The check runs after
6
+ * every operator/persona turn before the NEXT operator input lands so
7
+ * the compaction completes BEFORE the model would have rejected the
8
+ * request with a context-overflow error.
9
+ *
10
+ * Design choices:
11
+ *
12
+ * - Pure function. The caller passes (tokenCount, windowSize, env).
13
+ * The gate returns a verdict; the session module owns the side
14
+ * effect of invoking the summariser. Pure-function shape keeps the
15
+ * spec exhaustive and the call site readable.
16
+ *
17
+ * - Hysteresis: once a compaction lands, the marker resets the
18
+ * baseline token count to "summary + tail" — the gate looks at the
19
+ * POST-marker tokens only. This is enforced upstream by the caller
20
+ * passing the post-marker count; the gate itself has no memory.
21
+ *
22
+ * - Two env knobs:
23
+ * PUGI_AUTOCOMPACT_DISABLED=1 — kill switch
24
+ * PUGI_AUTOCOMPACT_THRESHOLD=N — float in (0, 1] (default 0.75)
25
+ * Anything outside (0, 1] is rejected and the gate falls back to
26
+ * the default. Bad input never crashes the REPL.
27
+ */
28
+ /** Default trip point as a fraction of the context window. */
29
+ export const DEFAULT_THRESHOLD = 0.75;
30
+ /**
31
+ * Decide whether to fire `/compact` automatically. Pure; safe to call
32
+ * after every turn.
33
+ */
34
+ export function evaluateAutoCompact(input) {
35
+ const env = input.env ?? process.env;
36
+ const threshold = resolveThreshold(env);
37
+ if (input.windowSize <= 0 || !Number.isFinite(input.windowSize)) {
38
+ return {
39
+ kind: 'skip',
40
+ reason: 'invalid-window',
41
+ tokenCount: input.tokenCount,
42
+ windowSize: input.windowSize,
43
+ threshold,
44
+ pressure: 0,
45
+ };
46
+ }
47
+ if (env['PUGI_AUTOCOMPACT_DISABLED'] === '1') {
48
+ return {
49
+ kind: 'skip',
50
+ reason: 'disabled',
51
+ tokenCount: input.tokenCount,
52
+ windowSize: input.windowSize,
53
+ threshold,
54
+ pressure: roundPressure(input.tokenCount / input.windowSize),
55
+ };
56
+ }
57
+ const pressure = roundPressure(input.tokenCount / input.windowSize);
58
+ if (pressure >= threshold) {
59
+ return {
60
+ kind: 'fire',
61
+ tokenCount: input.tokenCount,
62
+ windowSize: input.windowSize,
63
+ threshold,
64
+ pressure,
65
+ };
66
+ }
67
+ return {
68
+ kind: 'skip',
69
+ reason: 'below-threshold',
70
+ tokenCount: input.tokenCount,
71
+ windowSize: input.windowSize,
72
+ threshold,
73
+ pressure,
74
+ };
75
+ }
76
+ /**
77
+ * Resolve the threshold from env, clamping to the (0, 1] open-closed
78
+ * interval. Bad input silently falls back to DEFAULT_THRESHOLD so the
79
+ * REPL never crashes on a malformed environment variable.
80
+ */
81
+ function resolveThreshold(env) {
82
+ const raw = env['PUGI_AUTOCOMPACT_THRESHOLD'];
83
+ if (!raw)
84
+ return DEFAULT_THRESHOLD;
85
+ const parsed = Number.parseFloat(raw);
86
+ if (!Number.isFinite(parsed) || parsed <= 0 || parsed > 1) {
87
+ return DEFAULT_THRESHOLD;
88
+ }
89
+ return parsed;
90
+ }
91
+ function roundPressure(raw) {
92
+ if (!Number.isFinite(raw) || raw < 0)
93
+ return 0;
94
+ return Math.round(raw * 1000) / 1000;
95
+ }
96
+ //# sourceMappingURL=auto-trigger.js.map