A runnable demo of @copilotkit/bot-teams: a
Microsoft Teams bot backed by a CopilotKit BuiltInAgent that shows
streamed-by-edit replies, agent-rendered Adaptive Cards, and a
human-in-the-loop approval gate, testable locally in the Microsoft 365
Agents Playground with no Microsoft credentials. It needs an
OPENAI_API_KEY.
From this directory (after pnpm install at the repo root):
export OPENAI_API_KEY=sk-... # or add it to .env (see .env.example)
pnpm start # starts the bot on http://localhost:3978/api/messagesIn a second terminal:
pnpm playground # opens the M365 Agents Playground at http://localhost:56150Then, in the Playground:
- Ask anything → the agent replies, streaming in by message edit (a typing
indicator first, then text that fills in as it's edited, following Teams'
baseline post-then-
updateActivitystreaming model). - Ask for a summary, status, or any structured data → the agent calls
the
show_cardtool and posts an Adaptive Card (header, facts, table). - Ask it to "announce X to the team" → it drafts the message, posts an Approve/Reject card, and only sends after you approve (the card updates in place to ✅/🚫).
That exercises the CopilotKit bot engine and the Teams adapter end-to-end: streaming, agent-rendered Adaptive Cards, and human-in-the-loop.
app/index.tsx: the whole bot, covering an in-processBuiltInAgentruntime, thecreateBot({ adapters: [teams()] })wiring, anonMessagehandler that runs the agent, and the agent-facingshow_cardtool.app/human-in-the-loop/: theconfirm_writeapproval gate and the Adaptive Card it posts. This is user-land code, not SDK code.
By default the example serves an in-process BuiltInAgent. To point the bot at
a remote AG-UI endpoint (a deployed CopilotKit runtime, LangGraph, and so on)
instead, swap the agent factory to read a URL from the environment:
agent: (threadId) => {
const a = new SanitizingHttpAgent({ url: process.env.AGENT_URL! });
a.threadId = threadId;
return a;
},The Playground needs no credentials; real Teams does. The high-level path:
- Register the bot with Microsoft. Create an Entra app
registration
and note its Application (client) ID, Directory (tenant) ID, and a client
secret. Create an Azure Bot
resource
that uses that app, enable the Microsoft Teams channel, and set its
messaging endpoint to
https://<your-host>/api/messages. - Give the bot the credentials. Set
clientId/clientSecret/tenantId(the names the M365 Agents SDK reads) in the bot's environment. With them set, the bot acks each turn and runs the agent on a detached context, so HITL approvals can resume minutes later. - Build and upload the app package (below), then in Teams: Apps → Manage your apps → Upload a custom app.
The full step-by-step walkthrough is in the Microsoft Teams guide.
The app package is the manifest + icons you sideload into Teams. Build it with:
pnpm package # -> appPackage/appPackage.zipThe script (appPackage/package.mjs, dependency-free) reads your bot id from
MICROSOFT_APP_ID / CLIENT_ID / clientId (env or .env) and injects it into
the manifest, validates the manifest, and auto-generates placeholder icons if
they're missing, so the committed manifest.json stays a placeholder and you
never hardcode your id. See appPackage/README.md for
details.
The agent can read uploaded files and render charts. Upload a CSV and ask for a
pie/bar chart: the bot parses the data and calls render_chart, which posts a
native Teams chart (an Adaptive Card chart element, no image generation, no
headless browser). How the file reaches the bot depends on where it's uploaded,
because of a Teams limitation:
-
1:1 (personal) chat — the file is delivered to the bot inline (requires
supportsFiles: truein the manifest, already set). Works with no extra setup. -
Channel / group chat — Teams does not send the file to bots here, so the bot fetches it through Microsoft Graph. That needs two application permissions on the bot's Entra app, consented once by a tenant admin:
Files.Read.All— download the file from SharePoint.Group.Read.All(or the manifest's RSCChannelMessage.Read.Group, which a team owner can consent without a tenant admin) — read the channel message that references the file.
Without that consent the bot still works — it asks the user to paste the data inline (which also renders a chart). To verify the Graph chain in a tenant where you control consent before requesting it org-wide, run
scripts/verify-graph-channel.ts(see its header).
Charts render natively in the Teams client, so there's nothing extra to install
(no Chromium, no headless browser). Native charts need a Teams app manifest at
version 1.25+ (already set in appPackage/manifest.json).
The bot is a plain HTTP service: it serves POST /api/messages (plus a
/healthz liveness probe) and binds PORT, so it runs anywhere a Node process
does. Teams is an inbound webhook, so the service needs a public URL: point
your Azure Bot resource's messaging endpoint at https://<your-host>/api/messages.
This example consumes the @copilotkit/* packages via the workspace:*
protocol, so it always builds from the in-repo source — not the npm
registry. That decouples the deploy from publishing: a change to packages/**
redeploys with the new code immediately.
Because it's a workspace member, the deploy must run from the repo root so
the workspace and packages/** are visible. The bot runs its BuiltInAgent
runtime in-process (on RUNTIME_PORT, localhost-only), so it's a single
service — no separate runtime process. On Railway (or any host), set:
| Setting | Value |
|---|---|
| Root Directory | repo root (/) |
| Build Command | pnpm install && pnpm --filter teams-example build |
| Start Command | pnpm --filter teams-example start |
| Watch Paths | packages/**, examples/teams/**, pnpm-lock.yaml, package.json |
pnpm --filter teams-example build builds the workspace libs the example
imports (@copilotkit/bot, bot-teams, bot-ui, runtime) and everything
they depend on, via the Nx project graph — so tsx runs against fresh dist. The Watch Paths are
what make a packages/**-only change trigger a redeploy. On Railway, generate a
public domain on the service (Settings → Networking); it routes to $PORT,
which the bot listens on for /api/messages.
Copying this example out of the monorepo? Replace the
workspace:*ranges inpackage.jsonwith the published versions (e.g.@copilotkit/bot-teams: ^0.0.1) —workspace:*only resolves inside this monorepo.
Set the environment for wherever you deploy:
OPENAI_API_KEY(required): the bot runs aBuiltInAgentand exits at startup without it.OPENAI_MODEL(optional): defaults toopenai/gpt-5.5.clientId/clientSecret/tenantId: needed to reach real Teams (see above). The in-processBuiltInAgentruntime stays onRUNTIME_PORT(localhost-only, default 8200).
Note: the conversation store and pending HITL approvals are in-memory, so they do not survive a restart. Swap in a durable store before relying on long-lived approvals in production.