Chuyển đến nội dung chính

Documentation Index

Fetch the complete documentation index at: https://docs2.openclaw.ai/llms.txt

Use this file to discover all available pages before exploring further.

Trạng thái: hỗ trợ văn bản + tệp đính kèm DM; gửi tệp trong kênh/nhóm yêu cầu sharePointSiteId + quyền Graph (xem Gửi tệp trong cuộc trò chuyện nhóm). Poll được gửi qua Adaptive Cards. Hành động tin nhắn cung cấp rõ upload-file cho các lượt gửi ưu tiên tệp.

Plugin đi kèm

Microsoft Teams được phát hành dưới dạng Plugin đi kèm trong các bản phát hành OpenClaw hiện tại, vì vậy không cần cài đặt riêng trong bản dựng đóng gói thông thường. Nếu bạn đang dùng bản dựng cũ hơn hoặc bản cài đặt tùy chỉnh loại trừ Teams đi kèm, hãy cài đặt trực tiếp gói npm:
openclaw plugins install @openclaw/msteams
Dùng gói trần để theo dõi thẻ phát hành chính thức hiện tại. Chỉ ghim một phiên bản chính xác khi bạn cần bản cài đặt có thể tái lập. Checkout cục bộ (khi chạy từ repo git):
openclaw plugins install ./path/to/local/msteams-plugin
Chi tiết: Plugin

Thiết lập nhanh

@microsoft/teams.cli xử lý đăng ký bot, tạo manifest và tạo thông tin xác thực bằng một lệnh duy nhất. 1. Cài đặt và đăng nhập
npm install -g @microsoft/teams.cli@preview
teams login
teams status   # verify you're logged in and see your tenant info
Teams CLI hiện đang ở bản preview. Lệnh và cờ có thể thay đổi giữa các bản phát hành.
2. Khởi động tunnel (Teams không thể truy cập localhost) Cài đặt và xác thực devtunnel CLI nếu bạn chưa làm (hướng dẫn bắt đầu).
# One-time setup (persistent URL across sessions):
devtunnel create my-openclaw-bot --allow-anonymous
devtunnel port create my-openclaw-bot -p 3978 --protocol auto

# Each dev session:
devtunnel host my-openclaw-bot
# Your endpoint: https://<tunnel-id>.devtunnels.ms/api/messages
--allow-anonymous là bắt buộc vì Teams không thể xác thực với devtunnels. Mỗi yêu cầu bot đến vẫn được Teams SDK tự động xác thực.
Các lựa chọn thay thế: ngrok http 3978 hoặc tailscale funnel 3978 (nhưng chúng có thể thay đổi URL ở mỗi phiên). 3. Tạo ứng dụng
teams app create \
  --name "OpenClaw" \
  --endpoint "https://<your-tunnel-url>/api/messages"
Lệnh duy nhất này:
  • Tạo một ứng dụng Entra ID (Azure AD)
  • Tạo một client secret
  • Xây dựng và tải lên manifest ứng dụng Teams (kèm biểu tượng)
  • Đăng ký bot (mặc định do Teams quản lý - không cần gói đăng ký Azure)
Đầu ra sẽ hiển thị CLIENT_ID, CLIENT_SECRET, TENANT_IDTeams App ID - hãy ghi lại chúng cho các bước tiếp theo. Nó cũng đề nghị cài đặt ứng dụng trực tiếp trong Teams. 4. Cấu hình OpenClaw bằng thông tin xác thực từ đầu ra:
{
  channels: {
    msteams: {
      enabled: true,
      appId: "<CLIENT_ID>",
      appPassword: "<CLIENT_SECRET>",
      tenantId: "<TENANT_ID>",
      webhook: { port: 3978, path: "/api/messages" },
    },
  },
}
Hoặc dùng trực tiếp biến môi trường: MSTEAMS_APP_ID, MSTEAMS_APP_PASSWORD, MSTEAMS_TENANT_ID. 5. Cài đặt ứng dụng trong Teams teams app create sẽ nhắc bạn cài đặt ứng dụng - chọn “Install in Teams”. Nếu đã bỏ qua, bạn có thể lấy liên kết sau:
teams app get <teamsAppId> --install-link
6. Xác minh mọi thứ hoạt động
teams app doctor <teamsAppId>
Lệnh này chạy chẩn đoán trên đăng ký bot, cấu hình ứng dụng AAD, tính hợp lệ của manifest và thiết lập SSO. Đối với triển khai production, hãy cân nhắc dùng xác thực liên kết (chứng chỉ hoặc managed identity) thay vì client secret.
Cuộc trò chuyện nhóm bị chặn theo mặc định (channels.msteams.groupPolicy: "allowlist"). Để cho phép trả lời trong nhóm, đặt channels.msteams.groupAllowFrom, hoặc dùng groupPolicy: "open" để cho phép mọi thành viên (có cổng kiểm soát bằng nhắc đến).

Mục tiêu

  • Trò chuyện với OpenClaw qua DM Teams, cuộc trò chuyện nhóm hoặc kênh.
  • Giữ định tuyến xác định: câu trả lời luôn quay lại kênh mà chúng đến từ đó.
  • Mặc định dùng hành vi kênh an toàn (yêu cầu nhắc đến trừ khi được cấu hình khác).

Ghi cấu hình

Theo mặc định, Microsoft Teams được phép ghi các cập nhật cấu hình được kích hoạt bởi /config set|unset (yêu cầu commands.config: true). Tắt bằng:
{
  channels: { msteams: { configWrites: false } },
}

Kiểm soát truy cập (DM + nhóm)

