feat: стандартизація обробки повідомлень для всіх агентів

- Створено AgentConfig для уніфікації конфігурації агентів
- Додано універсальні функції: process_photo(), process_document(), process_voice()
- Створено handle_telegram_webhook() для будь-якого агента
- Рефакторинг telegram_webhook() та helion_telegram_webhook()
- Оновлено process_voice() для використання STT Service
- Додано реєстр агентів AGENT_REGISTRY
- Оновлено health endpoint для показу всіх агентів
- Додано інструкції для додавання нових агентів
This commit is contained in:
Apple
2025-11-21 01:50:18 -08:00
parent 28e0103301
commit 830d13b5a2

View File

@@ -8,6 +8,7 @@ import httpx
from pathlib import Path
from typing import Dict, Any, Optional
from datetime import datetime
from dataclasses import dataclass
from fastapi import APIRouter, HTTPException
from pydantic import BaseModel
@@ -31,63 +32,117 @@ router = APIRouter()
# ========================================
# Agent Configuration
# ========================================
@dataclass
class AgentConfig:
"""Конфігурація агента для стандартизації обробки повідомлень"""
agent_id: str
name: str
prompt_path: str
telegram_token_env: str
default_prompt: str
system_prompt: str = "" # Буде встановлено після завантаження
def load_prompt(self) -> str:
"""Завантажити system prompt з файлу"""
try:
p = Path(self.prompt_path)
if not p.exists():
logger.warning(f"{self.name} prompt file not found: {self.prompt_path}")
return self.default_prompt
prompt = p.read_text(encoding="utf-8")
logger.info(f"{self.name} system prompt loaded ({len(prompt)} chars)")
return prompt
except Exception as e:
logger.error(f"Error loading {self.name} prompt: {e}")
return self.default_prompt
def get_telegram_token(self) -> Optional[str]:
"""Отримати Telegram токен агента"""
return os.getenv(self.telegram_token_env)
def load_agent_config(agent_id: str, name: str, prompt_path: str,
telegram_token_env: str, default_prompt: str) -> AgentConfig:
"""Створити та завантажити конфігурацію агента"""
config = AgentConfig(
agent_id=agent_id,
name=name,
prompt_path=prompt_path,
telegram_token_env=telegram_token_env,
default_prompt=default_prompt,
system_prompt="" # Тимчасове значення
)
# Завантажити prompt
config.system_prompt = config.load_prompt()
return config
# ========================================
# Agent Configurations
# ========================================
# DAARWIZZ Configuration
# ========================================
DAARWIZZ_NAME = os.getenv("DAARWIZZ_NAME", "DAARWIZZ")
DAARWIZZ_PROMPT_PATH = os.getenv(
"DAARWIZZ_PROMPT_PATH",
str(Path(__file__).parent / "daarwizz_prompt.txt"),
DAARWIZZ_CONFIG = load_agent_config(
agent_id="daarwizz",
name=os.getenv("DAARWIZZ_NAME", "DAARWIZZ"),
prompt_path=os.getenv(
"DAARWIZZ_PROMPT_PATH",
str(Path(__file__).parent / "daarwizz_prompt.txt"),
),
telegram_token_env="TELEGRAM_BOT_TOKEN",
default_prompt=f"Ти — {os.getenv('DAARWIZZ_NAME', 'DAARWIZZ')}, AI-агент екосистеми DAARION.city. Допомагай учасникам з DAO-процесами."
)
def load_daarwizz_prompt() -> str:
"""Load DAARWIZZ system prompt from file"""
try:
p = Path(DAARWIZZ_PROMPT_PATH)
if not p.exists():
logger.warning(f"DAARWIZZ prompt file not found: {DAARWIZZ_PROMPT_PATH}")
return f"Ти — {DAARWIZZ_NAME}, AI-агент екосистеми DAARION.city. Допомагай учасникам з DAO-процесами."
prompt = p.read_text(encoding="utf-8")
logger.info(f"DAARWIZZ system prompt loaded ({len(prompt)} chars)")
return prompt
except Exception as e:
logger.error(f"Error loading DAARWIZZ prompt: {e}")
return f"Ти — {DAARWIZZ_NAME}, AI-агент екосистеми DAARION.city."
DAARWIZZ_SYSTEM_PROMPT = load_daarwizz_prompt()
# ========================================
# HELION Configuration
# ========================================
HELION_NAME = os.getenv("HELION_NAME", "Helion")
HELION_PROMPT_PATH = os.getenv(
"HELION_PROMPT_PATH",
str(Path(__file__).parent / "helion_prompt.txt"),
HELION_CONFIG = load_agent_config(
agent_id="helion",
name=os.getenv("HELION_NAME", "Helion"),
prompt_path=os.getenv(
"HELION_PROMPT_PATH",
str(Path(__file__).parent / "helion_prompt.txt"),
),
telegram_token_env="HELION_TELEGRAM_BOT_TOKEN",
default_prompt=f"Ти — {os.getenv('HELION_NAME', 'Helion')}, AI-агент платформи Energy Union. Допомагай учасникам з технологіями та токеномікою."
)
# Registry of all agents (для легкого додавання нових агентів)
#
# Щоб додати нового агента:
# 1. Створіть конфігурацію через load_agent_config():
# NEW_AGENT_CONFIG = load_agent_config(
# agent_id="new_agent",
# name=os.getenv("NEW_AGENT_NAME", "New Agent"),
# prompt_path=os.getenv("NEW_AGENT_PROMPT_PATH", str(Path(__file__).parent / "new_agent_prompt.txt")),
# telegram_token_env="NEW_AGENT_TELEGRAM_BOT_TOKEN",
# default_prompt="Ти — New Agent, AI-агент..."
# )
# 2. Додайте до реєстру:
# AGENT_REGISTRY["new_agent"] = NEW_AGENT_CONFIG
# 3. Створіть endpoint (опціонально, якщо потрібен окремий webhook):
# @router.post("/new_agent/telegram/webhook")
# async def new_agent_telegram_webhook(update: TelegramUpdate):
# return await handle_telegram_webhook(NEW_AGENT_CONFIG, update)
#
# Новий агент автоматично отримає:
# - Обробку фото через Swapper vision-8b
# - Обробку PDF документів
# - Обробку голосових повідомлень (коли буде реалізовано)
# - RAG запити по документам
# - Memory context
AGENT_REGISTRY: Dict[str, AgentConfig] = {
"daarwizz": DAARWIZZ_CONFIG,
"helion": HELION_CONFIG,
}
def load_helion_prompt() -> str:
"""Load Helion system prompt from file"""
try:
p = Path(HELION_PROMPT_PATH)
if not p.exists():
logger.warning(f"Helion prompt file not found: {HELION_PROMPT_PATH}")
return f"Ти — {HELION_NAME}, AI-агент платформи Energy Union. Допомагай учасникам з технологіями та токеномікою."
prompt = p.read_text(encoding="utf-8")
logger.info(f"Helion system prompt loaded ({len(prompt)} chars)")
return prompt
except Exception as e:
logger.error(f"Error loading Helion prompt: {e}")
return f"Ти — {HELION_NAME}, AI-агент платформи Energy Union."
HELION_SYSTEM_PROMPT = load_helion_prompt()
# Backward compatibility
DAARWIZZ_NAME = DAARWIZZ_CONFIG.name
DAARWIZZ_SYSTEM_PROMPT = DAARWIZZ_CONFIG.system_prompt
HELION_NAME = HELION_CONFIG.name
HELION_SYSTEM_PROMPT = HELION_CONFIG.system_prompt
# ========================================
@@ -126,6 +181,668 @@ def get_dao_id(chat_id: str, source: str) -> str:
return CHAT_TO_DAO.get(key, CHAT_TO_DAO["default"])
# ========================================
# Helper Functions
# ========================================
async def send_telegram_message(chat_id: str, text: str, bot_token: Optional[str] = None) -> bool:
"""
Відправити повідомлення в Telegram.
Args:
chat_id: ID чату
text: Текст повідомлення
bot_token: Telegram bot token (якщо None, використовується TELEGRAM_BOT_TOKEN)
Returns:
True якщо успішно, False інакше
"""
try:
token = bot_token or os.getenv("TELEGRAM_BOT_TOKEN")
if not token:
logger.error("TELEGRAM_BOT_TOKEN not set")
return False
url = f"https://api.telegram.org/bot{token}/sendMessage"
payload = {
"chat_id": chat_id,
"text": text,
"parse_mode": "Markdown"
}
async with httpx.AsyncClient() as client:
response = await client.post(url, json=payload, timeout=10.0)
response.raise_for_status()
return True
except Exception as e:
logger.error(f"Failed to send Telegram message: {e}")
return False
async def get_telegram_file_path(file_id: str, bot_token: Optional[str] = None) -> Optional[str]:
"""
Отримати шлях до файлу з Telegram API.
Args:
file_id: ID файлу з Telegram
bot_token: Telegram bot token (якщо None, використовується TELEGRAM_BOT_TOKEN)
Returns:
Шлях до файлу або None
"""
try:
token = bot_token or os.getenv("TELEGRAM_BOT_TOKEN")
if not token:
logger.error("TELEGRAM_BOT_TOKEN not set")
return None
url = f"https://api.telegram.org/bot{token}/getFile"
params = {"file_id": file_id}
async with httpx.AsyncClient() as client:
response = await client.get(url, params=params, timeout=10.0)
response.raise_for_status()
data = response.json()
if data.get("ok"):
return data.get("result", {}).get("file_path")
return None
except Exception as e:
logger.error(f"Failed to get Telegram file path: {e}")
return None
def format_qa_response(qa_list: list) -> str:
"""Форматувати список питань-відповідей для Telegram"""
if not qa_list:
return "Немає питань-відповідей."
result = "📋 **Питання та відповіді:**\n\n"
for i, qa in enumerate(qa_list, 1):
question = qa.get("question", "") if isinstance(qa, dict) else getattr(qa, "question", "")
answer = qa.get("answer", "") if isinstance(qa, dict) else getattr(qa, "answer", "")
result += f"**{i}. {question}**\n{answer}\n\n"
return result.strip()
def format_markdown_response(markdown: str) -> str:
"""Форматувати markdown відповідь для Telegram"""
if len(markdown) > TELEGRAM_SAFE_LENGTH:
return markdown[:TELEGRAM_SAFE_LENGTH] + "\n\n_... (відповідь обрізано)_"
return markdown
def format_chunks_response(chunks: list) -> str:
"""Форматувати список чанків для Telegram"""
if not chunks:
return "Немає фрагментів."
result = f"📄 **Знайдено {len(chunks)} фрагментів:**\n\n"
for i, chunk in enumerate(chunks[:5], 1): # Показуємо тільки перші 5
text = chunk.get("text", "") if isinstance(chunk, dict) else str(chunk)
if len(text) > 200:
text = text[:200] + "..."
result += f"**{i}.** {text}\n\n"
if len(chunks) > 5:
result += f"_... та ще {len(chunks) - 5} фрагментів_"
return result.strip()
# ========================================
# Universal Message Processing Functions
# ========================================
async def process_photo(
agent_config: AgentConfig,
update: TelegramUpdate,
chat_id: str,
user_id: str,
username: str,
dao_id: str,
photo: Dict[str, Any]
) -> Dict[str, Any]:
"""
Універсальна функція для обробки фото для будь-якого агента.
Args:
agent_config: Конфігурація агента
update: Telegram update об'єкт
chat_id: ID чату
user_id: ID користувача
username: Ім'я користувача
dao_id: ID DAO
photo: Об'єкт фото з Telegram
Returns:
Dict з результатом обробки
"""
# Telegram sends multiple sizes, get the largest one (last in array)
photo_obj = photo[-1] if isinstance(photo, list) else photo
file_id = photo_obj.get("file_id") if isinstance(photo_obj, dict) else None
if not file_id:
return {"ok": False, "error": "No file_id in photo"}
logger.info(f"{agent_config.name}: Photo from {username} (tg:{user_id}), file_id: {file_id}")
try:
# Get file path from Telegram
telegram_token = agent_config.get_telegram_token()
if not telegram_token:
return {"ok": False, "error": f"Telegram token not configured for {agent_config.name}"}
file_path = await get_telegram_file_path(file_id, telegram_token)
if not file_path:
raise HTTPException(status_code=400, detail="Failed to get file from Telegram")
# Build file URL
file_url = f"https://api.telegram.org/file/bot{telegram_token}/{file_path}"
# Send to Router with specialist_vision_8b model (Swapper)
router_request = {
"message": f"Опиши це зображення детально: {file_url}",
"mode": "chat",
"agent": agent_config.agent_id,
"metadata": {
"source": "telegram",
"dao_id": dao_id,
"user_id": f"tg:{user_id}",
"session_id": f"tg:{chat_id}:{dao_id}",
"username": username,
"chat_id": chat_id,
"file_id": file_id,
"file_url": file_url,
"has_image": True,
"use_llm": "specialist_vision_8b",
},
"context": {
"agent_name": agent_config.name,
"system_prompt": agent_config.system_prompt,
},
}
# Send to Router
logger.info(f"{agent_config.name}: Sending photo to Router with vision-8b: file_url={file_url[:50]}...")
response = await send_to_router(router_request)
# Extract response
if isinstance(response, dict) and response.get("ok"):
answer_text = response.get("data", {}).get("text") or response.get("response", "")
if answer_text:
# Photo processed successfully
await send_telegram_message(
chat_id,
f"✅ **Фото оброблено**\n\n{answer_text}",
telegram_token
)
# Save to memory for context
await memory_client.save_chat_turn(
agent_id=agent_config.agent_id,
team_id=dao_id,
user_id=f"tg:{user_id}",
message=f"[Photo: {file_id}]",
response=answer_text,
channel_id=chat_id,
scope="short_term"
)
return {"ok": True, "agent": agent_config.agent_id, "model": "specialist_vision_8b"}
else:
await send_telegram_message(
chat_id,
"Фото оброблено, але не вдалося отримати опис.",
telegram_token
)
return {"ok": False, "error": "No description in response"}
else:
error_msg = response.get("error", "Unknown error") if isinstance(response, dict) else "Router error"
logger.error(f"{agent_config.name}: Vision-8b error: {error_msg}")
await send_telegram_message(
chat_id,
f"Вибач, не вдалося обробити фото: {error_msg}",
telegram_token
)
return {"ok": False, "error": error_msg}
except Exception as e:
logger.error(f"{agent_config.name}: Photo processing failed: {e}", exc_info=True)
telegram_token = agent_config.get_telegram_token()
await send_telegram_message(
chat_id,
"Вибач, не вдалося обробити фото. Переконайся, що Swapper Service з vision-8b моделлю запущений.",
telegram_token
)
return {"ok": False, "error": "Photo processing failed"}
async def process_document(
agent_config: AgentConfig,
update: TelegramUpdate,
chat_id: str,
user_id: str,
username: str,
dao_id: str,
document: Dict[str, Any]
) -> Dict[str, Any]:
"""
Універсальна функція для обробки документів (PDF) для будь-якого агента.
Args:
agent_config: Конфігурація агента
update: Telegram update об'єкт
chat_id: ID чату
user_id: ID користувача
username: Ім'я користувача
dao_id: ID DAO
document: Об'єкт документа з Telegram
Returns:
Dict з результатом обробки
"""
mime_type = document.get("mime_type", "")
file_name = document.get("file_name", "")
file_id = document.get("file_id")
# Check if it's a PDF
is_pdf = (
mime_type == "application/pdf" or
(mime_type.startswith("application/") and file_name.lower().endswith(".pdf"))
)
if is_pdf and file_id:
logger.info(f"{agent_config.name}: PDF document from {username} (tg:{user_id}), file_id: {file_id}, file_name: {file_name}")
try:
telegram_token = agent_config.get_telegram_token()
if not telegram_token:
return {"ok": False, "error": f"Telegram token not configured for {agent_config.name}"}
file_path = await get_telegram_file_path(file_id, telegram_token)
if not file_path:
raise HTTPException(status_code=400, detail="Failed to get file from Telegram")
file_url = f"https://api.telegram.org/file/bot{telegram_token}/{file_path}"
await send_telegram_message(chat_id, "📄 Обробляю PDF-документ... Це може зайняти кілька секунд.", telegram_token)
session_id = f"telegram:{chat_id}"
result = await parse_document(
session_id=session_id,
doc_url=file_url,
file_name=file_name,
dao_id=dao_id,
user_id=f"tg:{user_id}",
output_mode="qa_pairs",
metadata={"username": username, "chat_id": chat_id}
)
if not result.success:
await send_telegram_message(chat_id, f"Вибач, не вдалося обробити документ: {result.error}", telegram_token)
return {"ok": False, "error": result.error}
# Format response for Telegram
answer_text = ""
if result.qa_pairs:
qa_list = [{"question": qa.question, "answer": qa.answer} for qa in result.qa_pairs]
answer_text = format_qa_response(qa_list)
elif result.markdown:
answer_text = format_markdown_response(result.markdown)
elif result.chunks_meta and result.chunks_meta.get("chunks"):
chunks = result.chunks_meta.get("chunks", [])
answer_text = format_chunks_response(chunks)
else:
answer_text = "✅ Документ успішно оброблено, але формат відповіді не розпізнано."
if not answer_text.endswith("_"):
answer_text += "\n\n💡 _Використай /ingest для імпорту документа у RAG_"
logger.info(f"{agent_config.name}: PDF parsing result: {len(answer_text)} chars, doc_id={result.doc_id}")
await send_telegram_message(chat_id, answer_text, telegram_token)
return {"ok": True, "agent": "parser", "mode": "doc_parse", "doc_id": result.doc_id}
except Exception as e:
logger.error(f"{agent_config.name}: PDF processing failed: {e}", exc_info=True)
telegram_token = agent_config.get_telegram_token()
await send_telegram_message(chat_id, "Вибач, не вдалося обробити PDF-документ. Переконайся, що файл не пошкоджений.", telegram_token)
return {"ok": False, "error": "PDF processing failed"}
elif document and not is_pdf:
telegram_token = agent_config.get_telegram_token()
await send_telegram_message(chat_id, "Наразі підтримуються тільки PDF-документи. Інші формати (docx, zip, тощо) будуть додані пізніше.", telegram_token)
return {"ok": False, "error": "Unsupported document type"}
return {"ok": False, "error": "No document to process"}
async def process_voice(
agent_config: AgentConfig,
update: TelegramUpdate,
chat_id: str,
user_id: str,
username: str,
dao_id: str,
media_obj: Dict[str, Any]
) -> Dict[str, Any]:
"""
Універсальна функція для обробки голосових повідомлень для будь-якого агента.
Використовує STT Service для розпізнавання мовлення.
Args:
agent_config: Конфігурація агента
update: Telegram update об'єкт
chat_id: ID чату
user_id: ID користувача
username: Ім'я користувача
dao_id: ID DAO
media_obj: Об'єкт голосового повідомлення з Telegram
Returns:
Dict з результатом обробки та розпізнаним текстом
"""
file_id = media_obj.get("file_id") if media_obj else None
if not file_id:
return {"ok": False, "error": "No file_id in voice/audio/video_note"}
logger.info(f"{agent_config.name}: Voice message from {username} (tg:{user_id}), file_id: {file_id}")
try:
telegram_token = agent_config.get_telegram_token()
if not telegram_token:
return {"ok": False, "error": f"Telegram token not configured for {agent_config.name}"}
# Отримуємо файл з Telegram
file_path = await get_telegram_file_path(file_id, telegram_token)
if not file_path:
raise HTTPException(status_code=400, detail="Failed to get file from Telegram")
# Завантажуємо файл
file_url = f"https://api.telegram.org/file/bot{telegram_token}/{file_path}"
async with httpx.AsyncClient(timeout=30.0) as client:
file_resp = await client.get(file_url)
file_resp.raise_for_status()
audio_bytes = file_resp.content
# Відправляємо в STT-сервіс
stt_service_url = os.getenv("STT_SERVICE_URL", "http://stt-service:9000")
files = {"file": ("voice.ogg", audio_bytes, "audio/ogg")}
async with httpx.AsyncClient(timeout=60.0) as client:
stt_resp = await client.post(f"{stt_service_url}/stt", files=files)
stt_resp.raise_for_status()
stt_data = stt_resp.json()
text = stt_data.get("text", "")
if not text:
await send_telegram_message(
chat_id,
"Вибач, не вдалося розпізнати голосове повідомлення. Спробуй надіслати текстом.",
telegram_token
)
return {"ok": False, "error": "STT returned empty text"}
logger.info(f"{agent_config.name}: STT result: {text[:100]}...")
# Повертаємо розпізнаний текст для подальшої обробки
return {"ok": True, "text": text, "agent": agent_config.agent_id, "mode": "voice_stt"}
except Exception as e:
logger.error(f"{agent_config.name}: Voice processing failed: {e}", exc_info=True)
telegram_token = agent_config.get_telegram_token()
await send_telegram_message(
chat_id,
"Вибач, не вдалося розпізнати голосове повідомлення. Спробуй надіслати текстом.",
telegram_token
)
return {"ok": False, "error": "Voice processing failed"}
# ========================================
# Universal Telegram Webhook Handler
# ========================================
async def handle_telegram_webhook(
agent_config: AgentConfig,
update: TelegramUpdate
) -> Dict[str, Any]:
"""
Універсальна функція для обробки Telegram webhook для будь-якого агента.
Args:
agent_config: Конфігурація агента
update: Telegram update об'єкт
Returns:
Dict з результатом обробки
"""
if not update.message:
raise HTTPException(status_code=400, detail="No message in update")
# Extract message details
from_user = update.message.get("from", {})
chat = update.message.get("chat", {})
user_id = str(from_user.get("id", "unknown"))
chat_id = str(chat.get("id", "unknown"))
username = from_user.get("username", "")
# Get DAO ID for this chat
dao_id = get_dao_id(chat_id, "telegram")
telegram_token = agent_config.get_telegram_token()
if not telegram_token:
raise HTTPException(status_code=500, detail=f"Telegram token not configured for {agent_config.name}")
# Check for /ingest command
text = update.message.get("text", "")
if text and text.strip().startswith("/ingest"):
session_id = f"telegram:{chat_id}"
# Check if there's a document in the message
document = update.message.get("document")
if document:
mime_type = document.get("mime_type", "")
file_name = document.get("file_name", "")
file_id = document.get("file_id")
is_pdf = (
mime_type == "application/pdf" or
(mime_type.startswith("application/") and file_name.lower().endswith(".pdf"))
)
if is_pdf and file_id:
try:
file_path = await get_telegram_file_path(file_id, telegram_token)
if file_path:
file_url = f"https://api.telegram.org/file/bot{telegram_token}/{file_path}"
await send_telegram_message(chat_id, "📥 Імпортую документ у RAG...", telegram_token)
result = await ingest_document(
session_id=session_id,
doc_url=file_url,
file_name=file_name,
dao_id=dao_id,
user_id=f"tg:{user_id}"
)
if result.success:
await send_telegram_message(
chat_id,
f"✅ **Документ імпортовано у RAG**\n\n"
f"📊 Фрагментів: {result.ingested_chunks}\n"
f"📁 DAO: {dao_id}\n\n"
f"Тепер ти можеш задавати питання по цьому документу!",
telegram_token
)
return {"ok": True, "chunks_count": result.ingested_chunks}
else:
await send_telegram_message(chat_id, f"Вибач, не вдалося імпортувати: {result.error}", telegram_token)
return {"ok": False, "error": result.error}
except Exception as e:
logger.error(f"{agent_config.name}: Ingest failed: {e}", exc_info=True)
await send_telegram_message(chat_id, "Вибач, не вдалося імпортувати документ.", telegram_token)
return {"ok": False, "error": "Ingest failed"}
# Try to get last parsed doc_id from session context
result = await ingest_document(
session_id=session_id,
dao_id=dao_id,
user_id=f"tg:{user_id}"
)
if result.success:
await send_telegram_message(
chat_id,
f"✅ **Документ імпортовано у RAG**\n\n"
f"📊 Фрагментів: {result.ingested_chunks}\n"
f"📁 DAO: {dao_id}\n\n"
f"Тепер ти можеш задавати питання по цьому документу!",
telegram_token
)
return {"ok": True, "chunks_count": result.ingested_chunks}
else:
await send_telegram_message(chat_id, "Спочатку надішли PDF-документ, а потім використай /ingest", telegram_token)
return {"ok": False, "error": result.error}
# Check if it's a document (PDF)
document = update.message.get("document")
if document:
result = await process_document(
agent_config, update, chat_id, user_id, username, dao_id, document
)
if result.get("ok"):
return result
# Check if it's a photo
photo = update.message.get("photo")
if photo:
result = await process_photo(
agent_config, update, chat_id, user_id, username, dao_id, photo
)
if result.get("ok"):
return result
# Check if it's a voice message
voice = update.message.get("voice")
audio = update.message.get("audio")
video_note = update.message.get("video_note")
text = ""
if voice or audio or video_note:
media_obj = voice or audio or video_note
result = await process_voice(
agent_config, update, chat_id, user_id, username, dao_id, media_obj
)
if result.get("ok") and result.get("text"):
# Отримали розпізнаний текст, продовжуємо обробку як текстове повідомлення
text = result.get("text")
elif result.get("ok"):
# STT успішний, але текст порожній
return result
else:
# Помилка STT
return result
# Get message text (якщо не було голосового повідомлення)
if not text:
text = update.message.get("text", "")
if not text:
raise HTTPException(status_code=400, detail="No text or voice in message")
logger.info(f"{agent_config.name} Telegram message from {username} (tg:{user_id}) in chat {chat_id}: {text[:50]}")
# Check if there's a document context for follow-up questions
session_id = f"telegram:{chat_id}"
doc_context = await get_doc_context(session_id)
# If there's a doc_id and the message looks like a question about the document
if doc_context and doc_context.doc_id:
# Check if it's a question (simple heuristic: contains question words or ends with ?)
is_question = (
"?" in text or
any(word in text.lower() for word in ["що", "як", "чому", "коли", "де", "хто", "чи"])
)
if is_question:
logger.info(f"{agent_config.name}: Follow-up question detected for doc_id={doc_context.doc_id}")
# Try RAG query first
rag_result = await ask_about_document(
session_id=session_id,
question=text,
doc_id=doc_context.doc_id,
dao_id=dao_id or doc_context.dao_id,
user_id=f"tg:{user_id}"
)
if rag_result.success and rag_result.answer:
# Truncate if too long for Telegram
answer = rag_result.answer
if len(answer) > TELEGRAM_SAFE_LENGTH:
answer = answer[:TELEGRAM_SAFE_LENGTH] + "\n\n_... (відповідь обрізано)_"
await send_telegram_message(chat_id, answer, telegram_token)
return {"ok": True, "agent": "parser", "mode": "rag_query"}
# Fall through to regular chat if RAG query fails
# Regular chat mode
# Fetch memory context
memory_context = await memory_client.get_context(
user_id=f"tg:{user_id}",
agent_id=agent_config.agent_id,
team_id=dao_id,
channel_id=chat_id,
limit=10
)
# Build request to Router
router_request = {
"message": text,
"mode": "chat",
"agent": agent_config.agent_id,
"metadata": {
"source": "telegram",
"dao_id": dao_id,
"user_id": f"tg:{user_id}",
"session_id": f"tg:{chat_id}:{dao_id}",
"username": username,
"chat_id": chat_id,
},
"context": {
"agent_name": agent_config.name,
"system_prompt": agent_config.system_prompt,
"memory": memory_context,
},
}
# Send to Router
logger.info(f"Sending to Router: agent={agent_config.agent_id}, dao={dao_id}, user=tg:{user_id}")
response = await send_to_router(router_request)
# Extract response
if isinstance(response, dict) and response.get("ok"):
answer_text = response.get("data", {}).get("text") or response.get("response", "")
if not answer_text:
answer_text = "Вибач, я зараз не можу відповісти."
# Truncate if too long for Telegram
if len(answer_text) > TELEGRAM_SAFE_LENGTH:
answer_text = answer_text[:TELEGRAM_SAFE_LENGTH] + "\n\n_... (відповідь обрізано)_"
# Send response back to Telegram
await send_telegram_message(chat_id, answer_text, telegram_token)
return {"ok": True, "agent": agent_config.agent_id}
else:
error_msg = response.get("error", "Unknown error") if isinstance(response, dict) else "Router error"
logger.error(f"Router error: {error_msg}")
await send_telegram_message(chat_id, f"Вибач, сталася помилка: {error_msg}", telegram_token)
return {"ok": False, "error": error_msg}
# ========================================
# Endpoints
# ========================================
@@ -133,7 +850,7 @@ def get_dao_id(chat_id: str, source: str) -> str:
@router.post("/telegram/webhook")
async def telegram_webhook(update: TelegramUpdate):
"""
Handle Telegram webhook.
Handle Telegram webhook for DAARWIZZ agent.
Telegram update format:
{
@@ -146,6 +863,16 @@ async def telegram_webhook(update: TelegramUpdate):
}
}
"""
try:
return await handle_telegram_webhook(DAARWIZZ_CONFIG, update)
except Exception as e:
logger.error(f"Error handling DAARWIZZ Telegram webhook: {e}", exc_info=True)
raise HTTPException(status_code=500, detail=str(e))
# Legacy code - will be removed after testing
async def _old_telegram_webhook(update: TelegramUpdate):
"""Стара версія - використовується для тестування"""
try:
if not update.message:
raise HTTPException(status_code=400, detail="No message in update")
@@ -758,6 +1485,16 @@ async def helion_telegram_webhook(update: TelegramUpdate):
"""
Handle Telegram webhook for Helion agent.
"""
try:
return await handle_telegram_webhook(HELION_CONFIG, update)
except Exception as e:
logger.error(f"Error handling Helion Telegram webhook: {e}", exc_info=True)
raise HTTPException(status_code=500, detail=str(e))
# Legacy code - will be removed after testing
async def _old_helion_telegram_webhook(update: TelegramUpdate):
"""Стара версія - використовується для тестування"""
try:
if not update.message:
raise HTTPException(status_code=400, detail="No message in update")
@@ -1110,17 +1847,17 @@ async def helion_telegram_webhook(update: TelegramUpdate):
@router.get("/health")
async def health():
"""Health check endpoint"""
agents_info = {}
for agent_id, config in AGENT_REGISTRY.items():
agents_info[agent_id] = {
"name": config.name,
"prompt_loaded": len(config.system_prompt) > 0,
"telegram_token_configured": config.get_telegram_token() is not None
}
return {
"status": "healthy",
"agents": {
"daarwizz": {
"name": DAARWIZZ_NAME,
"prompt_loaded": len(DAARWIZZ_SYSTEM_PROMPT) > 0
},
"helion": {
"name": HELION_NAME,
"prompt_loaded": len(HELION_SYSTEM_PROMPT) > 0
}
},
"agents": agents_info,
"agents_count": len(AGENT_REGISTRY),
"timestamp": datetime.utcnow().isoformat(),
}