This page covers the tool-call lifecycle: reading tool_use blocks from Claude's response, formatting tool_result blocks in your reply, and signaling errors. For the SDK abstraction that handles this automatically, see Tool Runner.
Simpler with Tool Runner: The manual tool handling described on this page is automatically managed by Tool Runner. Use this page when you need custom control over tool execution.
Claude's response differs based on whether it uses a client or server tool.
The response will have a stop_reason of tool_use and one or more tool_use content blocks that include:
id: A unique identifier for this particular tool use block. This will be used to match up the tool results later.name: The name of the tool being used.input: An object containing the input being passed to the tool, conforming to the tool's input_schema.When you receive a tool use response for a client tool, you should:
name, id, and input from the tool_use block.input.role of user, and a content block containing the tool_result type and the following information:
tool_use_id: The id of the tool use request this is a result for.content: The result of the tool, as a string (for example, "content": "15 degrees"), a list of nested content blocks (for example, "content": [{"type": "text", "text": "15 degrees"}]), or a list of document blocks (for example, "content": [{"type": "document", "source": {"type": "text", "media_type": "text/plain", "data": "15 degrees"}}]). These content blocks can use the text, image, or document types.is_error (optional): Set to true if the tool execution resulted in an error.Important formatting requirements:
For example, this will cause a 400 error:
{
"role": "user",
"content": [
{ "type": "text", "text": "Here are the results:" }, // ❌ Text before tool_result
{ "type": "tool_result", "tool_use_id": "toolu_01" /* ... */ }
]
}This is correct:
{
"role": "user",
"content": [
{ "type": "tool_result", "tool_use_id": "toolu_01" /* ... */ },
{ "type": "text", "text": "What should I do next?" } // ✅ Text after tool_result
]
}If you receive an error like "tool_use ids were found without tool_result blocks immediately after", check that your tool results are formatted correctly.
After receiving the tool result, Claude will use that information to continue generating a response to the original user prompt.
Claude executes the tool internally and incorporates the results directly into its response without requiring additional user interaction.
Differences from other APIs
Unlike APIs that separate tool use or use special roles like tool or function, the Claude API integrates tools directly into the user and assistant message structure.
Messages contain arrays of text, image, tool_use, and tool_result blocks. user messages include client content and tool_result, while assistant messages contain AI-generated content and tool_use.
There are a few different types of errors that can occur when using tools with Claude:
Was this page helpful?