@code-yeongyu/senpi 2026.5.13 → 2026.5.15

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (156) hide show
  1. package/CHANGELOG.md +1105 -1063
  2. package/dist/core/agent-session.d.ts.map +1 -1
  3. package/dist/core/agent-session.js +12 -5
  4. package/dist/core/agent-session.js.map +1 -1
  5. package/dist/core/bash-executor.d.ts.map +1 -1
  6. package/dist/core/bash-executor.js +1 -1
  7. package/dist/core/bash-executor.js.map +1 -1
  8. package/dist/core/compaction/compaction.d.ts.map +1 -1
  9. package/dist/core/compaction/compaction.js +2 -2
  10. package/dist/core/compaction/compaction.js.map +1 -1
  11. package/dist/core/export-html/index.d.ts.map +1 -1
  12. package/dist/core/export-html/index.js +8 -1
  13. package/dist/core/export-html/index.js.map +1 -1
  14. package/dist/core/extensions/builtin/anthropic-web-search/index.d.ts.map +1 -1
  15. package/dist/core/extensions/builtin/anthropic-web-search/index.js +20 -0
  16. package/dist/core/extensions/builtin/anthropic-web-search/index.js.map +1 -1
  17. package/dist/core/extensions/builtin/background-task/notification.d.ts +1 -1
  18. package/dist/core/extensions/builtin/background-task/notification.d.ts.map +1 -1
  19. package/dist/core/extensions/builtin/background-task/notification.js.map +1 -1
  20. package/dist/core/extensions/builtin/background-task/types.d.ts +2 -2
  21. package/dist/core/extensions/builtin/background-task/types.d.ts.map +1 -1
  22. package/dist/core/extensions/builtin/background-task/types.js +2 -2
  23. package/dist/core/extensions/builtin/background-task/types.js.map +1 -1
  24. package/dist/core/extensions/builtin/index.d.ts.map +1 -1
  25. package/dist/core/extensions/builtin/index.js +0 -18
  26. package/dist/core/extensions/builtin/index.js.map +1 -1
  27. package/dist/core/extensions/builtin/openai-web-search/index.d.ts.map +1 -1
  28. package/dist/core/extensions/builtin/openai-web-search/index.js +28 -0
  29. package/dist/core/extensions/builtin/openai-web-search/index.js.map +1 -1
  30. package/dist/core/extensions/builtin/system-messages.d.ts +6 -6
  31. package/dist/core/extensions/builtin/system-messages.d.ts.map +1 -1
  32. package/dist/core/extensions/builtin/system-messages.js +10 -10
  33. package/dist/core/extensions/builtin/system-messages.js.map +1 -1
  34. package/dist/core/extensions/builtin/todotools/continuation/prompt.d.ts +1 -1
  35. package/dist/core/extensions/builtin/todotools/continuation/prompt.d.ts.map +1 -1
  36. package/dist/core/extensions/builtin/todotools/continuation/prompt.js +1 -1
  37. package/dist/core/extensions/builtin/todotools/continuation/prompt.js.map +1 -1
  38. package/dist/core/extensions/builtin/todotools/state.d.ts +1 -1
  39. package/dist/core/extensions/builtin/todotools/state.d.ts.map +1 -1
  40. package/dist/core/extensions/builtin/todotools/state.js +2 -2
  41. package/dist/core/extensions/builtin/todotools/state.js.map +1 -1
  42. package/dist/core/extensions/builtin/todotools/system-messages.d.ts +3 -3
  43. package/dist/core/extensions/builtin/todotools/system-messages.d.ts.map +1 -1
  44. package/dist/core/extensions/builtin/todotools/system-messages.js +6 -6
  45. package/dist/core/extensions/builtin/todotools/system-messages.js.map +1 -1
  46. package/dist/core/resource-loader.d.ts.map +1 -1
  47. package/dist/core/resource-loader.js +0 -9
  48. package/dist/core/resource-loader.js.map +1 -1
  49. package/dist/core/sdk.d.ts +1 -1
  50. package/dist/core/sdk.d.ts.map +1 -1
  51. package/dist/core/sdk.js +1 -1
  52. package/dist/core/sdk.js.map +1 -1
  53. package/dist/core/session-manager.d.ts.map +1 -1
  54. package/dist/core/session-manager.js.map +1 -1
  55. package/dist/core/tools/render-utils.d.ts.map +1 -1
  56. package/dist/core/tools/render-utils.js +1 -1
  57. package/dist/core/tools/render-utils.js.map +1 -1
  58. package/dist/main.d.ts.map +1 -1
  59. package/dist/main.js +3 -2
  60. package/dist/main.js.map +1 -1
  61. package/dist/modes/interactive/components/assistant-message.d.ts +0 -3
  62. package/dist/modes/interactive/components/assistant-message.d.ts.map +1 -1
  63. package/dist/modes/interactive/components/assistant-message.js +3 -22
  64. package/dist/modes/interactive/components/assistant-message.js.map +1 -1
  65. package/dist/modes/interactive/components/bash-execution.d.ts.map +1 -1
  66. package/dist/modes/interactive/components/bash-execution.js +1 -1
  67. package/dist/modes/interactive/components/bash-execution.js.map +1 -1
  68. package/dist/modes/interactive/components/extension-selector.d.ts +2 -0
  69. package/dist/modes/interactive/components/extension-selector.d.ts.map +1 -1
  70. package/dist/modes/interactive/components/extension-selector.js +6 -1
  71. package/dist/modes/interactive/components/extension-selector.js.map +1 -1
  72. package/dist/modes/interactive/interactive-mode.d.ts +12 -0
  73. package/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  74. package/dist/modes/interactive/interactive-mode.js +43 -5
  75. package/dist/modes/interactive/interactive-mode.js.map +1 -1
  76. package/dist/modes/interactive/theme/theme.d.ts.map +1 -1
  77. package/dist/modes/interactive/theme/theme.js +2 -2
  78. package/dist/modes/interactive/theme/theme.js.map +1 -1
  79. package/dist/modes/print-mode.d.ts.map +1 -1
  80. package/dist/modes/print-mode.js +3 -11
  81. package/dist/modes/print-mode.js.map +1 -1
  82. package/dist/modes/provider-native-rendering.d.ts +5 -0
  83. package/dist/modes/provider-native-rendering.d.ts.map +1 -0
  84. package/dist/modes/provider-native-rendering.js +247 -0
  85. package/dist/modes/provider-native-rendering.js.map +1 -0
  86. package/dist/utils/ansi.d.ts +2 -0
  87. package/dist/utils/ansi.d.ts.map +1 -0
  88. package/dist/utils/ansi.js +52 -0
  89. package/dist/utils/ansi.js.map +1 -0
  90. package/dist/utils/html.d.ts +7 -0
  91. package/dist/utils/html.d.ts.map +1 -0
  92. package/dist/utils/html.js +40 -0
  93. package/dist/utils/html.js.map +1 -0
  94. package/dist/utils/mime.d.ts +1 -0
  95. package/dist/utils/mime.d.ts.map +1 -1
  96. package/dist/utils/mime.js +59 -16
  97. package/dist/utils/mime.js.map +1 -1
  98. package/dist/utils/syntax-highlight.d.ts +12 -0
  99. package/dist/utils/syntax-highlight.d.ts.map +1 -0
  100. package/dist/utils/syntax-highlight.js +118 -0
  101. package/dist/utils/syntax-highlight.js.map +1 -0
  102. package/dist/utils/tools-manager.d.ts.map +1 -1
  103. package/dist/utils/tools-manager.js +76 -7
  104. package/dist/utils/tools-manager.js.map +1 -1
  105. package/docs/agents.md +1 -1
  106. package/docs/sdk.md +25 -43
  107. package/examples/sdk/01-minimal.ts +14 -10
  108. package/examples/sdk/02-custom-model.ts +12 -8
  109. package/examples/sdk/03-custom-prompt.ts +24 -16
  110. package/examples/sdk/04-skills.ts +2 -2
  111. package/examples/sdk/05-tools.ts +8 -4
  112. package/examples/sdk/06-extensions.ts +11 -7
  113. package/examples/sdk/07-context-files.ts +2 -2
  114. package/examples/sdk/08-prompt-templates.ts +2 -2
  115. package/examples/sdk/09-api-keys-and-oauth.ts +8 -4
  116. package/examples/sdk/10-settings.ts +4 -4
  117. package/examples/sdk/11-sessions.ts +4 -0
  118. package/examples/sdk/12-full-control.ts +11 -7
  119. package/examples/sdk/README.md +6 -9
  120. package/package.json +7 -12
  121. package/dist/core/extensions/builtin/anthropic-code-execution/index.d.ts +0 -7
  122. package/dist/core/extensions/builtin/anthropic-code-execution/index.d.ts.map +0 -1
  123. package/dist/core/extensions/builtin/anthropic-code-execution/index.js +0 -79
  124. package/dist/core/extensions/builtin/anthropic-code-execution/index.js.map +0 -1
  125. package/dist/core/extensions/builtin/anthropic-computer-use/index.d.ts +0 -53
  126. package/dist/core/extensions/builtin/anthropic-computer-use/index.d.ts.map +0 -1
  127. package/dist/core/extensions/builtin/anthropic-computer-use/index.js +0 -676
  128. package/dist/core/extensions/builtin/anthropic-computer-use/index.js.map +0 -1
  129. package/dist/core/extensions/builtin/anthropic-text-editor/index.d.ts +0 -25
  130. package/dist/core/extensions/builtin/anthropic-text-editor/index.d.ts.map +0 -1
  131. package/dist/core/extensions/builtin/anthropic-text-editor/index.js +0 -244
  132. package/dist/core/extensions/builtin/anthropic-text-editor/index.js.map +0 -1
  133. package/dist/core/extensions/builtin/anthropic-tool-search/index.d.ts +0 -6
  134. package/dist/core/extensions/builtin/anthropic-tool-search/index.d.ts.map +0 -1
  135. package/dist/core/extensions/builtin/anthropic-tool-search/index.js +0 -112
  136. package/dist/core/extensions/builtin/anthropic-tool-search/index.js.map +0 -1
  137. package/dist/core/extensions/builtin/google-code-execution/index.d.ts +0 -7
  138. package/dist/core/extensions/builtin/google-code-execution/index.d.ts.map +0 -1
  139. package/dist/core/extensions/builtin/google-code-execution/index.js +0 -73
  140. package/dist/core/extensions/builtin/google-code-execution/index.js.map +0 -1
  141. package/dist/core/extensions/builtin/google-google-search/index.d.ts +0 -7
  142. package/dist/core/extensions/builtin/google-google-search/index.d.ts.map +0 -1
  143. package/dist/core/extensions/builtin/google-google-search/index.js +0 -83
  144. package/dist/core/extensions/builtin/google-google-search/index.js.map +0 -1
  145. package/dist/core/extensions/builtin/google-url-context/index.d.ts +0 -7
  146. package/dist/core/extensions/builtin/google-url-context/index.d.ts.map +0 -1
  147. package/dist/core/extensions/builtin/google-url-context/index.js +0 -82
  148. package/dist/core/extensions/builtin/google-url-context/index.js.map +0 -1
  149. package/dist/core/extensions/builtin/openai-api-parallel-tool-calls/index.d.ts +0 -6
  150. package/dist/core/extensions/builtin/openai-api-parallel-tool-calls/index.d.ts.map +0 -1
  151. package/dist/core/extensions/builtin/openai-api-parallel-tool-calls/index.js +0 -57
  152. package/dist/core/extensions/builtin/openai-api-parallel-tool-calls/index.js.map +0 -1
  153. package/dist/core/extensions/builtin/openai-code-interpreter/index.d.ts +0 -10
  154. package/dist/core/extensions/builtin/openai-code-interpreter/index.d.ts.map +0 -1
  155. package/dist/core/extensions/builtin/openai-code-interpreter/index.js +0 -95
  156. package/dist/core/extensions/builtin/openai-code-interpreter/index.js.map +0 -1
