acpx 0.3.1 → 0.4.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (35) hide show
  1. package/README.md +65 -16
  2. package/dist/{acp-jsonrpc-BNHXq7qK.js → acp-jsonrpc-BbBgC5gO.js} +15 -2
  3. package/dist/acp-jsonrpc-BbBgC5gO.js.map +1 -0
  4. package/dist/cli-idpWyCOs.js +176 -0
  5. package/dist/cli-idpWyCOs.js.map +1 -0
  6. package/dist/cli.d.ts +1 -118
  7. package/dist/cli.d.ts.map +1 -1
  8. package/dist/cli.js +235 -336
  9. package/dist/cli.js.map +1 -1
  10. package/dist/flags-CCcX9fZj.js +194 -0
  11. package/dist/flags-CCcX9fZj.js.map +1 -0
  12. package/dist/flows-BL1tSvZT.js +1551 -0
  13. package/dist/flows-BL1tSvZT.js.map +1 -0
  14. package/dist/flows.d.ts +292 -0
  15. package/dist/flows.d.ts.map +1 -0
  16. package/dist/flows.js +2 -0
  17. package/dist/{output-BmkPP7qE.js → output-Du3m6oPQ.js} +139 -30
  18. package/dist/output-Du3m6oPQ.js.map +1 -0
  19. package/dist/{output-render-DEAaMxg8.js → output-render-Bz58qaQn.js} +10 -10
  20. package/dist/output-render-Bz58qaQn.js.map +1 -0
  21. package/dist/{queue-ipc-EQLpBMKv.js → queue-ipc-CE8_QGX3.js} +258 -95
  22. package/dist/queue-ipc-CE8_QGX3.js.map +1 -0
  23. package/dist/{session-C2Q8ktsN.js → session-RO_LZUnv.js} +687 -138
  24. package/dist/session-RO_LZUnv.js.map +1 -0
  25. package/dist/types-CeRKmEQ1.d.ts +137 -0
  26. package/dist/types-CeRKmEQ1.d.ts.map +1 -0
  27. package/package.json +44 -16
  28. package/skills/acpx/SKILL.md +22 -5
  29. package/dist/acp-jsonrpc-BNHXq7qK.js.map +0 -1
  30. package/dist/output-BmkPP7qE.js.map +0 -1
  31. package/dist/output-render-DEAaMxg8.js.map +0 -1
  32. package/dist/queue-ipc-EQLpBMKv.js.map +0 -1
  33. package/dist/runtime-session-id-C544sPPL.js +0 -31
  34. package/dist/runtime-session-id-C544sPPL.js.map +0 -1
  35. package/dist/session-C2Q8ktsN.js.map +0 -1
@@ -1,4 +1,5 @@
1
- import { i as parsePromptStopReason, r as parseJsonRpcErrorMessage } from "./acp-jsonrpc-BNHXq7qK.js";
1
+ import { t as __exportAll } from "./rolldown-runtime-CiIaOW0V.js";
2
+ import { a as parsePromptStopReason, i as parseJsonRpcErrorMessage, t as extractSessionUpdateNotification } from "./acp-jsonrpc-BbBgC5gO.js";
2
3
  //#region src/jsonrpc-error.ts
3
4
  const OUTPUT_ERROR_JSONRPC_CODES = {
4
5
  NO_SESSION: -32002,
@@ -44,20 +45,129 @@ function buildJsonRpcErrorResponse(params) {
44
45
  };
45
46
  }
46
47
  //#endregion
48
+ //#region src/read-output-suppression.ts
49
+ const SUPPRESSED_READ_OUTPUT = "[read output suppressed]";
50
+ function inferToolKindFromTitle(title) {
51
+ const normalized = title?.trim().toLowerCase();
52
+ if (!normalized) return;
53
+ const head = normalized.split(":", 1)[0]?.trim();
54
+ if (!head) return;
55
+ if (head.includes("read") || head.includes("cat") || head.includes("open") || head.includes("view")) return "read";
56
+ }
57
+ function isReadLikeTool(tool) {
58
+ return tool.kind?.trim().toLowerCase() === "read" || inferToolKindFromTitle(tool.title) === "read";
59
+ }
60
+ //#endregion
47
61
  //#region src/output-json-formatter.ts
48
62
  const DEFAULT_JSON_SESSION_ID = "unknown";
