Files
hermes-agent/plugins/teams_pipeline/runtime.py
T
kshitij 5fba236644 chore: ruff auto-fix PLR6201 resweep — tuple → set in membership tests (#27355)
Six days after #23937 (608 fixes) the codebase had accumulated 241 new
PLR6201 violations. Same mechanical `x in (...)` → `x in {...}` fix,
same zero-risk profile: set lookup is O(1) vs O(n) for tuple and the
two are semantically equivalent for hashable scalar membership tests.

All 241 instances fixed via `ruff check --select PLR6201 --fix
--unsafe-fixes`, zero remaining. Every changed value is a hashable
scalar (str/int/None/enum/signal); no risk of unhashable runtime
errors. No behavior change.

Test plan:
- 119 files changed, +244/-244 (net zero) — exactly one-line edits
- `ruff check` clean afterward
- Compile checks pass on the largest touched files (cli.py, run_agent.py,
  gateway/run.py, gateway/platforms/discord.py, model_tools.py)
- Subset broad test run on tests/gateway/ tests/hermes_cli/ tests/agent/
  tests/tools/: 18187 passed, 59 pre-existing failures (verified against
  origin/main with the same shape — identical failure count, identical
  category — all xdist test-order flakes unrelated to this change)

Follows the same template as PR #23937 ([tracker: #23972](https://github.com/NousResearch/hermes-agent/issues/23972)).
2026-05-17 02:29:41 -07:00

136 lines
4.8 KiB
Python

"""Gateway runtime wiring for the Teams meeting pipeline plugin."""
from __future__ import annotations
import logging
from typing import Any
from gateway.config import Platform
from plugins.teams_pipeline.pipeline import TeamsMeetingPipeline
from plugins.teams_pipeline.store import TeamsPipelineStore, resolve_teams_pipeline_store_path
from plugins.teams_pipeline.subscriptions import build_graph_client
logger = logging.getLogger(__name__)
def _teams_delivery_is_configured(teams_extra: dict[str, Any], teams_delivery: dict[str, Any]) -> bool:
delivery_mode = str(
teams_delivery.get("mode")
or teams_delivery.get("delivery_mode")
or teams_extra.get("delivery_mode")
or ""
).strip().lower()
if delivery_mode == "incoming_webhook":
return bool(
teams_delivery.get("incoming_webhook_url")
or teams_extra.get("incoming_webhook_url")
)
if delivery_mode == "graph":
chat_id = teams_delivery.get("chat_id") or teams_extra.get("chat_id")
team_id = teams_delivery.get("team_id") or teams_extra.get("team_id")
channel_id = teams_delivery.get("channel_id") or teams_extra.get("channel_id")
return bool(chat_id or (team_id and channel_id))
return False
def build_pipeline_runtime_config(gateway_config: Any) -> dict[str, Any]:
"""Build pipeline config from gateway platform config.
Pipeline-specific knobs live under ``teams.extra.meeting_pipeline`` while
Teams delivery continues to source its target details from the existing
Teams platform config.
"""
teams_config = gateway_config.platforms.get(Platform("teams"))
teams_extra = dict((teams_config.extra or {}) if teams_config else {})
pipeline_config = dict(teams_extra.get("meeting_pipeline") or {})
if teams_config and teams_config.enabled:
teams_delivery = dict(pipeline_config.get("teams_delivery") or {})
delivery_mode = str(teams_extra.get("delivery_mode") or "").strip()
if delivery_mode:
teams_delivery["mode"] = delivery_mode
for key in (
"incoming_webhook_url",
"access_token",
"team_id",
"channel_id",
"chat_id",
):
value = teams_extra.get(key)
if value not in {None, ""}:
teams_delivery[key] = value
if teams_delivery:
teams_delivery["enabled"] = _teams_delivery_is_configured(teams_extra, teams_delivery)
pipeline_config["teams_delivery"] = teams_delivery
return pipeline_config
def build_pipeline_runtime(gateway: Any) -> TeamsMeetingPipeline:
teams_sender = None
teams_config = gateway.config.platforms.get(Platform("teams"))
pipeline_config = build_pipeline_runtime_config(gateway.config)
teams_delivery = dict(pipeline_config.get("teams_delivery") or {})
if teams_config and teams_config.enabled and teams_delivery.get("enabled"):
try:
from plugins.platforms.teams.adapter import TeamsSummaryWriter
except ImportError:
logger.debug(
"TeamsSummaryWriter unavailable; Teams outbound delivery remains disabled until the adapter layer is present."
)
else:
teams_sender = TeamsSummaryWriter(platform_config=teams_config)
return TeamsMeetingPipeline(
graph_client=build_graph_client(),
store=TeamsPipelineStore(resolve_teams_pipeline_store_path()),
config=pipeline_config,
teams_sender=teams_sender,
)
def bind_gateway_runtime(gateway: Any) -> bool:
"""Attach the Teams pipeline runtime to the msgraph webhook adapter."""
adapter = gateway.adapters.get(Platform.MSGRAPH_WEBHOOK)
if adapter is None:
return False
if getattr(gateway, "_teams_pipeline_runtime", None) is not None:
return True
try:
runtime = build_pipeline_runtime(gateway)
except Exception as exc:
error_message = str(exc)
gateway._teams_pipeline_runtime_error = error_message
logger.warning(
"Teams pipeline runtime unavailable: %s. Installing a drop-scheduler "
"so Graph notifications ack cleanly without piling up unbound.",
error_message,
)
async def _drop(notification: dict[str, Any], event: Any) -> None:
logger.debug(
"Dropping Graph notification because runtime is unavailable: id=%s resource=%s",
notification.get("id"),
notification.get("resource"),
)
adapter.set_notification_scheduler(_drop)
return False
async def _schedule(notification: dict[str, Any], event: Any) -> None:
await runtime.run_notification(notification)
adapter.set_notification_scheduler(_schedule)
gateway._teams_pipeline_runtime = runtime
gateway._teams_pipeline_runtime_error = None
return True