Looking for the chatbot template? It's now here.

Microsoft Teams

Configure the Microsoft Teams adapter with Azure Bot Service.

Installation

Terminal
pnpm add @chat-adapter/teams

Usage

The adapter auto-detects TEAMS_APP_ID, TEAMS_APP_PASSWORD, and TEAMS_APP_TENANT_ID from environment variables:

lib/bot.ts
import { Chat } from "chat";
import { createTeamsAdapter } from "@chat-adapter/teams";

const bot = new Chat({
  userName: "mybot",
  adapters: {
    teams: createTeamsAdapter({
      appType: "SingleTenant",
    }),
  },
});

bot.onNewMention(async (thread, message) => {
  await thread.post("Hello from Teams!");
});

Azure Bot setup

1. Create Azure Bot resource

  1. Go to portal.azure.com
  2. Click Create a resource
  3. Search for Azure Bot and select it
  4. Click Create and fill in:
    • Bot handle: Unique identifier for your bot
    • Subscription: Your Azure subscription
    • Resource group: Create new or use existing
    • Pricing tier: F0 (free) for testing
    • Type of App: Single Tenant (recommended for enterprise)
    • Creation type: Create new Microsoft App ID
  5. Click Review + create then Create

2. Get app credentials

  1. Go to your Bot resource then Configuration
  2. Copy Microsoft App ID as TEAMS_APP_ID
  3. Click Manage Password (next to Microsoft App ID)
  4. In the App Registration page, go to Certificates & secrets
  5. Click New client secret, add description, select expiry, click Add
  6. Copy the Value immediately (shown only once) as TEAMS_APP_PASSWORD
  7. Go to Overview and copy Directory (tenant) ID as TEAMS_APP_TENANT_ID

3. Configure messaging endpoint

  1. In your Azure Bot resource, go to Configuration
  2. Set Messaging endpoint to https://your-domain.com/api/webhooks/teams
  3. Click Apply

4. Enable Teams channel

  1. In your Azure Bot resource, go to Channels
  2. Click Microsoft Teams
  3. Accept the terms of service
  4. Click Apply

5. Create Teams app package

Create a manifest.json file:

manifest.json
{
  "$schema": "https://developer.microsoft.com/en-us/json-schemas/teams/v1.16/MicrosoftTeams.schema.json",
  "manifestVersion": "1.16",
  "version": "1.0.0",
  "id": "your_app_id_here",
  "packageName": "com.yourcompany.chatbot",
  "developer": {
    "name": "Your Company",
    "websiteUrl": "https://your-domain.com",
    "privacyUrl": "https://your-domain.com/privacy",
    "termsOfUseUrl": "https://your-domain.com/terms"
  },
  "name": {
    "short": "Chat Bot",
    "full": "Chat SDK Demo Bot"
  },
  "description": {
    "short": "A chat bot powered by Chat SDK",
    "full": "A chat bot powered by Chat SDK that responds to messages and commands."
  },
  "icons": {
    "outline": "outline.png",
    "color": "color.png"
  },
  "accentColor": "#FFFFFF",
  "bots": [
    {
      "botId": "your_app_id_here",
      "scopes": ["personal", "team", "groupchat"],
      "supportsFiles": false,
      "isNotificationOnly": false
    }
  ],
  "permissions": ["identity", "messageTeamMembers"],
  "validDomains": ["your-domain.com"]
}

Create icon files (32x32 outline.png and 192x192 color.png), then zip all three files together.

6. Upload app to Teams

For testing (sideloading):

  1. In Teams, click Apps in the sidebar
  2. Click Manage your apps then Upload an app
  3. Click Upload a custom app and select your zip file

For organization-wide deployment:

  1. Go to Teams Admin Center
  2. Go to Teams apps then Manage apps
  3. Click Upload new app and select your zip file
  4. Go to Setup policies to control who can use the app

Configuration

All options are auto-detected from environment variables when not provided.

OptionRequiredDescription
appIdNo*Azure Bot App ID. Auto-detected from TEAMS_APP_ID
appPasswordNo*Azure Bot App Password. Auto-detected from TEAMS_APP_PASSWORD
appTypeNo"MultiTenant" or "SingleTenant" (default: "MultiTenant")
appTenantIdFor SingleTenantAzure AD Tenant ID. Auto-detected from TEAMS_APP_TENANT_ID
loggerNoLogger instance (defaults to ConsoleLogger("info"))

*appId and appPassword are required — either via config or env vars.

Environment variables

.env.local
TEAMS_APP_ID=...
TEAMS_APP_PASSWORD=...
TEAMS_APP_TENANT_ID=...  # Required for SingleTenant

Features

FeatureSupported
MentionsYes
Reactions (add/remove)Read-only
Cards (Adaptive Cards)Yes
ModalsNo
StreamingPost+Edit fallback
DMsYes
Ephemeral messagesNo (DM fallback)
File uploadsYes
Typing indicatorNo
Message historyNo

Limitations

  • Adding reactions: Teams Bot Framework doesn't support bots adding reactions. Calling addReaction() or removeReaction() throws a NotImplementedError. The bot can still receive reaction events via onReaction().
  • Typing indicators: Not available via Bot Framework. startTyping() is a no-op.

Message history (fetchMessages)

Fetching message history requires the Microsoft Graph API with client credentials flow. To enable it:

  1. Set appTenantId in the adapter config
  2. Grant one of these Azure AD app permissions:
    • ChatMessage.Read.Chat
    • Chat.Read.All
    • Chat.Read.WhereInstalled

Without these permissions, fetchMessages will not be able to retrieve channel history.

Receiving all messages

By default, Teams bots only receive messages when directly @-mentioned. To receive all messages in a channel or group chat, add Resource-Specific Consent (RSC) permissions to your Teams app manifest:

manifest.json
{
  "authorization": {
    "permissions": {
      "resourceSpecific": [
        {
          "name": "ChannelMessage.Read.Group",
          "type": "Application"
        }
      ]
    }
  }
}

Alternatively, configure the bot in Azure to receive all messages.

Troubleshooting

"Unauthorized" error

  • Verify TEAMS_APP_ID and TEAMS_APP_PASSWORD are correct
  • For SingleTenant apps, ensure TEAMS_APP_TENANT_ID is set
  • Check that the messaging endpoint URL is correct in Azure

Bot not appearing in Teams

  • Verify the Teams channel is enabled in Azure Bot
  • Check that the app manifest is correctly configured
  • Ensure the app is installed in the workspace/team

Messages not received

  • Verify the messaging endpoint URL is correct
  • Check that your server is accessible from the internet
  • Review Azure Bot logs for errors