a7a2b4efa4
Backend
- Per-chat album scope for Immich commands (search/latest/memory/...): new
allowed_album_ids on CommandTrackerListener, threaded listener/page kwargs
through ProviderCommandHandler.handle; PATCH listener-scope endpoint.
- /search and /find accept a trailing page number; Immich client search_smart
/ search_metadata take a page param.
- Immich person-asset lookup switched from removed GET /api/people/{id}/assets
to POST /api/search/metadata with personIds (fixes /person command and
auto_organize rules silently returning zero candidates on Immich 1.106+).
- Auto_organize rule now sets the target album's thumbnail to the first added
image when missing (falls back to any asset type); failures do not fail the
rule. add_assets_to_album surfaces the Immich error body on non-2xx.
- EventLog.user_id / action_id / action_name columns with defensive migration
+ backfill. Status query filters by user_id directly; Immich/webhook paths
emit user_id explicitly. action_runner writes an action_success/partial/
failed event on each non-dry-run.
- Dashboard DELETE /api/status/events (scoped to user_id) + rendering live
tracker/provider/action names via FK join with snapshot fallback.
- PATCH /api/users/{id} for username/role change with last-admin guard.
- Deletion protection returns structured {message, entity, blocked_by}
(ApiError carries .blockedBy; frontend opens BlockedByModal).
- Backup prepare-restore → AppSetting markers + atomic write of
pending_restore.json; lifespan hook applies on next startup and archives
under data/applied_restores/. apply-restart sends SIGTERM so the lifespan
shutdown runs; NOTIFY_BRIDGE_SUPERVISED env override gates the button.
Manual POST /api/backup/files (same format as scheduled).
- New periodic-summary test path reuses shared collect_scheduled_assets
(limit=0) so test and future production code go through one primitive.
- Per-receiver locale for Telegram test messages (resolves
TelegramChat.language_override per chat instead of applying the first
receiver's locale to everyone).
- Bounded concurrency (semaphores) in NotificationDispatcher._preload_asset_data
and _refresh_telegram_chat_titles; chat title sweep extended to 24h since
save_chat_from_webhook covers active chats opportunistically.
- Telegram poller detects the \"webhook is active\" 409 and auto-calls
deleteWebhook for bots whose DB update_mode is polling (throttled per bot).
- TelegramClient.get_chat added (CLAUDE.md rule 6); set_album_thumbnail added.
- Seeds: rename \"Default Commands\" → \"Default Immich Commands\";
track_assets_removed default False.
Frontend
- Global provider selector visible when there is only one provider.
- Clear-events button + i18n + ConfirmModal on the dashboard; new icons/
labels/filters/colors for action_success / action_partial / action_failed.
- Auto-select first available tracking/template/command/config + bot on
create forms (trackers, command-trackers, targets, template/command
configs).
- Telegram target disable_url_preview defaults to true.
- BlockedByModal wired into 8 deletion flows; fetchAuth helper for
multipart/binary calls (reuses api()'s refresh + ApiError mapping).
- Immich tracker 'Checking links' parallelised (concurrency cap 6).
- Backup page: pending-restore banner + Apply-now / Apply-later modal,
restarting overlay polling /api/health, manual 'Create backup' button.
- Command-trackers listener row gets an 'Edit album scope' modal with
inherit/explicit multiselect.
- Users page: Edit user modal (username + role).
- parseDate helper for consistent UTC date rendering.
Migrations / schema
- event_log: + user_id, action_id, action_name (+ backfill user_id from
notification_tracker).
- command_tracker_listener: + allowed_album_ids.
309 lines
12 KiB
Python
309 lines
12 KiB
Python
"""Telegram long-polling service for bots in polling mode.
|
|
|
|
Uses APScheduler to run getUpdates periodically for each bot
|
|
with update_mode == "polling". Processes updates identically
|
|
to the webhook handler (auto-save chat, dispatch commands).
|
|
|
|
Ref-counted: only starts/stops polling for bots that have active
|
|
CommandTrackerListeners with enabled CommandTrackers.
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
import logging
|
|
from typing import Any
|
|
|
|
from sqlmodel import select
|
|
from sqlmodel.ext.asyncio.session import AsyncSession
|
|
|
|
from notify_bridge_core.notifications.telegram.client import TelegramClient
|
|
|
|
from ..database.engine import get_engine
|
|
from ..database.models import CommandTracker, CommandTrackerListener, TelegramBot, TelegramChat
|
|
from ..services.telegram import save_chat_from_webhook
|
|
from .scheduler import get_scheduler
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
# Track last update_id per bot to use as offset
|
|
_last_update_id: dict[int, int] = {}
|
|
|
|
# Throttle auto-reclaim attempts so we don't hammer deleteWebhook when a
|
|
# stubborn external instance keeps re-setting the webhook. (bot_id → unix ts)
|
|
_last_webhook_reclaim_at: dict[int, float] = {}
|
|
_WEBHOOK_RECLAIM_COOLDOWN_SECONDS = 60.0
|
|
|
|
# Phrase Telegram uses in the 409 response description for the
|
|
# "webhook is active" conflict. Matched case-insensitively so we don't
|
|
# depend on exact wording.
|
|
_WEBHOOK_CONFLICT_PHRASE = "webhook is active"
|
|
|
|
|
|
async def _get_bot_ids_with_active_listeners() -> set[int]:
|
|
"""Return bot IDs that have at least one active command tracker listener.
|
|
|
|
A bot is "active" if there is a CommandTrackerListener with
|
|
listener_type="telegram_bot" pointing to it, AND the associated
|
|
CommandTracker is enabled.
|
|
"""
|
|
engine = get_engine()
|
|
async with AsyncSession(engine) as session:
|
|
result = await session.exec(
|
|
select(CommandTrackerListener).where(
|
|
CommandTrackerListener.listener_type == "telegram_bot"
|
|
)
|
|
)
|
|
listeners = result.all()
|
|
|
|
active_bot_ids: set[int] = set()
|
|
tracker_ids = list({l.command_tracker_id for l in listeners})
|
|
if tracker_ids:
|
|
tracker_result = await session.exec(
|
|
select(CommandTracker).where(
|
|
CommandTracker.id.in_(tracker_ids),
|
|
CommandTracker.enabled == True, # noqa: E712
|
|
)
|
|
)
|
|
enabled_tracker_ids = {t.id for t in tracker_result.all()}
|
|
for listener in listeners:
|
|
if listener.command_tracker_id in enabled_tracker_ids:
|
|
active_bot_ids.add(listener.listener_id)
|
|
|
|
return active_bot_ids
|
|
|
|
|
|
async def start_command_listener_polling() -> None:
|
|
"""Schedule polling jobs only for bots with active command tracker listeners."""
|
|
active_bot_ids = await _get_bot_ids_with_active_listeners()
|
|
if not active_bot_ids:
|
|
_LOGGER.info("No bots with active command listeners to poll")
|
|
return
|
|
|
|
engine = get_engine()
|
|
async with AsyncSession(engine) as session:
|
|
result = await session.exec(
|
|
select(TelegramBot).where(
|
|
TelegramBot.update_mode == "polling",
|
|
TelegramBot.id.in_(active_bot_ids),
|
|
)
|
|
)
|
|
bots = result.all()
|
|
|
|
for bot in bots:
|
|
schedule_bot_polling(bot.id)
|
|
|
|
_LOGGER.info("Started command listener polling for %d bot(s)", len(bots))
|
|
|
|
|
|
async def start_bot_polling() -> None:
|
|
"""Schedule polling jobs for all bots with update_mode == 'polling'.
|
|
|
|
Deprecated: prefer start_command_listener_polling() which only starts
|
|
bots with active command tracker listeners.
|
|
"""
|
|
await start_command_listener_polling()
|
|
|
|
|
|
async def start_bot_if_needed(bot_id: int) -> None:
|
|
"""Start polling for a bot if it has active listeners and is not already running."""
|
|
engine = get_engine()
|
|
async with AsyncSession(engine) as session:
|
|
bot = await session.get(TelegramBot, bot_id)
|
|
if not bot or bot.update_mode != "polling":
|
|
return
|
|
|
|
active_bot_ids = await _get_bot_ids_with_active_listeners()
|
|
if bot_id in active_bot_ids:
|
|
schedule_bot_polling(bot_id)
|
|
|
|
|
|
async def stop_bot_if_unused(bot_id: int) -> None:
|
|
"""Stop polling for a bot if it has no enabled command tracker listeners."""
|
|
active_bot_ids = await _get_bot_ids_with_active_listeners()
|
|
if bot_id not in active_bot_ids:
|
|
unschedule_bot_polling(bot_id)
|
|
|
|
|
|
def schedule_bot_polling(bot_id: int) -> None:
|
|
"""Add a polling job for a bot (idempotent)."""
|
|
scheduler = get_scheduler()
|
|
job_id = f"telegram_poll_{bot_id}"
|
|
if scheduler.get_job(job_id):
|
|
return
|
|
scheduler.add_job(
|
|
_poll_bot,
|
|
"interval",
|
|
seconds=3,
|
|
id=job_id,
|
|
args=[bot_id],
|
|
replace_existing=True,
|
|
max_instances=1,
|
|
)
|
|
_LOGGER.info("Started polling for bot %d", bot_id)
|
|
|
|
|
|
def unschedule_bot_polling(bot_id: int) -> None:
|
|
"""Remove polling job for a bot."""
|
|
scheduler = get_scheduler()
|
|
job_id = f"telegram_poll_{bot_id}"
|
|
if scheduler.get_job(job_id):
|
|
scheduler.remove_job(job_id)
|
|
_LOGGER.info("Stopped polling for bot %d", bot_id)
|
|
|
|
|
|
async def _handle_webhook_conflict(bot_id: int, bot_token: str, description: str) -> None:
|
|
"""Reclaim a bot stuck behind an active webhook set by another instance.
|
|
|
|
Telegram's ``getUpdates`` returns 409 ``Conflict: can't use getUpdates
|
|
method while webhook is active`` whenever a webhook is currently
|
|
registered for the bot. Since this bot row has ``update_mode="polling"``
|
|
in our DB (that's the only reason we're polling it), the user's intent
|
|
is polling, so we drop the webhook and resume. Throttled to once per
|
|
minute per bot so a rival instance constantly re-registering the
|
|
webhook doesn't trigger a reclaim storm.
|
|
"""
|
|
import time
|
|
now = time.time()
|
|
last = _last_webhook_reclaim_at.get(bot_id, 0.0)
|
|
if now - last < _WEBHOOK_RECLAIM_COOLDOWN_SECONDS:
|
|
# Already logged recently; stay quiet until cooldown expires so the
|
|
# user gets one clear warning line per minute, not one every 3s.
|
|
return
|
|
_last_webhook_reclaim_at[bot_id] = now
|
|
|
|
from .http_session import get_http_session
|
|
http = await get_http_session()
|
|
client = TelegramClient(http, bot_token)
|
|
|
|
# Surface which URL stole the bot so the user can tell where it came from.
|
|
conflicting_url = ""
|
|
try:
|
|
info = await client.get_webhook_info()
|
|
if info.get("success"):
|
|
conflicting_url = info.get("result", {}).get("url", "") or ""
|
|
except Exception as err: # noqa: BLE001
|
|
_LOGGER.debug("getWebhookInfo during conflict recovery failed: %s", err)
|
|
|
|
_LOGGER.warning(
|
|
"Bot %d: webhook is active (url=%r) but this instance is in polling "
|
|
"mode — calling deleteWebhook to reclaim. Telegram said: %s",
|
|
bot_id, conflicting_url, description,
|
|
)
|
|
|
|
try:
|
|
del_result = await client.delete_webhook()
|
|
if del_result.get("success"):
|
|
_LOGGER.warning(
|
|
"Bot %d: webhook cleared; polling will resume on next tick",
|
|
bot_id,
|
|
)
|
|
# Reset offset so we don't skip updates that accumulated during the
|
|
# conflict window (Telegram held them until a client acknowledged).
|
|
_last_update_id.pop(bot_id, None)
|
|
else:
|
|
_LOGGER.error(
|
|
"Bot %d: deleteWebhook failed: %s",
|
|
bot_id, del_result.get("error"),
|
|
)
|
|
except Exception as err: # noqa: BLE001
|
|
_LOGGER.error("Bot %d: deleteWebhook raised: %s", bot_id, err)
|
|
|
|
|
|
async def _poll_bot(bot_id: int) -> None:
|
|
"""Fetch updates from Telegram and process them."""
|
|
engine = get_engine()
|
|
|
|
# Eagerly load bot data and close session before aiohttp work
|
|
# (cannot nest aiohttp inside active SQLAlchemy async session)
|
|
async with AsyncSession(engine) as session:
|
|
bot = await session.get(TelegramBot, bot_id)
|
|
if not bot or bot.update_mode != "polling":
|
|
unschedule_bot_polling(bot_id)
|
|
return
|
|
# Copy attributes before session closes to avoid detached-instance errors
|
|
from types import SimpleNamespace
|
|
bot_token = bot.token
|
|
bot_obj = SimpleNamespace(id=bot.id, name=bot.name, token=bot.token)
|
|
|
|
offset = _last_update_id.get(bot_id, 0)
|
|
|
|
try:
|
|
from .http_session import get_http_session
|
|
http = await get_http_session()
|
|
client = TelegramClient(http, bot_token)
|
|
result = await client.get_updates(
|
|
offset=offset + 1 if offset else None, limit=50,
|
|
)
|
|
if not result.get("success"):
|
|
err_text = str(result.get("error") or "")
|
|
# Detect the webhook-is-active conflict: another instance (or a
|
|
# stale registration) owns this bot's delivery, so getUpdates
|
|
# returns 409 and we get zero updates forever. Reclaim it —
|
|
# but only for bots the user explicitly set to polling mode.
|
|
if _WEBHOOK_CONFLICT_PHRASE in err_text.lower():
|
|
await _handle_webhook_conflict(bot_id, bot_token, err_text)
|
|
else:
|
|
_LOGGER.debug("Polling error for bot %d: %s", bot_id, err_text)
|
|
return
|
|
updates = result.get("result", [])
|
|
except Exception as e:
|
|
_LOGGER.debug("Polling error for bot %d: %s", bot_id, e)
|
|
return
|
|
|
|
if not updates:
|
|
return
|
|
|
|
# Update offset to latest
|
|
_last_update_id[bot_id] = updates[-1]["update_id"]
|
|
|
|
# Process each update
|
|
from ..commands.handler import handle_command, send_media_group, send_reply
|
|
|
|
for update in updates:
|
|
message = update.get("message")
|
|
if not message:
|
|
continue
|
|
|
|
chat_info = message.get("chat", {})
|
|
chat_id = str(chat_info.get("id", ""))
|
|
text = message.get("text", "")
|
|
from_user = message.get("from", {})
|
|
msg_language = from_user.get("language_code", "")
|
|
|
|
if not chat_id:
|
|
continue
|
|
|
|
# Auto-persist chat (fresh session per save)
|
|
try:
|
|
async with AsyncSession(engine) as save_session:
|
|
await save_chat_from_webhook(save_session, bot_obj.id, chat_info, language_code=msg_language)
|
|
await save_session.commit()
|
|
except Exception:
|
|
_LOGGER.debug("Failed to auto-save chat %s", chat_id, exc_info=True)
|
|
|
|
# Dispatch commands (only if chat has commands enabled)
|
|
if text and text.startswith("/"):
|
|
try:
|
|
async with AsyncSession(engine) as cmd_session:
|
|
chat_row = (await cmd_session.exec(
|
|
select(TelegramChat).where(
|
|
TelegramChat.bot_id == bot_obj.id,
|
|
TelegramChat.chat_id == chat_id,
|
|
)
|
|
)).first()
|
|
if not chat_row or not chat_row.commands_enabled:
|
|
continue
|
|
effective_lang = chat_row.language_override or msg_language
|
|
message_id = message.get("message_id")
|
|
responses = await handle_command(bot_obj, chat_id, text, language_code=effective_lang)
|
|
if responses:
|
|
for resp in responses:
|
|
if resp.text:
|
|
await send_reply(bot_token, chat_id, resp.text, reply_to_message_id=message_id)
|
|
if resp.media:
|
|
await send_media_group(bot_token, chat_id, resp.media, reply_to_message_id=message_id)
|
|
except Exception:
|
|
_LOGGER.error("Error handling command from bot %d", bot_id, exc_info=True)
|
|
|
|
|