Truy cập DM
  • Mặc định: channels.msteams.dmPolicy = "pairing". Người gửi không xác định sẽ bị bỏ qua cho đến khi được phê duyệt.
  • channels.msteams.allowFrom nên dùng AAD object ID ổn định hoặc nhóm truy cập người gửi tĩnh như accessGroup:core-team.
  • Không dựa vào khớp UPN/tên hiển thị cho allowlist - chúng có thể thay đổi. OpenClaw tắt khớp tên trực tiếp theo mặc định; bật rõ ràng bằng channels.msteams.dangerouslyAllowNameMatching: true.
  • Trình hướng dẫn có thể phân giải tên thành ID qua Microsoft Graph khi thông tin xác thực cho phép.
Truy cập nhóm
  • Mặc định: channels.msteams.groupPolicy = "allowlist" (bị chặn trừ khi bạn thêm groupAllowFrom). Dùng channels.defaults.groupPolicy để ghi đè mặc định khi chưa đặt.
  • channels.msteams.groupAllowFrom kiểm soát người gửi hoặc nhóm truy cập người gửi tĩnh nào có thể kích hoạt trong cuộc trò chuyện nhóm/kênh (fallback về channels.msteams.allowFrom).
  • Đặt groupPolicy: "open" để cho phép mọi thành viên (mặc định vẫn có cổng kiểm soát bằng nhắc đến).
  • Để không cho phép kênh nào, đặt channels.msteams.groupPolicy: "disabled".
Ví dụ:
{
  channels: {
    msteams: {
      groupPolicy: "allowlist",
      groupAllowFrom: ["00000000-0000-0000-0000-000000000000", "accessGroup:core-team"],
    },
  },
}
Allowlist Teams + kênh
  • Giới hạn phạm vi trả lời nhóm/kênh bằng cách liệt kê team và kênh dưới channels.msteams.teams.
  • Khóa nên dùng Teams conversation ID ổn định từ liên kết Teams, không dùng tên hiển thị có thể thay đổi.
  • Khi groupPolicy="allowlist" và có allowlist team, chỉ team/kênh được liệt kê mới được chấp nhận (có cổng kiểm soát bằng nhắc đến).
  • Trình hướng dẫn cấu hình chấp nhận mục Team/Channel và lưu chúng cho bạn.
  • Khi khởi động, OpenClaw phân giải tên allowlist team/kênh và người dùng thành ID (khi quyền Graph cho phép) và ghi log ánh xạ; tên team/kênh chưa phân giải được giữ nguyên như đã nhập nhưng mặc định bị bỏ qua khi định tuyến trừ khi bật channels.msteams.dangerouslyAllowNameMatching: true.
Ví dụ:
{
  channels: {
    msteams: {
      groupPolicy: "allowlist",
      teams: {
        "My Team": {
          channels: {
            General: { requireMention: true },
          },
        },
      },
    },
  },
}

Xác thực liên kết (chứng chỉ cộng với managed identity)

Đã thêm trong 2026.4.11
Đối với triển khai production, OpenClaw hỗ trợ xác thực liên kết như một lựa chọn thay thế an toàn hơn cho client secret. Có hai phương thức:

Tùy chọn A: Xác thực dựa trên chứng chỉ

Dùng chứng chỉ PEM đã đăng ký với app registration Entra ID của bạn. Thiết lập:
  1. Tạo hoặc lấy chứng chỉ (định dạng PEM với khóa riêng).
  2. Trong Entra ID → App Registration → Certificates & secretsCertificates → Tải lên chứng chỉ công khai.
Cấu hình:
{
  channels: {
    msteams: {
      enabled: true,
      appId: "<APP_ID>",
      tenantId: "<TENANT_ID>",
      authType: "federated",
      certificatePath: "/path/to/cert.pem",
      webhook: { port: 3978, path: "/api/messages" },
    },
  },
}
Biến môi trường:
  • MSTEAMS_AUTH_TYPE=federated
  • MSTEAMS_CERTIFICATE_PATH=/path/to/cert.pem

Tùy chọn B: Azure Managed Identity

Dùng Azure Managed Identity để xác thực không cần mật khẩu. Cách này lý tưởng cho triển khai trên hạ tầng Azure (AKS, App Service, Azure VM) nơi có managed identity. Cách hoạt động:
  1. Pod/VM của bot có managed identity (system-assigned hoặc user-assigned).
  2. Một federated identity credential liên kết managed identity với app registration Entra ID.
  3. Khi chạy, OpenClaw dùng @azure/identity để lấy token từ endpoint Azure IMDS (169.254.169.254).
  4. Token được chuyển cho Teams SDK để xác thực bot.
Điều kiện tiên quyết:
  • Hạ tầng Azure có bật managed identity (AKS workload identity, App Service, VM)
  • Federated identity credential được tạo trên app registration Entra ID
  • Quyền truy cập mạng tới IMDS (169.254.169.254:80) từ pod/VM
Cấu hình (system-assigned managed identity):
{
  channels: {
    msteams: {
      enabled: true,
      appId: "<APP_ID>",
      tenantId: "<TENANT_ID>",
      authType: "federated",
      useManagedIdentity: true,
      webhook: { port: 3978, path: "/api/messages" },
    },
  },
}
Cấu hình (danh tính được quản lý do người dùng gán):
{
  channels: {
    msteams: {
      enabled: true,
      appId: "<APP_ID>",
      tenantId: "<TENANT_ID>",
      authType: "federated",
      useManagedIdentity: true,
      managedIdentityClientId: "<MI_CLIENT_ID>",
      webhook: { port: 3978, path: "/api/messages" },
    },
  },
}
Biến môi trường:
  • MSTEAMS_AUTH_TYPE=federated
  • MSTEAMS_USE_MANAGED_IDENTITY=true
  • MSTEAMS_MANAGED_IDENTITY_CLIENT_ID=<client-id> (chỉ dành cho loại do người dùng gán)

Thiết lập danh tính khối lượng công việc AKS

