Loading...
  • Messages
  • Managed Agents
  • Admin
Search...
⌘K
Release notes
OverviewSystem prompts
Migration guides
Migrating to Managed Agents
Log in
Migrating to Managed Agents
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...

Solutions

  • AI agents
  • Code modernization
  • Coding
  • Customer support
  • Education
  • Financial services
  • Government
  • Life sciences

Partners

  • Amazon Bedrock
  • Google Cloud's Vertex AI

Learn

  • Blog
  • Courses
  • Use cases
  • Connectors
  • Customer stories
  • Engineering at Anthropic
  • Events
  • Powered by Claude
  • Service partners
  • Startups program

Company

  • Anthropic
  • Careers
  • Economic Futures
  • Research
  • News
  • Responsible Scaling Policy
  • Security and compliance
  • Transparency

Learn

  • Blog
  • Courses
  • Use cases
  • Connectors
  • Customer stories
  • Engineering at Anthropic
  • Events
  • Powered by Claude
  • Service partners
  • Startups program

Help and security

  • Availability
  • Status
  • Support
  • Discord

Terms and policies

  • Privacy policy
  • Responsible disclosure policy
  • Terms of service: Commercial
  • Terms of service: Consumer
  • Usage policy
Release notes/Migration guides

Migration

Move an existing agent built on the Messages API or the Claude Agent SDK to Claude Managed Agents.

Claude Managed Agents replaces your hand-written agent loop with managed infrastructure. This page covers what changes when you migrate from a custom loop built on the Messages API or from the Claude Agent SDK.

All Managed Agents API requests require the managed-agents-2026-04-01 beta header. The SDK sets the beta header automatically.

From a Messages API agent loop

If you built an agent by calling messages.create in a while loop, executing tool calls yourself, and appending results to the conversation history, most of that code goes away.

What you stop managing

BeforeAfter
You maintain the conversation history array and pass it back on every turn.The session stores history server-side. Send events, receive events.
You iterate tool_use content blocks, run each tool, and loop back with tool_result messages.Pre-built tools run inside the container automatically. You only handle custom tools through agent.custom_tool_use events.
You provision your own sandbox for running agent-generated code.The session container handles code execution, file operations, and bash.
You decide when the loop is done.The session emits session.status_idle when the agent has nothing more to do.

Code comparison

Before (Messages API loop, simplified):

messages = [{"role": "user", "content": task}]
while True:
    response = client.messages.create(
        model="claude-opus-4-7",
        max_tokens=1024,
        messages=messages,
        tools=tools,
    )
    messages.append({"role": "assistant", "content": response.content})
    if response.stop_reason == "end_turn":
        break
    for block in response.content:
        if block.type == "tool_use":
            result = execute_tool(block.name, block.input)
            messages.append(
                {
                    "role": "user",
                    "content": [
                        {
                            "type": "tool_result",
                            "tool_use_id": block.id,
                            "content": result,
                        }
                    ],
                }
            )

After (Claude Managed Agents):

agent = client.beta.agents.create(
    name="Task Runner",
    model="claude-opus-4-7",
    tools=[{"type": "agent_toolset_20260401"}],
)

session = client.beta.sessions.create(
    agent={"type": "agent", "id": agent.id, "version": agent.version},
    environment_id=environment.id,
)

with client.beta.sessions.events.stream(session.id) as stream:
    client.beta.sessions.events.send(
        session.id,
        events=[{"type": "user.message", "content": [{"type": "text", "text": task}]}],
    )
    for event in stream:
        if event.type == "session.status_idle":
            break

What you still control

  • System prompt and model: Same fields, now on the agent definition.
  • Custom tools: Still declared with JSON Schema. Execution moves from inline handling to responding to agent.custom_tool_use events. See Session event stream.
  • Context: You can still inject context through the system prompt, file resources, or skills.

From the Claude Agent SDK

If you built with the Claude Agent SDK, you're already working with agents, tools, and sessions as concepts. The difference is where they run: the SDK executes in a process you operate, while Managed Agents runs in Anthropic's infrastructure. Most of the migration is mapping SDK configuration objects to their API-side equivalents.

What changes

Agent SDKManaged Agents
ClaudeAgentOptions(...) constructed per runclient.beta.agents.create(...) once; the Agent is persisted and versioned server-side. See Agent setup.
async with ClaudeSDKClient(...) or query(...)client.beta.sessions.create(...) then send and receive events.
@tool-decorated functions dispatched automatically by the SDKDeclare as {"type": "custom", ...} on the Agent; your client handles agent.custom_tool_use events and replies with user.custom_tool_result. See Tools.
Built-in tools run in your process against your filesystem{"type": "agent_toolset_20260401"} runs the same tools inside the session container against /workspace.
cwd, add_dirs point at local pathsUpload or mount files as session resources.
system_prompt and the CLAUDE.md hierarchyA single system string on the Agent. Each update produces a new server-side version; pin sessions to a specific version to promote or roll back without a deploy. See Agent setup.
mcp_servers configured and authenticated in one placeDeclare servers on the Agent; provide credentials through a Vault on the Session.
permission_mode, can_use_toolPer-tool permission_policy; send user.tool_confirmation events for always_ask tools.

