dev-docs/architecture/setup-vanilla.md
This guide shows how to use CopilotKit without React or Angular — using the core API directly. This works with any framework (Vue, Svelte, vanilla JS, Node.js, etc).
graph LR
subgraph Your Code
App["Your Application"]
Sub["Event Subscribers"]
end
subgraph CopilotKit Core
Core["<b>CopilotKitCore</b>
<i>Orchestrator</i>"]
AR["AgentRegistry"]
RH["RunHandler"]
CS["ContextStore"]
end
subgraph Transport
Proxy["ProxiedAgent
<i>HTTP + SSE</i>"]
end
subgraph Your Server
Runtime["CopilotRuntime"]
end
App -->|creates| Core
App -->|subscribes| Sub
Sub -->|listens to| Core
Core --> AR
Core --> RH
Core --> CS
Core -->|creates| Proxy
Proxy -->|HTTP POST + SSE| Runtime
npm install @copilotkit/core
import { CopilotKitCore } from "@copilotkit/core";
const copilotKit = new CopilotKitCore({
runtimeUrl: "http://localhost:3000/api/copilotkit",
});
// Wait for runtime connection
const subscription = copilotKit.subscribe({
onRuntimeConnectionStatusChanged: async ({ status }) => {
if (status === "connected") {
// Agents are now available
const agent = copilotKit.getAgent("default");
// Add a user message
agent.addMessage({
id: crypto.randomUUID(),
role: "user",
content: "Hello, what can you do?",
});
// Run the agent
await copilotKit.runAgent({ agent });
}
},
});
That's it — CopilotKitCore handles connecting to the runtime, fetching agents, and managing the event lifecycle.
sequenceDiagram
participant App as Your Code
participant Core as CopilotKitCore
participant Runtime as CopilotRuntime
App->>Core: new CopilotKitCore({ runtimeUrl })
Core->>Runtime: GET /info
Runtime-->>Core: Available agents
Core->>App: onRuntimeConnectionStatusChanged("connected")
App->>Core: getAgent("default")
App->>Core: runAgent({ agent })
Core->>Runtime: POST /agent/default/run
Runtime-->>Core: SSE events stream
Core->>App: Messages update via subscription
The core provides a rich subscription system — this is how you react to changes without a framework:
const subscription = copilotKit.subscribe({
// Connection lifecycle
onRuntimeConnectionStatusChanged: ({ status }) => {
// "disconnected" | "connecting" | "connected" | "error"
updateConnectionUI(status);
},
// Agent availability
onAgentsChanged: ({ agents }) => {
console.log("Available agents:", Object.keys(agents));
// agents is Record<string, AbstractAgent>
},
// Tool execution
onToolExecutionStart: ({ toolName, args, agentId }) => {
showToolSpinner(toolName);
},
onToolExecutionEnd: ({ toolName, result, error }) => {
hideToolSpinner(toolName);
if (error) showError(error);
},
// Context changes
onContextChanged: ({ context }) => {
console.log("Context updated:", context);
},
// Suggestions
onSuggestionsChanged: ({ agentId, suggestions }) => {
renderSuggestionChips(suggestions);
},
// Errors
onError: ({ error, code, context }) => {
// code: "AGENT_CONNECT_FAILED" | "AGENT_RUN_FAILED" |
// "TOOL_HANDLER_FAILED" | "TOOL_ARGUMENT_PARSE_FAILED" |
// "RUNTIME_INFO_FETCH_FAILED"
console.error(`[${code}]`, error.message, context);
},
});
// Clean up when done
subscription.unsubscribe();
graph TB
subgraph "CopilotKitCore Events"
direction TB
subgraph Connection
RCS["onRuntimeConnectionStatusChanged"]
end
subgraph Agents
AC["onAgentsChanged"]
end
subgraph "Tool Execution"
TES["onToolExecutionStart"]
TEE["onToolExecutionEnd"]
end
subgraph Context
CC["onContextChanged"]
end
subgraph Suggestions
SC["onSuggestionsChanged"]
SSL["onSuggestionsStartedLoading"]
SFL["onSuggestionsFinishedLoading"]
end
subgraph Errors
ERR["onError"]
end
end
Your["Your subscriber"] --> RCS
Your --> AC
Your --> TES
Your --> TEE
Your --> CC
Your --> SC
Your --> ERR
In addition to core events, you can subscribe directly to an agent's events:
const agent = copilotKit.getAgent("default");
const agentSub = agent.subscribe({
// Messages changed (streaming text, new messages, etc.)
onMessagesChanged: ({ messages }) => {
renderChatMessages(messages);
},
// Agent state changed
onStateChanged: ({ state }) => {
updateStateDisplay(state);
},
// Run lifecycle
onRunInitialized: () => {
showTypingIndicator();
},
onRunFinalized: () => {
hideTypingIndicator();
},
onRunFailed: ({ error }) => {
showError(error);
},
// Granular event tracking
onToolCallStartEvent: ({ event }) => {
console.log("Tool call:", event.name);
},
onToolCallEndEvent: ({ toolCallArgs }) => {
console.log("Tool args:", toolCallArgs);
},
onToolCallResultEvent: ({ event }) => {
console.log("Tool result:", event.result);
},
});
// Clean up
agentSub.unsubscribe();
import { z } from "zod";
// Add a tool
copilotKit.addTool({
name: "getWeather",
description: "Get current weather for a location",
parameters: z.object({
city: z.string().describe("City name"),
unit: z.enum(["celsius", "fahrenheit"]).default("celsius"),
}),
handler: async ({ city, unit }) => {
const data = await fetch(`/api/weather?city=${city}&unit=${unit}`);
return await data.text();
},
followUp: true, // Agent will continue after getting the result
});
// Remove a tool
copilotKit.removeTool("getWeather");
// This tool is only available to the "research" agent
copilotKit.addTool({
name: "searchPapers",
description: "Search academic papers",
agentId: "research", // Only this agent can call it
parameters: z.object({ query: z.string() }),
handler: async ({ query }) => {
return await searchPapers(query);
},
});
// Add context (returns an ID for later removal)
const contextId = copilotKit.addContext({
description: "Current user session",
value: JSON.stringify({
userId: "user_123",
role: "admin",
currentPage: "/dashboard",
}),
});
// Update context (remove + re-add)
copilotKit.removeContext(contextId);
const newContextId = copilotKit.addContext({
description: "Current user session",
value: JSON.stringify({
userId: "user_123",
role: "admin",
currentPage: "/settings",
}),
});
// Remove when no longer relevant
copilotKit.removeContext(newContextId);
const copilotKit = new CopilotKitCore({
// Required
runtimeUrl: "http://localhost:3000/api/copilotkit",
// Authentication
headers: { Authorization: "Bearer my-token" },
credentials: "include", // Forward cookies
// Runtime transport mode
runtimeTransport: "rest", // "rest" (default) or "single"
// Custom properties forwarded to agents
properties: {
userId: "user_123",
environment: "production",
},
// Initial tools
tools: [
{
name: "myTool",
parameters: z.object({ input: z.string() }),
handler: async ({ input }) => `Result: ${input}`,
},
],
// Local agents (dev only — normally fetched from runtime)
agents__unsafe_dev_only: {
test: myLocalAgent,
},
});
graph TB
subgraph "CopilotKitCore Config"
direction TB
subgraph Required
URL["runtimeUrl"]
end
subgraph "Optional: Auth"
H["headers"]
C["credentials"]
end
subgraph "Optional: Transport"
RT["runtimeTransport
<i>'rest' or 'single'</i>"]
end
subgraph "Optional: Data"
P["properties"]
T["tools"]
SC["suggestionsConfig"]
end
subgraph "Optional: Dev"
AG["agents__unsafe_dev_only"]
end
end
For the simplest possible setup, you can skip CopilotKit entirely and use AG-UI's HttpAgent directly:
import { HttpAgent } from "@ag-ui/client";
const agent = new HttpAgent({
agentId: "my-agent",
url: "http://localhost:3000/api/copilotkit/agent/my-agent/run",
headers: { Authorization: "Bearer token" },
});
// Subscribe to messages
agent.subscribe({
onMessagesChanged: ({ messages }) => {
console.log("Messages:", messages);
},
});
// Send a message and run
agent.addMessage({
id: crypto.randomUUID(),
role: "user",
content: "Hello!",
});
await agent.runAgent();
When to use this: Only if you want zero abstraction and just need to talk to a single agent. You lose tools, context, suggestions, and multi-agent orchestration.
graph LR
subgraph "HttpAgent (Minimal)"
HA["HttpAgent"]
end
subgraph "CopilotKitCore (Full)"
CKC["CopilotKitCore"]
Tools["Tool Registry"]
Context["Context Store"]
Suggestions["Suggestions"]
Multi["Multi-Agent"]
CKC --> Tools
CKC --> Context
CKC --> Suggestions
CKC --> Multi
end
HA -->|HTTP + SSE| Server["Your Server"]
CKC -->|HTTP + SSE| Server
import { CopilotKitCore } from "@copilotkit/core";
import { z } from "zod";
// DOM elements
const messagesDiv = document.getElementById("messages")!;
const input = document.getElementById("input") as HTMLInputElement;
const sendBtn = document.getElementById("send")!;
const statusSpan = document.getElementById("status")!;
// Initialize CopilotKit
const copilotKit = new CopilotKitCore({
runtimeUrl: "/api/copilotkit",
});
let currentAgent: any = null;
// Subscribe to core events
copilotKit.subscribe({
onRuntimeConnectionStatusChanged: ({ status }) => {
statusSpan.textContent = status;
if (status === "connected") {
currentAgent = copilotKit.getAgent("default");
setupAgentSubscription();
input.disabled = false;
}
},
onToolExecutionStart: ({ toolName }) => {
appendMessage("system", `Running tool: ${toolName}...`);
},
onError: ({ error, code }) => {
appendMessage("error", `[${code}] ${error.message}`);
},
});
// Subscribe to agent messages
function setupAgentSubscription() {
currentAgent.subscribe({
onMessagesChanged: ({ messages }) => {
messagesDiv.innerHTML = "";
messages.forEach((msg) => appendMessage(msg.role, msg.content));
},
});
}
// Register a tool
copilotKit.addTool({
name: "getCurrentTime",
description: "Get the current date and time",
parameters: z.object({}),
handler: async () => new Date().toISOString(),
});
// Send message
async function sendMessage() {
const text = input.value.trim();
if (!text || !currentAgent) return;
input.value = "";
currentAgent.addMessage({
id: crypto.randomUUID(),
role: "user",
content: text,
});
await copilotKit.runAgent({ agent: currentAgent });
}
sendBtn.addEventListener("click", sendMessage);
input.addEventListener("keydown", (e) => {
if (e.key === "Enter") sendMessage();
});
// Helper
function appendMessage(role: string, content: string) {
const div = document.createElement("div");
div.className = `message ${role}`;
div.textContent = `${role}: ${content}`;
messagesDiv.appendChild(div);
}
<!-- index.html -->
<div id="app">
<div>Status: <span id="status">connecting...</span></div>
<div id="messages"></div>
<input id="input" disabled placeholder="Connecting..." />
<button id="send">Send</button>
</div>
<script type="module" src="./main.ts"></script>