Back to Copilotkit

CopilotKit Server-Side Tools

skills/runtime/references/server-side-tools.md

1.57.410.3 KB
Original Source

CopilotKit Server-Side Tools

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.

Setup

typescript
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> };
};

Core Patterns

Zod parameters (most common)

typescript
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[]>;

Valibot parameters (Standard Schema V1)

typescript
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}` }),
});

Graceful error handling inside execute

typescript
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 tool + client tool side by side

Server tools for I/O, client tools for UI. Both can coexist.

typescript
// 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>;
tsx
// 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>
  ),
});

Factory Mode — pass tools into the factory

Simple-Mode config.tools is ignored in Factory Mode.

typescript
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,
    });
  },
});

Common Mistakes

HIGH Using defineTool for tools that should render UI

Wrong:

typescript
defineTool({
  name: "showModal",
  description: "Show a confirmation modal to the user.",
  parameters: z.object({ title: z.string() }),
  execute: async () => "rendered",
});

Correct:

tsx
// 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.

MEDIUM Redefining AG-UI reserved names

Wrong:

typescript
defineTool({
  name: "AGUISendStateSnapshot",
  description: "My own snapshot tool.",
  parameters: z.object({ snapshot: z.any() }),
  execute: async () => ({ success: true }),
});

Correct:

typescript
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.

MEDIUM Throwing from execute without a result

Wrong:

typescript
defineTool({
  name: "runQuery",
  description: "Run a database query.",
  parameters: z.object({ sql: z.string() }),
  execute: async () => {
    throw new Error("db down");
  },
});

Correct:

typescript
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.

MEDIUM Passing a JSON-schema object as parameters

Wrong:

typescript
defineTool({
  name: "x",
  description: "...",
  parameters: {
    type: "object",
    properties: { q: { type: "string" } },
    required: ["q"],
  } as any,
  execute: async ({ q }) => q,
});

Correct:

typescript
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.

HIGH Unavailable in Factory Mode via config.tools

Wrong:

typescript
new BuiltInAgent({
  type: "tanstack",
  factory: myFactory,
  tools: [searchDocs], // ignored in Factory Mode
} as any);

Correct:

typescript
// 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.

MEDIUM Shared name between client and server tool

Wrong:

tsx
// 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:

tsx
// 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).

See also

  • copilotkit/built-in-agentconfig.tools only applies in Simple Mode
  • copilotkit/client-side-tools (react-core) — browser-side tools, paired decision
  • copilotkit/rendering-tool-calls (react-core) — rendering tool invocations in chat