Đối với các bản triển khai AKS sử dụng danh tính khối lượng công việc:
  1. Bật danh tính khối lượng công việc trên cụm AKS của bạn.
  2. Tạo thông tin xác thực danh tính liên kết trên đăng ký ứng dụng Entra ID:
    az ad app federated-credential create --id <APP_OBJECT_ID> --parameters '{
      "name": "my-bot-workload-identity",
      "issuer": "<AKS_OIDC_ISSUER_URL>",
      "subject": "system:serviceaccount:<NAMESPACE>:<SERVICE_ACCOUNT>",
      "audiences": ["api://AzureADTokenExchange"]
    }'
    
  3. Chú thích tài khoản dịch vụ Kubernetes bằng ID máy khách của ứng dụng:
    apiVersion: v1
    kind: ServiceAccount
    metadata:
      name: my-bot-sa
      annotations:
        azure.workload.identity/client-id: "<APP_CLIENT_ID>"
    
  4. Gắn nhãn pod để chèn danh tính khối lượng công việc:
    metadata:
      labels:
        azure.workload.identity/use: "true"
    
  5. Đảm bảo quyền truy cập mạng tới IMDS (169.254.169.254) - nếu dùng NetworkPolicy, hãy thêm quy tắc egress cho phép lưu lượng tới 169.254.169.254/32 trên cổng 80.

So sánh loại xác thực

Phương thứcCấu hìnhƯu điểmNhược điểm
Bí mật máy kháchappPasswordThiết lập đơn giảnCần xoay vòng bí mật, kém an toàn hơn
Chứng chỉauthType: "federated" + certificatePathKhông có bí mật dùng chung qua mạngTốn công quản lý chứng chỉ
Danh tính được quản lýauthType: "federated" + useManagedIdentityKhông dùng mật khẩu, không cần quản lý bí mậtCần hạ tầng Azure
Hành vi mặc định: Khi authType chưa được đặt, OpenClaw mặc định dùng xác thực bằng bí mật máy khách. Các cấu hình hiện có tiếp tục hoạt động mà không cần thay đổi.

Phát triển cục bộ (đường hầm)

Teams không thể truy cập localhost. Hãy dùng đường hầm phát triển cố định để URL của bạn giữ nguyên giữa các phiên:
# One-time setup:
devtunnel create my-openclaw-bot --allow-anonymous
devtunnel port create my-openclaw-bot -p 3978 --protocol auto

# Each dev session:
devtunnel host my-openclaw-bot
Các lựa chọn thay thế: ngrok http 3978 hoặc tailscale funnel 3978 (URL có thể thay đổi ở mỗi phiên). Nếu URL đường hầm của bạn thay đổi, hãy cập nhật điểm cuối:
teams app update <teamsAppId> --endpoint "https://<new-url>/api/messages"

Kiểm thử bot

Chạy chẩn đoán:
teams app doctor <teamsAppId>
Kiểm tra đăng ký bot, ứng dụng AAD, manifest và cấu hình SSO trong một lượt. Gửi tin nhắn kiểm thử:
  1. Cài đặt ứng dụng Teams (dùng liên kết cài đặt từ teams app get <id> --install-link)
  2. Tìm bot trong Teams và gửi DM
  3. Kiểm tra nhật ký Gateway để xem hoạt động gửi đến

Biến môi trường

Thay vào đó, tất cả khóa cấu hình đều có thể được đặt qua biến môi trường:
  • MSTEAMS_APP_ID
  • MSTEAMS_APP_PASSWORD
  • MSTEAMS_TENANT_ID
  • MSTEAMS_AUTH_TYPE (tùy chọn: "secret" hoặc "federated")
  • MSTEAMS_CERTIFICATE_PATH (liên kết + chứng chỉ)
  • MSTEAMS_CERTIFICATE_THUMBPRINT (tùy chọn, không bắt buộc cho xác thực)
  • MSTEAMS_USE_MANAGED_IDENTITY (liên kết + danh tính được quản lý)
  • MSTEAMS_MANAGED_IDENTITY_CLIENT_ID (chỉ MI do người dùng gán)

Hành động thông tin thành viên

OpenClaw cung cấp hành động member-info dựa trên Graph cho Microsoft Teams để agent và tự động hóa có thể phân giải chi tiết thành viên kênh (tên hiển thị, email, vai trò) trực tiếp từ Microsoft Graph. Yêu cầu:
  • Quyền RSC Member.Read.Group (đã có trong manifest được khuyến nghị)
  • Đối với tra cứu giữa các nhóm: quyền Ứng dụng Graph User.Read.All với sự chấp thuận của quản trị viên
Hành động này được kiểm soát bởi channels.msteams.actions.memberInfo (mặc định: bật khi có thông tin xác thực Graph).

Ngữ cảnh lịch sử

  • channels.msteams.historyLimit kiểm soát số lượng tin nhắn kênh/nhóm gần đây được đưa vào prompt.
  • Quay lui về messages.groupChat.historyLimit. Đặt 0 để tắt (mặc định 50).
  • Lịch sử luồng được lấy sẽ được lọc theo danh sách cho phép của người gửi (allowFrom / groupAllowFrom), nên việc gieo ngữ cảnh luồng chỉ bao gồm tin nhắn từ người gửi được phép.
  • Ngữ cảnh tệp đính kèm được trích dẫn (ReplyTo* bắt nguồn từ HTML trả lời của Teams) hiện được truyền nguyên như nhận được.
  • Nói cách khác, danh sách cho phép kiểm soát ai có thể kích hoạt agent; hiện nay chỉ có các đường dẫn ngữ cảnh bổ sung cụ thể được lọc.
  • Lịch sử DM có thể được giới hạn bằng channels.msteams.dmHistoryLimit (lượt người dùng). Ghi đè theo người dùng: channels.msteams.dms["<user_id>"].historyLimit.

Quyền RSC hiện tại của Teams (manifest)

