luckerr 0.41.0

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 (156) hide show
  1. package/README.md +267 -0
  2. package/README.zh-CN.md +237 -0
  3. package/dashboard/app.css +3022 -0
  4. package/dashboard/dist/app.js +30137 -0
  5. package/dashboard/dist/app.js.map +1 -0
  6. package/dashboard/dist/vendor-hljs.css +10 -0
  7. package/dashboard/dist/vendor-uplot.css +1 -0
  8. package/dashboard/index.html +19 -0
  9. package/data/deepseek-tokenizer.json.gz +0 -0
  10. package/dist/cli/acp-EOOAI4F5.js +712 -0
  11. package/dist/cli/acp-EOOAI4F5.js.map +1 -0
  12. package/dist/cli/chat-7J6GJXL2.js +51 -0
  13. package/dist/cli/chat-7J6GJXL2.js.map +1 -0
  14. package/dist/cli/chunk-2425HK6U.js +54 -0
  15. package/dist/cli/chunk-2425HK6U.js.map +1 -0
  16. package/dist/cli/chunk-25T6CVUP.js +172 -0
  17. package/dist/cli/chunk-25T6CVUP.js.map +1 -0
  18. package/dist/cli/chunk-2UQP6H6T.js +31 -0
  19. package/dist/cli/chunk-2UQP6H6T.js.map +1 -0
  20. package/dist/cli/chunk-56OAJILV.js +47 -0
  21. package/dist/cli/chunk-56OAJILV.js.map +1 -0
  22. package/dist/cli/chunk-5FTI4KXH.js +150 -0
  23. package/dist/cli/chunk-5FTI4KXH.js.map +1 -0
  24. package/dist/cli/chunk-5TWQD73O.js +2846 -0
  25. package/dist/cli/chunk-5TWQD73O.js.map +1 -0
  26. package/dist/cli/chunk-653BOCMK.js +40 -0
  27. package/dist/cli/chunk-653BOCMK.js.map +1 -0
  28. package/dist/cli/chunk-6ALJTWWQ.js +2663 -0
  29. package/dist/cli/chunk-6ALJTWWQ.js.map +1 -0
  30. package/dist/cli/chunk-6DRKA2IL.js +341 -0
  31. package/dist/cli/chunk-6DRKA2IL.js.map +1 -0
  32. package/dist/cli/chunk-6LV63NJV.js +634 -0
  33. package/dist/cli/chunk-6LV63NJV.js.map +1 -0
  34. package/dist/cli/chunk-74EX7SUH.js +25293 -0
  35. package/dist/cli/chunk-74EX7SUH.js.map +1 -0
  36. package/dist/cli/chunk-74U5RKTX.js +60611 -0
  37. package/dist/cli/chunk-74U5RKTX.js.map +1 -0
  38. package/dist/cli/chunk-ANJSUESV.js +143 -0
  39. package/dist/cli/chunk-ANJSUESV.js.map +1 -0
  40. package/dist/cli/chunk-DB2Z3DKZ.js +54 -0
  41. package/dist/cli/chunk-DB2Z3DKZ.js.map +1 -0
  42. package/dist/cli/chunk-DDIH3ZAA.js +400 -0
  43. package/dist/cli/chunk-DDIH3ZAA.js.map +1 -0
  44. package/dist/cli/chunk-ELN3Z3B2.js +621 -0
  45. package/dist/cli/chunk-ELN3Z3B2.js.map +1 -0
  46. package/dist/cli/chunk-F6BSQJGV.js +200 -0
  47. package/dist/cli/chunk-F6BSQJGV.js.map +1 -0
  48. package/dist/cli/chunk-FET2UAG5.js +246 -0
  49. package/dist/cli/chunk-FET2UAG5.js.map +1 -0
  50. package/dist/cli/chunk-FFJ342IJ.js +190 -0
  51. package/dist/cli/chunk-FFJ342IJ.js.map +1 -0
  52. package/dist/cli/chunk-GB3247B6.js +130 -0
  53. package/dist/cli/chunk-GB3247B6.js.map +1 -0
  54. package/dist/cli/chunk-HC2J4U3G.js +373 -0
  55. package/dist/cli/chunk-HC2J4U3G.js.map +1 -0
  56. package/dist/cli/chunk-HRUZAIHQ.js +42 -0
  57. package/dist/cli/chunk-HRUZAIHQ.js.map +1 -0
  58. package/dist/cli/chunk-J3ZJFUDL.js +308 -0
  59. package/dist/cli/chunk-J3ZJFUDL.js.map +1 -0
  60. package/dist/cli/chunk-J5XJHLWM.js +55 -0
  61. package/dist/cli/chunk-J5XJHLWM.js.map +1 -0
  62. package/dist/cli/chunk-JFGLMRZ6.js +160 -0
  63. package/dist/cli/chunk-JFGLMRZ6.js.map +1 -0
  64. package/dist/cli/chunk-JMBMLOBP.js +26 -0
  65. package/dist/cli/chunk-JMBMLOBP.js.map +1 -0
  66. package/dist/cli/chunk-JMWHXZEL.js +551 -0
  67. package/dist/cli/chunk-JMWHXZEL.js.map +1 -0
  68. package/dist/cli/chunk-KEQGPJBO.js +209 -0
  69. package/dist/cli/chunk-KEQGPJBO.js.map +1 -0
  70. package/dist/cli/chunk-M4K6U37F.js +232 -0
  71. package/dist/cli/chunk-M4K6U37F.js.map +1 -0
  72. package/dist/cli/chunk-MIJI2WMN.js +95 -0
  73. package/dist/cli/chunk-MIJI2WMN.js.map +1 -0
  74. package/dist/cli/chunk-MPAO3JNR.js +128 -0
  75. package/dist/cli/chunk-MPAO3JNR.js.map +1 -0
  76. package/dist/cli/chunk-PZOFBEDC.js +873 -0
  77. package/dist/cli/chunk-PZOFBEDC.js.map +1 -0
  78. package/dist/cli/chunk-RAILYQLN.js +46 -0
  79. package/dist/cli/chunk-RAILYQLN.js.map +1 -0
  80. package/dist/cli/chunk-RR35VQVT.js +90 -0
  81. package/dist/cli/chunk-RR35VQVT.js.map +1 -0
  82. package/dist/cli/chunk-RRA7VPW4.js +417 -0
  83. package/dist/cli/chunk-RRA7VPW4.js.map +1 -0
  84. package/dist/cli/chunk-RU36QVN3.js +452 -0
  85. package/dist/cli/chunk-RU36QVN3.js.map +1 -0
  86. package/dist/cli/chunk-RUBIINXR.js +1819 -0
  87. package/dist/cli/chunk-RUBIINXR.js.map +1 -0
  88. package/dist/cli/chunk-S4XVGLRW.js +499 -0
  89. package/dist/cli/chunk-S4XVGLRW.js.map +1 -0
  90. package/dist/cli/chunk-TUK7OWJA.js +51 -0
  91. package/dist/cli/chunk-TUK7OWJA.js.map +1 -0
  92. package/dist/cli/chunk-VALDDV76.js +580 -0
  93. package/dist/cli/chunk-VALDDV76.js.map +1 -0
  94. package/dist/cli/chunk-WQOGPYGN.js +11390 -0
  95. package/dist/cli/chunk-WQOGPYGN.js.map +1 -0
  96. package/dist/cli/chunk-WREKDFXT.js +34320 -0
  97. package/dist/cli/chunk-WREKDFXT.js.map +1 -0
  98. package/dist/cli/chunk-Y7XQU2EL.js +270 -0
  99. package/dist/cli/chunk-Y7XQU2EL.js.map +1 -0
  100. package/dist/cli/chunk-YBVCZJU4.js +54 -0
  101. package/dist/cli/chunk-YBVCZJU4.js.map +1 -0
  102. package/dist/cli/chunk-YLIHDXUQ.js +749 -0
  103. package/dist/cli/chunk-YLIHDXUQ.js.map +1 -0
  104. package/dist/cli/chunk-YV5XXFD7.js +767 -0
  105. package/dist/cli/chunk-YV5XXFD7.js.map +1 -0
  106. package/dist/cli/chunk-ZRCNIYRQ.js +101 -0
  107. package/dist/cli/chunk-ZRCNIYRQ.js.map +1 -0
  108. package/dist/cli/code-CRKVCMFZ.js +155 -0
  109. package/dist/cli/code-CRKVCMFZ.js.map +1 -0
  110. package/dist/cli/commands-QLMD3T7B.js +356 -0
  111. package/dist/cli/commands-QLMD3T7B.js.map +1 -0
  112. package/dist/cli/commit-53PP32NC.js +293 -0
  113. package/dist/cli/commit-53PP32NC.js.map +1 -0
  114. package/dist/cli/desktop-R6W5CLJ5.js +1046 -0
  115. package/dist/cli/desktop-R6W5CLJ5.js.map +1 -0
  116. package/dist/cli/devtools-YECO25QO.js +3719 -0
  117. package/dist/cli/devtools-YECO25QO.js.map +1 -0
  118. package/dist/cli/diff-LYNRCJZE.js +166 -0
  119. package/dist/cli/diff-LYNRCJZE.js.map +1 -0
  120. package/dist/cli/doctor-5IBP4R5J.js +28 -0
  121. package/dist/cli/doctor-5IBP4R5J.js.map +1 -0
  122. package/dist/cli/events-QN6KLN2V.js +340 -0
  123. package/dist/cli/events-QN6KLN2V.js.map +1 -0
  124. package/dist/cli/index.js +3500 -0
  125. package/dist/cli/index.js.map +1 -0
  126. package/dist/cli/mcp-FGKEH7RG.js +277 -0
  127. package/dist/cli/mcp-FGKEH7RG.js.map +1 -0
  128. package/dist/cli/mcp-browse-YCND4NWT.js +178 -0
  129. package/dist/cli/mcp-browse-YCND4NWT.js.map +1 -0
  130. package/dist/cli/mcp-inspect-V34J3VX5.js +143 -0
  131. package/dist/cli/mcp-inspect-V34J3VX5.js.map +1 -0
  132. package/dist/cli/package.json +3 -0
  133. package/dist/cli/prompt-I775PNKT.js +16 -0
  134. package/dist/cli/prompt-I775PNKT.js.map +1 -0
  135. package/dist/cli/prune-sessions-KGIIYD3P.js +44 -0
  136. package/dist/cli/prune-sessions-KGIIYD3P.js.map +1 -0
  137. package/dist/cli/replay-RDXLUAOE.js +292 -0
  138. package/dist/cli/replay-RDXLUAOE.js.map +1 -0
  139. package/dist/cli/run-RCAC2RYW.js +223 -0
  140. package/dist/cli/run-RCAC2RYW.js.map +1 -0
  141. package/dist/cli/server-FFU6TLYJ.js +3658 -0
  142. package/dist/cli/server-FFU6TLYJ.js.map +1 -0
  143. package/dist/cli/sessions-QT26MQAE.js +107 -0
  144. package/dist/cli/sessions-QT26MQAE.js.map +1 -0
  145. package/dist/cli/setup-VV4WKXHV.js +767 -0
  146. package/dist/cli/setup-VV4WKXHV.js.map +1 -0
  147. package/dist/cli/stats-JVZPQWAN.js +15 -0
  148. package/dist/cli/stats-JVZPQWAN.js.map +1 -0
  149. package/dist/cli/update-KYI3OVJP.js +15 -0
  150. package/dist/cli/update-KYI3OVJP.js.map +1 -0
  151. package/dist/cli/version-ANYORXTI.js +34 -0
  152. package/dist/cli/version-ANYORXTI.js.map +1 -0
  153. package/dist/index.d.ts +2557 -0
  154. package/dist/index.js +15000 -0
  155. package/dist/index.js.map +1 -0
  156. package/package.json +106 -0
