mirror of
https://github.com/jxxghp/MoviePilot.git
synced 2026-06-29 00:06:27 +08:00
170 lines
5.8 KiB
Python
170 lines
5.8 KiB
Python
from __future__ import annotations
|
||
|
||
import os
|
||
import shutil
|
||
from dataclasses import dataclass, field
|
||
from pathlib import Path
|
||
from typing import Literal
|
||
from urllib.parse import urlsplit, urlunsplit
|
||
|
||
|
||
PackageBackend = Literal["uv", "pip"]
|
||
|
||
|
||
@dataclass(frozen=True)
|
||
class PackageInstallRequest:
|
||
"""
|
||
Python 包安装请求,集中描述依赖文件、工具缓存、代理和本地 wheels 候选源。
|
||
"""
|
||
|
||
requirements_file: Path
|
||
python_bin: Path
|
||
find_links_dirs: list[Path] = field(default_factory=list)
|
||
constraints_file: Path | None = None
|
||
config_dir: Path = Path("/config")
|
||
package_cache_root: Path | None = None
|
||
pip_index_url: str | None = None
|
||
proxy_url: str | None = None
|
||
purpose: str = "plugin"
|
||
|
||
|
||
@dataclass(frozen=True)
|
||
class PackageInstallStrategy:
|
||
"""
|
||
单次安装尝试的完整执行信息,命令和日志展示命令分离以避免泄露凭据。
|
||
"""
|
||
|
||
strategy_name: str
|
||
backend: PackageBackend
|
||
command: list[str]
|
||
env: dict[str, str]
|
||
safe_log_command: list[str]
|
||
|
||
|
||
def redact_url(value: str) -> str:
|
||
"""
|
||
脱敏 URL 中的 userinfo,保留 scheme、host、path、query 便于定位镜像源。
|
||
"""
|
||
parsed = urlsplit(value)
|
||
if "@" not in parsed.netloc:
|
||
return value
|
||
host = parsed.netloc.rsplit("@", 1)[-1]
|
||
return urlunsplit((parsed.scheme, host, parsed.path, parsed.query, parsed.fragment))
|
||
|
||
|
||
def redact_command(command: list[str]) -> list[str]:
|
||
"""
|
||
脱敏命令参数中的 URL 凭据,用于日志展示。
|
||
"""
|
||
return [redact_url(item) if "://" in item else item for item in command]
|
||
|
||
|
||
def build_package_install_env(request: PackageInstallRequest, include_moviepilot_proxy: bool = True) -> dict[str, str]:
|
||
"""
|
||
构造 pip/uv 安装子进程环境,默认把包下载缓存放到持久化配置目录。
|
||
"""
|
||
env = os.environ.copy()
|
||
config_dir = Path(request.config_dir)
|
||
if request.package_cache_root:
|
||
package_cache_root = Path(request.package_cache_root)
|
||
env["PACKAGE_CACHE_ROOT"] = str(package_cache_root)
|
||
else:
|
||
package_cache_root = Path(env.get("PACKAGE_CACHE_ROOT") or config_dir / ".cache")
|
||
env.setdefault("PACKAGE_CACHE_ROOT", str(package_cache_root))
|
||
env.setdefault("PIP_CACHE_DIR", str(package_cache_root / "pip"))
|
||
env.setdefault("UV_CACHE_DIR", str(package_cache_root / "uv"))
|
||
proxy = (request.proxy_url or "").strip()
|
||
if proxy and include_moviepilot_proxy:
|
||
for key in ("HTTP_PROXY", "HTTPS_PROXY", "http_proxy", "https_proxy"):
|
||
env[key] = proxy
|
||
return env
|
||
|
||
|
||
def _find_uv(python_bin: Path) -> Path | None:
|
||
"""
|
||
优先使用解释器同目录 uv,保证虚拟环境内 wrapper 与真实安装环境一致。
|
||
"""
|
||
uv_name = "uv.exe" if os.name == "nt" else "uv"
|
||
sibling = python_bin.with_name(uv_name)
|
||
if sibling.exists():
|
||
return sibling
|
||
found = shutil.which("uv")
|
||
return Path(found) if found else None
|
||
|
||
|
||
def _base_install_args(request: PackageInstallRequest) -> list[str]:
|
||
args: list[str] = []
|
||
for directory in request.find_links_dirs:
|
||
args.extend(["--find-links", str(directory)])
|
||
if request.constraints_file:
|
||
args.extend(["-c", str(request.constraints_file)])
|
||
args.extend(["-r", str(request.requirements_file)])
|
||
return args
|
||
|
||
|
||
def _network_variants(request: PackageInstallRequest) -> list[tuple[str, bool, bool]]:
|
||
has_index = bool((request.pip_index_url or "").strip())
|
||
has_proxy = bool((request.proxy_url or "").strip())
|
||
variants: list[tuple[str, bool, bool]] = []
|
||
if has_index and has_proxy:
|
||
variants.append(("镜像+代理", True, True))
|
||
if has_index:
|
||
variants.append(("镜像", True, False))
|
||
if has_proxy:
|
||
variants.append(("代理", False, True))
|
||
variants.append(("直连", False, False))
|
||
return variants
|
||
|
||
|
||
def _build_uv_command(uv_bin: Path, request: PackageInstallRequest, use_index: bool) -> list[str]:
|
||
command = [str(uv_bin), "pip", "install", "--python", str(request.python_bin)]
|
||
if use_index and request.pip_index_url:
|
||
command.extend(["--default-index", request.pip_index_url])
|
||
command.extend(_base_install_args(request))
|
||
return command
|
||
|
||
|
||
def _build_pip_command(request: PackageInstallRequest, use_index: bool) -> list[str]:
|
||
command = [str(request.python_bin), "-m", "pip", "install"]
|
||
if use_index and request.pip_index_url:
|
||
command.extend(["-i", request.pip_index_url])
|
||
command.extend(_base_install_args(request))
|
||
return command
|
||
|
||
|
||
def build_package_install_strategies(request: PackageInstallRequest) -> list[PackageInstallStrategy]:
|
||
"""
|
||
按 uv 优先、pip 兜底顺序构造网络降级策略。
|
||
"""
|
||
strategies: list[PackageInstallStrategy] = []
|
||
variants = _network_variants(request)
|
||
uv_bin = _find_uv(Path(request.python_bin))
|
||
|
||
if uv_bin:
|
||
for variant_name, use_index, use_proxy in variants:
|
||
command = _build_uv_command(uv_bin, request, use_index)
|
||
env = build_package_install_env(request, include_moviepilot_proxy=use_proxy)
|
||
strategies.append(
|
||
PackageInstallStrategy(
|
||
strategy_name=f"uv:{variant_name}",
|
||
backend="uv",
|
||
command=command,
|
||
env=env,
|
||
safe_log_command=redact_command(command),
|
||
)
|
||
)
|
||
|
||
for variant_name, use_index, use_proxy in variants:
|
||
command = _build_pip_command(request, use_index)
|
||
env = build_package_install_env(request, include_moviepilot_proxy=use_proxy)
|
||
strategies.append(
|
||
PackageInstallStrategy(
|
||
strategy_name=f"pip:{variant_name}",
|
||
backend="pip",
|
||
command=command,
|
||
env=env,
|
||
safe_log_command=redact_command(command),
|
||
)
|
||
)
|
||
return strategies
|