from typing import Any, Generator, List, Optional, Tuple, Union from app import schemas from app.core.context import MediaInfo from app.core.event import eventmanager, Event from app.log import logger from app.modules import _MediaServerBase, _ModuleBase from app.modules.trimemedia.trimemedia import TrimeMedia from app.schemas import AuthCredentials, AuthInterceptCredentials from app.schemas.types import ChainEventType, MediaServerType, MediaType, ModuleType, SystemConfigKey, EventType class TrimeMediaModule(_ModuleBase, _MediaServerBase[TrimeMedia]): def init_module(self) -> None: """ 初始化模块 """ super().init_service( service_name=TrimeMedia.__name__.lower(), service_type=lambda conf: TrimeMedia( **conf.config, sync_libraries=conf.sync_libraries ), ) @eventmanager.register(EventType.ConfigChanged) def handle_config_changed(self, event: Event): """ 处理配置变更事件 :param event: 事件对象 """ if not event: return event_data: schemas.ConfigChangeEventData = event.event_data if event_data.key not in [SystemConfigKey.MediaServers.value]: return logger.info("配置变更,重新加载飞牛影视模块...") self.init_module() @staticmethod def get_name() -> str: return "飞牛影视" @staticmethod def get_type() -> ModuleType: """ 获取模块类型 """ return ModuleType.MediaServer @staticmethod def get_subtype() -> MediaServerType: """ 获取模块子类型 """ return MediaServerType.TrimeMedia @staticmethod def get_priority() -> int: """ 获取模块优先级,数字越小优先级越高,只有同一接口下优先级才生效 """ return 4 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_configured() and server.is_inactive(): logger.info(f"飞牛影视 {name} 连接断开,尝试重连 ...") server.reconnect() def stop(self): for server in self.get_instances().values(): if server.is_authenticated(): server.disconnect() def test(self) -> Optional[Tuple[bool, str]]: """ 测试模块连接性 """ if not self.get_instances(): return None for name, server in self.get_instances().items(): if not server.is_configured(): return False, f"飞牛影视配置不完整:{name}" if server.is_inactive() and not server.reconnect(): return False, f"无法连接飞牛影视:{name}" return True, "" def user_authenticate( self, credentials: AuthCredentials, service_name: Optional[str] = None ) -> Optional[AuthCredentials]: """ 使用飞牛影视用户辅助完成用户认证 :param credentials: 认证数据 :param service_name: 指定要认证的媒体服务器名称,若为 None 则认证所有服务 :return: 认证数据 """ # 飞牛影视认证 if not credentials or credentials.grant_type != "password": return None # 确定要认证的服务器列表 if service_name: # 如果指定了服务名,获取该服务实例 servers = ( [(service_name, server)] if (server := self.get_instance(service_name)) else [] ) else: # 如果没有指定服务名,遍历所有服务 servers = self.get_instances().items() # 遍历要认证的服务器 for name, server in servers: # 触发认证拦截事件 intercept_event = eventmanager.send_event( etype=ChainEventType.AuthIntercept, data=AuthInterceptCredentials( username=credentials.username, channel=self.get_name(), service=name, status="triggered", ), ) if intercept_event and intercept_event.event_data: intercept_data: AuthInterceptCredentials = intercept_event.event_data if intercept_data.cancel: continue token = server.authenticate(credentials.username, credentials.password) if token: credentials.channel = self.get_name() credentials.service = name credentials.token = token return credentials return None def webhook_parser( self, body: Any, form: Any, args: Any ) -> Optional[schemas.WebhookEventInfo]: """ 解析Webhook报文体 :param body: 请求体 :param form: 请求表单 :param args: 请求参数 :return: 字典,解析为消息时需要包含:title、text、image """ source = args.get("source") if source: server: TrimeMedia = self.get_instance(source) if not server: return None result = server.get_webhook_message(body) if result: result.server_name = source return result for server in self.get_instances().values(): if server: result = server.get_webhook_message(body) if result: return result return None def media_exists( self, mediainfo: MediaInfo, itemid: Optional[str] = None, server: Optional[str] = None, ) -> Optional[schemas.ExistMediaInfo]: """ 判断媒体文件是否存在 :param mediainfo: 识别的媒体信息 :param itemid: 媒体服务器ItemID :param server: 媒体服务器名称 :return: 如不存在返回None,存在时返回信息,包括每季已存在所有集{type: movie/tv, seasons: {season: [episodes]}} """ if server: servers = [(server, self.get_instance(server))] else: servers = self.get_instances().items() for name, s in servers: if not s: continue if mediainfo.type == MediaType.MOVIE: if itemid: movie = s.get_iteminfo(itemid) if movie: logger.info(f"媒体库 {name} 中找到了 {movie}") return schemas.ExistMediaInfo( type=MediaType.MOVIE, server_type="trimemedia", server=name, itemid=movie.item_id, ) movies = s.get_movies( title=mediainfo.title, year=mediainfo.year, tmdb_id=mediainfo.tmdb_id, ) if not movies: logger.info(f"{mediainfo.title_year} 没有在媒体库 {name} 中") continue else: logger.info(f"媒体库 {name} 中找到了 {movies}") return schemas.ExistMediaInfo( type=MediaType.MOVIE, server_type="trimemedia", server=name, itemid=movies[0].item_id, ) else: itemid, tvs = s.get_tv_episodes( title=mediainfo.title, year=mediainfo.year, tmdb_id=mediainfo.tmdb_id, item_id=itemid, ) if not tvs: logger.info(f"{mediainfo.title_year} 没有在媒体库 {name} 中") continue else: logger.info( f"{mediainfo.title_year} 在媒体库 {name} 中找到了这些季集:{tvs}" ) return schemas.ExistMediaInfo( type=MediaType.TV, seasons=tvs, server_type="trimemedia", server=name, itemid=itemid, ) return None def media_statistic( self, server: Optional[str] = None ) -> Optional[List[schemas.Statistic]]: """ 媒体数量统计 """ if server: server_obj: TrimeMedia = self.get_instance(server) if not server_obj: return None servers = [server_obj] else: servers = self.get_instances().values() media_statistics = [] for s in servers: media_statistic = s.get_medias_count() if not media_statistic: continue media_statistic.user_count = s.get_user_count() media_statistics.append(media_statistic) return media_statistics def mediaserver_librarys( self, server: Optional[str] = None, hidden: Optional[bool] = False, **kwargs ) -> Optional[List[schemas.MediaServerLibrary]]: """ 媒体库列表 """ server_obj: TrimeMedia = self.get_instance(server) if server_obj: return server_obj.get_librarys(hidden=hidden) return None def mediaserver_items( self, server: str, library_id: Union[str, int], start_index: Optional[int] = 0, limit: Optional[int] = -1, ) -> Optional[Generator]: """ 获取媒体服务器项目列表,支持分页和不分页逻辑,默认不分页获取所有数据 :param server: 媒体服务器名称 :param library_id: 媒体库ID,用于标识要获取的媒体库 :param start_index: 起始索引,用于分页获取数据。默认为 0,即从第一个项目开始获取 :param limit: 每次请求的最大项目数,用于分页。如果为 None 或 -1,则表示一次性获取所有数据,默认为 -1 :return: 返回一个生成器对象,用于逐步获取媒体服务器中的项目 """ server_obj: TrimeMedia = self.get_instance(server) if server_obj: return server_obj.get_items(library_id, start_index, limit) return None def mediaserver_iteminfo( self, server: str, item_id: str ) -> Optional[schemas.MediaServerItem]: """ 媒体库项目详情 """ server_obj: TrimeMedia = self.get_instance(server) if server_obj: return server_obj.get_iteminfo(item_id) return None def mediaserver_tv_episodes( self, server: str, item_id: Union[str, int] ) -> Optional[List[schemas.MediaServerSeasonInfo]]: """ 获取剧集信息 """ server_obj: TrimeMedia = self.get_instance(server) if not server_obj: return None _, seasoninfo = server_obj.get_tv_episodes(item_id=item_id) if not seasoninfo: return [] return [ schemas.MediaServerSeasonInfo(season=season, episodes=episodes) for season, episodes in seasoninfo.items() ] def mediaserver_playing( self, server: str, count: Optional[int] = 20, **kwargs ) -> List[schemas.MediaServerPlayItem]: """ 获取媒体服务器正在播放信息 """ server_obj: TrimeMedia = self.get_instance(server) if not server_obj: return [] return server_obj.get_resume(num=count) def mediaserver_play_url( self, server: str, item_id: Union[str, int] ) -> Optional[str]: """ 获取媒体库播放地址 """ server_obj: TrimeMedia = self.get_instance(server) if not server_obj: return None return server_obj.get_play_url(item_id) def mediaserver_latest( self, server: Optional[str] = None, count: Optional[int] = 20, **kwargs, ) -> List[schemas.MediaServerPlayItem]: """ 获取媒体服务器最新入库条目 """ server_obj: TrimeMedia = self.get_instance(server) if not server_obj: return [] return server_obj.get_latest(num=count) def mediaserver_latest_images( self, server: Optional[str] = None, count: Optional[int] = 20, remote: Optional[bool] = False, **kwargs, ) -> List[str]: """ 获取媒体服务器最新入库条目的图片 :param server: 媒体服务器名称 :param count: 获取数量 :param remote: True为外网链接, False为内网链接 :return: 图片链接列表 """ server_obj: TrimeMedia = self.get_instance(server) if not server_obj: return [] return server_obj.get_latest_backdrops(num=count, remote=remote)