Đây là các quyền resourceSpecific hiện có trong manifest ứng dụng Teams của chúng tôi. Chúng chỉ áp dụng bên trong nhóm/trò chuyện nơi ứng dụng được cài đặt. Đối với kênh (phạm vi nhóm):
  • ChannelMessage.Read.Group (Ứng dụng) - nhận tất cả tin nhắn kênh mà không cần @mention
  • ChannelMessage.Send.Group (Ứng dụng)
  • Member.Read.Group (Ứng dụng)
  • Owner.Read.Group (Ứng dụng)
  • ChannelSettings.Read.Group (Ứng dụng)
  • TeamMember.Read.Group (Ứng dụng)
  • TeamSettings.Read.Group (Ứng dụng)
Đối với trò chuyện nhóm:
  • ChatMessage.Read.Chat (Ứng dụng) - nhận tất cả tin nhắn trò chuyện nhóm mà không cần @mention
Để thêm quyền RSC qua Teams CLI:
teams app rsc add <teamsAppId> ChannelMessage.Read.Group --type Application

Ví dụ manifest Teams (đã lược bỏ thông tin nhạy cảm)

Ví dụ tối thiểu, hợp lệ với các trường bắt buộc. Thay thế ID và URL.
{
  $schema: "https://developer.microsoft.com/en-us/json-schemas/teams/v1.23/MicrosoftTeams.schema.json",
  manifestVersion: "1.23",
  version: "1.0.0",
  id: "00000000-0000-0000-0000-000000000000",
  name: { short: "OpenClaw" },
  developer: {
    name: "Your Org",
    websiteUrl: "https://example.com",
    privacyUrl: "https://example.com/privacy",
    termsOfUseUrl: "https://example.com/terms",
  },
  description: { short: "OpenClaw in Teams", full: "OpenClaw in Teams" },
  icons: { outline: "outline.png", color: "color.png" },
  accentColor: "#5B6DEF",
  bots: [
    {
      botId: "11111111-1111-1111-1111-111111111111",
      scopes: ["personal", "team", "groupChat"],
      isNotificationOnly: false,
      supportsCalling: false,
      supportsVideo: false,
      supportsFiles: true,
    },
  ],
  webApplicationInfo: {
    id: "11111111-1111-1111-1111-111111111111",
  },
  authorization: {
    permissions: {
      resourceSpecific: [
        { name: "ChannelMessage.Read.Group", type: "Application" },
        { name: "ChannelMessage.Send.Group", type: "Application" },
        { name: "Member.Read.Group", type: "Application" },
        { name: "Owner.Read.Group", type: "Application" },
        { name: "ChannelSettings.Read.Group", type: "Application" },
        { name: "TeamMember.Read.Group", type: "Application" },
        { name: "TeamSettings.Read.Group", type: "Application" },
        { name: "ChatMessage.Read.Chat", type: "Application" },
      ],
    },
  },
}

Lưu ý về manifest (các trường bắt buộc)

  • bots[].botId phải khớp với ID ứng dụng Azure Bot.
  • webApplicationInfo.id phải khớp với ID ứng dụng Azure Bot.
  • bots[].scopes phải bao gồm các bề mặt bạn định sử dụng (personal, team, groupChat).
  • bots[].supportsFiles: true là bắt buộc để xử lý tệp trong phạm vi cá nhân.
  • authorization.permissions.resourceSpecific phải bao gồm đọc/gửi kênh nếu bạn muốn lưu lượng kênh.

Cập nhật ứng dụng hiện có

Để cập nhật một ứng dụng Teams đã được cài đặt (ví dụ: để thêm quyền RSC):
# Download, edit, and re-upload the manifest
teams app manifest download <teamsAppId> manifest.json
# Edit manifest.json locally...
teams app manifest upload manifest.json <teamsAppId>
# Version is auto-bumped if content changed
Sau khi cập nhật, hãy cài đặt lại ứng dụng trong từng nhóm để quyền mới có hiệu lực, và thoát hoàn toàn rồi mở lại Teams (không chỉ đóng cửa sổ) để xóa siêu dữ liệu ứng dụng được lưu trong bộ nhớ đệm.

Khả năng: chỉ RSC so với Graph

Với chỉ Teams RSC (ứng dụng đã cài đặt, không có quyền Graph API)

Hoạt động:
  • Đọc nội dung văn bản của tin nhắn kênh.
  • Gửi nội dung văn bản của tin nhắn kênh.
  • Nhận tệp đính kèm trong cá nhân (DM).
Không hoạt động:
  • Nội dung hình ảnh hoặc tệp của kênh/nhóm (payload chỉ bao gồm stub HTML).
  • Tải xuống tệp đính kèm được lưu trong SharePoint/OneDrive.
  • Đọc lịch sử tin nhắn (ngoài sự kiện Webhook trực tiếp).

Với Teams RSC + quyền Ứng dụng Microsoft Graph

Bổ sung:
  • Tải xuống nội dung được lưu trữ (hình ảnh được dán vào tin nhắn).
  • Tải xuống tệp đính kèm được lưu trong SharePoint/OneDrive.
  • Đọc lịch sử tin nhắn kênh/trò chuyện qua Graph.

RSC so với Graph API

Khả năngQuyền RSCGraph API
Tin nhắn thời gian thựcCó (qua Webhook)Không (chỉ polling)
Tin nhắn lịch sửKhôngCó (có thể truy vấn lịch sử)
Độ phức tạp thiết lậpChỉ manifest ứng dụngCần sự chấp thuận của quản trị viên + luồng token
Hoạt động khi ngoại tuyếnKhông (phải đang chạy)Có (truy vấn bất cứ lúc nào)
Tóm lại: RSC dùng để lắng nghe thời gian thực; Graph API dùng để truy cập lịch sử. Để bắt kịp các tin nhắn bị bỏ lỡ khi ngoại tuyến, bạn cần Graph API với ChannelMessage.Read.All (cần sự chấp thuận của quản trị viên).

