dev-docs/architecture/setup-angular.md
This guide shows how to set up CopilotKit in an Angular app — from minimal to fully configured.
graph LR
subgraph Your Angular App
DI["<b>provideCopilotKit()</b>
<i>DI token</i>"]
Service["<b>CopilotKit Service</b>
<i>Injectable</i>"]
Store["<b>AgentStore</b>
<i>Signal-based state</i>"]
Comp["Your Components"]
end
subgraph Under the Hood
Core["CopilotKitCore
<i>Orchestrator</i>"]
Proxy["ProxiedAgent
<i>HTTP client</i>"]
end
subgraph Your Server
Runtime["CopilotRuntime
<i>Express / Hono</i>"]
end
DI -->|configures| Service
Service -->|wraps| Core
Comp -->|injects| Service
Service -->|creates| Store
Store -->|wraps| Proxy
Proxy -->|HTTP POST + SSE| Runtime
npm install @copilotkit/angular
// app.config.ts
import { ApplicationConfig } from "@angular/core";
import { provideCopilotKit } from "@copilotkit/angular";
export const appConfig: ApplicationConfig = {
providers: [
provideCopilotKit({
runtimeUrl: "/api/copilotkit",
}),
],
};
// chat.component.ts
import { Component, inject } from "@angular/core";
import { CopilotKit } from "@copilotkit/angular";
@Component({
selector: "app-chat",
template: `
<div>
<div *ngFor="let msg of agentStore.messages()">
<b>{{ msg.role }}:</b> {{ msg.content }}
</div>
<input #input (keydown.enter)="send(input.value); input.value = ''" />
</div>
`,
})
export class ChatComponent {
private copilotKit = inject(CopilotKit);
agentStore = this.copilotKit.getAgentStore(); // default agent
async send(message: string) {
this.agentStore.addMessage({
id: crypto.randomUUID(),
role: "user",
content: message,
});
await this.copilotKit.runAgent({ agent: this.agentStore.agent });
}
}
That's it — the DI token creates a CopilotKit service backed by CopilotKitCore, and AgentStore gives you signal-based reactive state.
sequenceDiagram
participant Config as app.config.ts
participant Service as CopilotKit Service
participant Core as CopilotKitCore
participant Runtime as Your Server
Config->>Service: provideCopilotKit({ runtimeUrl })
Service->>Core: new CopilotKitCore(config)
Core->>Runtime: GET /info
Runtime-->>Core: Available agents
Note over Service: Ready — inject anywhere
AgentStore uses Angular signals, so your templates react to changes automatically:
@Component({
template: `
@if (agentStore.isRunning()) {
<p>Agent is thinking...</p>
}
@for (msg of agentStore.messages(); track msg.id) {
<div [class]="msg.role">{{ msg.content }}</div>
}
<pre>{{ agentStore.state() | json }}</pre>
`,
})
export class ChatComponent {
private copilotKit = inject(CopilotKit);
agentStore = this.copilotKit.getAgentStore("my-agent");
}
| Signal | Type | What it tracks |
|---|---|---|
messages() | Message[] | All messages in the conversation |
isRunning() | boolean | Whether the agent is currently running |
state() | any | Agent state (arbitrary JSON) |
graph TB
subgraph AgentStore
Agent["AbstractAgent
<i>Subscribed to events</i>"]
MS["messages()
<i>Signal<Message[]></i>"]
IR["isRunning()
<i>Signal<boolean></i>"]
ST["state()
<i>Signal<any></i>"]
end
Agent -->|onMessagesChanged| MS
Agent -->|onRunStarted/Finished| IR
Agent -->|onStateChanged| ST
subgraph Template
T["Your template auto-updates"]
end
MS --> T
IR --> T
ST --> T
// In your component or service
import { CopilotKit } from "@copilotkit/angular";
import { z } from "zod";
@Component({
/* ... */
})
export class ProductComponent implements OnInit, OnDestroy {
private copilotKit = inject(CopilotKit);
ngOnInit() {
// Register a tool the agent can call
this.copilotKit.addTool({
name: "addToCart",
description: "Add a product to cart",
parameters: z.object({
productId: z.string(),
quantity: z.number().default(1),
}),
handler: async ({ productId, quantity }) => {
this.cartService.add(productId, quantity);
return `Added ${quantity} item(s)`;
},
});
}
ngOnDestroy() {
// Clean up when component is destroyed
this.copilotKit.removeTool("addToCart");
}
}
@Component({
/* ... */
})
export class DashboardComponent implements OnInit, OnDestroy {
private copilotKit = inject(CopilotKit);
private contextId?: string;
ngOnInit() {
this.contextId = this.copilotKit.addContext({
description: "Current dashboard metrics",
value: JSON.stringify({
revenue: this.metricsService.revenue(),
activeUsers: this.metricsService.activeUsers(),
}),
});
}
ngOnDestroy() {
if (this.contextId) {
this.copilotKit.removeContext(this.contextId);
}
}
}
Angular uses the AngularToolCall type for rendering tool calls:
import { AngularToolCall } from "@copilotkit/angular";
// Configure in provideCopilotKit
provideCopilotKit({
runtimeUrl: "/api/copilotkit",
renderToolCalls: [
{
name: "searchProducts",
// The Angular component receives the AngularToolCall
},
],
});
graph LR
IP["in-progress
<i>Args still streaming</i>"]
EX["executing
<i>Handler is running</i>"]
CO["complete
<i>Result ready</i>"]
IP --> EX --> CO
| Field | Type | Description |
|---|---|---|
status | "in-progress" | "executing" | "complete" | Current lifecycle stage |
name | string | Tool name |
args | Partial<T> or T | Tool arguments (partial while streaming) |
result | string | undefined | Result (only when complete) |
// app.config.ts
import { provideCopilotKit } from "@copilotkit/angular";
provideCopilotKit({
// Required
runtimeUrl: "/api/copilotkit",
// Authentication
headers: { Authorization: "Bearer token" },
// Custom properties forwarded to agents
properties: { userId: "123", plan: "pro" },
// Local agents for development
agents: { test: myTestAgent },
// Tools (can also add via service)
tools: [
{
name: "myTool",
parameters: z.object({ input: z.string() }),
handler: async ({ input }) => `Processed: ${input}`,
},
],
// Tool call rendering
renderToolCalls: [
/* ... */
],
// Frontend tools
frontendTools: [
/* ... */
],
// Human-in-the-loop
humanInTheLoop: [
/* ... */
],
});
graph TB
subgraph "provideCopilotKit() Config"
direction TB
subgraph Required
URL["runtimeUrl"]
end
subgraph "Optional: Auth"
H["headers"]
end
subgraph "Optional: Tools & Rendering"
T["tools"]
FT["frontendTools"]
RTC["renderToolCalls"]
HIL["humanInTheLoop"]
end
subgraph "Optional: Other"
P["properties"]
AG["agents"]
end
end
// app.config.ts
import { ApplicationConfig } from "@angular/core";
import { provideCopilotKit } from "@copilotkit/angular";
export const appConfig: ApplicationConfig = {
providers: [
provideCopilotKit({
runtimeUrl: "/api/copilotkit",
headers: { Authorization: `Bearer ${getToken()}` },
}),
],
};
// dashboard.component.ts
import { Component, inject, OnInit, OnDestroy } from "@angular/core";
import { CopilotKit } from "@copilotkit/angular";
import { z } from "zod";
@Component({
selector: "app-dashboard",
template: `
<div class="dashboard">
<app-metrics />
<div class="chat">
@if (agentStore.isRunning()) {
<div class="typing">Agent is thinking...</div>
}
@for (msg of agentStore.messages(); track msg.id) {
<div [class]="'message ' + msg.role">
{{ msg.content }}
</div>
}
<input
#input
placeholder="Ask about your metrics..."
(keydown.enter)="send(input.value); input.value = ''"
/>
</div>
</div>
`,
})
export class DashboardComponent implements OnInit, OnDestroy {
private copilotKit = inject(CopilotKit);
private metricsService = inject(MetricsService);
agentStore = this.copilotKit.getAgentStore();
private contextId?: string;
ngOnInit() {
// Provide context
this.contextId = this.copilotKit.addContext({
description: "Dashboard metrics",
value: JSON.stringify({
revenue: this.metricsService.revenue(),
users: this.metricsService.activeUsers(),
}),
});
// Register tool
this.copilotKit.addTool({
name: "filterMetrics",
description: "Filter dashboard metrics by date range",
parameters: z.object({
startDate: z.string(),
endDate: z.string(),
}),
handler: async ({ startDate, endDate }) => {
this.metricsService.setDateRange(startDate, endDate);
return `Filtered to ${startDate} - ${endDate}`;
},
});
}
ngOnDestroy() {
if (this.contextId) this.copilotKit.removeContext(this.contextId);
this.copilotKit.removeTool("filterMetrics");
}
async send(message: string) {
this.agentStore.addMessage({
id: crypto.randomUUID(),
role: "user",
content: message,
});
await this.copilotKit.runAgent({ agent: this.agentStore.agent });
}
}
| Aspect | React | Angular |
|---|---|---|
| Configuration | <CopilotKitProvider> JSX | provideCopilotKit() DI token |
| Service access | useCopilotKit() hook | inject(CopilotKit) |
| Agent state | useAgent() hook returns reactive values | AgentStore with Angular signals |
| Tool registration | useFrontendTool() hook (auto-cleanup) | addTool() / removeTool() (manual cleanup) |
| Context | useAgentContext() hook (auto-cleanup) | addContext() / removeContext() (manual cleanup) |
| Reactivity | React re-renders on state change | Angular signals trigger change detection |