import multiprocessing import os import signal import sys import threading from types import FrameType import uvicorn as uvicorn from PIL import Image from fastapi import FastAPI from fastapi.middleware.cors import CORSMiddleware from uvicorn import Config from app.utils.system import SystemUtils # 禁用输出 if SystemUtils.is_frozen(): sys.stdout = open(os.devnull, 'w') sys.stderr = open(os.devnull, 'w') from app.core.config import settings, global_vars from app.core.module import ModuleManager # SitesHelper涉及资源包拉取,提前引入并容错提示 try: from app.helper.sites import SitesHelper except ImportError as e: error_message = f"错误: {str(e)}\n站点认证及索引相关资源导入失败,请尝试重建容器或手动拉取资源" print(error_message, file=sys.stderr) sys.exit(1) from app.core.plugin import PluginManager from app.db.init import init_db, update_db, init_super_user from app.helper.thread import ThreadHelper from app.helper.display import DisplayHelper from app.helper.resource import ResourceHelper from app.helper.message import MessageHelper from app.scheduler import Scheduler from app.command import Command, CommandChian from app.schemas import Notification, NotificationType # App App = FastAPI(title=settings.PROJECT_NAME, openapi_url=f"{settings.API_V1_STR}/openapi.json") # 跨域 App.add_middleware( CORSMiddleware, allow_origins=settings.ALLOWED_HOSTS, allow_credentials=True, allow_methods=["*"], allow_headers=["*"], ) # uvicorn服务 Server = uvicorn.Server(Config(App, host=settings.HOST, port=settings.PORT, reload=settings.DEV, workers=multiprocessing.cpu_count())) def init_routers(): """ 初始化路由 """ from app.api.apiv1 import api_router from app.api.servarr import arr_router from app.api.servcookie import cookie_router # API路由 App.include_router(api_router, prefix=settings.API_V1_STR) # Radarr、Sonarr路由 App.include_router(arr_router, prefix="/api/v3") # CookieCloud路由 App.include_router(cookie_router, prefix="/cookiecloud") def start_frontend(): """ 启动前端服务 """ # 仅Windows可执行文件支持内嵌nginx if not SystemUtils.is_frozen() \ or not SystemUtils.is_windows(): return # 临时Nginx目录 nginx_path = settings.ROOT_PATH / 'nginx' if not nginx_path.exists(): return # 配置目录下的Nginx目录 run_nginx_dir = settings.CONFIG_PATH.with_name('nginx') if not run_nginx_dir.exists(): # 移动到配置目录 SystemUtils.move(nginx_path, run_nginx_dir) # 启动Nginx import subprocess subprocess.Popen("start nginx.exe", cwd=run_nginx_dir, shell=True) def stop_frontend(): """ 停止前端服务 """ if not SystemUtils.is_frozen() \ or not SystemUtils.is_windows(): return import subprocess subprocess.Popen(f"taskkill /f /im nginx.exe", shell=True) def start_tray(): """ 启动托盘图标 """ if not SystemUtils.is_frozen(): return if not SystemUtils.is_windows(): return def open_web(): """ 调用浏览器打开前端页面 """ import webbrowser webbrowser.open(f"http://localhost:{settings.NGINX_PORT}") def quit_app(): """ 退出程序 """ TrayIcon.stop() Server.should_exit = True import pystray # 托盘图标 TrayIcon = pystray.Icon( settings.PROJECT_NAME, icon=Image.open(settings.ROOT_PATH / 'app.ico'), menu=pystray.Menu( pystray.MenuItem( '打开', open_web, ), pystray.MenuItem( '退出', quit_app, ) ) ) # 启动托盘图标 threading.Thread(target=TrayIcon.run, daemon=True).start() def check_auth(): """ 检查认证状态 """ if SitesHelper().auth_level < 2: err_msg = "用户认证失败,站点相关功能将无法使用!" MessageHelper().put(f"注意:{err_msg}", title="用户认证", role="system") CommandChian().post_message( Notification( mtype=NotificationType.Manual, title="MoviePilot用户认证", text=err_msg, link=settings.MP_DOMAIN('#/site') ) ) def singal_handle(): """ 监听停止信号 """ def stop_event(signum: int, _: FrameType): """ SIGTERM信号处理 """ print(f"接收到停止信号:{signum},正在停止系统...") global_vars.stop_system() # 设置信号处理程序 signal.signal(signal.SIGTERM, stop_event) signal.signal(signal.SIGINT, stop_event) @App.on_event("shutdown") def shutdown_server(): """ 服务关闭 """ # 停止模块 ModuleManager().stop() # 停止插件 PluginManager().stop() PluginManager().stop_monitor() # 停止事件消费 Command().stop() # 停止虚拟显示 DisplayHelper().stop() # 停止定时服务 Scheduler().stop() # 停止线程池 ThreadHelper().shutdown() # 停止前端服务 stop_frontend() @App.on_event("startup") def start_module(): """ 启动模块 """ # 初始化超级管理员 init_super_user() # 虚拟显示 DisplayHelper() # 站点管理 SitesHelper() # 资源包检测 ResourceHelper() # 加载模块 ModuleManager() # 安装在线插件 PluginManager().install_online_plugin() # 加载插件 PluginManager().start() # 启动定时服务 Scheduler() # 启动事件消费 Command() # 初始化路由 init_routers() # 启动前端服务 start_frontend() # 检查认证状态 check_auth() # 监听停止信号 singal_handle() if __name__ == '__main__': # 启动托盘 start_tray() # 初始化数据库 init_db() # 更新数据库 update_db() # 启动API服务 Server.run()