Microsoft Teams
Estado: se admiten texto + archivos adjuntos en MD; el envío de archivos a canales/grupos requiere sharePointSiteId + permisos de Graph (consulte Sending files in group chats). Las encuestas se envían mediante Adaptive Cards. Las acciones de mensaje exponen una upload-file explícita para envíos priorizando archivos.
Complemento incluido
Sección titulada «Complemento incluido»Microsoft Teams se distribuye como un complemento incluido en las versiones actuales de OpenClaw, por lo que no se requiere una instalación separada en la compilación empaquetada normal.
Si está en una versión anterior o en una instalación personalizada que excluye el complemento Teams incluido, instale el paquete npm directamente:
openclaw plugins install @openclaw/msteamsUse el paquete básico para seguir la etiqueta de lanzamiento oficial actual. Fije una versión exacta solo cuando necesite una instalación reproducible.
Pago local (cuando se ejecuta desde un repositorio git):
openclaw plugins install ./path/to/local/msteams-pluginDetalles: Plugins
Configuración rápida
Sección titulada «Configuración rápida»El @microsoft/teams.cli gestiona el registro del bot, la creación del manifiesto y la generación de credenciales en un solo comando.
1. Instalar e iniciar sesión
npm install -g @microsoft/teams.cli@previewteams loginteams status # verify you're logged in and see your tenant info2. Iniciar un túnel (Teams no puede acceder a localhost)
Instale y autentique la CLI de devtunnel si aún no lo ha hecho (guía de introducción).
# One-time setup (persistent URL across sessions):devtunnel create my-openclaw-bot --allow-anonymousdevtunnel 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/messagesAlternativas: ngrok http 3978 o tailscale funnel 3978 (pero es posible que estos cambien las URL en cada sesión).
3. Crear la aplicación
teams app create \ --name "OpenClaw" \ --endpoint "https://<your-tunnel-url>/api/messages"Este único comando:
- Crea una aplicación Entra ID (Azure AD)
- Genera un secreto de cliente
- Compila y carga un manifiesto de aplicación de Teams (con iconos)
- Registra el bot (administrado por Teams de forma predeterminada: no se necesita suscripción a Azure)
La salida mostrará CLIENT_ID, CLIENT_SECRET, TENANT_ID y un Teams App ID; anótelos para los siguientes pasos. También ofrece instalar la aplicación directamente en Teams.
4. Configurar OpenClaw utilizando las credenciales de la salida:
{ channels: { msteams: { enabled: true, appId: "<CLIENT_ID>", appPassword: "<CLIENT_SECRET>", tenantId: "<TENANT_ID>", webhook: { port: 3978, path: "/api/messages" }, }, },}O use variables de entorno directamente: MSTEAMS_APP_ID, MSTEAMS_APP_PASSWORD, MSTEAMS_TENANT_ID.
5. Instalar la aplicación en Teams
teams app create le pedirá que instale la aplicación: seleccione “Instalar en Teams”. Si lo omitió, puede obtener el enlace más tarde:
teams app get <teamsAppId> --install-link6. Verificar que todo funciona
teams app doctor <teamsAppId>Esto ejecuta diagnósticos en el registro del bot, la configuración de la aplicación AAD, la validez del manifiesto y la configuración de SSO.
Para despliegues en producción, considere el uso de autenticación federada (certificado o identidad administrada) en lugar de secretos de cliente.
Objetivos
Sección titulada «Objetivos»- Hablar con OpenClaw mediante MDs de Teams, chats de grupo o canales.
- Mantener el enrutamiento determinista: las respuestas siempre vuelven al canal por el que llegaron.
- De forma predeterminada, adoptar un comportamiento de canal seguro (se requieren menciones a menos que se configure lo contrario).
Escrituras de configuración
Sección titulada «Escrituras de configuración»De forma predeterminada, se permite a Microsoft Teams escribir actualizaciones de configuración activadas por /config set|unset (requiere commands.config: true).
Desactivar con:
{ channels: { msteams: { configWrites: false } },}Control de acceso (MDs + grupos)
Sección titulada «Control de acceso (MDs + grupos)»Acceso a MDs
- Predeterminado:
channels.msteams.dmPolicy = "pairing". Los remitentes desconocidos se ignoran hasta que se aprueben. channels.msteams.allowFromdebe usar identificadores de objetos de AAD estables o grupos de acceso de remitentes estáticos comoaccessGroup:core-team.- No se base en la coincidencia de UPN/nombre para mostrar para las listas de permitidos: pueden cambiar. OpenClaw desactiva la coincidencia de nombres directa de forma predeterminada; actívelo explícitamente con
channels.msteams.dangerouslyAllowNameMatching: true. - El asistente puede resolver nombres a identificadores mediante Microsoft Graph cuando las credenciales lo permiten.
Acceso a grupos
- Predeterminado:
channels.msteams.groupPolicy = "allowlist"(bloqueado a menos que añadagroupAllowFrom). Usechannels.defaults.groupPolicypara anular el valor predeterminado cuando no esté establecido. channels.msteams.groupAllowFromcontrola qué remitentes o grupos de acceso de remitente estático pueden activarse en chats grupales/canales (recurre achannels.msteams.allowFrom).- Establezca
groupPolicy: "open"para permitir cualquier miembro (aún limitado por mención de forma predeterminada). - Para no permitir ningún canal, establezca
channels.msteams.groupPolicy: "disabled".
Ejemplo:
{ channels: { msteams: { groupPolicy: "allowlist", groupAllowFrom: ["00000000-0000-0000-0000-000000000000", "accessGroup:core-team"], }, },}Lista de permitidos de Teams + canales
- Alcance las respuestas de grupos/canales listando equipos y canales en
channels.msteams.teams. - Las claves deben usar ID de conversación de Teams estables de los enlaces de Teams, no nombres para mostrar mutables.
- Cuando
groupPolicy="allowlist"y hay una lista de permitidos de equipos, solo se aceptan los equipos/canales listados (restringido por mención). - El asistente de configuración acepta entradas
Team/Channely las almacena por usted. - Al iniciarse, OpenClaw resuelve los nombres de la lista de permitidos de equipos/canales y usuarios a ID (cuando los permisos de Graph lo permiten)
y registra la asignación; los nombres de equipos/canales no resueltos se mantienen tal como se escribieron pero se ignoran para el enrutamiento de manera predeterminada a menos que
channels.msteams.dangerouslyAllowNameMatching: trueesté habilitado.
Ejemplo:
{ channels: { msteams: { groupPolicy: "allowlist", teams: { "My Team": { channels: { General: { requireMention: true }, }, }, }, }, },}Configuración manual (sin la CLI de Teams)
Si no puede usar la CLI de Teams, puede configurar el bot manualmente a través de Azure Portal.
Cómo funciona
Sección titulada «Cómo funciona»- Asegúrese de que el complemento de Microsoft Teams esté disponible (incluido en las versiones actuales).
- Cree un Azure Bot (App ID + secreto + ID de inquilino).
- Compile un paquete de aplicación de Teams que haga referencia al bot e incluya los permisos RSC a continuación.
- Cargue/instale la aplicación de Teams en un equipo (o en el ámbito personal para MDs).
- Configure
msteamsen~/.openclaw/openclaw.json(o variables de entorno) e inicie la puerta de enlace. - La puerta de enlace escucha el tráfico del webhook de Bot Framework en
/api/messagesde manera predeterminada.
Paso 1: Crear Azure Bot
Sección titulada «Paso 1: Crear Azure Bot»-
Vaya a Create Azure Bot
-
Rellene la pestaña Aspectos básicos:
Campo Valor Identificador de bot El nombre de su bot, por ejemplo, openclaw-msteams(debe ser único)Suscripción Seleccione su suscripción de Azure Grupo de recursos Crear nuevo o usar existente Nivel de precios Gratis para desarrollo/pruebas Tipo de aplicación Single Tenant (recomendado: consulte la nota a continuación) Tipo de creación Crear nuevo Microsoft App ID
- Haga clic en Revisar y crear → Crear (espere ~1-2 minutos)
Paso 2: Obtener credenciales
Sección titulada «Paso 2: Obtener credenciales»- Vaya al recurso de su Azure Bot → Configuración
- Copie el Microsoft App ID → este es su
appId - Haga clic en Administrar contraseña → vaya al Registro de la aplicación
- En Certificados y secretos → Nuevo secreto de cliente → copie el Valor → este es su
appPassword - Vaya a Información general → copie el Id. de directorio (inquilino) → este es su
tenantId
Paso 3: Configurar el extremo de mensajería
Sección titulada «Paso 3: Configurar el extremo de mensajería»- En Azure Bot → Configuración
- Establezca el Extremo de mensajería en la URL de su webhook:
- Producción:
https://your-domain.com/api/messages - Desarrollo local: Use un túnel (consulte Local Development a continuación)
- Producción:
Paso 4: Habilitar el canal de Teams
Sección titulada «Paso 4: Habilitar el canal de Teams»- En Azure Bot → Canales
- Haga clic en Microsoft Teams → Configurar → Guardar
- Acepte los Términos de servicio
Paso 5: Crear el manifiesto de la aplicación de Teams
Sección titulada «Paso 5: Crear el manifiesto de la aplicación de Teams»- Incluya una entrada
botconbotId = <App ID>. - Ámbitos:
personal,team,groupChat. supportsFiles: true(requerido para el manejo de archivos en el ámbito personal).- Añada permisos RSC (consulte RSC Permissions).
- Cree iconos:
outline.png(32x32) ycolor.png(192x192). - Comprima los tres archivos juntos:
manifest.json,outline.png,color.png.
Paso 6: Configurar OpenClaw
Sección titulada «Paso 6: Configurar OpenClaw»{ channels: { msteams: { enabled: true, appId: "<APP_ID>", appPassword: "<APP_PASSWORD>", tenantId: "<TENANT_ID>", webhook: { port: 3978, path: "/api/messages" }, }, },}Variables de entorno: MSTEAMS_APP_ID, MSTEAMS_APP_PASSWORD, MSTEAMS_TENANT_ID.
Paso 7: Ejecutar el Gateway
Sección titulada «Paso 7: Ejecutar el Gateway»El canal de Teams se inicia automáticamente cuando el complemento está disponible y existe la configuración msteams con credenciales.
Autenticación federada (certificado más identidad administrada)
Sección titulada «Autenticación federada (certificado más identidad administrada)»Agregado en 2026.4.11
Para implementaciones en producción, OpenClaw admite autenticación federada como una alternativa más segura a los secretos de cliente. Hay dos métodos disponibles:
Opción A: Autenticación basada en certificados
Sección titulada «Opción A: Autenticación basada en certificados»Use un certificado PEM registrado con su registro de aplicación de Entra ID.
Configuración:
- Genere u obtenga un certificado (formato PEM con clave privada).
- En Entra ID → Registro de aplicación → Certificados y secretos → Certificados → Cargar el certificado público.
Configuración:
{ channels: { msteams: { enabled: true, appId: "<APP_ID>", tenantId: "<TENANT_ID>", authType: "federated", certificatePath: "/path/to/cert.pem", webhook: { port: 3978, path: "/api/messages" }, }, },}Variables de entorno:
MSTEAMS_AUTH_TYPE=federatedMSTEAMS_CERTIFICATE_PATH=/path/to/cert.pem
Opción B: Identidad administrada de Azure
Sección titulada «Opción B: Identidad administrada de Azure»Use Azure Managed Identity para la autenticación sin contraseña. Esto es ideal para implementaciones en la infraestructura de Azure (AKS, App Service, máquinas virtuales de Azure) donde hay disponible una identidad administrada.
Cómo funciona:
- El pod/VM del bot tiene una identidad administrada (asignada por el sistema o asignada por el usuario).
- Una credencial de identidad federada vincula la identidad administrada al registro de la aplicación de Entra ID.
- En tiempo de ejecución, OpenClaw usa
@azure/identitypara adquirir tokens desde el punto de conexión IMDS de Azure (169.254.169.254). - El token se pasa al SDK de Teams para la autenticación del bot.
Requisitos previos:
- Infraestructura de Azure con identidad administrada habilitada (identidad de carga de trabajo de AKS, App Service, VM)
- Credencial de identidad federada creada en el registro de la aplicación de Entra ID
- Acceso de red a IMDS (
169.254.169.254:80) desde el pod/VM
Config (identidad administrada asignada por el sistema):
{ channels: { msteams: { enabled: true, appId: "<APP_ID>", tenantId: "<TENANT_ID>", authType: "federated", useManagedIdentity: true, webhook: { port: 3978, path: "/api/messages" }, }, },}Config (identidad administrada asignada por el usuario):
{ channels: { msteams: { enabled: true, appId: "<APP_ID>", tenantId: "<TENANT_ID>", authType: "federated", useManagedIdentity: true, managedIdentityClientId: "<MI_CLIENT_ID>", webhook: { port: 3978, path: "/api/messages" }, }, },}Variables de entorno:
MSTEAMS_AUTH_TYPE=federatedMSTEAMS_USE_MANAGED_IDENTITY=trueMSTEAMS_MANAGED_IDENTITY_CLIENT_ID=<client-id>(solo para asignada por el usuario)
Configuración de la identidad de carga de trabajo de AKS
Sección titulada «Configuración de la identidad de carga de trabajo de AKS»Para implementaciones de AKS que usan identidad de carga de trabajo:
-
Habilite la identidad de carga de trabajo en su clúster de AKS.
-
Cree una credencial de identidad federada en el registro de la aplicación de Entra ID:
Ventana de terminal 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"]}' -
Anote la cuenta de servicio de Kubernetes con el ID de cliente de la aplicación:
apiVersion: v1kind: ServiceAccountmetadata:name: my-bot-saannotations:azure.workload.identity/client-id: "<APP_CLIENT_ID>" -
Etiquete el pod para la inyección de identidad de carga de trabajo:
metadata:labels:azure.workload.identity/use: "true" -
Asegure el acceso de red a IMDS (
169.254.169.254); si usa NetworkPolicy, agregue una regla de salida que permita el tráfico a169.254.169.254/32en el puerto 80.
Comparación de tipos de autenticación
Sección titulada «Comparación de tipos de autenticación»| Método | Config | Pros | Contras |
|---|---|---|---|
| Secreto de cliente | appPassword | Configuración sencilla | Se requiere rotación de secretos, menos segura |
| Certificado | authType: "federated" + certificatePath | Sin secreto compartido a través de la red | Sobrecarga de gestión de certificados |
| Identidad administrada | authType: "federated" + useManagedIdentity | Sin contraseña, sin secretos que gestionar | Se requiere infraestructura de Azure |
Comportamiento predeterminado: Cuando no se establece authType, OpenClaw usa de forma predeterminada la autenticación de secreto de cliente. Las configuraciones existentes continúan funcionando sin cambios.
Desarrollo local (túnel)
Sección titulada «Desarrollo local (túnel)»Teams no puede alcanzar localhost. Use un túnel de desarrollo persistente para que su URL se mantenga igual entre sesiones:
# One-time setup:devtunnel create my-openclaw-bot --allow-anonymousdevtunnel port create my-openclaw-bot -p 3978 --protocol auto
# Each dev session:devtunnel host my-openclaw-botAlternativas: ngrok http 3978 o tailscale funnel 3978 (las URL pueden cambiar en cada sesión).
Si la URL de su túnel cambia, actualice el punto de conexión:
teams app update <teamsAppId> --endpoint "https://<new-url>/api/messages"Probar el Bot
Sección titulada «Probar el Bot»Ejecutar diagnósticos:
teams app doctor <teamsAppId>Verifica el registro del bot, la aplicación AAD, el manifiesto y la configuración de SSO de una sola vez.
Enviar un mensaje de prueba:
- Instalar la aplicación de Teams (usar el enlace de instalación de
teams app get <id> --install-link) - Buscar el bot en Teams y enviar un MD
- Verificar los registros de la puerta de enlace para ver la actividad entrante
Variables de entorno
Sección titulada «Variables de entorno»Todas las claves de configuración se pueden establecer mediante variables de entorno:
MSTEAMS_APP_IDMSTEAMS_APP_PASSWORDMSTEAMS_TENANT_IDMSTEAMS_AUTH_TYPE(opcional:"secret"o"federated")MSTEAMS_CERTIFICATE_PATH(federado + certificado)MSTEAMS_CERTIFICATE_THUMBPRINT(opcional, no necesario para autenticación)MSTEAMS_USE_MANAGED_IDENTITY(federado + identidad administrada)MSTEAMS_MANAGED_IDENTITY_CLIENT_ID(solo MI asignada por el usuario)
Acción de información de miembro
Sección titulada «Acción de información de miembro»OpenClaw expone una acción member-info respaldada por Graph para Microsoft Teams, para que los agentes y automatizaciones puedan resolver detalles de miembros del canal (nombre para mostrar, correo electrónico, rol) directamente desde Microsoft Graph.
Requisitos:
- Permiso RSC
Member.Read.Group(ya incluido en el manifiesto recomendado) - Para búsquedas entre equipos: permiso de aplicación de Graph
User.Read.Allcon consentimiento de administrador
La acción está controlada por channels.msteams.actions.memberInfo (predeterminado: habilitado cuando las credenciales de Graph están disponibles).
Contexto de historial
Sección titulada «Contexto de historial»channels.msteams.historyLimitcontrola cuántos mensajes recientes del canal/grupo se incluyen en el aviso.- Recurre a
messages.groupChat.historyLimit. Establezca0para deshabilitar (predeterminado 50). - El historial de hilos recuperado se filtra mediante listas de permitidos de remitentes (
allowFrom/groupAllowFrom), por lo que la inicialización del contexto del hilo solo incluye mensajes de remitentes permitidos. - El contexto de datos adjuntos citados (
ReplyTo*derivado del HTML de respuesta de Teams) se pasa actualmente tal como se recibe. - En otras palabras, las listas permitidas controlan quién puede activar el agente; hoy solo se filtran rutas de contexto suplementarias específicas.
- El historial de MD se puede limitar con
channels.msteams.dmHistoryLimit(turnos de usuario). Invalidaciones por usuario:channels.msteams.dms["<user_id>"].historyLimit.
Permisos RSC actuales de Teams (manifiesto)
Sección titulada «Permisos RSC actuales de Teams (manifiesto)»Estos son los permisos resourceSpecific existentes en nuestro manifiesto de aplicación de Teams. Solo se aplican dentro del equipo/chat donde está instalada la aplicación.
Para canales (ámbito de equipo):
ChannelMessage.Read.Group(Application) - recibir todos los mensajes del canal sin @menciónChannelMessage.Send.Group(Application)Member.Read.Group(Application)Owner.Read.Group(Application)ChannelSettings.Read.Group(Application)TeamMember.Read.Group(Application)TeamSettings.Read.Group(Application)
Para chats de grupo:
ChatMessage.Read.Chat(Application) - recibir todos los mensajes de chat de grupo sin @mención
Para agregar permisos RSC a través de la CLI de Teams:
teams app rsc add <teamsAppId> ChannelMessage.Read.Group --type ApplicationManifiesto de Teams de ejemplo (censurado)
Sección titulada «Manifiesto de Teams de ejemplo (censurado)»Ejemplo mínimo y válido con los campos obligatorios. Reemplace los ID y las 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" }, ], }, },}Advertencias del manifiesto (campos obligatorios)
Sección titulada «Advertencias del manifiesto (campos obligatorios)»bots[].botIddebe coincidir con el ID de la aplicación de Azure Bot.webApplicationInfo.iddebe coincidir con el ID de la aplicación de Azure Bot.bots[].scopesdebe incluir las superficies que planea usar (personal,team,groupChat).bots[].supportsFiles: truees necesario para el manejo de archivos en el ámbito personal.authorization.permissions.resourceSpecificdebe incluir lectura/escritura del canal si desea tráfico del canal.
Actualizar una aplicación existente
Sección titulada «Actualizar una aplicación existente»Para actualizar una aplicación de Teams ya instalada (por ejemplo, para agregar permisos RSC):
# Download, edit, and re-upload the manifestteams app manifest download <teamsAppId> manifest.json# Edit manifest.json locally...teams app manifest upload manifest.json <teamsAppId># Version is auto-bumped if content changedDespués de actualizar, reinstale la aplicación en cada equipo para que los nuevos permisos surtan efecto y cierre y vuelva a abrir Teams por completo (no solo cierre la ventana) para borrar los metadatos de la aplicación en caché.
Actualización manual del manifiesto (sin CLI)
- Actualice su
manifest.jsoncon la nueva configuración - Incremente el campo
version(p. ej.,1.0.0→1.1.0) - Vuelva a comprimir el manifiesto con los iconos (
manifest.json,outline.png,color.png) - Suba el nuevo archivo zip:
- Centro de administración de Teams: Aplicaciones de Teams → Administrar aplicaciones → busque su aplicación → Subir nueva versión
- Carga lateral: En Teams → Aplicaciones → Administrar sus aplicaciones → Cargar una aplicación personalizada
Capacidades: Solo RSC frente a Graph
Sección titulada «Capacidades: Solo RSC frente a Graph»Con solo Teams RSC (aplicación instalada, sin permisos de API de Graph)
Sección titulada «Con solo Teams RSC (aplicación instalada, sin permisos de API de Graph)»Funciona:
- Leer el contenido de texto de los mensajes del canal.
- Enviar contenido de texto de mensajes al canal.
- Recibir archivos adjuntos personales (MD).
NO funciona:
- Contenido de imágenes o archivos de canales/grupos (la carga útil solo incluye un código auxiliar HTML).
- Descargar archivos adjuntos almacenados en SharePoint/OneDrive.
- Leer el historial de mensajes (más allá del evento de webhook en tiempo real).
Con Teams RSC + Permisos de aplicación de Microsoft Graph
Sección titulada «Con Teams RSC + Permisos de aplicación de Microsoft Graph»Añade:
- Descargar contenidos alojados (imágenes pegadas en los mensajes).
- Descargar archivos adjuntos almacenados en SharePoint/OneDrive.
- Leer el historial de mensajes de canales/chats a través de Graph.
RSC frente a API de Graph
Sección titulada «RSC frente a API de Graph»| Capacidad | Permisos RSC | API de Graph |
|---|---|---|
| Mensajes en tiempo real | Sí (vía webhook) | No (solo sondeo) |
| Mensajes históricos | No | Sí (puede consultar el historial) |
| Complejidad de configuración | Solo manifiesto de la aplicación | Requiere consentimiento de administrador + flujo de token |
| Funciona sin conexión | No (debe estar en ejecución) | Sí (consultar en cualquier momento) |
En resumen: RSC es para la escucha en tiempo real; la API de Graph es para el acceso histórico. Para ponerse al día con los mensajes perdidos mientras está sin conexión, necesita la API de Graph con ChannelMessage.Read.All (requiere consentimiento de administrador).
Medios e historial habilitados para Graph (necesario para canales)
Sección titulada «Medios e historial habilitados para Graph (necesario para canales)»Si necesita imágenes/archivos en canales o desea recuperar el historial de mensajes, debe habilitar los permisos de Microsoft Graph y otorgar el consentimiento del administrador.
- En Entra ID (Azure AD) Registro de aplicación, agregue Permisos de aplicación de Microsoft Graph:
ChannelMessage.Read.All(archivos adjuntos del canal + historial)Chat.Read.AlloChatMessage.Read.All(chats grupales)
- Otorgar consentimiento de administrador para el inquilino.
- Aumente la versión del manifiesto de la aplicación de Teams, cárguela de nuevo y reinstale la aplicación en Teams.
- Cierra completamente y vuelve a iniciar Teams para borrar los metadatos de la aplicación en caché.
Permiso adicional para menciones de usuario: Las @menciones de usuario funcionan de inmediato para los usuarios en la conversación. Sin embargo, si deseas buscar y mencionar dinámicamente a usuarios que no están en la conversación actual, agrega el permiso User.Read.All (Aplicación) y concede el consentimiento del administrador.
Limitaciones conocidas
Sección titulada «Limitaciones conocidas»Tiempos de espera de webhooks
Sección titulada «Tiempos de espera de webhooks»Teams entrega mensajes a través de un webhook HTTP. Si el procesamiento tarda demasiado (por ejemplo, respuestas lentas del LLM), es posible que veas:
- Tiempos de espera de la puerta de enlace
- Teams reintentando el mensaje (causando duplicados)
- Respuestas perdidas
OpenClaw maneja esto respondiendo rápidamente y enviando respuestas de manera proactiva, pero las respuestas muy lentas aún pueden causar problemas.
Formato
Sección titulada «Formato»El markdown de Teams es más limitado que el de Slack o Discord:
- El formato básico funciona: negrita, cursiva,
code, enlaces - El markdown complejo (tablas, listas anidadas) puede no renderizarse correctamente
- Las tarjetas adaptables (Adaptive Cards) son compatibles con encuestas y envíos de presentación semántica (ver a continuación)
Configuración
Sección titulada «Configuración»Configuraciones clave (ver /gateway/configuration para patrones de canal compartido):
channels.msteams.enabled: habilitar/deshabilitar el canal.channels.msteams.appId,channels.msteams.appPassword,channels.msteams.tenantId: credenciales del bot.channels.msteams.webhook.port(por defecto3978)channels.msteams.webhook.path(por defecto/api/messages)channels.msteams.dmPolicy:pairing | allowlist | open | disabled(por defecto: emparejamiento)channels.msteams.allowFrom: lista de permitidos para MD (se recomiendan los ID de objeto de AAD). El asistente resuelve los nombres a ID durante la configuración cuando el acceso a Graph está disponible.channels.msteams.dangerouslyAllowNameMatching: interruptor de emergencia para volver a habilitar la coincidencia mutable de UPN/nombre para mostrar y el enrutamiento directo por nombre de equipo/canal.channels.msteams.textChunkLimit: tamaño del fragmento de texto de salida.channels.msteams.chunkMode:length(por defecto) onewlinepara dividir en líneas en blanco (límites de párrafo) antes de la división por longitud.channels.msteams.mediaAllowHosts: lista de permitidos para hosts de archivos adjuntos entrantes (por defecto dominios de Microsoft/Teams).channels.msteams.mediaAuthAllowHosts: lista de permitidos para adjuntar encabezados de Autorización en reintentos de medios (por defecto hosts de Graph + Bot Framework).channels.msteams.requireMention: requerir @mención en canales/grupos (por defecto true).channels.msteams.replyStyle:thread | top-level(consulte Reply Style).channels.msteams.teams.<teamId>.replyStyle: anulación por equipo.channels.msteams.teams.<teamId>.requireMention: anulación por equipo.channels.msteams.teams.<teamId>.tools: anulaciones de política de herramienta predeterminadas por equipo (allow/deny/alsoAllow) usadas cuando falta una anulación de canal.channels.msteams.teams.<teamId>.toolsBySender: anulaciones de política de herramienta por remitente por equipo predeterminadas (se admite comodín"*").channels.msteams.teams.<teamId>.channels.<conversationId>.replyStyle: anulación por canal.channels.msteams.teams.<teamId>.channels.<conversationId>.requireMention: anulación por canal.channels.msteams.teams.<teamId>.channels.<conversationId>.tools: anulaciones de política de herramienta por canal (allow/deny/alsoAllow).channels.msteams.teams.<teamId>.channels.<conversationId>.toolsBySender: anulaciones de política de herramienta por remitente por canal (se admite comodín"*").- Las claves
toolsBySenderdeben usar prefijos explícitos:channel:,id:,e164:,username:,name:(las claves heredadas sin prefijo siguen asignándose solo aid:). channels.msteams.actions.memberInfo: activar o desactivar la acción de información de miembros respaldada por Graph (predeterminado: activado cuando las credenciales de Graph están disponibles).channels.msteams.authType: tipo de autenticación -"secret"(predeterminado) o"federated".channels.msteams.certificatePath: ruta al archivo de certificado PEM (autenticación federada + certificado).channels.msteams.certificateThumbprint: huella digital del certificado (opcional, no necesario para la autenticación).channels.msteams.useManagedIdentity: activar autenticación de identidad administrada (modo federado).channels.msteams.managedIdentityClientId: ID de cliente para identidad administrada asignada por el usuario.channels.msteams.sharePointSiteId: ID del sitio de SharePoint para cargar archivos en chats grupales/canales (consulte Enviar archivos en chats grupales).
Enrutamiento y sesiones
Sección titulada «Enrutamiento y sesiones»- Las claves de sesión siguen el formato estándar del agente (consulte /concepts/session):
- Los mensajes directos comparten la sesión principal (
agent:<agentId>:<mainKey>). - Los mensajes de canal/grupo usan el id. de conversación:
agent:<agentId>:msteams:channel:<conversationId>agent:<agentId>:msteams:group:<conversationId>
- Los mensajes directos comparten la sesión principal (
Estilo de respuesta: hilos vs. publicaciones
Sección titulada «Estilo de respuesta: hilos vs. publicaciones»Teams introdujo recientemente dos estilos de interfaz de usuario de canal sobre el mismo modelo de datos subyacente:
| Estilo | Descripción | Recomendado replyStyle |
|---|---|---|
| Publicaciones (clásico) | Los mensajes aparecen como tarjetas con respuestas en hilo debajo | thread (predeterminado) |
| Hilos (estilo Slack) | Los mensajes fluyen linealmente, más como en Slack | top-level |
El problema: La API de Teams no expone qué estilo de interfaz de usuario utiliza un canal. Si utiliza el replyStyle incorrecto:
threaden un canal estilo Hilos → las respuestas aparecen anidadas de forma incómodatop-levelen un canal estilo Publicaciones → las respuestas aparecen como publicaciones de nivel superior separadas en lugar de en el hilo
Solución: Configure replyStyle por canal según cómo esté configurado el canal:
{ channels: { msteams: { replyStyle: "thread", teams: { channels: { replyStyle: "top-level", }, }, }, }, }, },}Precedencia de resolución
Sección titulada «Precedencia de resolución»Cuando el bot envía una respuesta a un canal, replyStyle se resuelve desde la invalidación más específica hasta el valor predeterminado. El primer valor que no sea undefined gana:
- Por canal —
channels.msteams.teams.<teamId>.channels.<conversationId>.replyStyle - Por equipo —
channels.msteams.teams.<teamId>.replyStyle - Global —
channels.msteams.replyStyle - Predeterminado implícito — derivado de
requireMention:requireMention: true→threadrequireMention: false→top-level
Si establece requireMention: false globalmente sin un replyStyle explícito, las menciones en los canales estilo Publicaciones aparecerán como publicaciones de nivel superior incluso cuando el mensaje entrante fue una respuesta de hilo. Fije replyStyle: "thread" a nivel global, de equipo o de canal para evitar sorpresas.
Preservación del contexto del hilo
Sección titulada «Preservación del contexto del hilo»Cuando replyStyle: "thread" está en vigor y el bot fue @mencionado desde dentro de un hilo del canal, OpenClaw vuelve a adjuntar la raíz del hilo original a la referencia de conversación saliente (19:…@thread.tacv2;messageid=<root>) para que la respuesta aterrice dentro del mismo hilo. Esto es válido tanto para envíos en vivo (en turno) como para envíos proactivos realizados después de que el contexto de turno del Bot Framework haya expirado (por ejemplo, agentes de larga duración, respuestas a llamadas de herramientas en cola a través de mcp__openclaw__message).
La raíz del hilo se toma del threadId almacenado en la referencia de conversación. Las referencias almacenadas más antiguas que preceden a threadId vuelven a activityId (cualquier actividad entrante que haya sembrado por última vez la conversación), por lo que los despliegues existentes siguen funcionando sin necesidad de volver a sembrar.
Cuando replyStyle: "top-level" está en vigor, las entradas de hilo del canal se responden intencionalmente como nuevas publicaciones de nivel superior; no se adjunta ningún sufijo de hilo. Este es el comportamiento correcto para los canales de estilo Hilos; si ve publicaciones de nivel superior donde esperaba respuestas en hilo, su replyStyle está configurado incorrectamente para ese canal.
Archivos adjuntos e imágenes
Sección titulada «Archivos adjuntos e imágenes»Limitaciones actuales:
- Mensajes directos (DMs): Las imágenes y los archivos adjuntos funcionan a través de las API de archivos de bot de Teams.
- Canales/grupos: Los archivos adjuntos residen en el almacenamiento de M365 (SharePoint/OneDrive). La carga útil del webhook solo incluye un código auxiliar HTML, no los bytes reales del archivo. Se requieren permisos de la API de Graph para descargar los archivos adjuntos del canal.
- Para envíos explícitos de archivo primero, use
action=upload-fileconmedia/filePath/path; elmessageopcional se convierte en el texto/comentario adjunto, yfilenameanula el nombre cargado.
Sin permisos de Graph, los mensajes del canal con imágenes se recibirán solo como texto (el contenido de la imagen no es accesible para el bot).
De forma predeterminada, OpenClaw solo descarga medios de nombres de host de Microsoft/Teams. Anule con channels.msteams.mediaAllowHosts (use ["*"] para permitir cualquier host).
Los encabezados de autorización solo se adjuntan para los hosts en channels.msteams.mediaAuthAllowHosts (predeterminado en hosts de Graph + Bot Framework). Mantenga esta lista estricta (evite sufijos multiinquilino).
Envío de archivos en chats de grupo
Sección titulada «Envío de archivos en chats de grupo»Los bots pueden enviar archivos en mensajes directos (DM) usando el flujo FileConsentCard (integrado). Sin embargo, el envío de archivos en chats de grupo/canales requiere una configuración adicional:
| Contexto | Cómo se envían los archivos | Configuración necesaria |
|---|---|---|
| Mensajes directos (DMs) | FileConsentCard → usuario acepta → bot sube | Funciona de fábrica |
| Chats de grupo/canales | Subir a SharePoint → compartir enlace | Requiere sharePointSiteId + permisos de Graph |
| Imágenes (cualquier contexto) | Codificadas en Base64 en línea | Funciona de fábrica |
Por qué los chats de grupo necesitan SharePoint
Sección titulada «Por qué los chats de grupo necesitan SharePoint»Los bots no tienen una unidad personal de OneDrive (el punto de conexión de la API de Graph /me/drive no funciona para identidades de aplicaciones). Para enviar archivos en chats grupales/canales, el bot los carga en un sitio de SharePoint y crea un vínculo para compartir.
Configuración
Sección titulada «Configuración»-
Agregue permisos de la API de Graph en Entra ID (Azure AD) → Registro de aplicación:
Sites.ReadWrite.All(Aplicación): cargar archivos en SharePointChat.Read.All(Aplicación): opcional, permite vínculos para compartir por usuario
-
Conceda el consentimiento del administrador para el inquilino.
-
Obtenga su ID de sitio de SharePoint:
Ventana de terminal # 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" -
Configure OpenClaw:
{channels: {msteams: {// ... other config ...sharePointSiteId: "contoso.sharepoint.com,guid1,guid2",},},}
Comportamiento para compartir
Sección titulada «Comportamiento para compartir»| Permiso | Comportamiento para compartir |
|---|---|
Solo Sites.ReadWrite.All | Enlace para compartir en toda la organización (cualquiera en la organización puede acceder) |
Sites.ReadWrite.All + Chat.Read.All | Enlace para compartir por usuario (solo los miembros del chat pueden acceder) |
El uso compartido por usuario es más seguro, ya que solo los participantes del chat pueden acceder al archivo. Si falta el permiso Chat.Read.All, el bot recurre al uso compartido en toda la organización.
Comportamiento de reserva
Sección titulada «Comportamiento de reserva»| Escenario | Resultado |
|---|---|
Chat grupal + archivo + sharePointSiteId configurado | Subir a SharePoint, enviar enlace para compartir |
Chat grupal + archivo + sin sharePointSiteId | Intentar carga a OneDrive (puede fallar), enviar solo texto |
| Chat personal + archivo | Flujo FileConsentCard (funciona sin SharePoint) |
| Cualquier contexto + imagen | Codificado en Base64 en línea (funciona sin SharePoint) |
Ubicación de almacenamiento de archivos
Sección titulada «Ubicación de almacenamiento de archivos»Los archivos cargados se almacenan en una carpeta /OpenClawShared/ en la biblioteca de documentos predeterminada del sitio de SharePoint configurado.
Encuestas (Adaptive Cards)
Sección titulada «Encuestas (Adaptive Cards)»OpenClaw envía encuestas de Teams como Adaptive Cards (no hay una API de encuestas nativa de Teams).
- CLI:
openclaw message poll --channel msteams --target conversation:<id> ... - Los votos son registrados por la puerta de enlace en
~/.openclaw/msteams-polls.json. - La puerta de enlace debe permanecer en línea para registrar los votos.
- Las encuestas aún no publican automáticamente resúmenes de resultados (inspeccione el archivo de almacenamiento si es necesario).
Tarjetas de presentación
Sección titulada «Tarjetas de presentación»Envíe cargas útiles de presentación semántica a usuarios o conversaciones de Teams mediante la herramienta message, la CLI o la entrega normal de respuestas. OpenClaw las representa como tarjetas adaptables de Teams desde el contrato de presentación genérico.
El parámetro presentation acepta bloques semánticos. Cuando se proporciona presentation, el texto del mensaje es opcional. Los botones se representan como acciones de envío o URL de tarjeta adaptable. Los menús de selección aún no son nativos en el representador de Teams, por lo que OpenClaw los degrada a texto legible antes de la entrega.
Herramienta de agente:
{ action: "send", channel: "msteams", target: "user:<id>", presentation: { title: "Hello", blocks: [{ type: "text", text: "Hello!" }], },}CLI:
openclaw message send --channel msteams \ --presentation '{"title":"Hello","blocks":[{"type":"text","text":"Hello!"}]}'Para obtener detalles sobre los formatos de destino, consulte Formatos de destino a continuación.
Formatos de destino
Sección titulada «Formatos de destino»Los destinos de MSTeams usan prefijos para distinguir entre usuarios y conversaciones:
| Tipo de destino | Formato | Ejemplo |
|---|---|---|
| Usuario (por ID) | user:<aad-object-id> | user:40a1a0ed-4ff2-4164-a219-55518990c197 |
| Usuario (por nombre) | user:<display-name> | user:John Smith (requiere la API de Graph) |
| Grupo/canal | conversation:<conversation-id> | conversation:19:[email protected] |
| Grupo/canal (sin procesar) | <conversation-id> | 19:[email protected] (si contiene @thread) |
Ejemplos de CLI:
# Send to a user by IDopenclaw 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 channelopenclaw message send --channel msteams --target "conversation:19:[email protected]" --message "Hello"
# Send a presentation card to a conversation --presentation '{"title":"Hello","blocks":[{"type":"text","text":"Hello"}]}'Ejemplos de herramienta de agente:
{ action: "send", channel: "msteams", target: "user:John Smith", message: "Hello!",}{ action: "send", channel: "msteams", presentation: { title: "Hello", blocks: [{ type: "text", text: "Hello" }], },}Mensajería proactiva
Sección titulada «Mensajería proactiva»- Los mensajes proactivos solo son posibles después de que un usuario haya interactuado, porque almacenamos las referencias de conversación en ese momento.
- Consulte
/gateway/configurationparadmPolicyy el filtrado de lista de permitidos.
IDs de equipo y canal (Error común)
Sección titulada «IDs de equipo y canal (Error común)»El parámetro de consulta groupId en las URLs de Teams NO es el ID de equipo utilizado para la configuración. Extraiga los IDs de la ruta de la URL en su lugar:
URL del equipo:
https://teams.microsoft.com/l/team/19%3ABk4j...%40thread.tacv2/conversations?groupId=... └────────────────────────────┘ Team conversation ID (URL-decode this)URL del canal:
https://teams.microsoft.com/l/channel/19%3A15bc...%40thread.tacv2/ChannelName?groupId=... └─────────────────────────┘ Channel ID (URL-decode this)Para la configuración:
- Clave de equipo = segmento de ruta después de
/team/(decodificado en URL, p. ej.,19:[email protected]; los inquilinos antiguos pueden mostrar@thread.skype, que también es válido) - Clave de canal = segmento de ruta después de
/channel/(decodificado en URL) - Ignore el parámetro de consulta
groupIdpara el enrutamiento de OpenClaw. Es el ID de grupo de Microsoft Entra, no el ID de conversación de Bot Framework utilizado en las actividades entrantes de Teams.
Canales privados
Sección titulada «Canales privados»Los bots tienen soporte limitado en canales privados:
| Característica | Canales estándar | Canales privados |
|---|---|---|
| Instalación del bot | Sí | Limitado |
| Mensajes en tiempo real (webhook) | Sí | Puede que no funcione |
| Permisos RSC | Sí | Puede comportarse de manera diferente |
| @menciones | Sí | Si el bot es accesible |
| Historial de Graph API | Sí | Sí (con permisos) |
Soluciones alternativas si los canales privados no funcionan:
- Use canales estándar para las interacciones del bot
- Use MDs: los usuarios siempre pueden enviar mensajes directamente al bot
- Use la API de Graph para acceso histórico (requiere
ChannelMessage.Read.All)
Solución de problemas
Sección titulada «Solución de problemas»Problemas comunes
Sección titulada «Problemas comunes»- Las imágenes no se muestran en los canales: Faltan permisos de Graph o el consentimiento del administrador. Vuelva a instalar la aplicación de Teams y cierre/abra Teams por completo.
- Sin respuestas en el canal: se requieren menciones de forma predeterminada; establezca
channels.msteams.requireMention=falseo configure por equipo/canal. - Discrepancia de versión (Teams todavía muestra el manifiesto antiguo): elimine + vuelva a agregar la aplicación y cierre Teams por completo para actualizar.
- 401 Unauthorized del webhook: Se espera al probar manualmente sin el JWT de Azure, lo que significa que el punto de conexión es accesible pero la autenticación falló. Use Azure Web Chat para probar correctamente.
Errores de carga de manifiesto
Sección titulada «Errores de carga de manifiesto»- “El archivo de icono no puede estar vacío”: El manifiesto hace referencia a archivos de icono que tienen 0 bytes. Cree iconos PNG válidos (32x32 para
outline.png, 192x192 paracolor.png). - “webApplicationInfo.Id ya está en uso”: La aplicación todavía está instalada en otro equipo/chat. Encuéntrela y desinstálela primero, o espere 5-10 minutos para la propagación.
- “Algo salió mal” al cargar: Cargue a través de https://admin.teams.microsoft.com en su lugar, abra las DevTools del navegador (F12) → pestaña Network y verifique el cuerpo de la respuesta para ver el error real.
- Error al realizar la carga lateral: Intente “Cargar una aplicación en el catálogo de aplicaciones de su organización” en lugar de “Cargar una aplicación personalizada”: esto a menudo evita las restricciones de carga lateral.
Los permisos RSC no funcionan
Sección titulada «Los permisos RSC no funcionan»- Verifique que
webApplicationInfo.idcoincida exactamente con el ID de aplicación de su bot - Vuelva a cargar la aplicación y reinstálela en el equipo/chat
- Compruebe si el administrador de su organización ha bloqueado los permisos RSC
- Confirme que está utilizando el ámbito correcto:
ChannelMessage.Read.Grouppara equipos,ChatMessage.Read.Chatpara chats de grupo
Referencias
Sección titulada «Referencias»- Create Azure Bot - guía de configuración de Azure Bot
- Teams Developer Portal - crear/administrar aplicaciones de Teams
- Teams app manifest schema
- Receive channel messages with RSC
- RSC permissions reference
- Teams bot file handling (canal/grupo requiere Graph)
- Proactive messaging
- @microsoft/teams.cli - CLI de Teams para la gestión de bots
Relacionado
Sección titulada «Relacionado»- Channels Overview - todos los canales compatibles
- Pairing - flujo de autenticación y emparejamiento de MD
- Grupos - comportamiento del chat de grupo y filtrado de menciones
- Enrutamiento de canales - enrutamiento de sesiones para mensajes
- Seguridad - modelo de acceso y protección