packages/runtime/skills/runtime/references/server-side-tools.md
Server-side tools run in the runtime process. They are the right choice when the tool needs to touch server-only state: DB connections, API keys, filesystem, signed URLs.
defineTool returns a ToolDefinition. Pass an array of them to the Simple-Mode
BuiltInAgent.config.tools, or into the tools: option of chat() / streamText() inside
a Factory Mode factory.
import {
CopilotRuntime,
createCopilotRuntimeHandler,
BuiltInAgent,
defineTool,
} from "@copilotkit/runtime/v2";
import { z } from "zod";
const getInventory = defineTool({
name: "getInventory",
description: "Look up stock for a product SKU.",
parameters: z.object({ sku: z.string() }),
execute: async ({ sku }) => {
const row = await db.product.findUnique({ where: { sku } });
return { sku, inStock: row?.inStock ?? 0 };
},
});
const runtime = new CopilotRuntime({
agents: {
default: new BuiltInAgent({
model: "openai/gpt-4o",
maxSteps: 5,
tools: [getInventory],
}),
},
});
const handler = createCopilotRuntimeHandler({
runtime,
basePath: "/api/copilotkit",
});
export default { fetch: handler };
declare const db: {
product: { findUnique: (q: any) => Promise<{ inStock: number } | null> };
};
import { defineTool } from "@copilotkit/runtime/v2";
import { z } from "zod";
const searchDocs = defineTool({
name: "searchDocs",
description: "Search the internal docs index.",
parameters: z.object({
query: z.string().min(1),
limit: z.number().int().min(1).max(20).default(5),
}),
execute: async ({ query, limit }) => {
const results = await searchIndex(query, limit);
return { results };
},
});
declare const searchIndex: (q: string, n: number) => Promise<unknown[]>;
import { defineTool } from "@copilotkit/runtime/v2";
import * as v from "valibot";
const translate = defineTool({
name: "translate",
description: "Translate text between languages.",
parameters: v.object({
text: v.pipe(v.string(), v.minLength(1)),
target: v.picklist(["en", "es", "fr", "de"]),
}),
execute: async ({ text, target }) => ({ translated: `[${target}] ${text}` }),
});
import { defineTool } from "@copilotkit/runtime/v2";
import { z } from "zod";
const runQuery = defineTool({
name: "runQuery",
description: "Run an analytics query.",
parameters: z.object({ sql: z.string() }),
execute: async ({ sql }) => {
try {
return { rows: await warehouse.query(sql) };
} catch (e) {
return { error: String(e), retryable: true };
}
},
});
declare const warehouse: { query: (sql: string) => Promise<unknown[]> };
Server tools for I/O, client tools for UI. Both can coexist.
// server
import { defineTool } from "@copilotkit/runtime/v2";
import { z } from "zod";
export const fetchOrder = defineTool({
name: "fetchOrder",
description: "Fetch order details from the orders service.",
parameters: z.object({ orderId: z.string() }),
execute: async ({ orderId }) => fetchOrderFromService(orderId),
});
declare const fetchOrderFromService: (id: string) => Promise<unknown>;
// client — a render-only tool lets the LLM display a modal
import { useComponent } from "@copilotkit/react-core/v2";
import { z } from "zod";
useComponent({
name: "showOrderDetails",
parameters: z.object({ orderId: z.string(), status: z.string() }),
// Schema fields arrive DIRECTLY as props (InferRenderProps<TSchema>) —
// no { args } wrapper. See packages/react-core/src/v2/hooks/use-component.tsx.
render: ({ orderId, status }) => (
<div className="modal">
Order {orderId} — {status}
</div>
),
});
Simple-Mode config.tools is ignored in Factory Mode.
import {
BuiltInAgent,
convertToolDefinitionsToVercelAITools,
convertMessagesToVercelAISDKMessages,
defineTool,
} from "@copilotkit/runtime/v2";
import { streamText } from "ai";
import { openai } from "@ai-sdk/openai";
import { z } from "zod";
const searchDocs = defineTool({
name: "searchDocs",
description: "Search the internal docs index.",
parameters: z.object({ query: z.string() }),
execute: async ({ query }) => ({ results: [] }),
});
new BuiltInAgent({
type: "aisdk",
factory: ({ input, abortSignal }) => {
const serverTools = convertToolDefinitionsToVercelAITools([searchDocs]);
return streamText({
model: openai("gpt-4o"),
messages: convertMessagesToVercelAISDKMessages(input.messages),
tools: serverTools,
abortSignal,
});
},
});
Wrong:
defineTool({
name: "showModal",
description: "Show a confirmation modal to the user.",
parameters: z.object({ title: z.string() }),
execute: async () => "rendered",
});
Correct:
// Keep UI on the client — frontend tool with a renderer
import { useFrontendTool } from "@copilotkit/react-core/v2";
import { z } from "zod";
useFrontendTool({
name: "showModal",
parameters: z.object({ title: z.string() }),
handler: async (args) => ({ confirmed: true }),
});
Server tools execute on the server and stream only results back. The browser never sees a
TOOL_CALL_START for a server tool, so there is nothing to mount a renderer against.
Source: dev-docs/architecture/plugin-points.md:36-77;
docs/content/docs/integrations/built-in-agent/server-tools.mdx:9-14.
Wrong:
defineTool({
name: "AGUISendStateSnapshot",
description: "My own snapshot tool.",
parameters: z.object({ snapshot: z.any() }),
execute: async () => ({ success: true }),
});
Correct:
defineTool({
name: "mySnapshotExport",
description: "Export a user-facing state snapshot.",
parameters: z.object({ snapshot: z.any() }),
execute: async () => ({ success: true }),
});
AGUISendStateSnapshot and AGUISendStateDelta are auto-injected by BuiltInAgent in
Simple Mode — redefining them silently overwrites the built-ins.
Source: packages/runtime/src/agent/index.ts:1139-1177.
Wrong:
defineTool({
name: "runQuery",
description: "Run a database query.",
parameters: z.object({ sql: z.string() }),
execute: async () => {
throw new Error("db down");
},
});
Correct:
defineTool({
name: "runQuery",
description: "Run a database query.",
parameters: z.object({ sql: z.string() }),
execute: async ({ sql }) => {
try {
return await db.query(sql);
} catch (e) {
return { error: String(e), retryable: true };
}
},
});
Thrown errors kill the run; unserializable results (class instances, circular refs) become
the string "[Unserializable tool result from X]". Return a plain-object error shape
instead and let the LLM retry.
Source: packages/runtime/src/agent/index.ts:1469-1474.
Wrong:
defineTool({
name: "x",
description: "...",
parameters: {
type: "object",
properties: { q: { type: "string" } },
required: ["q"],
} as any,
execute: async ({ q }) => q,
});
Correct:
import { z } from "zod";
defineTool({
name: "x",
description: "...",
parameters: z.object({ q: z.string() }),
execute: async ({ q }) => q,
});
parameters must be a Standard Schema V1 validator (Zod, Valibot, ArkType, ...). Plain
JSON Schema throws in schemaToJsonSchema(). Also, Standard Schema V1 preserves static
types — execute's arg type is inferred.
Source: packages/runtime/src/agent/index.ts:633-659.
Wrong:
new BuiltInAgent({
type: "tanstack",
factory: myFactory,
tools: [searchDocs], // ignored in Factory Mode
} as any);
Correct:
// Factory Mode — AI SDK factory: convert defineTool → Vercel AI SDK tools
import {
BuiltInAgent,
convertToolDefinitionsToVercelAITools,
convertMessagesToVercelAISDKMessages,
} from "@copilotkit/runtime/v2";
import { streamText } from "ai";
import { openai } from "@ai-sdk/openai";
new BuiltInAgent({
type: "aisdk",
factory: ({ input, abortSignal }) => {
const tools = convertToolDefinitionsToVercelAITools([searchDocs]);
return streamText({
model: openai("gpt-4o"),
messages: convertMessagesToVercelAISDKMessages(input.messages),
tools,
abortSignal,
});
},
});
// Factory Mode — TanStack AI factory: defineTool output is NOT a TanStack tool.
// There is no built-in converter in @copilotkit/runtime for TanStack. Either
// redefine the tool with TanStack's `toolDefinition()` API from `@tanstack/ai`,
// or write a small adapter that translates your `defineTool` output into
// TanStack's tool shape before passing it into `chat({ tools })`.
Factory Mode ignores config.tools. Wire server tools through the factory's LLM call —
AI SDK has convertToolDefinitionsToVercelAITools([...]) out of the box; TanStack AI has
its own toolDefinition() API you need to build the tools with directly.
Source: packages/runtime/src/agent/index.ts:1581-1671.
Wrong:
// frontend
useFrontendTool({
name: "getWeather",
parameters: z.object({ city: z.string() }),
handler,
});
// server
defineTool({
name: "getWeather",
parameters: z.object({ city: z.string() }),
execute,
});
// Server silently wins on the merge — handler never fires
Correct:
// Pick one side and give tools distinct names if both sides need their own
useFrontendTool({ name: "getWeatherClientSide" /* ... */ });
defineTool({ name: "getWeatherServer" /* ... */ });
On collisions, config.tools (server) overwrites frontend-registered tools. The LLM sees
only one getWeather — the server version.
Source: packages/runtime/src/agent/index.ts (tool merge).
copilotkit/built-in-agent — config.tools only applies in Simple Modecopilotkit/client-side-tools (react-core) — browser-side tools, paired decisioncopilotkit/rendering-tool-calls (react-core) — rendering tool invocations in chat