@@ -0,0 +1,166 @@
1
+ #!/usr/bin/env node
2
+ import { createRequire as __cr } from 'node:module'; if (typeof globalThis.require === 'undefined') { globalThis.require = __cr(import.meta.url); }
3
+ import {
4
+ RecordView
5
+ } from "./chunk-YBVCZJU4.js";
6
+ import {
7
+ diffTranscripts,
8
+ findNextDivergence,
9
+ findPrevDivergence,
10
+ renderMarkdown,
11
+ renderSummaryTable
12
+ } from "./chunk-J3ZJFUDL.js";
13
+ import {
14
+ readTranscript
15
+ } from "./chunk-FFJ342IJ.js";
16
+ import {
17
+ Box_default,
18
+ Static,
19
+ Text,
20
+ render_default,
21
+ require_react,
22
+ use_app_default,
23
+ use_input_default
24
+ } from "./chunk-WREKDFXT.js";
25
+ import "./chunk-DDIH3ZAA.js";
26
+ import "./chunk-25T6CVUP.js";
27
+ import {
28
+ t
29
+ } from "./chunk-5TWQD73O.js";
30
+ import "./chunk-6ALJTWWQ.js";
31
+ import "./chunk-ANJSUESV.js";
32
+ import "./chunk-JMWHXZEL.js";
33
+ import {
34
+ __toESM
35
+ } from "./chunk-TUK7OWJA.js";
36
+
37
+ // src/cli/commands/diff.ts
38
+ import { writeFileSync } from "fs";
39
+ import { basename } from "path";
40
+ var import_react2 = __toESM(require_react(), 1);
41
+
42
+ // src/cli/ui/DiffApp.tsx
43
+ var import_react = __toESM(require_react(), 1);
44
+ function DiffApp({ report }) {
45
+ const { exit } = use_app_default();
46
+ const maxIdx = Math.max(0, report.pairs.length - 1);
47
+ const initialIdx = report.firstDivergenceTurn ? report.pairs.findIndex((p) => p.turn === report.firstDivergenceTurn) : 0;
48
+ const [idx, setIdx] = (0, import_react.useState)(Math.max(0, initialIdx));
49
+ use_input_default((input, key) => {
50
+ if (input === "q" || key.ctrl && input === "c") {
51
+ exit();
52
+ return;
53
+ }
54
+ if (input === "j" || key.downArrow || input === " " || key.return) {
55
+ setIdx((i) => Math.min(maxIdx, i + 1));
56
+ } else if (input === "k" || key.upArrow) {
57
+ setIdx((i) => Math.max(0, i - 1));
58
+ } else if (input === "g") {
59
+ setIdx(0);
60
+ } else if (input === "G") {
61
+ setIdx(maxIdx);
62
+ } else if (input === "n") {
63
+ const next = findNextDivergence(report.pairs, idx);
64
+ if (next !== -1) setIdx(next);
65
+ } else if (input === "N" || input === "p") {
66
+ const prev = findPrevDivergence(report.pairs, idx);
67
+ if (prev !== -1) setIdx(prev);
68
+ }
69
+ });
70
+ const pair = report.pairs[idx];
71
+ return /* @__PURE__ */ import_react.default.createElement(Box_default, { flexDirection: "column" }, /* @__PURE__ */ import_react.default.createElement(DiffHeader, { report }), /* @__PURE__ */ import_react.default.createElement(Box_default, { marginTop: 1, paddingX: 1, justifyContent: "space-between" }, /* @__PURE__ */ import_react.default.createElement(Text, { color: "cyan", bold: true }, t("diffApp.turnLabel", {
72
+ turn: pair?.turn ?? "?",
73
+ current: idx + 1,
74
+ total: report.pairs.length
75
+ })), /* @__PURE__ */ import_react.default.createElement(Text, null, pair ? /* @__PURE__ */ import_react.default.createElement(KindBadge, { kind: pair.kind }) : null)), /* @__PURE__ */ import_react.default.createElement(Box_default, { flexDirection: "row", marginTop: 1 }, /* @__PURE__ */ import_react.default.createElement(Pane, { label: report.a.label, headerColor: "blue", records: paneRecords(pair, "a") }), /* @__PURE__ */ import_react.default.createElement(Pane, { label: report.b.label, headerColor: "magenta", records: paneRecords(pair, "b") })), pair?.divergenceNote ? /* @__PURE__ */ import_react.default.createElement(Box_default, { marginTop: 1, paddingX: 1 }, /* @__PURE__ */ import_react.default.createElement(Text, { color: "yellow" }, "\u2605 "), /* @__PURE__ */ import_react.default.createElement(Text, null, pair.divergenceNote)) : null, /* @__PURE__ */ import_react.default.createElement(Box_default, { marginTop: 1, paddingX: 1, borderStyle: "single", borderColor: "gray" }, /* @__PURE__ */ import_react.default.createElement(Text, { dimColor: true }, /* @__PURE__ */ import_react.default.createElement(Text, { bold: true }, "j"), "/", /* @__PURE__ */ import_react.default.createElement(Text, { bold: true }, "\\u2193"), " next \\u00b7 ", /* @__PURE__ */ import_react.default.createElement(Text, { bold: true }, "k"), "/", /* @__PURE__ */ import_react.default.createElement(Text, { bold: true }, "\\u2191"), " prev \\u00b7 ", /* @__PURE__ */ import_react.default.createElement(Text, { bold: true }, "n"), " next-diverge \\u00b7", " ", /* @__PURE__ */ import_react.default.createElement(Text, { bold: true }, "N"), "/", /* @__PURE__ */ import_react.default.createElement(Text, { bold: true }, "p"), " prev-diverge \\u00b7 ", /* @__PURE__ */ import_react.default.createElement(Text, { bold: true }, "g"), "/", /* @__PURE__ */ import_react.default.createElement(Text, { bold: true }, "G"), " first/last \\u00b7 ", /* @__PURE__ */ import_react.default.createElement(Text, { bold: true }, "q"), " quit")));
76
+ }
77
+ function DiffHeader({ report }) {
78
+ const a = report.a;
79
+ const b = report.b;
80
+ const cacheDelta = b.stats.cacheHitRatio - a.stats.cacheHitRatio;
81
+ const costDelta = a.stats.totalCostUsd > 0 ? (b.stats.totalCostUsd - a.stats.totalCostUsd) / a.stats.totalCostUsd * 100 : 0;
82
+ const aStable = a.stats.prefixHashes.length <= 1;
83
+ const bStable = b.stats.prefixHashes.length <= 1;
84
+ let prefixLine = null;
85
+ if (aStable !== bStable) {
86
+ const stableLabel = aStable ? report.a.label : report.b.label;
87
+ const churnLabel = aStable ? report.b.label : report.a.label;
88
+ const churnCount = aStable ? b.stats.prefixHashes.length : a.stats.prefixHashes.length;
89
+ prefixLine = `${stableLabel} stayed byte-stable; ${churnLabel} churned ${churnCount} distinct prefixes.`;
90
+ } else if (a.stats.prefixHashes[0] && a.stats.prefixHashes[0] === b.stats.prefixHashes[0]) {
91
+ prefixLine = `shared prefix hash ${a.stats.prefixHashes[0].slice(0, 12)}\u2026 \u2014 cache delta attributable to log stability, not prompt change.`;
92
+ }
93
+ return /* @__PURE__ */ import_react.default.createElement(Box_default, { flexDirection: "column", borderStyle: "round", borderColor: "cyan", paddingX: 1 }, /* @__PURE__ */ import_react.default.createElement(Box_default, { justifyContent: "space-between" }, /* @__PURE__ */ import_react.default.createElement(Text, null, /* @__PURE__ */ import_react.default.createElement(Text, { color: "cyan", bold: true }, t("diffApp.title")), /* @__PURE__ */ import_react.default.createElement(Text, { dimColor: true }, " \\u00b7 A="), /* @__PURE__ */ import_react.default.createElement(Text, { color: "blue" }, a.label), /* @__PURE__ */ import_react.default.createElement(Text, { dimColor: true }, " vs B="), /* @__PURE__ */ import_react.default.createElement(Text, { color: "magenta" }, b.label)), /* @__PURE__ */ import_react.default.createElement(Text, { dimColor: true }, t("diffApp.turnsAligned", { count: report.pairs.length }))), /* @__PURE__ */ import_react.default.createElement(Box_default, { marginTop: 1, gap: 3 }, /* @__PURE__ */ import_react.default.createElement(Text, null, /* @__PURE__ */ import_react.default.createElement(Text, { dimColor: true }, "cache "), /* @__PURE__ */ import_react.default.createElement(Text, null, (a.stats.cacheHitRatio * 100).toFixed(1), "%"), /* @__PURE__ */ import_react.default.createElement(Text, { dimColor: true }, " \u2192 "), /* @__PURE__ */ import_react.default.createElement(Text, null, (b.stats.cacheHitRatio * 100).toFixed(1), "%"), /* @__PURE__ */ import_react.default.createElement(Text, { color: cacheDelta >= 0 ? "green" : "red", bold: true }, " ", cacheDelta >= 0 ? "+" : "", (cacheDelta * 100).toFixed(1), "pp")), /* @__PURE__ */ import_react.default.createElement(Text, null, /* @__PURE__ */ import_react.default.createElement(Text, { dimColor: true }, "cost "), /* @__PURE__ */ import_react.default.createElement(Text, null, "$", a.stats.totalCostUsd.toFixed(6)), /* @__PURE__ */ import_react.default.createElement(Text, { dimColor: true }, " \u2192 "), /* @__PURE__ */ import_react.default.createElement(Text, null, "$", b.stats.totalCostUsd.toFixed(6)), /* @__PURE__ */ import_react.default.createElement(Text, { color: costDelta <= 0 ? "green" : "red", bold: true }, " ", costDelta >= 0 ? "+" : "", costDelta.toFixed(1), "%")), /* @__PURE__ */ import_react.default.createElement(Text, null, /* @__PURE__ */ import_react.default.createElement(Text, { dimColor: true }, "model calls "), /* @__PURE__ */ import_react.default.createElement(Text, null, a.stats.turns, " \u2192 ", b.stats.turns))), prefixLine ? /* @__PURE__ */ import_react.default.createElement(Box_default, { marginTop: 1 }, /* @__PURE__ */ import_react.default.createElement(Text, { dimColor: true, italic: true }, prefixLine)) : null);
94
+ }
95
+ function Pane({
96
+ label,
97
+ headerColor,
98
+ records
99
+ }) {
100
+ return /* @__PURE__ */ import_react.default.createElement(
101
+ Box_default,
102
+ {
103
+ flexDirection: "column",
104
+ flexGrow: 1,
105
+ paddingX: 1,
106
+ borderStyle: "single",
107
+ borderColor: headerColor
108
+ },
109
+ /* @__PURE__ */ import_react.default.createElement(Text, { color: headerColor, bold: true }, label),
110
+ records.length === 0 ? /* @__PURE__ */ import_react.default.createElement(Box_default, { marginTop: 1 }, /* @__PURE__ */ import_react.default.createElement(Text, { dimColor: true, italic: true }, t("diffApp.paneEmpty"))) : /* @__PURE__ */ import_react.default.createElement(Static, { items: records.map((rec, i) => ({ key: `${label}-${i}`, rec })) }, ({ key, rec }) => /* @__PURE__ */ import_react.default.createElement(RecordView, { key, rec, compact: true }))
111
+ );
112
+ }
113
+ function KindBadge({ kind }) {
114
+ if (kind === "match") {
115
+ return /* @__PURE__ */ import_react.default.createElement(Text, { color: "green" }, t("diffApp.kindMatch"));
116
+ }
117
+ if (kind === "diverge") {
118
+ return /* @__PURE__ */ import_react.default.createElement(Text, { color: "yellow" }, t("diffApp.kindDiverge"));
119
+ }
120
+ if (kind === "only_in_a") {
121
+ return /* @__PURE__ */ import_react.default.createElement(Text, { color: "blue" }, t("diffApp.kindOnlyInA"));
122
+ }
123
+ return /* @__PURE__ */ import_react.default.createElement(Text, { color: "magenta" }, t("diffApp.kindOnlyInB"));
124
+ }
125
+ function paneRecords(pair, side) {
126
+ if (!pair) return [];
127
+ const tools = side === "a" ? pair.aTools : pair.bTools;
128
+ const assistant = side === "a" ? pair.aAssistant : pair.bAssistant;
129
+ const out = [...tools];
130
+ if (assistant) out.push(assistant);
131
+ return out;
132
+ }
133
+
134
+ // src/cli/commands/diff.ts
135
+ async function diffCommand(opts) {
136
+ const aParsed = readTranscript(opts.a);
137
+ const bParsed = readTranscript(opts.b);
138
+ const report = diffTranscripts(
139
+ { label: opts.labelA ?? basename(opts.a), parsed: aParsed },
140
+ { label: opts.labelB ?? basename(opts.b), parsed: bParsed }
141
+ );
142
+ const wantMarkdown = !!opts.mdPath;
143
+ const wantPrint = opts.print || !process.stdout.isTTY;
144
+ const wantTui = opts.tui || !wantPrint && !wantMarkdown;
145
+ if (wantMarkdown) {
146
+ console.log(renderSummaryTable(report));
147
+ const md = renderMarkdown(report);
148
+ writeFileSync(opts.mdPath, md, "utf8");
149
+ console.log(`
150
+ markdown report written to ${opts.mdPath}`);
151
+ return;
152
+ }
153
+ if (wantTui) {
154
+ const { waitUntilExit } = render_default(import_react2.default.createElement(DiffApp, { report }), {
155
+ exitOnCtrlC: true,
156
+ patchConsole: false
157
+ });
158
+ await waitUntilExit();
159
+ return;
160
+ }
161
+ console.log(renderSummaryTable(report));
162
+ }
163
+ export {
164
+ diffCommand
165
+ };
166
+ //# sourceMappingURL=diff-LYNRCJZE.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/cli/commands/diff.ts","../../src/cli/ui/DiffApp.tsx"],"sourcesContent":["import { writeFileSync } from \"node:fs\";\nimport { basename } from \"node:path\";\nimport { render } from \"ink\";\nimport React from \"react\";\nimport { diffTranscripts, renderMarkdown, renderSummaryTable } from \"../../transcript/diff.js\";\nimport { readTranscript } from \"../../transcript/log.js\";\nimport { DiffApp } from \"../ui/DiffApp.js\";\n\nexport interface DiffOptions {\n a: string;\n b: string;\n mdPath?: string;\n labelA?: string;\n labelB?: string;\n /** Force stdout summary table (no Ink TUI). Auto when stdout isn't a TTY. */\n print?: boolean;\n /** Force the TUI even when stdout isn't a TTY (rare). */\n tui?: boolean;\n}\n\nexport async function diffCommand(opts: DiffOptions): Promise<void> {\n const aParsed = readTranscript(opts.a);\n const bParsed = readTranscript(opts.b);\n\n const report = diffTranscripts(\n { label: opts.labelA ?? basename(opts.a), parsed: aParsed },\n { label: opts.labelB ?? basename(opts.b), parsed: bParsed },\n );\n\n const wantMarkdown = !!opts.mdPath;\n const wantPrint = opts.print || !process.stdout.isTTY;\n const wantTui = opts.tui || (!wantPrint && !wantMarkdown);\n\n if (wantMarkdown) {\n // Markdown export implies the user wants an artifact, not a TUI.\n // Still echo the stdout summary to confirm the action.\n console.log(renderSummaryTable(report));\n const md = renderMarkdown(report);\n writeFileSync(opts.mdPath!, md, \"utf8\");\n console.log(`\\nmarkdown report written to ${opts.mdPath}`);\n return;\n }\n\n if (wantTui) {\n const { waitUntilExit } = render(React.createElement(DiffApp, { report }), {\n exitOnCtrlC: true,\n patchConsole: false,\n });\n await waitUntilExit();\n return;\n }\n\n // stdout fallback (piped, --print, or non-TTY)\n console.log(renderSummaryTable(report));\n}\n","/**\n * Ink TUI for `luckerr diff`. Split-pane: A on the left, B on the right,\n * shared cursor. Header shows aggregate deltas; footer shows the current\n * pair's divergence note (if any) + key cheat sheet.\n *\n * j/k moves the cursor by one turn; n/N jumps to the next/prev divergent\n * turn — which is the whole point of a diff tool. Quit with q.\n *\n * Pure navigation lives in src/diff.ts (findNextDivergence / findPrevDivergence).\n */\n\nimport { Box, Static, Text, useApp, useInput } from \"ink\";\nimport React, { useState } from \"react\";\nimport { t } from \"../../i18n/index.js\";\nimport {\n type DiffReport,\n type TurnPair,\n findNextDivergence,\n findPrevDivergence,\n} from \"../../transcript/diff.js\";\nimport { RecordView } from \"./RecordView.js\";\n\nexport interface DiffAppProps {\n report: DiffReport;\n}\n\nexport function DiffApp({ report }: DiffAppProps) {\n const { exit } = useApp();\n const maxIdx = Math.max(0, report.pairs.length - 1);\n // Start at the first divergence when one exists — that's the user's most\n // likely destination. Falls back to idx 0 for fully-matching diffs.\n const initialIdx = report.firstDivergenceTurn\n ? report.pairs.findIndex((p) => p.turn === report.firstDivergenceTurn)\n : 0;\n const [idx, setIdx] = useState(Math.max(0, initialIdx));\n\n useInput((input, key) => {\n if (input === \"q\" || (key.ctrl && input === \"c\")) {\n exit();\n return;\n }\n if (input === \"j\" || key.downArrow || input === \" \" || key.return) {\n setIdx((i) => Math.min(maxIdx, i + 1));\n } else if (input === \"k\" || key.upArrow) {\n setIdx((i) => Math.max(0, i - 1));\n } else if (input === \"g\") {\n setIdx(0);\n } else if (input === \"G\") {\n setIdx(maxIdx);\n } else if (input === \"n\") {\n const next = findNextDivergence(report.pairs, idx);\n if (next !== -1) setIdx(next);\n } else if (input === \"N\" || input === \"p\") {\n const prev = findPrevDivergence(report.pairs, idx);\n if (prev !== -1) setIdx(prev);\n }\n });\n\n const pair = report.pairs[idx];\n\n return (\n <Box flexDirection=\"column\">\n <DiffHeader report={report} />\n\n <Box marginTop={1} paddingX={1} justifyContent=\"space-between\">\n <Text color=\"cyan\" bold>\n {t(\"diffApp.turnLabel\", {\n turn: pair?.turn ?? \"?\",\n current: idx + 1,\n total: report.pairs.length,\n })}\n </Text>\n <Text>{pair ? <KindBadge kind={pair.kind} /> : null}</Text>\n </Box>\n\n <Box flexDirection=\"row\" marginTop={1}>\n <Pane label={report.a.label} headerColor=\"blue\" records={paneRecords(pair, \"a\")} />\n <Pane label={report.b.label} headerColor=\"magenta\" records={paneRecords(pair, \"b\")} />\n </Box>\n\n {pair?.divergenceNote ? (\n <Box marginTop={1} paddingX={1}>\n <Text color=\"yellow\">★ </Text>\n <Text>{pair.divergenceNote}</Text>\n </Box>\n ) : null}\n\n <Box marginTop={1} paddingX={1} borderStyle=\"single\" borderColor=\"gray\">\n <Text dimColor>\n <Text bold>j</Text>/<Text bold>\\u2193</Text> next \\u00b7 <Text bold>k</Text>/\n <Text bold>\\u2191</Text> prev \\u00b7 <Text bold>n</Text> next-diverge \\u00b7{\" \"}\n <Text bold>N</Text>/<Text bold>p</Text> prev-diverge \\u00b7 <Text bold>g</Text>/\n <Text bold>G</Text> first/last \\u00b7 <Text bold>q</Text> quit\n </Text>\n </Box>\n </Box>\n );\n}\n\n// ----------------------------------------------------------------------------\n\nfunction DiffHeader({ report }: { report: DiffReport }) {\n const a = report.a;\n const b = report.b;\n\n const cacheDelta = b.stats.cacheHitRatio - a.stats.cacheHitRatio;\n const costDelta =\n a.stats.totalCostUsd > 0\n ? ((b.stats.totalCostUsd - a.stats.totalCostUsd) / a.stats.totalCostUsd) * 100\n : 0;\n\n // Prefix stability one-liner (same logic as the stdout summary).\n const aStable = a.stats.prefixHashes.length <= 1;\n const bStable = b.stats.prefixHashes.length <= 1;\n let prefixLine: string | null = null;\n if (aStable !== bStable) {\n const stableLabel = aStable ? report.a.label : report.b.label;\n const churnLabel = aStable ? report.b.label : report.a.label;\n const churnCount = aStable ? b.stats.prefixHashes.length : a.stats.prefixHashes.length;\n prefixLine = `${stableLabel} stayed byte-stable; ${churnLabel} churned ${churnCount} distinct prefixes.`;\n } else if (a.stats.prefixHashes[0] && a.stats.prefixHashes[0] === b.stats.prefixHashes[0]) {\n prefixLine = `shared prefix hash ${a.stats.prefixHashes[0].slice(0, 12)}… — cache delta attributable to log stability, not prompt change.`;\n }\n\n return (\n <Box flexDirection=\"column\" borderStyle=\"round\" borderColor=\"cyan\" paddingX={1}>\n <Box justifyContent=\"space-between\">\n <Text>\n <Text color=\"cyan\" bold>\n {t(\"diffApp.title\")}\n </Text>\n <Text dimColor> \\u00b7 A=</Text>\n <Text color=\"blue\">{a.label}</Text>\n <Text dimColor> vs B=</Text>\n <Text color=\"magenta\">{b.label}</Text>\n </Text>\n <Text dimColor>{t(\"diffApp.turnsAligned\", { count: report.pairs.length })}</Text>\n </Box>\n\n <Box marginTop={1} gap={3}>\n <Text>\n <Text dimColor>cache </Text>\n <Text>{(a.stats.cacheHitRatio * 100).toFixed(1)}%</Text>\n <Text dimColor> → </Text>\n <Text>{(b.stats.cacheHitRatio * 100).toFixed(1)}%</Text>\n <Text color={cacheDelta >= 0 ? \"green\" : \"red\"} bold>\n {\" \"}\n {cacheDelta >= 0 ? \"+\" : \"\"}\n {(cacheDelta * 100).toFixed(1)}pp\n </Text>\n </Text>\n <Text>\n <Text dimColor>cost </Text>\n <Text>${a.stats.totalCostUsd.toFixed(6)}</Text>\n <Text dimColor> → </Text>\n <Text>${b.stats.totalCostUsd.toFixed(6)}</Text>\n <Text color={costDelta <= 0 ? \"green\" : \"red\"} bold>\n {\" \"}\n {costDelta >= 0 ? \"+\" : \"\"}\n {costDelta.toFixed(1)}%\n </Text>\n </Text>\n <Text>\n <Text dimColor>model calls </Text>\n <Text>\n {a.stats.turns} → {b.stats.turns}\n </Text>\n </Text>\n </Box>\n\n {prefixLine ? (\n <Box marginTop={1}>\n <Text dimColor italic>\n {prefixLine}\n </Text>\n </Box>\n ) : null}\n </Box>\n );\n}\n\nfunction Pane({\n label,\n headerColor,\n records,\n}: {\n label: string;\n headerColor: \"blue\" | \"magenta\";\n records: TurnPair[\"aTools\"];\n}) {\n return (\n <Box\n flexDirection=\"column\"\n flexGrow={1}\n paddingX={1}\n borderStyle=\"single\"\n borderColor={headerColor}\n >\n <Text color={headerColor} bold>\n {label}\n </Text>\n {records.length === 0 ? (\n <Box marginTop={1}>\n <Text dimColor italic>\n {t(\"diffApp.paneEmpty\")}\n </Text>\n </Box>\n ) : (\n <Static items={records.map((rec, i) => ({ key: `${label}-${i}`, rec }))}>\n {({ key, rec }) => <RecordView key={key} rec={rec} compact />}\n </Static>\n )}\n </Box>\n );\n}\n\nfunction KindBadge({ kind }: { kind: TurnPair[\"kind\"] }) {\n if (kind === \"match\") {\n return <Text color=\"green\">{t(\"diffApp.kindMatch\")}</Text>;\n }\n if (kind === \"diverge\") {\n return <Text color=\"yellow\">{t(\"diffApp.kindDiverge\")}</Text>;\n }\n if (kind === \"only_in_a\") {\n return <Text color=\"blue\">{t(\"diffApp.kindOnlyInA\")}</Text>;\n }\n return <Text color=\"magenta\">{t(\"diffApp.kindOnlyInB\")}</Text>;\n}\n\n// ----------------------------------------------------------------------------\n\nfunction paneRecords(pair: TurnPair | undefined, side: \"a\" | \"b\"): TurnPair[\"aTools\"] {\n if (!pair) return [];\n const tools = side === \"a\" ? pair.aTools : pair.bTools;\n const assistant = side === \"a\" ? pair.aAssistant : pair.bAssistant;\n const out: TurnPair[\"aTools\"] = [...tools];\n if (assistant) out.push(assistant);\n return out;\n}\n"],"mappings":"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","names":["import_react","React","React"]}
@@ -0,0 +1,28 @@
1
+ #!/usr/bin/env node
2
+ import { createRequire as __cr } from 'node:module'; if (typeof globalThis.require === 'undefined') { globalThis.require = __cr(import.meta.url); }
3
+ import {
4
+ doctorCommand,
5
+ formatDoctorJson,
6
+ runDoctorChecks
7
+ } from "./chunk-RU36QVN3.js";
8
+ import "./chunk-74EX7SUH.js";
9
+ import "./chunk-F6BSQJGV.js";
10
+ import "./chunk-2UQP6H6T.js";
11
+ import "./chunk-ZRCNIYRQ.js";
12
+ import "./chunk-PZOFBEDC.js";
13
+ import "./chunk-KEQGPJBO.js";
14
+ import "./chunk-S4XVGLRW.js";
15
+ import "./chunk-DDIH3ZAA.js";
16
+ import "./chunk-25T6CVUP.js";
17
+ import "./chunk-Y7XQU2EL.js";
18
+ import "./chunk-5TWQD73O.js";
19
+ import "./chunk-6ALJTWWQ.js";
20
+ import "./chunk-JMWHXZEL.js";
21
+ import "./chunk-MPAO3JNR.js";
22
+ import "./chunk-TUK7OWJA.js";
23
+ export {
24
+ doctorCommand,
25
+ formatDoctorJson,
26
+ runDoctorChecks
27
+ };
28
+ //# sourceMappingURL=doctor-5IBP4R5J.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":[],"sourcesContent":[],"mappings":"","names":[]}
@@ -0,0 +1,340 @@
1
+ #!/usr/bin/env node
2
+ import { createRequire as __cr } from 'node:module'; if (typeof globalThis.require === 'undefined') { globalThis.require = __cr(import.meta.url); }
3
+ import {
4
+ readEventLogFile
5
+ } from "./chunk-J5XJHLWM.js";
6
+ import {
7
+ eventLogPath
8
+ } from "./chunk-DB2Z3DKZ.js";
9
+ import "./chunk-Y7XQU2EL.js";
10
+ import "./chunk-TUK7OWJA.js";
11
+
12
+ // src/core/reducers.ts
13
+ function emptyConversation() {
14
+ return { messages: [], pendingToolCalls: [] };
15
+ }
16
+ function emptyBudget(capUsd = null) {
17
+ return {
18
+ spentUsd: 0,
19
+ capUsd,
20
+ promptTokens: 0,
21
+ completionTokens: 0,
22
+ cacheHitTokens: 0,
23
+ cacheMissTokens: 0,
24
+ warned: false,
25
+ blocked: false
26
+ };
27
+ }
28
+ function emptyPlan() {
29
+ return { steps: [], body: null, submittedTurn: null };
30
+ }
31
+ function emptyWorkspace() {
32
+ return { filesTouched: /* @__PURE__ */ new Map(), lastCheckpointId: null };
33
+ }
34
+ function emptyCapabilities() {
35
+ return { tools: [] };
36
+ }
37
+ function emptyStatus() {
38
+ return { current: null };
39
+ }
40
+ function emptySessionMeta() {
41
+ return {
42
+ name: null,
43
+ openedAt: null,
44
+ resumedFromTurn: null,
45
+ currentTurn: 0,
46
+ lastError: null
47
+ };
48
+ }
49
+ function emptyProjections(capUsd = null) {
50
+ return {
51
+ conversation: emptyConversation(),
52
+ budget: emptyBudget(capUsd),
53
+ plan: emptyPlan(),
54
+ workspace: emptyWorkspace(),
55
+ capabilities: emptyCapabilities(),
56
+ status: emptyStatus(),
57
+ session: emptySessionMeta()
58
+ };
59
+ }
60
+ var conversation = (v, ev) => {
61
+ switch (ev.type) {
62
+ case "user.message": {
63
+ const msg = { role: "user", content: ev.text };
64
+ return { ...v, messages: [...v.messages, msg] };
65
+ }
66
+ case "model.final": {
67
+ const msg = { role: "assistant", content: ev.content };
68
+ if (ev.toolCalls.length > 0) msg.tool_calls = [...ev.toolCalls];
69
+ if (ev.reasoningContent !== void 0) msg.reasoning_content = ev.reasoningContent;
70
+ return { ...v, messages: [...v.messages, msg] };
71
+ }
72
+ case "tool.intent":
73
+ return {
74
+ ...v,
75
+ pendingToolCalls: [...v.pendingToolCalls, { callId: ev.callId, name: ev.name }]
76
+ };
77
+ case "tool.result": {
78
+ const msg = { role: "tool", content: ev.output, tool_call_id: ev.callId };
79
+ return {
80
+ messages: [...v.messages, msg],
81
+ pendingToolCalls: v.pendingToolCalls.filter((c) => c.callId !== ev.callId)
82
+ };
83
+ }
84
+ case "tool.denied": {
85
+ const msg = {
86
+ role: "tool",
87
+ content: `denied: ${ev.reason}`,
88
+ tool_call_id: ev.callId
89
+ };
90
+ return {
91
+ messages: [...v.messages, msg],
92
+ pendingToolCalls: v.pendingToolCalls.filter((c) => c.callId !== ev.callId)
93
+ };
94
+ }
95
+ case "session.compacted":
96
+ return { messages: [...ev.replacementMessages], pendingToolCalls: [] };
97
+ default:
98
+ return v;
99
+ }
100
+ };
101
+ var budget = (v, ev) => {
102
+ switch (ev.type) {
103
+ case "model.final": {
104
+ const u = ev.usage;
105
+ return {
106
+ ...v,
107
+ spentUsd: v.spentUsd + ev.costUsd,
108
+ promptTokens: v.promptTokens + (u.prompt_tokens ?? 0),
109
+ completionTokens: v.completionTokens + (u.completion_tokens ?? 0),
110
+ cacheHitTokens: v.cacheHitTokens + (u.prompt_cache_hit_tokens ?? 0),
111
+ cacheMissTokens: v.cacheMissTokens + (u.prompt_cache_miss_tokens ?? 0)
112
+ };
113
+ }
114
+ case "policy.budget.warning":
115
+ return { ...v, warned: true };
116
+ case "policy.budget.blocked":
117
+ return { ...v, blocked: true };
118
+ default:
119
+ return v;
120
+ }
121
+ };
122
+ var plan = (v, ev) => {
123
+ switch (ev.type) {
124
+ case "plan.submitted": {
125
+ const steps = ev.steps.map((s) => ({
126
+ id: s.id,
127
+ title: s.title,
128
+ action: s.action,
129
+ risk: s.risk,
130
+ completed: false
131
+ }));
132
+ return { steps, body: ev.body, submittedTurn: ev.turn };
133
+ }
134
+ case "plan.step.completed": {
135
+ if (!v.steps.some((s) => s.id === ev.stepId)) return v;
136
+ return {
137
+ ...v,
138
+ steps: v.steps.map(
139
+ (s) => s.id === ev.stepId ? { ...s, completed: true, notes: ev.notes } : s
140
+ )
141
+ };
142
+ }
143
+ default:
144
+ return v;
145
+ }
146
+ };
147
+ var workspace = (v, ev) => {
148
+ switch (ev.type) {
149
+ case "effect.file.touched": {
150
+ const next = new Map(v.filesTouched);
151
+ next.set(ev.path, ev.mode);
152
+ return { ...v, filesTouched: next };
153
+ }
154
+ case "checkpoint.created":
155
+ return { ...v, lastCheckpointId: ev.checkpointId };
156
+ default:
157
+ return v;
158
+ }
159
+ };
160
+ var capabilities = (v, ev) => {
161
+ switch (ev.type) {
162
+ case "capability.registered": {
163
+ const filtered = v.tools.filter((t) => t.name !== ev.name);
164
+ return { tools: [...filtered, { name: ev.name, permission: ev.permission }] };
165
+ }
166
+ case "capability.removed":
167
+ return { tools: v.tools.filter((t) => t.name !== ev.name) };
168
+ default:
169
+ return v;
170
+ }
171
+ };
172
+ var STATUS_CLEARING = /* @__PURE__ */ new Set([
173
+ "model.delta",
174
+ "model.final",
175
+ "tool.dispatched",
176
+ "tool.result",
177
+ "tool.denied",
178
+ "error"
179
+ ]);
180
+ var status = (v, ev) => {
181
+ if (ev.type === "status") return { current: ev.text };
182
+ if (STATUS_CLEARING.has(ev.type) && v.current !== null) return { current: null };
183
+ return v;
184
+ };
185
+ var sessionMeta = (v, ev) => {
186
+ let next = v;
187
+ if (ev.turn > next.currentTurn) next = { ...next, currentTurn: ev.turn };
188
+ switch (ev.type) {
189
+ case "session.opened":
190
+ return {
191
+ ...next,
192
+ name: ev.name,
193
+ openedAt: ev.ts,
194
+ resumedFromTurn: ev.resumedFromTurn
195
+ };
196
+ case "error":
197
+ return { ...next, lastError: ev.message };
198
+ default:
199
+ return next;
200
+ }
201
+ };
202
+ function apply(state, ev) {
203
+ return {
204
+ conversation: conversation(state.conversation, ev),
205
+ budget: budget(state.budget, ev),
206
+ plan: plan(state.plan, ev),
207
+ workspace: workspace(state.workspace, ev),
208
+ capabilities: capabilities(state.capabilities, ev),
209
+ status: status(state.status, ev),
210
+ session: sessionMeta(state.session, ev)
211
+ };
212
+ }
213
+ function replay(events, capUsd = null) {
214
+ let s = emptyProjections(capUsd);
215
+ for (const ev of events) s = apply(s, ev);
216
+ return s;
217
+ }
218
+
219
+ // src/cli/commands/events.ts
220
+ function eventsCommand(opts) {
221
+ const path = eventLogPath(opts.name);
222
+ let events = readEventLogFile(path);
223
+ if (events.length === 0) {
224
+ console.error(`no events for session "${opts.name}"`);
225
+ console.error(`looked at: ${path}`);
226
+ console.error("(sessions auto-create the sidecar on first turn \u2014 has this session run yet?)");
227
+ process.exit(1);
228
+ return;
229
+ }
230
+ if (opts.type) events = events.filter((e) => e.type === opts.type);
231
+ if (opts.since !== void 0 && Number.isFinite(opts.since)) {
232
+ events = events.filter((e) => e.id >= opts.since);
233
+ }
234
+ if (opts.tail !== void 0 && Number.isFinite(opts.tail) && opts.tail > 0) {
235
+ events = events.slice(-opts.tail);
236
+ }
237
+ if (opts.projection) {
238
+ const p = replay(events);
239
+ console.log(JSON.stringify(p, mapReplacer, 2));
240
+ return;
241
+ }
242
+ if (opts.json) {
243
+ for (const e of events) console.log(JSON.stringify(e));
244
+ return;
245
+ }
246
+ console.log(`[events] ${opts.name} ${events.length} event(s) ${path}`);
247
+ console.log("");
248
+ for (const e of events) console.log(formatEvent(e));
249
+ }
250
+ function formatEvent(e) {
251
+ const id = String(e.id).padStart(4);
252
+ const turn = `t${e.turn}`.padEnd(4);
253
+ const ts = e.ts.replace("T", " ").replace(/\.\d+Z$/, "");
254
+ const type = e.type.padEnd(22);
255
+ return `[${id}] ${turn} ${ts} ${type} ${detailsFor(e)}`;
256
+ }
257
+ function detailsFor(e) {
258
+ switch (e.type) {
259
+ case "user.message":
260
+ return quote(e.text, 80);
261
+ case "slash.invoked":
262
+ return `/${e.name} ${e.args}`.trim();
263
+ case "model.turn.started":
264
+ return `model=${e.model} effort=${e.reasoningEffort} prefix=${e.prefixHash.slice(0, 8)}`;
265
+ case "model.delta":
266
+ return `${e.channel} ${quote(e.text, 60)}`;
267
+ case "model.final": {
268
+ const u = e.usage;
269
+ const tokens = `in=${u.prompt_tokens ?? 0} out=${u.completion_tokens ?? 0}`;
270
+ const tail = e.forcedSummary ? " [forcedSummary]" : "";
271
+ return `cost=$${e.costUsd.toFixed(4)} ${tokens}${tail}`;
272
+ }
273
+ case "tool.intent":
274
+ return `${e.callId} ${e.name} args=${truncate(e.args, 60)}`;
275
+ case "tool.dispatched":
276
+ return e.callId;
277
+ case "tool.denied":
278
+ return `${e.callId} reason=${e.reason}`;
279
+ case "tool.result":
280
+ return `${e.callId} ${e.ok ? "ok" : "err"} ${e.output.length}B${e.truncated ? " [trunc]" : ""}`;
281
+ case "tool.call":
282
+ return `${e.name} args=${truncate(JSON.stringify(e.args), 60)}`;
283
+ case "tool.confirm.allow":
284
+ return `${e.kind} ${quote(e.payload.command, 60)}`;
285
+ case "tool.confirm.deny":
286
+ return `${e.kind} ${quote(e.payload.command, 60)}${e.denyContext ? ` \u2014 ${truncate(e.denyContext, 40)}` : ""}`;
287
+ case "tool.confirm.always_allow":
288
+ return `${e.kind} ${quote(e.payload.command, 60)} prefix=${truncate(e.prefix, 30)}`;
289
+ case "effect.file.touched":
290
+ return `${e.mode} ${e.path} (${e.bytes}B)`;
291
+ case "effect.memory.written":
292
+ return `${e.scope}:${e.key}`;
293
+ case "plan.submitted":
294
+ return `${e.steps.length} step(s)`;
295
+ case "plan.step.completed":
296
+ return `${e.stepId}${e.title ? ` \u2014 ${e.title}` : ""}`;
297
+ case "checkpoint.created":
298
+ return `${e.checkpointId} "${e.name}" ${e.fileCount} file(s) ${e.bytes}B`;
299
+ case "checkpoint.restored":
300
+ return `${e.checkpointId} restored=${e.restored} removed=${e.removed} skipped=${e.skipped}`;
301
+ case "hook.fired":
302
+ return `${e.phase} ${e.hookName} \u2192 ${e.outcome}`;
303
+ case "policy.budget.warning":
304
+ return `$${e.spentUsd.toFixed(4)} / $${e.capUsd.toFixed(2)}`;
305
+ case "policy.budget.blocked":
306
+ return `$${e.spentUsd.toFixed(4)} / $${e.capUsd.toFixed(2)} BLOCKED`;
307
+ case "policy.escalated":
308
+ return `${e.fromModel} \u2192 ${e.toModel} (${e.reason})${e.rationale ? ` "${e.rationale}"` : ""}`;
309
+ case "session.opened":
310
+ return `${e.name} resumed-from-turn=${e.resumedFromTurn}`;
311
+ case "session.compacted":
312
+ return `${e.beforeMessages} \u2192 ${e.afterMessages} msgs (${e.reason})`;
313
+ case "capability.registered":
314
+ return `${e.name} ${e.permission}`;
315
+ case "capability.removed":
316
+ return e.name;
317
+ case "status":
318
+ return quote(e.text, 80);
319
+ case "error":
320
+ return `${e.recoverable ? "[recoverable] " : ""}${quote(e.message, 80)}`;
321
+ default:
322
+ return "";
323
+ }
324
+ }
325
+ function quote(s, max) {
326
+ const flat = s.replace(/\s+/g, " ").trim();
327
+ return flat.length <= max ? `"${flat}"` : `"${flat.slice(0, max - 1)}\u2026"`;
328
+ }
329
+ function truncate(s, max) {
330
+ return s.length <= max ? s : `${s.slice(0, max - 1)}\u2026`;
331
+ }
332
+ function mapReplacer(_key, value) {
333
+ if (value instanceof Map) return Object.fromEntries(value);
334
+ if (value instanceof Set) return [...value];
335
+ return value;
336
+ }
337
+ export {
338
+ eventsCommand
339
+ };
340
+ //# sourceMappingURL=events-QN6KLN2V.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/core/reducers.ts","../../src/cli/commands/events.ts"],"sourcesContent":["/** Pure projection reducers over the Event log — deterministic, no I/O, no mutation. */\n\nimport type { ChatMessage } from \"../types.js\";\nimport type {\n BudgetView,\n CapabilityView,\n ConversationView,\n Event,\n PlanStepView,\n PlanView,\n ProjectionSet,\n Reducer,\n SessionMetaView,\n StatusView,\n WorkspaceView,\n} from \"./events.js\";\n\nexport function emptyConversation(): ConversationView {\n return { messages: [], pendingToolCalls: [] };\n}\n\nexport function emptyBudget(capUsd: number | null = null): BudgetView {\n return {\n spentUsd: 0,\n capUsd,\n promptTokens: 0,\n completionTokens: 0,\n cacheHitTokens: 0,\n cacheMissTokens: 0,\n warned: false,\n blocked: false,\n };\n}\n\nexport function emptyPlan(): PlanView {\n return { steps: [], body: null, submittedTurn: null };\n}\n\nexport function emptyWorkspace(): WorkspaceView {\n return { filesTouched: new Map(), lastCheckpointId: null };\n}\n\nexport function emptyCapabilities(): CapabilityView {\n return { tools: [] };\n}\n\nexport function emptyStatus(): StatusView {\n return { current: null };\n}\n\nexport function emptySessionMeta(): SessionMetaView {\n return {\n name: null,\n openedAt: null,\n resumedFromTurn: null,\n currentTurn: 0,\n lastError: null,\n };\n}\n\nexport function emptyProjections(capUsd: number | null = null): ProjectionSet {\n return {\n conversation: emptyConversation(),\n budget: emptyBudget(capUsd),\n plan: emptyPlan(),\n workspace: emptyWorkspace(),\n capabilities: emptyCapabilities(),\n status: emptyStatus(),\n session: emptySessionMeta(),\n };\n}\n\nexport const conversation: Reducer<ConversationView> = (v, ev) => {\n switch (ev.type) {\n case \"user.message\": {\n const msg: ChatMessage = { role: \"user\", content: ev.text };\n return { ...v, messages: [...v.messages, msg] };\n }\n case \"model.final\": {\n const msg: ChatMessage = { role: \"assistant\", content: ev.content };\n if (ev.toolCalls.length > 0) msg.tool_calls = [...ev.toolCalls];\n if (ev.reasoningContent !== undefined) msg.reasoning_content = ev.reasoningContent;\n return { ...v, messages: [...v.messages, msg] };\n }\n case \"tool.intent\":\n return {\n ...v,\n pendingToolCalls: [...v.pendingToolCalls, { callId: ev.callId, name: ev.name }],\n };\n case \"tool.result\": {\n const msg: ChatMessage = { role: \"tool\", content: ev.output, tool_call_id: ev.callId };\n return {\n messages: [...v.messages, msg],\n pendingToolCalls: v.pendingToolCalls.filter((c) => c.callId !== ev.callId),\n };\n }\n case \"tool.denied\": {\n const msg: ChatMessage = {\n role: \"tool\",\n content: `denied: ${ev.reason}`,\n tool_call_id: ev.callId,\n };\n return {\n messages: [...v.messages, msg],\n pendingToolCalls: v.pendingToolCalls.filter((c) => c.callId !== ev.callId),\n };\n }\n case \"session.compacted\":\n return { messages: [...ev.replacementMessages], pendingToolCalls: [] };\n default:\n return v;\n }\n};\n\nexport const budget: Reducer<BudgetView> = (v, ev) => {\n switch (ev.type) {\n case \"model.final\": {\n const u = ev.usage;\n return {\n ...v,\n spentUsd: v.spentUsd + ev.costUsd,\n promptTokens: v.promptTokens + (u.prompt_tokens ?? 0),\n completionTokens: v.completionTokens + (u.completion_tokens ?? 0),\n cacheHitTokens: v.cacheHitTokens + (u.prompt_cache_hit_tokens ?? 0),\n cacheMissTokens: v.cacheMissTokens + (u.prompt_cache_miss_tokens ?? 0),\n };\n }\n case \"policy.budget.warning\":\n return { ...v, warned: true };\n case \"policy.budget.blocked\":\n return { ...v, blocked: true };\n default:\n return v;\n }\n};\n\nexport const plan: Reducer<PlanView> = (v, ev) => {\n switch (ev.type) {\n case \"plan.submitted\": {\n const steps: PlanStepView[] = ev.steps.map((s) => ({\n id: s.id,\n title: s.title,\n action: s.action,\n risk: s.risk,\n completed: false,\n }));\n return { steps, body: ev.body, submittedTurn: ev.turn };\n }\n case \"plan.step.completed\": {\n if (!v.steps.some((s) => s.id === ev.stepId)) return v;\n return {\n ...v,\n steps: v.steps.map((s) =>\n s.id === ev.stepId ? { ...s, completed: true, notes: ev.notes } : s,\n ),\n };\n }\n default:\n return v;\n }\n};\n\nexport const workspace: Reducer<WorkspaceView> = (v, ev) => {\n switch (ev.type) {\n case \"effect.file.touched\": {\n const next = new Map(v.filesTouched);\n next.set(ev.path, ev.mode);\n return { ...v, filesTouched: next };\n }\n case \"checkpoint.created\":\n return { ...v, lastCheckpointId: ev.checkpointId };\n default:\n return v;\n }\n};\n\nexport const capabilities: Reducer<CapabilityView> = (v, ev) => {\n switch (ev.type) {\n case \"capability.registered\": {\n const filtered = v.tools.filter((t) => t.name !== ev.name);\n return { tools: [...filtered, { name: ev.name, permission: ev.permission }] };\n }\n case \"capability.removed\":\n return { tools: v.tools.filter((t) => t.name !== ev.name) };\n default:\n return v;\n }\n};\n\nconst STATUS_CLEARING: ReadonlySet<Event[\"type\"]> = new Set([\n \"model.delta\",\n \"model.final\",\n \"tool.dispatched\",\n \"tool.result\",\n \"tool.denied\",\n \"error\",\n]);\n\nexport const status: Reducer<StatusView> = (v, ev) => {\n if (ev.type === \"status\") return { current: ev.text };\n if (STATUS_CLEARING.has(ev.type) && v.current !== null) return { current: null };\n return v;\n};\n\nexport const sessionMeta: Reducer<SessionMetaView> = (v, ev) => {\n let next = v;\n if (ev.turn > next.currentTurn) next = { ...next, currentTurn: ev.turn };\n switch (ev.type) {\n case \"session.opened\":\n return {\n ...next,\n name: ev.name,\n openedAt: ev.ts,\n resumedFromTurn: ev.resumedFromTurn,\n };\n case \"error\":\n return { ...next, lastError: ev.message };\n default:\n return next;\n }\n};\n\nexport function apply(state: ProjectionSet, ev: Event): ProjectionSet {\n return {\n conversation: conversation(state.conversation, ev),\n budget: budget(state.budget, ev),\n plan: plan(state.plan, ev),\n workspace: workspace(state.workspace, ev),\n capabilities: capabilities(state.capabilities, ev),\n status: status(state.status, ev),\n session: sessionMeta(state.session, ev),\n };\n}\n\nexport function replay(events: Iterable<Event>, capUsd: number | null = null): ProjectionSet {\n let s = emptyProjections(capUsd);\n for (const ev of events) s = apply(s, ev);\n return s;\n}\n","import { eventLogPath } from \"../../adapters/event-sink-jsonl.js\";\nimport { readEventLogFile } from \"../../adapters/event-source-jsonl.js\";\nimport type { Event } from \"../../core/events.js\";\nimport { replay as replayReducers } from \"../../core/reducers.js\";\n\nexport interface EventsOptions {\n name: string;\n type?: string;\n since?: number;\n tail?: number;\n json?: boolean;\n projection?: boolean;\n}\n\nexport function eventsCommand(opts: EventsOptions): void {\n const path = eventLogPath(opts.name);\n let events = readEventLogFile(path);\n\n if (events.length === 0) {\n console.error(`no events for session \"${opts.name}\"`);\n console.error(`looked at: ${path}`);\n console.error(\"(sessions auto-create the sidecar on first turn — has this session run yet?)\");\n process.exit(1);\n return;\n }\n\n if (opts.type) events = events.filter((e) => e.type === opts.type);\n if (opts.since !== undefined && Number.isFinite(opts.since)) {\n events = events.filter((e) => e.id >= opts.since!);\n }\n if (opts.tail !== undefined && Number.isFinite(opts.tail) && opts.tail > 0) {\n events = events.slice(-opts.tail);\n }\n\n if (opts.projection) {\n const p = replayReducers(events);\n console.log(JSON.stringify(p, mapReplacer, 2));\n return;\n }\n\n if (opts.json) {\n for (const e of events) console.log(JSON.stringify(e));\n return;\n }\n\n console.log(`[events] ${opts.name} ${events.length} event(s) ${path}`);\n console.log(\"\");\n for (const e of events) console.log(formatEvent(e));\n}\n\nfunction formatEvent(e: Event): string {\n const id = String(e.id).padStart(4);\n const turn = `t${e.turn}`.padEnd(4);\n const ts = e.ts.replace(\"T\", \" \").replace(/\\.\\d+Z$/, \"\");\n const type = e.type.padEnd(22);\n return `[${id}] ${turn} ${ts} ${type} ${detailsFor(e)}`;\n}\n\nfunction detailsFor(e: Event): string {\n switch (e.type) {\n case \"user.message\":\n return quote(e.text, 80);\n case \"slash.invoked\":\n return `/${e.name} ${e.args}`.trim();\n case \"model.turn.started\":\n return `model=${e.model} effort=${e.reasoningEffort} prefix=${e.prefixHash.slice(0, 8)}`;\n case \"model.delta\":\n return `${e.channel} ${quote(e.text, 60)}`;\n case \"model.final\": {\n const u = e.usage;\n const tokens = `in=${u.prompt_tokens ?? 0} out=${u.completion_tokens ?? 0}`;\n const tail = e.forcedSummary ? \" [forcedSummary]\" : \"\";\n return `cost=$${e.costUsd.toFixed(4)} ${tokens}${tail}`;\n }\n case \"tool.intent\":\n return `${e.callId} ${e.name} args=${truncate(e.args, 60)}`;\n case \"tool.dispatched\":\n return e.callId;\n case \"tool.denied\":\n return `${e.callId} reason=${e.reason}`;\n case \"tool.result\":\n return `${e.callId} ${e.ok ? \"ok\" : \"err\"} ${e.output.length}B${e.truncated ? \" [trunc]\" : \"\"}`;\n case \"tool.call\":\n return `${e.name} args=${truncate(JSON.stringify(e.args), 60)}`;\n case \"tool.confirm.allow\":\n return `${e.kind} ${quote(e.payload.command, 60)}`;\n case \"tool.confirm.deny\":\n return `${e.kind} ${quote(e.payload.command, 60)}${e.denyContext ? ` — ${truncate(e.denyContext, 40)}` : \"\"}`;\n case \"tool.confirm.always_allow\":\n return `${e.kind} ${quote(e.payload.command, 60)} prefix=${truncate(e.prefix, 30)}`;\n case \"effect.file.touched\":\n return `${e.mode} ${e.path} (${e.bytes}B)`;\n case \"effect.memory.written\":\n return `${e.scope}:${e.key}`;\n case \"plan.submitted\":\n return `${e.steps.length} step(s)`;\n case \"plan.step.completed\":\n return `${e.stepId}${e.title ? ` — ${e.title}` : \"\"}`;\n case \"checkpoint.created\":\n return `${e.checkpointId} \"${e.name}\" ${e.fileCount} file(s) ${e.bytes}B`;\n case \"checkpoint.restored\":\n return `${e.checkpointId} restored=${e.restored} removed=${e.removed} skipped=${e.skipped}`;\n case \"hook.fired\":\n return `${e.phase} ${e.hookName} → ${e.outcome}`;\n case \"policy.budget.warning\":\n return `$${e.spentUsd.toFixed(4)} / $${e.capUsd.toFixed(2)}`;\n case \"policy.budget.blocked\":\n return `$${e.spentUsd.toFixed(4)} / $${e.capUsd.toFixed(2)} BLOCKED`;\n case \"policy.escalated\":\n return `${e.fromModel} → ${e.toModel} (${e.reason})${e.rationale ? ` \"${e.rationale}\"` : \"\"}`;\n case \"session.opened\":\n return `${e.name} resumed-from-turn=${e.resumedFromTurn}`;\n case \"session.compacted\":\n return `${e.beforeMessages} → ${e.afterMessages} msgs (${e.reason})`;\n case \"capability.registered\":\n return `${e.name} ${e.permission}`;\n case \"capability.removed\":\n return e.name;\n case \"status\":\n return quote(e.text, 80);\n case \"error\":\n return `${e.recoverable ? \"[recoverable] \" : \"\"}${quote(e.message, 80)}`;\n default:\n return \"\";\n }\n}\n\nfunction quote(s: string, max: number): string {\n const flat = s.replace(/\\s+/g, \" \").trim();\n return flat.length <= max ? `\"${flat}\"` : `\"${flat.slice(0, max - 1)}…\"`;\n}\n\nfunction truncate(s: string, max: number): string {\n return s.length <= max ? s : `${s.slice(0, max - 1)}…`;\n}\n\n/** WorkspaceView holds files in a Map; default JSON.stringify drops it. */\nfunction mapReplacer(_key: string, value: unknown): unknown {\n if (value instanceof Map) return Object.fromEntries(value);\n if (value instanceof Set) return [...value];\n return value;\n}\n"],"mappings":"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","names":[]}