mirror of
https://github.com/jxxghp/MoviePilot.git
synced 2026-05-11 09:59:51 +08:00
Compare commits
24 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
1336b2136d | ||
|
|
b20e21e700 | ||
|
|
c27ab4a4c7 | ||
|
|
d9e6532325 | ||
|
|
049f16ba01 | ||
|
|
6541458326 | ||
|
|
9f2912426b | ||
|
|
fde33d267a | ||
|
|
ef7f0afa37 | ||
|
|
bea77a8243 | ||
|
|
b984b83870 | ||
|
|
2153ad48db | ||
|
|
c9c43fde74 | ||
|
|
e2c9742f64 | ||
|
|
3d459a40f7 | ||
|
|
5675cd5b11 | ||
|
|
74a4d0bd66 | ||
|
|
2b8c313019 | ||
|
|
62fb6b80a3 | ||
|
|
eea86528d8 | ||
|
|
84e6abb659 | ||
|
|
da2c755b6d | ||
|
|
51f39be9bc | ||
|
|
21b762e75c |
@@ -11,7 +11,7 @@ ENV LANG="C.UTF-8" \
|
|||||||
PORT=3001 \
|
PORT=3001 \
|
||||||
NGINX_PORT=3000 \
|
NGINX_PORT=3000 \
|
||||||
PROXY_HOST="" \
|
PROXY_HOST="" \
|
||||||
MOVIEPILOT_AUTO_UPDATE=release \
|
MOVIEPILOT_AUTO_UPDATE=false \
|
||||||
AUTH_SITE="iyuu" \
|
AUTH_SITE="iyuu" \
|
||||||
IYUU_SIGN=""
|
IYUU_SIGN=""
|
||||||
WORKDIR "/app"
|
WORKDIR "/app"
|
||||||
|
|||||||
@@ -192,10 +192,7 @@ def reset_plugin(plugin_id: str, _: schemas.TokenPayload = Depends(verify_token)
|
|||||||
# 删除插件所有数据
|
# 删除插件所有数据
|
||||||
PluginManager().delete_plugin_data(plugin_id)
|
PluginManager().delete_plugin_data(plugin_id)
|
||||||
# 重新生效插件
|
# 重新生效插件
|
||||||
PluginManager().init_plugin(plugin_id, {
|
PluginManager().reload_plugin(plugin_id)
|
||||||
"enabled": False,
|
|
||||||
"enable": False
|
|
||||||
})
|
|
||||||
# 注册插件服务
|
# 注册插件服务
|
||||||
Scheduler().update_plugin_job(plugin_id)
|
Scheduler().update_plugin_job(plugin_id)
|
||||||
# 注册插件API
|
# 注册插件API
|
||||||
|
|||||||
@@ -316,34 +316,34 @@ class SearchChain(ChainBase):
|
|||||||
self.progress.update(value=0,
|
self.progress.update(value=0,
|
||||||
text=f"开始搜索,共 {total_num} 个站点 ...",
|
text=f"开始搜索,共 {total_num} 个站点 ...",
|
||||||
key=ProgressKey.Search)
|
key=ProgressKey.Search)
|
||||||
# 多线程
|
|
||||||
executor = ThreadPoolExecutor(max_workers=len(indexer_sites))
|
|
||||||
all_task = []
|
|
||||||
for site in indexer_sites:
|
|
||||||
if area == "imdbid":
|
|
||||||
# 搜索IMDBID
|
|
||||||
task = executor.submit(self.search_torrents, site=site,
|
|
||||||
keywords=[mediainfo.imdb_id] if mediainfo else None,
|
|
||||||
mtype=mediainfo.type if mediainfo else None,
|
|
||||||
page=page)
|
|
||||||
else:
|
|
||||||
# 搜索标题
|
|
||||||
task = executor.submit(self.search_torrents, site=site,
|
|
||||||
keywords=keywords,
|
|
||||||
mtype=mediainfo.type if mediainfo else None,
|
|
||||||
page=page)
|
|
||||||
all_task.append(task)
|
|
||||||
# 结果集
|
# 结果集
|
||||||
results = []
|
results = []
|
||||||
for future in as_completed(all_task):
|
# 多线程
|
||||||
finish_count += 1
|
with ThreadPoolExecutor(max_workers=len(indexer_sites)) as executor:
|
||||||
result = future.result()
|
all_task = []
|
||||||
if result:
|
for site in indexer_sites:
|
||||||
results.extend(result)
|
if area == "imdbid":
|
||||||
logger.info(f"站点搜索进度:{finish_count} / {total_num}")
|
# 搜索IMDBID
|
||||||
self.progress.update(value=finish_count / total_num * 100,
|
task = executor.submit(self.search_torrents, site=site,
|
||||||
text=f"正在搜索{keywords or ''},已完成 {finish_count} / {total_num} 个站点 ...",
|
keywords=[mediainfo.imdb_id] if mediainfo else None,
|
||||||
key=ProgressKey.Search)
|
mtype=mediainfo.type if mediainfo else None,
|
||||||
|
page=page)
|
||||||
|
else:
|
||||||
|
# 搜索标题
|
||||||
|
task = executor.submit(self.search_torrents, site=site,
|
||||||
|
keywords=keywords,
|
||||||
|
mtype=mediainfo.type if mediainfo else None,
|
||||||
|
page=page)
|
||||||
|
all_task.append(task)
|
||||||
|
for future in as_completed(all_task):
|
||||||
|
finish_count += 1
|
||||||
|
result = future.result()
|
||||||
|
if result:
|
||||||
|
results.extend(result)
|
||||||
|
logger.info(f"站点搜索进度:{finish_count} / {total_num}")
|
||||||
|
self.progress.update(value=finish_count / total_num * 100,
|
||||||
|
text=f"正在搜索{keywords or ''},已完成 {finish_count} / {total_num} 个站点 ...",
|
||||||
|
key=ProgressKey.Search)
|
||||||
# 计算耗时
|
# 计算耗时
|
||||||
end_time = datetime.now()
|
end_time = datetime.now()
|
||||||
# 更新进度
|
# 更新进度
|
||||||
|
|||||||
@@ -110,30 +110,24 @@ class SiteChain(ChainBase):
|
|||||||
domain = StringUtils.get_url_domain(site.url)
|
domain = StringUtils.get_url_domain(site.url)
|
||||||
url = f"https://api.{domain}/api/member/profile"
|
url = f"https://api.{domain}/api/member/profile"
|
||||||
headers = {
|
headers = {
|
||||||
"Content-Type": "application/json",
|
|
||||||
"User-Agent": user_agent,
|
"User-Agent": user_agent,
|
||||||
"Accept": "application/json, text/plain, */*",
|
"Accept": "application/json, text/plain, */*",
|
||||||
"Authorization": site.token
|
"x-api-key": site.apikey,
|
||||||
}
|
}
|
||||||
res = RequestUtils(
|
res = RequestUtils(
|
||||||
headers=headers,
|
headers=headers,
|
||||||
proxies=settings.PROXY if site.proxy else None,
|
proxies=settings.PROXY if site.proxy else None,
|
||||||
timeout=site.timeout or 15
|
timeout=site.timeout or 15
|
||||||
).post_res(url=url)
|
).post_res(url=url)
|
||||||
if res and res.status_code == 200:
|
if res is None:
|
||||||
user_info = res.json()
|
return False, "无法打开网站!"
|
||||||
if user_info and user_info.get("data"):
|
if res.status_code == 200:
|
||||||
# 更新最后访问时间
|
user_info = res.json() or {}
|
||||||
res = RequestUtils(headers=headers,
|
if user_info.get("data"):
|
||||||
timeout=site.timeout or 15,
|
return True, "连接成功"
|
||||||
proxies=settings.PROXY if site.proxy else None,
|
return False, user_info.get("message", "鉴权已过期或无效")
|
||||||
referer=f"{site.url}index"
|
else:
|
||||||
).post_res(url=f"https://api.{domain}/api/member/updateLastBrowse")
|
return False, f"错误:{res.status_code} {res.reason}"
|
||||||
if res:
|
|
||||||
return True, "连接成功"
|
|
||||||
else:
|
|
||||||
return True, f"连接成功,但更新状态失败"
|
|
||||||
return False, "鉴权已过期或无效"
|
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def __yema_test(site: Site) -> Tuple[bool, str]:
|
def __yema_test(site: Site) -> Tuple[bool, str]:
|
||||||
|
|||||||
@@ -179,9 +179,9 @@ class SubscribeChain(ChainBase):
|
|||||||
text = f"评分:{mediainfo.vote_average}"
|
text = f"评分:{mediainfo.vote_average}"
|
||||||
# 群发
|
# 群发
|
||||||
if mediainfo.type == MediaType.TV:
|
if mediainfo.type == MediaType.TV:
|
||||||
link = settings.MP_DOMAIN('#/subscribe-tv?tab=mysub')
|
link = settings.MP_DOMAIN('#/subscribe/tv?tab=mysub')
|
||||||
else:
|
else:
|
||||||
link = settings.MP_DOMAIN('#/subscribe-movie?tab=mysub')
|
link = settings.MP_DOMAIN('#/subscribe/movie?tab=mysub')
|
||||||
self.post_message(Notification(mtype=NotificationType.Subscribe,
|
self.post_message(Notification(mtype=NotificationType.Subscribe,
|
||||||
title=f"{mediainfo.title_year} {metainfo.season} 已添加订阅",
|
title=f"{mediainfo.title_year} {metainfo.season} 已添加订阅",
|
||||||
text=text,
|
text=text,
|
||||||
@@ -922,9 +922,9 @@ class SubscribeChain(ChainBase):
|
|||||||
self.subscribeoper.delete(subscribe.id)
|
self.subscribeoper.delete(subscribe.id)
|
||||||
# 发送通知
|
# 发送通知
|
||||||
if mediainfo.type == MediaType.TV:
|
if mediainfo.type == MediaType.TV:
|
||||||
link = settings.MP_DOMAIN('#/subscribe-tv?tab=mysub')
|
link = settings.MP_DOMAIN('#/subscribe/tv?tab=mysub')
|
||||||
else:
|
else:
|
||||||
link = settings.MP_DOMAIN('#/subscribe-movie?tab=mysub')
|
link = settings.MP_DOMAIN('#/subscribe/movie?tab=mysub')
|
||||||
self.post_message(Notification(mtype=NotificationType.Subscribe,
|
self.post_message(Notification(mtype=NotificationType.Subscribe,
|
||||||
title=f'{mediainfo.title_year} {meta.season} 已完成{msgstr}',
|
title=f'{mediainfo.title_year} {meta.season} 已完成{msgstr}',
|
||||||
image=mediainfo.get_message_image(),
|
image=mediainfo.get_message_image(),
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import copy
|
||||||
import importlib
|
import importlib
|
||||||
import threading
|
import threading
|
||||||
import traceback
|
import traceback
|
||||||
@@ -11,8 +12,7 @@ from app.chain.subscribe import SubscribeChain
|
|||||||
from app.chain.system import SystemChain
|
from app.chain.system import SystemChain
|
||||||
from app.chain.transfer import TransferChain
|
from app.chain.transfer import TransferChain
|
||||||
from app.core.config import settings
|
from app.core.config import settings
|
||||||
from app.core.event import Event as ManagerEvent
|
from app.core.event import Event as ManagerEvent, eventmanager, EventManager
|
||||||
from app.core.event import eventmanager, EventManager
|
|
||||||
from app.core.plugin import PluginManager
|
from app.core.plugin import PluginManager
|
||||||
from app.helper.message import MessageHelper
|
from app.helper.message import MessageHelper
|
||||||
from app.helper.thread import ThreadHelper
|
from app.helper.thread import ThreadHelper
|
||||||
@@ -194,7 +194,7 @@ class Command(metaclass=Singleton):
|
|||||||
# 插件事件
|
# 插件事件
|
||||||
self.threader.submit(
|
self.threader.submit(
|
||||||
self.pluginmanager.run_plugin_method,
|
self.pluginmanager.run_plugin_method,
|
||||||
class_name, method_name, event
|
class_name, method_name, copy.deepcopy(event)
|
||||||
)
|
)
|
||||||
|
|
||||||
else:
|
else:
|
||||||
@@ -217,7 +217,7 @@ class Command(metaclass=Singleton):
|
|||||||
if hasattr(class_obj, method_name):
|
if hasattr(class_obj, method_name):
|
||||||
self.threader.submit(
|
self.threader.submit(
|
||||||
getattr(class_obj, method_name),
|
getattr(class_obj, method_name),
|
||||||
event
|
copy.deepcopy(event)
|
||||||
)
|
)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
logger.error(f"事件处理出错:{str(e)} - {traceback.format_exc()}")
|
logger.error(f"事件处理出错:{str(e)} - {traceback.format_exc()}")
|
||||||
|
|||||||
@@ -229,7 +229,7 @@ class Settings(BaseSettings):
|
|||||||
# Github代理服务器,格式:https://mirror.ghproxy.com/
|
# Github代理服务器,格式:https://mirror.ghproxy.com/
|
||||||
GITHUB_PROXY: Optional[str] = ''
|
GITHUB_PROXY: Optional[str] = ''
|
||||||
# 自动检查和更新站点资源包(站点索引、认证等)
|
# 自动检查和更新站点资源包(站点索引、认证等)
|
||||||
AUTO_UPDATE_RESOURCE: bool = True
|
AUTO_UPDATE_RESOURCE: bool = False
|
||||||
# 元数据识别缓存过期时间(小时)
|
# 元数据识别缓存过期时间(小时)
|
||||||
META_CACHE_EXPIRE: int = 0
|
META_CACHE_EXPIRE: int = 0
|
||||||
# 是否启用DOH解析域名
|
# 是否启用DOH解析域名
|
||||||
|
|||||||
@@ -71,7 +71,10 @@ class ReleaseGroupsMatcher(metaclass=Singleton):
|
|||||||
"ultrahd": [],
|
"ultrahd": [],
|
||||||
"others": ['B(?:MDru|eyondHD|TN)', 'C(?:fandora|trlhd|MRG)', 'DON', 'EVO', 'FLUX', 'HONE(?:|yG)',
|
"others": ['B(?:MDru|eyondHD|TN)', 'C(?:fandora|trlhd|MRG)', 'DON', 'EVO', 'FLUX', 'HONE(?:|yG)',
|
||||||
'N(?:oGroup|T(?:b|G))', 'PandaMoon', 'SMURF', 'T(?:EPES|aengoo|rollHD )'],
|
'N(?:oGroup|T(?:b|G))', 'PandaMoon', 'SMURF', 'T(?:EPES|aengoo|rollHD )'],
|
||||||
"anime": ['ANi', 'HYSUB', 'KTXP', 'LoliHouse', 'MCE', 'Nekomoe kissaten', '(?:Lilith|NC)-Raws', '织梦字幕组']
|
"anime": ['ANi', 'HYSUB', 'KTXP', 'LoliHouse', 'MCE', 'Nekomoe kissaten', 'SweetSub', 'MingY',
|
||||||
|
'(?:Lilith|NC)-Raws', '织梦字幕组', '枫叶字幕组', '猎户手抄部', '喵萌奶茶屋', '漫猫字幕社',
|
||||||
|
'霜庭云花Sub', '北宇治字幕组', '氢气烤肉架', '云歌字幕组', '萌樱字幕组','极影字幕社','悠哈璃羽字幕社',
|
||||||
|
'❀拨雪寻春❀', '沸羊羊(?:制作|字幕组)', '(?:桜|樱)都字幕组',]
|
||||||
}
|
}
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
|
|||||||
@@ -219,12 +219,13 @@ class FileTransferModule(_ModuleBase):
|
|||||||
"""
|
"""
|
||||||
# 字幕正则式
|
# 字幕正则式
|
||||||
_zhcn_sub_re = r"([.\[(](((zh[-_])?(cn|ch[si]|sg|sc))|zho?" \
|
_zhcn_sub_re = r"([.\[(](((zh[-_])?(cn|ch[si]|sg|sc))|zho?" \
|
||||||
r"|chinese|(cn|ch[si]|sg|zho?|eng)[-_&](cn|ch[si]|sg|zho?|eng)" \
|
r"|chinese|(cn|ch[si]|sg|zho?|eng)[-_&]?(cn|ch[si]|sg|zho?|eng)" \
|
||||||
r"|简[体中]?)[.\])])" \
|
r"|简[体中]?)[.\])])" \
|
||||||
r"|([\u4e00-\u9fa5]{0,3}[中双][\u4e00-\u9fa5]{0,2}[字文语][\u4e00-\u9fa5]{0,3})" \
|
r"|([\u4e00-\u9fa5]{0,3}[中双][\u4e00-\u9fa5]{0,2}[字文语][\u4e00-\u9fa5]{0,3})" \
|
||||||
r"|简体|简中|JPSC" \
|
r"|简体|简中|JPSC" \
|
||||||
r"|(?<![a-z0-9])gb(?![a-z0-9])"
|
r"|(?<![a-z0-9])gb(?![a-z0-9])"
|
||||||
_zhtw_sub_re = r"([.\[(](((zh[-_])?(hk|tw|cht|tc))" \
|
_zhtw_sub_re = r"([.\[(](((zh[-_])?(hk|tw|cht|tc))" \
|
||||||
|
r"|(cht|eng)[-_&]?(cht|eng)" \
|
||||||
r"|繁[体中]?)[.\])])" \
|
r"|繁[体中]?)[.\])])" \
|
||||||
r"|繁体中[文字]|中[文字]繁体|繁体|JPTC" \
|
r"|繁体中[文字]|中[文字]繁体|繁体|JPTC" \
|
||||||
r"|(?<![a-z0-9])big5(?![a-z0-9])"
|
r"|(?<![a-z0-9])big5(?![a-z0-9])"
|
||||||
|
|||||||
@@ -193,7 +193,6 @@ class MTorrentSpider:
|
|||||||
'id': torrent_id
|
'id': torrent_id
|
||||||
},
|
},
|
||||||
'header': {
|
'header': {
|
||||||
'Content-Type': 'application/json',
|
|
||||||
'User-Agent': f'{self._ua}',
|
'User-Agent': f'{self._ua}',
|
||||||
'Accept': 'application/json, text/plain, */*',
|
'Accept': 'application/json, text/plain, */*',
|
||||||
'x-api-key': self._apikey
|
'x-api-key': self._apikey
|
||||||
|
|||||||
@@ -186,7 +186,7 @@ class StringUtils:
|
|||||||
忽略特殊字符
|
忽略特殊字符
|
||||||
"""
|
"""
|
||||||
# 需要忽略的特殊字符
|
# 需要忽略的特殊字符
|
||||||
CONVERT_EMPTY_CHARS = r"[、.。,,·::;;!!'’\"“”()()\[\]【】「」\-——\+\|\\_/&#~~]"
|
CONVERT_EMPTY_CHARS = r"[、.。,,·::;;!!'’\"“”()()\[\]【】「」\-—―\+\|\\_/&#~~]"
|
||||||
if not text:
|
if not text:
|
||||||
return text
|
return text
|
||||||
if not isinstance(text, list):
|
if not isinstance(text, list):
|
||||||
|
|||||||
@@ -20,7 +20,7 @@ DOH_RESOLVERS=1.0.0.1,1.1.1.1,9.9.9.9,149.112.112.112
|
|||||||
# 元数据识别缓存过期时间,数字型,单位小时,0为系统默认(大内存模式为7天,滞则为3天),调大该值可减少themoviedb的访问次数
|
# 元数据识别缓存过期时间,数字型,单位小时,0为系统默认(大内存模式为7天,滞则为3天),调大该值可减少themoviedb的访问次数
|
||||||
META_CACHE_EXPIRE=0
|
META_CACHE_EXPIRE=0
|
||||||
# 自动检查和更新站点资源包(索引、认证等)
|
# 自动检查和更新站点资源包(索引、认证等)
|
||||||
AUTO_UPDATE_RESOURCE=true
|
AUTO_UPDATE_RESOURCE=false
|
||||||
# 【*】API密钥,建议更换复杂字符串,有Jellyseerr/Overseerr、媒体服务器Webhook等配置以及部分支持API_TOKEN的API中使用
|
# 【*】API密钥,建议更换复杂字符串,有Jellyseerr/Overseerr、媒体服务器Webhook等配置以及部分支持API_TOKEN的API中使用
|
||||||
API_TOKEN=moviepilot
|
API_TOKEN=moviepilot
|
||||||
# 登录页面电影海报,tmdb/bing,tmdb要求能正常连接api.themoviedb.org
|
# 登录页面电影海报,tmdb/bing,tmdb要求能正常连接api.themoviedb.org
|
||||||
|
|||||||
@@ -58,5 +58,5 @@ pystray~=0.19.5
|
|||||||
pyotp~=2.9.0
|
pyotp~=2.9.0
|
||||||
Pinyin2Hanzi~=0.1.1
|
Pinyin2Hanzi~=0.1.1
|
||||||
pywebpush~=2.0.0
|
pywebpush~=2.0.0
|
||||||
py115~=0.0.4
|
py115j~=0.0.6
|
||||||
oss2~=2.18.6
|
oss2~=2.18.6
|
||||||
@@ -1 +1 @@
|
|||||||
APP_VERSION = 'v1.9.12'
|
APP_VERSION = 'v1.9.19'
|
||||||
|
|||||||
Reference in New Issue
Block a user