website/docs/guides/operate-teams-meeting-pipeline.md
Use this guide after you have already enabled the feature from Teams Meetings.
This page covers:
hermes teams-pipeline validate
Use this first after any config change.
hermes teams-pipeline token-health
hermes teams-pipeline token-health --force-refresh
Use --force-refresh when you suspect stale auth state.
hermes teams-pipeline subscriptions
hermes teams-pipeline maintain-subscriptions
hermes teams-pipeline maintain-subscriptions --dry-run
Microsoft Graph subscriptions expire in at most 72 hours. If nothing renews them, meeting notifications silently stop after 3 days and the pipeline looks "broken." This is the #1 operational failure mode for any Graph-backed integration.
You MUST run maintain-subscriptions on a schedule. Pick one of these three options:
Hermes ships a built-in cron scheduler. The --no-agent mode runs a script as the job (rather than using an LLM), and --script must point at a file under ~/.hermes/scripts/. First create the script:
mkdir -p ~/.hermes/scripts
cat > ~/.hermes/scripts/maintain-teams-subscriptions.sh <<'EOF'
#!/usr/bin/env bash
exec hermes teams-pipeline maintain-subscriptions
EOF
chmod +x ~/.hermes/scripts/maintain-teams-subscriptions.sh
Then register a script-only cron job that runs every 12 hours (gives 6x headroom against the 72h expiry window):
hermes cron create "0 */12 * * *" \
--name "teams-pipeline-maintain-subscriptions" \
--no-agent \
--script maintain-teams-subscriptions.sh \
--deliver local
Verify it was registered and inspect the next run time:
hermes cron list
hermes cron status # scheduler status
Create /etc/systemd/system/hermes-teams-pipeline-maintain.service:
[Unit]
Description=Hermes Teams pipeline subscription maintenance
After=network-online.target
[Service]
Type=oneshot
User=hermes
EnvironmentFile=/etc/hermes/env
ExecStart=/usr/local/bin/hermes teams-pipeline maintain-subscriptions
And /etc/systemd/system/hermes-teams-pipeline-maintain.timer:
[Unit]
Description=Run Hermes Teams pipeline subscription maintenance every 12 hours
[Timer]
OnBootSec=5min
OnUnitActiveSec=12h
Persistent=true
[Install]
WantedBy=timers.target
Enable:
sudo systemctl daemon-reload
sudo systemctl enable --now hermes-teams-pipeline-maintain.timer
systemctl list-timers hermes-teams-pipeline-maintain.timer
0 */12 * * * /usr/local/bin/hermes teams-pipeline maintain-subscriptions >> /var/log/hermes/teams-pipeline-maintain.log 2>&1
Make sure the cron environment has the MSGRAPH_* credentials. Simplest fix: source ~/.hermes/.env at the top of a wrapper script that crontab calls.
After you've set up the schedule, check renewal activity after the first scheduled run:
hermes teams-pipeline subscriptions # should show expirationDateTime advanced
hermes teams-pipeline maintain-subscriptions --dry-run # should show "0 expiring soon" most of the time
If you ever see your Graph webhook mysteriously "stop working" after exactly ~72 hours, this is the first thing to check: did the renewal job actually run?
hermes teams-pipeline list
hermes teams-pipeline list --status failed
hermes teams-pipeline show <job-id>
hermes teams-pipeline run <job-id>
hermes teams-pipeline fetch --meeting-id <meeting-id>
hermes teams-pipeline fetch --join-web-url "<join-url>"
Run these in order:
hermes teams-pipeline validate
hermes teams-pipeline token-health --force-refresh
hermes teams-pipeline subscriptions
Then trigger or wait for a real meeting event and confirm:
hermes teams-pipeline list
hermes teams-pipeline show <job-id>
hermes teams-pipeline maintain-subscriptions --dry-runhermes teams-pipeline list --status failedhermes teams-pipeline validateCheck:
msgraph_webhook is enabled/msgraph/webhookMSGRAPH_WEBHOOK_CLIENT_STATECheck:
ffmpeg availability if recording fallback is enabledCheck:
platforms.teams.enabled: truedelivery_modeincoming_webhook_url for webhook modechat_id or team_id plus channel_id for Graph modeCheck:
hermes teams-pipeline runmsgraph_webhook is enabled and reachable from the public internetMSGRAPH_WEBHOOK_CLIENT_STATE is set and matches subscriptionsffmpeg is installed if recording fallback is enabledhermes teams-pipeline validate returns an OK snapshothermes teams-pipeline token-health --force-refresh succeedsmaintain-subscriptions is scheduled (Hermes cron, systemd timer, or crontab — see Automating subscription renewal). Without this, Graph subscriptions silently expire within 72 hours.| Mode | Use when | Tradeoff |
|---|---|---|
incoming_webhook | you only need simple posting into Teams | simplest setup, less control |
graph | you need channel or chat posting through Graph | more control, more auth and target config |
Fill this out before rollout:
| Item | Value |
|---|---|
| Public notification URL | |
| Graph tenant ID | |
| Graph client ID | |
| Webhook client state | |
| Transcript resource subscription | |
| Recording resource subscription | |
| Teams delivery mode | |
| Teams chat ID or team/channel | |
| Notion database ID | |
| Linear team ID | |
| Store path override, if any | |
| Owner for daily checks |
Use this before changing the deployment:
| Question | Answer |
|---|---|
| Are we changing the public webhook URL? | |
| Are we rotating Graph credentials? | |
| Are we changing Teams delivery mode? | |
| Are we moving to a new Teams chat or channel? | |
| Do subscriptions need to be recreated or renewed? | |
| Do we need a fresh end-to-end verification run? |