from pathlib import Path from typing import Set, Tuple, Optional, Union, List, Dict from torrentool.torrent import Torrent from transmission_rpc import File from app import schemas from app.core.cache import FileCache from app.core.config import settings from app.core.metainfo import MetaInfo from app.log import logger from app.modules import _ModuleBase, _DownloaderBase from app.modules.transmission.transmission import Transmission from app.schemas import DownloaderTorrent from app.schemas.types import ( DownloadTaskState, DownloaderType, ModuleType, TorrentQueryStatus, TorrentStatus, ) from app.utils.string import StringUtils _TRANSMISSION_DOWNLOADING_STATES = { "download_pending", "downloading", } _TRANSMISSION_PAUSED_STATES = { "stopped", } class TransmissionModule(_ModuleBase, _DownloaderBase[Transmission]): def init_module(self) -> None: """ 初始化模块 """ super().init_service(service_name=Transmission.__name__.lower(), service_type=Transmission) @staticmethod def get_name() -> str: return "Transmission" @staticmethod def get_type() -> ModuleType: """ 获取模块类型 """ return ModuleType.Downloader @staticmethod def get_subtype() -> DownloaderType: """ 获取模块子类型 """ return DownloaderType.Transmission @staticmethod def get_priority() -> int: """ 获取模块优先级,数字越小优先级越高,只有同一接口下优先级才生效 """ return 2 def stop(self): pass def test(self) -> Optional[Tuple[bool, str]]: """ 测试模块连接性 """ if not self.get_instances(): return None for name, server in self.get_instances().items(): if server.is_inactive(): server.reconnect() if not server.transfer_info(): return False, f"无法连接Transmission下载器:{name}" return True, "" def init_setting(self) -> Tuple[str, Union[str, bool]]: pass def scheduler_job(self) -> None: """ 定时任务,每10分钟调用一次 """ # 定时重连 for name, server in self.get_instances().items(): if server.is_inactive(): logger.info(f"Transmission下载器 {name} 连接断开,尝试重连 ...") server.reconnect() def download(self, content: Union[Path, str, bytes], download_dir: Path, cookie: str, episodes: Set[int] = None, category: Optional[str] = None, label: Optional[str] = None, downloader: Optional[str] = None) -> Optional[Tuple[Optional[str], Optional[str], Optional[str], str]]: """ 根据种子文件,选择并添加下载任务 :param content: 种子文件地址或者磁力链接或种子内容 :param download_dir: 下载目录 :param cookie: cookie :param episodes: 需要下载的集数 :param category: 分类,TR中未使用 :param label: 标签 :param downloader: 下载器 :return: 下载器名称、种子Hash、种子文件布局、错误原因 """ def __get_torrent_info() -> Tuple[Optional[Torrent], Optional[bytes]]: """ 获取种子名称 """ torrent_info, torrent_content = None, None try: if isinstance(content, Path): if content.exists(): torrent_content = content.read_bytes() else: # 读取缓存的种子文件 torrent_content = FileCache().get(content.as_posix(), region="torrents") else: torrent_content = content if torrent_content: # 检查是否为磁力链接 if StringUtils.is_magnet_link(torrent_content): return None, torrent_content else: torrent_info = Torrent.from_string(torrent_content) return torrent_info, torrent_content except Exception as e: logger.error(f"获取种子名称失败:{e}") return None, None if not content: return None, None, None, "下载内容为空" # 读取种子的名称 torrent_from_file, content = __get_torrent_info() # 检查是否为磁力链接 is_magnet = isinstance(content, str) and content.startswith("magnet:") or isinstance(content, bytes) and content.startswith( b"magnet:") if not torrent_from_file and not is_magnet: return None, None, None, f"添加种子任务失败:无法读取种子文件" # 获取下载器 server: Transmission = self.get_instance(downloader) if not server: return None # 如果要选择文件则先暂停 is_paused = True if episodes else False # 标签 if label: labels = label.split(',') elif settings.TORRENT_TAG: labels = settings.TORRENT_TAG.split(',') else: labels = None # 添加任务 added_torrent = server.add_torrent( content=content, download_dir=self.normalize_path(download_dir, downloader), is_paused=is_paused, labels=labels, cookie=cookie ) # TR 始终使用原始种子布局, 返回"Original" torrent_layout = "Original" if not added_torrent: # 查询所有下载器的种子 torrents, error = server.get_torrents() if error: return None, None, None, "无法连接transmission下载器" if torrents: try: for torrent in torrents: # 名称与大小相等则认为是同一个种子 if torrent.name == getattr(torrent_from_file, 'name', '') and torrent.total_size == getattr(torrent_from_file, 'total_size', 0): torrent_hash = torrent.hashString logger.warn(f"下载器中已存在该种子任务:{torrent_hash} - {torrent.name}") # 给种子打上标签 if settings.TORRENT_TAG: logger.info(f"给种子 {torrent_hash} 打上标签:{settings.TORRENT_TAG}") # 种子标签 labels = [str(tag).strip() for tag in torrent.labels] if hasattr(torrent, "labels") else [] if "已整理" in labels: labels.remove("已整理") server.set_torrent_tag(ids=torrent_hash, tags=labels) if settings.TORRENT_TAG and settings.TORRENT_TAG not in labels: labels.append(settings.TORRENT_TAG) server.set_torrent_tag(ids=torrent_hash, tags=labels) return downloader or self.get_default_config_name(), torrent_hash, torrent_layout, f"下载任务已存在" finally: torrents.clear() del torrents return None, None, None, f"添加种子任务失败:{content}" else: torrent_hash = added_torrent.hashString if is_paused: # 选择文件 torrent_files = server.get_files(torrent_hash) if not torrent_files: return downloader or self.get_default_config_name(), torrent_hash, torrent_layout, "获取种子文件失败,下载任务可能在暂停状态" # 需要的文件信息 file_ids = [] unwanted_file_ids = [] try: for torrent_file in torrent_files: file_id = torrent_file.id file_name = torrent_file.name meta_info = MetaInfo(file_name) if not meta_info.episode_list: unwanted_file_ids.append(file_id) continue selected = set(meta_info.episode_list).issubset(set(episodes)) if not selected: unwanted_file_ids.append(file_id) continue file_ids.append(file_id) # 选择文件 server.set_files(torrent_hash, file_ids) server.set_unwanted_files(torrent_hash, unwanted_file_ids) # 开始任务 server.start_torrents(torrent_hash) finally: torrent_files.clear() del torrent_files return downloader or self.get_default_config_name(), torrent_hash, torrent_layout, "添加下载任务成功" else: return downloader or self.get_default_config_name(), torrent_hash, torrent_layout, "添加下载任务成功" def list_torrents(self, status: TorrentStatus = None, hashs: Union[list, str] = None, downloader: Optional[str] = None, include_all_tags: bool = False, ) -> Optional[List[DownloaderTorrent]]: """ 获取下载器种子列表 :param status: 种子状态 :param hashs: 种子Hash :param downloader: 下载器 :param include_all_tags: 是否包含未打内置标签的下载任务 :return: 下载器中符合状态的种子列表 """ # 获取下载器 if downloader: server: Transmission = self.get_instance(downloader) if not server: return None servers = {downloader: server} else: servers: Dict[str, Transmission] = self.get_instances() ret_torrents = [] query_status = self.__normalize_query_status(status) query_tags = None if include_all_tags else settings.TORRENT_TAG def __get_torrent_attr(torrent_data, *attr_names): """ 兼容 transmission-rpc 新旧字段名。 """ for attr_name in attr_names: try: return getattr(torrent_data, attr_name) except (AttributeError, KeyError, TypeError, ValueError): continue return None def __get_torrent_progress(torrent_data) -> float: """ 获取任务进度。 """ return __get_torrent_attr(torrent_data, "progress", "percent_done") or 0 def __get_torrent_size(torrent_data) -> int: """ 获取任务大小。 """ return __get_torrent_attr(torrent_data, "total_size", "totalSize") or 0 def __get_torrent_labels(torrent_data) -> str: """ 获取任务标签。 """ return ",".join(getattr(torrent_data, "labels", None) or []) def __get_torrent_path(torrent_data) -> Path: """ 获取任务内容路径。 """ return Path(torrent_data.download_dir) / torrent_data.name def __build_torrent(downloader_name: str, torrent_data) -> DownloaderTorrent: """ 构造统一下载器任务对象。 """ meta = MetaInfo(torrent_data.name) dlspeed = __get_torrent_attr( torrent_data, "rate_download", "rateDownload" ) or 0 upspeed = __get_torrent_attr( torrent_data, "rate_upload", "rateUpload" ) or 0 left_until_done = __get_torrent_attr( torrent_data, "left_until_done", "leftUntilDone" ) or 0 torrent_path = __get_torrent_path(torrent_data) ratio_limit = __get_torrent_attr(torrent_data, "seed_ratio_limit", "seedRatioLimit") seeding_time_limit = __get_torrent_attr(torrent_data, "seed_idle_limit", "seedIdleLimit") return DownloaderTorrent( downloader=downloader_name, hash=torrent_data.hashString, title=torrent_data.name, name=meta.name, year=meta.year, season_episode=meta.season_episode, path=Path(self.normalize_return_path(torrent_path, downloader_name)), save_path=self.normalize_return_path( Path(torrent_data.download_dir), downloader_name ) if getattr(torrent_data, "download_dir", None) else None, content_path=self.normalize_return_path(torrent_path, downloader_name), progress=__get_torrent_progress(torrent_data), size=__get_torrent_size(torrent_data), state=self.__normalize_torrent_state(torrent_data.status), dlspeed=StringUtils.str_filesize(dlspeed), upspeed=StringUtils.str_filesize(upspeed), tags=__get_torrent_labels(torrent_data), download_limit=__get_torrent_attr(torrent_data, "download_limit", "downloadLimit"), upload_limit=__get_torrent_attr(torrent_data, "upload_limit", "uploadLimit"), ratio_limit=ratio_limit, seeding_time_limit=seeding_time_limit, left_time=StringUtils.str_secends( left_until_done / dlspeed ) if dlspeed > 0 else '' ) if hashs: # 按Hash获取 for name, server in servers.items(): torrents, _ = server.get_torrents(ids=hashs, tags=query_tags) or [] try: for torrent_info in torrents: ret_torrents.append(__build_torrent(name, torrent_info)) finally: torrents.clear() del torrents elif query_status == TorrentQueryStatus.TRANSFER: # 获取已完成且未整理的 for name, server in servers.items(): torrents = server.get_completed_torrents(tags=query_tags) or [] try: for torrent_info in torrents: # 含"已整理"tag的不处理 if "已整理" in torrent_info.labels or []: continue # 下载路径 path = torrent_info.download_dir # 无法获取下载路径的不处理 if not path: logger.debug(f"未获取到 {torrent_info.name} 下载保存路径") continue ret_torrents.append(__build_torrent(name, torrent_info)) finally: torrents.clear() del torrents elif query_status == TorrentQueryStatus.DOWNLOADING: # 获取正在下载的任务 for name, server in servers.items(): torrents = server.get_downloading_torrents(tags=query_tags) or [] try: for torrent_info in torrents: ret_torrents.append(__build_torrent(name, torrent_info)) finally: torrents.clear() del torrents elif query_status in ( TorrentQueryStatus.ALL, TorrentQueryStatus.COMPLETED, TorrentQueryStatus.PAUSED, ): # 获取完整任务列表,由 MoviePilot 统一归一实际下载器状态。 for name, server in servers.items(): torrents, _ = server.get_torrents(tags=query_tags) or [] try: for torrent_info in torrents: torrent_state = self.__normalize_torrent_state(torrent_info.status) if ( query_status != TorrentQueryStatus.ALL and torrent_state != query_status.value ): continue ret_torrents.append(__build_torrent(name, torrent_info)) finally: torrents.clear() del torrents else: return None return ret_torrents # noqa @staticmethod def __normalize_query_status( status: Optional[Union[TorrentStatus, TorrentQueryStatus, str]] ) -> TorrentQueryStatus: """ 归一任务查询状态。 """ status_value = getattr(status, "value", status) status_text = str(status_value or "").strip().lower() if not status_text or status_text in {"all", "全部"}: return TorrentQueryStatus.ALL if status_text in { TorrentStatus.TRANSFER.value, TorrentQueryStatus.TRANSFER.value, "transfer", }: return TorrentQueryStatus.TRANSFER if status_text in { TorrentStatus.DOWNLOADING.value, TorrentQueryStatus.DOWNLOADING.value, "downloading", }: return TorrentQueryStatus.DOWNLOADING if status_text in { TorrentQueryStatus.COMPLETED.value, "complete", "seeding", "完成", "已完成", }: return TorrentQueryStatus.COMPLETED if status_text in {TorrentQueryStatus.PAUSED.value, "pause", "暂停", "已暂停"}: return TorrentQueryStatus.PAUSED return TorrentQueryStatus.ALL @staticmethod def __normalize_torrent_state(status: Optional[str]) -> str: """ 归一 Transmission 原始任务状态。 """ status_text = str(status or "").strip().lower() if status_text in _TRANSMISSION_PAUSED_STATES: return DownloadTaskState.PAUSED.value if status_text in _TRANSMISSION_DOWNLOADING_STATES: return DownloadTaskState.DOWNLOADING.value return DownloadTaskState.COMPLETED.value def transfer_completed(self, hashs: str, downloader: Optional[str] = None) -> None: """ 转移完成后的处理 :param hashs: 种子Hash :param downloader: 下载器 """ # 获取下载器 server: Transmission = self.get_instance(downloader) if not server: return None # 获取原标签 org_tags = server.get_torrent_tags(ids=hashs) # 种子打上已整理标签 if org_tags: tags = org_tags + ['已整理'] else: tags = ['已整理'] server.set_torrent_tag(ids=hashs, tags=tags) return None def remove_torrents(self, hashs: Union[str, list], delete_file: Optional[bool] = True, downloader: Optional[str] = None) -> Optional[bool]: """ 删除下载器种子 :param hashs: 种子Hash :param delete_file: 是否删除文件 :param downloader: 下载器 :return: bool """ # 获取下载器 server: Transmission = self.get_instance(downloader) if not server: return None return server.delete_torrents(delete_file=delete_file, ids=hashs) def set_torrents_tag(self, hashs: Union[str, list], tags: list, downloader: Optional[str] = None) -> Optional[bool]: """ 设置种子标签 :param hashs: 种子Hash :param tags: 标签列表 :param downloader: 下载器 :return: bool """ # 获取下载器 server: Transmission = self.get_instance(downloader) if not server: return None # 获取原标签,TR默认会覆盖,需追加 org_tags = server.get_torrent_tags(ids=hashs) return server.set_torrent_tag(ids=hashs, tags=tags, org_tags=org_tags) def update_torrent( self, hash_string: str, downloader: Optional[str] = None, download_limit: Optional[float] = None, upload_limit: Optional[float] = None, tracker_list: Optional[list] = None, save_path: Optional[str] = None, category: Optional[str] = None, ratio_limit: Optional[float] = None, seeding_time_limit: Optional[int] = None, ) -> Optional[Dict[str, bool]]: """ 修改下载任务属性。 :param hash_string: 种子Hash :param downloader: 下载器 :param download_limit: 下载限速,单位 KB/s :param upload_limit: 上传限速,单位 KB/s :param tracker_list: Tracker URL列表 :param save_path: 保存目录 :param category: 分类,Transmission 不支持 :param ratio_limit: 分享率限制 :param seeding_time_limit: 做种时间限制,单位分钟 :return: 各项修改结果 """ server: Transmission = self.get_instance(downloader) if not server: return None results = {} if any( value is not None for value in (download_limit, upload_limit, ratio_limit, seeding_time_limit) ): change_result = server.change_torrent( hash_string=hash_string, download_limit=download_limit, upload_limit=upload_limit, ratio_limit=ratio_limit, seeding_time_limit=seeding_time_limit, ) results["limits"] = change_result if save_path is not None: results["save_path"] = server.set_torrent_location( hash_string=hash_string, location=self.normalize_path(Path(save_path), downloader), ) if tracker_list is not None: results["trackers"] = server.update_tracker( hash_string=hash_string, tracker_list=tracker_list ) if category is not None: results["category"] = False return results def get_torrent_trackers( self, hash_string: str, downloader: Optional[str] = None, ) -> Optional[Dict[str, List[str]]]: """ 查询下载任务Tracker列表。 :param hash_string: 种子Hash :param downloader: 下载器 :return: 下载器名称到Tracker列表的映射 """ if downloader: server: Transmission = self.get_instance(downloader) if not server: return None servers = {downloader: server} else: servers: Dict[str, Transmission] = self.get_instances() ret_trackers = {} for name, server in servers.items(): trackers = server.get_trackers(hash_string) if trackers is not None: ret_trackers[name] = trackers return ret_trackers def start_torrents(self, hashs: Union[list, str], downloader: Optional[str] = None) -> Optional[bool]: """ 开始下载 :param hashs: 种子Hash :param downloader: 下载器 :return: bool """ # 获取下载器 server: Transmission = self.get_instance(downloader) if not server: return None return server.start_torrents(ids=hashs) def stop_torrents(self, hashs: Union[list, str], downloader: Optional[str] = None) -> Optional[bool]: """ 停止下载 :param hashs: 种子Hash :param downloader: 下载器 :return: bool """ # 获取下载器 server: Transmission = self.get_instance(downloader) if not server: return None return server.stop_torrents(ids=hashs) def torrent_files(self, tid: str, downloader: Optional[str] = None) -> Optional[List[File]]: """ 获取种子文件列表 """ # 获取下载器 server: Transmission = self.get_instance(downloader) if not server: return None return server.get_files(tid=tid) def downloader_info(self, downloader: Optional[str] = None) -> Optional[List[schemas.DownloaderInfo]]: """ 下载器信息 """ if downloader: server: Transmission = self.get_instance(downloader) if not server: return None servers = [server] else: servers = self.get_instances().values() # 调用Qbittorrent API查询实时信息 ret_info = [] for server in servers: info = server.transfer_info() if not info: continue ret_info.append(schemas.DownloaderInfo( download_speed=info.download_speed, upload_speed=info.upload_speed, download_size=info.current_stats.downloaded_bytes, upload_size=info.current_stats.uploaded_bytes )) return ret_info