@code-yeongyu/senpi 2026.5.23 → 2026.5.24

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 (104) hide show
  1. package/CHANGELOG.md +26 -0
  2. package/dist/core/extensions/builtin/history-search/filter.d.ts +3 -0
  3. package/dist/core/extensions/builtin/history-search/filter.d.ts.map +1 -0
  4. package/dist/core/extensions/builtin/history-search/filter.js +22 -0
  5. package/dist/core/extensions/builtin/history-search/filter.js.map +1 -0
  6. package/dist/core/extensions/builtin/history-search/index.d.ts +7 -0
  7. package/dist/core/extensions/builtin/history-search/index.d.ts.map +1 -0
  8. package/dist/core/extensions/builtin/history-search/index.js +45 -0
  9. package/dist/core/extensions/builtin/history-search/index.js.map +1 -0
  10. package/dist/core/extensions/builtin/history-search/indexer.d.ts +3 -0
  11. package/dist/core/extensions/builtin/history-search/indexer.d.ts.map +1 -0
  12. package/dist/core/extensions/builtin/history-search/indexer.js +161 -0
  13. package/dist/core/extensions/builtin/history-search/indexer.js.map +1 -0
  14. package/dist/core/extensions/builtin/history-search/overlay.d.ts +30 -0
  15. package/dist/core/extensions/builtin/history-search/overlay.d.ts.map +1 -0
  16. package/dist/core/extensions/builtin/history-search/overlay.js +115 -0
  17. package/dist/core/extensions/builtin/history-search/overlay.js.map +1 -0
  18. package/dist/core/extensions/builtin/history-search/types.d.ts +8 -0
  19. package/dist/core/extensions/builtin/history-search/types.d.ts.map +1 -0
  20. package/dist/core/extensions/builtin/history-search/types.js +2 -0
  21. package/dist/core/extensions/builtin/history-search/types.js.map +1 -0
  22. package/dist/core/extensions/builtin/index.d.ts.map +1 -1
  23. package/dist/core/extensions/builtin/index.js +4 -0
  24. package/dist/core/extensions/builtin/index.js.map +1 -1
  25. package/dist/core/extensions/builtin/session-observer/index.d.ts +5 -0
  26. package/dist/core/extensions/builtin/session-observer/index.d.ts.map +1 -0
  27. package/dist/core/extensions/builtin/session-observer/index.js +36 -0
  28. package/dist/core/extensions/builtin/session-observer/index.js.map +1 -0
  29. package/dist/core/extensions/builtin/session-observer/loader.d.ts +3 -0
  30. package/dist/core/extensions/builtin/session-observer/loader.d.ts.map +1 -0
  31. package/dist/core/extensions/builtin/session-observer/loader.js +20 -0
  32. package/dist/core/extensions/builtin/session-observer/loader.js.map +1 -0
  33. package/dist/core/extensions/builtin/session-observer/overlay-format.d.ts +7 -0
  34. package/dist/core/extensions/builtin/session-observer/overlay-format.d.ts.map +1 -0
  35. package/dist/core/extensions/builtin/session-observer/overlay-format.js +30 -0
  36. package/dist/core/extensions/builtin/session-observer/overlay-format.js.map +1 -0
  37. package/dist/core/extensions/builtin/session-observer/overlay.d.ts +51 -0
  38. package/dist/core/extensions/builtin/session-observer/overlay.d.ts.map +1 -0
  39. package/dist/core/extensions/builtin/session-observer/overlay.js +239 -0
  40. package/dist/core/extensions/builtin/session-observer/overlay.js.map +1 -0
  41. package/dist/core/extensions/builtin/session-observer/scanner.d.ts +10 -0
  42. package/dist/core/extensions/builtin/session-observer/scanner.d.ts.map +1 -0
  43. package/dist/core/extensions/builtin/session-observer/scanner.js +140 -0
  44. package/dist/core/extensions/builtin/session-observer/scanner.js.map +1 -0
  45. package/dist/core/extensions/builtin/session-observer/text.d.ts +7 -0
  46. package/dist/core/extensions/builtin/session-observer/text.d.ts.map +1 -0
  47. package/dist/core/extensions/builtin/session-observer/text.js +37 -0
  48. package/dist/core/extensions/builtin/session-observer/text.js.map +1 -0
  49. package/dist/core/extensions/builtin/session-observer/transcript-entries.d.ts +7 -0
  50. package/dist/core/extensions/builtin/session-observer/transcript-entries.d.ts.map +1 -0
  51. package/dist/core/extensions/builtin/session-observer/transcript-entries.js +71 -0
  52. package/dist/core/extensions/builtin/session-observer/transcript-entries.js.map +1 -0
  53. package/dist/core/extensions/builtin/session-observer/transcript-format.d.ts +11 -0
  54. package/dist/core/extensions/builtin/session-observer/transcript-format.d.ts.map +1 -0
  55. package/dist/core/extensions/builtin/session-observer/transcript-format.js +65 -0
  56. package/dist/core/extensions/builtin/session-observer/transcript-format.js.map +1 -0
  57. package/dist/core/extensions/builtin/session-observer/transcript.d.ts +4 -0
  58. package/dist/core/extensions/builtin/session-observer/transcript.d.ts.map +1 -0
  59. package/dist/core/extensions/builtin/session-observer/transcript.js +81 -0
  60. package/dist/core/extensions/builtin/session-observer/transcript.js.map +1 -0
  61. package/dist/core/extensions/builtin/session-observer/types.d.ts +33 -0
  62. package/dist/core/extensions/builtin/session-observer/types.d.ts.map +1 -0
  63. package/dist/core/extensions/builtin/session-observer/types.js +2 -0
  64. package/dist/core/extensions/builtin/session-observer/types.js.map +1 -0
  65. package/dist/core/extensions/runner.d.ts.map +1 -1
  66. package/dist/core/extensions/runner.js +1 -0
  67. package/dist/core/extensions/runner.js.map +1 -1
  68. package/dist/core/keybindings.d.ts +10 -0
  69. package/dist/core/keybindings.d.ts.map +1 -1
  70. package/dist/core/keybindings.js +3 -0
  71. package/dist/core/keybindings.js.map +1 -1
  72. package/dist/core/package-manager.d.ts.map +1 -1
  73. package/dist/core/package-manager.js +16 -4
  74. package/dist/core/package-manager.js.map +1 -1
  75. package/dist/modes/interactive/components/footer.d.ts.map +1 -1
  76. package/dist/modes/interactive/components/footer.js +74 -63
  77. package/dist/modes/interactive/components/footer.js.map +1 -1
  78. package/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  79. package/dist/modes/interactive/interactive-mode.js +18 -0
  80. package/dist/modes/interactive/interactive-mode.js.map +1 -1
  81. package/dist/utils/paths.d.ts +1 -0
  82. package/dist/utils/paths.d.ts.map +1 -1
  83. package/dist/utils/paths.js +8 -0
  84. package/dist/utils/paths.js.map +1 -1
  85. package/docs/terminal-setup.md +6 -0
  86. package/node_modules/@earendil-works/pi-agent-core/package.json +2 -2
  87. package/node_modules/@earendil-works/pi-ai/dist/models.generated.d.ts.map +1 -1
  88. package/node_modules/@earendil-works/pi-ai/dist/models.generated.js +1 -1
  89. package/node_modules/@earendil-works/pi-ai/dist/models.generated.js.map +1 -1
  90. package/node_modules/@earendil-works/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
  91. package/node_modules/@earendil-works/pi-ai/dist/providers/anthropic.js +53 -11
  92. package/node_modules/@earendil-works/pi-ai/dist/providers/anthropic.js.map +1 -1
  93. package/node_modules/@earendil-works/pi-ai/package.json +1 -1
  94. package/node_modules/@earendil-works/pi-tui/dist/native-modifiers.d.ts +3 -0
  95. package/node_modules/@earendil-works/pi-tui/dist/native-modifiers.d.ts.map +1 -0
  96. package/node_modules/@earendil-works/pi-tui/dist/native-modifiers.js +53 -0
  97. package/node_modules/@earendil-works/pi-tui/dist/native-modifiers.js.map +1 -0
  98. package/node_modules/@earendil-works/pi-tui/dist/terminal.d.ts +2 -0
  99. package/node_modules/@earendil-works/pi-tui/dist/terminal.d.ts.map +1 -1
  100. package/node_modules/@earendil-works/pi-tui/dist/terminal.js +13 -1
  101. package/node_modules/@earendil-works/pi-tui/dist/terminal.js.map +1 -1
  102. package/node_modules/@earendil-works/pi-tui/package.json +2 -2
  103. package/npm-shrinkwrap.json +12 -12
  104. package/package.json +4 -4
