VOOZH about

URL: https://platform.claude.com/docs/en/build-with-claude/task-budgets

⇱ Task budgets - Claude API Docs


Task budgets (beta)
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...

This feature is eligible for Zero Data Retention (ZDR). When your organization has a ZDR arrangement, data sent through this feature is not stored after the API response is returned.

Task budgets let you tell Claude how many tokens it has for a full agentic loop, including thinking, tool calls, tool results, and output. The model sees a running countdown and uses it to prioritize work and finish gracefully as the budget is consumed.

Task budgets are in beta on Claude Fable 5, Claude Mythos 5, Claude Opus 4.8, and Claude Opus 4.7. Set the task-budgets-2026-03-13 beta header to opt in.

When to use task budgets

Task budgets work best for agentic workflows where Claude makes multiple tool calls and decisions before finalizing its output to await the next human response. Use them when:

  • You want Claude to self-regulate token spend on long-horizon tasks.
  • You have a predictable per-task cost or latency ceiling to enforce.
  • You want the model to finish gracefully (summarize findings, report progress) as it approaches the budget rather than cutting off mid-action.

Task budgets complement the effort parameter: effort controls how thoroughly Claude reasons about each step, while task budgets cap the total work Claude can do across an agentic loop.

Setting a task budget

Add task_budget to and include the beta header:

Was this page helpful?

output_config
client = anthropic.Anthropic()

with client.beta.messages.stream(
 model="claude-opus-4-8",
 max_tokens=128000,
 output_config={
 "effort": "high",
 "task_budget": {"type": "tokens", "total": 64000},
 },
 messages=[
 {"role": "user", "content": "Review the codebase and propose a refactor plan."}
 ],
 betas=["task-budgets-2026-03-13"],
) as stream:
 response = stream.get_final_message()

print(response.usage)

The task_budget object has three fields:

How the budget countdown works

Claude sees a budget-countdown marker injected server-side throughout the conversation. The marker shows how many tokens remain in the current agentic loop and updates as the model generates thinking, tool calls, and output, and as it processes tool results. Claude uses this signal to pace itself and finish gracefully as the budget is consumed.

The countdown is visible only to the model. API responses do not include a remaining-budget field: there is no task_budget information in the response usage object, and SDKs have no accessor for it. To track spend client-side, sum token usage across the requests in your loop as shown in Measure your current usage, or pass your own figure forward with remaining when carrying a budget across compaction.

The countdown reflects tokens Claude has processed in the current agentic loop, not tokens you resend between turns. If your client sends the full conversation history on every follow-up request, your client-side token count may differ from the budget Claude is tracking. If you also decrement remaining while resending full history, the model sees an under-reported budget and the countdown drops faster than it should, causing Claude to wrap up earlier than the budget actually allows. Set a generous budget and let the model self-regulate against the countdown rather than trying to mirror it client-side.

Worked example: budget counting across turns

The task budget counts what Claude sees (thinking, tool calls and results, and text), not what's in your request payload. In an agentic loop your client resends the full conversation on every request, so the payload grows turn over turn, but the budget only decrements by the tokens Claude sees this turn.

Consider a loop with task_budget: {type: "tokens", total: 100000} and a single bash tool.

Turn 1. You send the initial request:

{
 "messages": [
 { "role": "user", "content": "Audit this repo for security issues and report findings." }
 ]
}

Claude thinks, then emits a tool call and stops with stop_reason: "tool_use":

{
 "role": "assistant",
 "content": [
 {
 "type": "thinking",
 "thinking": "I'll start by listing dependencies to look for known-vulnerable packages..."
 },
 {
 "type": "tool_use",
 "id": "toolu_01",
 "name": "bash",
 "input": { "command": "cat package.json && npm audit --json" }
 }
 ]
}

Suppose this assistant turn (thinking plus the tool call) totals 5,000 generated tokens. The countdown Claude saw during generation ended near remaining ≈ 95,000.

Turn 2. Your client executes the tool, then resends the full history with the tool result appended:

{
 "messages": [
 { "role": "user", "content": "Audit this repo for security issues and report findings." },
 {
 "role": "assistant",
 "content": [
 { "type": "thinking", "thinking": "I'll start by listing dependencies..." },
 {
 "type": "tool_use",
 "id": "toolu_01",
 "name": "bash",
 "input": { "command": "cat package.json && npm audit --json" }
 }
 ]
 },
 {
 "role": "user",
 "content": [
 {
 "type": "tool_result",
 "tool_use_id": "toolu_01",
 "content": "<2,800 tokens of npm audit output>"
 }
 ]
 }
 ]
}

