Contributing

June 22, 2026 · View on GitHub

Reporting issues

For bugs, feature requests, documentation issues, or new adapter requests, pick an issue template. Each one asks for the information we need to triage the report.

For questions and getting help, see SUPPORT.md. Security vulnerabilities should be reported privately — see SECURITY.md. Do not file a public issue for security issues.

Building your own adapter

Want to add Chat SDK support for a platform that isn't covered by the official adapters? See Building a community adapter for a walkthrough of the Adapter interface, testing, packaging, and getting your adapter listed on chat-sdk.dev.

Package conventions

The repo uses konsistent (configured in .github/konsistent.json, run via pnpm konsistent) to enforce a consistent public surface across adapter and state packages.

Adapter packages (packages/adapter-*) must:

  • Live in src/index.ts with a sibling src/types.ts
  • Import the Adapter type from chat
  • Export a ${Name}Adapter class that implements Adapter
  • Export a create${Name}Adapter factory function whose parameter is typed ${Name}AdapterConfig
  • Export the ${Name}AdapterConfig type from ./types

State packages (packages/state-*) must:

  • Import the StateAdapter type from chat
  • Export a ${Name}StateAdapter class that implements StateAdapter
  • Export a create${Name}State factory function whose parameter is typed ${Name}StateAdapterOptions
  • Export the ${Name}StateAdapterOptions type

${Name} is the kebab-case package suffix in PascalCase — most cases are mechanical (discordDiscord), but a few overrides live in kebabToPascalMap in the config (e.g. gchatGoogleChat, whatsappWhatsApp).

Example apps (examples/*) must:

  • Be marked "private": true (they are never published)
  • Name their package.json name field example-* (e.g. example-nextjs-chat)

The example-* name lets the ignore glob in .changeset/config.json exclude every example from versioning and publishing automatically, so adding a new example needs no changeset and no edit to the changesets config. A test in packages/integration-tests asserts every examples/* package follows this convention and is in the resolved changesets ignore list.

Signed Commits

All commits to this repository must be signed and verified. Pull requests with unsigned commits will not be merged.

GitHub has a guide on setting this up: Signing commits. The easiest path is usually signing with SSH using a key you've already added to GitHub.

Verify your setup by checking that new commits show a "Verified" badge on github.com.

Commit messages

We follow Conventional Commitsfeat:, fix:, docs:, chore:, etc., optionally with a scope (e.g., fix(slack): ...). The release workflow's auto-generated version PRs also use this convention (chore(release): version packages), so keeping new commits consistent makes changelogs and release PRs predictable.

Development

Testing

Run all unit tests across every package in a single Vitest Workspace run:

pnpm test:workspace

This produces one combined report covering all 11 unit-test packages. Integration tests (@chat-adapter/integration-tests) are excluded since they require platform credentials.

You can also run tests per-package via Turborepo:

# All packages (including integration tests)
pnpm test

# Single package
pnpm --filter chat test
pnpm --filter @chat-adapter/slack test

Other commands

pnpm check       # Check all packages (linting and formatting)
pnpm typecheck   # Type-check all packages
pnpm knip        # Check for unused exports/dependencies
pnpm konsistent  # Enforce adapter/state-package shape conventions
pnpm validate    # Run everything (knip, lint, typecheck, test, build)

Changesets

This project uses Changesets for version management. Every PR that changes a package's behavior must include a changeset.

Adding a Changeset

When you make a change that should be released (bug fix, new feature, breaking change), run:

pnpm changeset

This interactive CLI will ask:

  1. Which packages changed? — Select affected packages (space to select, enter to confirm)
  2. Bump type?major (breaking), minor (feature), or patch (fix)
  3. Summary — A brief description for the changelog

This creates a markdown file in .changeset/ describing your change. Commit this file with your PR.

When to Add a Changeset

  • Do add for: bug fixes, new features, breaking changes, dependency updates affecting behavior
  • Don't add for: documentation changes, internal refactors, test changes, CI updates, or changes to example apps (examples/* are private and ignored by changesets)

Changeset Types

TypeWhen to UseVersion Bump
patchBug fixes, minor improvements4.0.04.0.1
minorNew features (backward compatible)4.0.04.1.0
majorBreaking changes4.0.05.0.0

All packages in this monorepo use fixed versioning — they always share the same version number, and any release bumps every package together.

Updating documentation

User-facing docs live in apps/docs/content/docs/ and render at chat-sdk.dev/docs. When a PR changes behavior, public APIs, or environment variables, update the relevant page(s) in the same PR.

To preview docs locally:

pnpm --filter docs dev

Preview Branch Testing

The example app includes a middleware that can proxy webhook requests to a preview branch deployment. This allows testing preview branches with real webhook traffic from Slack/Teams/GChat.

Setup

  1. Deploy a preview branch to Vercel (e.g., https://chat-sdk-git-feature-branch.vercel.app)
  2. Go to /settings on the production deployment
  3. Enter the preview branch URL and save

To disable

Clear the URL on the settings page.

Files

  • examples/nextjs-chat/src/middleware.ts - The proxy middleware
  • examples/nextjs-chat/src/app/settings/page.tsx - Settings UI
  • examples/nextjs-chat/src/app/api/settings/preview-branch/route.ts - API to get/set the URL