diff --git a/app/agent/tools/base.py b/app/agent/tools/base.py index e99f0e7e..5e7ee549 100644 --- a/app/agent/tools/base.py +++ b/app/agent/tools/base.py @@ -26,12 +26,13 @@ class MoviePilotTool(BaseTool): async def _arun(self, **kwargs) -> str: raise NotImplementedError - def _send_tool_message(self, message: str, **kwargs): + def _send_tool_message(self, message: str, title: str = None, **kwargs): """发送工具执行消息""" try: self._message_helper.put( message=message, - role="system" + role="system", + title=title or "工具执行" ) except Exception as e: logger.error(f"发送工具消息失败: {e}") diff --git a/app/agent/tools/impl/add_download.py b/app/agent/tools/impl/add_download.py index bad4508e..44c8275a 100644 --- a/app/agent/tools/impl/add_download.py +++ b/app/agent/tools/impl/add_download.py @@ -20,12 +20,12 @@ class AddDownloadTool(MoviePilotTool): logger.info(f"执行工具: {self.name}, 参数: torrent_title={torrent_title}, torrent_url={torrent_url}, downloader={downloader}, save_path={save_path}, labels={labels}") # 发送工具执行说明 - self._send_tool_message(f"正在添加下载任务: {torrent_title}", "info") + self._send_tool_message(f"正在添加下载任务: {torrent_title}", title="添加下载") try: if not torrent_title or not torrent_url: error_message = "错误:必须提供种子标题和下载链接" - self._send_tool_message(error_message, "error") + self._send_tool_message(error_message, title="下载失败") return error_message # 使用DownloadChain添加下载 @@ -42,17 +42,22 @@ class AddDownloadTool(MoviePilotTool): meta_info=meta_info ) - did = download_chain.download_single(context=context, downloader=downloader, - save_path=save_path, label=labels) + did = download_chain.download_single( + context=context, + downloader=downloader, + save_path=save_path, + label=labels + ) if did: success_message = f"成功添加下载任务:{torrent_title}" - self._send_tool_message(success_message, "success") + self._send_tool_message(success_message, title="下载成功") return success_message else: error_message = "添加下载任务失败" - self._send_tool_message(error_message, "error") + self._send_tool_message(error_message, title="下载失败") return error_message except Exception as e: error_message = f"添加下载任务时发生错误: {str(e)}" - self._send_tool_message(error_message, "error") + logger.error(f"添加下载任务失败: {e}", exc_info=True) + self._send_tool_message(error_message, title="下载失败") return error_message diff --git a/app/agent/tools/impl/add_subscribe.py b/app/agent/tools/impl/add_subscribe.py index 1baa2433..162eeb02 100644 --- a/app/agent/tools/impl/add_subscribe.py +++ b/app/agent/tools/impl/add_subscribe.py @@ -17,21 +17,36 @@ class AddSubscribeTool(MoviePilotTool): logger.info(f"执行工具: {self.name}, 参数: title={title}, year={year}, media_type={media_type}, season={season}, tmdb_id={tmdb_id}") # 发送工具执行说明 - self._send_tool_message(f"正在添加订阅: {title} ({year}) - {media_type}", "info") + self._send_tool_message(f"正在添加订阅: {title} ({year}) - {media_type}", title="添加订阅") try: subscribe_chain = SubscribeChain() - sid, message = subscribe_chain.add(mtype=MediaType(media_type), title=title, year=year, - tmdbid=tmdb_id, season=season, username=self._user_id) + # 转换 tmdb_id 为整数 + tmdbid_int = None + if tmdb_id: + try: + tmdbid_int = int(tmdb_id) + except (ValueError, TypeError): + logger.warning(f"无效的 tmdb_id: {tmdb_id},将忽略") + + sid, message = subscribe_chain.add( + mtype=MediaType(media_type), + title=title, + year=year, + tmdbid=tmdbid_int, + season=season, + username=self._user_id + ) if sid: success_message = f"成功添加订阅:{title} ({year})" - self._send_tool_message(success_message, "success") + self._send_tool_message(success_message, title="订阅成功") return success_message else: error_message = f"添加订阅失败:{message}" - self._send_tool_message(error_message, "error") + self._send_tool_message(error_message, title="订阅失败") return error_message except Exception as e: error_message = f"添加订阅时发生错误: {str(e)}" - self._send_tool_message(error_message, "error") + logger.error(f"添加订阅失败: {e}", exc_info=True) + self._send_tool_message(error_message, title="订阅失败") return error_message diff --git a/app/agent/tools/impl/get_recommendations.py b/app/agent/tools/impl/get_recommendations.py index f305e766..a5b38bac 100644 --- a/app/agent/tools/impl/get_recommendations.py +++ b/app/agent/tools/impl/get_recommendations.py @@ -32,8 +32,9 @@ class GetRecommendationsTool(MoviePilotTool): results = recommend_chain.bangumi_calendar(limit=limit) if results: - return json.dumps([r.dict() for r in results], ensure_ascii=False, indent=2) + # 使用 to_dict() 方法 + return json.dumps([r.to_dict() if hasattr(r, 'to_dict') else r.dict() if hasattr(r, 'dict') else r.model_dump() if hasattr(r, 'model_dump') else r for r in results], ensure_ascii=False, indent=2) return "未找到推荐内容。" except Exception as e: - logger.error(f"获取推荐失败: {e}") + logger.error(f"获取推荐失败: {e}", exc_info=True) return f"获取推荐时发生错误: {str(e)}" diff --git a/app/agent/tools/impl/query_downloads.py b/app/agent/tools/impl/query_downloads.py index 946430a4..220b8ccf 100644 --- a/app/agent/tools/impl/query_downloads.py +++ b/app/agent/tools/impl/query_downloads.py @@ -3,7 +3,7 @@ import json from typing import Optional -from app.db.download_oper import DownloadOper +from app.chain.download import DownloadChain from app.log import logger from app.agent.tools.base import MoviePilotTool @@ -16,8 +16,9 @@ class QueryDownloadsTool(MoviePilotTool): status: Optional[str] = "all") -> str: logger.info(f"执行工具: {self.name}, 参数: downloader={downloader}, status={status}") try: - download_oper = DownloadOper() - downloads = download_oper.list() + download_chain = DownloadChain() + # 使用 DownloadChain.downloading 方法获取正在下载的任务 + downloads = download_chain.downloading(name=downloader) filtered_downloads = [] for dl in downloads: if downloader and dl.downloader != downloader: @@ -26,8 +27,8 @@ class QueryDownloadsTool(MoviePilotTool): continue filtered_downloads.append(dl) if filtered_downloads: - return json.dumps([d.dict() for d in filtered_downloads], ensure_ascii=False, indent=2) + return json.dumps([d.dict() if hasattr(d, 'dict') else d.model_dump() if hasattr(d, 'model_dump') else d for d in filtered_downloads], ensure_ascii=False, indent=2) return "未找到相关下载任务。" except Exception as e: - logger.error(f"查询下载失败: {e}") + logger.error(f"查询下载失败: {e}", exc_info=True) return f"查询下载时发生错误: {str(e)}" diff --git a/app/agent/tools/impl/query_media_library.py b/app/agent/tools/impl/query_media_library.py index d1cf6bd1..def948cb 100644 --- a/app/agent/tools/impl/query_media_library.py +++ b/app/agent/tools/impl/query_media_library.py @@ -26,8 +26,8 @@ class QueryMediaLibraryTool(MoviePilotTool): continue filtered_medias.append(media) if filtered_medias: - return json.dumps([m.dict() for m in filtered_medias], ensure_ascii=False, indent=2) + return json.dumps([m.to_dict() if hasattr(m, 'to_dict') else m.dict() if hasattr(m, 'dict') else m.model_dump() if hasattr(m, 'model_dump') else m for m in filtered_medias], ensure_ascii=False, indent=2) return "媒体库中未找到相关媒体。" except Exception as e: - logger.error(f"查询媒体库失败: {e}") + logger.error(f"查询媒体库失败: {e}", exc_info=True) return f"查询媒体库时发生错误: {str(e)}" diff --git a/app/agent/tools/impl/query_subscribes.py b/app/agent/tools/impl/query_subscribes.py index 2dd80aa3..0edcf214 100644 --- a/app/agent/tools/impl/query_subscribes.py +++ b/app/agent/tools/impl/query_subscribes.py @@ -20,14 +20,14 @@ class QuerySubscribesTool(MoviePilotTool): subscribes = subscribe_oper.list() filtered_subscribes = [] for sub in subscribes: - if status != "all" and sub.status != status: + if status != "all" and sub.state != status: continue if media_type != "all" and sub.type != media_type: continue filtered_subscribes.append(sub) if filtered_subscribes: - return json.dumps([s.dict() for s in filtered_subscribes], ensure_ascii=False, indent=2) + return json.dumps([s.to_dict() for s in filtered_subscribes], ensure_ascii=False, indent=2) return "未找到相关订阅。" except Exception as e: - logger.error(f"查询订阅失败: {e}") + logger.error(f"查询订阅失败: {e}", exc_info=True) return f"查询订阅时发生错误: {str(e)}" diff --git a/app/agent/tools/impl/search_media.py b/app/agent/tools/impl/search_media.py index 73e95c09..2e1b00d2 100644 --- a/app/agent/tools/impl/search_media.py +++ b/app/agent/tools/impl/search_media.py @@ -4,7 +4,9 @@ import json from typing import Optional from app.chain.media import MediaChain +from app.core.metainfo import MetaInfo from app.log import logger +from app.schemas.types import MediaType from app.agent.tools.base import MoviePilotTool @@ -17,26 +19,58 @@ class SearchMediaTool(MoviePilotTool): logger.info(f"执行工具: {self.name}, 参数: title={title}, year={year}, media_type={media_type}, season={season}") # 发送工具执行说明 - self._send_tool_message(f"正在搜索媒体资源: {title}" + (f" ({year})" if year else ""), "info") + self._send_tool_message(f"正在搜索媒体资源: {title}" + (f" ({year})" if year else ""), title="搜索中") try: media_chain = MediaChain() - results = media_chain.search_media(title=title, year=year, mtype=media_type, season=season) + # 构建搜索标题 + search_title = title + if year: + search_title = f"{title} {year}" + if media_type: + search_title = f"{search_title} {media_type}" + if season: + search_title = f"{search_title} S{season:02d}" + + # 使用 MediaChain.search 方法 + meta, results = media_chain.search(title=search_title) + + # 过滤结果 if results: - result_message = f"找到 {len(results)} 个相关媒体资源" - self._send_tool_message(result_message, "success") + filtered_results = [] + for result in results: + if year and result.year != year: + continue + if media_type: + try: + if result.type != MediaType(media_type): + continue + except: + pass + if season and result.season != season: + continue + filtered_results.append(result) - # 发送详细结果 - for i, result in enumerate(results[:5]): # 只显示前5个结果 - media_info = f"{i+1}. {result.title} ({result.year}) - {result.type}" - self._send_tool_message(media_info, "info") - - return json.dumps([r.dict() for r in results], ensure_ascii=False, indent=2) + if filtered_results: + result_message = f"找到 {len(filtered_results)} 个相关媒体资源" + self._send_tool_message(result_message, title="搜索成功") + + # 发送详细结果 + for i, result in enumerate(filtered_results[:5]): # 只显示前5个结果 + media_info = f"{i+1}. {result.title} ({result.year}) - {result.type.value if result.type else '未知'}" + self._send_tool_message(media_info, title="搜索结果") + + return json.dumps([r.to_dict() for r in filtered_results], ensure_ascii=False, indent=2) + else: + error_message = f"未找到符合条件的媒体资源: {title}" + self._send_tool_message(error_message, title="搜索完成") + return error_message else: error_message = f"未找到相关媒体资源: {title}" - self._send_tool_message(error_message, "warning") + self._send_tool_message(error_message, title="搜索完成") return error_message except Exception as e: error_message = f"搜索媒体失败: {str(e)}" - self._send_tool_message(error_message, "error") + logger.error(f"搜索媒体失败: {e}", exc_info=True) + self._send_tool_message(error_message, title="搜索失败") return error_message diff --git a/app/agent/tools/impl/search_torrents.py b/app/agent/tools/impl/search_torrents.py index 631009c5..0c515fd4 100644 --- a/app/agent/tools/impl/search_torrents.py +++ b/app/agent/tools/impl/search_torrents.py @@ -19,36 +19,44 @@ class SearchTorrentsTool(MoviePilotTool): logger.info(f"执行工具: {self.name}, 参数: title={title}, year={year}, media_type={media_type}, season={season}, sites={sites}") # 发送工具执行说明 - self._send_tool_message(f"正在搜索种子资源: {title}" + (f" ({year})" if year else ""), "info") + self._send_tool_message(f"正在搜索种子资源: {title}" + (f" ({year})" if year else ""), title="搜索种子") try: search_chain = SearchChain() torrents = search_chain.search_by_title(title=title, sites=sites) filtered_torrents = [] for torrent in torrents: - if year and torrent.meta_info.year != year: + # torrent 是 Context 对象,需要通过 meta_info 和 media_info 访问属性 + if year and torrent.meta_info and torrent.meta_info.year != year: continue - if media_type and torrent.media_info and torrent.media_info.type != MediaType(media_type): - continue - if season and torrent.meta_info.begin_season != season: + if media_type and torrent.media_info: + try: + if torrent.media_info.type != MediaType(media_type): + continue + except: + pass + if season and torrent.meta_info and torrent.meta_info.begin_season != season: continue filtered_torrents.append(torrent) if filtered_torrents: result_message = f"找到 {len(filtered_torrents)} 个相关种子资源" - self._send_tool_message(result_message, "success") + self._send_tool_message(result_message, title="搜索成功") # 发送详细结果 for i, torrent in enumerate(filtered_torrents[:5]): # 只显示前5个结果 - torrent_info = f"{i+1}. {torrent.title} - {torrent.site_name}" - self._send_tool_message(torrent_info, "info") + torrent_title = torrent.torrent_info.title if torrent.torrent_info else torrent.meta_info.title if torrent.meta_info else "未知" + site_name = torrent.torrent_info.site_name if torrent.torrent_info else "未知站点" + torrent_info = f"{i+1}. {torrent_title} - {site_name}" + self._send_tool_message(torrent_info, title="搜索结果") - return json.dumps([t.dict() for t in filtered_torrents], ensure_ascii=False, indent=2) + return json.dumps([t.to_dict() for t in filtered_torrents], ensure_ascii=False, indent=2) else: error_message = f"未找到相关种子资源: {title}" - self._send_tool_message(error_message, "warning") + self._send_tool_message(error_message, title="搜索完成") return error_message except Exception as e: error_message = f"搜索种子时发生错误: {str(e)}" - self._send_tool_message(error_message, "error") + logger.error(f"搜索种子失败: {e}", exc_info=True) + self._send_tool_message(error_message, title="搜索失败") return error_message