π File detail
utils/streamlinedTransform.ts
π― Use case
This file lives under βutils/β, which covers cross-cutting helpers (shell, tempfiles, settings, messages, process input, β¦). On the API surface it exposes createStreamlinedTransformer and shouldIncludeInStreamlined β mainly functions, hooks, or classes. Dependencies touch src. What the file header says: Transforms SDK messages for streamlined output mode. Streamlined mode is a "distillation-resistant" output format that: - Keeps text messages intact - Summarizes tool calls with cumulative counts (resets when text appears) - Omits thinking content - Strips tool list and model inf.
Generated from folder role, exports, dependency roots, and inline comments β not hand-reviewed for every path.
π§ Inline summary
Transforms SDK messages for streamlined output mode. Streamlined mode is a "distillation-resistant" output format that: - Keeps text messages intact - Summarizes tool calls with cumulative counts (resets when text appears) - Omits thinking content - Strips tool list and model info from init messages
π€ Exports (heuristic)
createStreamlinedTransformershouldIncludeInStreamlined
π External import roots
Package roots from from "β¦" (relative paths omitted).
src
π₯οΈ Source preview
/**
* Transforms SDK messages for streamlined output mode.
*
* Streamlined mode is a "distillation-resistant" output format that:
* - Keeps text messages intact
* - Summarizes tool calls with cumulative counts (resets when text appears)
* - Omits thinking content
* - Strips tool list and model info from init messages
*/
import type { SDKAssistantMessage } from 'src/entrypoints/agentSdkTypes.js'
import type { StdoutMessage } from 'src/entrypoints/sdk/controlTypes.js'
import { FILE_EDIT_TOOL_NAME } from 'src/tools/FileEditTool/constants.js'
import { FILE_READ_TOOL_NAME } from 'src/tools/FileReadTool/prompt.js'
import { FILE_WRITE_TOOL_NAME } from 'src/tools/FileWriteTool/prompt.js'
import { GLOB_TOOL_NAME } from 'src/tools/GlobTool/prompt.js'
import { GREP_TOOL_NAME } from 'src/tools/GrepTool/prompt.js'
import { LIST_MCP_RESOURCES_TOOL_NAME } from 'src/tools/ListMcpResourcesTool/prompt.js'
import { LSP_TOOL_NAME } from 'src/tools/LSPTool/prompt.js'
import { NOTEBOOK_EDIT_TOOL_NAME } from 'src/tools/NotebookEditTool/constants.js'
import { TASK_STOP_TOOL_NAME } from 'src/tools/TaskStopTool/prompt.js'
import { WEB_SEARCH_TOOL_NAME } from 'src/tools/WebSearchTool/prompt.js'
import { extractTextContent } from 'src/utils/messages.js'
import { SHELL_TOOL_NAMES } from 'src/utils/shell/shellToolUtils.js'
import { capitalize } from 'src/utils/stringUtils.js'
type ToolCounts = {
searches: number
reads: number
writes: number
commands: number
other: number
}
/**
* Tool categories for summarization.
*/
const SEARCH_TOOLS = [
GREP_TOOL_NAME,
GLOB_TOOL_NAME,
WEB_SEARCH_TOOL_NAME,
LSP_TOOL_NAME,
]
const READ_TOOLS = [FILE_READ_TOOL_NAME, LIST_MCP_RESOURCES_TOOL_NAME]
const WRITE_TOOLS = [
FILE_WRITE_TOOL_NAME,
FILE_EDIT_TOOL_NAME,
NOTEBOOK_EDIT_TOOL_NAME,
]
const COMMAND_TOOLS = [...SHELL_TOOL_NAMES, 'Tmux', TASK_STOP_TOOL_NAME]
function categorizeToolName(toolName: string): keyof ToolCounts {
if (SEARCH_TOOLS.some(t => toolName.startsWith(t))) return 'searches'
if (READ_TOOLS.some(t => toolName.startsWith(t))) return 'reads'
if (WRITE_TOOLS.some(t => toolName.startsWith(t))) return 'writes'
if (COMMAND_TOOLS.some(t => toolName.startsWith(t))) return 'commands'
return 'other'
}
function createEmptyToolCounts(): ToolCounts {
return {
searches: 0,
reads: 0,
writes: 0,
commands: 0,
other: 0,
}
}
/**
* Generate a summary text for tool counts.
*/
function getToolSummaryText(counts: ToolCounts): string | undefined {
const parts: string[] = []
// Use similar phrasing to collapseReadSearch.ts
if (counts.searches > 0) {
parts.push(
`searched ${counts.searches} ${counts.searches === 1 ? 'pattern' : 'patterns'}`,
)
}
if (counts.reads > 0) {
parts.push(`read ${counts.reads} ${counts.reads === 1 ? 'file' : 'files'}`)
}
if (counts.writes > 0) {
parts.push(
`wrote ${counts.writes} ${counts.writes === 1 ? 'file' : 'files'}`,
)
}
if (counts.commands > 0) {
parts.push(
`ran ${counts.commands} ${counts.commands === 1 ? 'command' : 'commands'}`,
)
}
if (counts.other > 0) {
parts.push(`${counts.other} other ${counts.other === 1 ? 'tool' : 'tools'}`)
}
if (parts.length === 0) {
return undefined
}
return capitalize(parts.join(', '))
}
/**
* Count tool uses in an assistant message and add to existing counts.
*/
function accumulateToolUses(
message: SDKAssistantMessage,
counts: ToolCounts,
): void {
const content = message.message.content
if (!Array.isArray(content)) {
return
}
for (const block of content) {
if (block.type === 'tool_use' && 'name' in block) {
const category = categorizeToolName(block.name as string)
counts[category]++
}
}
}
/**
* Create a stateful transformer that accumulates tool counts between text messages.
* Tool counts reset when a message with text content is encountered.
*/
export function createStreamlinedTransformer(): (
message: StdoutMessage,
) => StdoutMessage | null {
let cumulativeCounts = createEmptyToolCounts()
return function transformToStreamlined(
message: StdoutMessage,
): StdoutMessage | null {
switch (message.type) {
case 'assistant': {
const content = message.message.content
const text = Array.isArray(content)
? extractTextContent(content, '\n').trim()
: ''
// Accumulate tool counts from this message
accumulateToolUses(message, cumulativeCounts)
if (text.length > 0) {
// Text message: emit text only, reset counts
cumulativeCounts = createEmptyToolCounts()
return {
type: 'streamlined_text',
text,
session_id: message.session_id,
uuid: message.uuid,
}
}
// Tool-only message: emit cumulative tool summary
const toolSummary = getToolSummaryText(cumulativeCounts)
if (!toolSummary) {
return null
}
return {
type: 'streamlined_tool_use_summary',
tool_summary: toolSummary,
session_id: message.session_id,
uuid: message.uuid,
}
}
case 'result':
// Keep result messages as-is (they have structured_output, permission_denials)
return message
case 'system':
case 'user':
case 'stream_event':
case 'tool_progress':
case 'auth_status':
case 'rate_limit_event':
case 'control_response':
case 'control_request':
case 'control_cancel_request':
case 'keep_alive':
return null
default:
return null
}
}
}
/**
* Check if a message should be included in streamlined output.
* Useful for filtering before transformation.
*/
export function shouldIncludeInStreamlined(message: StdoutMessage): boolean {
return message.type === 'assistant' || message.type === 'result'
}