groupToolUses.ts
utils/groupToolUses.ts
183
Lines
5537
Bytes
3
Exports
4
Imports
10
Keywords
What this is
This page documents one file from the repository and includes its full source so you can read it without leaving the docs site.
Beginner explanation
This file is one piece of the larger system. Its name, directory, imports, and exports show where it fits. Start by reading the exports and related files first.
How it is used
Start from the exports list and related files. Those are the easiest clues for where this file fits into the system.
Expert explanation
Architecturally, this file intersects with tool-system. It contains 183 lines, 4 detected imports, and 3 detected exports.
Important relationships
Detected exports
MessageWithoutProgressGroupingResultapplyGrouping
Keywords
messagesgroupinfotoolsmessagecontenttooltoolnamegroupsmessageid
Detected imports
@anthropic-ai/sdk/resources/beta/messages/messages.mjs@anthropic-ai/sdk/resources/messages/messages.mjs../Tool.js../types/message.js
Source notes
This page embeds the full file contents. Small or leaf files are still indexed honestly instead of being over-explained.
Full source
import type { BetaToolUseBlock } from '@anthropic-ai/sdk/resources/beta/messages/messages.mjs'
import type { ToolResultBlockParam } from '@anthropic-ai/sdk/resources/messages/messages.mjs'
import type { Tools } from '../Tool.js'
import type {
GroupedToolUseMessage,
NormalizedAssistantMessage,
NormalizedMessage,
NormalizedUserMessage,
ProgressMessage,
RenderableMessage,
} from '../types/message.js'
export type MessageWithoutProgress = Exclude<NormalizedMessage, ProgressMessage>
export type GroupingResult = {
messages: RenderableMessage[]
}
// Cache the set of tool names that support grouped rendering, keyed by the
// tools array reference. The tools array is stable across renders (only
// replaced on MCP connect/disconnect), so this avoids rebuilding the set on
// every call. WeakMap lets old entries be GC'd when the array is replaced.
const GROUPING_CACHE = new WeakMap<Tools, Set<string>>()
function getToolsWithGrouping(tools: Tools): Set<string> {
let cached = GROUPING_CACHE.get(tools)
if (!cached) {
cached = new Set(tools.filter(t => t.renderGroupedToolUse).map(t => t.name))
GROUPING_CACHE.set(tools, cached)
}
return cached
}
function getToolUseInfo(
msg: MessageWithoutProgress,
): { messageId: string; toolUseId: string; toolName: string } | null {
if (msg.type === 'assistant' && msg.message.content[0]?.type === 'tool_use') {
const content = msg.message.content[0]
return {
messageId: msg.message.id,
toolUseId: content.id,
toolName: content.name,
}
}
return null
}
/**
* Groups tool uses by message.id (same API response) if the tool supports grouped rendering.
* Only groups 2+ tools of the same type from the same message.
* Also collects corresponding tool_results and attaches them to the grouped message.
* When verbose is true, skips grouping so messages render at original positions.
*/
export function applyGrouping(
messages: MessageWithoutProgress[],
tools: Tools,
verbose: boolean = false,
): GroupingResult {
// In verbose mode, don't group - each message renders at its original position
if (verbose) {
return {
messages: messages,
}
}
const toolsWithGrouping = getToolsWithGrouping(tools)
// First pass: group tool uses by message.id + tool name
const groups = new Map<
string,
NormalizedAssistantMessage<BetaToolUseBlock>[]
>()
for (const msg of messages) {
const info = getToolUseInfo(msg)
if (info && toolsWithGrouping.has(info.toolName)) {
const key = `${info.messageId}:${info.toolName}`
const group = groups.get(key) ?? []
group.push(msg as NormalizedAssistantMessage<BetaToolUseBlock>)
groups.set(key, group)
}
}
// Identify valid groups (2+ items) and collect their tool use IDs
const validGroups = new Map<
string,
NormalizedAssistantMessage<BetaToolUseBlock>[]
>()
const groupedToolUseIds = new Set<string>()
for (const [key, group] of groups) {
if (group.length >= 2) {
validGroups.set(key, group)
for (const msg of group) {
const info = getToolUseInfo(msg)
if (info) {
groupedToolUseIds.add(info.toolUseId)
}
}
}
}
// Collect result messages for grouped tool_uses
// Map from tool_use_id to the user message containing that result
const resultsByToolUseId = new Map<string, NormalizedUserMessage>()
for (const msg of messages) {
if (msg.type === 'user') {
for (const content of msg.message.content) {
if (
content.type === 'tool_result' &&
groupedToolUseIds.has(content.tool_use_id)
) {
resultsByToolUseId.set(content.tool_use_id, msg)
}
}
}
}
// Second pass: build output, emitting each group only once
const result: RenderableMessage[] = []
const emittedGroups = new Set<string>()
for (const msg of messages) {
const info = getToolUseInfo(msg)
if (info) {
const key = `${info.messageId}:${info.toolName}`
const group = validGroups.get(key)
if (group) {
if (!emittedGroups.has(key)) {
emittedGroups.add(key)
const firstMsg = group[0]!
// Collect results for this group
const results: NormalizedUserMessage[] = []
for (const assistantMsg of group) {
const toolUseId = (
assistantMsg.message.content[0] as { id: string }
).id
const resultMsg = resultsByToolUseId.get(toolUseId)
if (resultMsg) {
results.push(resultMsg)
}
}
const groupedMessage: GroupedToolUseMessage = {
type: 'grouped_tool_use',
toolName: info.toolName,
messages: group,
results,
displayMessage: firstMsg,
uuid: `grouped-${firstMsg.uuid}`,
timestamp: firstMsg.timestamp,
messageId: info.messageId,
}
result.push(groupedMessage)
}
continue
}
}
// Skip user messages whose tool_results are all grouped
if (msg.type === 'user') {
const toolResults = msg.message.content.filter(
(c): c is ToolResultBlockParam => c.type === 'tool_result',
)
if (toolResults.length > 0) {
const allGrouped = toolResults.every(tr =>
groupedToolUseIds.has(tr.tool_use_id),
)
if (allGrouped) {
continue
}
}
}
result.push(msg)
}
return { messages: result }
}