The resent turn-1 user and assistant messages are not counted again, but the 2,800-token tool result is new content Claude sees this turn and counts against the budget. Claude spends another 4,000 tokens on thinking and a second tool call (grep -rn "eval(" src/). The countdown ends near remaining ≈ 88,200.

Turn 3. Full history resent again with the second tool result (1,200 tokens of grep output) appended. Claude writes a 6,000-token final findings report and stops with stop_reason: "end_turn". remaining ≈ 81,000.

Putting the three turns side by side makes the distinction between payload size and budget spend explicit:

TurnRequest payload (approx. input tokens you sent)Tokens counted against budget this turnBudget remaining after
1~205,000 (thinking + tool_use)~95,000
2~7,800 (turn 1 history + tool result)6,800 (2,800 tool result + 4,000 thinking and tool_use)~88,200
3~13,000 (full history + second tool result)7,200 (1,200 tool result + 6,000 text)~81,000
Total~20,820 sent across requests19,000 counted against budgetN/A

Your client sent the turn-1 user message three times and the turn-1 assistant message twice, but each was counted once. The budget spent 19,000 of 100,000 tokens, even though the cumulative payload your client transmitted was larger and the prompt-cached input on turns 2 and 3 was larger still.

Carrying a budget across compaction with remaining

If your agentic loop compacts or rewrites context between requests (for example, by summarizing earlier turns), the server has no memory of how much budget was spent before compaction. Pass remaining on the next request so the countdown continues from where you left off rather than resetting to total:

output_config = {
 "effort": "high",
 "task_budget": {
 "type": "tokens",
 "total": 128000,
 "remaining": 128000 - tokens_spent_so_far,
 },
}

For loops that resend the full uncompacted history on every turn, omit remaining and let the server track the countdown.

Task budgets are advisory, not enforced

Task budgets are a soft hint, not a hard cap. Claude may occasionally exceed the budget if it is in the middle of an action that would be more disruptive to interrupt than to finish. The enforced limit on total output tokens is still max_tokens, which truncates the response with stop_reason: "max_tokens" when reached.

For a hard cap on cost or latency, combine task budgets with a reasonable max_tokens value:

Because task_budget spans the full agentic loop (potentially many requests) while max_tokens caps each individual request, the two values are independent; one is not required to be at or below the other.

A budget that is too small for the task can cause refusal-like behavior. When Claude sees a budget that is clearly insufficient for the work being asked (for example, a 20,000-token budget for a multi-hour agentic coding task), it may decline to attempt the task at all, scope it down aggressively, or stop early with a partial result rather than start work it cannot finish. If you observe unexpected refusals or premature stops after setting a budget, raise the budget before debugging other parameters. Size budgets against your actual task-length distribution rather than a fixed default; see Choosing a budget.

Choosing a budget

The right budget depends on how much work your agentic loop currently does. Rather than guessing, measure your existing token usage first and then tune from there.

Measure your current usage

Run a representative sample of tasks without task_budget set and record the total tokens Claude spends per task. For an agentic loop, sum usage.output_tokens plus thinking and tool-result tokens across every request in the loop:

def run_task_and_count_tokens(messages: list) -> int:
 """Runs an agentic loop to completion and returns total tokens spent."""
 total_spend = 0
 while True:
 with client.beta.messages.stream(
 model="claude-opus-4-8",
 max_tokens=128000,
 messages=messages,
 tools=tools,
 betas=["task-budgets-2026-03-13"],
 ) as stream:
 response = stream.get_final_message()
 # Count what Claude generated this turn (output covers text + thinking + tool calls).
 # Tool-result tokens also count against the budget; add the token count of the
 # tool_result blocks you append below if you want client-side tracking to match
 # the server-side countdown.
 total_spend += response.usage.output_tokens
 if response.stop_reason == "end_turn":
 return total_spend
 # Append the assistant turn and your tool results, then continue the loop.
 messages += [
 {"role": "assistant", "content": response.content},
 {"role": "user", "content": run_tools(response.content)},
 ]

Run this across a representative set of tasks and record the distribution. Start with the p99 of your per-task token spend to understand how providing the model with a task budget may modify the model's behavior, then test up or down as needed.

The minimum accepted task_budget.total is 20,000 tokens; values below the minimum return a 400 error.

Interaction with other parameters

Feature support

ModelSupport
Claude Fable 5Beta (set task-budgets-2026-03-13 header)
Claude Mythos 5Beta (set task-budgets-2026-03-13 header)
Claude Opus 4.8Beta (set task-budgets-2026-03-13 header)
Claude Opus 4.7Beta (set task-budgets-2026-03-13 header)
Claude Opus 4.6Not supported
Claude Sonnet 4.6Not supported
Claude Haiku 4.5Not supported

Task budgets are not supported on Claude Code or Cowork surfaces. Use task budgets directly via the Messages API on a supported model.