Phương tiện + lịch sử có Graph (bắt buộc cho kênh)

Nếu bạn cần hình ảnh/tệp trong kênh hoặc muốn lấy lịch sử tin nhắn, bạn phải bật quyền Microsoft Graph và cấp sự chấp thuận của quản trị viên.
  1. Trong Đăng ký ứng dụng Entra ID (Azure AD), thêm quyền Ứng dụng Microsoft Graph:
    • ChannelMessage.Read.All (tệp đính kèm kênh + lịch sử)
    • Chat.Read.All hoặc ChatMessage.Read.All (trò chuyện nhóm)
  2. Cấp sự chấp thuận của quản trị viên cho tenant.
  3. Tăng phiên bản manifest của ứng dụng Teams, tải lên lại và cài đặt lại ứng dụng trong Teams.
  4. Thoát hoàn toàn rồi mở lại Teams để xóa siêu dữ liệu ứng dụng được lưu trong bộ nhớ đệm.
Quyền bổ sung cho lượt nhắc người dùng: @mention người dùng hoạt động ngay mặc định đối với người dùng trong cuộc trò chuyện. Tuy nhiên, nếu bạn muốn tìm kiếm và nhắc động những người dùng không có trong cuộc trò chuyện hiện tại, hãy thêm quyền User.Read.All (Ứng dụng) và cấp sự chấp thuận của quản trị viên.

Hạn chế đã biết

Hết thời gian chờ Webhook

Teams gửi tin nhắn qua Webhook HTTP. Nếu xử lý mất quá nhiều thời gian (ví dụ: phản hồi LLM chậm), bạn có thể thấy:
  • Gateway hết thời gian chờ
  • Teams thử gửi lại tin nhắn (gây trùng lặp)
  • Phản hồi bị bỏ
OpenClaw xử lý việc này bằng cách trả về nhanh và chủ động gửi phản hồi, nhưng các phản hồi rất chậm vẫn có thể gây ra sự cố.

Định dạng

Markdown của Teams hạn chế hơn Slack hoặc Discord:
  • Định dạng cơ bản hoạt động: in đậm, in nghiêng, code, liên kết
  • Markdown phức tạp (bảng, danh sách lồng nhau) có thể không hiển thị đúng
  • Adaptive Cards được hỗ trợ cho thăm dò ý kiến và các lượt gửi trình bày ngữ nghĩa (xem bên dưới)

Cấu hình

Các thiết lập chính (xem /gateway/configuration để biết các mẫu kênh dùng chung):
  • channels.msteams.enabled: bật/tắt kênh.
  • channels.msteams.appId, channels.msteams.appPassword, channels.msteams.tenantId: thông tin xác thực bot.
  • channels.msteams.webhook.port (mặc định 3978)
  • channels.msteams.webhook.path (mặc định /api/messages)
  • channels.msteams.dmPolicy: pairing | allowlist | open | disabled (mặc định: pairing)
  • channels.msteams.allowFrom: danh sách cho phép DM (khuyến nghị dùng ID đối tượng AAD). Trình hướng dẫn sẽ phân giải tên thành ID trong quá trình thiết lập khi có quyền truy cập Graph.
  • channels.msteams.dangerouslyAllowNameMatching: công tắc phá kính để bật lại việc khớp UPN/tên hiển thị có thể thay đổi và định tuyến trực tiếp theo tên nhóm/kênh.
  • channels.msteams.textChunkLimit: kích thước đoạn văn bản gửi đi.
  • channels.msteams.chunkMode: length (mặc định) hoặc newline để tách theo dòng trống (ranh giới đoạn văn) trước khi tách theo độ dài.
  • channels.msteams.mediaAllowHosts: danh sách cho phép đối với máy chủ tệp đính kèm đến (mặc định là các miền Microsoft/Teams).
  • channels.msteams.mediaAuthAllowHosts: danh sách cho phép để đính kèm header Authorization khi thử lại media (mặc định là các máy chủ Graph + Bot Framework).
  • channels.msteams.requireMention: yêu cầu @mention trong kênh/nhóm (mặc định là true).
  • channels.msteams.replyStyle: thread | top-level (xem Kiểu trả lời).
  • channels.msteams.teams.<teamId>.replyStyle: ghi đè theo từng nhóm.
  • channels.msteams.teams.<teamId>.requireMention: ghi đè theo từng nhóm.
  • channels.msteams.teams.<teamId>.tools: các ghi đè chính sách công cụ mặc định theo từng nhóm (allow/deny/alsoAllow) được dùng khi thiếu ghi đè theo kênh.
  • channels.msteams.teams.<teamId>.toolsBySender: các ghi đè chính sách công cụ mặc định theo từng nhóm, theo từng người gửi (hỗ trợ ký tự đại diện "*").
  • channels.msteams.teams.<teamId>.channels.<conversationId>.replyStyle: ghi đè theo từng kênh.
  • channels.msteams.teams.<teamId>.channels.<conversationId>.requireMention: ghi đè theo từng kênh.
  • channels.msteams.teams.<teamId>.channels.<conversationId>.tools: các ghi đè chính sách công cụ theo từng kênh (allow/deny/alsoAllow).
  • channels.msteams.teams.<teamId>.channels.<conversationId>.toolsBySender: các ghi đè chính sách công cụ theo từng kênh, theo từng người gửi (hỗ trợ ký tự đại diện "*").
  • Khóa toolsBySender nên dùng tiền tố rõ ràng: channel:, id:, e164:, username:, name: (các khóa cũ không có tiền tố vẫn chỉ ánh xạ tới id:).
  • channels.msteams.actions.memberInfo: bật hoặc tắt hành động thông tin thành viên được hỗ trợ bởi Graph (mặc định: bật khi có thông tin xác thực Graph).
  • channels.msteams.authType: loại xác thực - "secret" (mặc định) hoặc "federated".
  • channels.msteams.certificatePath: đường dẫn tới tệp chứng chỉ PEM (xác thực liên kết + chứng chỉ).
  • channels.msteams.certificateThumbprint: dấu vân tay chứng chỉ (tùy chọn, không bắt buộc để xác thực).
  • channels.msteams.useManagedIdentity: bật xác thực managed identity (chế độ liên kết).
  • channels.msteams.managedIdentityClientId: ID máy khách cho managed identity do người dùng gán.
  • channels.msteams.sharePointSiteId: ID site SharePoint để tải tệp lên trong chat nhóm/kênh (xem Gửi tệp trong chat nhóm).

