Documentation Index
Fetch the complete documentation index at: https://docs.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). Cuộc thăm dò được gửi qua Adaptive Cards. Hành động tin nhắn cung cấp upload-file rõ ràng 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, nên 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 một gói npm hiện tại khi gói đó được phát hành:
openclaw plugins install @openclaw/msteams
Nếu npm báo gói do OpenClaw sở hữu là đã ngừng khuyến nghị, hãy dùng bản dựng OpenClaw
đóng gói hiện tại hoặc đường dẫn checkout cục bộ cho đến khi một gói npm mới hơn được
phát hành.
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 trong 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 xem trước. 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 thực hiện (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 đi vào vẫn được Teams SDK tự động xác thực.
Phương án thay thế: ngrok http 3978 hoặc tailscale funnel 3978 (nhưng các công cụ này có thể thay đổi URL trong 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 đăng ký Azure)
Đầu ra sẽ hiển thị CLIENT_ID, CLIENT_SECRET, TENANT_ID và một Teams App ID — hãy ghi lại các giá trị này cho các bước tiếp theo. Lệnh cũng đề xuất 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 các 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 secrets.
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 kiểm soát bằng mention).
Mục tiêu
- Trao đổi với OpenClaw qua DM Teams, cuộc trò chuyện nhóm hoặc kênh.
- Giữ định tuyến xác định: phản hồ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 mention 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 bản 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 bị bỏ qua cho đến khi được phê duyệt.
channels.msteams.allowFrom nên dùng các ID đối tượng AAD ổn định.
- 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 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 (vẫn được kiểm soát bằng mention theo mặc định).
- Để không cho phép kênh nào, đặt
channels.msteams.groupPolicy: "disabled".
Ví dụ:
{
channels: {
msteams: {
groupPolicy: "allowlist",
groupAllowFrom: ["user@org.com"],
},
},
}
Teams + allowlist kênh
- Giới hạn phạm vi phản hồi nhóm/kênh bằng cách liệt kê teams và kênh trong
channels.msteams.teams.
- Khóa nên dùng team IDs và channel conversation IDs ổn định.
- Khi
groupPolicy="allowlist" và có allowlist teams, chỉ các teams/kênh được liệt kê mới được chấp nhận (được kiểm soát bằng mention).
- Trình hướng dẫn cấu hình chấp nhận các mục
Team/Channel và lưu chúng cho bạn.
- Khi khởi động, OpenClaw phân giải tên team/channel và tên allowlist người dùng thành ID (khi quyền Graph cho phép)
và ghi log ánh xạ; tên team/channel chưa phân giải được giữ nguyên như đã nhập nhưng mặc định bị bỏ qua trong đị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)
Được thêm vào 2026.4.11
Đối với triển khai production, OpenClaw hỗ trợ xác thực liên kết như một phương án thay thế an toàn hơn cho client secrets. Có hai phương pháp:
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:
- Tạo hoặc lấy một chứng chỉ (định dạng PEM với khóa riêng).
- Trong Entra ID → App Registration → Certificates & secrets → Certificates → 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 các triển khai trên hạ tầng Azure (AKS, App Service, Azure VMs) nơi có managed identity.
Cách hoạt động:
- Pod/VM của bot có managed identity (system-assigned hoặc user-assigned).
- Một federated identity credential liên kết managed identity với app registration Entra ID.
- Khi chạy, OpenClaw dùng
@azure/identity để lấy token từ endpoint Azure IMDS (169.254.169.254).
- Token được truyền tới Teams SDK để xác thực bot.
Điều kiện tiên quyết:
- Hạ tầng Azure đã bật managed identity (AKS workload identity, App Service, VM)
- Đã tạo federated identity credential trên app registration Entra ID
- Pod/VM có quyền truy cập mạng tới IMDS (
169.254.169.254:80)
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 danh tính do người dùng gán)
Thiết lập AKS Workload Identity
Đối với các triển khai AKS dùng workload identity:
-
Bật workload identity trên cụm AKS của bạn.
-
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"]
}'
-
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>"
-
Gắn nhãn pod để chèn workload identity:
metadata:
labels:
azure.workload.identity/use: "true"
-
Đả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ức | Cấu hình | Ưu điểm | Nhược điểm |
|---|
| Client secret | appPassword | Thiết lập đơn giản | Cần xoay vòng bí mật, kém an toàn hơn |
| Chứng chỉ | authType: "federated" + certificatePath | Không có bí mật dùng chung qua mạng | Tốn công quản lý chứng chỉ |
| Managed Identity | authType: "federated" + useManagedIdentity | Không mật khẩu, không cần quản lý bí mật | Cần hạ tầng Azure |
Hành vi mặc định: Khi không đặt authType, OpenClaw mặc định dùng xác thực client secret. 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ộ (tạo đườ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:
# Thiết lập một lần:
devtunnel create my-openclaw-bot --allow-anonymous
devtunnel port create my-openclaw-bot -p 3978 --protocol auto
# Mỗi phiên phát triển:
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 endpoint:
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ử:
- Cài đặt ứng dụng Teams (dùng liên kết cài đặt từ
teams app get <id> --install-link)
- Tìm bot trong Teams và gửi DM
- Kiểm tra nhật ký gateway để xem hoạt động đến
Biến môi trường
Thay vào đó, mọi 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 (federated + certificate)
MSTEAMS_CERTIFICATE_THUMBPRINT (tùy chọn, không bắt buộc để xác thực)
MSTEAMS_USE_MANAGED_IDENTITY (federated + managed identity)
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 được Graph hỗ trợ 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 liên nhóm: quyền Ứng dụng Graph
User.Read.All với sự đồng ý 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 gói vào prompt.
- Dự phòng về
messages.groupChat.historyLimit. Đặt 0 để tắt (mặc định 50).
- Lịch sử luồng đã lấy được lọc theo danh sách cho phép người gửi (
allowFrom / groupAllowFrom), vì vậy việc gieo ngữ cảnh luồng chỉ bao gồm tin nhắn từ những người gửi được phép.
- Ngữ cảnh tệp đính kèm được trích dẫn (
ReplyTo* được suy ra từ HTML trả lời của Teams) hiện được truyền qua như đã nhận.
- 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 đường dẫn ngữ cảnh bổ sung cụ thể mới đượ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 từng người dùng: channels.msteams.dms["<user_id>"].historyLimit.
Quyền Teams RSC hiện tại (manifest)
Đây là các quyền resourceSpecific hiện có trong manifest ứng dụng Teams của chúng ta. Chúng chỉ áp dụng bên trong nhóm/cuộc 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 (Application) - nhận mọi tin nhắn kênh mà không cần @mention
ChannelMessage.Send.Group (Application)
Member.Read.Group (Application)
Owner.Read.Group (Application)
ChannelSettings.Read.Group (Application)
TeamMember.Read.Group (Application)
TeamSettings.Read.Group (Application)
Đối với cuộc trò chuyện nhóm:
ChatMessage.Read.Chat (Application) - nhận mọi 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 (đã biên tập)
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 Azure Bot App ID.
webApplicationInfo.id phải khớp với Azure Bot App ID.
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 quyền đọc/gửi kênh nếu bạn muốn có 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):
# Tải xuống, chỉnh sửa và tải lại manifest
teams app manifest download <teamsAppId> manifest.json
# Chỉnh sửa manifest.json cục bộ...
teams app manifest upload manifest.json <teamsAppId>
# Phiên bản được tự động tăng nếu nội dung thay đổi
Sau khi cập nhật, hãy cài đặt lại ứng dụng trong từng nhóm để các quyền mới có hiệu lực, và thoát hoàn toàn rồi khởi chạy lại Teams (không chỉ đóng cửa sổ) để xóa siêu dữ liệu ứng dụng đã 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 cá nhân (DM).
Không hoạt động:
- Nội dung hình ảnh hoặc tệp trong kênh/nhóm (payload chỉ bao gồm stub HTML).
- Tải xuống tệp đính kèm được lưu trữ 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 trữ trong SharePoint/OneDrive.
- Đọc lịch sử tin nhắn kênh/cuộc trò chuyện qua Graph.
RSC so với Graph API
| Khả năng | Quyền RSC | Graph API |
|---|
| Tin nhắn thời gian thực | Có (qua Webhook) | Không (chỉ polling) |
| Tin nhắn lịch sử | Không | Có (có thể truy vấn lịch sử) |
| Độ phức tạp thiết lập | Chỉ manifest ứng dụng | Cần sự đồng ý của quản trị viên + luồng token |
| Hoạt động ngoại tuyến | Khô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 theo 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ự đồng ý 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ự đồng ý của quản trị viên.
- Trong Entra ID (Azure AD) App Registration, 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 (cuộc trò chuyện nhóm)
- Cấp sự đồng ý của quản trị viên cho tenant.
- Tăng phiên bản manifest của ứng dụng Teams, tải lại lên và cài đặt lại ứng dụng trong Teams.
- Thoát hoàn toàn rồi khởi chạy lại Teams để xóa siêu dữ liệu ứng dụng đã lưu trong bộ nhớ đệm.
Quyền bổ sung cho đề cập người dùng: @mention người dùng hoạt động sẵn đố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 động và đề cập 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 (Application) và cấp sự đồng ý của quản trị viên.
Giới hạn đã biết
Hết thời gian chờ Webhook
Teams chuyển phát tin nhắn qua Webhook HTTP. Nếu quá trình xử lý mất quá lâu (ví dụ: phản hồi LLM chậm), bạn có thể thấy:
- Gateway hết thời gian chờ
- Teams thử lại tin nhắn (gây trùng lặp)
- Phản hồi bị bỏ qua
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 sự cố.
Định dạng
Markdown của Teams bị giới hạn hơn Slack hoặc Discord:
- Định dạng cơ bản hoạt động: đậm, 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 khảo sát và các lần 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 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 team/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 chia đoạn theo độ dài.
channels.msteams.mediaAllowHosts: danh sách cho phép các 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 gắn header Authorization khi thử lại media (mặc định là Graph + máy chủ Bot Framework).
channels.msteams.requireMention: yêu cầu @mention trong kênh/nhóm (mặc định true).
channels.msteams.replyStyle: thread | top-level (xem Kiểu trả lời).
channels.msteams.teams.<teamId>.replyStyle: ghi đè theo từng team.
channels.msteams.teams.<teamId>.requireMention: ghi đè theo từng team.
channels.msteams.teams.<teamId>.tools: ghi đè chính sách công cụ mặc định theo từng team (allow/deny/alsoAllow) được dùng khi thiếu ghi đè theo kênh.
channels.msteams.teams.<teamId>.toolsBySender: ghi đè chính sách công cụ mặc định theo từng team và 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: ghi đè chính sách công cụ theo từng kênh (allow/deny/alsoAllow).
channels.msteams.teams.<teamId>.channels.<conversationId>.toolsBySender: ghi đè chính sách công cụ theo từng kênh và từng người gửi (hỗ trợ ký tự đại diện "*").
- Các khóa
toolsBySender nên dùng tiền tố rõ ràng:
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 dựa trên Graph (mặc định: bật khi có thông tin xác thực Graph).
channels.msteams.authType: kiểu 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 bằng danh tính được quản lý (chế độ liên kết).
channels.msteams.managedIdentityClientId: client ID cho danh tính được quản lý do người dùng gán.
channels.msteams.sharePointSiteId: ID site SharePoint để tải tệp lên trong cuộc trò chuyện nhóm/kênh (xem Gửi tệp trong cuộc trò chuyện nhóm).
Định tuyến & 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 UI kênh trên cùng một mô hình dữ liệu bên dưới:
| Kiểu | Mô 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 phản hồi theo luồng bên dưới | thread (mặc định) |
| Luồng (giống Slack) | Tin nhắn chảy tuyến tính, giống Slack hơn | top-level |
Vấn đề: Teams API không cho biết kênh đang dùng kiểu UI nào. Nếu bạn dùng sai replyStyle:
thread trong kênh kiểu Luồng → phản hồi xuất hiện lồng nhau một cách khó xử
top-level trong kênh kiểu Bài đăng → phản hồi xuất hiện như các bài đăng cấp cao riêng biệt thay vì nằm 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",
},
},
},
},
},
},
}
Tệp đính kèm & 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 API tệp bot của Teams.
- 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 stub HTML, 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 của kênh.
- Với các lần gửi ưu tiên tệp rõ ràng, dùng
action=upload-file với media / filePath / path; message tùy chọn sẽ 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 gắn 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ố nhiều tenant).
Gửi tệp trong cuộc trò chuyện 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 cuộc trò chuyện nhóm/kênh cần thiết lập bổ sung:
| Ngữ cảnh | Cách gửi tệp | Thiết lập cần thiết |
|---|
| DM | FileConsentCard → người dùng chấp nhận → bot tải lên | Hoạt động ngay |
| Cuộc trò chuyện nhóm/kênh | Tải lên SharePoint → chia sẻ liên kết | Cần sharePointSiteId + quyền Graph |
| Hình ảnh (mọi ngữ cảnh) | Nội tuyến mã hóa Base64 | Hoạt động ngay |
Vì sao cuộc trò chuyện 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 cuộc trò chuyện 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
-
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
-
Cấp quyền đồng ý của quản trị viên cho tenant.
-
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"
-
Cấu hình OpenClaw:
{
channels: {
msteams: {
// ... other config ...
sharePointSiteId: "contoso.sharepoint.com,guid1,guid2",
},
},
}
Hành vi chia sẻ
| Quyền | Hành vi chia sẻ |
|---|
Chỉ Sites.ReadWrite.All | Liê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.All | Liên kết chia sẻ theo từng người dùng (chỉ thành viên cuộc trò chuyện có thể truy cập) |
Chia sẻ theo từng người dùng an toàn hơn vì chỉ người tham gia cuộc trò chuyện mới có thể truy cập tệp. Nếu thiếu quyền Chat.Read.All, bot chuyển sang chia sẻ toàn tổ chức.
Hành vi dự phòng
| Tình huống | Kết quả |
|---|
Cuộc trò chuyện nhóm + tệp + đã cấu hình sharePointSiteId | Tải lên SharePoint, gửi liên kết chia sẻ |
Cuộc trò chuyện nhóm + tệp + không có sharePointSiteId | Thử tải lên OneDrive (có thể thất bại), chỉ gửi văn bản |
| Trò chuyện cá nhân + tệp | Luồng FileConsentCard (hoạt động không cần SharePoint) |
| Mọi ngữ cảnh + hình ảnh | Nội tuyến mã hóa 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.
Khảo sát (Adaptive Cards)
OpenClaw gửi khảo sát Teams dưới dạng Adaptive Cards (không có API khảo sát Teams 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 nhận phiếu bầu.
- Khảo sát 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 tới 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ạng | Ví 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 (cần Graph API) |
| Nhóm/kênh | conversation:<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ụ về công cụ của 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 sẽ mặc định được phân giải theo nhóm hoặc team. Luôn dùng user: khi nhắm mục tiêu đến người dùng bằng 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ì chúng tôi lưu tham chiếu cuộc trò chuyện tại thời điểm đó.
- Xem
/gateway/configuration để biết dmPolicy và kiểm soát bằng danh sách cho phép.
ID team và kênh (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=...
└────────────────────────────┘
ID team (giải mã URL phần này)
URL kênh:
https://teams.microsoft.com/l/channel/19%3A15bc...%40thread.tacv2/ChannelName?groupId=...
└─────────────────────────┘
ID kênh (giải mã URL phần này)
Đối với cấu hình:
- ID team = đoạn đường dẫn sau
/team/ (đã giải mã URL, ví dụ: 19:Bk4j...@thread.tacv2)
- ID kênh = đoạn đường dẫn sau
/channel/ (đã giải mã URL)
- Bỏ qua tham số truy vấn
groupId
Kênh riêng tư
Bot chỉ được hỗ trợ hạn chế trong các kênh riêng tư:
| Tính năng | Kênh tiêu chuẩn | Kênh riêng tư |
|---|
| Cài đặt bot | Có | Hạn chế |
| Tin nhắn thời gian thực (Webhook) | Có | Có thể không hoạt động |
| Quyền RSC | Có | Có thể hoạt động khác |
| @mentions | Có | Nếu bot có thể truy cập |
| Lịch sử Graph API | Có | Có (với quyền phù hợp) |
Cách xử lý nếu kênh riêng tư không hoạt động:
- Dùng kênh tiêu chuẩn cho tương tác với bot
- Dùng DM - người dùng luôn có thể nhắn trực tiếp cho bot
- Dùng Graph API để truy cập lịch sử (yêu cầu
ChannelMessage.Read.All)
Khắc phục sự cố
Vấn đề 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 rồi thoát hoàn toàn/mở lại Teams.
- Không có phản hồi trong kênh: theo mặc định cần có mention; đặt
channels.msteams.requireMention=false hoặc cấu hình theo từng team/kênh.
- 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 trong team/chat khác. Tìm và gỡ cài đặt trước, hoặc chờ 5-10 phút để thay đổi được 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
- Xác minh
webApplicationInfo.id khớp chính xác với App ID của bot
- Tải lại ứng dụng lên và cài đặt lại trong team/chat
- Kiểm tra xem quản trị viên tổ chức của bạn có chặn quyền RSC hay không
- Xác nhận bạn đang dùng đúng phạm vi:
ChannelMessage.Read.Group cho team, ChatMessage.Read.Chat cho chat nhóm
Tài liệu tham khảo
Liên quan