Two checklists: what makes a complete package, and what external setup is needed when adding a new framework.
Everything below should exist in showcase/integrations/<slug>/:
-
manifest.yaml— name, slug, category, language, features, demos,deployed: false,generative_ui,interaction_modalities, and optionallymanaged_platform -
package.json— dependencies including@copilotkit/react-core,zod,tailwindcss -
tsconfig.json -
next.config.ts -
postcss.config.mjs
-
layout.tsx— importsglobals.css,copilotkit-overrides.css,@copilotkit/react-core/v2/styles.css -
globals.css— NO* { margin: 0; padding: 0; }reset (onlybox-sizing: border-box) -
copilotkit-overrides.css— separate file for CopilotKit class overrides (survives Tailwind v4 purging) -
api/copilotkit/route.ts— runtime endpoint -
api/health/route.ts— health check endpoint -
error-boundary.tsx— DemoErrorBoundary component
One per declared feature. Each demo must:
- Use
CopilotKitprovider withruntimeUrl="/api/copilotkit"and correctagentname - Use
@copilotkit/react-core/v2imports (NOT@copilotkitnext/) - For
CopilotChatdemos: wrapper div withpx-6for horizontal padding (matches Dojo) - For
CopilotSidebardemos: no extra padding needed (sidebar has built-inpx-8) - Use
h-fullnoth-screen(demos render in iframes) - Use inline styles for dynamic content (Tailwind v4 purges classes it can't statically find)
- Include
useConfigureSuggestionswith relevant suggestions
- Agent code in
src/agents/(Python) orsrc/lib/(TypeScript) - One agent per feature (names must match the
agentprop in demo pages) -
langgraph.json(Python) or equivalent config - Pin framework versions — see "Dependency Pinning" below
-
Dockerfile— multi-stage, starts both agent backend and Next.js frontend -
entrypoint.sh— starts agent server and Next.js, waits for both
-
playwright.config.ts -
tests/— one E2E test per demo (basic: load → send message → get response) -
qa/— manual QA checklist per demo
- Logo SVG at
showcase/shell/public/logos/<slug>.svg
Two directories hold integration code, and they play different roles. Understanding the relationship is critical before adding or modifying a package.
examples/integrations/<name>/— the Dojo example. This is the dep-pinning source of truth: minimal, focused agent code used to prove a framework works against CopilotKit/AG-UI. The weekly drift-detection workflow and the "Always pin agent framework and SDK versions to exact versions from the working Dojo example" rule (see "Dependency Pinning" below) both treat this directory as canonical.showcase/integrations/<slug>/— the full triple-duty integration:- Partner-facing demo (lives on
showcase.copilotkit.dev) - Cloneable starter source (extracted on-demand via
extract-starter.ts) - Iframe-embedded experience inside the public showcase shell
- Partner-facing demo (lives on
examples/integrations/<name>/ ──(migrate-integration-examples.ts)──▶ showcase/integrations/<slug>/src/agents/
showcase/integrations/<slug>/ ──(extract-starter.ts)────────────────▶ standalone starter (on-demand)
showcase/scripts/migrate-integration-examples.tscopies agent code fromexamples/integrations/<name>/intoshowcase/integrations/<slug>/src/agents/. It never runs in reverse.showcase/scripts/extract-starter.tsextracts a clean standalone starter from any integration on demand, dereferencing symlinks and stripping test/CI artifacts.- Do not hand-edit agent code inside
showcase/integrations/<slug>/src/agents/if the package has a Dojo counterpart — fix it upstream inexamples/integrations/<name>/and re-run the migration script.
Five packages exist only in showcase and have no examples/integrations/<name>/ sibling:
ag2claude-sdk-pythonclaude-sdk-typescriptlangroidspring-ai
These are authored directly in showcase/integrations/<slug>/ and are exempt from the pin-to-Dojo rule — there is no Dojo to pin to. They still must pin exact versions (see "Dependency Pinning"), but the reference is whatever the framework's own examples or release notes recommend, not a sibling examples/integrations/ directory.
Several packages have different names in examples/integrations/ vs showcase/integrations/. The aliasing is historical — showcase standardized on shorter, marketing-friendly slugs while the Dojo kept the original framework-canonical names.
showcase/integrations/ slug |
examples/integrations/ name |
Why different |
|---|---|---|
google-adk |
adk |
Showcase prefixes with vendor for disambiguation |
langgraph-typescript |
langgraph-js |
Showcase prefers full language name (-typescript) |
ms-agent-dotnet |
ms-agent-framework-dotnet |
Showcase shortens -framework- out of the slug |
ms-agent-python |
ms-agent-framework-python |
Same — shorter slug in showcase |
strands |
strands-python |
Showcase drops the language suffix (no TS variant exists) |
When running migrate-integration-examples.ts or reasoning about drift, remember that the script internally maps these aliases — don't "fix" them by renaming one side.
- Create service in the CopilotKit Showcase Railway project, US-West region
- Type: Docker (image from GHCR, not source build)
- Image URL:
ghcr.io/copilotkit/showcase-<slug>:latest - Health check path:
/api/health - Link shared variables group (contains API keys)
- Set
NODE_ENV=production,NEXT_PUBLIC_BASE_URL=https://showcase.copilotkit.dev
- Ensure
RAILWAY_TOKENsecret exists in the repo
- Add slug to
workflow_dispatch.inputs.service.options - Add change detection filter for
showcase/integrations/<slug>/** - Add build job: build Docker image → push to GHCR → trigger Railway deploy
- Wire up the
RAILWAY_TOKENsecret
- Run
npx tsx showcase/scripts/generate-registry.tsto regenerateregistry.json - Verify the integration appears on the Integrations page
- Verify demos load in the drawer (Preview tab)
- Verify Railway service is healthy:
curl https://showcase-<slug>-production.up.railway.app/api/health - Verify all demos respond: visit each
/demos/<id>route - Set
deployed: trueinmanifest.yaml - Verify constraint validation passes:
npx tsx showcase/scripts/validate-constraints.ts <slug> - Regenerate registry:
npx tsx showcase/scripts/generate-registry.ts - Commit and push — stack nav chip will light up automatically
- If the framework name in the stack nav differs from
manifest.yamlname, verifystartsWithmatching works - Demo content (Code/Docs tabs): run
npx tsx showcase/scripts/generate-demo-content.tsif it exists
Always pin agent framework and SDK versions to exact versions from the working Dojo example. Do not use floating ranges like >=0.3.0 — they resolve to different versions over time and silently break APIs.
Why this matters: langchain>=0.3.0 resolved to 0.3.x which lacked create_agent. The Dojo uses langchain==1.2.0 where it exists. A floating range that worked at scaffold time broke on the next Docker build when a different version was pulled.
What to pin:
- Agent framework packages (langchain, langgraph, @mastra/core, etc.)
- CopilotKit SDK packages (copilotkit, @copilotkit/runtime, etc.)
- LLM provider SDKs (langchain-openai, @ai-sdk/openai, etc.)
What can float:
- Standard utilities (zod, react, next) — these have stable APIs
- Dev dependencies (playwright, typescript, tailwind)
Where to find correct versions:
- Check the corresponding Dojo example at
examples/integrations/<slug>/ - Use exact versions from its
requirements.txt/pyproject.toml/package.json - The weekly drift detection workflow will flag when pinned versions fall behind
LangGraph supports two agent authoring styles, and showcase uses both. When touching a LangGraph package — or adding a new one — decide the style explicitly and match the existing sibling's idioms.
- Node-based — hand-rolled
StateGraphwithaddNode(...), explicit edges, and custom routing logic. Maximum control; more code to maintain. - Prebuilt —
create_react_agent/create_agenthelpers that wrap the common ReAct pattern. Minimal code; less flexibility.
| Package | Style | Evidence |
|---|---|---|
showcase/integrations/langgraph-python |
Prebuilt | create_react_agent in src/agents/main.py:53 |
showcase/integrations/langgraph-fastapi |
Prebuilt | create_react_agent in src/agents/src/agent.py:166 |
showcase/integrations/langgraph-typescript |
Node-based | StateGraph in src/agent/graph.ts:271 |
The ag-ui/apps/dojo/ e2e tests exclusively exercise node-based graphs. This means prebuilt-agent coverage is thin in the Dojo even though two of the three LangGraph packages users clone from showcase are prebuilt.
Cross-reference the action inventory for the full breakdown of which AG-UI features are exercised where: https://www.notion.so/3443aa38185281b5a1dfc6e0890264e1.
- When adding a new LangGraph-based package, decide the authoring style explicitly and match the idioms of the corresponding showcase sibling (Python → prebuilt, TypeScript → node-based) unless you have a concrete reason to diverge.
- If you do diverge, document why in the package's README and add an entry to the table above.
- Do not silently convert a package between styles — it's a public API change for anyone who cloned the starter.
This distinction only applies to LangGraph today. Other frameworks (CrewAI, Mastra, etc.) have their own framework-specific authoring idioms — out of scope for this section.
| Gotcha | Fix |
|---|---|
| CSS classes purged by Tailwind v4 | Put CopilotKit overrides in copilotkit-overrides.css, not globals.css |
* { margin: 0; padding: 0; } |
NEVER use this reset — it strips CopilotKit's internal padding |
| Chat messages flush to edges | Add px-6 to the CopilotChat wrapper div |
h-screen in demos |
Use h-full — demos render inside iframes |
| Dynamic content unstyled | Use inline style={} not Tailwind classes for agent-generated content |
| Stale lockfile | Run pnpm install after changing package.json, commit the lockfile |
| Stack chip not lighting up | Check deployed: true in manifest and registry name matching |
| Agent import errors in Docker | Pin framework deps to exact Dojo versions — floating ranges resolve differently over time |