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.
- package/README.md +267 -0
- package/README.zh-CN.md +237 -0
- package/dashboard/app.css +3022 -0
- package/dashboard/dist/app.js +30137 -0
- package/dashboard/dist/app.js.map +1 -0
- package/dashboard/dist/vendor-hljs.css +10 -0
- package/dashboard/dist/vendor-uplot.css +1 -0
- package/dashboard/index.html +19 -0
- package/data/deepseek-tokenizer.json.gz +0 -0
- package/dist/cli/acp-EOOAI4F5.js +712 -0
- package/dist/cli/acp-EOOAI4F5.js.map +1 -0
- package/dist/cli/chat-7J6GJXL2.js +51 -0
- package/dist/cli/chat-7J6GJXL2.js.map +1 -0
- package/dist/cli/chunk-2425HK6U.js +54 -0
- package/dist/cli/chunk-2425HK6U.js.map +1 -0
- package/dist/cli/chunk-25T6CVUP.js +172 -0
- package/dist/cli/chunk-25T6CVUP.js.map +1 -0
- package/dist/cli/chunk-2UQP6H6T.js +31 -0
- package/dist/cli/chunk-2UQP6H6T.js.map +1 -0
- package/dist/cli/chunk-56OAJILV.js +47 -0
- package/dist/cli/chunk-56OAJILV.js.map +1 -0
- package/dist/cli/chunk-5FTI4KXH.js +150 -0
- package/dist/cli/chunk-5FTI4KXH.js.map +1 -0
- package/dist/cli/chunk-5TWQD73O.js +2846 -0
- package/dist/cli/chunk-5TWQD73O.js.map +1 -0
- package/dist/cli/chunk-653BOCMK.js +40 -0
- package/dist/cli/chunk-653BOCMK.js.map +1 -0
- package/dist/cli/chunk-6ALJTWWQ.js +2663 -0
- package/dist/cli/chunk-6ALJTWWQ.js.map +1 -0
- package/dist/cli/chunk-6DRKA2IL.js +341 -0
- package/dist/cli/chunk-6DRKA2IL.js.map +1 -0
- package/dist/cli/chunk-6LV63NJV.js +634 -0
- package/dist/cli/chunk-6LV63NJV.js.map +1 -0
- package/dist/cli/chunk-74EX7SUH.js +25293 -0
- package/dist/cli/chunk-74EX7SUH.js.map +1 -0
- package/dist/cli/chunk-74U5RKTX.js +60611 -0
- package/dist/cli/chunk-74U5RKTX.js.map +1 -0
- package/dist/cli/chunk-ANJSUESV.js +143 -0
- package/dist/cli/chunk-ANJSUESV.js.map +1 -0
- package/dist/cli/chunk-DB2Z3DKZ.js +54 -0
- package/dist/cli/chunk-DB2Z3DKZ.js.map +1 -0
- package/dist/cli/chunk-DDIH3ZAA.js +400 -0
- package/dist/cli/chunk-DDIH3ZAA.js.map +1 -0
- package/dist/cli/chunk-ELN3Z3B2.js +621 -0
- package/dist/cli/chunk-ELN3Z3B2.js.map +1 -0
- package/dist/cli/chunk-F6BSQJGV.js +200 -0
- package/dist/cli/chunk-F6BSQJGV.js.map +1 -0
- package/dist/cli/chunk-FET2UAG5.js +246 -0
- package/dist/cli/chunk-FET2UAG5.js.map +1 -0
- package/dist/cli/chunk-FFJ342IJ.js +190 -0
- package/dist/cli/chunk-FFJ342IJ.js.map +1 -0
- package/dist/cli/chunk-GB3247B6.js +130 -0
- package/dist/cli/chunk-GB3247B6.js.map +1 -0
- package/dist/cli/chunk-HC2J4U3G.js +373 -0
- package/dist/cli/chunk-HC2J4U3G.js.map +1 -0
- package/dist/cli/chunk-HRUZAIHQ.js +42 -0
- package/dist/cli/chunk-HRUZAIHQ.js.map +1 -0
- package/dist/cli/chunk-J3ZJFUDL.js +308 -0
- package/dist/cli/chunk-J3ZJFUDL.js.map +1 -0
- package/dist/cli/chunk-J5XJHLWM.js +55 -0
- package/dist/cli/chunk-J5XJHLWM.js.map +1 -0
- package/dist/cli/chunk-JFGLMRZ6.js +160 -0
- package/dist/cli/chunk-JFGLMRZ6.js.map +1 -0
- package/dist/cli/chunk-JMBMLOBP.js +26 -0
- package/dist/cli/chunk-JMBMLOBP.js.map +1 -0
- package/dist/cli/chunk-JMWHXZEL.js +551 -0
- package/dist/cli/chunk-JMWHXZEL.js.map +1 -0
- package/dist/cli/chunk-KEQGPJBO.js +209 -0
- package/dist/cli/chunk-KEQGPJBO.js.map +1 -0
- package/dist/cli/chunk-M4K6U37F.js +232 -0
- package/dist/cli/chunk-M4K6U37F.js.map +1 -0
- package/dist/cli/chunk-MIJI2WMN.js +95 -0
- package/dist/cli/chunk-MIJI2WMN.js.map +1 -0
- package/dist/cli/chunk-MPAO3JNR.js +128 -0
- package/dist/cli/chunk-MPAO3JNR.js.map +1 -0
- package/dist/cli/chunk-PZOFBEDC.js +873 -0
- package/dist/cli/chunk-PZOFBEDC.js.map +1 -0
- package/dist/cli/chunk-RAILYQLN.js +46 -0
- package/dist/cli/chunk-RAILYQLN.js.map +1 -0
- package/dist/cli/chunk-RR35VQVT.js +90 -0
- package/dist/cli/chunk-RR35VQVT.js.map +1 -0
- package/dist/cli/chunk-RRA7VPW4.js +417 -0
- package/dist/cli/chunk-RRA7VPW4.js.map +1 -0
- package/dist/cli/chunk-RU36QVN3.js +452 -0
- package/dist/cli/chunk-RU36QVN3.js.map +1 -0
- package/dist/cli/chunk-RUBIINXR.js +1819 -0
- package/dist/cli/chunk-RUBIINXR.js.map +1 -0
- package/dist/cli/chunk-S4XVGLRW.js +499 -0
- package/dist/cli/chunk-S4XVGLRW.js.map +1 -0
- package/dist/cli/chunk-TUK7OWJA.js +51 -0
- package/dist/cli/chunk-TUK7OWJA.js.map +1 -0
- package/dist/cli/chunk-VALDDV76.js +580 -0
- package/dist/cli/chunk-VALDDV76.js.map +1 -0
- package/dist/cli/chunk-WQOGPYGN.js +11390 -0
- package/dist/cli/chunk-WQOGPYGN.js.map +1 -0
- package/dist/cli/chunk-WREKDFXT.js +34320 -0
- package/dist/cli/chunk-WREKDFXT.js.map +1 -0
- package/dist/cli/chunk-Y7XQU2EL.js +270 -0
- package/dist/cli/chunk-Y7XQU2EL.js.map +1 -0
- package/dist/cli/chunk-YBVCZJU4.js +54 -0
- package/dist/cli/chunk-YBVCZJU4.js.map +1 -0
- package/dist/cli/chunk-YLIHDXUQ.js +749 -0
- package/dist/cli/chunk-YLIHDXUQ.js.map +1 -0
- package/dist/cli/chunk-YV5XXFD7.js +767 -0
- package/dist/cli/chunk-YV5XXFD7.js.map +1 -0
- package/dist/cli/chunk-ZRCNIYRQ.js +101 -0
- package/dist/cli/chunk-ZRCNIYRQ.js.map +1 -0
- package/dist/cli/code-CRKVCMFZ.js +155 -0
- package/dist/cli/code-CRKVCMFZ.js.map +1 -0
- package/dist/cli/commands-QLMD3T7B.js +356 -0
- package/dist/cli/commands-QLMD3T7B.js.map +1 -0
- package/dist/cli/commit-53PP32NC.js +293 -0
- package/dist/cli/commit-53PP32NC.js.map +1 -0
- package/dist/cli/desktop-R6W5CLJ5.js +1046 -0
- package/dist/cli/desktop-R6W5CLJ5.js.map +1 -0
- package/dist/cli/devtools-YECO25QO.js +3719 -0
- package/dist/cli/devtools-YECO25QO.js.map +1 -0
- package/dist/cli/diff-LYNRCJZE.js +166 -0
- package/dist/cli/diff-LYNRCJZE.js.map +1 -0
- package/dist/cli/doctor-5IBP4R5J.js +28 -0
- package/dist/cli/doctor-5IBP4R5J.js.map +1 -0
- package/dist/cli/events-QN6KLN2V.js +340 -0
- package/dist/cli/events-QN6KLN2V.js.map +1 -0
- package/dist/cli/index.js +3500 -0
- package/dist/cli/index.js.map +1 -0
- package/dist/cli/mcp-FGKEH7RG.js +277 -0
- package/dist/cli/mcp-FGKEH7RG.js.map +1 -0
- package/dist/cli/mcp-browse-YCND4NWT.js +178 -0
- package/dist/cli/mcp-browse-YCND4NWT.js.map +1 -0
- package/dist/cli/mcp-inspect-V34J3VX5.js +143 -0
- package/dist/cli/mcp-inspect-V34J3VX5.js.map +1 -0
- package/dist/cli/package.json +3 -0
- package/dist/cli/prompt-I775PNKT.js +16 -0
- package/dist/cli/prompt-I775PNKT.js.map +1 -0
- package/dist/cli/prune-sessions-KGIIYD3P.js +44 -0
- package/dist/cli/prune-sessions-KGIIYD3P.js.map +1 -0
- package/dist/cli/replay-RDXLUAOE.js +292 -0
- package/dist/cli/replay-RDXLUAOE.js.map +1 -0
- package/dist/cli/run-RCAC2RYW.js +223 -0
- package/dist/cli/run-RCAC2RYW.js.map +1 -0
- package/dist/cli/server-FFU6TLYJ.js +3658 -0
- package/dist/cli/server-FFU6TLYJ.js.map +1 -0
- package/dist/cli/sessions-QT26MQAE.js +107 -0
- package/dist/cli/sessions-QT26MQAE.js.map +1 -0
- package/dist/cli/setup-VV4WKXHV.js +767 -0
- package/dist/cli/setup-VV4WKXHV.js.map +1 -0
- package/dist/cli/stats-JVZPQWAN.js +15 -0
- package/dist/cli/stats-JVZPQWAN.js.map +1 -0
- package/dist/cli/update-KYI3OVJP.js +15 -0
- package/dist/cli/update-KYI3OVJP.js.map +1 -0
- package/dist/cli/version-ANYORXTI.js +34 -0
- package/dist/cli/version-ANYORXTI.js.map +1 -0
- package/dist/index.d.ts +2557 -0
- package/dist/index.js +15000 -0
- package/dist/index.js.map +1 -0
- 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":[]}
|