Định tuyến và phiên

  • Khóa phiên tuân theo định dạng agent tiêu chuẩn (xem /concepts/session):
    • Tin nhắn trực tiếp dùng chung phiên chính (agent:<agentId>:<mainKey>).
    • Tin nhắn kênh/nhóm dùng conversation id:
      • agent:<agentId>:msteams:channel:<conversationId>
      • agent:<agentId>:msteams:group:<conversationId>

Kiểu trả lời: luồng so với bài đăng

Teams gần đây đã giới thiệu hai kiểu giao diện kênh trên cùng mô hình dữ liệu nền:
KiểuMô tảreplyStyle khuyến nghị
Bài đăng (cổ điển)Tin nhắn xuất hiện dưới dạng thẻ với trả lời dạng luồng bên dướithread (mặc định)
Luồng (giống Slack)Tin nhắn chảy tuyến tính, giống Slack hơntop-level
Vấn đề: Teams API không cho biết một kênh đang dùng kiểu giao diện nào. Nếu bạn dùng sai replyStyle:
  • thread trong kênh kiểu Luồng → trả lời xuất hiện lồng nhau một cách gượng ép
  • top-level trong kênh kiểu Bài đăng → trả lời xuất hiện thành các bài đăng cấp cao nhất riêng biệt thay vì trong luồng
Giải pháp: Cấu hình replyStyle theo từng kênh dựa trên cách kênh được thiết lập:
{
  channels: {
    msteams: {
      replyStyle: "thread",
      teams: {
        "19:abc...@thread.tacv2": {
          channels: {
            "19:xyz...@thread.tacv2": {
              replyStyle: "top-level",
            },
          },
        },
      },
    },
  },
}

Thứ tự ưu tiên phân giải

Khi bot gửi trả lời vào một kênh, replyStyle được phân giải từ ghi đè cụ thể nhất xuống đến mặc định. Giá trị đầu tiên không phải undefined sẽ thắng:
  1. Theo từng kênhchannels.msteams.teams.<teamId>.channels.<conversationId>.replyStyle
  2. Theo từng nhómchannels.msteams.teams.<teamId>.replyStyle
  3. Toàn cụcchannels.msteams.replyStyle
  4. Mặc định ngầm định — suy ra từ requireMention:
    • requireMention: truethread
    • requireMention: falsetop-level
Nếu bạn đặt requireMention: false ở phạm vi toàn cục mà không có replyStyle rõ ràng, các lượt nhắc đến trong kênh kiểu Bài đăng sẽ xuất hiện dưới dạng bài đăng cấp cao nhất ngay cả khi tin nhắn đến là trả lời trong luồng. Ghim replyStyle: "thread" ở cấp toàn cục, nhóm hoặc kênh để tránh bất ngờ.

Bảo toàn ngữ cảnh luồng

Khi replyStyle: "thread" có hiệu lực và bot được @mentioned từ bên trong một luồng kênh, OpenClaw sẽ gắn lại gốc luồng ban đầu vào tham chiếu cuộc trò chuyện gửi đi (19:…@thread.tacv2;messageid=<root>) để trả lời được đưa vào cùng luồng. Điều này áp dụng cho cả lượt gửi trực tiếp (trong lượt) và lượt gửi chủ động được thực hiện sau khi ngữ cảnh lượt Bot Framework đã hết hạn (ví dụ: agent chạy lâu, trả lời lệnh gọi công cụ được xếp hàng qua mcp__openclaw__message). Gốc luồng được lấy từ threadId đã lưu trên tham chiếu cuộc trò chuyện. Các tham chiếu cũ được lưu trước khi có threadId sẽ quay về dùng activityId (bất kỳ hoạt động đến nào gần nhất đã khởi tạo cuộc trò chuyện), vì vậy các triển khai hiện có tiếp tục hoạt động mà không cần khởi tạo lại. Khi replyStyle: "top-level" có hiệu lực, các tin nhắn đến từ luồng kênh sẽ được trả lời có chủ ý dưới dạng bài đăng cấp cao nhất mới — không gắn hậu tố luồng. Đây là hành vi đúng cho các kênh kiểu Luồng; nếu bạn thấy bài đăng cấp cao nhất ở nơi bạn mong đợi trả lời dạng luồng, replyStyle của bạn đang được đặt sai cho kênh đó.

Tệp đính kèm và hình ảnh

Giới hạn hiện tại:
  • DM: Hình ảnh và tệp đính kèm hoạt động qua Teams bot file APIs.
  • Kênh/nhóm: Tệp đính kèm nằm trong bộ nhớ M365 (SharePoint/OneDrive). Payload Webhook chỉ bao gồm một HTML stub, không phải byte tệp thực tế. Cần quyền Graph API để tải xuống tệp đính kèm trong kênh.
  • Với các lượt gửi rõ ràng ưu tiên tệp, dùng action=upload-file với media / filePath / path; message tùy chọn trở thành văn bản/bình luận đi kèm, và filename ghi đè tên đã tải lên.