63
+ function asRecord$1(value) {
64
+ if (!value || typeof value !== "object" || Array.isArray(value)) return;
65
+ return value;
66
+ }
67
+ function jsonRpcIdKey(value) {
68
+ if (typeof value === "string") return `s:${value}`;
69
+ if (typeof value === "number" && Number.isFinite(value)) return `n:${value}`;
70
+ }
71
+ function sanitizeReadResult(result) {
72
+ const record = asRecord$1(result);
73
+ if (!record || typeof record.content !== "string") return result;
74
+ return {
75
+ ...record,
76
+ content: SUPPRESSED_READ_OUTPUT
77
+ };
78
+ }
79
+ function sanitizeToolContent(content) {
80
+ if (!Array.isArray(content)) return content;
81
+ return [{
82
+ type: "content",
83
+ content: {
84
+ type: "text",
85
+ text: SUPPRESSED_READ_OUTPUT
86
+ }
87
+ }];
88
+ }
89
+ function sanitizeToolMessage(message) {
90
+ const root = asRecord$1(message);
91
+ const params = asRecord$1(root?.params);
92
+ const update = asRecord$1(params?.update);
93
+ if (!root || !params || !update) return message;
94
+ return {
95
+ ...root,
96
+ params: {
97
+ ...params,
98
+ update: {
99
+ ...update,
100
+ rawOutput: Object.prototype.hasOwnProperty.call(update, "rawOutput") && update.rawOutput !== void 0 ? { content: SUPPRESSED_READ_OUTPUT } : update.rawOutput,
101
+ content: Object.prototype.hasOwnProperty.call(update, "content") && update.content !== void 0 ? sanitizeToolContent(update.content) : update.content
102
+ }
103
+ }
104
+ };
105
+ }
49
106
  var JsonOutputFormatter = class {
50
107
  stdout;
108
+ suppressReads;
51
109
  sessionId;
52
- constructor(stdout, context) {
110
+ requestMethodById = /* @__PURE__ */ new Map();
111
+ toolStateById = /* @__PURE__ */ new Map();
112
+ constructor(stdout, suppressReads, context) {
53
113
  this.stdout = stdout;
114
+ this.suppressReads = suppressReads;
54
115
  this.sessionId = context?.sessionId?.trim() || DEFAULT_JSON_SESSION_ID;
55
116
  }
56
117
  setContext(context) {
57
118
  this.sessionId = context.sessionId?.trim() || this.sessionId || DEFAULT_JSON_SESSION_ID;
58
119
  }
59
120
  onAcpMessage(message) {
60
- this.stdout.write(`${JSON.stringify(message)}\n`);
121
+ this.stdout.write(`${JSON.stringify(this.sanitizeMessage(message))}\n`);
122
+ }
123
+ sanitizeMessage(message) {
124
+ if (!this.suppressReads) return message;
125
+ const sanitizedResponse = this.sanitizeReadResponse(message);
126
+ if (sanitizedResponse !== message) return sanitizedResponse;
127
+ const sanitizedToolMessage = this.sanitizeReadToolMessage(message);
128
+ if (sanitizedToolMessage !== message) return sanitizedToolMessage;
129
+ this.trackRequestMethod(message);
130
+ return message;
131
+ }
132
+ trackRequestMethod(message) {
133
+ const candidate = message;
134
+ if (typeof candidate.method !== "string") return;
135
+ const idKey = jsonRpcIdKey(candidate.id);
136
+ if (!idKey) return;
137
+ this.requestMethodById.set(idKey, candidate.method);
138
+ }
139
+ sanitizeReadResponse(message) {
140
+ const candidate = message;
141
+ const idKey = jsonRpcIdKey(candidate.id);
142
+ if (!idKey || !Object.hasOwn(candidate, "result")) return message;
143
+ const method = this.requestMethodById.get(idKey);
144
+ this.requestMethodById.delete(idKey);
145
+ if (method !== "fs/read_text_file") return message;
146
+ const root = asRecord$1(message);
147
+ if (!root) return message;
148
+ return {
149
+ ...root,
150
+ result: sanitizeReadResult(candidate.result)
151
+ };
152
+ }
153
+ sanitizeReadToolMessage(message) {
154
+ const root = asRecord$1(message);
155
+ if (root?.method !== "session/update") return message;
156
+ const params = asRecord$1(root.params);
157
+ const update = asRecord$1(params?.update);
158
+ if (!params || !update) return message;
159
+ const sessionUpdate = update.sessionUpdate;
160
+ if (sessionUpdate !== "tool_call" && sessionUpdate !== "tool_call_update") return message;
161
+ const toolCallId = typeof update.toolCallId === "string" ? update.toolCallId : void 0;
162
+ if (!toolCallId) return message;
163
+ const previous = this.toolStateById.get(toolCallId) ?? {};
164
+ const current = {
165
+ title: typeof update.title === "string" ? update.title : previous.title,
166
+ kind: typeof update.kind === "string" || update.kind === null ? update.kind : previous.kind
167
+ };
168
+ this.toolStateById.set(toolCallId, current);
169
+ if (!isReadLikeTool(current)) return message;
170
+ return sanitizeToolMessage(message);
61
171
  }
62
172
  onError(params) {
63
173
  this.stdout.write(`${JSON.stringify(buildJsonRpcErrorResponse({
@@ -73,11 +183,12 @@ var JsonOutputFormatter = class {
73
183
  }
74
184
  flush() {}
75
185
  };
76
- function createJsonOutputFormatter(stdout, context) {
77
- return new JsonOutputFormatter(stdout, context);
186
+ function createJsonOutputFormatter(stdout, suppressReads = false, context) {
187
+ return new JsonOutputFormatter(stdout, suppressReads, context);
78
188
  }
79
189
  //#endregion
80
190
  //#region src/output.ts
191
+ var output_exports = /* @__PURE__ */ __exportAll({ createOutputFormatter: () => createOutputFormatter });
81
192
  const MAX_THOUGHT_CHARS = 900;
82
193
  const MAX_INLINE_CHARS = 220;
83
194
  const MAX_OUTPUT_CHARS = 2e3;
@@ -113,26 +224,15 @@ function asRecord(value) {
113
224
  if (!value || typeof value !== "object" || Array.isArray(value)) return;
114
225
  return value;
115
226
  }
116
- function extractSessionUpdate(message) {
117
- if (!Object.hasOwn(message, "method")) return;
118
- if (message.method !== "session/update") return;
119
- const params = asRecord(message.params);
120
- if (!params) return;
121
- const sessionId = typeof params.sessionId === "string" ? params.sessionId : null;
122
- if (!sessionId) return;
123
- const update = asRecord(params.update);
124
- if (!update || typeof update.sessionUpdate !== "string") return;
125
- return {
126
- sessionId,
127
- update
128
- };
129
- }
130
227
  function extractJsonRpcMethod(message) {
131
228
  return Object.hasOwn(message, "method") ? message.method?.toString() : void 0;
132
229
  }
133
230
  function collapseWhitespace(value) {
134
231
  return value.replace(/\s+/g, " ").trim();
135
232
  }
233
+ function normalizeLineEndings(value) {
234
+ return value.replace(/\r\n?/g, "\n");
235
+ }
136
236
  function truncate(value, maxChars) {
137
237
  if (value.length <= maxChars) return value;
138
238
  if (maxChars <= 3) return value.slice(0, maxChars);
@@ -308,6 +408,10 @@ function summarizeToolOutput(rawOutput, content) {
308
408
  if (fragments.length === 0) return;
309
409
  return fragments.join("\n\n");
310
410
  }
411
+ function renderToolOutput(state, suppressReads) {
412
+ if (suppressReads && isReadLikeTool(state)) return SUPPRESSED_READ_OUTPUT;
413
+ return summarizeToolOutput(state.rawOutput, state.content);
414
+ }
311
415
  function limitOutputBlock(value) {
312
416
  const normalized = value.replace(/\r\n/g, "\n").trim();
313
417
  if (!normalized) return "";
@@ -324,18 +428,20 @@ function limitOutputBlock(value) {
324
428
  var TextOutputFormatter = class {
325
429
  stdout;
326
430
  useColor;
431
+ suppressReads;
327
432
  toolStates = /* @__PURE__ */ new Map();
328
433
  thoughtBuffer = "";
329
434
  wroteAny = false;
330
435
  atLineStart = true;
331
436
  section = null;
332
- constructor(stdout) {
437
+ constructor(stdout, suppressReads) {
333
438
  this.stdout = stdout;
334
439
  this.useColor = Boolean(stdout.isTTY);
440
+ this.suppressReads = suppressReads;
335
441
  }
336
442
  setContext(_context) {}
337
443
  onAcpMessage(message) {
338
- const notification = extractSessionUpdate(message);
444
+ const notification = extractSessionUpdateNotification(message);
339
445
  if (notification) {
340
446
  this.renderSessionUpdate(notification);
341
447
  return;
@@ -431,11 +537,13 @@ var TextOutputFormatter = class {
431
537
  this.write(text);
432
538
  }
433
539
  flushThoughtBuffer() {
434
- const thought = truncate(collapseWhitespace(this.thoughtBuffer), MAX_THOUGHT_CHARS);
540
+ const thought = truncate(normalizeLineEndings(this.thoughtBuffer).trim(), MAX_THOUGHT_CHARS);
435
541
  this.thoughtBuffer = "";
436
542
  if (!thought) return;
437
543
  this.beginSection("thought");
438
- this.writeLine(this.dim(`[thinking] ${thought}`));
544
+ const [firstLine, ...restLines] = thought.split("\n");
545
+ this.writeLine(this.dim(`[thinking] ${firstLine}`));
546
+ for (const line of restLines) this.writeLine(this.dim(` ${line}`));
439
547
  }
440
548
  renderToolUpdate(update) {
441
549
  const state = this.getOrCreateToolState(update.toolCallId);
@@ -479,7 +587,7 @@ var TextOutputFormatter = class {
479
587
  kind: state.kind,
480
588
  input: summarizeToolInput(state.rawInput),
481
589
  files: formatLocations(state.locations),
482
- output: summarizeToolOutput(state.rawOutput, state.content)
590
+ output: renderToolOutput(state, this.suppressReads)
483
591
  };
484
592
  return safeJson(signaturePayload, 0) ?? JSON.stringify(signaturePayload);
485
593
  }
@@ -504,7 +612,7 @@ var TextOutputFormatter = class {
504
612
  if (input) this.writeLine(` input: ${input}`);
505
613
  const files = formatLocations(state.locations);
506
614
  if (files) this.writeLine(` files: ${files}`);
507
- const output = summarizeToolOutput(state.rawOutput, state.content);
615
+ const output = renderToolOutput(state, this.suppressReads);
508
616
  if (output) {
509
617
  this.writeLine(" output:");
510
618
  this.writeLine(indentBlock(limitOutputBlock(output), " "));
@@ -538,7 +646,7 @@ var QuietOutputFormatter = class {
538
646
  }
539
647
  setContext(_context) {}
540
648
  onAcpMessage(message) {
541
- const update = extractSessionUpdate(message);
649
+ const update = extractSessionUpdateNotification(message);
542
650
  if (update?.update.sessionUpdate === "agent_message_chunk" && update.update.content.type === "text") {
543
651
  this.chunks.push(update.update.content.text);
544
652
  return;
@@ -556,14 +664,15 @@ var QuietOutputFormatter = class {
556
664
  };
557
665
  function createOutputFormatter(format, options = {}) {
558
666
  const stdout = options.stdout ?? process.stdout;
667
+ const suppressReads = options.suppressReads === true;
559
668
  switch (format) {
560
- case "text": return new TextOutputFormatter(stdout);
561
- case "json": return createJsonOutputFormatter(stdout, options.jsonContext);
669
+ case "text": return new TextOutputFormatter(stdout, suppressReads);
670
+ case "json": return createJsonOutputFormatter(stdout, suppressReads, options.jsonContext);
562
671
  case "quiet": return new QuietOutputFormatter(stdout);
563
672
  default: throw new Error("Unsupported output format");
564
673
  }
565
674
  }
566
675
  //#endregion
567
- export { createOutputFormatter };
676
+ export { output_exports as n, createOutputFormatter as t };
568
677
 
569
- //# sourceMappingURL=output-BmkPP7qE.js.map
678
+ //# sourceMappingURL=output-Du3m6oPQ.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"output-Du3m6oPQ.js","names":["asRecord"],"sources":["../src/jsonrpc-error.ts","../src/read-output-suppression.ts","../src/output-json-formatter.ts","../src/output.ts"],"sourcesContent":["import type { OutputErrorAcpPayload, OutputErrorCode, OutputErrorOrigin } from \"./types.js\";\n\nexport const OUTPUT_ERROR_JSONRPC_CODES: Record<OutputErrorCode, number> = {\n NO_SESSION: -32002,\n TIMEOUT: -32070,\n PERMISSION_DENIED: -32071,\n PERMISSION_PROMPT_UNAVAILABLE: -32072,\n RUNTIME: -32603,\n USAGE: -32602,\n};\n\ntype JsonRpcErrorObject = {\n code: number;\n message: string;\n data?: unknown;\n};\n\nexport type BuildJsonRpcErrorParams = {\n id?: string | number | null;\n outputCode: OutputErrorCode;\n detailCode?: string;\n origin?: OutputErrorOrigin;\n message: string;\n retryable?: boolean;\n timestamp?: string;\n sessionId?: string;\n acp?: OutputErrorAcpPayload;\n};\n\nfunction hasValidAcpError(\n acp: OutputErrorAcpPayload | undefined,\n): acp is { code: number; message: string; data?: unknown } {\n return Boolean(\n acp &&\n Number.isFinite(acp.code) &&\n typeof acp.message === \"string\" &&\n acp.message.trim().length > 0,\n );\n}\n\nfunction buildFallbackData(params: BuildJsonRpcErrorParams): Record<string, unknown> {\n const data: Record<string, unknown> = {\n acpxCode: params.outputCode,\n detailCode: params.detailCode,\n origin: params.origin,\n retryable: params.retryable,\n timestamp: params.timestamp,\n sessionId: params.sessionId,\n };\n\n for (const [key, value] of Object.entries(data)) {\n if (value === undefined) {\n delete data[key];\n }\n }\n\n return data;\n}\n\nfunction buildErrorObject(params: BuildJsonRpcErrorParams): JsonRpcErrorObject {\n if (hasValidAcpError(params.acp)) {\n return {\n code: params.acp.code,\n message: params.acp.message,\n ...(params.acp.data !== undefined ? { data: params.acp.data } : {}),\n };\n }\n\n const data = buildFallbackData(params);\n return {\n code: OUTPUT_ERROR_JSONRPC_CODES[params.outputCode] ?? -32603,\n message: params.message,\n ...(Object.keys(data).length > 0 ? { data } : {}),\n };\n}\n\nexport function buildJsonRpcErrorResponse(params: BuildJsonRpcErrorParams): {\n jsonrpc: \"2.0\";\n id: string | number | null;\n error: JsonRpcErrorObject;\n} {\n return {\n jsonrpc: \"2.0\",\n id: params.id ?? null,\n error: buildErrorObject(params),\n };\n}\n","export const SUPPRESSED_READ_OUTPUT = \"[read output suppressed]\";\n\nexport type ReadLikeToolDescriptor = {\n title?: string;\n kind?: string | null;\n};\n\nfunction inferToolKindFromTitle(title: string | undefined): string | undefined {\n const normalized = title?.trim().toLowerCase();\n if (!normalized) {\n return undefined;\n }\n\n const head = normalized.split(\":\", 1)[0]?.trim();\n if (!head) {\n return undefined;\n }\n\n if (\n head.includes(\"read\") ||\n head.includes(\"cat\") ||\n head.includes(\"open\") ||\n head.includes(\"view\")\n ) {\n return \"read\";\n }\n\n return undefined;\n}\n\nexport function isReadLikeTool(tool: ReadLikeToolDescriptor): boolean {\n return (\n tool.kind?.trim().toLowerCase() === \"read\" || inferToolKindFromTitle(tool.title) === \"read\"\n );\n}\n","import { buildJsonRpcErrorResponse } from \"./jsonrpc-error.js\";\nimport { isReadLikeTool, SUPPRESSED_READ_OUTPUT } from \"./read-output-suppression.js\";\nimport type {\n OutputErrorAcpPayload,\n OutputErrorCode,\n OutputErrorOrigin,\n OutputFormatter,\n OutputFormatterContext,\n} from \"./types.js\";\n\ntype WritableLike = {\n write(chunk: string): void;\n};\n\ntype JsonRpcRequestMessage = {\n jsonrpc?: unknown;\n id?: unknown;\n method?: unknown;\n};\n\ntype JsonRpcResponseMessage = {\n jsonrpc?: unknown;\n id?: unknown;\n result?: unknown;\n};\n\nconst DEFAULT_JSON_SESSION_ID = \"unknown\";\n\nfunction asRecord(value: unknown): Record<string, unknown> | undefined {\n if (!value || typeof value !== \"object\" || Array.isArray(value)) {\n return undefined;\n }\n return value as Record<string, unknown>;\n}\n\nfunction jsonRpcIdKey(value: unknown): string | undefined {\n if (typeof value === \"string\") {\n return `s:${value}`;\n }\n if (typeof value === \"number\" && Number.isFinite(value)) {\n return `n:${value}`;\n }\n return undefined;\n}\n\nfunction sanitizeReadResult(result: unknown): unknown {\n const record = asRecord(result);\n if (!record || typeof record.content !== \"string\") {\n return result;\n }\n return {\n ...record,\n content: SUPPRESSED_READ_OUTPUT,\n };\n}\n\nfunction sanitizeToolContent(content: unknown): unknown {\n if (!Array.isArray(content)) {\n return content;\n }\n\n return [\n {\n type: \"content\",\n content: {\n type: \"text\",\n text: SUPPRESSED_READ_OUTPUT,\n },\n },\n ];\n}\n\nfunction sanitizeToolMessage(message: unknown): unknown {\n const root = asRecord(message);\n const params = asRecord(root?.params);\n const update = asRecord(params?.update);\n if (!root || !params || !update) {\n return message;\n }\n\n return {\n ...root,\n params: {\n ...params,\n update: {\n ...update,\n rawOutput:\n Object.prototype.hasOwnProperty.call(update, \"rawOutput\") &&\n update.rawOutput !== undefined\n ? { content: SUPPRESSED_READ_OUTPUT }\n : update.rawOutput,\n content:\n Object.prototype.hasOwnProperty.call(update, \"content\") && update.content !== undefined\n ? sanitizeToolContent(update.content)\n : update.content,\n },\n },\n };\n}\n\nclass JsonOutputFormatter implements OutputFormatter {\n private readonly stdout: WritableLike;\n private readonly suppressReads: boolean;\n private sessionId: string;\n private readonly requestMethodById = new Map<string, string>();\n private readonly toolStateById = new Map<string, { title?: string; kind?: string | null }>();\n\n constructor(stdout: WritableLike, suppressReads: boolean, context?: OutputFormatterContext) {\n this.stdout = stdout;\n this.suppressReads = suppressReads;\n this.sessionId = context?.sessionId?.trim() || DEFAULT_JSON_SESSION_ID;\n }\n\n setContext(context: OutputFormatterContext): void {\n this.sessionId = context.sessionId?.trim() || this.sessionId || DEFAULT_JSON_SESSION_ID;\n }\n\n onAcpMessage(message: unknown): void {\n this.stdout.write(`${JSON.stringify(this.sanitizeMessage(message))}\\n`);\n }\n\n private sanitizeMessage(message: unknown): unknown {\n if (!this.suppressReads) {\n return message;\n }\n\n const sanitizedResponse = this.sanitizeReadResponse(message);\n if (sanitizedResponse !== message) {\n return sanitizedResponse;\n }\n\n const sanitizedToolMessage = this.sanitizeReadToolMessage(message);\n if (sanitizedToolMessage !== message) {\n return sanitizedToolMessage;\n }\n\n this.trackRequestMethod(message);\n return message;\n }\n\n private trackRequestMethod(message: unknown): void {\n const candidate = message as JsonRpcRequestMessage;\n if (typeof candidate.method !== \"string\") {\n return;\n }\n const idKey = jsonRpcIdKey(candidate.id);\n if (!idKey) {\n return;\n }\n this.requestMethodById.set(idKey, candidate.method);\n }\n\n private sanitizeReadResponse(message: unknown): unknown {\n const candidate = message as JsonRpcResponseMessage;\n const idKey = jsonRpcIdKey(candidate.id);\n if (!idKey || !Object.hasOwn(candidate, \"result\")) {\n return message;\n }\n\n const method = this.requestMethodById.get(idKey);\n this.requestMethodById.delete(idKey);\n if (method !== \"fs/read_text_file\") {\n return message;\n }\n\n const root = asRecord(message);\n if (!root) {\n return message;\n }\n\n return {\n ...root,\n result: sanitizeReadResult(candidate.result),\n };\n }\n\n private sanitizeReadToolMessage(message: unknown): unknown {\n const root = asRecord(message);\n if (root?.method !== \"session/update\") {\n return message;\n }\n\n const params = asRecord(root.params);\n const update = asRecord(params?.update);\n if (!params || !update) {\n return message;\n }\n\n const sessionUpdate = update.sessionUpdate;\n if (sessionUpdate !== \"tool_call\" && sessionUpdate !== \"tool_call_update\") {\n return message;\n }\n\n const toolCallId = typeof update.toolCallId === \"string\" ? update.toolCallId : undefined;\n if (!toolCallId) {\n return message;\n }\n\n const previous = this.toolStateById.get(toolCallId) ?? {};\n const current = {\n title: typeof update.title === \"string\" ? update.title : previous.title,\n kind: typeof update.kind === \"string\" || update.kind === null ? update.kind : previous.kind,\n };\n this.toolStateById.set(toolCallId, current);\n\n if (!isReadLikeTool(current)) {\n return message;\n }\n\n return sanitizeToolMessage(message);\n }\n\n onError(params: {\n code: OutputErrorCode;\n detailCode?: string;\n origin?: OutputErrorOrigin;\n message: string;\n retryable?: boolean;\n acp?: OutputErrorAcpPayload;\n timestamp?: string;\n }): void {\n this.stdout.write(\n `${JSON.stringify(\n buildJsonRpcErrorResponse({\n outputCode: params.code,\n detailCode: params.detailCode,\n origin: params.origin,\n message: params.message,\n retryable: params.retryable,\n timestamp: params.timestamp,\n sessionId: this.sessionId,\n acp: params.acp,\n }),\n )}\\n`,\n );\n }\n\n flush(): void {\n // no-op for streaming output\n }\n}\n\nexport function createJsonOutputFormatter(\n stdout: WritableLike,\n suppressReads = false,\n context?: OutputFormatterContext,\n): OutputFormatter {\n return new JsonOutputFormatter(stdout, suppressReads, context);\n}\n","import type {\n AnyMessage,\n ContentBlock,\n SessionNotification,\n ToolCall,\n ToolCallContent,\n ToolCallLocation,\n ToolCallStatus,\n ToolCallUpdate,\n} from \"@agentclientprotocol/sdk\";\nimport {\n extractSessionUpdateNotification,\n parseJsonRpcErrorMessage,\n parsePromptStopReason,\n} from \"./acp-jsonrpc.js\";\nimport { createJsonOutputFormatter } from \"./output-json-formatter.js\";\nimport { isReadLikeTool, SUPPRESSED_READ_OUTPUT } from \"./read-output-suppression.js\";\nimport type {\n AcpJsonRpcMessage,\n ClientOperation,\n OutputErrorAcpPayload,\n OutputErrorCode,\n OutputFormatterContext,\n OutputFormat,\n OutputFormatter,\n OutputErrorOrigin,\n} from \"./types.js\";\n\ntype WritableLike = {\n write(chunk: string): void;\n isTTY?: boolean;\n};\n\ntype OutputFormatterOptions = {\n stdout?: WritableLike;\n jsonContext?: OutputFormatterContext;\n suppressReads?: boolean;\n};\n\ntype NormalizedToolStatus = ToolCallStatus | \"unknown\";\n\ntype FormatterSection = \"assistant\" | \"thought\" | \"tool\" | \"plan\" | \"client\" | \"done\";\n\ntype ToolRenderState = {\n id: string;\n title?: string;\n status?: ToolCallStatus | null;\n kind?: string | null;\n locations?: Array<ToolCallLocation> | null;\n rawInput?: unknown;\n rawOutput?: unknown;\n content?: Array<ToolCallContent> | null;\n startedPrinted: boolean;\n finalSignature?: string;\n};\n\nconst MAX_THOUGHT_CHARS = 900;\nconst MAX_INLINE_CHARS = 220;\nconst MAX_OUTPUT_CHARS = 2_000;\nconst MAX_OUTPUT_LINES = 28;\nconst MAX_LOCATION_ITEMS = 5;\nconst OUTPUT_PRIORITY_KEYS = [\n \"stdout\",\n \"stderr\",\n \"output\",\n \"content\",\n \"text\",\n \"message\",\n \"result\",\n \"response\",\n \"value\",\n] as const;\n\nfunction asStatus(status: ToolCallStatus | null | undefined): NormalizedToolStatus {\n return status ?? \"unknown\";\n}\n\nfunction isFinalStatus(status: NormalizedToolStatus): status is \"completed\" | \"failed\" {\n return status === \"completed\" || status === \"failed\";\n}\n\nfunction toStatusLabel(status: NormalizedToolStatus): string {\n switch (status) {\n case \"in_progress\":\n return \"running\";\n case \"pending\":\n return \"pending\";\n case \"completed\":\n return \"completed\";\n case \"failed\":\n return \"failed\";\n default:\n return \"running\";\n }\n}\n\nfunction asRecord(value: unknown): Record<string, unknown> | undefined {\n if (!value || typeof value !== \"object\" || Array.isArray(value)) {\n return undefined;\n }\n return value as Record<string, unknown>;\n}\n\nfunction extractJsonRpcMethod(message: AnyMessage): string | undefined {\n return Object.hasOwn(message, \"method\")\n ? (message as { method?: unknown }).method?.toString()\n : undefined;\n}\n\nfunction collapseWhitespace(value: string): string {\n return value.replace(/\\s+/g, \" \").trim();\n}\n\nfunction normalizeLineEndings(value: string): string {\n return value.replace(/\\r\\n?/g, \"\\n\");\n}\n\nfunction truncate(value: string, maxChars: number): string {\n if (value.length <= maxChars) {\n return value;\n }\n if (maxChars <= 3) {\n return value.slice(0, maxChars);\n }\n return `${value.slice(0, maxChars - 3)}...`;\n}\n\nfunction toInline(value: string, maxChars = MAX_INLINE_CHARS): string {\n return truncate(collapseWhitespace(value), maxChars);\n}\n\nfunction indentBlock(value: string, prefix: string): string {\n return value\n .split(\"\\n\")\n .map((line) => `${prefix}${line}`)\n .join(\"\\n\");\n}\n\nfunction dedupeStrings(values: string[]): string[] {\n const seen = new Set<string>();\n const result: string[] = [];\n\n for (const value of values) {\n if (seen.has(value)) {\n continue;\n }\n seen.add(value);\n result.push(value);\n }\n\n return result;\n}\n\nfunction safeJson(value: unknown, spacing: number): string | undefined {\n const seen = new WeakSet();\n\n try {\n return JSON.stringify(\n value,\n (_key, entry: unknown) => {\n if (typeof entry === \"bigint\") {\n return `${entry}n`;\n }\n if (typeof entry === \"function\") {\n return `[Function ${entry.name || \"anonymous\"}]`;\n }\n if (typeof entry === \"symbol\") {\n return entry.toString();\n }\n if (entry && typeof entry === \"object\") {\n if (seen.has(entry)) {\n return \"[Circular]\";\n }\n seen.add(entry);\n }\n return entry;\n },\n spacing,\n );\n } catch {\n return undefined;\n }\n}\n\nfunction readFirstString(source: Record<string, unknown>, keys: string[]): string | undefined {\n for (const key of keys) {\n const value = source[key];\n if (typeof value === \"string\" && value.trim()) {\n return value.trim();\n }\n }\n return undefined;\n}\n\nfunction readFirstStringArray(\n source: Record<string, unknown>,\n keys: string[],\n): string[] | undefined {\n for (const key of keys) {\n const value = source[key];\n if (!Array.isArray(value)) {\n continue;\n }\n const entries = value\n .map((entry) => (typeof entry === \"string\" ? entry.trim() : \"\"))\n .filter((entry) => entry.length > 0);\n if (entries.length > 0) {\n return entries;\n }\n }\n return undefined;\n}\n\nfunction summarizeToolInput(rawInput: unknown): string | undefined {\n if (rawInput == null) {\n return undefined;\n }\n\n if (\n typeof rawInput === \"string\" ||\n typeof rawInput === \"number\" ||\n typeof rawInput === \"boolean\"\n ) {\n return toInline(String(rawInput));\n }\n\n const record = asRecord(rawInput);\n if (record) {\n const command = readFirstString(record, [\"command\", \"cmd\", \"program\"]);\n const args = readFirstStringArray(record, [\"args\", \"arguments\"]);\n if (command) {\n const invocation = [command, ...(args ?? [])].join(\" \");\n return toInline(invocation);\n }\n\n const location = readFirstString(record, [\n \"path\",\n \"file\",\n \"filePath\",\n \"filepath\",\n \"target\",\n \"uri\",\n \"url\",\n ]);\n if (location) {\n return toInline(location);\n }\n\n const query = readFirstString(record, [\"query\", \"pattern\", \"text\", \"search\"]);\n if (query) {\n return toInline(query);\n }\n }\n\n const json = safeJson(rawInput, 0);\n return json ? toInline(json) : undefined;\n}\n\nfunction formatLocations(\n locations: Array<ToolCallLocation> | null | undefined,\n): string | undefined {\n if (!locations || locations.length === 0) {\n return undefined;\n }\n\n const unique = new Set<string>();\n for (const location of locations) {\n const path = location.path?.trim();\n if (!path) {\n continue;\n }\n\n const line =\n typeof location.line === \"number\" && Number.isFinite(location.line)\n ? `:${Math.max(1, Math.trunc(location.line))}`\n : \"\";\n unique.add(`${path}${line}`);\n }\n\n const items = [...unique];\n if (items.length === 0) {\n return undefined;\n }\n\n const visible = items.slice(0, MAX_LOCATION_ITEMS);\n const hidden = items.length - visible.length;\n if (hidden <= 0) {\n return visible.join(\", \");\n }\n\n return `${visible.join(\", \")}, +${hidden} more`;\n}\n\nfunction summarizeDiff(path: string, oldText: string | null | undefined, newText: string): string {\n const oldLines = oldText ? oldText.split(\"\\n\").length : 0;\n const newLines = newText.split(\"\\n\").length;\n const delta = newLines - oldLines;\n\n if (delta === 0) {\n return `diff ${path} (line count unchanged)`;\n }\n\n const signedDelta = `${delta > 0 ? \"+\" : \"\"}${delta}`;\n return `diff ${path} (${signedDelta} lines)`;\n}\n\nfunction textFromContentBlock(content: ContentBlock): string | undefined {\n switch (content.type) {\n case \"text\":\n return content.text;\n case \"resource_link\":\n return content.title ?? content.name ?? content.uri;\n case \"resource\": {\n if (\"text\" in content.resource && typeof content.resource.text === \"string\") {\n return content.resource.text;\n }\n const uri = content.resource.uri;\n const mimeType = content.resource.mimeType;\n return `[resource] ${uri}${mimeType ? ` (${mimeType})` : \"\"}`;\n }\n case \"image\":\n return `[image] ${content.mimeType}`;\n case \"audio\":\n return `[audio] ${content.mimeType}`;\n default:\n return undefined;\n }\n}\n\nfunction summarizeToolContent(\n content: Array<ToolCallContent> | null | undefined,\n): string | undefined {\n if (!content || content.length === 0) {\n return undefined;\n }\n\n const fragments: string[] = [];\n\n for (const entry of content) {\n if (entry.type === \"content\") {\n const text = textFromContentBlock(entry.content);\n if (text && text.trim()) {\n fragments.push(text.trimEnd());\n }\n continue;\n }\n\n if (entry.type === \"diff\") {\n fragments.push(summarizeDiff(entry.path, entry.oldText, entry.newText));\n continue;\n }\n\n if (entry.type === \"terminal\") {\n fragments.push(`[terminal] ${entry.terminalId}`);\n }\n }\n\n const unique = dedupeStrings(\n fragments.map((fragment) => fragment.trim()).filter((fragment) => fragment.length > 0),\n );\n if (unique.length === 0) {\n return undefined;\n }\n\n return unique.join(\"\\n\\n\");\n}\n\nfunction extractOutputText(\n value: unknown,\n depth = 0,\n seen = new Set<unknown>(),\n): string | undefined {\n if (value == null) {\n return undefined;\n }\n\n if (typeof value === \"string\") {\n const trimmed = value.trimEnd();\n return trimmed.length > 0 ? trimmed : undefined;\n }\n\n if (typeof value === \"number\" || typeof value === \"boolean\") {\n return String(value);\n }\n\n if (depth >= 4) {\n return undefined;\n }\n\n if (Array.isArray(value)) {\n const parts = value\n .map((entry) => extractOutputText(entry, depth + 1, seen))\n .filter((entry): entry is string => Boolean(entry));\n if (parts.length === 0) {\n return undefined;\n }\n return dedupeStrings(parts).join(\"\\n\");\n }\n\n const record = asRecord(value);\n if (!record) {\n return undefined;\n }\n if (seen.has(record)) {\n return undefined;\n }\n seen.add(record);\n\n const preferred: string[] = [];\n for (const key of OUTPUT_PRIORITY_KEYS) {\n if (!(key in record)) {\n continue;\n }\n const extracted = extractOutputText(record[key], depth + 1, seen);\n if (extracted) {\n preferred.push(extracted);\n }\n }\n\n const uniquePreferred = dedupeStrings(preferred);\n if (uniquePreferred.length > 0) {\n return uniquePreferred.join(\"\\n\");\n }\n\n const json = safeJson(record, 2);\n if (!json || json === \"{}\") {\n return undefined;\n }\n return json;\n}\n\nfunction summarizeToolOutput(\n rawOutput: unknown,\n content: Array<ToolCallContent> | null | undefined,\n): string | undefined {\n const outputFromRaw = extractOutputText(rawOutput);\n const outputFromContent = summarizeToolContent(content);\n\n const fragments = dedupeStrings(\n [outputFromRaw, outputFromContent]\n .map((fragment) => fragment?.trim())\n .filter((fragment): fragment is string => Boolean(fragment)),\n );\n\n if (fragments.length === 0) {\n return undefined;\n }\n\n return fragments.join(\"\\n\\n\");\n}\n\nfunction renderToolOutput(state: ToolRenderState, suppressReads: boolean): string | undefined {\n if (suppressReads && isReadLikeTool(state)) {\n return SUPPRESSED_READ_OUTPUT;\n }\n\n return summarizeToolOutput(state.rawOutput, state.content);\n}\n\nfunction limitOutputBlock(value: string): string {\n const normalized = value.replace(/\\r\\n/g, \"\\n\").trim();\n if (!normalized) {\n return \"\";\n }\n\n const lines = normalized.split(\"\\n\");\n const visible = lines.slice(0, MAX_OUTPUT_LINES);\n let result = visible.join(\"\\n\");\n\n if (lines.length > visible.length) {\n const hidden = lines.length - visible.length;\n result += `\\n... (${hidden} more lines)`;\n }\n\n if (result.length > MAX_OUTPUT_CHARS) {\n result = `${result.slice(0, MAX_OUTPUT_CHARS - 3)}...`;\n }\n\n return result;\n}\n\nclass TextOutputFormatter implements OutputFormatter {\n private readonly stdout: WritableLike;\n private readonly useColor: boolean;\n private readonly suppressReads: boolean;\n private readonly toolStates = new Map<string, ToolRenderState>();\n private thoughtBuffer = \"\";\n private wroteAny = false;\n private atLineStart = true;\n private section: FormatterSection | null = null;\n\n constructor(stdout: WritableLike, suppressReads: boolean) {\n this.stdout = stdout;\n this.useColor = Boolean(stdout.isTTY);\n this.suppressReads = suppressReads;\n }\n\n setContext(_context: OutputFormatterContext): void {\n // no-op for text mode\n }\n\n onAcpMessage(message: AcpJsonRpcMessage): void {\n const notification = extractSessionUpdateNotification(message);\n if (notification) {\n this.renderSessionUpdate(notification);\n return;\n }\n\n const method = extractJsonRpcMethod(message);\n if (method && method !== \"session/prompt\" && method !== \"session/cancel\") {\n this.onClientOperation({\n method: method as ClientOperation[\"method\"],\n status: \"running\",\n summary: method,\n timestamp: new Date().toISOString(),\n });\n return;\n }\n\n const stopReason = parsePromptStopReason(message);\n if (stopReason) {\n this.renderDone(stopReason);\n return;\n }\n\n const errorMessage = parseJsonRpcErrorMessage(message);\n if (errorMessage) {\n this.onError({\n code: \"RUNTIME\",\n origin: \"acp\",\n message: errorMessage,\n });\n }\n }\n\n private renderSessionUpdate(notification: SessionNotification): void {\n const update = notification.update;\n if (update.sessionUpdate !== \"agent_thought_chunk\") {\n this.flushThoughtBuffer();\n }\n\n switch (update.sessionUpdate) {\n case \"agent_message_chunk\": {\n if (update.content.type === \"text\") {\n this.writeAssistantChunk(update.content.text);\n }\n return;\n }\n case \"agent_thought_chunk\": {\n if (update.content.type === \"text\") {\n this.thoughtBuffer += update.content.text;\n }\n return;\n }\n case \"tool_call\": {\n this.renderToolUpdate(update);\n return;\n }\n case \"tool_call_update\": {\n this.renderToolUpdate(update);\n return;\n }\n case \"plan\": {\n this.beginSection(\"plan\");\n this.writeLine(this.bold(\"[plan]\"));\n for (const entry of update.entries) {\n this.writeLine(` - [${entry.status}] ${entry.content}`);\n }\n return;\n }\n default:\n return;\n }\n }\n\n private renderDone(stopReason: string): void {\n this.flushThoughtBuffer();\n this.beginSection(\"done\");\n this.writeLine(this.dim(`[done] ${stopReason}`));\n }\n\n onError(params: {\n code: OutputErrorCode;\n detailCode?: string;\n origin?: OutputErrorOrigin;\n message: string;\n retryable?: boolean;\n acp?: OutputErrorAcpPayload;\n timestamp?: string;\n }): void {\n this.flushThoughtBuffer();\n this.beginSection(\"done\");\n this.writeLine(this.formatAnsi(`[error] ${params.code}: ${params.message}`, \"31\"));\n }\n\n onClientOperation(operation: ClientOperation): void {\n this.flushThoughtBuffer();\n this.beginSection(\"client\");\n\n const normalizedStatus: NormalizedToolStatus =\n operation.status === \"completed\"\n ? \"completed\"\n : operation.status === \"failed\"\n ? \"failed\"\n : \"in_progress\";\n const statusText = this.colorStatus(operation.status, normalizedStatus);\n this.writeLine(`${this.bold(\"[client]\")} ${operation.summary} (${statusText})`);\n if (operation.details && operation.details.trim().length > 0) {\n this.writeLine(\" details:\");\n this.writeLine(indentBlock(operation.details, \" \"));\n }\n }\n\n flush(): void {\n this.flushThoughtBuffer();\n if (!this.atLineStart) {\n this.write(\"\\n\");\n }\n }\n\n private write(chunk: string): void {\n if (!chunk) {\n return;\n }\n this.stdout.write(chunk);\n this.wroteAny = true;\n this.atLineStart = chunk.endsWith(\"\\n\");\n }\n\n private writeLine(line: string): void {\n this.write(`${line}\\n`);\n }\n\n private beginSection(next: Exclude<FormatterSection, \"assistant\">): void {\n if (!this.atLineStart) {\n this.write(\"\\n\");\n }\n if (this.wroteAny) {\n this.write(\"\\n\");\n }\n this.section = next;\n }\n\n private writeAssistantChunk(text: string): void {\n if (!text) {\n return;\n }\n this.section = \"assistant\";\n this.write(text);\n }\n\n private flushThoughtBuffer(): void {\n const thought = truncate(normalizeLineEndings(this.thoughtBuffer).trim(), MAX_THOUGHT_CHARS);\n this.thoughtBuffer = \"\";\n if (!thought) {\n return;\n }\n\n this.beginSection(\"thought\");\n const [firstLine, ...restLines] = thought.split(\"\\n\");\n this.writeLine(this.dim(`[thinking] ${firstLine}`));\n for (const line of restLines) {\n this.writeLine(this.dim(` ${line}`));\n }\n }\n\n private renderToolUpdate(update: ToolCall | ToolCallUpdate): void {\n const state = this.getOrCreateToolState(update.toolCallId);\n this.mergeToolState(state, update);\n\n const status = asStatus(state.status);\n if (isFinalStatus(status)) {\n const signature = this.toolSignature(state);\n if (signature !== state.finalSignature) {\n state.finalSignature = signature;\n this.renderFinalToolState(state, status);\n }\n return;\n }\n\n if (state.startedPrinted) {\n return;\n }\n\n state.startedPrinted = true;\n this.renderStartingToolState(state, status);\n }\n\n private getOrCreateToolState(toolCallId: string): ToolRenderState {\n const existing = this.toolStates.get(toolCallId);\n if (existing) {\n return existing;\n }\n\n const created: ToolRenderState = {\n id: toolCallId,\n startedPrinted: false,\n };\n this.toolStates.set(toolCallId, created);\n return created;\n }\n\n private mergeToolState(state: ToolRenderState, update: ToolCall | ToolCallUpdate): void {\n if (typeof update.title === \"string\" && update.title.trim().length > 0) {\n state.title = update.title;\n }\n\n if (update.status !== undefined) {\n state.status = update.status;\n }\n if (update.kind !== undefined) {\n state.kind = update.kind;\n }\n if (update.locations !== undefined) {\n state.locations = update.locations;\n }\n if (update.rawInput !== undefined) {\n state.rawInput = update.rawInput;\n }\n if (update.rawOutput !== undefined) {\n state.rawOutput = update.rawOutput;\n }\n if (update.content !== undefined) {\n state.content = update.content;\n }\n }\n\n private toolSignature(state: ToolRenderState): string {\n const signaturePayload = {\n title: state.title,\n status: state.status,\n kind: state.kind,\n input: summarizeToolInput(state.rawInput),\n files: formatLocations(state.locations),\n output: renderToolOutput(state, this.suppressReads),\n };\n\n return safeJson(signaturePayload, 0) ?? JSON.stringify(signaturePayload);\n }\n\n private renderStartingToolState(\n state: ToolRenderState,\n status: Exclude<NormalizedToolStatus, \"completed\" | \"failed\">,\n ): void {\n this.beginSection(\"tool\");\n\n const title = state.title ?? state.id;\n const label = status === \"pending\" ? \"pending\" : \"running\";\n const statusText = this.colorStatus(label, status);\n this.writeLine(`${this.bold(\"[tool]\")} ${title} (${statusText})`);\n\n const input = summarizeToolInput(state.rawInput);\n if (input) {\n this.writeLine(` input: ${input}`);\n }\n\n const files = formatLocations(state.locations);\n if (files) {\n this.writeLine(` files: ${files}`);\n }\n }\n\n private renderFinalToolState(state: ToolRenderState, status: \"completed\" | \"failed\"): void {\n this.beginSection(\"tool\");\n\n const title = state.title ?? state.id;\n const statusText = this.colorStatus(toStatusLabel(status), status);\n this.writeLine(`${this.bold(\"[tool]\")} ${title} (${statusText})`);\n\n if (state.kind) {\n this.writeLine(` kind: ${state.kind}`);\n }\n\n const input = summarizeToolInput(state.rawInput);\n if (input) {\n this.writeLine(` input: ${input}`);\n }\n\n const files = formatLocations(state.locations);\n if (files) {\n this.writeLine(` files: ${files}`);\n }\n\n const output = renderToolOutput(state, this.suppressReads);\n if (output) {\n this.writeLine(\" output:\");\n this.writeLine(indentBlock(limitOutputBlock(output), \" \"));\n }\n }\n\n private formatAnsi(text: string, code: string): string {\n if (!this.useColor) {\n return text;\n }\n return `\\u001b[${code}m${text}\\u001b[0m`;\n }\n\n private bold(text: string): string {\n return this.formatAnsi(text, \"1\");\n }\n\n private dim(text: string): string {\n return this.formatAnsi(text, \"2\");\n }\n\n private colorStatus(text: string, status: NormalizedToolStatus): string {\n if (!this.useColor) {\n return text;\n }\n\n switch (status) {\n case \"completed\":\n return this.formatAnsi(text, \"32\");\n case \"failed\":\n return this.formatAnsi(text, \"31\");\n case \"pending\":\n case \"in_progress\":\n case \"unknown\":\n default:\n return this.formatAnsi(text, \"33\");\n }\n }\n}\n\nclass QuietOutputFormatter implements OutputFormatter {\n private readonly stdout: WritableLike;\n private chunks: string[] = [];\n private flushed = false;\n\n constructor(stdout: WritableLike) {\n this.stdout = stdout;\n }\n\n setContext(_context: OutputFormatterContext): void {\n // no-op for quiet mode\n }\n\n onAcpMessage(message: AcpJsonRpcMessage): void {\n const update = extractSessionUpdateNotification(message);\n if (\n update?.update.sessionUpdate === \"agent_message_chunk\" &&\n update.update.content.type === \"text\"\n ) {\n this.chunks.push(update.update.content.text);\n return;\n }\n\n if (parsePromptStopReason(message)) {\n this.flushBufferedOutput();\n }\n }\n\n onError(_params: {\n code: OutputErrorCode;\n detailCode?: string;\n origin?: OutputErrorOrigin;\n message: string;\n retryable?: boolean;\n acp?: OutputErrorAcpPayload;\n timestamp?: string;\n }): void {\n // no-op in quiet mode\n }\n\n flush(): void {\n // no-op for streaming output\n }\n\n private flushBufferedOutput(): void {\n if (this.flushed) {\n return;\n }\n\n this.flushed = true;\n const text = this.chunks.join(\"\");\n this.stdout.write(text.endsWith(\"\\n\") ? text : `${text}\\n`);\n }\n}\n\nexport function createOutputFormatter(\n format: OutputFormat,\n options: OutputFormatterOptions = {},\n): OutputFormatter {\n const stdout = options.stdout ?? process.stdout;\n const suppressReads = options.suppressReads === true;\n\n switch (format) {\n case \"text\":\n return new TextOutputFormatter(stdout, suppressReads);\n case \"json\":\n return createJsonOutputFormatter(stdout, suppressReads, options.jsonContext);\n case \"quiet\":\n return new QuietOutputFormatter(stdout);\n default: {\n const exhaustive: never = format;\n void exhaustive;\n throw new Error(\"Unsupported output format\");\n }\n }\n}\n"],"mappings":"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"}
@@ -1,11 +1,16 @@
1
1
  import { t as __exportAll } from "./rolldown-runtime-CiIaOW0V.js";
2
- import { n as probeQueueOwnerHealth } from "./queue-ipc-EQLpBMKv.js";
3
- import { n as normalizeRuntimeSessionId } from "./runtime-session-id-C544sPPL.js";
2
+ import { N as normalizeRuntimeSessionId, s as probeQueueOwnerHealth } from "./queue-ipc-CE8_QGX3.js";
4
3
  import path from "node:path";
4
+ //#region src/cli/json-output.ts
5
+ function emitJsonResult(format, payload) {
6
+ if (format !== "json") return false;
7
+ process.stdout.write(`${JSON.stringify(payload)}\n`);
8
+ return true;
9
+ }
10
+ //#endregion
5
11
  //#region src/cli/output-render.ts
6
12
  var output_render_exports = /* @__PURE__ */ __exportAll({
7
13
  agentSessionIdPayload: () => agentSessionIdPayload,
8
- emitJsonResult: () => emitJsonResult,
9
14
  formatPromptSessionBannerLine: () => formatPromptSessionBannerLine,
10
15
  printClosedSessionByFormat: () => printClosedSessionByFormat,
11
16
  printCreatedSessionBanner: () => printCreatedSessionBanner,
@@ -26,11 +31,6 @@ function formatRoutedFrom(sessionCwd, currentCwd) {
26
31
  async function resolveSessionConnectionStatus(record) {
27
32
  return (await probeQueueOwnerHealth(record.acpxRecordId)).healthy ? "connected" : "needs reconnect";
28
33
  }
29
- function emitJsonResult(format, payload) {
30
- if (format !== "json") return false;
31
- process.stdout.write(`${JSON.stringify(payload)}\n`);
32
- return true;
33
- }
34
34
  function printSessionsByFormat(sessions, format) {
35
35
  if (format === "json") {
36
36
  process.stdout.write(`${JSON.stringify(sessions)}\n`);
@@ -134,6 +134,6 @@ function agentSessionIdPayload(agentSessionId) {
134
134
  return { agentSessionId: normalized };
135
135
  }
136
136
  //#endregion
137
- export { output_render_exports as n, formatPromptSessionBannerLine as t };
137
+ export { emitJsonResult as i, formatPromptSessionBannerLine as n, output_render_exports as r, agentSessionIdPayload as t };
138
138
 
139
- //# sourceMappingURL=output-render-DEAaMxg8.js.map
139
+ //# sourceMappingURL=output-render-Bz58qaQn.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"output-render-Bz58qaQn.js","names":[],"sources":["../src/cli/json-output.ts","../src/cli/output-render.ts"],"sourcesContent":["import type { OutputFormat } from \"../types.js\";\n\nexport function emitJsonResult(format: OutputFormat, payload: unknown): boolean {\n if (format !== \"json\") {\n return false;\n }\n process.stdout.write(`${JSON.stringify(payload)}\\n`);\n return true;\n}\n","import path from \"node:path\";\nimport { probeQueueOwnerHealth } from \"../queue-ipc.js\";\nimport { normalizeRuntimeSessionId } from \"../runtime-session-id.js\";\nimport type { OutputFormat, SessionRecord } from \"../types.js\";\nimport { emitJsonResult } from \"./json-output.js\";\n\nfunction formatSessionLabel(record: SessionRecord): string {\n return record.name ?? \"cwd\";\n}\n\nfunction formatRoutedFrom(sessionCwd: string, currentCwd: string): string | undefined {\n const relative = path.relative(sessionCwd, currentCwd);\n if (!relative || relative === \".\") {\n return undefined;\n }\n return relative.startsWith(\".\") ? relative : `.${path.sep}${relative}`;\n}\n\ntype SessionConnectionStatus = \"connected\" | \"needs reconnect\";\n\nasync function resolveSessionConnectionStatus(\n record: SessionRecord,\n): Promise<SessionConnectionStatus> {\n const health = await probeQueueOwnerHealth(record.acpxRecordId);\n return health.healthy ? \"connected\" : \"needs reconnect\";\n}\n\nexport function printSessionsByFormat(sessions: SessionRecord[], format: OutputFormat): void {\n if (format === \"json\") {\n process.stdout.write(`${JSON.stringify(sessions)}\\n`);\n return;\n }\n\n if (format === \"quiet\") {\n for (const session of sessions) {\n const closedMarker = session.closed ? \" [closed]\" : \"\";\n process.stdout.write(`${session.acpxRecordId}${closedMarker}\\n`);\n }\n return;\n }\n\n if (sessions.length === 0) {\n process.stdout.write(\"No sessions\\n\");\n return;\n }\n\n for (const session of sessions) {\n const closedMarker = session.closed ? \" [closed]\" : \"\";\n process.stdout.write(\n `${session.acpxRecordId}${closedMarker}\\t${session.name ?? \"-\"}\\t${session.cwd}\\t${session.lastUsedAt}\\n`,\n );\n }\n}\n\nexport function printClosedSessionByFormat(record: SessionRecord, format: OutputFormat): void {\n if (\n emitJsonResult(format, {\n action: \"session_closed\",\n acpxRecordId: record.acpxRecordId,\n acpxSessionId: record.acpSessionId,\n agentSessionId: record.agentSessionId,\n })\n ) {\n return;\n }\n\n if (format === \"quiet\") {\n return;\n }\n\n process.stdout.write(`${record.acpxRecordId}\\n`);\n}\n\nexport function printNewSessionByFormat(\n record: SessionRecord,\n replaced: SessionRecord | undefined,\n format: OutputFormat,\n): void {\n if (\n emitJsonResult(format, {\n action: \"session_ensured\",\n created: true,\n acpxRecordId: record.acpxRecordId,\n acpxSessionId: record.acpSessionId,\n agentSessionId: record.agentSessionId,\n name: record.name,\n replacedSessionId: replaced?.acpxRecordId,\n })\n ) {\n return;\n }\n\n if (format === \"quiet\") {\n process.stdout.write(`${record.acpxRecordId}\\n`);\n return;\n }\n\n if (replaced) {\n process.stdout.write(`${record.acpxRecordId}\\t(replaced ${replaced.acpxRecordId})\\n`);\n return;\n }\n\n process.stdout.write(`${record.acpxRecordId}\\n`);\n}\n\nexport function printEnsuredSessionByFormat(\n record: SessionRecord,\n created: boolean,\n format: OutputFormat,\n): void {\n if (\n emitJsonResult(format, {\n action: \"session_ensured\",\n created,\n acpxRecordId: record.acpxRecordId,\n acpxSessionId: record.acpSessionId,\n agentSessionId: record.agentSessionId,\n name: record.name,\n })\n ) {\n return;\n }\n\n if (format === \"quiet\") {\n process.stdout.write(`${record.acpxRecordId}\\n`);\n return;\n }\n\n const action = created ? \"created\" : \"existing\";\n process.stdout.write(`${record.acpxRecordId}\\t(${action})\\n`);\n}\n\nexport function printQueuedPromptByFormat(\n result: {\n sessionId: string;\n requestId: string;\n },\n format: OutputFormat,\n): void {\n if (\n emitJsonResult(format, {\n action: \"prompt_queued\",\n acpxRecordId: result.sessionId,\n requestId: result.requestId,\n })\n ) {\n return;\n }\n\n if (format === \"quiet\") {\n return;\n }\n\n process.stdout.write(`[queued] ${result.requestId}\\n`);\n}\n\nexport function formatPromptSessionBannerLine(\n record: SessionRecord,\n currentCwd: string,\n connectionStatus: SessionConnectionStatus = \"needs reconnect\",\n): string {\n const label = formatSessionLabel(record);\n const normalizedSessionCwd = path.resolve(record.cwd);\n const normalizedCurrentCwd = path.resolve(currentCwd);\n const routedFrom =\n normalizedSessionCwd === normalizedCurrentCwd\n ? undefined\n : formatRoutedFrom(normalizedSessionCwd, normalizedCurrentCwd);\n const status = connectionStatus;\n\n if (routedFrom) {\n return `[acpx] session ${label} (${record.acpxRecordId}) · ${normalizedSessionCwd} (routed from ${routedFrom}) · agent ${status}`;\n }\n\n return `[acpx] session ${label} (${record.acpxRecordId}) · ${normalizedSessionCwd} · agent ${status}`;\n}\n\nexport async function printPromptSessionBanner(\n record: SessionRecord,\n currentCwd: string,\n format: OutputFormat,\n jsonStrict = false,\n): Promise<void> {\n if (format === \"quiet\" || (jsonStrict && format === \"json\")) {\n return;\n }\n\n const status = await resolveSessionConnectionStatus(record);\n process.stderr.write(`${formatPromptSessionBannerLine(record, currentCwd, status)}\\n`);\n}\n\nexport function printCreatedSessionBanner(\n record: SessionRecord,\n agentName: string,\n format: OutputFormat,\n jsonStrict = false,\n): void {\n if (format === \"quiet\" || (jsonStrict && format === \"json\")) {\n return;\n }\n\n const label = formatSessionLabel(record);\n process.stderr.write(`[acpx] created session ${label} (${record.acpxRecordId})\\n`);\n process.stderr.write(`[acpx] agent: ${agentName}\\n`);\n process.stderr.write(`[acpx] cwd: ${record.cwd}\\n`);\n}\n\nexport function agentSessionIdPayload(agentSessionId: string | undefined): {\n agentSessionId?: string;\n} {\n const normalized = normalizeRuntimeSessionId(agentSessionId);\n if (!normalized) {\n return {};\n }\n\n return { agentSessionId: normalized };\n}\n"],"mappings":";;;;AAEA,SAAgB,eAAe,QAAsB,SAA2B;AAC9E,KAAI,WAAW,OACb,QAAO;AAET,SAAQ,OAAO,MAAM,GAAG,KAAK,UAAU,QAAQ,CAAC,IAAI;AACpD,QAAO;;;;;;;;;;;;;;;ACDT,SAAS,mBAAmB,QAA+B;AACzD,QAAO,OAAO,QAAQ;;AAGxB,SAAS,iBAAiB,YAAoB,YAAwC;CACpF,MAAM,WAAW,KAAK,SAAS,YAAY,WAAW;AACtD,KAAI,CAAC,YAAY,aAAa,IAC5B;AAEF,QAAO,SAAS,WAAW,IAAI,GAAG,WAAW,IAAI,KAAK,MAAM;;AAK9D,eAAe,+BACb,QACkC;AAElC,SADe,MAAM,sBAAsB,OAAO,aAAa,EACjD,UAAU,cAAc;;AAGxC,SAAgB,sBAAsB,UAA2B,QAA4B;AAC3F,KAAI,WAAW,QAAQ;AACrB,UAAQ,OAAO,MAAM,GAAG,KAAK,UAAU,SAAS,CAAC,IAAI;AACrD;;AAGF,KAAI,WAAW,SAAS;AACtB,OAAK,MAAM,WAAW,UAAU;GAC9B,MAAM,eAAe,QAAQ,SAAS,cAAc;AACpD,WAAQ,OAAO,MAAM,GAAG,QAAQ,eAAe,aAAa,IAAI;;AAElE;;AAGF,KAAI,SAAS,WAAW,GAAG;AACzB,UAAQ,OAAO,MAAM,gBAAgB;AACrC;;AAGF,MAAK,MAAM,WAAW,UAAU;EAC9B,MAAM,eAAe,QAAQ,SAAS,cAAc;AACpD,UAAQ,OAAO,MACb,GAAG,QAAQ,eAAe,aAAa,IAAI,QAAQ,QAAQ,IAAI,IAAI,QAAQ,IAAI,IAAI,QAAQ,WAAW,IACvG;;;AAIL,SAAgB,2BAA2B,QAAuB,QAA4B;AAC5F,KACE,eAAe,QAAQ;EACrB,QAAQ;EACR,cAAc,OAAO;EACrB,eAAe,OAAO;EACtB,gBAAgB,OAAO;EACxB,CAAC,CAEF;AAGF,KAAI,WAAW,QACb;AAGF,SAAQ,OAAO,MAAM,GAAG,OAAO,aAAa,IAAI;;AAGlD,SAAgB,wBACd,QACA,UACA,QACM;AACN,KACE,eAAe,QAAQ;EACrB,QAAQ;EACR,SAAS;EACT,cAAc,OAAO;EACrB,eAAe,OAAO;EACtB,gBAAgB,OAAO;EACvB,MAAM,OAAO;EACb,mBAAmB,UAAU;EAC9B,CAAC,CAEF;AAGF,KAAI,WAAW,SAAS;AACtB,UAAQ,OAAO,MAAM,GAAG,OAAO,aAAa,IAAI;AAChD;;AAGF,KAAI,UAAU;AACZ,UAAQ,OAAO,MAAM,GAAG,OAAO,aAAa,cAAc,SAAS,aAAa,KAAK;AACrF;;AAGF,SAAQ,OAAO,MAAM,GAAG,OAAO,aAAa,IAAI;;AAGlD,SAAgB,4BACd,QACA,SACA,QACM;AACN,KACE,eAAe,QAAQ;EACrB,QAAQ;EACR;EACA,cAAc,OAAO;EACrB,eAAe,OAAO;EACtB,gBAAgB,OAAO;EACvB,MAAM,OAAO;EACd,CAAC,CAEF;AAGF,KAAI,WAAW,SAAS;AACtB,UAAQ,OAAO,MAAM,GAAG,OAAO,aAAa,IAAI;AAChD;;CAGF,MAAM,SAAS,UAAU,YAAY;AACrC,SAAQ,OAAO,MAAM,GAAG,OAAO,aAAa,KAAK,OAAO,KAAK;;AAG/D,SAAgB,0BACd,QAIA,QACM;AACN,KACE,eAAe,QAAQ;EACrB,QAAQ;EACR,cAAc,OAAO;EACrB,WAAW,OAAO;EACnB,CAAC,CAEF;AAGF,KAAI,WAAW,QACb;AAGF,SAAQ,OAAO,MAAM,YAAY,OAAO,UAAU,IAAI;;AAGxD,SAAgB,8BACd,QACA,YACA,mBAA4C,mBACpC;CACR,MAAM,QAAQ,mBAAmB,OAAO;CACxC,MAAM,uBAAuB,KAAK,QAAQ,OAAO,IAAI;CACrD,MAAM,uBAAuB,KAAK,QAAQ,WAAW;CACrD,MAAM,aACJ,yBAAyB,uBACrB,KAAA,IACA,iBAAiB,sBAAsB,qBAAqB;CAClE,MAAM,SAAS;AAEf,KAAI,WACF,QAAO,kBAAkB,MAAM,IAAI,OAAO,aAAa,MAAM,qBAAqB,gBAAgB,WAAW,YAAY;AAG3H,QAAO,kBAAkB,MAAM,IAAI,OAAO,aAAa,MAAM,qBAAqB,WAAW;;AAG/F,eAAsB,yBACpB,QACA,YACA,QACA,aAAa,OACE;AACf,KAAI,WAAW,WAAY,cAAc,WAAW,OAClD;CAGF,MAAM,SAAS,MAAM,+BAA+B,OAAO;AAC3D,SAAQ,OAAO,MAAM,GAAG,8BAA8B,QAAQ,YAAY,OAAO,CAAC,IAAI;;AAGxF,SAAgB,0BACd,QACA,WACA,QACA,aAAa,OACP;AACN,KAAI,WAAW,WAAY,cAAc,WAAW,OAClD;CAGF,MAAM,QAAQ,mBAAmB,OAAO;AACxC,SAAQ,OAAO,MAAM,0BAA0B,MAAM,IAAI,OAAO,aAAa,KAAK;AAClF,SAAQ,OAAO,MAAM,iBAAiB,UAAU,IAAI;AACpD,SAAQ,OAAO,MAAM,eAAe,OAAO,IAAI,IAAI;;AAGrD,SAAgB,sBAAsB,gBAEpC;CACA,MAAM,aAAa,0BAA0B,eAAe;AAC5D,KAAI,CAAC,WACH,QAAO,EAAE;AAGX,QAAO,EAAE,gBAAgB,YAAY"}