mirror of
https://github.com/jxxghp/MoviePilot.git
synced 2026-03-20 03:57:30 +08:00
418 lines
17 KiB
Python
418 lines
17 KiB
Python
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 TransferTorrent, DownloadingTorrent
|
||
from app.schemas.types import TorrentStatus, ModuleType, DownloaderType
|
||
from app.utils.string import StringUtils
|
||
|
||
|
||
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
|
||
) -> Optional[List[Union[TransferTorrent, DownloadingTorrent]]]:
|
||
"""
|
||
获取下载器种子列表
|
||
:param status: 种子状态
|
||
:param hashs: 种子Hash
|
||
:param downloader: 下载器
|
||
: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 = []
|
||
if hashs:
|
||
# 按Hash获取
|
||
for name, server in servers.items():
|
||
torrents, _ = server.get_torrents(ids=hashs, tags=settings.TORRENT_TAG) or []
|
||
try:
|
||
for torrent in torrents:
|
||
ret_torrents.append(TransferTorrent(
|
||
downloader=name,
|
||
title=torrent.name,
|
||
path=Path(torrent.download_dir) / torrent.name,
|
||
hash=torrent.hashString,
|
||
size=torrent.total_size,
|
||
tags=",".join(torrent.labels or []),
|
||
progress=torrent.progress
|
||
))
|
||
finally:
|
||
torrents.clear()
|
||
del torrents
|
||
elif status == TorrentStatus.TRANSFER:
|
||
# 获取已完成且未整理的
|
||
for name, server in servers.items():
|
||
torrents = server.get_completed_torrents(tags=settings.TORRENT_TAG) or []
|
||
try:
|
||
for torrent in torrents:
|
||
# 含"已整理"tag的不处理
|
||
if "已整理" in torrent.labels or []:
|
||
continue
|
||
# 下载路径
|
||
path = torrent.download_dir
|
||
# 无法获取下载路径的不处理
|
||
if not path:
|
||
logger.debug(f"未获取到 {torrent.name} 下载保存路径")
|
||
continue
|
||
ret_torrents.append(TransferTorrent(
|
||
downloader=name,
|
||
title=torrent.name,
|
||
path=Path(torrent.download_dir) / torrent.name,
|
||
hash=torrent.hashString,
|
||
tags=",".join(torrent.labels or []),
|
||
progress=torrent.progress,
|
||
state="paused" if torrent.status == "stopped" else "downloading",
|
||
))
|
||
finally:
|
||
torrents.clear()
|
||
del torrents
|
||
elif status == TorrentStatus.DOWNLOADING:
|
||
# 获取正在下载的任务
|
||
for name, server in servers.items():
|
||
torrents = server.get_downloading_torrents(tags=settings.TORRENT_TAG) or []
|
||
try:
|
||
for torrent in torrents:
|
||
meta = MetaInfo(torrent.name)
|
||
dlspeed = torrent.rate_download if hasattr(torrent, "rate_download") else torrent.rateDownload
|
||
upspeed = torrent.rate_upload if hasattr(torrent, "rate_upload") else torrent.rateUpload
|
||
ret_torrents.append(DownloadingTorrent(
|
||
downloader=name,
|
||
hash=torrent.hashString,
|
||
title=torrent.name,
|
||
name=meta.name,
|
||
year=meta.year,
|
||
season_episode=meta.season_episode,
|
||
progress=torrent.progress,
|
||
size=torrent.total_size,
|
||
state="paused" if torrent.status == "stopped" else "downloading",
|
||
dlspeed=StringUtils.str_filesize(dlspeed),
|
||
upspeed=StringUtils.str_filesize(upspeed),
|
||
left_time=StringUtils.str_secends(torrent.left_until_done / dlspeed) if dlspeed > 0 else ''
|
||
))
|
||
finally:
|
||
torrents.clear()
|
||
del torrents
|
||
else:
|
||
return None
|
||
return ret_torrents # noqa
|
||
|
||
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 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
|