Nếu không có quyền Graph, tin nhắn kênh có hình ảnh sẽ được nhận dưới dạng chỉ văn bản (bot không thể truy cập nội dung hình ảnh). Theo mặc định, OpenClaw chỉ tải media xuống từ tên máy chủ Microsoft/Teams. Ghi đè bằng channels.msteams.mediaAllowHosts (dùng ["*"] để cho phép mọi máy chủ). Header Authorization chỉ được đính kèm cho các máy chủ trong channels.msteams.mediaAuthAllowHosts (mặc định là máy chủ Graph + Bot Framework). Giữ danh sách này nghiêm ngặt (tránh hậu tố đa tenant).

Gửi tệp trong chat nhóm

Bot có thể gửi tệp trong DM bằng luồng FileConsentCard (tích hợp sẵn). Tuy nhiên, gửi tệp trong chat nhóm/kênh cần thiết lập bổ sung:
Ngữ cảnhCách gửi tệpThiết lập cần thiết
DMFileConsentCard → người dùng chấp nhận → bot tải lênHoạt động ngay
Chat nhóm/kênhTải lên SharePoint → chia sẻ liên kếtCần sharePointSiteId + quyền Graph
Hình ảnh (mọi ngữ cảnh)Mã hóa inline bằng Base64Hoạt động ngay

Vì sao chat nhóm cần SharePoint

Bot không có ổ OneDrive cá nhân (endpoint Graph API /me/drive không hoạt động với danh tính ứng dụng). Để gửi tệp trong chat nhóm/kênh, bot tải lên một site SharePoint và tạo liên kết chia sẻ.

Thiết lập

  1. Thêm quyền Graph API trong Entra ID (Azure AD) → App Registration:
    • Sites.ReadWrite.All (Application) - tải tệp lên SharePoint
    • Chat.Read.All (Application) - tùy chọn, bật liên kết chia sẻ theo từng người dùng
  2. Cấp sự đồng ý của quản trị viên cho tenant.
  3. Lấy ID site SharePoint của bạn:
    # Via Graph Explorer or curl with a valid token:
    curl -H "Authorization: Bearer $TOKEN" \
      "https://graph.microsoft.com/v1.0/sites/{hostname}:/{site-path}"
    
    # Example: for a site at "contoso.sharepoint.com/sites/BotFiles"
    curl -H "Authorization: Bearer $TOKEN" \
      "https://graph.microsoft.com/v1.0/sites/contoso.sharepoint.com:/sites/BotFiles"
    
    # Response includes: "id": "contoso.sharepoint.com,guid1,guid2"
    
  4. Cấu hình OpenClaw:
    {
      channels: {
        msteams: {
          // ... other config ...
          sharePointSiteId: "contoso.sharepoint.com,guid1,guid2",
        },
      },
    }
    

Hành vi chia sẻ

QuyềnHành vi chia sẻ
Chỉ Sites.ReadWrite.AllLiên kết chia sẻ toàn tổ chức (bất kỳ ai trong tổ chức đều có thể truy cập)
Sites.ReadWrite.All + Chat.Read.AllLiên kết chia sẻ theo từng người dùng (chỉ thành viên chat có thể truy cập)
Chia sẻ theo từng người dùng an toàn hơn vì chỉ những người tham gia chat mới có thể truy cập tệp. Nếu thiếu quyền Chat.Read.All, bot quay về chia sẻ toàn tổ chức.

Hành vi dự phòng

Kịch bảnKết quả
Chat nhóm + tệp + đã cấu hình sharePointSiteIdTải lên SharePoint, gửi liên kết chia sẻ
Chat nhóm + tệp + không có sharePointSiteIdThử tải lên OneDrive (có thể thất bại), chỉ gửi văn bản
Chat cá nhân + tệpLuồng FileConsentCard (hoạt động không cần SharePoint)
Mọi ngữ cảnh + hình ảnhMã hóa inline bằng Base64 (hoạt động không cần SharePoint)

Vị trí lưu trữ tệp

Các tệp đã tải lên được lưu trong thư mục /OpenClawShared/ trong thư viện tài liệu mặc định của site SharePoint đã cấu hình.

Thăm dò ý kiến (Adaptive Cards)

OpenClaw gửi thăm dò ý kiến Teams dưới dạng Adaptive Cards (không có Teams poll API gốc).
  • CLI: openclaw message poll --channel msteams --target conversation:<id> ...
  • Phiếu bầu được Gateway ghi lại trong ~/.openclaw/msteams-polls.json.
  • Gateway phải duy trì trực tuyến để ghi lại phiếu bầu.
  • Hiện các cuộc thăm dò chưa tự động đăng tóm tắt kết quả (kiểm tra tệp lưu trữ nếu cần).

Thẻ trình bày

Gửi payload trình bày ngữ nghĩa đến người dùng hoặc cuộc trò chuyện Teams bằng công cụ message hoặc CLI. OpenClaw hiển thị chúng dưới dạng Teams Adaptive Cards từ hợp đồng trình bày chung. Tham số presentation chấp nhận các khối ngữ nghĩa. Khi cung cấp presentation, văn bản tin nhắn là tùy chọn. Công cụ agent:
{
  action: "send",
  channel: "msteams",
  target: "user:<id>",
  presentation: {
    title: "Hello",
    blocks: [{ type: "text", text: "Hello!" }],
  },
}
CLI:
openclaw message send --channel msteams \
  --target "conversation:19:abc...@thread.tacv2" \
  --presentation '{"title":"Hello","blocks":[{"type":"text","text":"Hello!"}]}'
Để biết chi tiết định dạng đích, xem Định dạng đích bên dưới.

Định dạng đích

Đích MSTeams dùng tiền tố để phân biệt giữa người dùng và cuộc trò chuyện:
Loại đíchĐịnh dạngVí dụ
Người dùng (theo ID)user:<aad-object-id>user:40a1a0ed-4ff2-4164-a219-55518990c197
Người dùng (theo tên)user:<display-name>user:John Smith (yêu cầu Graph API)
Nhóm/kênhconversation:<conversation-id>conversation:19:abc123...@thread.tacv2
Nhóm/kênh (thô)<conversation-id>19:abc123...@thread.tacv2 (nếu chứa @thread)
Ví dụ CLI:
# Send to a user by ID
openclaw message send --channel msteams --target "user:40a1a0ed-..." --message "Hello"

