From f320e2a1ef9ad115d10315ea5ff8fa56f20894f3 Mon Sep 17 00:00:00 2001 From: wnlen <62139570+wnlen@users.noreply.github.com> Date: Wed, 14 Jan 2026 16:29:10 +0800 Subject: [PATCH] Update update.sh --- update.sh | 272 ++++++++++++++++++++++++++++-------------------------- 1 file changed, 142 insertions(+), 130 deletions(-) diff --git a/update.sh b/update.sh index 31c7fad..7cc6629 100644 --- a/update.sh +++ b/update.sh @@ -1,12 +1,17 @@ -#!/bin/bash +#!/usr/bin/env bash +set -euo pipefail #################### 脚本初始化任务 #################### -# 获取脚本工作目录绝对路径 -export Server_Dir=$(cd $(dirname "${BASH_SOURCE[0]}") && pwd) +Server_Dir="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)" # 加载.env变量文件 -source $Server_Dir/.env +if [ ! -f "$Server_Dir/.env" ]; then + echo -e "\033[31m[ERROR]\033[0m 未找到 .env:$Server_Dir/.env" + exit 1 +fi +# shellcheck disable=SC1090 +source "$Server_Dir/.env" #################### 变量设置 #################### @@ -14,195 +19,202 @@ Conf_Dir="$Server_Dir/conf" Temp_Dir="$Server_Dir/temp" Log_Dir="$Server_Dir/logs" -# 将 CLASH_URL 变量的值赋给 URL 变量,并检查 CLASH_URL 是否为空 -URL=${CLASH_URL:?Error: CLASH_URL variable is not set or empty} +mkdir -p "$Conf_Dir" "$Temp_Dir" "$Log_Dir" + +URL="${CLASH_URL:?Error: CLASH_URL variable is not set or empty}" # 获取 CLASH_SECRET 值,若未设置则尝试读取旧配置,否则生成随机数 -Secret=${CLASH_SECRET:-} +Secret="${CLASH_SECRET:-}" if [ -z "$Secret" ] && [ -f "$Conf_Dir/config.yaml" ]; then - Secret=$(awk -F': ' '/^secret:/{print $2; exit}' "$Conf_Dir/config.yaml") + Secret="$(awk -F': ' '/^secret:/{print $2; exit}' "$Conf_Dir/config.yaml" || true)" fi if [ -z "$Secret" ]; then - Secret=$(openssl rand -hex 32) + Secret="$(openssl rand -hex 32)" fi -# 设置默认值 -CLASH_HTTP_PORT=${CLASH_HTTP_PORT:-7890} -CLASH_SOCKS_PORT=${CLASH_SOCKS_PORT:-7891} -CLASH_REDIR_PORT=${CLASH_REDIR_PORT:-7892} -CLASH_LISTEN_IP=${CLASH_LISTEN_IP:-0.0.0.0} -CLASH_ALLOW_LAN=${CLASH_ALLOW_LAN:-false} -EXTERNAL_CONTROLLER_ENABLED=${EXTERNAL_CONTROLLER_ENABLED:-true} -EXTERNAL_CONTROLLER=${EXTERNAL_CONTROLLER:-127.0.0.1:9090} -ALLOW_INSECURE_TLS=${ALLOW_INSECURE_TLS:-false} +CLASH_HTTP_PORT="${CLASH_HTTP_PORT:-7890}" +CLASH_SOCKS_PORT="${CLASH_SOCKS_PORT:-7891}" +CLASH_REDIR_PORT="${CLASH_REDIR_PORT:-7892}" +CLASH_LISTEN_IP="${CLASH_LISTEN_IP:-0.0.0.0}" +CLASH_ALLOW_LAN="${CLASH_ALLOW_LAN:-false}" +EXTERNAL_CONTROLLER_ENABLED="${EXTERNAL_CONTROLLER_ENABLED:-true}" +EXTERNAL_CONTROLLER="${EXTERNAL_CONTROLLER:-127.0.0.1:9090}" +ALLOW_INSECURE_TLS="${ALLOW_INSECURE_TLS:-false}" +CLASH_HEADERS="${CLASH_HEADERS:-}" +# 工具脚本 +# shellcheck disable=SC1090 source "$Server_Dir/scripts/port_utils.sh" -CLASH_HTTP_PORT=$(resolve_port_value "HTTP" "$CLASH_HTTP_PORT") -CLASH_SOCKS_PORT=$(resolve_port_value "SOCKS" "$CLASH_SOCKS_PORT") -CLASH_REDIR_PORT=$(resolve_port_value "REDIR" "$CLASH_REDIR_PORT") -EXTERNAL_CONTROLLER=$(resolve_host_port "External Controller" "$EXTERNAL_CONTROLLER" "0.0.0.0") +CLASH_HTTP_PORT="$(resolve_port_value "HTTP" "$CLASH_HTTP_PORT")" +CLASH_SOCKS_PORT="$(resolve_port_value "SOCKS" "$CLASH_SOCKS_PORT")" +CLASH_REDIR_PORT="$(resolve_port_value "REDIR" "$CLASH_REDIR_PORT")" +EXTERNAL_CONTROLLER="$(resolve_host_port "External Controller" "$EXTERNAL_CONTROLLER" "0.0.0.0")" +# shellcheck disable=SC1090 source "$Server_Dir/scripts/config_utils.sh" -#################### 函数定义 #################### +#################### action / if_success #################### -# 自定义action函数,实现通用action功能 -success() { - echo -en "\\033[60G[\\033[1;32m OK \\033[0;39m]\r" - return 0 -} +success() { echo -en "\\033[60G[\\033[1;32m OK \\033[0;39m]\r"; return 0; } +failure() { local rc=$?; echo -en "\\033[60G[\\033[1;31mFAILED\\033[0;39m]\r"; [ -x /bin/plymouth ] && /bin/plymouth --details; return $rc; } +action() { local STRING rc; STRING=$1; echo -n "$STRING "; shift; "$@" && success || failure; rc=$?; echo; return $rc; } -failure() { - local rc=$? - echo -en "\\033[60G[\\033[1;31mFAILED\\033[0;39m]\r" - [ -x /bin/plymouth ] && /bin/plymouth --details - return $rc -} - -action() { - local STRING rc - - STRING=$1 - echo -n "$STRING " - shift - "$@" && success $"$STRING" || failure $"$STRING" - rc=$? - echo - return $rc -} - -# 判断命令是否正常执行 函数 if_success() { - local ReturnStatus=$3 - if [ $ReturnStatus -eq 0 ]; then - action "$1" /bin/true - else - action "$2" /bin/false - exit 1 - fi + local ok_msg="$1" fail_msg="$2" st="$3" + if [ "$st" -eq 0 ]; then + action "$ok_msg" /bin/true + else + action "$fail_msg" /bin/false + exit 1 + fi } #################### 任务执行 #################### -## 临时取消环境变量 -unset http_proxy -unset https_proxy -unset no_proxy -unset HTTP_PROXY -unset HTTPS_PROXY -unset NO_PROXY +# 临时取消环境变量(避免被自身代理影响下载) +unset http_proxy https_proxy no_proxy HTTP_PROXY HTTPS_PROXY NO_PROXY || true -## Clash 订阅地址检测及配置文件下载 echo -e '\n正在检测订阅地址...' Text1="Clash订阅地址可访问!" Text2="Clash订阅地址不可访问!" -# 构建检测 curl 命令,添加自定义请求头 CHECK_CMD=(curl -o /dev/null -L -sS --retry 5 -m 10 --connect-timeout 10 -w "%{http_code}") if [ "$ALLOW_INSECURE_TLS" = "true" ]; then - CHECK_CMD+=(-k) - echo -e "\033[33m[WARN] 已启用不安全的 TLS 下载(跳过证书校验)\033[0m" + CHECK_CMD+=(-k) + echo -e "\033[33m[WARN] 已启用不安全的 TLS 下载(跳过证书校验)\033[0m" fi if [ -n "$CLASH_HEADERS" ]; then - CHECK_CMD+=(-H "$CLASH_HEADERS") + CHECK_CMD+=(-H "$CLASH_HEADERS") fi CHECK_CMD+=("$URL") -# 检查订阅地址 -status_code=$("${CHECK_CMD[@]}") +status_code="$("${CHECK_CMD[@]}")" echo "$status_code" | grep -E '^[23][0-9]{2}$' &>/dev/null ReturnStatus=$? -if_success $Text1 $Text2 $ReturnStatus +if_success "$Text1" "$Text2" "$ReturnStatus" -# 拉取更新config.yml文件 echo -e '\n正在下载Clash配置文件...' -Text3="配置文件config.yaml下载成功!" -Text4="配置文件config.yaml下载失败,退出更新!" +Text3="配置文件下载成功!" +Text4="配置文件下载失败,退出更新!" -# 构建 curl 命令,添加自定义请求头 -CURL_CMD=(curl -L -sS --retry 5 -m 10 -o "$Temp_Dir/clash.yaml") +CURL_CMD=(curl -L -sS --retry 5 -m 20 -o "$Temp_Dir/clash.yaml") if [ "$ALLOW_INSECURE_TLS" = "true" ]; then - CURL_CMD+=(-k) + CURL_CMD+=(-k) fi if [ -n "$CLASH_HEADERS" ]; then - CURL_CMD+=(-H "$CLASH_HEADERS") + CURL_CMD+=(-H "$CLASH_HEADERS") fi CURL_CMD+=("$URL") -# 尝试使用curl进行下载 -"${CURL_CMD[@]}" +"${CURL_CMD[@]}" || true ReturnStatus=$? + if [ $ReturnStatus -ne 0 ]; then - # 如果使用curl下载失败,尝试使用wget进行下载 - WGET_CMD=(wget -q -O "$Temp_Dir/clash.yaml") - if [ "$ALLOW_INSECURE_TLS" = "true" ]; then - WGET_CMD+=(--no-check-certificate) - fi - if [ -n "$CLASH_HEADERS" ]; then - WGET_CMD+=(--header="$CLASH_HEADERS") - fi - WGET_CMD+=("$URL") - - for i in {1..10} - do - "${WGET_CMD[@]}" - ReturnStatus=$? - if [ $ReturnStatus -eq 0 ]; then - break - else - continue - fi - done + WGET_CMD=(wget -q -O "$Temp_Dir/clash.yaml") + if [ "$ALLOW_INSECURE_TLS" = "true" ]; then + WGET_CMD+=(--no-check-certificate) + fi + if [ -n "$CLASH_HEADERS" ]; then + WGET_CMD+=(--header="$CLASH_HEADERS") + fi + WGET_CMD+=("$URL") + + for _ in {1..10}; do + "${WGET_CMD[@]}" && ReturnStatus=0 && break || ReturnStatus=$? + done fi -if_success $Text3 $Text4 $ReturnStatus +if_success "$Text3" "$Text4" "$ReturnStatus" -# 重命名clash配置文件 -\cp -a $Temp_Dir/clash.yaml $Temp_Dir/clash_config.yaml +# 基础内容校验(避免 HTML/空文件) +if ! grep -Eq '^(proxies:|proxy-groups:|rules:|mixed-port:|port:)' "$Temp_Dir/clash.yaml"; then + echo -e "\033[31m[ERROR]\033[0m 下载内容不像 Clash 配置(缺少关键字段),请检查订阅是否返回了网页/登录页/错误信息。" + echo -e "可执行:head -n 20 $Temp_Dir/clash.yaml 查看内容" + exit 1 +fi -## 判断订阅内容是否符合clash配置文件标准,尝试转换(需 subconverter 可执行文件支持) -source $Server_Dir/scripts/resolve_subconverter.sh -if [ "$Subconverter_Ready" = "true" ]; then - echo -e '\n判断订阅内容是否符合clash配置文件标准:' - export SUBCONVERTER_BIN="$Subconverter_Bin" - bash $Server_Dir/scripts/clash_profile_conversion.sh - sleep 3 +\cp -a "$Temp_Dir/clash.yaml" "$Temp_Dir/clash_config.yaml" + +# subconverter +# shellcheck disable=SC1090 +source "$Server_Dir/scripts/resolve_subconverter.sh" +if [ "${Subconverter_Ready:-false}" = "true" ]; then + echo -e '\n判断订阅内容是否符合clash配置文件标准:' + export SUBCONVERTER_BIN="$Subconverter_Bin" + bash "$Server_Dir/scripts/clash_profile_conversion.sh" + sleep 1 else - echo -e "\033[33m[WARN] 未检测到可用的 subconverter,跳过订阅转换\033[0m" + echo -e "\033[33m[WARN] 未检测到可用的 subconverter,跳过订阅转换\033[0m" fi -## Clash 配置文件重新格式化及配置 -sed -n '/^proxies:/,$p' $Temp_Dir/clash_config.yaml > $Temp_Dir/proxy.txt +# ========= 生成最终 config.yaml ========= +# 兼容两类订阅: +# A) 全量 config(包含 port/mixed-port 等),直接用订阅为主 +# B) 仅节点列表(含 proxies:),用 templete + proxies 合并 +FULL_CONFIG=false +if grep -Eq '^(port:|mixed-port:|socks-port:|redir-port:)' "$Temp_Dir/clash_config.yaml"; then + FULL_CONFIG=true +fi -# 合并形成新的config.yaml,并替换配置占位符 -cat $Temp_Dir/templete_config.yaml > $Temp_Dir/config.yaml -cat $Temp_Dir/proxy.txt >> $Temp_Dir/config.yaml +if [ "$FULL_CONFIG" = "true" ]; then + echo -e "\n检测到订阅为【全量配置】模式,直接使用订阅生成 config.yaml" + \cp -a "$Temp_Dir/clash_config.yaml" "$Temp_Dir/config.yaml" +else + echo -e "\n检测到订阅为【节点/片段】模式,使用 templete 合并 proxies" + if [ ! -f "$Temp_Dir/templete_config.yaml" ]; then + echo -e "\033[31m[ERROR]\033[0m 未找到 templete_config.yaml:$Temp_Dir/templete_config.yaml" + exit 1 + fi -# 替换配置文件中的占位符为环境变量值 -sed -i "s/CLASH_HTTP_PORT_PLACEHOLDER/${CLASH_HTTP_PORT}/g" $Temp_Dir/config.yaml -sed -i "s/CLASH_SOCKS_PORT_PLACEHOLDER/${CLASH_SOCKS_PORT}/g" $Temp_Dir/config.yaml -sed -i "s/CLASH_REDIR_PORT_PLACEHOLDER/${CLASH_REDIR_PORT}/g" $Temp_Dir/config.yaml -sed -i "s/CLASH_LISTEN_IP_PLACEHOLDER/${CLASH_LISTEN_IP}/g" $Temp_Dir/config.yaml -sed -i "s/CLASH_ALLOW_LAN_PLACEHOLDER/${CLASH_ALLOW_LAN}/g" $Temp_Dir/config.yaml + sed -n '/^proxies:/,$p' "$Temp_Dir/clash_config.yaml" > "$Temp_Dir/proxy.txt" + cat "$Temp_Dir/templete_config.yaml" > "$Temp_Dir/config.yaml" + cat "$Temp_Dir/proxy.txt" >> "$Temp_Dir/config.yaml" +fi -# 配置 external-controller +# 替换占位符(仅在 templete 模式才会命中;全量模式下无害) +sed -i "s/CLASH_HTTP_PORT_PLACEHOLDER/${CLASH_HTTP_PORT}/g" "$Temp_Dir/config.yaml" +sed -i "s/CLASH_SOCKS_PORT_PLACEHOLDER/${CLASH_SOCKS_PORT}/g" "$Temp_Dir/config.yaml" +sed -i "s/CLASH_REDIR_PORT_PLACEHOLDER/${CLASH_REDIR_PORT}/g" "$Temp_Dir/config.yaml" +sed -i "s/CLASH_LISTEN_IP_PLACEHOLDER/${CLASH_LISTEN_IP}/g" "$Temp_Dir/config.yaml" +sed -i "s/CLASH_ALLOW_LAN_PLACEHOLDER/${CLASH_ALLOW_LAN}/g" "$Temp_Dir/config.yaml" + +# external-controller(全量 config 也允许覆盖/写入) if [ "$EXTERNAL_CONTROLLER_ENABLED" = "true" ]; then - sed -i "s/EXTERNAL_CONTROLLER_PLACEHOLDER/${EXTERNAL_CONTROLLER}/g" $Temp_Dir/config.yaml + # 如果已经有 external-controller 则替换;没有则追加 + if grep -qE '^external-controller:' "$Temp_Dir/config.yaml"; then + sed -i "s@^external-controller:.*@external-controller: ${EXTERNAL_CONTROLLER}@g" "$Temp_Dir/config.yaml" + else + echo "external-controller: ${EXTERNAL_CONTROLLER}" >> "$Temp_Dir/config.yaml" + fi else - # 如果禁用 external-controller,则注释掉该行 - sed -i "s/external-controller: 'EXTERNAL_CONTROLLER_PLACEHOLDER'/# external-controller: disabled/g" $Temp_Dir/config.yaml + # 禁用:若存在则注释 + sed -i "s@^external-controller:.*@# external-controller: disabled@g" "$Temp_Dir/config.yaml" || true fi apply_tun_config "$Temp_Dir/config.yaml" apply_mixin_config "$Temp_Dir/config.yaml" "$Server_Dir" -\cp $Temp_Dir/config.yaml $Conf_Dir/ +\cp "$Temp_Dir/config.yaml" "$Conf_Dir/config.yaml" -# Configure Clash Dashboard -Work_Dir=$(cd $(dirname $0); pwd) +# Dashboard +Work_Dir="$(cd "$(dirname "$0")" && pwd)" Dashboard_Dir="${Work_Dir}/dashboard/public" if [ "$EXTERNAL_CONTROLLER_ENABLED" = "true" ]; then - sed -ri "s@^# external-ui:.*@external-ui: ${Dashboard_Dir}@g" $Conf_Dir/config.yaml + # 若有 external-ui 注释行则替换;否则追加 + if grep -qE '^(#\s*)?external-ui:' "$Conf_Dir/config.yaml"; then + sed -ri "s@^(#\s*)?external-ui:.*@external-ui: ${Dashboard_Dir}@g" "$Conf_Dir/config.yaml" + else + echo "external-ui: ${Dashboard_Dir}" >> "$Conf_Dir/config.yaml" + fi fi -sed -r -i '/^secret: /s@(secret: ).*@\1'${Secret}'@g' $Conf_Dir/config.yaml + +# 写入 secret(用 awk 重写,避免 sed 转义问题) +tmpfile="$(mktemp)" +awk -v sec="$Secret" ' + BEGIN{done=0} + /^secret:/ {print "secret: " sec; done=1; next} + {print} + END{ if(done==0) print "secret: " sec } +' "$Conf_Dir/config.yaml" > "$tmpfile" +mv "$tmpfile" "$Conf_Dir/config.yaml" echo -e "\n订阅更新完成,如需生效请执行: bash restart.sh\n"