@@ -1 +1 @@
1
- {"version":3,"file":"print-mode.d.ts","sourceRoot":"","sources":["../../src/modes/print-mode.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,KAAK,EAAoB,YAAY,EAAE,MAAM,uBAAuB,CAAC;AAC5E,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,kCAAkC,CAAC;AAI5E;;GAEG;AACH,MAAM,WAAW,gBAAgB;IAChC,yEAAyE;IACzE,IAAI,EAAE,MAAM,GAAG,MAAM,CAAC;IACtB,+DAA+D;IAC/D,QAAQ,CAAC,EAAE,MAAM,EAAE,CAAC;IACpB,wDAAwD;IACxD,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,8CAA8C;IAC9C,aAAa,CAAC,EAAE,YAAY,EAAE,CAAC;CAC/B;AAED;;;GAGG;AACH,wBAAsB,YAAY,CAAC,WAAW,EAAE,mBAAmB,EAAE,OAAO,EAAE,gBAAgB,GAAG,OAAO,CAAC,MAAM,CAAC,CAkI/G","sourcesContent":["/**\n * Print mode (single-shot): Send prompts, output result, exit.\n *\n * Used for:\n * - `pi -p \"prompt\"` - text output\n * - `senpi --mode json \"prompt\"` - JSON event stream\n */\n\nimport type { AssistantMessage, ImageContent } from \"@earendil-works/pi-ai\";\nimport type { AgentSessionRuntime } from \"../core/agent-session-runtime.js\";\nimport { flushRawStdout, writeRawStdout } from \"../core/output-guard.js\";\nimport { killTrackedDetachedChildren } from \"../utils/shell.js\";\n\n/**\n * Options for print mode.\n */\nexport interface PrintModeOptions {\n\t/** Output mode: \"text\" for final response only, \"json\" for all events */\n\tmode: \"text\" | \"json\";\n\t/** Array of additional prompts to send after initialMessage */\n\tmessages?: string[];\n\t/** First message to send (may contain @file content) */\n\tinitialMessage?: string;\n\t/** Images to attach to the initial message */\n\tinitialImages?: ImageContent[];\n}\n\n/**\n * Run in print (single-shot) mode.\n * Sends prompts to the agent and outputs the result.\n */\nexport async function runPrintMode(runtimeHost: AgentSessionRuntime, options: PrintModeOptions): Promise<number> {\n\tconst { mode, messages = [], initialMessage, initialImages } = options;\n\tlet exitCode = 0;\n\tlet session = runtimeHost.session;\n\tlet unsubscribe: (() => void) | undefined;\n\tlet disposed = false;\n\tconst signalCleanupHandlers: Array<() => void> = [];\n\n\tconst disposeRuntime = async (): Promise<void> => {\n\t\tif (disposed) return;\n\t\tdisposed = true;\n\t\tunsubscribe?.();\n\t\tawait runtimeHost.dispose();\n\t};\n\n\tconst registerSignalHandlers = (): void => {\n\t\tconst signals: NodeJS.Signals[] = [\"SIGTERM\"];\n\t\tif (process.platform !== \"win32\") {\n\t\t\tsignals.push(\"SIGHUP\");\n\t\t}\n\n\t\tfor (const signal of signals) {\n\t\t\tconst handler = () => {\n\t\t\t\tkillTrackedDetachedChildren();\n\t\t\t\tvoid disposeRuntime().finally(() => {\n\t\t\t\t\tprocess.exit(signal === \"SIGHUP\" ? 129 : 143);\n\t\t\t\t});\n\t\t\t};\n\t\t\tprocess.on(signal, handler);\n\t\t\tsignalCleanupHandlers.push(() => process.off(signal, handler));\n\t\t}\n\t};\n\n\tregisterSignalHandlers();\n\n\truntimeHost.setRebindSession(async () => {\n\t\tawait rebindSession();\n\t});\n\n\tconst rebindSession = async (): Promise<void> => {\n\t\tsession = runtimeHost.session;\n\t\tawait session.bindExtensions({\n\t\t\tcommandContextActions: {\n\t\t\t\twaitForIdle: () => session.agent.waitForIdle(),\n\t\t\t\tnewSession: async (newSessionOptions) => runtimeHost.newSession(newSessionOptions),\n\t\t\t\tfork: async (entryId, forkOptions) => {\n\t\t\t\t\tconst result = await runtimeHost.fork(entryId, forkOptions);\n\t\t\t\t\treturn { cancelled: result.cancelled };\n\t\t\t\t},\n\t\t\t\tnavigateTree: async (targetId, navigateOptions) => {\n\t\t\t\t\tconst result = await session.navigateTree(targetId, {\n\t\t\t\t\t\tsummarize: navigateOptions?.summarize,\n\t\t\t\t\t\tcustomInstructions: navigateOptions?.customInstructions,\n\t\t\t\t\t\treplaceInstructions: navigateOptions?.replaceInstructions,\n\t\t\t\t\t\tlabel: navigateOptions?.label,\n\t\t\t\t\t});\n\t\t\t\t\treturn { cancelled: result.cancelled };\n\t\t\t\t},\n\t\t\t\tswitchSession: async (sessionPath, switchOptions) => {\n\t\t\t\t\treturn runtimeHost.switchSession(sessionPath, switchOptions);\n\t\t\t\t},\n\t\t\t\treload: async () => {\n\t\t\t\t\tawait session.reload();\n\t\t\t\t},\n\t\t\t},\n\t\t\tonError: (err) => {\n\t\t\t\tconsole.error(`Extension error (${err.extensionPath}): ${err.error}`);\n\t\t\t},\n\t\t});\n\n\t\tunsubscribe?.();\n\t\tunsubscribe = session.subscribe((event) => {\n\t\t\tif (mode === \"json\") {\n\t\t\t\twriteRawStdout(`${JSON.stringify(event)}\\n`);\n\t\t\t}\n\t\t});\n\t};\n\n\ttry {\n\t\tif (mode === \"json\") {\n\t\t\tconst header = session.sessionManager.getHeader();\n\t\t\tif (header) {\n\t\t\t\twriteRawStdout(`${JSON.stringify(header)}\\n`);\n\t\t\t}\n\t\t}\n\n\t\tawait rebindSession();\n\n\t\tif (initialMessage) {\n\t\t\tawait session.prompt(initialMessage, { images: initialImages });\n\t\t}\n\n\t\tfor (const message of messages) {\n\t\t\tawait session.prompt(message);\n\t\t}\n\n\t\tif (mode === \"text\") {\n\t\t\tconst state = session.state;\n\t\t\tconst lastMessage = state.messages[state.messages.length - 1];\n\n\t\t\tif (lastMessage?.role === \"assistant\") {\n\t\t\t\tconst assistantMsg = lastMessage as AssistantMessage;\n\t\t\t\tif (assistantMsg.stopReason === \"error\" || assistantMsg.stopReason === \"aborted\") {\n\t\t\t\t\tconsole.error(assistantMsg.errorMessage || `Request ${assistantMsg.stopReason}`);\n\t\t\t\t\texitCode = 1;\n\t\t\t\t} else {\n\t\t\t\t\tfor (const content of assistantMsg.content) {\n\t\t\t\t\t\tif (content.type === \"text\") {\n\t\t\t\t\t\t\twriteRawStdout(`${content.text}\\n`);\n\t\t\t\t\t\t} else if (content.type === \"providerNative\") {\n\t\t\t\t\t\t\tconst providerPrefix = assistantMsg.provider ? `${assistantMsg.provider} · ` : \"\";\n\t\t\t\t\t\t\twriteRawStdout(`▸ ${providerPrefix}providerNative · ${content.subtype}\\n`);\n\t\t\t\t\t\t\twriteRawStdout(`${stringifyProviderNative(content.raw)}\\n`);\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn exitCode;\n\t} catch (error: unknown) {\n\t\tconsole.error(error instanceof Error ? error.message : String(error));\n\t\treturn 1;\n\t} finally {\n\t\tfor (const cleanup of signalCleanupHandlers) {\n\t\t\tcleanup();\n\t\t}\n\t\tawait disposeRuntime();\n\t\tawait flushRawStdout();\n\t}\n}\n\nfunction stringifyProviderNative(raw: unknown): string {\n\ttry {\n\t\treturn JSON.stringify(raw, null, 2) ?? \"null\";\n\t} catch {\n\t\treturn String(raw);\n\t}\n}\n"]}
1
+ {"version":3,"file":"print-mode.d.ts","sourceRoot":"","sources":["../../src/modes/print-mode.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,KAAK,EAAoB,YAAY,EAAE,MAAM,uBAAuB,CAAC;AAC5E,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,kCAAkC,CAAC;AAK5E;;GAEG;AACH,MAAM,WAAW,gBAAgB;IAChC,yEAAyE;IACzE,IAAI,EAAE,MAAM,GAAG,MAAM,CAAC;IACtB,+DAA+D;IAC/D,QAAQ,CAAC,EAAE,MAAM,EAAE,CAAC;IACpB,wDAAwD;IACxD,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,8CAA8C;IAC9C,aAAa,CAAC,EAAE,YAAY,EAAE,CAAC;CAC/B;AAED;;;GAGG;AACH,wBAAsB,YAAY,CAAC,WAAW,EAAE,mBAAmB,EAAE,OAAO,EAAE,gBAAgB,GAAG,OAAO,CAAC,MAAM,CAAC,CAiI/G","sourcesContent":["/**\n * Print mode (single-shot): Send prompts, output result, exit.\n *\n * Used for:\n * - `pi -p \"prompt\"` - text output\n * - `senpi --mode json \"prompt\"` - JSON event stream\n */\n\nimport type { AssistantMessage, ImageContent } from \"@earendil-works/pi-ai\";\nimport type { AgentSessionRuntime } from \"../core/agent-session-runtime.js\";\nimport { flushRawStdout, writeRawStdout } from \"../core/output-guard.js\";\nimport { killTrackedDetachedChildren } from \"../utils/shell.js\";\nimport { formatProviderNativeBody, formatProviderNativeSummary } from \"./provider-native-rendering.js\";\n\n/**\n * Options for print mode.\n */\nexport interface PrintModeOptions {\n\t/** Output mode: \"text\" for final response only, \"json\" for all events */\n\tmode: \"text\" | \"json\";\n\t/** Array of additional prompts to send after initialMessage */\n\tmessages?: string[];\n\t/** First message to send (may contain @file content) */\n\tinitialMessage?: string;\n\t/** Images to attach to the initial message */\n\tinitialImages?: ImageContent[];\n}\n\n/**\n * Run in print (single-shot) mode.\n * Sends prompts to the agent and outputs the result.\n */\nexport async function runPrintMode(runtimeHost: AgentSessionRuntime, options: PrintModeOptions): Promise<number> {\n\tconst { mode, messages = [], initialMessage, initialImages } = options;\n\tlet exitCode = 0;\n\tlet session = runtimeHost.session;\n\tlet unsubscribe: (() => void) | undefined;\n\tlet disposed = false;\n\tconst signalCleanupHandlers: Array<() => void> = [];\n\n\tconst disposeRuntime = async (): Promise<void> => {\n\t\tif (disposed) return;\n\t\tdisposed = true;\n\t\tunsubscribe?.();\n\t\tawait runtimeHost.dispose();\n\t};\n\n\tconst registerSignalHandlers = (): void => {\n\t\tconst signals: NodeJS.Signals[] = [\"SIGTERM\"];\n\t\tif (process.platform !== \"win32\") {\n\t\t\tsignals.push(\"SIGHUP\");\n\t\t}\n\n\t\tfor (const signal of signals) {\n\t\t\tconst handler = () => {\n\t\t\t\tkillTrackedDetachedChildren();\n\t\t\t\tvoid disposeRuntime().finally(() => {\n\t\t\t\t\tprocess.exit(signal === \"SIGHUP\" ? 129 : 143);\n\t\t\t\t});\n\t\t\t};\n\t\t\tprocess.on(signal, handler);\n\t\t\tsignalCleanupHandlers.push(() => process.off(signal, handler));\n\t\t}\n\t};\n\n\tregisterSignalHandlers();\n\n\truntimeHost.setRebindSession(async () => {\n\t\tawait rebindSession();\n\t});\n\n\tconst rebindSession = async (): Promise<void> => {\n\t\tsession = runtimeHost.session;\n\t\tawait session.bindExtensions({\n\t\t\tcommandContextActions: {\n\t\t\t\twaitForIdle: () => session.agent.waitForIdle(),\n\t\t\t\tnewSession: async (newSessionOptions) => runtimeHost.newSession(newSessionOptions),\n\t\t\t\tfork: async (entryId, forkOptions) => {\n\t\t\t\t\tconst result = await runtimeHost.fork(entryId, forkOptions);\n\t\t\t\t\treturn { cancelled: result.cancelled };\n\t\t\t\t},\n\t\t\t\tnavigateTree: async (targetId, navigateOptions) => {\n\t\t\t\t\tconst result = await session.navigateTree(targetId, {\n\t\t\t\t\t\tsummarize: navigateOptions?.summarize,\n\t\t\t\t\t\tcustomInstructions: navigateOptions?.customInstructions,\n\t\t\t\t\t\treplaceInstructions: navigateOptions?.replaceInstructions,\n\t\t\t\t\t\tlabel: navigateOptions?.label,\n\t\t\t\t\t});\n\t\t\t\t\treturn { cancelled: result.cancelled };\n\t\t\t\t},\n\t\t\t\tswitchSession: async (sessionPath, switchOptions) => {\n\t\t\t\t\treturn runtimeHost.switchSession(sessionPath, switchOptions);\n\t\t\t\t},\n\t\t\t\treload: async () => {\n\t\t\t\t\tawait session.reload();\n\t\t\t\t},\n\t\t\t},\n\t\t\tonError: (err) => {\n\t\t\t\tconsole.error(`Extension error (${err.extensionPath}): ${err.error}`);\n\t\t\t},\n\t\t});\n\n\t\tunsubscribe?.();\n\t\tunsubscribe = session.subscribe((event) => {\n\t\t\tif (mode === \"json\") {\n\t\t\t\twriteRawStdout(`${JSON.stringify(event)}\\n`);\n\t\t\t}\n\t\t});\n\t};\n\n\ttry {\n\t\tif (mode === \"json\") {\n\t\t\tconst header = session.sessionManager.getHeader();\n\t\t\tif (header) {\n\t\t\t\twriteRawStdout(`${JSON.stringify(header)}\\n`);\n\t\t\t}\n\t\t}\n\n\t\tawait rebindSession();\n\n\t\tif (initialMessage) {\n\t\t\tawait session.prompt(initialMessage, { images: initialImages });\n\t\t}\n\n\t\tfor (const message of messages) {\n\t\t\tawait session.prompt(message);\n\t\t}\n\n\t\tif (mode === \"text\") {\n\t\t\tconst state = session.state;\n\t\t\tconst lastMessage = state.messages[state.messages.length - 1];\n\n\t\t\tif (lastMessage?.role === \"assistant\") {\n\t\t\t\tconst assistantMsg = lastMessage as AssistantMessage;\n\t\t\t\tif (assistantMsg.stopReason === \"error\" || assistantMsg.stopReason === \"aborted\") {\n\t\t\t\t\tconsole.error(assistantMsg.errorMessage || `Request ${assistantMsg.stopReason}`);\n\t\t\t\t\texitCode = 1;\n\t\t\t\t} else {\n\t\t\t\t\tfor (const content of assistantMsg.content) {\n\t\t\t\t\t\tif (content.type === \"text\") {\n\t\t\t\t\t\t\twriteRawStdout(`${content.text}\\n`);\n\t\t\t\t\t\t} else if (content.type === \"providerNative\") {\n\t\t\t\t\t\t\twriteRawStdout(`${formatProviderNativeSummary(assistantMsg, content, false)}\\n`);\n\t\t\t\t\t\t\twriteRawStdout(`${formatProviderNativeBody(content, false)}\\n`);\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn exitCode;\n\t} catch (error: unknown) {\n\t\tconsole.error(error instanceof Error ? error.message : String(error));\n\t\treturn 1;\n\t} finally {\n\t\tfor (const cleanup of signalCleanupHandlers) {\n\t\t\tcleanup();\n\t\t}\n\t\tawait disposeRuntime();\n\t\tawait flushRawStdout();\n\t}\n}\n"]}
@@ -7,6 +7,7 @@
7
7
  */
8
8
  import { flushRawStdout, writeRawStdout } from "../core/output-guard.js";
9
9
  import { killTrackedDetachedChildren } from "../utils/shell.js";
10
+ import { formatProviderNativeBody, formatProviderNativeSummary } from "./provider-native-rendering.js";
10
11
  /**
11
12
  * Run in print (single-shot) mode.
12
13
  * Sends prompts to the agent and outputs the result.
@@ -111,9 +112,8 @@ export async function runPrintMode(runtimeHost, options) {
111
112
  writeRawStdout(`${content.text}\n`);
112
113
  }
113
114
  else if (content.type === "providerNative") {
114
- const providerPrefix = assistantMsg.provider ? `${assistantMsg.provider} · ` : "";
115
- writeRawStdout(`▸ ${providerPrefix}providerNative · ${content.subtype}\n`);
116
- writeRawStdout(`${stringifyProviderNative(content.raw)}\n`);
115
+ writeRawStdout(`${formatProviderNativeSummary(assistantMsg, content, false)}\n`);
116
+ writeRawStdout(`${formatProviderNativeBody(content, false)}\n`);
117
117
  }
118
118
  }
119
119
  }
@@ -133,12 +133,4 @@ export async function runPrintMode(runtimeHost, options) {
133
133
  await flushRawStdout();
134
134
  }
135
135
  }
136
- function stringifyProviderNative(raw) {
137
- try {
138
- return JSON.stringify(raw, null, 2) ?? "null";
139
- }
140
- catch {
141
- return String(raw);
142
- }
143
- }
144
136
  //# sourceMappingURL=print-mode.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"print-mode.js","sourceRoot":"","sources":["../../src/modes/print-mode.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Print mode (single-shot): Send prompts, output result, exit.\n *\n * Used for:\n * - `pi -p \"prompt\"` - text output\n * - `senpi --mode json \"prompt\"` - JSON event stream\n */\n\nimport type { AssistantMessage, ImageContent } from \"@earendil-works/pi-ai\";\nimport type { AgentSessionRuntime } from \"../core/agent-session-runtime.js\";\nimport { flushRawStdout, writeRawStdout } from \"../core/output-guard.js\";\nimport { killTrackedDetachedChildren } from \"../utils/shell.js\";\n\n/**\n * Options for print mode.\n */\nexport interface PrintModeOptions {\n\t/** Output mode: \"text\" for final response only, \"json\" for all events */\n\tmode: \"text\" | \"json\";\n\t/** Array of additional prompts to send after initialMessage */\n\tmessages?: string[];\n\t/** First message to send (may contain @file content) */\n\tinitialMessage?: string;\n\t/** Images to attach to the initial message */\n\tinitialImages?: ImageContent[];\n}\n\n/**\n * Run in print (single-shot) mode.\n * Sends prompts to the agent and outputs the result.\n */\nexport async function runPrintMode(runtimeHost: AgentSessionRuntime, options: PrintModeOptions): Promise<number> {\n\tconst { mode, messages = [], initialMessage, initialImages } = options;\n\tlet exitCode = 0;\n\tlet session = runtimeHost.session;\n\tlet unsubscribe: (() => void) | undefined;\n\tlet disposed = false;\n\tconst signalCleanupHandlers: Array<() => void> = [];\n\n\tconst disposeRuntime = async (): Promise<void> => {\n\t\tif (disposed) return;\n\t\tdisposed = true;\n\t\tunsubscribe?.();\n\t\tawait runtimeHost.dispose();\n\t};\n\n\tconst registerSignalHandlers = (): void => {\n\t\tconst signals: NodeJS.Signals[] = [\"SIGTERM\"];\n\t\tif (process.platform !== \"win32\") {\n\t\t\tsignals.push(\"SIGHUP\");\n\t\t}\n\n\t\tfor (const signal of signals) {\n\t\t\tconst handler = () => {\n\t\t\t\tkillTrackedDetachedChildren();\n\t\t\t\tvoid disposeRuntime().finally(() => {\n\t\t\t\t\tprocess.exit(signal === \"SIGHUP\" ? 129 : 143);\n\t\t\t\t});\n\t\t\t};\n\t\t\tprocess.on(signal, handler);\n\t\t\tsignalCleanupHandlers.push(() => process.off(signal, handler));\n\t\t}\n\t};\n\n\tregisterSignalHandlers();\n\n\truntimeHost.setRebindSession(async () => {\n\t\tawait rebindSession();\n\t});\n\n\tconst rebindSession = async (): Promise<void> => {\n\t\tsession = runtimeHost.session;\n\t\tawait session.bindExtensions({\n\t\t\tcommandContextActions: {\n\t\t\t\twaitForIdle: () => session.agent.waitForIdle(),\n\t\t\t\tnewSession: async (newSessionOptions) => runtimeHost.newSession(newSessionOptions),\n\t\t\t\tfork: async (entryId, forkOptions) => {\n\t\t\t\t\tconst result = await runtimeHost.fork(entryId, forkOptions);\n\t\t\t\t\treturn { cancelled: result.cancelled };\n\t\t\t\t},\n\t\t\t\tnavigateTree: async (targetId, navigateOptions) => {\n\t\t\t\t\tconst result = await session.navigateTree(targetId, {\n\t\t\t\t\t\tsummarize: navigateOptions?.summarize,\n\t\t\t\t\t\tcustomInstructions: navigateOptions?.customInstructions,\n\t\t\t\t\t\treplaceInstructions: navigateOptions?.replaceInstructions,\n\t\t\t\t\t\tlabel: navigateOptions?.label,\n\t\t\t\t\t});\n\t\t\t\t\treturn { cancelled: result.cancelled };\n\t\t\t\t},\n\t\t\t\tswitchSession: async (sessionPath, switchOptions) => {\n\t\t\t\t\treturn runtimeHost.switchSession(sessionPath, switchOptions);\n\t\t\t\t},\n\t\t\t\treload: async () => {\n\t\t\t\t\tawait session.reload();\n\t\t\t\t},\n\t\t\t},\n\t\t\tonError: (err) => {\n\t\t\t\tconsole.error(`Extension error (${err.extensionPath}): ${err.error}`);\n\t\t\t},\n\t\t});\n\n\t\tunsubscribe?.();\n\t\tunsubscribe = session.subscribe((event) => {\n\t\t\tif (mode === \"json\") {\n\t\t\t\twriteRawStdout(`${JSON.stringify(event)}\\n`);\n\t\t\t}\n\t\t});\n\t};\n\n\ttry {\n\t\tif (mode === \"json\") {\n\t\t\tconst header = session.sessionManager.getHeader();\n\t\t\tif (header) {\n\t\t\t\twriteRawStdout(`${JSON.stringify(header)}\\n`);\n\t\t\t}\n\t\t}\n\n\t\tawait rebindSession();\n\n\t\tif (initialMessage) {\n\t\t\tawait session.prompt(initialMessage, { images: initialImages });\n\t\t}\n\n\t\tfor (const message of messages) {\n\t\t\tawait session.prompt(message);\n\t\t}\n\n\t\tif (mode === \"text\") {\n\t\t\tconst state = session.state;\n\t\t\tconst lastMessage = state.messages[state.messages.length - 1];\n\n\t\t\tif (lastMessage?.role === \"assistant\") {\n\t\t\t\tconst assistantMsg = lastMessage as AssistantMessage;\n\t\t\t\tif (assistantMsg.stopReason === \"error\" || assistantMsg.stopReason === \"aborted\") {\n\t\t\t\t\tconsole.error(assistantMsg.errorMessage || `Request ${assistantMsg.stopReason}`);\n\t\t\t\t\texitCode = 1;\n\t\t\t\t} else {\n\t\t\t\t\tfor (const content of assistantMsg.content) {\n\t\t\t\t\t\tif (content.type === \"text\") {\n\t\t\t\t\t\t\twriteRawStdout(`${content.text}\\n`);\n\t\t\t\t\t\t} else if (content.type === \"providerNative\") {\n\t\t\t\t\t\t\tconst providerPrefix = assistantMsg.provider ? `${assistantMsg.provider} · ` : \"\";\n\t\t\t\t\t\t\twriteRawStdout(`▸ ${providerPrefix}providerNative · ${content.subtype}\\n`);\n\t\t\t\t\t\t\twriteRawStdout(`${stringifyProviderNative(content.raw)}\\n`);\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn exitCode;\n\t} catch (error: unknown) {\n\t\tconsole.error(error instanceof Error ? error.message : String(error));\n\t\treturn 1;\n\t} finally {\n\t\tfor (const cleanup of signalCleanupHandlers) {\n\t\t\tcleanup();\n\t\t}\n\t\tawait disposeRuntime();\n\t\tawait flushRawStdout();\n\t}\n}\n\nfunction stringifyProviderNative(raw: unknown): string {\n\ttry {\n\t\treturn JSON.stringify(raw, null, 2) ?? \"null\";\n\t} catch {\n\t\treturn String(raw);\n\t}\n}\n"]}
