mirror of
https://github.com/jxxghp/MoviePilot.git
synced 2026-05-07 05:42:40 +08:00
451 lines
10 KiB
Bash
Executable File
451 lines
10 KiB
Bash
Executable File
#!/usr/bin/env bash
|
||
|
||
set -euo pipefail
|
||
|
||
REPO_URL="https://github.com/jxxghp/MoviePilot.git"
|
||
REPO_REF="v2"
|
||
WORKDIR="$PWD"
|
||
APP_DIR_NAME="MoviePilot"
|
||
LINK_CLI="true"
|
||
LINK_PATH=""
|
||
CONFIG_DIR=""
|
||
RUN_WIZARD="true"
|
||
START_AFTER_INSTALL="true"
|
||
NON_INTERACTIVE="false"
|
||
OS_NAME="Unknown"
|
||
PYTHON_BIN=""
|
||
PROMPT_INPUT="/dev/stdin"
|
||
PROMPT_OUTPUT="/dev/stdout"
|
||
HAS_TTY="false"
|
||
PATH_RC_FILE=""
|
||
PATH_UPDATED="false"
|
||
|
||
usage() {
|
||
cat <<EOF
|
||
Usage: $(basename "$0") [OPTIONS]
|
||
|
||
Options:
|
||
--workdir PATH 克隆与安装的目标目录,默认当前目录
|
||
--app-dir NAME MoviePilot 目录名,默认 ${APP_DIR_NAME}
|
||
--repo-url URL 主项目仓库地址
|
||
--config-dir PATH 配置目录,默认使用程序目录外的系统配置目录
|
||
--link-path PATH 全局 moviepilot 软链接位置
|
||
--no-link-cli 安装完成后不创建全局 moviepilot 命令
|
||
--no-wizard 跳过 moviepilot setup 的交互式初始化向导
|
||
--no-start 安装完成后不自动启动服务
|
||
--non-interactive 非交互模式,直接使用传入参数
|
||
-h, --help 显示帮助
|
||
|
||
Examples:
|
||
$(basename "$0")
|
||
$(basename "$0") --workdir ~/Projects
|
||
$(basename "$0") --config-dir ~/.config/moviepilot-local
|
||
$(basename "$0") --non-interactive --workdir ~/Projects --no-start
|
||
EOF
|
||
}
|
||
|
||
repo_dirty() {
|
||
(
|
||
cd "$1"
|
||
git status --porcelain 2>/dev/null | grep -q .
|
||
)
|
||
}
|
||
|
||
sync_repo() {
|
||
if [[ ! -d "$APP_DIR/.git" ]]; then
|
||
echo "==> 克隆 MoviePilot 到 $APP_DIR"
|
||
git clone --branch "$REPO_REF" "$REPO_URL" "$APP_DIR"
|
||
return
|
||
fi
|
||
|
||
echo "==> 复用已有 MoviePilot 仓库: $APP_DIR"
|
||
if repo_dirty "$APP_DIR"; then
|
||
echo "检测到现有仓库包含未提交改动,已停止自动更新。" >&2
|
||
echo "请先清理 $APP_DIR 的本地修改,或换一个新的安装目录后重试。" >&2
|
||
exit 1
|
||
fi
|
||
|
||
(
|
||
cd "$APP_DIR"
|
||
echo "==> 更新本地仓库到 origin/$REPO_REF"
|
||
git fetch --tags origin "$REPO_REF"
|
||
if git show-ref --verify --quiet "refs/heads/$REPO_REF"; then
|
||
git checkout "$REPO_REF"
|
||
else
|
||
git checkout -b "$REPO_REF" "origin/$REPO_REF"
|
||
fi
|
||
git pull --ff-only origin "$REPO_REF"
|
||
)
|
||
}
|
||
|
||
default_config_dir() {
|
||
case "$OS_NAME" in
|
||
macOS)
|
||
printf '%s\n' "$HOME/Library/Application Support/MoviePilot"
|
||
;;
|
||
*)
|
||
printf '%s\n' "${XDG_CONFIG_HOME:-$HOME/.config}/moviepilot"
|
||
;;
|
||
esac
|
||
}
|
||
|
||
setup_prompt_io() {
|
||
if [[ -t 0 && -t 1 ]]; then
|
||
HAS_TTY="true"
|
||
return
|
||
fi
|
||
|
||
if [[ -r /dev/tty && -w /dev/tty ]]; then
|
||
PROMPT_INPUT="/dev/tty"
|
||
PROMPT_OUTPUT="/dev/tty"
|
||
HAS_TTY="true"
|
||
fi
|
||
}
|
||
|
||
detect_os() {
|
||
local uname_s
|
||
uname_s="$(uname -s)"
|
||
|
||
case "$uname_s" in
|
||
Darwin)
|
||
OS_NAME="macOS"
|
||
if command -v brew >/dev/null 2>&1; then
|
||
LINK_PATH="$(brew --prefix)/bin/moviepilot"
|
||
else
|
||
LINK_PATH="/usr/local/bin/moviepilot"
|
||
fi
|
||
;;
|
||
Linux)
|
||
if grep -qi microsoft /proc/version 2>/dev/null; then
|
||
OS_NAME="Linux (WSL)"
|
||
else
|
||
OS_NAME="Linux"
|
||
fi
|
||
LINK_PATH="/usr/local/bin/moviepilot"
|
||
;;
|
||
MINGW*|MSYS*|CYGWIN*)
|
||
OS_NAME="Windows"
|
||
;;
|
||
*)
|
||
OS_NAME="$uname_s"
|
||
LINK_PATH="/usr/local/bin/moviepilot"
|
||
;;
|
||
esac
|
||
|
||
if [[ -z "$CONFIG_DIR" ]]; then
|
||
CONFIG_DIR="$(default_config_dir)"
|
||
fi
|
||
}
|
||
|
||
find_python() {
|
||
if command -v python3 >/dev/null 2>&1; then
|
||
command -v python3
|
||
return 0
|
||
fi
|
||
if command -v python >/dev/null 2>&1; then
|
||
command -v python
|
||
return 0
|
||
fi
|
||
return 1
|
||
}
|
||
|
||
python_version_ok() {
|
||
local python_bin="$1"
|
||
"$python_bin" - <<'PY' >/dev/null 2>&1
|
||
import sys
|
||
raise SystemExit(0 if sys.version_info >= (3, 12) else 1)
|
||
PY
|
||
}
|
||
|
||
python_install_hint() {
|
||
case "$OS_NAME" in
|
||
macOS)
|
||
echo "请先安装 Git、curl 和 Python 3.12,例如:brew install git curl python@3.12" >&2
|
||
;;
|
||
Linux*)
|
||
echo "请先安装 Git、curl 和 Python 3.12,并确保包含 venv 模块。" >&2
|
||
echo "例如 Debian/Ubuntu: sudo apt install git curl python3.12 python3.12-venv" >&2
|
||
echo "例如 Fedora/RHEL: sudo dnf install git curl python3.12" >&2
|
||
;;
|
||
Windows)
|
||
echo "推荐在 WSL、Linux 或 macOS 终端中运行此脚本。" >&2
|
||
;;
|
||
*)
|
||
echo "请先安装 Git、curl 和 Python 3.12。" >&2
|
||
;;
|
||
esac
|
||
}
|
||
|
||
require_prereqs() {
|
||
if [[ "$OS_NAME" == "Windows" ]]; then
|
||
echo "检测到当前环境为 Windows shell,建议改用 WSL、Linux 或 macOS 终端运行。" >&2
|
||
exit 1
|
||
fi
|
||
|
||
if ! command -v git >/dev/null 2>&1; then
|
||
echo "未找到 git。" >&2
|
||
python_install_hint
|
||
exit 1
|
||
fi
|
||
|
||
if ! command -v curl >/dev/null 2>&1; then
|
||
echo "未找到 curl。" >&2
|
||
python_install_hint
|
||
exit 1
|
||
fi
|
||
|
||
PYTHON_BIN="$(find_python || true)"
|
||
if [[ -z "$PYTHON_BIN" ]] || ! python_version_ok "$PYTHON_BIN"; then
|
||
echo "未找到可用的 Python 3.12+ 解释器。" >&2
|
||
python_install_hint
|
||
exit 1
|
||
fi
|
||
}
|
||
|
||
prompt_text() {
|
||
local label="$1"
|
||
local default_value="${2:-}"
|
||
local answer=""
|
||
|
||
if [[ -n "$default_value" ]]; then
|
||
printf '%s [%s]: ' "$label" "$default_value" >"$PROMPT_OUTPUT"
|
||
else
|
||
printf '%s: ' "$label" >"$PROMPT_OUTPUT"
|
||
fi
|
||
|
||
IFS= read -r answer <"$PROMPT_INPUT" || true
|
||
if [[ -z "$answer" ]]; then
|
||
answer="$default_value"
|
||
fi
|
||
printf '%s\n' "$answer"
|
||
}
|
||
|
||
prompt_yes_no() {
|
||
local label="$1"
|
||
local default_value="${2:-y}"
|
||
local answer=""
|
||
local prompt="[y/N]"
|
||
|
||
if [[ "$default_value" == "y" ]]; then
|
||
prompt="[Y/n]"
|
||
fi
|
||
|
||
while true; do
|
||
printf '%s %s: ' "$label" "$prompt" >"$PROMPT_OUTPUT"
|
||
IFS= read -r answer <"$PROMPT_INPUT" || true
|
||
answer="$(printf '%s' "$answer" | tr '[:upper:]' '[:lower:]')"
|
||
if [[ -z "$answer" ]]; then
|
||
answer="$default_value"
|
||
fi
|
||
case "$answer" in
|
||
y|yes) return 0 ;;
|
||
n|no) return 1 ;;
|
||
esac
|
||
printf '请输入 y 或 n。\n' >"$PROMPT_OUTPUT"
|
||
done
|
||
}
|
||
|
||
run_interactive_guide() {
|
||
printf '==> 当前系统: %s\n' "$OS_NAME" >"$PROMPT_OUTPUT"
|
||
printf '==> 将自动拉取 MoviePilot,并下载前端 release、资源文件与本地 Node 运行时\n' >"$PROMPT_OUTPUT"
|
||
|
||
WORKDIR="$(prompt_text "安装目录" "$WORKDIR")"
|
||
APP_DIR_NAME="$(prompt_text "主项目目录名" "$APP_DIR_NAME")"
|
||
CONFIG_DIR="$(prompt_text "配置目录" "$CONFIG_DIR")"
|
||
|
||
if prompt_yes_no "安装过程中进入 MoviePilot 初始化向导" "y"; then
|
||
RUN_WIZARD="true"
|
||
else
|
||
RUN_WIZARD="false"
|
||
fi
|
||
|
||
if prompt_yes_no "安装完成后立即启动前后端服务" "y"; then
|
||
START_AFTER_INSTALL="true"
|
||
else
|
||
START_AFTER_INSTALL="false"
|
||
fi
|
||
}
|
||
|
||
ensure_link_path() {
|
||
if [[ "$LINK_CLI" != "true" ]]; then
|
||
return
|
||
fi
|
||
|
||
if [[ -z "$LINK_PATH" ]]; then
|
||
LINK_PATH="/usr/local/bin/moviepilot"
|
||
fi
|
||
|
||
local link_dir
|
||
link_dir="$(dirname "$LINK_PATH")"
|
||
if mkdir -p "$link_dir" 2>/dev/null && [[ -w "$link_dir" ]]; then
|
||
return
|
||
fi
|
||
|
||
LINK_PATH="$HOME/.local/bin/moviepilot"
|
||
mkdir -p "$(dirname "$LINK_PATH")"
|
||
}
|
||
|
||
detect_rc_file() {
|
||
local shell_name
|
||
shell_name="$(basename "${SHELL:-}")"
|
||
case "$shell_name" in
|
||
zsh)
|
||
printf '%s\n' "$HOME/.zshrc"
|
||
;;
|
||
bash)
|
||
printf '%s\n' "$HOME/.bashrc"
|
||
;;
|
||
*)
|
||
printf '%s\n' "$HOME/.profile"
|
||
;;
|
||
esac
|
||
}
|
||
|
||
ensure_path_configured() {
|
||
if [[ "$LINK_CLI" != "true" ]]; then
|
||
return
|
||
fi
|
||
|
||
local bin_dir
|
||
bin_dir="$(dirname "$LINK_PATH")"
|
||
export PATH="$bin_dir:$PATH"
|
||
|
||
if [[ "$bin_dir" != "$HOME/.local/bin" ]]; then
|
||
return
|
||
fi
|
||
|
||
PATH_RC_FILE="$(detect_rc_file)"
|
||
local export_line='export PATH="$HOME/.local/bin:$PATH"'
|
||
mkdir -p "$(dirname "$PATH_RC_FILE")"
|
||
touch "$PATH_RC_FILE"
|
||
|
||
if ! grep -Fqs "$export_line" "$PATH_RC_FILE"; then
|
||
{
|
||
printf '\n# MoviePilot CLI\n'
|
||
printf '%s\n' "$export_line"
|
||
} >>"$PATH_RC_FILE"
|
||
PATH_UPDATED="true"
|
||
fi
|
||
}
|
||
|
||
while [[ $# -gt 0 ]]; do
|
||
case "$1" in
|
||
--workdir)
|
||
WORKDIR="$2"
|
||
shift 2
|
||
;;
|
||
--app-dir)
|
||
APP_DIR_NAME="$2"
|
||
shift 2
|
||
;;
|
||
--repo-url)
|
||
REPO_URL="$2"
|
||
shift 2
|
||
;;
|
||
--config-dir)
|
||
CONFIG_DIR="$2"
|
||
shift 2
|
||
;;
|
||
--link-path)
|
||
LINK_PATH="$2"
|
||
shift 2
|
||
;;
|
||
--no-link-cli)
|
||
LINK_CLI="false"
|
||
shift
|
||
;;
|
||
--no-wizard)
|
||
RUN_WIZARD="false"
|
||
shift
|
||
;;
|
||
--no-start)
|
||
START_AFTER_INSTALL="false"
|
||
shift
|
||
;;
|
||
--non-interactive)
|
||
NON_INTERACTIVE="true"
|
||
shift
|
||
;;
|
||
-h|--help)
|
||
usage
|
||
exit 0
|
||
;;
|
||
*)
|
||
echo "未知参数: $1" >&2
|
||
usage
|
||
exit 1
|
||
;;
|
||
esac
|
||
done
|
||
|
||
detect_os
|
||
setup_prompt_io
|
||
require_prereqs
|
||
ensure_link_path
|
||
|
||
if [[ "$NON_INTERACTIVE" != "true" && "$HAS_TTY" == "true" ]]; then
|
||
run_interactive_guide
|
||
ensure_link_path
|
||
elif [[ "$RUN_WIZARD" == "true" && "$HAS_TTY" != "true" ]]; then
|
||
echo "==> 未检测到可用终端输入,已跳过初始化向导。安装完成后可手动执行:moviepilot setup --wizard"
|
||
RUN_WIZARD="false"
|
||
fi
|
||
|
||
mkdir -p "$WORKDIR"
|
||
WORKDIR="$(cd "$WORKDIR" && pwd)"
|
||
APP_DIR="$WORKDIR/$APP_DIR_NAME"
|
||
sync_repo
|
||
|
||
cd "$APP_DIR"
|
||
echo "==> 执行本地环境安装与初始化"
|
||
SETUP_ARGS=(setup --config-dir "$CONFIG_DIR")
|
||
if [[ "$RUN_WIZARD" == "true" ]]; then
|
||
SETUP_ARGS+=(--wizard)
|
||
fi
|
||
if [[ "$HAS_TTY" == "true" ]]; then
|
||
./moviepilot "${SETUP_ARGS[@]}" <"$PROMPT_INPUT"
|
||
else
|
||
./moviepilot "${SETUP_ARGS[@]}"
|
||
fi
|
||
|
||
if [[ "$LINK_CLI" == "true" ]]; then
|
||
echo "==> 创建全局 moviepilot 命令到 $LINK_PATH"
|
||
ln -sf "$APP_DIR/moviepilot" "$LINK_PATH"
|
||
ensure_path_configured
|
||
fi
|
||
|
||
if [[ "$START_AFTER_INSTALL" == "true" ]]; then
|
||
echo "==> 启动 MoviePilot 前后端服务"
|
||
./moviepilot start
|
||
fi
|
||
|
||
cat <<EOF
|
||
==> 安装完成
|
||
|
||
系统环境: $OS_NAME
|
||
项目目录: $APP_DIR
|
||
配置目录: $CONFIG_DIR
|
||
Python 解释器: $PYTHON_BIN
|
||
CLI 命令: ${LINK_CLI:-false}
|
||
CLI 路径: ${LINK_PATH:-未创建}
|
||
|
||
使用方式:
|
||
moviepilot status
|
||
moviepilot logs --frontend
|
||
moviepilot logs --stdio
|
||
moviepilot config path
|
||
|
||
完整 CLI 文档:
|
||
$APP_DIR/docs/cli.md
|
||
EOF
|
||
|
||
if [[ "$LINK_CLI" == "true" && "$(dirname "$LINK_PATH")" == "$HOME/.local/bin" ]]; then
|
||
echo
|
||
echo "PATH 说明:"
|
||
if [[ "$PATH_UPDATED" == "true" ]]; then
|
||
echo " 已将 ~/.local/bin 写入 $PATH_RC_FILE"
|
||
fi
|
||
echo " 如果当前终端仍提示找不到 moviepilot,请重新打开终端,或执行:"
|
||
echo " source ${PATH_RC_FILE:-$HOME/.profile}"
|
||
fi
|