# Send to a user by display name (triggers Graph API lookup)
openclaw message send --channel msteams --target "user:John Smith" --message "Hello"

# Send to a group chat or channel
openclaw message send --channel msteams --target "conversation:19:abc...@thread.tacv2" --message "Hello"

# Send a presentation card to a conversation
openclaw message send --channel msteams --target "conversation:19:abc...@thread.tacv2" \
  --presentation '{"title":"Hello","blocks":[{"type":"text","text":"Hello"}]}'
Ví dụ công cụ agent:
{
  action: "send",
  channel: "msteams",
  target: "user:John Smith",
  message: "Hello!",
}
{
  action: "send",
  channel: "msteams",
  target: "conversation:19:abc...@thread.tacv2",
  presentation: {
    title: "Hello",
    blocks: [{ type: "text", text: "Hello" }],
  },
}
Nếu không có tiền tố user:, tên mặc định được phân giải theo nhóm hoặc team. Luôn dùng user: khi nhắm đến người theo tên hiển thị.

Nhắn tin chủ động

  • Tin nhắn chủ động chỉ có thể thực hiện sau khi người dùng đã tương tác, vì tại thời điểm đó chúng tôi lưu trữ tham chiếu cuộc trò chuyện.
  • Xem /gateway/configuration để biết dmPolicy và kiểm soát bằng danh sách cho phép.

ID Team và Channel (lỗi thường gặp)

Tham số truy vấn groupId trong URL Teams KHÔNG phải là ID team dùng cho cấu hình. Thay vào đó, hãy trích xuất ID từ đường dẫn URL: URL Team:
https://teams.microsoft.com/l/team/19%3ABk4j...%40thread.tacv2/conversations?groupId=...
                                    └────────────────────────────┘
                                    Team conversation ID (URL-decode this)
URL Channel:
https://teams.microsoft.com/l/channel/19%3A15bc...%40thread.tacv2/ChannelName?groupId=...
                                      └─────────────────────────┘
                                      Channel ID (URL-decode this)
Đối với cấu hình:
  • Khóa team = đoạn đường dẫn sau /team/ (đã giải mã URL, ví dụ 19:Bk4j...@thread.tacv2; các tenant cũ hơn có thể hiển thị @thread.skype, cũng hợp lệ)
  • Khóa channel = đoạn đường dẫn sau /channel/ (đã giải mã URL)
  • Bỏ qua tham số truy vấn groupId cho định tuyến OpenClaw. Đây là ID nhóm Microsoft Entra, không phải ID cuộc trò chuyện Bot Framework được dùng trong hoạt động Teams đến.

Kênh riêng tư

Bot có hỗ trợ hạn chế trong các kênh riêng tư:
Tính năngKênh tiêu chuẩnKênh riêng tư
Cài đặt botHạn chế
Tin nhắn thời gian thực (webhook)Có thể không hoạt động
Quyền RSCCó thể hoạt động khác đi
@mentionsNếu bot có thể truy cập
Lịch sử Graph APICó (với quyền phù hợp)
Cách xử lý nếu kênh riêng tư không hoạt động:
  1. Dùng kênh tiêu chuẩn cho tương tác bot
  2. Dùng DM - người dùng luôn có thể nhắn trực tiếp cho bot
  3. Dùng Graph API để truy cập lịch sử (yêu cầu ChannelMessage.Read.All)

Khắc phục sự cố

Sự cố thường gặp

  • Hình ảnh không hiển thị trong kênh: Thiếu quyền Graph hoặc sự đồng ý của quản trị viên. Cài đặt lại ứng dụng Teams và thoát hoàn toàn/mở lại Teams.
  • Không có phản hồi trong kênh: mặc định yêu cầu nhắc đến; đặt channels.msteams.requireMention=false hoặc cấu hình theo từng team/channel.
  • Không khớp phiên bản (Teams vẫn hiển thị manifest cũ): gỡ bỏ + thêm lại ứng dụng và thoát hoàn toàn Teams để làm mới.
  • 401 Unauthorized từ webhook: Dự kiến xảy ra khi kiểm thử thủ công không có Azure JWT - nghĩa là endpoint có thể truy cập được nhưng xác thực thất bại. Dùng Azure Web Chat để kiểm thử đúng cách.

Lỗi tải manifest lên

  • “Icon file cannot be empty”: Manifest tham chiếu đến các tệp biểu tượng có kích thước 0 byte. Tạo biểu tượng PNG hợp lệ (32x32 cho outline.png, 192x192 cho color.png).
  • “webApplicationInfo.Id already in use”: Ứng dụng vẫn được cài đặt trong team/chat khác. Tìm và gỡ cài đặt trước, hoặc chờ 5-10 phút để thay đổi lan truyền.
  • “Something went wrong” khi tải lên: Thay vào đó hãy tải lên qua https://admin.teams.microsoft.com, mở DevTools của trình duyệt (F12) → tab Network, rồi kiểm tra phần thân phản hồi để xem lỗi thực tế.
  • Sideload thất bại: Thử “Upload an app to your org’s app catalog” thay vì “Upload a custom app” - cách này thường vượt qua được các hạn chế sideload.

Quyền RSC không hoạt động

  1. Xác minh webApplicationInfo.id khớp chính xác với App ID của bot
  2. Tải lại ứng dụng lên và cài đặt lại trong team/chat
  3. Kiểm tra xem quản trị viên tổ chức của bạn có chặn quyền RSC không
  4. Xác nhận bạn đang dùng đúng phạm vi: ChannelMessage.Read.Group cho teams, ChatMessage.Read.Chat cho group chats

Tham khảo

Liên quan