1
+ {"version":3,"file":"print-mode.js","sourceRoot":"","sources":["../../src/modes/print-mode.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAIH,OAAO,EAAE,cAAc,EAAE,cAAc,EAAE,MAAM,yBAAyB,CAAC;AACzE,OAAO,EAAE,2BAA2B,EAAE,MAAM,mBAAmB,CAAC;AAChE,OAAO,EAAE,wBAAwB,EAAE,2BAA2B,EAAE,MAAM,gCAAgC,CAAC;AAgBvG;;;GAGG;AACH,MAAM,CAAC,KAAK,UAAU,YAAY,CAAC,WAAgC,EAAE,OAAyB,EAAmB;IAChH,MAAM,EAAE,IAAI,EAAE,QAAQ,GAAG,EAAE,EAAE,cAAc,EAAE,aAAa,EAAE,GAAG,OAAO,CAAC;IACvE,IAAI,QAAQ,GAAG,CAAC,CAAC;IACjB,IAAI,OAAO,GAAG,WAAW,CAAC,OAAO,CAAC;IAClC,IAAI,WAAqC,CAAC;IAC1C,IAAI,QAAQ,GAAG,KAAK,CAAC;IACrB,MAAM,qBAAqB,GAAsB,EAAE,CAAC;IAEpD,MAAM,cAAc,GAAG,KAAK,IAAmB,EAAE,CAAC;QACjD,IAAI,QAAQ;YAAE,OAAO;QACrB,QAAQ,GAAG,IAAI,CAAC;QAChB,WAAW,EAAE,EAAE,CAAC;QAChB,MAAM,WAAW,CAAC,OAAO,EAAE,CAAC;IAAA,CAC5B,CAAC;IAEF,MAAM,sBAAsB,GAAG,GAAS,EAAE,CAAC;QAC1C,MAAM,OAAO,GAAqB,CAAC,SAAS,CAAC,CAAC;QAC9C,IAAI,OAAO,CAAC,QAAQ,KAAK,OAAO,EAAE,CAAC;YAClC,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QACxB,CAAC;QAED,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE,CAAC;YAC9B,MAAM,OAAO,GAAG,GAAG,EAAE,CAAC;gBACrB,2BAA2B,EAAE,CAAC;gBAC9B,KAAK,cAAc,EAAE,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC;oBACnC,OAAO,CAAC,IAAI,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;gBAAA,CAC9C,CAAC,CAAC;YAAA,CACH,CAAC;YACF,OAAO,CAAC,EAAE,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;YAC5B,qBAAqB,CAAC,IAAI,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC,CAAC;QAChE,CAAC;IAAA,CACD,CAAC;IAEF,sBAAsB,EAAE,CAAC;IAEzB,WAAW,CAAC,gBAAgB,CAAC,KAAK,IAAI,EAAE,CAAC;QACxC,MAAM,aAAa,EAAE,CAAC;IAAA,CACtB,CAAC,CAAC;IAEH,MAAM,aAAa,GAAG,KAAK,IAAmB,EAAE,CAAC;QAChD,OAAO,GAAG,WAAW,CAAC,OAAO,CAAC;QAC9B,MAAM,OAAO,CAAC,cAAc,CAAC;YAC5B,qBAAqB,EAAE;gBACtB,WAAW,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,KAAK,CAAC,WAAW,EAAE;gBAC9C,UAAU,EAAE,KAAK,EAAE,iBAAiB,EAAE,EAAE,CAAC,WAAW,CAAC,UAAU,CAAC,iBAAiB,CAAC;gBAClF,IAAI,EAAE,KAAK,EAAE,OAAO,EAAE,WAAW,EAAE,EAAE,CAAC;oBACrC,MAAM,MAAM,GAAG,MAAM,WAAW,CAAC,IAAI,CAAC,OAAO,EAAE,WAAW,CAAC,CAAC;oBAC5D,OAAO,EAAE,SAAS,EAAE,MAAM,CAAC,SAAS,EAAE,CAAC;gBAAA,CACvC;gBACD,YAAY,EAAE,KAAK,EAAE,QAAQ,EAAE,eAAe,EAAE,EAAE,CAAC;oBAClD,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,YAAY,CAAC,QAAQ,EAAE;wBACnD,SAAS,EAAE,eAAe,EAAE,SAAS;wBACrC,kBAAkB,EAAE,eAAe,EAAE,kBAAkB;wBACvD,mBAAmB,EAAE,eAAe,EAAE,mBAAmB;wBACzD,KAAK,EAAE,eAAe,EAAE,KAAK;qBAC7B,CAAC,CAAC;oBACH,OAAO,EAAE,SAAS,EAAE,MAAM,CAAC,SAAS,EAAE,CAAC;gBAAA,CACvC;gBACD,aAAa,EAAE,KAAK,EAAE,WAAW,EAAE,aAAa,EAAE,EAAE,CAAC;oBACpD,OAAO,WAAW,CAAC,aAAa,CAAC,WAAW,EAAE,aAAa,CAAC,CAAC;gBAAA,CAC7D;gBACD,MAAM,EAAE,KAAK,IAAI,EAAE,CAAC;oBACnB,MAAM,OAAO,CAAC,MAAM,EAAE,CAAC;gBAAA,CACvB;aACD;YACD,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC;gBACjB,OAAO,CAAC,KAAK,CAAC,oBAAoB,GAAG,CAAC,aAAa,MAAM,GAAG,CAAC,KAAK,EAAE,CAAC,CAAC;YAAA,CACtE;SACD,CAAC,CAAC;QAEH,WAAW,EAAE,EAAE,CAAC;QAChB,WAAW,GAAG,OAAO,CAAC,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC;YAC1C,IAAI,IAAI,KAAK,MAAM,EAAE,CAAC;gBACrB,cAAc,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;YAC9C,CAAC;QAAA,CACD,CAAC,CAAC;IAAA,CACH,CAAC;IAEF,IAAI,CAAC;QACJ,IAAI,IAAI,KAAK,MAAM,EAAE,CAAC;YACrB,MAAM,MAAM,GAAG,OAAO,CAAC,cAAc,CAAC,SAAS,EAAE,CAAC;YAClD,IAAI,MAAM,EAAE,CAAC;gBACZ,cAAc,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;YAC/C,CAAC;QACF,CAAC;QAED,MAAM,aAAa,EAAE,CAAC;QAEtB,IAAI,cAAc,EAAE,CAAC;YACpB,MAAM,OAAO,CAAC,MAAM,CAAC,cAAc,EAAE,EAAE,MAAM,EAAE,aAAa,EAAE,CAAC,CAAC;QACjE,CAAC;QAED,KAAK,MAAM,OAAO,IAAI,QAAQ,EAAE,CAAC;YAChC,MAAM,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;QAC/B,CAAC;QAED,IAAI,IAAI,KAAK,MAAM,EAAE,CAAC;YACrB,MAAM,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC;YAC5B,MAAM,WAAW,GAAG,KAAK,CAAC,QAAQ,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;YAE9D,IAAI,WAAW,EAAE,IAAI,KAAK,WAAW,EAAE,CAAC;gBACvC,MAAM,YAAY,GAAG,WAA+B,CAAC;gBACrD,IAAI,YAAY,CAAC,UAAU,KAAK,OAAO,IAAI,YAAY,CAAC,UAAU,KAAK,SAAS,EAAE,CAAC;oBAClF,OAAO,CAAC,KAAK,CAAC,YAAY,CAAC,YAAY,IAAI,WAAW,YAAY,CAAC,UAAU,EAAE,CAAC,CAAC;oBACjF,QAAQ,GAAG,CAAC,CAAC;gBACd,CAAC;qBAAM,CAAC;oBACP,KAAK,MAAM,OAAO,IAAI,YAAY,CAAC,OAAO,EAAE,CAAC;wBAC5C,IAAI,OAAO,CAAC,IAAI,KAAK,MAAM,EAAE,CAAC;4BAC7B,cAAc,CAAC,GAAG,OAAO,CAAC,IAAI,IAAI,CAAC,CAAC;wBACrC,CAAC;6BAAM,IAAI,OAAO,CAAC,IAAI,KAAK,gBAAgB,EAAE,CAAC;4BAC9C,cAAc,CAAC,GAAG,2BAA2B,CAAC,YAAY,EAAE,OAAO,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC;4BACjF,cAAc,CAAC,GAAG,wBAAwB,CAAC,OAAO,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC;wBACjE,CAAC;oBACF,CAAC;gBACF,CAAC;YACF,CAAC;QACF,CAAC;QAED,OAAO,QAAQ,CAAC;IACjB,CAAC;IAAC,OAAO,KAAc,EAAE,CAAC;QACzB,OAAO,CAAC,KAAK,CAAC,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;QACtE,OAAO,CAAC,CAAC;IACV,CAAC;YAAS,CAAC;QACV,KAAK,MAAM,OAAO,IAAI,qBAAqB,EAAE,CAAC;YAC7C,OAAO,EAAE,CAAC;QACX,CAAC;QACD,MAAM,cAAc,EAAE,CAAC;QACvB,MAAM,cAAc,EAAE,CAAC;IACxB,CAAC;AAAA,CACD","sourcesContent":["/**\n * Print mode (single-shot): Send prompts, output result, exit.\n *\n * Used for:\n * - `pi -p \"prompt\"` - text output\n * - `senpi --mode json \"prompt\"` - JSON event stream\n */\n\nimport type { AssistantMessage, ImageContent } from \"@earendil-works/pi-ai\";\nimport type { AgentSessionRuntime } from \"../core/agent-session-runtime.js\";\nimport { flushRawStdout, writeRawStdout } from \"../core/output-guard.js\";\nimport { killTrackedDetachedChildren } from \"../utils/shell.js\";\nimport { formatProviderNativeBody, formatProviderNativeSummary } from \"./provider-native-rendering.js\";\n\n/**\n * Options for print mode.\n */\nexport interface PrintModeOptions {\n\t/** Output mode: \"text\" for final response only, \"json\" for all events */\n\tmode: \"text\" | \"json\";\n\t/** Array of additional prompts to send after initialMessage */\n\tmessages?: string[];\n\t/** First message to send (may contain @file content) */\n\tinitialMessage?: string;\n\t/** Images to attach to the initial message */\n\tinitialImages?: ImageContent[];\n}\n\n/**\n * Run in print (single-shot) mode.\n * Sends prompts to the agent and outputs the result.\n */\nexport async function runPrintMode(runtimeHost: AgentSessionRuntime, options: PrintModeOptions): Promise<number> {\n\tconst { mode, messages = [], initialMessage, initialImages } = options;\n\tlet exitCode = 0;\n\tlet session = runtimeHost.session;\n\tlet unsubscribe: (() => void) | undefined;\n\tlet disposed = false;\n\tconst signalCleanupHandlers: Array<() => void> = [];\n\n\tconst disposeRuntime = async (): Promise<void> => {\n\t\tif (disposed) return;\n\t\tdisposed = true;\n\t\tunsubscribe?.();\n\t\tawait runtimeHost.dispose();\n\t};\n\n\tconst registerSignalHandlers = (): void => {\n\t\tconst signals: NodeJS.Signals[] = [\"SIGTERM\"];\n\t\tif (process.platform !== \"win32\") {\n\t\t\tsignals.push(\"SIGHUP\");\n\t\t}\n\n\t\tfor (const signal of signals) {\n\t\t\tconst handler = () => {\n\t\t\t\tkillTrackedDetachedChildren();\n\t\t\t\tvoid disposeRuntime().finally(() => {\n\t\t\t\t\tprocess.exit(signal === \"SIGHUP\" ? 129 : 143);\n\t\t\t\t});\n\t\t\t};\n\t\t\tprocess.on(signal, handler);\n\t\t\tsignalCleanupHandlers.push(() => process.off(signal, handler));\n\t\t}\n\t};\n\n\tregisterSignalHandlers();\n\n\truntimeHost.setRebindSession(async () => {\n\t\tawait rebindSession();\n\t});\n\n\tconst rebindSession = async (): Promise<void> => {\n\t\tsession = runtimeHost.session;\n\t\tawait session.bindExtensions({\n\t\t\tcommandContextActions: {\n\t\t\t\twaitForIdle: () => session.agent.waitForIdle(),\n\t\t\t\tnewSession: async (newSessionOptions) => runtimeHost.newSession(newSessionOptions),\n\t\t\t\tfork: async (entryId, forkOptions) => {\n\t\t\t\t\tconst result = await runtimeHost.fork(entryId, forkOptions);\n\t\t\t\t\treturn { cancelled: result.cancelled };\n\t\t\t\t},\n\t\t\t\tnavigateTree: async (targetId, navigateOptions) => {\n\t\t\t\t\tconst result = await session.navigateTree(targetId, {\n\t\t\t\t\t\tsummarize: navigateOptions?.summarize,\n\t\t\t\t\t\tcustomInstructions: navigateOptions?.customInstructions,\n\t\t\t\t\t\treplaceInstructions: navigateOptions?.replaceInstructions,\n\t\t\t\t\t\tlabel: navigateOptions?.label,\n\t\t\t\t\t});\n\t\t\t\t\treturn { cancelled: result.cancelled };\n\t\t\t\t},\n\t\t\t\tswitchSession: async (sessionPath, switchOptions) => {\n\t\t\t\t\treturn runtimeHost.switchSession(sessionPath, switchOptions);\n\t\t\t\t},\n\t\t\t\treload: async () => {\n\t\t\t\t\tawait session.reload();\n\t\t\t\t},\n\t\t\t},\n\t\t\tonError: (err) => {\n\t\t\t\tconsole.error(`Extension error (${err.extensionPath}): ${err.error}`);\n\t\t\t},\n\t\t});\n\n\t\tunsubscribe?.();\n\t\tunsubscribe = session.subscribe((event) => {\n\t\t\tif (mode === \"json\") {\n\t\t\t\twriteRawStdout(`${JSON.stringify(event)}\\n`);\n\t\t\t}\n\t\t});\n\t};\n\n\ttry {\n\t\tif (mode === \"json\") {\n\t\t\tconst header = session.sessionManager.getHeader();\n\t\t\tif (header) {\n\t\t\t\twriteRawStdout(`${JSON.stringify(header)}\\n`);\n\t\t\t}\n\t\t}\n\n\t\tawait rebindSession();\n\n\t\tif (initialMessage) {\n\t\t\tawait session.prompt(initialMessage, { images: initialImages });\n\t\t}\n\n\t\tfor (const message of messages) {\n\t\t\tawait session.prompt(message);\n\t\t}\n\n\t\tif (mode === \"text\") {\n\t\t\tconst state = session.state;\n\t\t\tconst lastMessage = state.messages[state.messages.length - 1];\n\n\t\t\tif (lastMessage?.role === \"assistant\") {\n\t\t\t\tconst assistantMsg = lastMessage as AssistantMessage;\n\t\t\t\tif (assistantMsg.stopReason === \"error\" || assistantMsg.stopReason === \"aborted\") {\n\t\t\t\t\tconsole.error(assistantMsg.errorMessage || `Request ${assistantMsg.stopReason}`);\n\t\t\t\t\texitCode = 1;\n\t\t\t\t} else {\n\t\t\t\t\tfor (const content of assistantMsg.content) {\n\t\t\t\t\t\tif (content.type === \"text\") {\n\t\t\t\t\t\t\twriteRawStdout(`${content.text}\\n`);\n\t\t\t\t\t\t} else if (content.type === \"providerNative\") {\n\t\t\t\t\t\t\twriteRawStdout(`${formatProviderNativeSummary(assistantMsg, content, false)}\\n`);\n\t\t\t\t\t\t\twriteRawStdout(`${formatProviderNativeBody(content, false)}\\n`);\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn exitCode;\n\t} catch (error: unknown) {\n\t\tconsole.error(error instanceof Error ? error.message : String(error));\n\t\treturn 1;\n\t} finally {\n\t\tfor (const cleanup of signalCleanupHandlers) {\n\t\t\tcleanup();\n\t\t}\n\t\tawait disposeRuntime();\n\t\tawait flushRawStdout();\n\t}\n}\n"]}
@@ -0,0 +1,5 @@
1
+ import type { AssistantMessage, ProviderNativeContent } from "@earendil-works/pi-ai";
2
+ export declare function stringifyProviderNative(raw: unknown): string;
3
+ export declare function formatProviderNativeSummary(message: AssistantMessage, content: ProviderNativeContent, expanded: boolean): string;
4
+ export declare function formatProviderNativeBody(content: ProviderNativeContent, expanded: boolean): string;
5
+ //# sourceMappingURL=provider-native-rendering.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"provider-native-rendering.d.ts","sourceRoot":"","sources":["../../src/modes/provider-native-rendering.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,gBAAgB,EAAE,qBAAqB,EAAE,MAAM,uBAAuB,CAAC;AA+PrF,wBAAgB,uBAAuB,CAAC,GAAG,EAAE,OAAO,GAAG,MAAM,CAM5D;AAED,wBAAgB,2BAA2B,CAC1C,OAAO,EAAE,gBAAgB,EACzB,OAAO,EAAE,qBAAqB,EAC9B,QAAQ,EAAE,OAAO,GACf,MAAM,CASR;AAED,wBAAgB,wBAAwB,CAAC,OAAO,EAAE,qBAAqB,EAAE,QAAQ,EAAE,OAAO,GAAG,MAAM,CAWlG","sourcesContent":["import type { AssistantMessage, ProviderNativeContent } from \"@earendil-works/pi-ai\";\n\ntype NativeRecord = Record<string, unknown>;\n\ninterface SearchSource {\n\ttitle?: string;\n\turl?: string;\n\tsnippet?: string;\n\tstatus?: string;\n}\n\nfunction isRecord(value: unknown): value is NativeRecord {\n\treturn typeof value === \"object\" && value !== null && !Array.isArray(value);\n}\n\nfunction readRecord(record: NativeRecord | undefined, key: string): NativeRecord | undefined {\n\tconst value = record?.[key];\n\treturn isRecord(value) ? value : undefined;\n}\n\nfunction readArray(record: NativeRecord | undefined, key: string): unknown[] {\n\tconst value = record?.[key];\n\treturn Array.isArray(value) ? value : [];\n}\n\nfunction readString(record: NativeRecord | undefined, key: string): string | undefined {\n\tconst value = record?.[key];\n\treturn typeof value === \"string\" ? value : undefined;\n}\n\nfunction readStringArray(record: NativeRecord | undefined, key: string): string[] {\n\treturn readArray(record, key).filter((value): value is string => typeof value === \"string\");\n}\n\nfunction formatProviderName(message: AssistantMessage): string {\n\treturn message.provider ? `${message.provider} · ` : \"\";\n}\n\nfunction pluralize(count: number, singular: string, plural = `${singular}s`): string {\n\treturn `${count} ${count === 1 ? singular : plural}`;\n}\n\nfunction shorten(value: string, max: number): string {\n\treturn value.length <= max ? value : `${value.slice(0, max - 1)}…`;\n}\n\nfunction quoted(value: string): string {\n\treturn JSON.stringify(value);\n}\n\nfunction unique(values: string[]): string[] {\n\treturn [...new Set(values)];\n}\n\nfunction getWebSearchResults(raw: unknown): NativeRecord[] | undefined {\n\tif (!isRecord(raw) || !Array.isArray(raw.content)) {\n\t\treturn undefined;\n\t}\n\n\treturn raw.content.filter(isRecord).filter((item) => readString(item, \"type\") === \"web_search_result\");\n}\n\nfunction sourceFromRecord(record: NativeRecord | undefined): SearchSource | undefined {\n\tif (!record) return undefined;\n\tconst title = readString(record, \"title\") ?? readString(record, \"name\");\n\tconst url =\n\t\treadString(record, \"url\") ??\n\t\treadString(record, \"uri\") ??\n\t\treadString(record, \"retrievedUrl\") ??\n\t\treadString(record, \"sourceUrl\");\n\tconst snippet =\n\t\treadString(record, \"snippet\") ??\n\t\treadString(record, \"text\") ??\n\t\treadString(record, \"summary\") ??\n\t\treadString(record, \"page_age\");\n\tconst status = readString(record, \"status\") ?? readString(record, \"urlRetrievalStatus\");\n\tif (!title && !url && !snippet && !status) return undefined;\n\treturn { ...(title && { title }), ...(url && { url }), ...(snippet && { snippet }), ...(status && { status }) };\n}\n\nfunction sourceFromGoogleGroundingChunk(record: NativeRecord): SearchSource | undefined {\n\treturn sourceFromRecord(readRecord(record, \"web\") ?? record);\n}\n\nfunction formatQueries(queries: string[]): string[] {\n\tconst values = unique(queries);\n\tif (values.length === 0) return [];\n\tif (values.length === 1) return [`query: ${quoted(values[0])}`];\n\treturn [\"queries:\", ...values.map((query) => `- ${quoted(query)}`)];\n}\n\nfunction formatSources(sources: SearchSource[], label: \"result\" | \"source\" | \"url\", expanded: boolean): string[] {\n\tif (sources.length === 0) return [];\n\n\tconst visibleSources = expanded ? sources : sources.slice(0, 3);\n\tconst lines = [pluralize(sources.length, label)];\n\tfor (const source of visibleSources) {\n\t\tconst title = source.title ? shorten(source.title, 100) : undefined;\n\t\tconst url = source.url ? shorten(source.url, 120) : undefined;\n\t\tconst status = source.status ? ` (${source.status})` : \"\";\n\t\tif (title && url) {\n\t\t\tlines.push(`${title} ${url}${status}`);\n\t\t} else if (title) {\n\t\t\tlines.push(`${title}${status}`);\n\t\t} else if (url) {\n\t\t\tlines.push(`${url}${status}`);\n\t\t} else if (source.status) {\n\t\t\tlines.push(source.status);\n\t\t}\n\t\tif (source.snippet) {\n\t\t\tlines.push(` ${shorten(source.snippet, 160)}`);\n\t\t}\n\t}\n\tif (!expanded && sources.length > visibleSources.length) {\n\t\tlines.push(`… ${sources.length - visibleSources.length} more ${label}s`);\n\t}\n\treturn lines;\n}\n\nfunction formatServerToolUseBody(raw: unknown): string | undefined {\n\tconst record = isRecord(raw) ? raw : undefined;\n\tconst input = readRecord(record, \"input\");\n\tconst query = readString(input, \"query\");\n\tif (query) {\n\t\treturn `query: ${quoted(query)}`;\n\t}\n\tconst name = readString(record, \"name\");\n\tif (name) {\n\t\treturn `name: ${name}`;\n\t}\n\treturn undefined;\n}\n\nfunction formatAnthropicWebSearchResultBody(raw: unknown, expanded: boolean): string | undefined {\n\tconst results = getWebSearchResults(raw);\n\tif (!results) {\n\t\treturn undefined;\n\t}\n\tif (results.length === 0) {\n\t\treturn \"0 results\";\n\t}\n\n\tconst sources = results.flatMap((result) => {\n\t\tconst source = sourceFromRecord(result);\n\t\treturn source ? [source] : [];\n\t});\n\treturn formatSources(sources, \"result\", expanded).join(\"\\n\");\n}\n\nfunction formatOpenAiWebSearchCallBody(raw: unknown, expanded: boolean): string | undefined {\n\tconst record = isRecord(raw) ? raw : undefined;\n\tif (!record) return undefined;\n\n\tconst action = readRecord(record, \"action\");\n\tconst status = readString(record, \"status\");\n\tconst queries = [\n\t\t...(readString(action, \"query\") ? [readString(action, \"query\") as string] : []),\n\t\t...readStringArray(action, \"queries\"),\n\t];\n\tconst sources = readArray(action, \"sources\").flatMap((rawSource) => {\n\t\tconst source = sourceFromRecord(isRecord(rawSource) ? rawSource : undefined);\n\t\treturn source ? [source] : [];\n\t});\n\n\tconst lines = [\n\t\t...(status ? [`status: ${status}`] : []),\n\t\t...formatQueries(queries),\n\t\t...formatSources(sources, \"source\", expanded),\n\t];\n\treturn lines.length > 0 ? lines.join(\"\\n\") : undefined;\n}\n\nfunction formatGoogleGroundingMetadataBody(raw: unknown, expanded: boolean): string | undefined {\n\tconst record = isRecord(raw) ? raw : undefined;\n\tif (!record) return undefined;\n\n\tconst queries = readStringArray(record, \"webSearchQueries\");\n\tconst sources = readArray(record, \"groundingChunks\").flatMap((rawChunk) => {\n\t\tconst source = isRecord(rawChunk) ? sourceFromGoogleGroundingChunk(rawChunk) : undefined;\n\t\treturn source ? [source] : [];\n\t});\n\n\tconst lines = [...formatQueries(queries), ...formatSources(sources, \"source\", expanded)];\n\treturn lines.length > 0 ? lines.join(\"\\n\") : undefined;\n}\n\nfunction formatGoogleUrlContextMetadataBody(raw: unknown, expanded: boolean): string | undefined {\n\tconst record = isRecord(raw) ? raw : undefined;\n\tif (!record) return undefined;\n\n\tconst sources = readArray(record, \"urlMetadata\").flatMap((rawUrl) => {\n\t\tconst source = sourceFromRecord(isRecord(rawUrl) ? rawUrl : undefined);\n\t\treturn source ? [source] : [];\n\t});\n\n\tconst lines = formatSources(sources, \"url\", expanded);\n\treturn lines.length > 0 ? lines.join(\"\\n\") : undefined;\n}\n\nfunction formatSpecializedProviderNativeSummary(\n\tmessage: AssistantMessage,\n\tcontent: ProviderNativeContent,\n\texpanded: boolean,\n): string | undefined {\n\tconst provider = formatProviderName(message);\n\tconst marker = expanded ? \"▾\" : \"▸\";\n\tif (content.subtype === \"server_tool_use\") {\n\t\tconst raw = isRecord(content.raw) ? content.raw : undefined;\n\t\tconst name = readString(raw, \"name\");\n\t\tif (name) {\n\t\t\treturn `${marker} ${provider}${name} · server_tool_use`;\n\t\t}\n\t}\n\tif (content.subtype === \"web_search_tool_result\") {\n\t\treturn `${marker} ${provider}web_search results`;\n\t}\n\tif (content.subtype === \"web_search_call\") {\n\t\tconst raw = isRecord(content.raw) ? content.raw : undefined;\n\t\tconst status = readString(raw, \"status\");\n\t\treturn `${marker} ${provider}web_search${status ? ` · ${status}` : \" · web_search_call\"}`;\n\t}\n\tif (content.subtype === \"groundingMetadata\") {\n\t\tconst body = formatGoogleGroundingMetadataBody(content.raw, expanded);\n\t\tif (body) {\n\t\t\treturn `${marker} ${provider}google_search results`;\n\t\t}\n\t}\n\tif (content.subtype === \"urlContextMetadata\") {\n\t\tconst body = formatGoogleUrlContextMetadataBody(content.raw, expanded);\n\t\tif (body) {\n\t\t\treturn `${marker} ${provider}url_context results`;\n\t\t}\n\t}\n\treturn undefined;\n}\n\nfunction formatSpecializedProviderNativeBody(content: ProviderNativeContent, expanded: boolean): string | undefined {\n\tif (content.subtype === \"server_tool_use\") {\n\t\treturn formatServerToolUseBody(content.raw);\n\t}\n\tif (content.subtype === \"web_search_tool_result\") {\n\t\treturn formatAnthropicWebSearchResultBody(content.raw, expanded);\n\t}\n\tif (content.subtype === \"web_search_call\") {\n\t\treturn formatOpenAiWebSearchCallBody(content.raw, expanded);\n\t}\n\tif (content.subtype === \"groundingMetadata\") {\n\t\treturn formatGoogleGroundingMetadataBody(content.raw, expanded);\n\t}\n\tif (content.subtype === \"urlContextMetadata\") {\n\t\treturn formatGoogleUrlContextMetadataBody(content.raw, expanded);\n\t}\n\treturn undefined;\n}\n\nexport function stringifyProviderNative(raw: unknown): string {\n\ttry {\n\t\treturn JSON.stringify(raw, null, 2) ?? \"null\";\n\t} catch {\n\t\treturn String(raw);\n\t}\n}\n\nexport function formatProviderNativeSummary(\n\tmessage: AssistantMessage,\n\tcontent: ProviderNativeContent,\n\texpanded: boolean,\n): string {\n\tconst specialized = formatSpecializedProviderNativeSummary(message, content, expanded);\n\tif (specialized) {\n\t\treturn specialized;\n\t}\n\n\tconst provider = formatProviderName(message);\n\tconst marker = expanded ? \"▾\" : \"▸\";\n\treturn `${marker} ${provider}providerNative · ${content.subtype}`;\n}\n\nexport function formatProviderNativeBody(content: ProviderNativeContent, expanded: boolean): string {\n\tconst specialized = formatSpecializedProviderNativeBody(content, expanded);\n\tif (specialized) {\n\t\treturn specialized;\n\t}\n\n\tconst rawJson = stringifyProviderNative(content.raw);\n\tif (expanded || rawJson.length <= 2000) {\n\t\treturn rawJson;\n\t}\n\treturn `${rawJson.slice(0, 2000)}…`;\n}\n"]}
@@ -0,0 +1,247 @@
1
+ function isRecord(value) {
2
+ return typeof value === "object" && value !== null && !Array.isArray(value);
3
+ }
4
+ function readRecord(record, key) {
5
+ const value = record?.[key];
6
+ return isRecord(value) ? value : undefined;
7
+ }
8
+ function readArray(record, key) {
9
+ const value = record?.[key];
10
+ return Array.isArray(value) ? value : [];
11
+ }
12
+ function readString(record, key) {
13
+ const value = record?.[key];
14
+ return typeof value === "string" ? value : undefined;
15
+ }
16
+ function readStringArray(record, key) {
17
+ return readArray(record, key).filter((value) => typeof value === "string");
18
+ }
19
+ function formatProviderName(message) {
20
+ return message.provider ? `${message.provider} · ` : "";
21
+ }
22
+ function pluralize(count, singular, plural = `${singular}s`) {
23
+ return `${count} ${count === 1 ? singular : plural}`;
24
+ }
25
+ function shorten(value, max) {
26
+ return value.length <= max ? value : `${value.slice(0, max - 1)}…`;
27
+ }
28
+ function quoted(value) {
29
+ return JSON.stringify(value);
30
+ }
31
+ function unique(values) {
32
+ return [...new Set(values)];
33
+ }
34
+ function getWebSearchResults(raw) {
35
+ if (!isRecord(raw) || !Array.isArray(raw.content)) {
36
+ return undefined;
37
+ }
38
+ return raw.content.filter(isRecord).filter((item) => readString(item, "type") === "web_search_result");
39
+ }
40
+ function sourceFromRecord(record) {
41
+ if (!record)
42
+ return undefined;
43
+ const title = readString(record, "title") ?? readString(record, "name");
44
+ const url = readString(record, "url") ??
45
+ readString(record, "uri") ??
46
+ readString(record, "retrievedUrl") ??
47
+ readString(record, "sourceUrl");
48
+ const snippet = readString(record, "snippet") ??
49
+ readString(record, "text") ??
50
+ readString(record, "summary") ??
51
+ readString(record, "page_age");
52
+ const status = readString(record, "status") ?? readString(record, "urlRetrievalStatus");
53
+ if (!title && !url && !snippet && !status)
54
+ return undefined;
55
+ return { ...(title && { title }), ...(url && { url }), ...(snippet && { snippet }), ...(status && { status }) };
56
+ }
57
+ function sourceFromGoogleGroundingChunk(record) {
58
+ return sourceFromRecord(readRecord(record, "web") ?? record);
59
+ }
60
+ function formatQueries(queries) {
61
+ const values = unique(queries);
62
+ if (values.length === 0)
63
+ return [];
64
+ if (values.length === 1)
65
+ return [`query: ${quoted(values[0])}`];
66
+ return ["queries:", ...values.map((query) => `- ${quoted(query)}`)];
67
+ }
68
+ function formatSources(sources, label, expanded) {
69
+ if (sources.length === 0)
70
+ return [];
71
+ const visibleSources = expanded ? sources : sources.slice(0, 3);
72
+ const lines = [pluralize(sources.length, label)];
73
+ for (const source of visibleSources) {
74
+ const title = source.title ? shorten(source.title, 100) : undefined;
75
+ const url = source.url ? shorten(source.url, 120) : undefined;
76
+ const status = source.status ? ` (${source.status})` : "";
77
+ if (title && url) {
78
+ lines.push(`${title} ${url}${status}`);
79
+ }
80
+ else if (title) {
81
+ lines.push(`${title}${status}`);
82
+ }
83
+ else if (url) {
84
+ lines.push(`${url}${status}`);
85
+ }
86
+ else if (source.status) {
87
+ lines.push(source.status);
88
+ }
89
+ if (source.snippet) {
90
+ lines.push(` ${shorten(source.snippet, 160)}`);
91
+ }
92
+ }
93
+ if (!expanded && sources.length > visibleSources.length) {
94
+ lines.push(`… ${sources.length - visibleSources.length} more ${label}s`);
95
+ }
96
+ return lines;
97
+ }
98
+ function formatServerToolUseBody(raw) {
99
+ const record = isRecord(raw) ? raw : undefined;
100
+ const input = readRecord(record, "input");
101
+ const query = readString(input, "query");
102
+ if (query) {
103
+ return `query: ${quoted(query)}`;
104
+ }
105
+ const name = readString(record, "name");
106
+ if (name) {
107
+ return `name: ${name}`;
108
+ }
109
+ return undefined;
110
+ }
111
+ function formatAnthropicWebSearchResultBody(raw, expanded) {
112
+ const results = getWebSearchResults(raw);
113
+ if (!results) {
114
+ return undefined;
115
+ }
116
+ if (results.length === 0) {
117
+ return "0 results";
118
+ }
119
+ const sources = results.flatMap((result) => {
120
+ const source = sourceFromRecord(result);
121
+ return source ? [source] : [];
122
+ });
123
+ return formatSources(sources, "result", expanded).join("\n");
124
+ }
125
+ function formatOpenAiWebSearchCallBody(raw, expanded) {
126
+ const record = isRecord(raw) ? raw : undefined;
127
+ if (!record)
128
+ return undefined;
129
+ const action = readRecord(record, "action");
130
+ const status = readString(record, "status");
131
+ const queries = [
132
+ ...(readString(action, "query") ? [readString(action, "query")] : []),
133
+ ...readStringArray(action, "queries"),
134
+ ];
135
+ const sources = readArray(action, "sources").flatMap((rawSource) => {
136
+ const source = sourceFromRecord(isRecord(rawSource) ? rawSource : undefined);
137
+ return source ? [source] : [];
138
+ });
139
+ const lines = [
140
+ ...(status ? [`status: ${status}`] : []),
141
+ ...formatQueries(queries),
142
+ ...formatSources(sources, "source", expanded),
143
+ ];
144
+ return lines.length > 0 ? lines.join("\n") : undefined;
145
+ }
146
+ function formatGoogleGroundingMetadataBody(raw, expanded) {
147
+ const record = isRecord(raw) ? raw : undefined;
148
+ if (!record)
149
+ return undefined;
150
+ const queries = readStringArray(record, "webSearchQueries");
151
+ const sources = readArray(record, "groundingChunks").flatMap((rawChunk) => {
152
+ const source = isRecord(rawChunk) ? sourceFromGoogleGroundingChunk(rawChunk) : undefined;
153
+ return source ? [source] : [];
154
+ });
155
+ const lines = [...formatQueries(queries), ...formatSources(sources, "source", expanded)];
156
+ return lines.length > 0 ? lines.join("\n") : undefined;
157
+ }
158
+ function formatGoogleUrlContextMetadataBody(raw, expanded) {
159
+ const record = isRecord(raw) ? raw : undefined;
160
+ if (!record)
161
+ return undefined;
162
+ const sources = readArray(record, "urlMetadata").flatMap((rawUrl) => {
163
+ const source = sourceFromRecord(isRecord(rawUrl) ? rawUrl : undefined);
164
+ return source ? [source] : [];
165
+ });
166
+ const lines = formatSources(sources, "url", expanded);
167
+ return lines.length > 0 ? lines.join("\n") : undefined;
168
+ }
169
+ function formatSpecializedProviderNativeSummary(message, content, expanded) {
170
+ const provider = formatProviderName(message);
171
+ const marker = expanded ? "▾" : "▸";
172
+ if (content.subtype === "server_tool_use") {
173
+ const raw = isRecord(content.raw) ? content.raw : undefined;
174
+ const name = readString(raw, "name");
175
+ if (name) {
176
+ return `${marker} ${provider}${name} · server_tool_use`;
177
+ }
178
+ }
179
+ if (content.subtype === "web_search_tool_result") {
180
+ return `${marker} ${provider}web_search results`;
181
+ }
182
+ if (content.subtype === "web_search_call") {
183
+ const raw = isRecord(content.raw) ? content.raw : undefined;
184
+ const status = readString(raw, "status");
185
+ return `${marker} ${provider}web_search${status ? ` · ${status}` : " · web_search_call"}`;
186
+ }
187
+ if (content.subtype === "groundingMetadata") {
188
+ const body = formatGoogleGroundingMetadataBody(content.raw, expanded);
189
+ if (body) {
190
+ return `${marker} ${provider}google_search results`;
191
+ }
192
+ }
193
+ if (content.subtype === "urlContextMetadata") {
194
+ const body = formatGoogleUrlContextMetadataBody(content.raw, expanded);
195
+ if (body) {
196
+ return `${marker} ${provider}url_context results`;
197
+ }
198
+ }
199
+ return undefined;
200
+ }
201
+ function formatSpecializedProviderNativeBody(content, expanded) {
202
+ if (content.subtype === "server_tool_use") {
203
+ return formatServerToolUseBody(content.raw);
204
+ }
205
+ if (content.subtype === "web_search_tool_result") {
206
+ return formatAnthropicWebSearchResultBody(content.raw, expanded);
207
+ }
208
+ if (content.subtype === "web_search_call") {
209
+ return formatOpenAiWebSearchCallBody(content.raw, expanded);
210
+ }
211
+ if (content.subtype === "groundingMetadata") {
212
+ return formatGoogleGroundingMetadataBody(content.raw, expanded);
213
+ }
214
+ if (content.subtype === "urlContextMetadata") {
215
+ return formatGoogleUrlContextMetadataBody(content.raw, expanded);
216
+ }
217
+ return undefined;
218
+ }
219
+ export function stringifyProviderNative(raw) {
220
+ try {
221
+ return JSON.stringify(raw, null, 2) ?? "null";
222
+ }
223
+ catch {
224
+ return String(raw);
225
+ }
226
+ }
227
+ export function formatProviderNativeSummary(message, content, expanded) {
228
+ const specialized = formatSpecializedProviderNativeSummary(message, content, expanded);
229
+ if (specialized) {
230
+ return specialized;
231
+ }
232
+ const provider = formatProviderName(message);
233
+ const marker = expanded ? "▾" : "▸";
234
+ return `${marker} ${provider}providerNative · ${content.subtype}`;
235
+ }
236
+ export function formatProviderNativeBody(content, expanded) {
237
+ const specialized = formatSpecializedProviderNativeBody(content, expanded);
238
+ if (specialized) {
239
+ return specialized;
240
+ }
241
+ const rawJson = stringifyProviderNative(content.raw);
242
+ if (expanded || rawJson.length <= 2000) {
243
+ return rawJson;
244
+ }
245
+ return `${rawJson.slice(0, 2000)}…`;
246
+ }
247
+ //# sourceMappingURL=provider-native-rendering.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"provider-native-rendering.js","sourceRoot":"","sources":["../../src/modes/provider-native-rendering.ts"],"names":[],"mappings":"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","sourcesContent":["import type { AssistantMessage, ProviderNativeContent } from \"@earendil-works/pi-ai\";\n\ntype NativeRecord = Record<string, unknown>;\n\ninterface SearchSource {\n\ttitle?: string;\n\turl?: string;\n\tsnippet?: string;\n\tstatus?: string;\n}\n\nfunction isRecord(value: unknown): value is NativeRecord {\n\treturn typeof value === \"object\" && value !== null && !Array.isArray(value);\n}\n\nfunction readRecord(record: NativeRecord | undefined, key: string): NativeRecord | undefined {\n\tconst value = record?.[key];\n\treturn isRecord(value) ? value : undefined;\n}\n\nfunction readArray(record: NativeRecord | undefined, key: string): unknown[] {\n\tconst value = record?.[key];\n\treturn Array.isArray(value) ? value : [];\n}\n\nfunction readString(record: NativeRecord | undefined, key: string): string | undefined {\n\tconst value = record?.[key];\n\treturn typeof value === \"string\" ? value : undefined;\n}\n\nfunction readStringArray(record: NativeRecord | undefined, key: string): string[] {\n\treturn readArray(record, key).filter((value): value is string => typeof value === \"string\");\n}\n\nfunction formatProviderName(message: AssistantMessage): string {\n\treturn message.provider ? `${message.provider} · ` : \"\";\n}\n\nfunction pluralize(count: number, singular: string, plural = `${singular}s`): string {\n\treturn `${count} ${count === 1 ? singular : plural}`;\n}\n\nfunction shorten(value: string, max: number): string {\n\treturn value.length <= max ? value : `${value.slice(0, max - 1)}…`;\n}\n\nfunction quoted(value: string): string {\n\treturn JSON.stringify(value);\n}\n\nfunction unique(values: string[]): string[] {\n\treturn [...new Set(values)];\n}\n\nfunction getWebSearchResults(raw: unknown): NativeRecord[] | undefined {\n\tif (!isRecord(raw) || !Array.isArray(raw.content)) {\n\t\treturn undefined;\n\t}\n\n\treturn raw.content.filter(isRecord).filter((item) => readString(item, \"type\") === \"web_search_result\");\n}\n\nfunction sourceFromRecord(record: NativeRecord | undefined): SearchSource | undefined {\n\tif (!record) return undefined;\n\tconst title = readString(record, \"title\") ?? readString(record, \"name\");\n\tconst url =\n\t\treadString(record, \"url\") ??\n\t\treadString(record, \"uri\") ??\n\t\treadString(record, \"retrievedUrl\") ??\n\t\treadString(record, \"sourceUrl\");\n\tconst snippet =\n\t\treadString(record, \"snippet\") ??\n\t\treadString(record, \"text\") ??\n\t\treadString(record, \"summary\") ??\n\t\treadString(record, \"page_age\");\n\tconst status = readString(record, \"status\") ?? readString(record, \"urlRetrievalStatus\");\n\tif (!title && !url && !snippet && !status) return undefined;\n\treturn { ...(title && { title }), ...(url && { url }), ...(snippet && { snippet }), ...(status && { status }) };\n}\n\nfunction sourceFromGoogleGroundingChunk(record: NativeRecord): SearchSource | undefined {\n\treturn sourceFromRecord(readRecord(record, \"web\") ?? record);\n}\n\nfunction formatQueries(queries: string[]): string[] {\n\tconst values = unique(queries);\n\tif (values.length === 0) return [];\n\tif (values.length === 1) return [`query: ${quoted(values[0])}`];\n\treturn [\"queries:\", ...values.map((query) => `- ${quoted(query)}`)];\n}\n\nfunction formatSources(sources: SearchSource[], label: \"result\" | \"source\" | \"url\", expanded: boolean): string[] {\n\tif (sources.length === 0) return [];\n\n\tconst visibleSources = expanded ? sources : sources.slice(0, 3);\n\tconst lines = [pluralize(sources.length, label)];\n\tfor (const source of visibleSources) {\n\t\tconst title = source.title ? shorten(source.title, 100) : undefined;\n\t\tconst url = source.url ? shorten(source.url, 120) : undefined;\n\t\tconst status = source.status ? ` (${source.status})` : \"\";\n\t\tif (title && url) {\n\t\t\tlines.push(`${title} ${url}${status}`);\n\t\t} else if (title) {\n\t\t\tlines.push(`${title}${status}`);\n\t\t} else if (url) {\n\t\t\tlines.push(`${url}${status}`);\n\t\t} else if (source.status) {\n\t\t\tlines.push(source.status);\n\t\t}\n\t\tif (source.snippet) {\n\t\t\tlines.push(` ${shorten(source.snippet, 160)}`);\n\t\t}\n\t}\n\tif (!expanded && sources.length > visibleSources.length) {\n\t\tlines.push(`… ${sources.length - visibleSources.length} more ${label}s`);\n\t}\n\treturn lines;\n}\n\nfunction formatServerToolUseBody(raw: unknown): string | undefined {\n\tconst record = isRecord(raw) ? raw : undefined;\n\tconst input = readRecord(record, \"input\");\n\tconst query = readString(input, \"query\");\n\tif (query) {\n\t\treturn `query: ${quoted(query)}`;\n\t}\n\tconst name = readString(record, \"name\");\n\tif (name) {\n\t\treturn `name: ${name}`;\n\t}\n\treturn undefined;\n}\n\nfunction formatAnthropicWebSearchResultBody(raw: unknown, expanded: boolean): string | undefined {\n\tconst results = getWebSearchResults(raw);\n\tif (!results) {\n\t\treturn undefined;\n\t}\n\tif (results.length === 0) {\n\t\treturn \"0 results\";\n\t}\n\n\tconst sources = results.flatMap((result) => {\n\t\tconst source = sourceFromRecord(result);\n\t\treturn source ? [source] : [];\n\t});\n\treturn formatSources(sources, \"result\", expanded).join(\"\\n\");\n}\n\nfunction formatOpenAiWebSearchCallBody(raw: unknown, expanded: boolean): string | undefined {\n\tconst record = isRecord(raw) ? raw : undefined;\n\tif (!record) return undefined;\n\n\tconst action = readRecord(record, \"action\");\n\tconst status = readString(record, \"status\");\n\tconst queries = [\n\t\t...(readString(action, \"query\") ? [readString(action, \"query\") as string] : []),\n\t\t...readStringArray(action, \"queries\"),\n\t];\n\tconst sources = readArray(action, \"sources\").flatMap((rawSource) => {\n\t\tconst source = sourceFromRecord(isRecord(rawSource) ? rawSource : undefined);\n\t\treturn source ? [source] : [];\n\t});\n\n\tconst lines = [\n\t\t...(status ? [`status: ${status}`] : []),\n\t\t...formatQueries(queries),\n\t\t...formatSources(sources, \"source\", expanded),\n\t];\n\treturn lines.length > 0 ? lines.join(\"\\n\") : undefined;\n}\n\nfunction formatGoogleGroundingMetadataBody(raw: unknown, expanded: boolean): string | undefined {\n\tconst record = isRecord(raw) ? raw : undefined;\n\tif (!record) return undefined;\n\n\tconst queries = readStringArray(record, \"webSearchQueries\");\n\tconst sources = readArray(record, \"groundingChunks\").flatMap((rawChunk) => {\n\t\tconst source = isRecord(rawChunk) ? sourceFromGoogleGroundingChunk(rawChunk) : undefined;\n\t\treturn source ? [source] : [];\n\t});\n\n\tconst lines = [...formatQueries(queries), ...formatSources(sources, \"source\", expanded)];\n\treturn lines.length > 0 ? lines.join(\"\\n\") : undefined;\n}\n\nfunction formatGoogleUrlContextMetadataBody(raw: unknown, expanded: boolean): string | undefined {\n\tconst record = isRecord(raw) ? raw : undefined;\n\tif (!record) return undefined;\n\n\tconst sources = readArray(record, \"urlMetadata\").flatMap((rawUrl) => {\n\t\tconst source = sourceFromRecord(isRecord(rawUrl) ? rawUrl : undefined);\n\t\treturn source ? [source] : [];\n\t});\n\n\tconst lines = formatSources(sources, \"url\", expanded);\n\treturn lines.length > 0 ? lines.join(\"\\n\") : undefined;\n}\n\nfunction formatSpecializedProviderNativeSummary(\n\tmessage: AssistantMessage,\n\tcontent: ProviderNativeContent,\n\texpanded: boolean,\n): string | undefined {\n\tconst provider = formatProviderName(message);\n\tconst marker = expanded ? \"▾\" : \"▸\";\n\tif (content.subtype === \"server_tool_use\") {\n\t\tconst raw = isRecord(content.raw) ? content.raw : undefined;\n\t\tconst name = readString(raw, \"name\");\n\t\tif (name) {\n\t\t\treturn `${marker} ${provider}${name} · server_tool_use`;\n\t\t}\n\t}\n\tif (content.subtype === \"web_search_tool_result\") {\n\t\treturn `${marker} ${provider}web_search results`;\n\t}\n\tif (content.subtype === \"web_search_call\") {\n\t\tconst raw = isRecord(content.raw) ? content.raw : undefined;\n\t\tconst status = readString(raw, \"status\");\n\t\treturn `${marker} ${provider}web_search${status ? ` · ${status}` : \" · web_search_call\"}`;\n\t}\n\tif (content.subtype === \"groundingMetadata\") {\n\t\tconst body = formatGoogleGroundingMetadataBody(content.raw, expanded);\n\t\tif (body) {\n\t\t\treturn `${marker} ${provider}google_search results`;\n\t\t}\n\t}\n\tif (content.subtype === \"urlContextMetadata\") {\n\t\tconst body = formatGoogleUrlContextMetadataBody(content.raw, expanded);\n\t\tif (body) {\n\t\t\treturn `${marker} ${provider}url_context results`;\n\t\t}\n\t}\n\treturn undefined;\n}\n\nfunction formatSpecializedProviderNativeBody(content: ProviderNativeContent, expanded: boolean): string | undefined {\n\tif (content.subtype === \"server_tool_use\") {\n\t\treturn formatServerToolUseBody(content.raw);\n\t}\n\tif (content.subtype === \"web_search_tool_result\") {\n\t\treturn formatAnthropicWebSearchResultBody(content.raw, expanded);\n\t}\n\tif (content.subtype === \"web_search_call\") {\n\t\treturn formatOpenAiWebSearchCallBody(content.raw, expanded);\n\t}\n\tif (content.subtype === \"groundingMetadata\") {\n\t\treturn formatGoogleGroundingMetadataBody(content.raw, expanded);\n\t}\n\tif (content.subtype === \"urlContextMetadata\") {\n\t\treturn formatGoogleUrlContextMetadataBody(content.raw, expanded);\n\t}\n\treturn undefined;\n}\n\nexport function stringifyProviderNative(raw: unknown): string {\n\ttry {\n\t\treturn JSON.stringify(raw, null, 2) ?? \"null\";\n\t} catch {\n\t\treturn String(raw);\n\t}\n}\n\nexport function formatProviderNativeSummary(\n\tmessage: AssistantMessage,\n\tcontent: ProviderNativeContent,\n\texpanded: boolean,\n): string {\n\tconst specialized = formatSpecializedProviderNativeSummary(message, content, expanded);\n\tif (specialized) {\n\t\treturn specialized;\n\t}\n\n\tconst provider = formatProviderName(message);\n\tconst marker = expanded ? \"▾\" : \"▸\";\n\treturn `${marker} ${provider}providerNative · ${content.subtype}`;\n}\n\nexport function formatProviderNativeBody(content: ProviderNativeContent, expanded: boolean): string {\n\tconst specialized = formatSpecializedProviderNativeBody(content, expanded);\n\tif (specialized) {\n\t\treturn specialized;\n\t}\n\n\tconst rawJson = stringifyProviderNative(content.raw);\n\tif (expanded || rawJson.length <= 2000) {\n\t\treturn rawJson;\n\t}\n\treturn `${rawJson.slice(0, 2000)}…`;\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export declare function stripAnsi(value: string): string;
2
+ //# sourceMappingURL=ansi.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ansi.d.ts","sourceRoot":"","sources":["../../src/utils/ansi.ts"],"names":[],"mappings":"AA6CA,wBAAgB,SAAS,CAAC,KAAK,EAAE,MAAM,GAAG,MAAM,CAc/C","sourcesContent":["/*\n * Portions of this file are derived from:\n * - ansi-regex (https://github.com/chalk/ansi-regex)\n * - strip-ansi (https://github.com/chalk/strip-ansi)\n *\n * MIT License\n *\n * Copyright (c) Sindre Sorhus <sindresorhus@gmail.com> (https://sindresorhus.com)\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n */\n\nfunction ansiRegex({ onlyFirst = false }: { onlyFirst?: boolean } = {}): RegExp {\n\t// Valid string terminator sequences are BEL, ESC\\, and 0x9c\n\tconst ST = \"(?:\\\\u0007|\\\\u001B\\\\u005C|\\\\u009C)\";\n\n\t// OSC sequences only: ESC ] ... ST (non-greedy until the first ST)\n\tconst osc = `(?:\\\\u001B\\\\][\\\\s\\\\S]*?${ST})`;\n\n\t// CSI and related: ESC/C1, optional intermediates, optional params (supports ; and :) then final byte\n\tconst csi = \"[\\\\u001B\\\\u009B][[\\\\]()#;?]*(?:\\\\d{1,4}(?:[;:]\\\\d{0,4})*)?[\\\\dA-PR-TZcf-nq-uy=><~]\";\n\n\tconst pattern = `${osc}|${csi}`;\n\n\treturn new RegExp(pattern, onlyFirst ? undefined : \"g\");\n}\n\nconst regex = ansiRegex();\n\nexport function stripAnsi(value: string): string {\n\tif (typeof value !== \"string\") {\n\t\tthrow new TypeError(`Expected a \\`string\\`, got \\`${typeof value}\\``);\n\t}\n\n\t// Fast path: ANSI codes require ESC (7-bit) or CSI (8-bit) introducer\n\tif (!value.includes(\"\\u001B\") && !value.includes(\"\\u009B\")) {\n\t\treturn value;\n\t}\n\n\t// Even though the regex is global, we don't need to reset the `.lastIndex`\n\t// because unlike `.exec()` and `.test()`, `.replace()` does it automatically\n\t// and doing it manually has a performance penalty.\n\treturn value.replace(regex, \"\");\n}\n"]}
@@ -0,0 +1,52 @@
1
+ /*
2
+ * Portions of this file are derived from:
3
+ * - ansi-regex (https://github.com/chalk/ansi-regex)
4
+ * - strip-ansi (https://github.com/chalk/strip-ansi)
5
+ *
6
+ * MIT License
7
+ *
8
+ * Copyright (c) Sindre Sorhus <sindresorhus@gmail.com> (https://sindresorhus.com)
9
+ *
10
+ * Permission is hereby granted, free of charge, to any person obtaining a copy
11
+ * of this software and associated documentation files (the "Software"), to deal
12
+ * in the Software without restriction, including without limitation the rights
13
+ * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
14
+ * copies of the Software, and to permit persons to whom the Software is
15
+ * furnished to do so, subject to the following conditions:
16
+ *
17
+ * The above copyright notice and this permission notice shall be included in all
18
+ * copies or substantial portions of the Software.
19
+ *
20
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
21
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
22
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
23
+ * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
24
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
25
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
26
+ * SOFTWARE.
27
+ */
28
+ function ansiRegex({ onlyFirst = false } = {}) {
29
+ // Valid string terminator sequences are BEL, ESC\, and 0x9c
30
+ const ST = "(?:\\u0007|\\u001B\\u005C|\\u009C)";
31
+ // OSC sequences only: ESC ] ... ST (non-greedy until the first ST)
32
+ const osc = `(?:\\u001B\\][\\s\\S]*?${ST})`;
33
+ // CSI and related: ESC/C1, optional intermediates, optional params (supports ; and :) then final byte
34
+ const csi = "[\\u001B\\u009B][[\\]()#;?]*(?:\\d{1,4}(?:[;:]\\d{0,4})*)?[\\dA-PR-TZcf-nq-uy=><~]";
35
+ const pattern = `${osc}|${csi}`;
36
+ return new RegExp(pattern, onlyFirst ? undefined : "g");
37
+ }
38
+ const regex = ansiRegex();
39
+ export function stripAnsi(value) {
40
+ if (typeof value !== "string") {
41
+ throw new TypeError(`Expected a \`string\`, got \`${typeof value}\``);
42
+ }
43
+ // Fast path: ANSI codes require ESC (7-bit) or CSI (8-bit) introducer
44
+ if (!value.includes("\u001B") && !value.includes("\u009B")) {
45
+ return value;
46
+ }
47
+ // Even though the regex is global, we don't need to reset the `.lastIndex`
48
+ // because unlike `.exec()` and `.test()`, `.replace()` does it automatically
49
+ // and doing it manually has a performance penalty.
50
+ return value.replace(regex, "");
51
+ }
52
+ //# sourceMappingURL=ansi.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ansi.js","sourceRoot":"","sources":["../../src/utils/ansi.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;GA0BG;AAEH,SAAS,SAAS,CAAC,EAAE,SAAS,GAAG,KAAK,EAAE,GAA4B,EAAE,EAAU;IAC/E,4DAA4D;IAC5D,MAAM,EAAE,GAAG,oCAAoC,CAAC;IAEhD,mEAAmE;IACnE,MAAM,GAAG,GAAG,0BAA0B,EAAE,GAAG,CAAC;IAE5C,sGAAsG;IACtG,MAAM,GAAG,GAAG,oFAAoF,CAAC;IAEjG,MAAM,OAAO,GAAG,GAAG,GAAG,IAAI,GAAG,EAAE,CAAC;IAEhC,OAAO,IAAI,MAAM,CAAC,OAAO,EAAE,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;AAAA,CACxD;AAED,MAAM,KAAK,GAAG,SAAS,EAAE,CAAC;AAE1B,MAAM,UAAU,SAAS,CAAC,KAAa,EAAU;IAChD,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;QAC/B,MAAM,IAAI,SAAS,CAAC,gCAAgC,OAAO,KAAK,IAAI,CAAC,CAAC;IACvE,CAAC;IAED,sEAAsE;IACtE,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,QAAQ,CAAC,EAAE,CAAC;QAC5D,OAAO,KAAK,CAAC;IACd,CAAC;IAED,2EAA2E;IAC3E,6EAA6E;IAC7E,mDAAmD;IACnD,OAAO,KAAK,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;AAAA,CAChC","sourcesContent":["/*\n * Portions of this file are derived from:\n * - ansi-regex (https://github.com/chalk/ansi-regex)\n * - strip-ansi (https://github.com/chalk/strip-ansi)\n *\n * MIT License\n *\n * Copyright (c) Sindre Sorhus <sindresorhus@gmail.com> (https://sindresorhus.com)\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n */\n\nfunction ansiRegex({ onlyFirst = false }: { onlyFirst?: boolean } = {}): RegExp {\n\t// Valid string terminator sequences are BEL, ESC\\, and 0x9c\n\tconst ST = \"(?:\\\\u0007|\\\\u001B\\\\u005C|\\\\u009C)\";\n\n\t// OSC sequences only: ESC ] ... ST (non-greedy until the first ST)\n\tconst osc = `(?:\\\\u001B\\\\][\\\\s\\\\S]*?${ST})`;\n\n\t// CSI and related: ESC/C1, optional intermediates, optional params (supports ; and :) then final byte\n\tconst csi = \"[\\\\u001B\\\\u009B][[\\\\]()#;?]*(?:\\\\d{1,4}(?:[;:]\\\\d{0,4})*)?[\\\\dA-PR-TZcf-nq-uy=><~]\";\n\n\tconst pattern = `${osc}|${csi}`;\n\n\treturn new RegExp(pattern, onlyFirst ? undefined : \"g\");\n}\n\nconst regex = ansiRegex();\n\nexport function stripAnsi(value: string): string {\n\tif (typeof value !== \"string\") {\n\t\tthrow new TypeError(`Expected a \\`string\\`, got \\`${typeof value}\\``);\n\t}\n\n\t// Fast path: ANSI codes require ESC (7-bit) or CSI (8-bit) introducer\n\tif (!value.includes(\"\\u001B\") && !value.includes(\"\\u009B\")) {\n\t\treturn value;\n\t}\n\n\t// Even though the regex is global, we don't need to reset the `.lastIndex`\n\t// because unlike `.exec()` and `.test()`, `.replace()` does it automatically\n\t// and doing it manually has a performance penalty.\n\treturn value.replace(regex, \"\");\n}\n"]}
@@ -0,0 +1,7 @@
1
+ export interface DecodedHtmlEntity {
2
+ text: string;
3
+ length: number;
4
+ }
5
+ export declare function decodeHtmlEntity(entity: string): string | undefined;
6
+ export declare function decodeHtmlEntityAt(html: string, index: number): DecodedHtmlEntity | undefined;
7
+ //# sourceMappingURL=html.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"html.d.ts","sourceRoot":"","sources":["../../src/utils/html.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,iBAAiB;IACjC,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC;CACf;AASD,wBAAgB,gBAAgB,CAAC,MAAM,EAAE,MAAM,GAAG,MAAM,GAAG,SAAS,CAuBnE;AAED,wBAAgB,kBAAkB,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,iBAAiB,GAAG,SAAS,CAa7F","sourcesContent":["export interface DecodedHtmlEntity {\n\ttext: string;\n\tlength: number;\n}\n\nfunction decodeCodePoint(codePoint: number): string | undefined {\n\tif (!Number.isInteger(codePoint) || codePoint < 0 || codePoint > 0x10ffff) {\n\t\treturn undefined;\n\t}\n\treturn String.fromCodePoint(codePoint);\n}\n\nexport function decodeHtmlEntity(entity: string): string | undefined {\n\tswitch (entity) {\n\t\tcase \"amp\":\n\t\t\treturn \"&\";\n\t\tcase \"lt\":\n\t\t\treturn \"<\";\n\t\tcase \"gt\":\n\t\t\treturn \">\";\n\t\tcase \"quot\":\n\t\t\treturn '\"';\n\t\tcase \"apos\":\n\t\t\treturn \"'\";\n\t}\n\n\tif (entity.startsWith(\"#x\") || entity.startsWith(\"#X\")) {\n\t\treturn decodeCodePoint(Number.parseInt(entity.slice(2), 16));\n\t}\n\n\tif (entity.startsWith(\"#\")) {\n\t\treturn decodeCodePoint(Number.parseInt(entity.slice(1), 10));\n\t}\n\n\treturn undefined;\n}\n\nexport function decodeHtmlEntityAt(html: string, index: number): DecodedHtmlEntity | undefined {\n\tconst semicolonIndex = html.indexOf(\";\", index + 1);\n\tif (semicolonIndex === -1 || semicolonIndex - index > 16) {\n\t\treturn undefined;\n\t}\n\n\tconst entity = html.slice(index + 1, semicolonIndex);\n\tconst decoded = decodeHtmlEntity(entity);\n\tif (decoded === undefined) {\n\t\treturn undefined;\n\t}\n\n\treturn { text: decoded, length: semicolonIndex - index + 1 };\n}\n"]}
@@ -0,0 +1,40 @@
1
+ function decodeCodePoint(codePoint) {
2
+ if (!Number.isInteger(codePoint) || codePoint < 0 || codePoint > 0x10ffff) {
3
+ return undefined;
4
+ }
5
+ return String.fromCodePoint(codePoint);
6
+ }
7
+ export function decodeHtmlEntity(entity) {
8
+ switch (entity) {
9
+ case "amp":
10
+ return "&";
11
+ case "lt":
12
+ return "<";
13
+ case "gt":
14
+ return ">";
15
+ case "quot":
16
+ return '"';
17
+ case "apos":
18
+ return "'";
19
+ }
20
+ if (entity.startsWith("#x") || entity.startsWith("#X")) {
21
+ return decodeCodePoint(Number.parseInt(entity.slice(2), 16));
22
+ }
23
+ if (entity.startsWith("#")) {
24
+ return decodeCodePoint(Number.parseInt(entity.slice(1), 10));
25
+ }
26
+ return undefined;
27
+ }
28
+ export function decodeHtmlEntityAt(html, index) {
29
+ const semicolonIndex = html.indexOf(";", index + 1);
30
+ if (semicolonIndex === -1 || semicolonIndex - index > 16) {
31
+ return undefined;
32
+ }
33
+ const entity = html.slice(index + 1, semicolonIndex);
34
+ const decoded = decodeHtmlEntity(entity);
35
+ if (decoded === undefined) {
36
+ return undefined;
37
+ }
38
+ return { text: decoded, length: semicolonIndex - index + 1 };
39
+ }
40
+ //# sourceMappingURL=html.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"html.js","sourceRoot":"","sources":["../../src/utils/html.ts"],"names":[],"mappings":"AAKA,SAAS,eAAe,CAAC,SAAiB,EAAsB;IAC/D,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,SAAS,CAAC,IAAI,SAAS,GAAG,CAAC,IAAI,SAAS,GAAG,QAAQ,EAAE,CAAC;QAC3E,OAAO,SAAS,CAAC;IAClB,CAAC;IACD,OAAO,MAAM,CAAC,aAAa,CAAC,SAAS,CAAC,CAAC;AAAA,CACvC;AAED,MAAM,UAAU,gBAAgB,CAAC,MAAc,EAAsB;IACpE,QAAQ,MAAM,EAAE,CAAC;QAChB,KAAK,KAAK;YACT,OAAO,GAAG,CAAC;QACZ,KAAK,IAAI;YACR,OAAO,GAAG,CAAC;QACZ,KAAK,IAAI;YACR,OAAO,GAAG,CAAC;QACZ,KAAK,MAAM;YACV,OAAO,GAAG,CAAC;QACZ,KAAK,MAAM;YACV,OAAO,GAAG,CAAC;IACb,CAAC;IAED,IAAI,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,EAAE,CAAC;QACxD,OAAO,eAAe,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC;IAC9D,CAAC;IAED,IAAI,MAAM,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;QAC5B,OAAO,eAAe,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC;IAC9D,CAAC;IAED,OAAO,SAAS,CAAC;AAAA,CACjB;AAED,MAAM,UAAU,kBAAkB,CAAC,IAAY,EAAE,KAAa,EAAiC;IAC9F,MAAM,cAAc,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,KAAK,GAAG,CAAC,CAAC,CAAC;IACpD,IAAI,cAAc,KAAK,CAAC,CAAC,IAAI,cAAc,GAAG,KAAK,GAAG,EAAE,EAAE,CAAC;QAC1D,OAAO,SAAS,CAAC;IAClB,CAAC;IAED,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,CAAC,EAAE,cAAc,CAAC,CAAC;IACrD,MAAM,OAAO,GAAG,gBAAgB,CAAC,MAAM,CAAC,CAAC;IACzC,IAAI,OAAO,KAAK,SAAS,EAAE,CAAC;QAC3B,OAAO,SAAS,CAAC;IAClB,CAAC;IAED,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,EAAE,cAAc,GAAG,KAAK,GAAG,CAAC,EAAE,CAAC;AAAA,CAC7D","sourcesContent":["export interface DecodedHtmlEntity {\n\ttext: string;\n\tlength: number;\n}\n\nfunction decodeCodePoint(codePoint: number): string | undefined {\n\tif (!Number.isInteger(codePoint) || codePoint < 0 || codePoint > 0x10ffff) {\n\t\treturn undefined;\n\t}\n\treturn String.fromCodePoint(codePoint);\n}\n\nexport function decodeHtmlEntity(entity: string): string | undefined {\n\tswitch (entity) {\n\t\tcase \"amp\":\n\t\t\treturn \"&\";\n\t\tcase \"lt\":\n\t\t\treturn \"<\";\n\t\tcase \"gt\":\n\t\t\treturn \">\";\n\t\tcase \"quot\":\n\t\t\treturn '\"';\n\t\tcase \"apos\":\n\t\t\treturn \"'\";\n\t}\n\n\tif (entity.startsWith(\"#x\") || entity.startsWith(\"#X\")) {\n\t\treturn decodeCodePoint(Number.parseInt(entity.slice(2), 16));\n\t}\n\n\tif (entity.startsWith(\"#\")) {\n\t\treturn decodeCodePoint(Number.parseInt(entity.slice(1), 10));\n\t}\n\n\treturn undefined;\n}\n\nexport function decodeHtmlEntityAt(html: string, index: number): DecodedHtmlEntity | undefined {\n\tconst semicolonIndex = html.indexOf(\";\", index + 1);\n\tif (semicolonIndex === -1 || semicolonIndex - index > 16) {\n\t\treturn undefined;\n\t}\n\n\tconst entity = html.slice(index + 1, semicolonIndex);\n\tconst decoded = decodeHtmlEntity(entity);\n\tif (decoded === undefined) {\n\t\treturn undefined;\n\t}\n\n\treturn { text: decoded, length: semicolonIndex - index + 1 };\n}\n"]}
@@ -1,2 +1,3 @@
1
+ export declare function detectSupportedImageMimeType(buffer: Uint8Array): string | null;
1
2
  export declare function detectSupportedImageMimeTypeFromFile(filePath: string): Promise<string | null>;
2
3
  //# sourceMappingURL=mime.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"mime.d.ts","sourceRoot":"","sources":["../../src/utils/mime.ts"],"names":[],"mappings":"AAOA,wBAAsB,oCAAoC,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CAsBnG","sourcesContent":["import { open } from \"node:fs/promises\";\nimport { fileTypeFromBuffer } from \"file-type\";\n\nconst IMAGE_MIME_TYPES = new Set([\"image/jpeg\", \"image/png\", \"image/gif\", \"image/webp\"]);\n\nconst FILE_TYPE_SNIFF_BYTES = 4100;\n\nexport async function detectSupportedImageMimeTypeFromFile(filePath: string): Promise<string | null> {\n\tconst fileHandle = await open(filePath, \"r\");\n\ttry {\n\t\tconst buffer = Buffer.alloc(FILE_TYPE_SNIFF_BYTES);\n\t\tconst { bytesRead } = await fileHandle.read(buffer, 0, FILE_TYPE_SNIFF_BYTES, 0);\n\t\tif (bytesRead === 0) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst fileType = await fileTypeFromBuffer(buffer.subarray(0, bytesRead));\n\t\tif (!fileType) {\n\t\t\treturn null;\n\t\t}\n\n\t\tif (!IMAGE_MIME_TYPES.has(fileType.mime)) {\n\t\t\treturn null;\n\t\t}\n\n\t\treturn fileType.mime;\n\t} finally {\n\t\tawait fileHandle.close();\n\t}\n}\n"]}
1
+ {"version":3,"file":"mime.d.ts","sourceRoot":"","sources":["../../src/utils/mime.ts"],"names":[],"mappings":"AAKA,wBAAgB,4BAA4B,CAAC,MAAM,EAAE,UAAU,GAAG,MAAM,GAAG,IAAI,CAc9E;AAED,wBAAsB,oCAAoC,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CASnG","sourcesContent":["import { open } from \"node:fs/promises\";\n\nconst IMAGE_TYPE_SNIFF_BYTES = 4100;\nconst PNG_SIGNATURE = [0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a];\n\nexport function detectSupportedImageMimeType(buffer: Uint8Array): string | null {\n\tif (startsWith(buffer, [0xff, 0xd8, 0xff])) {\n\t\treturn buffer[3] === 0xf7 ? null : \"image/jpeg\";\n\t}\n\tif (startsWith(buffer, PNG_SIGNATURE)) {\n\t\treturn isPng(buffer) && !isAnimatedPng(buffer) ? \"image/png\" : null;\n\t}\n\tif (startsWithAscii(buffer, 0, \"GIF\")) {\n\t\treturn \"image/gif\";\n\t}\n\tif (startsWithAscii(buffer, 0, \"RIFF\") && startsWithAscii(buffer, 8, \"WEBP\")) {\n\t\treturn \"image/webp\";\n\t}\n\treturn null;\n}\n\nexport async function detectSupportedImageMimeTypeFromFile(filePath: string): Promise<string | null> {\n\tconst fileHandle = await open(filePath, \"r\");\n\ttry {\n\t\tconst buffer = Buffer.alloc(IMAGE_TYPE_SNIFF_BYTES);\n\t\tconst { bytesRead } = await fileHandle.read(buffer, 0, IMAGE_TYPE_SNIFF_BYTES, 0);\n\t\treturn detectSupportedImageMimeType(buffer.subarray(0, bytesRead));\n\t} finally {\n\t\tawait fileHandle.close();\n\t}\n}\n\nfunction isPng(buffer: Uint8Array): boolean {\n\treturn (\n\t\tbuffer.length >= 16 && readUint32BE(buffer, PNG_SIGNATURE.length) === 13 && startsWithAscii(buffer, 12, \"IHDR\")\n\t);\n}\n\nfunction isAnimatedPng(buffer: Uint8Array): boolean {\n\tlet offset = PNG_SIGNATURE.length;\n\twhile (offset + 8 <= buffer.length) {\n\t\tconst chunkLength = readUint32BE(buffer, offset);\n\t\tconst chunkTypeOffset = offset + 4;\n\t\tif (startsWithAscii(buffer, chunkTypeOffset, \"acTL\")) return true;\n\t\tif (startsWithAscii(buffer, chunkTypeOffset, \"IDAT\")) return false;\n\n\t\tconst nextOffset = offset + 8 + chunkLength + 4;\n\t\tif (nextOffset <= offset || nextOffset > buffer.length) return false;\n\t\toffset = nextOffset;\n\t}\n\treturn false;\n}\n\nfunction readUint32BE(buffer: Uint8Array, offset: number): number {\n\treturn (\n\t\t(buffer[offset] ?? 0) * 0x1000000 +\n\t\t((buffer[offset + 1] ?? 0) << 16) +\n\t\t((buffer[offset + 2] ?? 0) << 8) +\n\t\t(buffer[offset + 3] ?? 0)\n\t);\n}\n\nfunction startsWith(buffer: Uint8Array, bytes: number[]): boolean {\n\tif (buffer.length < bytes.length) return false;\n\treturn bytes.every((byte, index) => buffer[index] === byte);\n}\n\nfunction startsWithAscii(buffer: Uint8Array, offset: number, text: string): boolean {\n\tif (buffer.length < offset + text.length) return false;\n\tfor (let index = 0; index < text.length; index++) {\n\t\tif (buffer[offset + index] !== text.charCodeAt(index)) return false;\n\t}\n\treturn true;\n}\n"]}