Code comparison

Before (Agent SDK):

from claude_agent_sdk import (
    ClaudeAgentOptions,
    ClaudeSDKClient,
    create_sdk_mcp_server,
    tool,
)


@tool("get_weather", "Get the current weather for a city.", {"city": str})
async def get_weather(args: dict) -> dict:
    return {"content": [{"type": "text", "text": f"{args['city']}: 18°C, clear"}]}


options = ClaudeAgentOptions(
    model="claude-opus-4-7",
    system_prompt="You are a concise weather assistant.",
    mcp_servers={
        "weather": create_sdk_mcp_server("weather", "1.0", tools=[get_weather])
    },
)

async with ClaudeSDKClient(options=options) as agent:
    await agent.query("What's the weather in Tokyo?")
    async for msg in agent.receive_response():
        print(msg)

After (Managed Agents):

from anthropic import Anthropic

client = Anthropic()

agent = client.beta.agents.create(
    name="weather-agent",
    model="claude-opus-4-7",
    system="You are a concise weather assistant.",
    tools=[
        {
            "type": "custom",
            "name": "get_weather",
            "description": "Get the current weather for a city.",
            "input_schema": {
                "type": "object",
                "properties": {"city": {"type": "string"}},
                "required": ["city"],
            },
        }
    ],
)
environment = client.beta.environments.create(
    name="weather-env",
    config={"type": "cloud", "networking": {"type": "unrestricted"}},
)

session = client.beta.sessions.create(
    agent={"type": "agent", "id": agent.id, "version": agent.version},
    environment_id=environment.id,
)


def get_weather(city: str) -> str:
    return f"{city}: 18°C, clear"


with client.beta.sessions.events.stream(session.id) as stream:
    client.beta.sessions.events.send(
        session.id,
        events=[
            {
                "type": "user.message",
                "content": [{"type": "text", "text": "What's the weather in Tokyo?"}],
            }
        ],
    )
    for ev in stream:
        if ev.type == "agent.message":
            print("".join(block.text for block in ev.content if block.type == "text"))
        elif ev.type == "agent.custom_tool_use":
            result = get_weather(**ev.input)
            client.beta.sessions.events.send(
                session.id,
                events=[
                    {
                        "type": "user.custom_tool_result",
                        "custom_tool_use_id": ev.id,
                        "content": [{"type": "text", "text": result}],
                    }
                ],
            )
        elif (
            ev.type == "session.status_idle"
            and ev.stop_reason
            and ev.stop_reason.type == "end_turn"
        ):
            break

The Agent and Environment are created once and reused across sessions. The tool function still runs in your process; the difference is that you read the agent.custom_tool_use event and send the result explicitly instead of the SDK dispatching it for you.

Features that move to your client

The tradeoff for Anthropic running the agent loop is that a few things the SDK handled automatically become your client's responsibility.

SDK featureManaged Agents approach
Plan modeRun a planning-only session first, then a second session to run the plan.
Output styles, slash commandsApply in your client before sending user.message or after receiving agent.message.
PreToolUse / PostToolUse hooksYour client already sees every agent.custom_tool_use event before responding; put the logic there. For built-in tools, use permission_policy: always_ask.
max_turnsCount turns client-side.

Migration checklist

  1. Create an environment with the networking and runtimes your agent needs.
  2. Port your system prompt and tool selection to an agent definition.
  3. Replace your loop with sessions.create and sessions.events.stream.
  4. For any local files the agent reads, upload them through the Files API and mount them as resources.
  5. For any custom tool handlers, move execution into your event loop as responses to agent.custom_tool_use events.
  6. Verify with a test session before pointing production traffic at the new flow.

Migrating between model versions

When a new Claude model is released, migrating a Claude Managed Agents integration is typically a one-field change: update model on your agent definition and the change takes effect on the next session you create.

ant beta:agents update \
  --agent-id "$AGENT_ID" \
  --version "$AGENT_VERSION" \
  --model claude-opus-4-7

Most model-level behavior changes documented in the Messages API migration guide do not require action on your side:

  • Request parameter changes (max_tokens defaults, thinking configuration) are handled by the Claude Managed Agents runtime. These fields are not exposed on the agent definition.
  • Assistant message prefilling does not exist in the event-based session model, so its removal on newer models is a no-op.
  • Tool argument JSON escaping is parsed by the runtime before you receive agent.custom_tool_use events. You see structured data, not raw strings.

The behavior descriptions in the Messages API guide (what the model does differently) still apply. The migration steps (how to change your request code) do not.

Was this page helpful?

  • From a Messages API agent loop
  • What you stop managing
  • Code comparison
  • What you still control
  • From the Claude Agent SDK
  • What changes
  • Code comparison
  • Features that move to your client
  • Migration checklist
  • Migrating between model versions