mirror of
https://github.com/jxxghp/MoviePilot.git
synced 2026-03-20 03:57:30 +08:00
81 lines
3.9 KiB
Python
81 lines
3.9 KiB
Python
"""查询下载工具"""
|
|
|
|
import json
|
|
from typing import Optional, Type
|
|
|
|
from pydantic import BaseModel, Field
|
|
|
|
from app.agent.tools.base import MoviePilotTool
|
|
from app.chain.download import DownloadChain
|
|
from app.log import logger
|
|
|
|
|
|
class QueryDownloadsInput(BaseModel):
|
|
"""查询下载工具的输入参数模型"""
|
|
explanation: str = Field(..., description="Clear explanation of why this tool is being used in the current context")
|
|
downloader: Optional[str] = Field(None,
|
|
description="Name of specific downloader to query (optional, if not provided queries all configured downloaders)")
|
|
status: Optional[str] = Field("all",
|
|
description="Filter downloads by status: 'downloading' for active downloads, 'completed' for finished downloads, 'paused' for paused downloads, 'all' for all downloads")
|
|
|
|
|
|
class QueryDownloadsTool(MoviePilotTool):
|
|
name: str = "query_downloads"
|
|
description: str = "Query download status and list all active download tasks. Shows download progress, completion status, and task details from configured downloaders."
|
|
args_schema: Type[BaseModel] = QueryDownloadsInput
|
|
|
|
async def run(self, downloader: Optional[str] = None,
|
|
status: Optional[str] = "all", **kwargs) -> str:
|
|
logger.info(f"执行工具: {self.name}, 参数: downloader={downloader}, status={status}")
|
|
try:
|
|
download_chain = DownloadChain()
|
|
# 使用 DownloadChain.downloading 方法获取正在下载的任务
|
|
downloads = download_chain.downloading(name=downloader)
|
|
filtered_downloads = []
|
|
for dl in downloads:
|
|
if downloader and dl.downloader != downloader:
|
|
continue
|
|
if status != "all" and dl.status != status:
|
|
continue
|
|
filtered_downloads.append(dl)
|
|
if filtered_downloads:
|
|
# 限制最多20条结果
|
|
total_count = len(filtered_downloads)
|
|
limited_downloads = filtered_downloads[:20]
|
|
# 精简字段,只保留关键信息
|
|
simplified_downloads = []
|
|
for d in limited_downloads:
|
|
simplified = {
|
|
"downloader": d.downloader,
|
|
"hash": d.hash,
|
|
"title": d.title,
|
|
"name": d.name,
|
|
"year": d.year,
|
|
"season_episode": d.season_episode,
|
|
"size": d.size,
|
|
"progress": d.progress,
|
|
"state": d.state,
|
|
"upspeed": d.upspeed,
|
|
"dlspeed": d.dlspeed,
|
|
"left_time": d.left_time
|
|
}
|
|
# 精简 media 字段
|
|
if d.media:
|
|
simplified["media"] = {
|
|
"tmdbid": d.media.get("tmdbid"),
|
|
"type": d.media.get("type"),
|
|
"title": d.media.get("title"),
|
|
"season": d.media.get("season"),
|
|
"episode": d.media.get("episode")
|
|
}
|
|
simplified_downloads.append(simplified)
|
|
result_json = json.dumps(simplified_downloads, ensure_ascii=False, indent=2)
|
|
# 如果结果被裁剪,添加提示信息
|
|
if total_count > 20:
|
|
return f"注意:查询结果共找到 {total_count} 条,为节省上下文空间,仅显示前 20 条结果。\n\n{result_json}"
|
|
return result_json
|
|
return "未找到相关下载任务。"
|
|
except Exception as e:
|
|
logger.error(f"查询下载失败: {e}", exc_info=True)
|
|
return f"查询下载时发生错误: {str(e)}"
|