skills/runtime/references/built-in-agent.md
BuiltInAgent has two modes:
{ model, apiKey, prompt, tools, mcpServers, maxSteps, ... }.
Convenient for quickstarts. Simple Mode auto-injects the AGUISendStateSnapshot /
AGUISendStateDelta state tools; Factory Mode does not.Use Factory Mode with TanStack AI for new code.
Factory Mode with TanStack AI (preferred default):
import {
CopilotRuntime,
createCopilotRuntimeHandler,
BuiltInAgent,
convertInputToTanStackAI,
} from "@copilotkit/runtime/v2";
import { chat } from "@tanstack/ai";
import { openaiText } from "@tanstack/ai-openai";
const agent = new BuiltInAgent({
type: "tanstack",
factory: ({ input, abortController }) => {
const { messages, systemPrompts } = convertInputToTanStackAI(input);
systemPrompts.unshift("You are a helpful assistant.");
return chat({
adapter: openaiText("gpt-4o"),
messages,
systemPrompts,
abortController,
});
},
});
const runtime = new CopilotRuntime({ agents: { default: agent } });
const handler = createCopilotRuntimeHandler({
runtime,
basePath: "/api/copilotkit",
});
export default { fetch: handler };
Simple Mode (quickstart only):
import {
BuiltInAgent,
CopilotRuntime,
createCopilotRuntimeHandler,
} from "@copilotkit/runtime/v2";
const agent = new BuiltInAgent({
model: "openai/gpt-4o",
apiKey: process.env.OPENAI_API_KEY,
prompt: "You are a helpful assistant.",
maxSteps: 5, // enable the tool-call loop
});
const runtime = new CopilotRuntime({ agents: { default: agent } });
const handler = createCopilotRuntimeHandler({
runtime,
basePath: "/api/copilotkit",
});
export default { fetch: handler };
import {
BuiltInAgent,
convertMessagesToVercelAISDKMessages,
convertToolsToVercelAITools,
} from "@copilotkit/runtime/v2";
import { streamText, stepCountIs } from "ai";
import { anthropic } from "@ai-sdk/anthropic";
const agent = new BuiltInAgent({
type: "aisdk",
factory: ({ input, abortSignal }) => {
const messages = convertMessagesToVercelAISDKMessages(input.messages);
const tools = convertToolsToVercelAITools(input.tools);
return streamText({
model: anthropic("claude-sonnet-4-5-20250929"),
messages,
tools,
abortSignal,
stopWhen: stepCountIs(5),
});
},
});
import {
CopilotRuntime,
BuiltInAgent,
convertInputToTanStackAI,
} from "@copilotkit/runtime/v2";
import { chat } from "@tanstack/ai";
import { openaiText } from "@tanstack/ai-openai";
const runtime = new CopilotRuntime({
agents: ({ request }) => {
const tenantId = request.headers.get("x-tenant-id") ?? "default";
return {
default: new BuiltInAgent({
type: "tanstack",
factory: ({ input, abortController }) => {
const { messages, systemPrompts } = convertInputToTanStackAI(input);
systemPrompts.unshift(`You are the ${tenantId} assistant.`);
return chat({
adapter: openaiText("gpt-4o"),
messages,
systemPrompts,
abortController,
});
},
}),
};
},
});
new BuiltInAgent({
model: "openai/gpt-4o",
maxSteps: 5,
mcpServers: [
{ type: "http", url: "https://mcp.example.com/mcp" },
{
type: "sse",
url: "https://mcp.example.com/sse",
headers: { Authorization: `Bearer ${process.env.MCP_TOKEN}` },
},
],
});
"provider/model" or "provider:model". Supported providers: openai, anthropic,
google (aliases gemini, google-gemini), vertex. The bare model id ("gpt-4o") is
rejected.
new BuiltInAgent({ model: "openai/gpt-4o" });
new BuiltInAgent({ model: "anthropic/claude-sonnet-4.5" });
new BuiltInAgent({ model: "google/gemini-2.5-pro" });
Wrong:
const agent = new BuiltInAgent({
model: "openai/gpt-4o",
prompt: "You are a helpful assistant.",
});
Correct:
import { BuiltInAgent, convertInputToTanStackAI } from "@copilotkit/runtime/v2";
import { chat } from "@tanstack/ai";
import { openaiText } from "@tanstack/ai-openai";
const agent = new BuiltInAgent({
type: "tanstack",
factory: ({ input, abortController }) => {
const { messages, systemPrompts } = convertInputToTanStackAI(input);
systemPrompts.unshift("You are a helpful assistant.");
return chat({
adapter: openaiText("gpt-4o"),
messages,
systemPrompts,
abortController,
});
},
});
Factory Mode with TanStack AI is the canonical in-tree default (see
examples/v2/react-router/app/routes/api.copilotkit.$.tsx) and is AG-UI-native. Simple
Mode is fine for quickstarts but reaches its ceiling on anything non-standard.
Source: examples/v2/react-router/app/routes/api.copilotkit.$.tsx; maintainer Phase 4c.
Wrong:
new BuiltInAgent({
model: "openai/gpt-4o",
tools: [searchTool],
// maxSteps defaults to undefined → AI SDK stops after one generation; tool results
// are never fed back. Set maxSteps: N to enable the tool-call loop.
});
Correct:
new BuiltInAgent({
model: "openai/gpt-4o",
tools: [searchTool],
maxSteps: 5,
});
maxSteps defaults to undefined, so stopWhen is undefined and the AI SDK's own
default applies — streamText stops after a single generation, the tool call happens,
but results are never fed back for a second turn. Set maxSteps: N to install
stepCountIs(N) and enable the tool-call loop up to N steps.
Source: packages/runtime/src/agent/index.ts:988-990.
Wrong:
new BuiltInAgent({ model: "gpt-4o" });
Correct:
new BuiltInAgent({ model: "openai/gpt-4o" });
// Also valid: "openai:gpt-4o"
resolveModel throws Invalid model string "gpt-4o". Use "openai/gpt-5", "anthropic/claude-sonnet-4.5", or "google/gemini-2.5-pro". when the provider separator
is missing.
Source: packages/runtime/src/agent/index.ts:186-204.
Wrong:
// One shared instance across tenants
const agent = new BuiltInAgent({ model: "openai/gpt-4o" });
new CopilotRuntime({ agents: { default: agent } });
Correct:
// Use the agents-as-factory form for per-request instances
new CopilotRuntime({
agents: ({ request }) => ({
default: new BuiltInAgent({ model: "openai/gpt-4o" }),
}),
});
A single BuiltInAgent instance guards against concurrent run() with
"Agent is already running. Call abortRun() first or create a new instance." Multi-tenant
servers that share one instance see errors on the second concurrent user.
Source: packages/runtime/src/agent/index.ts:895-898.
Wrong:
new BuiltInAgent({
type: "tanstack",
factory: ({ input, abortController }) => {
const { messages, systemPrompts } = convertInputToTanStackAI(input);
return chat({
adapter: openaiText("gpt-4o"),
messages,
systemPrompts,
abortController,
});
},
});
// Frontend uses useAgent + shared state — but no state-tool calls come back
Correct (AI SDK factory — defineTool output converts via
convertToolDefinitionsToVercelAITools):
import {
BuiltInAgent,
convertMessagesToVercelAISDKMessages,
convertToolDefinitionsToVercelAITools,
defineTool,
} from "@copilotkit/runtime/v2";
import { streamText } from "ai";
import { openai } from "@ai-sdk/openai";
import { z } from "zod";
const sendStateSnapshot = defineTool({
name: "AGUISendStateSnapshot",
description: "Replace the entire application state with a new snapshot",
parameters: z.object({
snapshot: z.any().describe("The complete new state object"),
}),
execute: async ({ snapshot }) => ({ success: true, snapshot }),
});
const sendStateDelta = defineTool({
name: "AGUISendStateDelta",
description:
"Apply incremental updates to application state using JSON Patch operations",
// MUST mirror the Simple-Mode auto-injected schema (src/agent/index.ts:1140-1176)
// or the frontend's state handler won't recognize the payload.
parameters: z.object({
delta: z
.array(
z.object({
op: z.enum(["add", "replace", "remove"]),
path: z.string(), // JSON Pointer, e.g. "/foo/bar"
value: z.any().optional(), // required for add/replace, ignored for remove
}),
)
.describe("Array of JSON Patch operations"),
}),
execute: async ({ delta }) => ({ success: true, delta }),
});
// If you don't want to hand-wire this, use Simple Mode — it auto-injects both
// AGUISendStateSnapshot and AGUISendStateDelta with the correct JSON Patch schema.
// Source: packages/runtime/src/agent/index.ts:1140-1176
new BuiltInAgent({
type: "aisdk",
factory: ({ input, abortSignal }) =>
streamText({
model: openai("gpt-4o"),
messages: convertMessagesToVercelAISDKMessages(input.messages),
tools: convertToolDefinitionsToVercelAITools([
sendStateSnapshot,
sendStateDelta,
]),
abortSignal,
}),
});
Only Simple Mode auto-injects the AG-UI state tools. In Factory Mode you must register
them by hand or shared-state updates never reach the LLM. defineTool produces a Standard
Schema V1 + execute shape — use convertToolDefinitionsToVercelAITools([...]) to adapt
it to the AI SDK's streamText({ tools }). TanStack AI factories cannot consume
defineTool output directly; either redefine the tools with toolDefinition() from
@tanstack/ai, or switch to the AI SDK factory above.
Source: docs/snippets/shared/backend/custom-agent.mdx:495-588.
Wrong:
new BuiltInAgent({
type: "tanstack",
factory: myFactory,
tools: [t1, t2], // ignored in Factory Mode
});
Correct:
new BuiltInAgent({
type: "tanstack",
factory: ({ input, abortController }) => {
const { messages, systemPrompts } = convertInputToTanStackAI(input);
return chat({
adapter: openaiText("gpt-4o"),
messages,
systemPrompts,
tools: [t1, t2],
abortController,
});
},
});
Factory Mode ignores config.tools, config.mcpServers, config.prompt entirely — the
factory owns the call. Wire tools inside chat({ tools }) for TanStack AI, or via
convertToolsToVercelAITools(input.tools) / convertToolDefinitionsToVercelAITools([...])
for AI SDK.
Source: packages/runtime/src/agent/index.ts:1581-1671.
Wrong:
new BuiltInAgent({
type: "tanstack",
factory: ({ input, abortController }) => {
const { messages, systemPrompts } = convertInputToTanStackAI(input);
return chat({
adapter: anthropicText("claude-sonnet-4-5-20250929"),
messages,
systemPrompts,
modelOptions: { thinking: { type: "enabled", budgetTokens: 10000 } },
abortController,
});
},
});
// expecting REASONING_START / REASONING_MESSAGE_CONTENT / REASONING_END — nothing arrives
Correct:
import {
BuiltInAgent,
convertMessagesToVercelAISDKMessages,
} from "@copilotkit/runtime/v2";
import { streamText } from "ai";
import { anthropic } from "@ai-sdk/anthropic";
new BuiltInAgent({
type: "aisdk",
factory: ({ input, abortSignal }) =>
streamText({
model: anthropic("claude-sonnet-4-5-20250929"),
messages: convertMessagesToVercelAISDKMessages(input.messages),
providerOptions: {
anthropic: { thinking: { type: "enabled", budgetTokens: 10000 } },
},
abortSignal,
}),
});
The TanStack AI converter does NOT surface REASONING_START /
REASONING_MESSAGE_CONTENT / REASONING_END events — even with a thinking-capable model.
Use AI SDK when the frontend needs a reasoning UI.
Source: docs/snippets/shared/backend/custom-agent.mdx:315-317 (warn callout).
Wrong:
// Client sends { role: "system", content: "You are..." } and expects it prefixed
new BuiltInAgent({ model: "openai/gpt-4o" });
Correct:
// Either set the server-side prompt
new BuiltInAgent({ model: "openai/gpt-4o", prompt: "You are..." });
// or opt in explicitly
new BuiltInAgent({ model: "openai/gpt-4o", forwardSystemMessages: true });
forwardSystemMessages and forwardDeveloperMessages default to false. System/developer
messages from the AG-UI input are dropped unless opted in.
Source: packages/runtime/src/agent/index.ts:440-456,809-815.
Wrong:
factory: (ctx) => {
ctx.abortController.abort(); // JSDoc says don't
return streamText({
/* ... */
});
};
Correct:
factory: (ctx) => streamText({ /* ... */, abortSignal: ctx.abortSignal });
// Externally, from outside the factory:
agent.abortRun();
The JSDoc on AgentFactoryContext.abortController explicitly warns against calling
.abort() on it inside the factory — use agent.abortRun() or pass abortSignal to the
downstream fetch/LLM call.
Source: packages/runtime/src/agent/index.ts:670-672.
copilotkit/server-side-tools — defineTool powers config.tools in Simple Modecopilotkit/setup-endpoint — mount the runtime that hosts this agentcopilotkit/wiring-external-agents — alternative when you want an external framework