@@ -0,0 +1,140 @@
1
+ import { readdir, readFile, stat } from "node:fs/promises";
2
+ import { basename, isAbsolute, join, relative, resolve } from "node:path";
3
+ import { getSessionsDir } from "../../../../config.js";
4
+ import { parseSessionEntries } from "../../../session-manager.js";
5
+ import { compactWhitespace, getTextContent } from "./text.js";
6
+ export function resolveSessionHudRoot(currentSessionDir, defaultSessionsRoot = getSessionsDir(), pathImpl = { resolve, relative, isAbsolute }) {
7
+ const defaultRoot = pathImpl.resolve(defaultSessionsRoot);
8
+ if (!currentSessionDir)
9
+ return defaultRoot;
10
+ const current = pathImpl.resolve(currentSessionDir);
11
+ if (current === defaultRoot)
12
+ return defaultRoot;
13
+ const rel = pathImpl.relative(defaultRoot, current);
14
+ if (rel && !rel.startsWith("..") && !pathImpl.isAbsolute(rel))
15
+ return defaultRoot;
16
+ return current;
17
+ }
18
+ function hasErrorCode(error, code) {
19
+ return error instanceof Error && "code" in error && error.code === code;
20
+ }
21
+ async function readDirIfExists(path) {
22
+ try {
23
+ return await readdir(path);
24
+ }
25
+ catch (error) {
26
+ if (hasErrorCode(error, "ENOENT"))
27
+ return [];
28
+ throw error;
29
+ }
30
+ }
31
+ async function statIfFile(path) {
32
+ try {
33
+ const fileStat = await stat(path);
34
+ return fileStat.isFile() ? fileStat : undefined;
35
+ }
36
+ catch (error) {
37
+ if (hasErrorCode(error, "ENOENT"))
38
+ return undefined;
39
+ throw error;
40
+ }
41
+ }
42
+ async function collectFilesInDir(dir) {
43
+ const names = await readDirIfExists(dir);
44
+ const files = [];
45
+ for (const name of names) {
46
+ if (!name.endsWith(".jsonl"))
47
+ continue;
48
+ const file = join(dir, name);
49
+ if (await statIfFile(file))
50
+ files.push(file);
51
+ }
52
+ return files;
53
+ }
54
+ async function discoverSessionFiles(root) {
55
+ const names = await readDirIfExists(root);
56
+ const files = [...(await collectFilesInDir(root))];
57
+ for (const name of names) {
58
+ if (name.endsWith(".jsonl"))
59
+ continue;
60
+ const dir = join(root, name);
61
+ try {
62
+ const dirStat = await stat(dir);
63
+ if (dirStat.isDirectory())
64
+ files.push(...(await collectFilesInDir(dir)));
65
+ }
66
+ catch (error) {
67
+ if (!hasErrorCode(error, "ENOENT"))
68
+ throw error;
69
+ }
70
+ }
71
+ return files;
72
+ }
73
+ function firstHeader(entries, filePath) {
74
+ const header = entries[0];
75
+ if (header?.type === "session")
76
+ return header;
77
+ return { type: "session", id: basename(filePath, ".jsonl"), timestamp: new Date(0).toISOString(), cwd: "" };
78
+ }
79
+ function lastUserText(entries) {
80
+ for (let index = entries.length - 1; index >= 0; index -= 1) {
81
+ const entry = entries[index];
82
+ if (entry?.type !== "message")
83
+ continue;
84
+ const message = entry.message;
85
+ if (message.role !== "user")
86
+ continue;
87
+ const text = compactWhitespace(getTextContent(message.content));
88
+ if (text)
89
+ return text;
90
+ }
91
+ return "(no user prompt)";
92
+ }
93
+ function latestMessageTimestamp(entries, fallback) {
94
+ for (let index = entries.length - 1; index >= 0; index -= 1) {
95
+ const entry = entries[index];
96
+ if (!entry || entry.type === "session")
97
+ continue;
98
+ const timestamp = Date.parse(entry.timestamp);
99
+ if (Number.isFinite(timestamp))
100
+ return timestamp;
101
+ }
102
+ return fallback;
103
+ }
104
+ async function summarizeSession(filePath, currentSessionFile) {
105
+ const [content, fileStat] = await Promise.all([readFile(filePath, "utf-8"), stat(filePath)]);
106
+ const entries = parseSessionEntries(content);
107
+ const header = firstHeader(entries, filePath);
108
+ if (!header)
109
+ return undefined;
110
+ const messageCount = entries.filter((entry) => entry.type === "message").length;
111
+ return {
112
+ id: header.id,
113
+ shortId: header.id.length <= 8 ? header.id : header.id.slice(0, 8),
114
+ path: filePath,
115
+ cwd: header.cwd,
116
+ createdAt: Date.parse(header.timestamp),
117
+ modifiedAt: latestMessageTimestamp(entries, fileStat.mtime.getTime()),
118
+ messageCount,
119
+ lastUserText: lastUserText(entries),
120
+ isCurrent: currentSessionFile === filePath,
121
+ };
122
+ }
123
+ export async function scanSessionHudEntries(root, currentSessionFile) {
124
+ const files = await discoverSessionFiles(root);
125
+ const sessions = [];
126
+ for (const file of files) {
127
+ try {
128
+ const session = await summarizeSession(file, currentSessionFile);
129
+ if (session)
130
+ sessions.push(session);
131
+ }
132
+ catch (error) {
133
+ if (!hasErrorCode(error, "ENOENT"))
134
+ throw error;
135
+ }
136
+ }
137
+ sessions.sort((left, right) => right.modifiedAt - left.modifiedAt);
138
+ return sessions;
139
+ }
140
+ //# sourceMappingURL=scanner.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scanner.js","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/scanner.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Stats } from \"node:fs\";\nimport { readdir, readFile, stat } from \"node:fs/promises\";\nimport { basename, isAbsolute, join, relative, resolve } from \"node:path\";\nimport { getSessionsDir } from \"../../../../config.ts\";\nimport type { FileEntry, SessionHeader } from \"../../../session-manager.ts\";\nimport { parseSessionEntries } from \"../../../session-manager.ts\";\nimport { compactWhitespace, getTextContent } from \"./text.ts\";\nimport type { SessionHudEntry } from \"./types.ts\";\n\ninterface PathLike {\n\tresolve(path: string): string;\n\trelative(from: string, to: string): string;\n\tisAbsolute(path: string): boolean;\n}\n\nexport function resolveSessionHudRoot(\n\tcurrentSessionDir: string,\n\tdefaultSessionsRoot: string = getSessionsDir(),\n\tpathImpl: PathLike = { resolve, relative, isAbsolute },\n): string {\n\tconst defaultRoot = pathImpl.resolve(defaultSessionsRoot);\n\tif (!currentSessionDir) return defaultRoot;\n\tconst current = pathImpl.resolve(currentSessionDir);\n\tif (current === defaultRoot) return defaultRoot;\n\tconst rel = pathImpl.relative(defaultRoot, current);\n\tif (rel && !rel.startsWith(\"..\") && !pathImpl.isAbsolute(rel)) return defaultRoot;\n\treturn current;\n}\n\nfunction hasErrorCode(error: unknown, code: string): boolean {\n\treturn error instanceof Error && \"code\" in error && error.code === code;\n}\n\nasync function readDirIfExists(path: string): Promise<readonly string[]> {\n\ttry {\n\t\treturn await readdir(path);\n\t} catch (error) {\n\t\tif (hasErrorCode(error, \"ENOENT\")) return [];\n\t\tthrow error;\n\t}\n}\n\nasync function statIfFile(path: string): Promise<Stats | undefined> {\n\ttry {\n\t\tconst fileStat = await stat(path);\n\t\treturn fileStat.isFile() ? fileStat : undefined;\n\t} catch (error) {\n\t\tif (hasErrorCode(error, \"ENOENT\")) return undefined;\n\t\tthrow error;\n\t}\n}\n\nasync function collectFilesInDir(dir: string): Promise<readonly string[]> {\n\tconst names = await readDirIfExists(dir);\n\tconst files: string[] = [];\n\tfor (const name of names) {\n\t\tif (!name.endsWith(\".jsonl\")) continue;\n\t\tconst file = join(dir, name);\n\t\tif (await statIfFile(file)) files.push(file);\n\t}\n\treturn files;\n}\n\nasync function discoverSessionFiles(root: string): Promise<readonly string[]> {\n\tconst names = await readDirIfExists(root);\n\tconst files: string[] = [...(await collectFilesInDir(root))];\n\tfor (const name of names) {\n\t\tif (name.endsWith(\".jsonl\")) continue;\n\t\tconst dir = join(root, name);\n\t\ttry {\n\t\t\tconst dirStat = await stat(dir);\n\t\t\tif (dirStat.isDirectory()) files.push(...(await collectFilesInDir(dir)));\n\t\t} catch (error) {\n\t\t\tif (!hasErrorCode(error, \"ENOENT\")) throw error;\n\t\t}\n\t}\n\treturn files;\n}\n\nfunction firstHeader(entries: readonly FileEntry[], filePath: string): SessionHeader | undefined {\n\tconst header = entries[0];\n\tif (header?.type === \"session\") return header;\n\treturn { type: \"session\", id: basename(filePath, \".jsonl\"), timestamp: new Date(0).toISOString(), cwd: \"\" };\n}\n\nfunction lastUserText(entries: readonly FileEntry[]): string {\n\tfor (let index = entries.length - 1; index >= 0; index -= 1) {\n\t\tconst entry = entries[index];\n\t\tif (entry?.type !== \"message\") continue;\n\t\tconst message = entry.message;\n\t\tif (message.role !== \"user\") continue;\n\t\tconst text = compactWhitespace(getTextContent(message.content));\n\t\tif (text) return text;\n\t}\n\treturn \"(no user prompt)\";\n}\n\nfunction latestMessageTimestamp(entries: readonly FileEntry[], fallback: number): number {\n\tfor (let index = entries.length - 1; index >= 0; index -= 1) {\n\t\tconst entry = entries[index];\n\t\tif (!entry || entry.type === \"session\") continue;\n\t\tconst timestamp = Date.parse(entry.timestamp);\n\t\tif (Number.isFinite(timestamp)) return timestamp;\n\t}\n\treturn fallback;\n}\n\nasync function summarizeSession(\n\tfilePath: string,\n\tcurrentSessionFile: string | undefined,\n): Promise<SessionHudEntry | undefined> {\n\tconst [content, fileStat] = await Promise.all([readFile(filePath, \"utf-8\"), stat(filePath)]);\n\tconst entries = parseSessionEntries(content);\n\tconst header = firstHeader(entries, filePath);\n\tif (!header) return undefined;\n\tconst messageCount = entries.filter((entry) => entry.type === \"message\").length;\n\treturn {\n\t\tid: header.id,\n\t\tshortId: header.id.length <= 8 ? header.id : header.id.slice(0, 8),\n\t\tpath: filePath,\n\t\tcwd: header.cwd,\n\t\tcreatedAt: Date.parse(header.timestamp),\n\t\tmodifiedAt: latestMessageTimestamp(entries, fileStat.mtime.getTime()),\n\t\tmessageCount,\n\t\tlastUserText: lastUserText(entries),\n\t\tisCurrent: currentSessionFile === filePath,\n\t};\n}\n\nexport async function scanSessionHudEntries(\n\troot: string,\n\tcurrentSessionFile?: string,\n): Promise<readonly SessionHudEntry[]> {\n\tconst files = await discoverSessionFiles(root);\n\tconst sessions: SessionHudEntry[] = [];\n\tfor (const file of files) {\n\t\ttry {\n\t\t\tconst session = await summarizeSession(file, currentSessionFile);\n\t\t\tif (session) sessions.push(session);\n\t\t} catch (error) {\n\t\t\tif (!hasErrorCode(error, \"ENOENT\")) throw error;\n\t\t}\n\t}\n\tsessions.sort((left, right) => right.modifiedAt - left.modifiedAt);\n\treturn sessions;\n}\n"]}
@@ -0,0 +1,7 @@
1
+ import type { ImageContent, TextContent } from "@earendil-works/pi-ai";
2
+ export declare const INDENT = " ";
3
+ export declare function sanitizeLine(text: string, width: number): string;
4
+ export declare function compactWhitespace(text: string): string;
5
+ export declare function getTextContent(content: string | readonly (TextContent | ImageContent)[]): string;
6
+ export declare function formatSessionDate(date: Date): string;
7
+ //# sourceMappingURL=text.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"text.d.ts","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/text.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,YAAY,EAAE,WAAW,EAAE,MAAM,uBAAuB,CAAC;AAGvE,eAAO,MAAM,MAAM,SAAS,CAAC;AAE7B,wBAAgB,YAAY,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,MAAM,CAEhE;AAED,wBAAgB,iBAAiB,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAEtD;AAED,wBAAgB,cAAc,CAAC,OAAO,EAAE,MAAM,GAAG,SAAS,CAAC,WAAW,GAAG,YAAY,CAAC,EAAE,GAAG,MAAM,CAMhG;AAED,wBAAgB,iBAAiB,CAAC,IAAI,EAAE,IAAI,GAAG,MAAM,CAcpD","sourcesContent":["import type { ImageContent, TextContent } from \"@earendil-works/pi-ai\";\nimport { truncateToWidth } from \"@earendil-works/pi-tui\";\n\nexport const INDENT = \" \";\n\nexport function sanitizeLine(text: string, width: number): string {\n\treturn truncateToWidth(text.replace(/\\t/g, \" \").replace(/\\r/g, \"\"), Math.max(1, width), \"…\");\n}\n\nexport function compactWhitespace(text: string): string {\n\treturn text.replace(/[\\r\\n\\t ]+/g, \" \").trim();\n}\n\nexport function getTextContent(content: string | readonly (TextContent | ImageContent)[]): string {\n\tif (typeof content === \"string\") return content;\n\treturn content\n\t\t.filter((block): block is TextContent => block.type === \"text\")\n\t\t.map((block) => block.text)\n\t\t.join(\"\\n\");\n}\n\nexport function formatSessionDate(date: Date): string {\n\tconst now = new Date();\n\tconst diffMs = now.getTime() - date.getTime();\n\tconst diffMins = Math.floor(diffMs / 60_000);\n\tconst diffHours = Math.floor(diffMs / 3_600_000);\n\tconst diffDays = Math.floor(diffMs / 86_400_000);\n\n\tif (diffMins < 1) return \"now\";\n\tif (diffMins < 60) return `${diffMins}m`;\n\tif (diffHours < 24) return `${diffHours}h`;\n\tif (diffDays < 7) return `${diffDays}d`;\n\tif (diffDays < 30) return `${Math.floor(diffDays / 7)}w`;\n\tif (diffDays < 365) return `${Math.floor(diffDays / 30)}mo`;\n\treturn `${Math.floor(diffDays / 365)}y`;\n}\n"]}
@@ -0,0 +1,37 @@
1
+ import { truncateToWidth } from "@earendil-works/pi-tui";
2
+ export const INDENT = " ";
3
+ export function sanitizeLine(text, width) {
4
+ return truncateToWidth(text.replace(/\t/g, " ").replace(/\r/g, ""), Math.max(1, width), "…");
5
+ }
6
+ export function compactWhitespace(text) {
7
+ return text.replace(/[\r\n\t ]+/g, " ").trim();
8
+ }
9
+ export function getTextContent(content) {
10
+ if (typeof content === "string")
11
+ return content;
12
+ return content
13
+ .filter((block) => block.type === "text")
14
+ .map((block) => block.text)
15
+ .join("\n");
16
+ }
17
+ export function formatSessionDate(date) {
18
+ const now = new Date();
19
+ const diffMs = now.getTime() - date.getTime();
20
+ const diffMins = Math.floor(diffMs / 60_000);
21
+ const diffHours = Math.floor(diffMs / 3_600_000);
22
+ const diffDays = Math.floor(diffMs / 86_400_000);
23
+ if (diffMins < 1)
24
+ return "now";
25
+ if (diffMins < 60)
26
+ return `${diffMins}m`;
27
+ if (diffHours < 24)
28
+ return `${diffHours}h`;
29
+ if (diffDays < 7)
30
+ return `${diffDays}d`;
31
+ if (diffDays < 30)
32
+ return `${Math.floor(diffDays / 7)}w`;
33
+ if (diffDays < 365)
34
+ return `${Math.floor(diffDays / 30)}mo`;
35
+ return `${Math.floor(diffDays / 365)}y`;
36
+ }
37
+ //# sourceMappingURL=text.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"text.js","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/text.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AAEzD,MAAM,CAAC,MAAM,MAAM,GAAG,MAAM,CAAC;AAE7B,MAAM,UAAU,YAAY,CAAC,IAAY,EAAE,KAAa,EAAU;IACjE,OAAO,eAAe,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,KAAK,CAAC,EAAE,KAAG,CAAC,CAAC;AAAA,CAC/F;AAED,MAAM,UAAU,iBAAiB,CAAC,IAAY,EAAU;IACvD,OAAO,IAAI,CAAC,OAAO,CAAC,aAAa,EAAE,GAAG,CAAC,CAAC,IAAI,EAAE,CAAC;AAAA,CAC/C;AAED,MAAM,UAAU,cAAc,CAAC,OAAyD,EAAU;IACjG,IAAI,OAAO,OAAO,KAAK,QAAQ;QAAE,OAAO,OAAO,CAAC;IAChD,OAAO,OAAO;SACZ,MAAM,CAAC,CAAC,KAAK,EAAwB,EAAE,CAAC,KAAK,CAAC,IAAI,KAAK,MAAM,CAAC;SAC9D,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC;SAC1B,IAAI,CAAC,IAAI,CAAC,CAAC;AAAA,CACb;AAED,MAAM,UAAU,iBAAiB,CAAC,IAAU,EAAU;IACrD,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC;IACvB,MAAM,MAAM,GAAG,GAAG,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC,OAAO,EAAE,CAAC;IAC9C,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,MAAM,CAAC,CAAC;IAC7C,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,SAAS,CAAC,CAAC;IACjD,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,UAAU,CAAC,CAAC;IAEjD,IAAI,QAAQ,GAAG,CAAC;QAAE,OAAO,KAAK,CAAC;IAC/B,IAAI,QAAQ,GAAG,EAAE;QAAE,OAAO,GAAG,QAAQ,GAAG,CAAC;IACzC,IAAI,SAAS,GAAG,EAAE;QAAE,OAAO,GAAG,SAAS,GAAG,CAAC;IAC3C,IAAI,QAAQ,GAAG,CAAC;QAAE,OAAO,GAAG,QAAQ,GAAG,CAAC;IACxC,IAAI,QAAQ,GAAG,EAAE;QAAE,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,GAAG,CAAC,CAAC,GAAG,CAAC;IACzD,IAAI,QAAQ,GAAG,GAAG;QAAE,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,GAAG,EAAE,CAAC,IAAI,CAAC;IAC5D,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,GAAG,GAAG,CAAC,GAAG,CAAC;AAAA,CACxC","sourcesContent":["import type { ImageContent, TextContent } from \"@earendil-works/pi-ai\";\nimport { truncateToWidth } from \"@earendil-works/pi-tui\";\n\nexport const INDENT = \" \";\n\nexport function sanitizeLine(text: string, width: number): string {\n\treturn truncateToWidth(text.replace(/\\t/g, \" \").replace(/\\r/g, \"\"), Math.max(1, width), \"…\");\n}\n\nexport function compactWhitespace(text: string): string {\n\treturn text.replace(/[\\r\\n\\t ]+/g, \" \").trim();\n}\n\nexport function getTextContent(content: string | readonly (TextContent | ImageContent)[]): string {\n\tif (typeof content === \"string\") return content;\n\treturn content\n\t\t.filter((block): block is TextContent => block.type === \"text\")\n\t\t.map((block) => block.text)\n\t\t.join(\"\\n\");\n}\n\nexport function formatSessionDate(date: Date): string {\n\tconst now = new Date();\n\tconst diffMs = now.getTime() - date.getTime();\n\tconst diffMins = Math.floor(diffMs / 60_000);\n\tconst diffHours = Math.floor(diffMs / 3_600_000);\n\tconst diffDays = Math.floor(diffMs / 86_400_000);\n\n\tif (diffMins < 1) return \"now\";\n\tif (diffMins < 60) return `${diffMins}m`;\n\tif (diffHours < 24) return `${diffHours}h`;\n\tif (diffDays < 7) return `${diffDays}d`;\n\tif (diffDays < 30) return `${Math.floor(diffDays / 7)}w`;\n\tif (diffDays < 365) return `${Math.floor(diffDays / 30)}mo`;\n\treturn `${Math.floor(diffDays / 365)}y`;\n}\n"]}
@@ -0,0 +1,7 @@
1
+ import type { ToolCall, ToolResultMessage } from "@earendil-works/pi-ai";
2
+ import type { TranscriptRenderOptions } from "./types.ts";
3
+ export declare function renderThinkingEntry(lines: string[], text: string, expanded: boolean, selected: boolean, options: TranscriptRenderOptions): void;
4
+ export declare function renderTextEntry(lines: string[], label: string, text: string, expanded: boolean, selected: boolean, options: TranscriptRenderOptions): void;
5
+ export declare function renderUserEntry(lines: string[], label: string, text: string, expanded: boolean, selected: boolean, options: TranscriptRenderOptions): void;
6
+ export declare function renderToolEntry(lines: string[], call: ToolCall, result: ToolResultMessage<unknown> | undefined, expanded: boolean, selected: boolean, options: TranscriptRenderOptions): void;
7
+ //# sourceMappingURL=transcript-entries.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"transcript-entries.d.ts","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/transcript-entries.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,QAAQ,EAAE,iBAAiB,EAAE,MAAM,uBAAuB,CAAC;AAYzE,OAAO,KAAK,EAAE,uBAAuB,EAAE,MAAM,YAAY,CAAC;AAE1D,wBAAgB,mBAAmB,CAClC,KAAK,EAAE,MAAM,EAAE,EACf,IAAI,EAAE,MAAM,EACZ,QAAQ,EAAE,OAAO,EACjB,QAAQ,EAAE,OAAO,EACjB,OAAO,EAAE,uBAAuB,GAC9B,IAAI,CAiBN;AAED,wBAAgB,eAAe,CAC9B,KAAK,EAAE,MAAM,EAAE,EACf,KAAK,EAAE,MAAM,EACb,IAAI,EAAE,MAAM,EACZ,QAAQ,EAAE,OAAO,EACjB,QAAQ,EAAE,OAAO,EACjB,OAAO,EAAE,uBAAuB,GAC9B,IAAI,CAUN;AAED,wBAAgB,eAAe,CAC9B,KAAK,EAAE,MAAM,EAAE,EACf,KAAK,EAAE,MAAM,EACb,IAAI,EAAE,MAAM,EACZ,QAAQ,EAAE,OAAO,EACjB,QAAQ,EAAE,OAAO,EACjB,OAAO,EAAE,uBAAuB,GAC9B,IAAI,CAaN;AA4BD,wBAAgB,eAAe,CAC9B,KAAK,EAAE,MAAM,EAAE,EACf,IAAI,EAAE,QAAQ,EACd,MAAM,EAAE,iBAAiB,CAAC,OAAO,CAAC,GAAG,SAAS,EAC9C,QAAQ,EAAE,OAAO,EACjB,QAAQ,EAAE,OAAO,EACjB,OAAO,EAAE,uBAAuB,GAC9B,IAAI,CAON","sourcesContent":["import type { ToolCall, ToolResultMessage } from \"@earendil-works/pi-ai\";\nimport { getMarkdownTheme, theme } from \"../../../../modes/interactive/theme/theme.ts\";\nimport { compactWhitespace, INDENT, sanitizeLine } from \"./text.ts\";\nimport {\n\tcontentWidth,\n\tformatToolArgs,\n\tMAX_EXPANDED_LINES,\n\tMAX_THINKING_COLLAPSED,\n\trenderMarkdown,\n\trenderPreview,\n\ttoolResultText,\n} from \"./transcript-format.ts\";\nimport type { TranscriptRenderOptions } from \"./types.ts\";\n\nexport function renderThinkingEntry(\n\tlines: string[],\n\ttext: string,\n\texpanded: boolean,\n\tselected: boolean,\n\toptions: TranscriptRenderOptions,\n): void {\n\tconst cursor = selected ? theme.fg(\"accent\", \"▶\") : \" \";\n\tlines.push(\"\");\n\tlines.push(\n\t\t`${cursor} ${theme.fg(\"dim\", \"💭 Thinking\")}${!expanded && text.length > MAX_THINKING_COLLAPSED ? theme.fg(\"dim\", \" ↵\") : \"\"}`,\n\t);\n\tconst displayText =\n\t\texpanded || text.length <= MAX_THINKING_COLLAPSED ? text : `${text.slice(0, MAX_THINKING_COLLAPSED)}...`;\n\tif (!expanded) {\n\t\trenderPreview(lines, displayText, options.width, \"thinkingText\");\n\t\treturn;\n\t}\n\tconst rendered = renderMarkdown(displayText, options.width, options.markdownTheme ?? getMarkdownTheme());\n\tfor (const line of rendered.slice(0, MAX_EXPANDED_LINES)) lines.push(line);\n\tif (rendered.length > MAX_EXPANDED_LINES) {\n\t\tlines.push(`${INDENT}${theme.fg(\"dim\", `... ${rendered.length - MAX_EXPANDED_LINES} more lines`)}`);\n\t}\n}\n\nexport function renderTextEntry(\n\tlines: string[],\n\tlabel: string,\n\ttext: string,\n\texpanded: boolean,\n\tselected: boolean,\n\toptions: TranscriptRenderOptions,\n): void {\n\tconst cursor = selected ? theme.fg(\"accent\", \"▶\") : \" \";\n\tlines.push(\"\");\n\tlines.push(`${cursor} ${theme.fg(\"muted\", label)}`);\n\tif (!expanded) {\n\t\trenderPreview(lines, text, options.width, \"dim\");\n\t\treturn;\n\t}\n\tfor (const line of renderMarkdown(text, options.width, options.markdownTheme ?? getMarkdownTheme()))\n\t\tlines.push(line);\n}\n\nexport function renderUserEntry(\n\tlines: string[],\n\tlabel: string,\n\ttext: string,\n\texpanded: boolean,\n\tselected: boolean,\n\toptions: TranscriptRenderOptions,\n): void {\n\tconst cursor = selected ? theme.fg(\"accent\", \"▶\") : \" \";\n\tlines.push(\"\");\n\tif (expanded) {\n\t\tlines.push(`${cursor} ${theme.fg(\"dim\", `[${label}]`)}`);\n\t\tfor (const line of renderMarkdown(text, options.width, options.markdownTheme ?? getMarkdownTheme()))\n\t\t\tlines.push(line);\n\t\treturn;\n\t}\n\tconst normalized = compactWhitespace(text);\n\tlines.push(\n\t\t`${cursor} ${theme.fg(\"dim\", `[${label}]`)} ${theme.fg(\"muted\", sanitizeLine(normalized, contentWidth(options.width)))}`,\n\t);\n}\n\nfunction renderToolResult(\n\tlines: string[],\n\tresult: ToolResultMessage<unknown> | undefined,\n\texpanded: boolean,\n\toptions: TranscriptRenderOptions,\n): void {\n\tif (!result) return;\n\tconst text = toolResultText(result);\n\tif (!text) {\n\t\tlines.push(`${INDENT}${theme.fg(\"success\", \"✓ done\")}`);\n\t\treturn;\n\t}\n\tconst resultLines = text.split(\"\\n\");\n\tconst maxLines = expanded ? 20 : 3;\n\tconst color = result.isError ? \"error\" : \"dim\";\n\tconst marker = result.isError ? \"✗\" : \"✓\";\n\tlines.push(\n\t\t`${INDENT}${theme.fg(result.isError ? \"error\" : \"success\", marker)} ${theme.fg(color, sanitizeLine(resultLines[0] ?? \"\", contentWidth(options.width)))}`,\n\t);\n\tfor (const line of resultLines.slice(1, maxLines)) {\n\t\tlines.push(`${INDENT} ${theme.fg(color, sanitizeLine(line, contentWidth(options.width)))}`);\n\t}\n\tif (resultLines.length > maxLines)\n\t\tlines.push(`${INDENT} ${theme.fg(\"dim\", `... ${resultLines.length - maxLines} more`)}`);\n}\n\nexport function renderToolEntry(\n\tlines: string[],\n\tcall: ToolCall,\n\tresult: ToolResultMessage<unknown> | undefined,\n\texpanded: boolean,\n\tselected: boolean,\n\toptions: TranscriptRenderOptions,\n): void {\n\tconst cursor = selected ? theme.fg(\"accent\", \"▶\") : \" \";\n\tlines.push(\"\");\n\tlines.push(`${cursor} ${theme.fg(\"accent\", \"▸\")} ${theme.bold(theme.fg(\"muted\", call.name))}`);\n\tconst args = formatToolArgs(call);\n\tif (args) lines.push(`${INDENT}${theme.fg(\"dim\", sanitizeLine(args, contentWidth(options.width)))}`);\n\trenderToolResult(lines, result, expanded, options);\n}\n"]}
@@ -0,0 +1,71 @@
1
+ import { getMarkdownTheme, theme } from "../../../../modes/interactive/theme/theme.js";
2
+ import { compactWhitespace, INDENT, sanitizeLine } from "./text.js";
3
+ import { contentWidth, formatToolArgs, MAX_EXPANDED_LINES, MAX_THINKING_COLLAPSED, renderMarkdown, renderPreview, toolResultText, } from "./transcript-format.js";
4
+ export function renderThinkingEntry(lines, text, expanded, selected, options) {
5
+ const cursor = selected ? theme.fg("accent", "▶") : " ";
6
+ lines.push("");
7
+ lines.push(`${cursor} ${theme.fg("dim", "💭 Thinking")}${!expanded && text.length > MAX_THINKING_COLLAPSED ? theme.fg("dim", " ↵") : ""}`);
8
+ const displayText = expanded || text.length <= MAX_THINKING_COLLAPSED ? text : `${text.slice(0, MAX_THINKING_COLLAPSED)}...`;
9
+ if (!expanded) {
10
+ renderPreview(lines, displayText, options.width, "thinkingText");
11
+ return;
12
+ }
13
+ const rendered = renderMarkdown(displayText, options.width, options.markdownTheme ?? getMarkdownTheme());
14
+ for (const line of rendered.slice(0, MAX_EXPANDED_LINES))
15
+ lines.push(line);
16
+ if (rendered.length > MAX_EXPANDED_LINES) {
17
+ lines.push(`${INDENT}${theme.fg("dim", `... ${rendered.length - MAX_EXPANDED_LINES} more lines`)}`);
18
+ }
19
+ }
20
+ export function renderTextEntry(lines, label, text, expanded, selected, options) {
21
+ const cursor = selected ? theme.fg("accent", "▶") : " ";
22
+ lines.push("");
23
+ lines.push(`${cursor} ${theme.fg("muted", label)}`);
24
+ if (!expanded) {
25
+ renderPreview(lines, text, options.width, "dim");
26
+ return;
27
+ }
28
+ for (const line of renderMarkdown(text, options.width, options.markdownTheme ?? getMarkdownTheme()))
29
+ lines.push(line);
30
+ }
31
+ export function renderUserEntry(lines, label, text, expanded, selected, options) {
32
+ const cursor = selected ? theme.fg("accent", "▶") : " ";
33
+ lines.push("");
34
+ if (expanded) {
35
+ lines.push(`${cursor} ${theme.fg("dim", `[${label}]`)}`);
36
+ for (const line of renderMarkdown(text, options.width, options.markdownTheme ?? getMarkdownTheme()))
37
+ lines.push(line);
38
+ return;
39
+ }
40
+ const normalized = compactWhitespace(text);
41
+ lines.push(`${cursor} ${theme.fg("dim", `[${label}]`)} ${theme.fg("muted", sanitizeLine(normalized, contentWidth(options.width)))}`);
42
+ }
43
+ function renderToolResult(lines, result, expanded, options) {
44
+ if (!result)
45
+ return;
46
+ const text = toolResultText(result);
47
+ if (!text) {
48
+ lines.push(`${INDENT}${theme.fg("success", "✓ done")}`);
49
+ return;
50
+ }
51
+ const resultLines = text.split("\n");
52
+ const maxLines = expanded ? 20 : 3;
53
+ const color = result.isError ? "error" : "dim";
54
+ const marker = result.isError ? "✗" : "✓";
55
+ lines.push(`${INDENT}${theme.fg(result.isError ? "error" : "success", marker)} ${theme.fg(color, sanitizeLine(resultLines[0] ?? "", contentWidth(options.width)))}`);
56
+ for (const line of resultLines.slice(1, maxLines)) {
57
+ lines.push(`${INDENT} ${theme.fg(color, sanitizeLine(line, contentWidth(options.width)))}`);
58
+ }
59
+ if (resultLines.length > maxLines)
60
+ lines.push(`${INDENT} ${theme.fg("dim", `... ${resultLines.length - maxLines} more`)}`);
61
+ }
62
+ export function renderToolEntry(lines, call, result, expanded, selected, options) {
63
+ const cursor = selected ? theme.fg("accent", "▶") : " ";
64
+ lines.push("");
65
+ lines.push(`${cursor} ${theme.fg("accent", "▸")} ${theme.bold(theme.fg("muted", call.name))}`);
66
+ const args = formatToolArgs(call);
67
+ if (args)
68
+ lines.push(`${INDENT}${theme.fg("dim", sanitizeLine(args, contentWidth(options.width)))}`);
69
+ renderToolResult(lines, result, expanded, options);
70
+ }
71
+ //# sourceMappingURL=transcript-entries.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"transcript-entries.js","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/transcript-entries.ts"],"names":[],"mappings":"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","sourcesContent":["import type { ToolCall, ToolResultMessage } from \"@earendil-works/pi-ai\";\nimport { getMarkdownTheme, theme } from \"../../../../modes/interactive/theme/theme.ts\";\nimport { compactWhitespace, INDENT, sanitizeLine } from \"./text.ts\";\nimport {\n\tcontentWidth,\n\tformatToolArgs,\n\tMAX_EXPANDED_LINES,\n\tMAX_THINKING_COLLAPSED,\n\trenderMarkdown,\n\trenderPreview,\n\ttoolResultText,\n} from \"./transcript-format.ts\";\nimport type { TranscriptRenderOptions } from \"./types.ts\";\n\nexport function renderThinkingEntry(\n\tlines: string[],\n\ttext: string,\n\texpanded: boolean,\n\tselected: boolean,\n\toptions: TranscriptRenderOptions,\n): void {\n\tconst cursor = selected ? theme.fg(\"accent\", \"▶\") : \" \";\n\tlines.push(\"\");\n\tlines.push(\n\t\t`${cursor} ${theme.fg(\"dim\", \"💭 Thinking\")}${!expanded && text.length > MAX_THINKING_COLLAPSED ? theme.fg(\"dim\", \" ↵\") : \"\"}`,\n\t);\n\tconst displayText =\n\t\texpanded || text.length <= MAX_THINKING_COLLAPSED ? text : `${text.slice(0, MAX_THINKING_COLLAPSED)}...`;\n\tif (!expanded) {\n\t\trenderPreview(lines, displayText, options.width, \"thinkingText\");\n\t\treturn;\n\t}\n\tconst rendered = renderMarkdown(displayText, options.width, options.markdownTheme ?? getMarkdownTheme());\n\tfor (const line of rendered.slice(0, MAX_EXPANDED_LINES)) lines.push(line);\n\tif (rendered.length > MAX_EXPANDED_LINES) {\n\t\tlines.push(`${INDENT}${theme.fg(\"dim\", `... ${rendered.length - MAX_EXPANDED_LINES} more lines`)}`);\n\t}\n}\n\nexport function renderTextEntry(\n\tlines: string[],\n\tlabel: string,\n\ttext: string,\n\texpanded: boolean,\n\tselected: boolean,\n\toptions: TranscriptRenderOptions,\n): void {\n\tconst cursor = selected ? theme.fg(\"accent\", \"▶\") : \" \";\n\tlines.push(\"\");\n\tlines.push(`${cursor} ${theme.fg(\"muted\", label)}`);\n\tif (!expanded) {\n\t\trenderPreview(lines, text, options.width, \"dim\");\n\t\treturn;\n\t}\n\tfor (const line of renderMarkdown(text, options.width, options.markdownTheme ?? getMarkdownTheme()))\n\t\tlines.push(line);\n}\n\nexport function renderUserEntry(\n\tlines: string[],\n\tlabel: string,\n\ttext: string,\n\texpanded: boolean,\n\tselected: boolean,\n\toptions: TranscriptRenderOptions,\n): void {\n\tconst cursor = selected ? theme.fg(\"accent\", \"▶\") : \" \";\n\tlines.push(\"\");\n\tif (expanded) {\n\t\tlines.push(`${cursor} ${theme.fg(\"dim\", `[${label}]`)}`);\n\t\tfor (const line of renderMarkdown(text, options.width, options.markdownTheme ?? getMarkdownTheme()))\n\t\t\tlines.push(line);\n\t\treturn;\n\t}\n\tconst normalized = compactWhitespace(text);\n\tlines.push(\n\t\t`${cursor} ${theme.fg(\"dim\", `[${label}]`)} ${theme.fg(\"muted\", sanitizeLine(normalized, contentWidth(options.width)))}`,\n\t);\n}\n\nfunction renderToolResult(\n\tlines: string[],\n\tresult: ToolResultMessage<unknown> | undefined,\n\texpanded: boolean,\n\toptions: TranscriptRenderOptions,\n): void {\n\tif (!result) return;\n\tconst text = toolResultText(result);\n\tif (!text) {\n\t\tlines.push(`${INDENT}${theme.fg(\"success\", \"✓ done\")}`);\n\t\treturn;\n\t}\n\tconst resultLines = text.split(\"\\n\");\n\tconst maxLines = expanded ? 20 : 3;\n\tconst color = result.isError ? \"error\" : \"dim\";\n\tconst marker = result.isError ? \"✗\" : \"✓\";\n\tlines.push(\n\t\t`${INDENT}${theme.fg(result.isError ? \"error\" : \"success\", marker)} ${theme.fg(color, sanitizeLine(resultLines[0] ?? \"\", contentWidth(options.width)))}`,\n\t);\n\tfor (const line of resultLines.slice(1, maxLines)) {\n\t\tlines.push(`${INDENT} ${theme.fg(color, sanitizeLine(line, contentWidth(options.width)))}`);\n\t}\n\tif (resultLines.length > maxLines)\n\t\tlines.push(`${INDENT} ${theme.fg(\"dim\", `... ${resultLines.length - maxLines} more`)}`);\n}\n\nexport function renderToolEntry(\n\tlines: string[],\n\tcall: ToolCall,\n\tresult: ToolResultMessage<unknown> | undefined,\n\texpanded: boolean,\n\tselected: boolean,\n\toptions: TranscriptRenderOptions,\n): void {\n\tconst cursor = selected ? theme.fg(\"accent\", \"▶\") : \" \";\n\tlines.push(\"\");\n\tlines.push(`${cursor} ${theme.fg(\"accent\", \"▸\")} ${theme.bold(theme.fg(\"muted\", call.name))}`);\n\tconst args = formatToolArgs(call);\n\tif (args) lines.push(`${INDENT}${theme.fg(\"dim\", sanitizeLine(args, contentWidth(options.width)))}`);\n\trenderToolResult(lines, result, expanded, options);\n}\n"]}
@@ -0,0 +1,11 @@
1
+ import type { ToolCall, ToolResultMessage } from "@earendil-works/pi-ai";
2
+ import { type MarkdownTheme } from "@earendil-works/pi-tui";
3
+ export declare const MAX_COLLAPSED_LINES = 3;
4
+ export declare const MAX_EXPANDED_LINES = 100;
5
+ export declare const MAX_THINKING_COLLAPSED = 200;
6
+ export declare function contentWidth(width: number): number;
7
+ export declare function renderMarkdown(text: string, width: number, markdownTheme: MarkdownTheme): readonly string[];
8
+ export declare function toolResultText(result: ToolResultMessage<unknown>): string;
9
+ export declare function formatToolArgs(call: ToolCall): string;
10
+ export declare function renderPreview(lines: string[], text: string, width: number, color: "dim" | "thinkingText"): void;
11
+ //# sourceMappingURL=transcript-format.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"transcript-format.d.ts","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/transcript-format.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,QAAQ,EAAE,iBAAiB,EAAE,MAAM,uBAAuB,CAAC;AACzE,OAAO,EAAY,KAAK,aAAa,EAAE,MAAM,wBAAwB,CAAC;AAItE,eAAO,MAAM,mBAAmB,IAAI,CAAC;AACrC,eAAO,MAAM,kBAAkB,MAAM,CAAC;AACtC,eAAO,MAAM,sBAAsB,MAAM,CAAC;AAI1C,wBAAgB,YAAY,CAAC,KAAK,EAAE,MAAM,GAAG,MAAM,CAElD;AAED,wBAAgB,cAAc,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,aAAa,EAAE,aAAa,GAAG,SAAS,MAAM,EAAE,CAG3G;AAED,wBAAgB,cAAc,CAAC,MAAM,EAAE,iBAAiB,CAAC,OAAO,CAAC,GAAG,MAAM,CAMzE;AAWD,wBAAgB,cAAc,CAAC,IAAI,EAAE,QAAQ,GAAG,MAAM,CAmBrD;AAED,wBAAgB,aAAa,CAAC,KAAK,EAAE,MAAM,EAAE,EAAE,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,KAAK,EAAE,KAAK,GAAG,cAAc,GAAG,IAAI,CAS/G","sourcesContent":["import type { ToolCall, ToolResultMessage } from \"@earendil-works/pi-ai\";\nimport { Markdown, type MarkdownTheme } from \"@earendil-works/pi-tui\";\nimport { theme } from \"../../../../modes/interactive/theme/theme.ts\";\nimport { INDENT, sanitizeLine } from \"./text.ts\";\n\nexport const MAX_COLLAPSED_LINES = 3;\nexport const MAX_EXPANDED_LINES = 100;\nexport const MAX_THINKING_COLLAPSED = 200;\n\nconst MAX_TOOL_ARGS_CHARS = 500;\n\nexport function contentWidth(width: number): number {\n\treturn Math.max(20, width - INDENT.length - 4);\n}\n\nexport function renderMarkdown(text: string, width: number, markdownTheme: MarkdownTheme): readonly string[] {\n\tconst markdown = new Markdown(text, 0, 0, markdownTheme);\n\treturn markdown.render(Math.max(40, width - INDENT.length - 4)).map((line) => `${INDENT}${line.trimEnd()}`);\n}\n\nexport function toolResultText(result: ToolResultMessage<unknown>): string {\n\treturn result.content\n\t\t.filter((part) => part.type === \"text\")\n\t\t.map((part) => part.text)\n\t\t.join(\"\\n\")\n\t\t.trim();\n}\n\nfunction safeJson(value: unknown): string {\n\ttry {\n\t\treturn JSON.stringify(value) ?? \"\";\n\t} catch (error) {\n\t\tif (error instanceof TypeError) return \"[unserializable]\";\n\t\tthrow error;\n\t}\n}\n\nexport function formatToolArgs(call: ToolCall): string {\n\tconst args = call.arguments;\n\tconst path = args.path;\n\tif ((call.name === \"read\" || call.name === \"write\" || call.name === \"edit\") && typeof path === \"string\") {\n\t\treturn `path: ${path}`;\n\t}\n\tconst command = args.command;\n\tif (call.name === \"bash\" && typeof command === \"string\") return command.replace(/\\t/g, \" \");\n\tconst parts: string[] = [];\n\tlet total = 0;\n\tfor (const [key, value] of Object.entries(args)) {\n\t\tif (key.startsWith(\"_\")) continue;\n\t\tconst encoded = typeof value === \"string\" ? value : safeJson(value);\n\t\tconst part = `${key}: ${encoded}`;\n\t\tif (total + part.length > MAX_TOOL_ARGS_CHARS) break;\n\t\tparts.push(part);\n\t\ttotal += part.length;\n\t}\n\treturn parts.join(\", \");\n}\n\nexport function renderPreview(lines: string[], text: string, width: number, color: \"dim\" | \"thinkingText\"): void {\n\tconst textLines = text.split(\"\\n\");\n\tconst maxWidth = contentWidth(width);\n\tfor (const line of textLines.slice(0, MAX_COLLAPSED_LINES)) {\n\t\tlines.push(`${INDENT}${theme.fg(color, sanitizeLine(line, maxWidth))}`);\n\t}\n\tif (textLines.length > MAX_COLLAPSED_LINES) {\n\t\tlines.push(`${INDENT}${theme.fg(\"dim\", `... ${textLines.length - MAX_COLLAPSED_LINES} more lines`)}`);\n\t}\n}\n"]}
@@ -0,0 +1,65 @@
1
+ import { Markdown } from "@earendil-works/pi-tui";
2
+ import { theme } from "../../../../modes/interactive/theme/theme.js";
3
+ import { INDENT, sanitizeLine } from "./text.js";
4
+ export const MAX_COLLAPSED_LINES = 3;
5
+ export const MAX_EXPANDED_LINES = 100;
6
+ export const MAX_THINKING_COLLAPSED = 200;
7
+ const MAX_TOOL_ARGS_CHARS = 500;
8
+ export function contentWidth(width) {
9
+ return Math.max(20, width - INDENT.length - 4);
10
+ }
11
+ export function renderMarkdown(text, width, markdownTheme) {
12
+ const markdown = new Markdown(text, 0, 0, markdownTheme);
13
+ return markdown.render(Math.max(40, width - INDENT.length - 4)).map((line) => `${INDENT}${line.trimEnd()}`);
14
+ }
15
+ export function toolResultText(result) {
16
+ return result.content
17
+ .filter((part) => part.type === "text")
18
+ .map((part) => part.text)
19
+ .join("\n")
20
+ .trim();
21
+ }
22
+ function safeJson(value) {
23
+ try {
24
+ return JSON.stringify(value) ?? "";
25
+ }
26
+ catch (error) {
27
+ if (error instanceof TypeError)
28
+ return "[unserializable]";
29
+ throw error;
30
+ }
31
+ }
32
+ export function formatToolArgs(call) {
33
+ const args = call.arguments;
34
+ const path = args.path;
35
+ if ((call.name === "read" || call.name === "write" || call.name === "edit") && typeof path === "string") {
36
+ return `path: ${path}`;
37
+ }
38
+ const command = args.command;
39
+ if (call.name === "bash" && typeof command === "string")
40
+ return command.replace(/\t/g, " ");
41
+ const parts = [];
42
+ let total = 0;
43
+ for (const [key, value] of Object.entries(args)) {
44
+ if (key.startsWith("_"))
45
+ continue;
46
+ const encoded = typeof value === "string" ? value : safeJson(value);
47
+ const part = `${key}: ${encoded}`;
48
+ if (total + part.length > MAX_TOOL_ARGS_CHARS)
49
+ break;
50
+ parts.push(part);
51
+ total += part.length;
52
+ }
53
+ return parts.join(", ");
54
+ }
55
+ export function renderPreview(lines, text, width, color) {
56
+ const textLines = text.split("\n");
57
+ const maxWidth = contentWidth(width);
58
+ for (const line of textLines.slice(0, MAX_COLLAPSED_LINES)) {
59
+ lines.push(`${INDENT}${theme.fg(color, sanitizeLine(line, maxWidth))}`);
60
+ }
61
+ if (textLines.length > MAX_COLLAPSED_LINES) {
62
+ lines.push(`${INDENT}${theme.fg("dim", `... ${textLines.length - MAX_COLLAPSED_LINES} more lines`)}`);
63
+ }
64
+ }
65
+ //# sourceMappingURL=transcript-format.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"transcript-format.js","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/transcript-format.ts"],"names":[],"mappings":"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","sourcesContent":["import type { ToolCall, ToolResultMessage } from \"@earendil-works/pi-ai\";\nimport { Markdown, type MarkdownTheme } from \"@earendil-works/pi-tui\";\nimport { theme } from \"../../../../modes/interactive/theme/theme.ts\";\nimport { INDENT, sanitizeLine } from \"./text.ts\";\n\nexport const MAX_COLLAPSED_LINES = 3;\nexport const MAX_EXPANDED_LINES = 100;\nexport const MAX_THINKING_COLLAPSED = 200;\n\nconst MAX_TOOL_ARGS_CHARS = 500;\n\nexport function contentWidth(width: number): number {\n\treturn Math.max(20, width - INDENT.length - 4);\n}\n\nexport function renderMarkdown(text: string, width: number, markdownTheme: MarkdownTheme): readonly string[] {\n\tconst markdown = new Markdown(text, 0, 0, markdownTheme);\n\treturn markdown.render(Math.max(40, width - INDENT.length - 4)).map((line) => `${INDENT}${line.trimEnd()}`);\n}\n\nexport function toolResultText(result: ToolResultMessage<unknown>): string {\n\treturn result.content\n\t\t.filter((part) => part.type === \"text\")\n\t\t.map((part) => part.text)\n\t\t.join(\"\\n\")\n\t\t.trim();\n}\n\nfunction safeJson(value: unknown): string {\n\ttry {\n\t\treturn JSON.stringify(value) ?? \"\";\n\t} catch (error) {\n\t\tif (error instanceof TypeError) return \"[unserializable]\";\n\t\tthrow error;\n\t}\n}\n\nexport function formatToolArgs(call: ToolCall): string {\n\tconst args = call.arguments;\n\tconst path = args.path;\n\tif ((call.name === \"read\" || call.name === \"write\" || call.name === \"edit\") && typeof path === \"string\") {\n\t\treturn `path: ${path}`;\n\t}\n\tconst command = args.command;\n\tif (call.name === \"bash\" && typeof command === \"string\") return command.replace(/\\t/g, \" \");\n\tconst parts: string[] = [];\n\tlet total = 0;\n\tfor (const [key, value] of Object.entries(args)) {\n\t\tif (key.startsWith(\"_\")) continue;\n\t\tconst encoded = typeof value === \"string\" ? value : safeJson(value);\n\t\tconst part = `${key}: ${encoded}`;\n\t\tif (total + part.length > MAX_TOOL_ARGS_CHARS) break;\n\t\tparts.push(part);\n\t\ttotal += part.length;\n\t}\n\treturn parts.join(\", \");\n}\n\nexport function renderPreview(lines: string[], text: string, width: number, color: \"dim\" | \"thinkingText\"): void {\n\tconst textLines = text.split(\"\\n\");\n\tconst maxWidth = contentWidth(width);\n\tfor (const line of textLines.slice(0, MAX_COLLAPSED_LINES)) {\n\t\tlines.push(`${INDENT}${theme.fg(color, sanitizeLine(line, maxWidth))}`);\n\t}\n\tif (textLines.length > MAX_COLLAPSED_LINES) {\n\t\tlines.push(`${INDENT}${theme.fg(\"dim\", `... ${textLines.length - MAX_COLLAPSED_LINES} more lines`)}`);\n\t}\n}\n"]}
@@ -0,0 +1,4 @@
1
+ import type { SessionMessageEntry } from "../../../session-manager.ts";
2
+ import type { RenderedTranscript, TranscriptRenderOptions } from "./types.ts";
3
+ export declare function renderTranscript(entries: readonly SessionMessageEntry[], options: TranscriptRenderOptions): RenderedTranscript;
4
+ //# sourceMappingURL=transcript.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"transcript.d.ts","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/transcript.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,6BAA6B,CAAC;AAGvE,OAAO,KAAK,EAAE,kBAAkB,EAAE,uBAAuB,EAAoB,MAAM,YAAY,CAAC;AA8BhG,wBAAgB,gBAAgB,CAC/B,OAAO,EAAE,SAAS,mBAAmB,EAAE,EACvC,OAAO,EAAE,uBAAuB,GAC9B,kBAAkB,CAwEpB","sourcesContent":["import type { ToolResultMessage } from \"@earendil-works/pi-ai\";\nimport type { SessionMessageEntry } from \"../../../session-manager.ts\";\nimport { getTextContent } from \"./text.ts\";\nimport { renderTextEntry, renderThinkingEntry, renderToolEntry, renderUserEntry } from \"./transcript-entries.ts\";\nimport type { RenderedTranscript, TranscriptRenderOptions, ViewerEntryRange } from \"./types.ts\";\n\nfunction pushRange(\n\tranges: ViewerEntryRange[],\n\tlines: readonly string[],\n\tstart: number,\n\tkind: ViewerEntryRange[\"kind\"],\n): void {\n\tranges.push({ lineStart: start, lineCount: lines.length - start, kind });\n}\n\nfunction collectToolResults(entries: readonly SessionMessageEntry[]): Map<string, ToolResultMessage<unknown>> {\n\tconst results = new Map<string, ToolResultMessage<unknown>>();\n\tfor (const entry of entries) {\n\t\tif (entry.message.role === \"toolResult\") results.set(entry.message.toolCallId, entry.message);\n\t}\n\treturn results;\n}\n\nfunction pushAssistantRange(\n\tranges: ViewerEntryRange[],\n\tlines: string[],\n\tstart: number,\n\tblockType: \"thinking\" | \"text\" | \"toolCall\",\n): void {\n\tif (blockType === \"thinking\") pushRange(ranges, lines, start, \"thinking\");\n\telse if (blockType === \"toolCall\") pushRange(ranges, lines, start, \"tool\");\n\telse pushRange(ranges, lines, start, \"response\");\n}\n\nexport function renderTranscript(\n\tentries: readonly SessionMessageEntry[],\n\toptions: TranscriptRenderOptions,\n): RenderedTranscript {\n\tconst lines: string[] = [];\n\tconst ranges: ViewerEntryRange[] = [];\n\tconst toolResults = collectToolResults(entries);\n\tlet entryIndex = 0;\n\tfor (const entry of entries) {\n\t\tconst message = entry.message;\n\t\tif (message.role === \"toolResult\") continue;\n\t\tif (message.role === \"assistant\") {\n\t\t\tif (message.content.length === 0 && message.errorMessage) {\n\t\t\t\tconst start = lines.length;\n\t\t\t\trenderTextEntry(lines, \"Error\", message.errorMessage, true, entryIndex === options.selectedIndex, options);\n\t\t\t\tpushRange(ranges, lines, start, \"response\");\n\t\t\t\tentryIndex += 1;\n\t\t\t}\n\t\t\tfor (const block of message.content) {\n\t\t\t\tconst start = lines.length;\n\t\t\t\tconst expanded = options.expandedEntries.has(entryIndex);\n\t\t\t\tconst selected = entryIndex === options.selectedIndex;\n\t\t\t\tif (block.type === \"thinking\" && block.thinking.trim())\n\t\t\t\t\trenderThinkingEntry(lines, block.thinking.trim(), expanded, selected, options);\n\t\t\t\telse if (block.type === \"text\" && block.text.trim())\n\t\t\t\t\trenderTextEntry(lines, \"Response\", block.text.trim(), expanded, selected, options);\n\t\t\t\telse if (block.type === \"toolCall\")\n\t\t\t\t\trenderToolEntry(lines, block, toolResults.get(block.id), expanded, selected, options);\n\t\t\t\telse continue;\n\t\t\t\tpushAssistantRange(ranges, lines, start, block.type);\n\t\t\t\tentryIndex += 1;\n\t\t\t}\n\t\t} else if (message.role === \"user\") {\n\t\t\tconst text = getTextContent(message.content).trim();\n\t\t\tif (!text) continue;\n\t\t\tconst start = lines.length;\n\t\t\trenderUserEntry(\n\t\t\t\tlines,\n\t\t\t\t\"User\",\n\t\t\t\ttext,\n\t\t\t\toptions.expandedEntries.has(entryIndex),\n\t\t\t\tentryIndex === options.selectedIndex,\n\t\t\t\toptions,\n\t\t\t);\n\t\t\tpushRange(ranges, lines, start, \"user\");\n\t\t\tentryIndex += 1;\n\t\t} else if (message.role === \"custom\") {\n\t\t\tconst text = getTextContent(message.content).trim();\n\t\t\tif (!text) continue;\n\t\t\tconst start = lines.length;\n\t\t\trenderUserEntry(\n\t\t\t\tlines,\n\t\t\t\tmessage.customType,\n\t\t\t\ttext,\n\t\t\t\toptions.expandedEntries.has(entryIndex),\n\t\t\t\tentryIndex === options.selectedIndex,\n\t\t\t\toptions,\n\t\t\t);\n\t\t\tpushRange(ranges, lines, start, \"system\");\n\t\t\tentryIndex += 1;\n\t\t} else if (message.role === \"bashExecution\") {\n\t\t\tconst start = lines.length;\n\t\t\trenderUserEntry(\n\t\t\t\tlines,\n\t\t\t\t\"Bash\",\n\t\t\t\t`$ ${message.command}\\n${message.output}`,\n\t\t\t\toptions.expandedEntries.has(entryIndex),\n\t\t\t\tentryIndex === options.selectedIndex,\n\t\t\t\toptions,\n\t\t\t);\n\t\t\tpushRange(ranges, lines, start, \"tool\");\n\t\t\tentryIndex += 1;\n\t\t}\n\t}\n\treturn { lines, ranges };\n}\n"]}
@@ -0,0 +1,81 @@
1
+ import { getTextContent } from "./text.js";
2
+ import { renderTextEntry, renderThinkingEntry, renderToolEntry, renderUserEntry } from "./transcript-entries.js";
3
+ function pushRange(ranges, lines, start, kind) {
4
+ ranges.push({ lineStart: start, lineCount: lines.length - start, kind });
5
+ }
6
+ function collectToolResults(entries) {
7
+ const results = new Map();
8
+ for (const entry of entries) {
9
+ if (entry.message.role === "toolResult")
10
+ results.set(entry.message.toolCallId, entry.message);
11
+ }
12
+ return results;
13
+ }
14
+ function pushAssistantRange(ranges, lines, start, blockType) {
15
+ if (blockType === "thinking")
16
+ pushRange(ranges, lines, start, "thinking");
17
+ else if (blockType === "toolCall")
18
+ pushRange(ranges, lines, start, "tool");
19
+ else
20
+ pushRange(ranges, lines, start, "response");
21
+ }
22
+ export function renderTranscript(entries, options) {
23
+ const lines = [];
24
+ const ranges = [];
25
+ const toolResults = collectToolResults(entries);
26
+ let entryIndex = 0;
27
+ for (const entry of entries) {
28
+ const message = entry.message;
29
+ if (message.role === "toolResult")
30
+ continue;
31
+ if (message.role === "assistant") {
32
+ if (message.content.length === 0 && message.errorMessage) {
33
+ const start = lines.length;
34
+ renderTextEntry(lines, "Error", message.errorMessage, true, entryIndex === options.selectedIndex, options);
35
+ pushRange(ranges, lines, start, "response");
36
+ entryIndex += 1;
37
+ }
38
+ for (const block of message.content) {
39
+ const start = lines.length;
40
+ const expanded = options.expandedEntries.has(entryIndex);
41
+ const selected = entryIndex === options.selectedIndex;
42
+ if (block.type === "thinking" && block.thinking.trim())
43
+ renderThinkingEntry(lines, block.thinking.trim(), expanded, selected, options);
44
+ else if (block.type === "text" && block.text.trim())
45
+ renderTextEntry(lines, "Response", block.text.trim(), expanded, selected, options);
46
+ else if (block.type === "toolCall")
47
+ renderToolEntry(lines, block, toolResults.get(block.id), expanded, selected, options);
48
+ else
49
+ continue;
50
+ pushAssistantRange(ranges, lines, start, block.type);
51
+ entryIndex += 1;
52
+ }
53
+ }
54
+ else if (message.role === "user") {
55
+ const text = getTextContent(message.content).trim();
56
+ if (!text)
57
+ continue;
58
+ const start = lines.length;
59
+ renderUserEntry(lines, "User", text, options.expandedEntries.has(entryIndex), entryIndex === options.selectedIndex, options);
60
+ pushRange(ranges, lines, start, "user");
61
+ entryIndex += 1;
62
+ }
63
+ else if (message.role === "custom") {
64
+ const text = getTextContent(message.content).trim();
65
+ if (!text)
66
+ continue;
67
+ const start = lines.length;
68
+ renderUserEntry(lines, message.customType, text, options.expandedEntries.has(entryIndex), entryIndex === options.selectedIndex, options);
69
+ pushRange(ranges, lines, start, "system");
70
+ entryIndex += 1;
71
+ }
72
+ else if (message.role === "bashExecution") {
73
+ const start = lines.length;
74
+ renderUserEntry(lines, "Bash", `$ ${message.command}\n${message.output}`, options.expandedEntries.has(entryIndex), entryIndex === options.selectedIndex, options);
75
+ pushRange(ranges, lines, start, "tool");
76
+ entryIndex += 1;
77
+ }
78
+ }
79
+ return { lines, ranges };
80
+ }
81
+ //# sourceMappingURL=transcript.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"transcript.js","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/transcript.ts"],"names":[],"mappings":"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","sourcesContent":["import type { ToolResultMessage } from \"@earendil-works/pi-ai\";\nimport type { SessionMessageEntry } from \"../../../session-manager.ts\";\nimport { getTextContent } from \"./text.ts\";\nimport { renderTextEntry, renderThinkingEntry, renderToolEntry, renderUserEntry } from \"./transcript-entries.ts\";\nimport type { RenderedTranscript, TranscriptRenderOptions, ViewerEntryRange } from \"./types.ts\";\n\nfunction pushRange(\n\tranges: ViewerEntryRange[],\n\tlines: readonly string[],\n\tstart: number,\n\tkind: ViewerEntryRange[\"kind\"],\n): void {\n\tranges.push({ lineStart: start, lineCount: lines.length - start, kind });\n}\n\nfunction collectToolResults(entries: readonly SessionMessageEntry[]): Map<string, ToolResultMessage<unknown>> {\n\tconst results = new Map<string, ToolResultMessage<unknown>>();\n\tfor (const entry of entries) {\n\t\tif (entry.message.role === \"toolResult\") results.set(entry.message.toolCallId, entry.message);\n\t}\n\treturn results;\n}\n\nfunction pushAssistantRange(\n\tranges: ViewerEntryRange[],\n\tlines: string[],\n\tstart: number,\n\tblockType: \"thinking\" | \"text\" | \"toolCall\",\n): void {\n\tif (blockType === \"thinking\") pushRange(ranges, lines, start, \"thinking\");\n\telse if (blockType === \"toolCall\") pushRange(ranges, lines, start, \"tool\");\n\telse pushRange(ranges, lines, start, \"response\");\n}\n\nexport function renderTranscript(\n\tentries: readonly SessionMessageEntry[],\n\toptions: TranscriptRenderOptions,\n): RenderedTranscript {\n\tconst lines: string[] = [];\n\tconst ranges: ViewerEntryRange[] = [];\n\tconst toolResults = collectToolResults(entries);\n\tlet entryIndex = 0;\n\tfor (const entry of entries) {\n\t\tconst message = entry.message;\n\t\tif (message.role === \"toolResult\") continue;\n\t\tif (message.role === \"assistant\") {\n\t\t\tif (message.content.length === 0 && message.errorMessage) {\n\t\t\t\tconst start = lines.length;\n\t\t\t\trenderTextEntry(lines, \"Error\", message.errorMessage, true, entryIndex === options.selectedIndex, options);\n\t\t\t\tpushRange(ranges, lines, start, \"response\");\n\t\t\t\tentryIndex += 1;\n\t\t\t}\n\t\t\tfor (const block of message.content) {\n\t\t\t\tconst start = lines.length;\n\t\t\t\tconst expanded = options.expandedEntries.has(entryIndex);\n\t\t\t\tconst selected = entryIndex === options.selectedIndex;\n\t\t\t\tif (block.type === \"thinking\" && block.thinking.trim())\n\t\t\t\t\trenderThinkingEntry(lines, block.thinking.trim(), expanded, selected, options);\n\t\t\t\telse if (block.type === \"text\" && block.text.trim())\n\t\t\t\t\trenderTextEntry(lines, \"Response\", block.text.trim(), expanded, selected, options);\n\t\t\t\telse if (block.type === \"toolCall\")\n\t\t\t\t\trenderToolEntry(lines, block, toolResults.get(block.id), expanded, selected, options);\n\t\t\t\telse continue;\n\t\t\t\tpushAssistantRange(ranges, lines, start, block.type);\n\t\t\t\tentryIndex += 1;\n\t\t\t}\n\t\t} else if (message.role === \"user\") {\n\t\t\tconst text = getTextContent(message.content).trim();\n\t\t\tif (!text) continue;\n\t\t\tconst start = lines.length;\n\t\t\trenderUserEntry(\n\t\t\t\tlines,\n\t\t\t\t\"User\",\n\t\t\t\ttext,\n\t\t\t\toptions.expandedEntries.has(entryIndex),\n\t\t\t\tentryIndex === options.selectedIndex,\n\t\t\t\toptions,\n\t\t\t);\n\t\t\tpushRange(ranges, lines, start, \"user\");\n\t\t\tentryIndex += 1;\n\t\t} else if (message.role === \"custom\") {\n\t\t\tconst text = getTextContent(message.content).trim();\n\t\t\tif (!text) continue;\n\t\t\tconst start = lines.length;\n\t\t\trenderUserEntry(\n\t\t\t\tlines,\n\t\t\t\tmessage.customType,\n\t\t\t\ttext,\n\t\t\t\toptions.expandedEntries.has(entryIndex),\n\t\t\t\tentryIndex === options.selectedIndex,\n\t\t\t\toptions,\n\t\t\t);\n\t\t\tpushRange(ranges, lines, start, \"system\");\n\t\t\tentryIndex += 1;\n\t\t} else if (message.role === \"bashExecution\") {\n\t\t\tconst start = lines.length;\n\t\t\trenderUserEntry(\n\t\t\t\tlines,\n\t\t\t\t\"Bash\",\n\t\t\t\t`$ ${message.command}\\n${message.output}`,\n\t\t\t\toptions.expandedEntries.has(entryIndex),\n\t\t\t\tentryIndex === options.selectedIndex,\n\t\t\t\toptions,\n\t\t\t);\n\t\t\tpushRange(ranges, lines, start, \"tool\");\n\t\t\tentryIndex += 1;\n\t\t}\n\t}\n\treturn { lines, ranges };\n}\n"]}
@@ -0,0 +1,33 @@
1
+ import type { MarkdownTheme } from "@earendil-works/pi-tui";
2
+ import type { SessionMessageEntry } from "../../../session-manager.ts";
3
+ export interface SessionHudEntry {
4
+ readonly id: string;
5
+ readonly shortId: string;
6
+ readonly path: string;
7
+ readonly cwd: string;
8
+ readonly createdAt: number;
9
+ readonly modifiedAt: number;
10
+ readonly messageCount: number;
11
+ readonly lastUserText: string;
12
+ readonly isCurrent: boolean;
13
+ }
14
+ export interface TranscriptSnapshot {
15
+ readonly entries: readonly SessionMessageEntry[];
16
+ readonly model?: string;
17
+ }
18
+ export interface ViewerEntryRange {
19
+ readonly lineStart: number;
20
+ readonly lineCount: number;
21
+ readonly kind: "thinking" | "response" | "tool" | "user" | "system";
22
+ }
23
+ export interface RenderedTranscript {
24
+ readonly lines: readonly string[];
25
+ readonly ranges: readonly ViewerEntryRange[];
26
+ }
27
+ export interface TranscriptRenderOptions {
28
+ readonly width: number;
29
+ readonly selectedIndex: number;
30
+ readonly expandedEntries: ReadonlySet<number>;
31
+ readonly markdownTheme?: MarkdownTheme;
32
+ }
33
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/types.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;AAC5D,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,6BAA6B,CAAC;AAEvE,MAAM,WAAW,eAAe;IAC/B,QAAQ,CAAC,EAAE,EAAE,MAAM,CAAC;IACpB,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAC;IACzB,QAAQ,CAAC,IAAI,EAAE,MAAM,CAAC;IACtB,QAAQ,CAAC,GAAG,EAAE,MAAM,CAAC;IACrB,QAAQ,CAAC,SAAS,EAAE,MAAM,CAAC;IAC3B,QAAQ,CAAC,UAAU,EAAE,MAAM,CAAC;IAC5B,QAAQ,CAAC,YAAY,EAAE,MAAM,CAAC;IAC9B,QAAQ,CAAC,YAAY,EAAE,MAAM,CAAC;IAC9B,QAAQ,CAAC,SAAS,EAAE,OAAO,CAAC;CAC5B;AAED,MAAM,WAAW,kBAAkB;IAClC,QAAQ,CAAC,OAAO,EAAE,SAAS,mBAAmB,EAAE,CAAC;IACjD,QAAQ,CAAC,KAAK,CAAC,EAAE,MAAM,CAAC;CACxB;AAED,MAAM,WAAW,gBAAgB;IAChC,QAAQ,CAAC,SAAS,EAAE,MAAM,CAAC;IAC3B,QAAQ,CAAC,SAAS,EAAE,MAAM,CAAC;IAC3B,QAAQ,CAAC,IAAI,EAAE,UAAU,GAAG,UAAU,GAAG,MAAM,GAAG,MAAM,GAAG,QAAQ,CAAC;CACpE;AAED,MAAM,WAAW,kBAAkB;IAClC,QAAQ,CAAC,KAAK,EAAE,SAAS,MAAM,EAAE,CAAC;IAClC,QAAQ,CAAC,MAAM,EAAE,SAAS,gBAAgB,EAAE,CAAC;CAC7C;AAED,MAAM,WAAW,uBAAuB;IACvC,QAAQ,CAAC,KAAK,EAAE,MAAM,CAAC;IACvB,QAAQ,CAAC,aAAa,EAAE,MAAM,CAAC;IAC/B,QAAQ,CAAC,eAAe,EAAE,WAAW,CAAC,MAAM,CAAC,CAAC;IAC9C,QAAQ,CAAC,aAAa,CAAC,EAAE,aAAa,CAAC;CACvC","sourcesContent":["import type { MarkdownTheme } from \"@earendil-works/pi-tui\";\nimport type { SessionMessageEntry } from \"../../../session-manager.ts\";\n\nexport interface SessionHudEntry {\n\treadonly id: string;\n\treadonly shortId: string;\n\treadonly path: string;\n\treadonly cwd: string;\n\treadonly createdAt: number;\n\treadonly modifiedAt: number;\n\treadonly messageCount: number;\n\treadonly lastUserText: string;\n\treadonly isCurrent: boolean;\n}\n\nexport interface TranscriptSnapshot {\n\treadonly entries: readonly SessionMessageEntry[];\n\treadonly model?: string;\n}\n\nexport interface ViewerEntryRange {\n\treadonly lineStart: number;\n\treadonly lineCount: number;\n\treadonly kind: \"thinking\" | \"response\" | \"tool\" | \"user\" | \"system\";\n}\n\nexport interface RenderedTranscript {\n\treadonly lines: readonly string[];\n\treadonly ranges: readonly ViewerEntryRange[];\n}\n\nexport interface TranscriptRenderOptions {\n\treadonly width: number;\n\treadonly selectedIndex: number;\n\treadonly expandedEntries: ReadonlySet<number>;\n\treadonly markdownTheme?: MarkdownTheme;\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../src/core/extensions/builtin/session-observer/types.ts"],"names":[],"mappings":"","sourcesContent":["import type { MarkdownTheme } from \"@earendil-works/pi-tui\";\nimport type { SessionMessageEntry } from \"../../../session-manager.ts\";\n\nexport interface SessionHudEntry {\n\treadonly id: string;\n\treadonly shortId: string;\n\treadonly path: string;\n\treadonly cwd: string;\n\treadonly createdAt: number;\n\treadonly modifiedAt: number;\n\treadonly messageCount: number;\n\treadonly lastUserText: string;\n\treadonly isCurrent: boolean;\n}\n\nexport interface TranscriptSnapshot {\n\treadonly entries: readonly SessionMessageEntry[];\n\treadonly model?: string;\n}\n\nexport interface ViewerEntryRange {\n\treadonly lineStart: number;\n\treadonly lineCount: number;\n\treadonly kind: \"thinking\" | \"response\" | \"tool\" | \"user\" | \"system\";\n}\n\nexport interface RenderedTranscript {\n\treadonly lines: readonly string[];\n\treadonly ranges: readonly ViewerEntryRange[];\n}\n\nexport interface TranscriptRenderOptions {\n\treadonly width: number;\n\treadonly selectedIndex: number;\n\treadonly expandedEntries: ReadonlySet<number>;\n\treadonly markdownTheme?: MarkdownTheme;\n}\n"]}