Update start.sh

This commit is contained in:
wnlen
2026-01-14 16:15:48 +08:00
committed by GitHub
parent 17dbb9ccb3
commit 4fd7deb711

153
start.sh
View File

@ -1,30 +1,35 @@
#!/bin/bash #!/usr/bin/env bash
set -euo pipefail
############################################
# Clash for Linux - start.sh (Full Version)
############################################
# 加载系统函数库(Only for RHEL Linux) # 加载系统函数库(Only for RHEL Linux)
# [ -f /etc/init.d/functions ] && source /etc/init.d/functions [ -f /etc/init.d/functions ] && source /etc/init.d/functions
#################### 脚本初始化任务 #################### #################### 脚本初始化任务 ####################
# 获取脚本工作目录绝对路径 # 获取脚本工作目录绝对路径
export Server_Dir=$(cd $(dirname "${BASH_SOURCE[0]}") && pwd) export Server_Dir
Server_Dir="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
# 加载.env变量文件 # 加载.env变量文件
source $Server_Dir/.env source "$Server_Dir/.env"
# 给二进制启动程序、脚本等添加可执行权限 # 给二进制启动程序、脚本等添加可执行权限
chmod +x $Server_Dir/bin/* 2>/dev/null chmod +x "$Server_Dir/bin/"* 2>/dev/null || true
chmod +x $Server_Dir/scripts/* 2>/dev/null chmod +x "$Server_Dir/scripts/"* 2>/dev/null || true
if [ -f "$Server_Dir/tools/subconverter/subconverter" ]; then if [ -f "$Server_Dir/tools/subconverter/subconverter" ]; then
chmod +x $Server_Dir/tools/subconverter/subconverter 2>/dev/null chmod +x "$Server_Dir/tools/subconverter/subconverter" 2>/dev/null || true
fi fi
#################### 变量设置 #################### #################### 变量设置 ####################
Conf_Dir="$Server_Dir/conf" Conf_Dir="$Server_Dir/conf"
Temp_Dir="$Server_Dir/temp" Temp_Dir="$Server_Dir/temp"
Log_Dir="$Server_Dir/logs" Log_Dir="$Server_Dir/logs"
PID_FILE="${CLASH_PID_FILE:-$Temp_Dir/clash.pid}" PID_FILE="${CLASH_PID_FILE:-$Temp_Dir/clash.pid}"
# 将 CLASH_URL 变量的值赋给 URL 变量,并检查 CLASH_URL 是否为空 # 将 CLASH_URL 变量的值赋给 URL 变量,并检查 CLASH_URL 是否为空
@ -39,38 +44,38 @@ CLASH_SOCKS_PORT=${CLASH_SOCKS_PORT:-7891}
CLASH_REDIR_PORT=${CLASH_REDIR_PORT:-7892} CLASH_REDIR_PORT=${CLASH_REDIR_PORT:-7892}
CLASH_LISTEN_IP=${CLASH_LISTEN_IP:-0.0.0.0} CLASH_LISTEN_IP=${CLASH_LISTEN_IP:-0.0.0.0}
CLASH_ALLOW_LAN=${CLASH_ALLOW_LAN:-false} CLASH_ALLOW_LAN=${CLASH_ALLOW_LAN:-false}
EXTERNAL_CONTROLLER_ENABLED=${EXTERNAL_CONTROLLER_ENABLED:-true} EXTERNAL_CONTROLLER_ENABLED=${EXTERNAL_CONTROLLER_ENABLED:-true}
EXTERNAL_CONTROLLER=${EXTERNAL_CONTROLLER:-127.0.0.1:9090} EXTERNAL_CONTROLLER=${EXTERNAL_CONTROLLER:-127.0.0.1:9090}
ALLOW_INSECURE_TLS=${ALLOW_INSECURE_TLS:-false} ALLOW_INSECURE_TLS=${ALLOW_INSECURE_TLS:-false}
# 端口与配置工具
source "$Server_Dir/scripts/port_utils.sh" source "$Server_Dir/scripts/port_utils.sh"
CLASH_HTTP_PORT=$(resolve_port_value "HTTP" "$CLASH_HTTP_PORT") CLASH_HTTP_PORT="$(resolve_port_value "HTTP" "$CLASH_HTTP_PORT")"
CLASH_SOCKS_PORT=$(resolve_port_value "SOCKS" "$CLASH_SOCKS_PORT") CLASH_SOCKS_PORT="$(resolve_port_value "SOCKS" "$CLASH_SOCKS_PORT")"
CLASH_REDIR_PORT=$(resolve_port_value "REDIR" "$CLASH_REDIR_PORT") CLASH_REDIR_PORT="$(resolve_port_value "REDIR" "$CLASH_REDIR_PORT")"
EXTERNAL_CONTROLLER=$(resolve_host_port "External Controller" "$EXTERNAL_CONTROLLER" "0.0.0.0") EXTERNAL_CONTROLLER="$(resolve_host_port "External Controller" "$EXTERNAL_CONTROLLER" "0.0.0.0")"
source "$Server_Dir/scripts/config_utils.sh" source "$Server_Dir/scripts/config_utils.sh"
#################### 函数定义 #################### #################### 函数定义 ####################
# 自定义action函数实现通用action功能 # 自定义action函数实现通用action功能
success() { success() {
echo -en "\\033[60G[\\033[1;32m OK \\033[0;39m]\r" echo -en "\033[60G[\033[1;32m OK \033[0;39m]\r"
return 0 return 0
} }
failure() { failure() {
local rc=$? local rc=$?
echo -en "\\033[60G[\\033[1;31mFAILED\\033[0;39m]\r" echo -en "\033[60G[\033[1;31mFAILED\033[0;39m]\r"
[ -x /bin/plymouth ] && /bin/plymouth --details [ -x /bin/plymouth ] && /bin/plymouth --details
return $rc return $rc
} }
action() { action() {
local STRING rc local STRING rc
STRING=$1 STRING=$1
echo -n "$STRING " echo -n "$STRING "
shift shift
@ -83,7 +88,7 @@ action() {
# 判断命令是否正常执行 函数 # 判断命令是否正常执行 函数
if_success() { if_success() {
local ReturnStatus=$3 local ReturnStatus=$3
if [ $ReturnStatus -eq 0 ]; then if [ "$ReturnStatus" -eq 0 ]; then
action "$1" /bin/true action "$1" /bin/true
else else
action "$2" /bin/false action "$2" /bin/false
@ -91,33 +96,26 @@ if_success() {
fi fi
} }
#################### 任务执行 #################### #################### 任务执行 ####################
mkdir -p "$Conf_Dir" "$Temp_Dir" "$Log_Dir"
## 获取CPU架构信息 ## 获取CPU架构信息
# Source the script to get CPU architecture source "$Server_Dir/scripts/get_cpu_arch.sh"
source $Server_Dir/scripts/get_cpu_arch.sh
# Check if we obtained CPU architecture # Check if we obtained CPU architecture
if [[ -z "$CpuArch" ]]; then if [[ -z "${CpuArch:-}" ]]; then
echo "Failed to obtain CPU architecture" echo "Failed to obtain CPU architecture"
exit 1 exit 1
fi fi
source "$Server_Dir/scripts/resolve_clash.sh" source "$Server_Dir/scripts/resolve_clash.sh"
## 临时取消环境变量 ## 临时取消环境变量
unset http_proxy unset http_proxy https_proxy no_proxy HTTP_PROXY HTTPS_PROXY NO_PROXY || true
unset https_proxy
unset no_proxy
unset HTTP_PROXY
unset HTTPS_PROXY
unset NO_PROXY
## Clash 订阅地址检测及配置文件下载 ## Clash 订阅地址检测及配置文件下载
# 检查url是否有效 # 检查url是否有效
echo -e '\n正在检测订阅地址...' echo -e '\n正在检测订阅地址...'
Text1="Clash订阅地址可访问" Text1="Clash订阅地址可访问"
@ -125,20 +123,22 @@ Text2="Clash订阅地址不可访问"
# 构建检测 curl 命令,添加自定义请求头 # 构建检测 curl 命令,添加自定义请求头
CHECK_CMD=(curl -o /dev/null -L -sS --retry 5 -m 10 --connect-timeout 10 -w "%{http_code}") CHECK_CMD=(curl -o /dev/null -L -sS --retry 5 -m 10 --connect-timeout 10 -w "%{http_code}")
if [ "$ALLOW_INSECURE_TLS" = "true" ]; then if [ "$ALLOW_INSECURE_TLS" = "true" ]; then
CHECK_CMD+=(-k) CHECK_CMD+=(-k)
echo -e "\033[33m[WARN] 已启用不安全的 TLS 下载(跳过证书校验)\033[0m" echo -e "\033[33m[WARN] 已启用不安全的 TLS 下载(跳过证书校验)\033[0m"
fi fi
if [ -n "$CLASH_HEADERS" ]; then
if [ -n "${CLASH_HEADERS:-}" ]; then
CHECK_CMD+=(-H "$CLASH_HEADERS") CHECK_CMD+=(-H "$CLASH_HEADERS")
fi fi
CHECK_CMD+=("$URL") CHECK_CMD+=("$URL")
# 检查订阅地址 status_code="$("${CHECK_CMD[@]}")"
status_code=$("${CHECK_CMD[@]}")
echo "$status_code" | grep -E '^[23][0-9]{2}$' &>/dev/null echo "$status_code" | grep -E '^[23][0-9]{2}$' &>/dev/null
ReturnStatus=$? ReturnStatus=$?
if_success $Text1 $Text2 $ReturnStatus if_success "$Text1" "$Text2" "$ReturnStatus"
# 拉取更新config.yml文件 # 拉取更新config.yml文件
echo -e '\n正在下载Clash配置文件...' echo -e '\n正在下载Clash配置文件...'
@ -147,30 +147,36 @@ Text4="配置文件config.yaml下载失败退出启动"
# 构建 curl 命令,添加自定义请求头 # 构建 curl 命令,添加自定义请求头
CURL_CMD=(curl -L -sS --retry 5 -m 10 -o "$Temp_Dir/clash.yaml") CURL_CMD=(curl -L -sS --retry 5 -m 10 -o "$Temp_Dir/clash.yaml")
if [ "$ALLOW_INSECURE_TLS" = "true" ]; then if [ "$ALLOW_INSECURE_TLS" = "true" ]; then
CURL_CMD+=(-k) CURL_CMD+=(-k)
fi fi
if [ -n "$CLASH_HEADERS" ]; then
if [ -n "${CLASH_HEADERS:-}" ]; then
CURL_CMD+=(-H "$CLASH_HEADERS") CURL_CMD+=(-H "$CLASH_HEADERS")
fi fi
CURL_CMD+=("$URL") CURL_CMD+=("$URL")
# 尝试使用curl进行下载 # 尝试使用curl进行下载
"${CURL_CMD[@]}" "${CURL_CMD[@]}"
ReturnStatus=$? ReturnStatus=$?
if [ $ReturnStatus -ne 0 ]; then if [ $ReturnStatus -ne 0 ]; then
# 如果使用curl下载失败尝试使用wget进行下载 # 如果使用curl下载失败尝试使用wget进行下载
WGET_CMD=(wget -q -O "$Temp_Dir/clash.yaml") WGET_CMD=(wget -q -O "$Temp_Dir/clash.yaml")
if [ "$ALLOW_INSECURE_TLS" = "true" ]; then if [ "$ALLOW_INSECURE_TLS" = "true" ]; then
WGET_CMD+=(--no-check-certificate) WGET_CMD+=(--no-check-certificate)
fi fi
if [ -n "$CLASH_HEADERS" ]; then
if [ -n "${CLASH_HEADERS:-}" ]; then
WGET_CMD+=(--header="$CLASH_HEADERS") WGET_CMD+=(--header="$CLASH_HEADERS")
fi fi
WGET_CMD+=("$URL") WGET_CMD+=("$URL")
for i in {1..10} for i in {1..10}; do
do
"${WGET_CMD[@]}" "${WGET_CMD[@]}"
ReturnStatus=$? ReturnStatus=$?
if [ $ReturnStatus -eq 0 ]; then if [ $ReturnStatus -eq 0 ]; then
@ -180,69 +186,71 @@ if [ $ReturnStatus -ne 0 ]; then
fi fi
done done
fi fi
if_success $Text3 $Text4 $ReturnStatus
if_success "$Text3" "$Text4" "$ReturnStatus"
# 重命名clash配置文件 # 重命名clash配置文件
\cp -a $Temp_Dir/clash.yaml $Temp_Dir/clash_config.yaml \cp -a "$Temp_Dir/clash.yaml" "$Temp_Dir/clash_config.yaml"
## 判断订阅内容是否符合clash配置文件标准尝试转换需 subconverter 可执行文件支持) ## 判断订阅内容是否符合clash配置文件标准尝试转换需 subconverter 可执行文件支持)
source $Server_Dir/scripts/resolve_subconverter.sh source "$Server_Dir/scripts/resolve_subconverter.sh"
if [ "$Subconverter_Ready" = "true" ]; then
if [ "${Subconverter_Ready:-false}" = "true" ]; then
echo -e '\n判断订阅内容是否符合clash配置文件标准:' echo -e '\n判断订阅内容是否符合clash配置文件标准:'
export SUBCONVERTER_BIN="$Subconverter_Bin" export SUBCONVERTER_BIN="$Subconverter_Bin"
bash $Server_Dir/scripts/clash_profile_conversion.sh bash "$Server_Dir/scripts/clash_profile_conversion.sh"
sleep 3 sleep 3
else else
echo -e "\033[33m[WARN] 未检测到可用的 subconverter跳过订阅转换\033[0m" echo -e "\033[33m[WARN] 未检测到可用的 subconverter跳过订阅转换\033[0m"
fi fi
## Clash 配置文件重新格式化及配置 ## Clash 配置文件重新格式化及配置
# 取出代理相关配置 # 取出代理相关配置
#sed -n '/^proxies:/,$p' $Temp_Dir/clash.yaml > $Temp_Dir/proxy.txt sed -n '/^proxies:/,$p' "$Temp_Dir/clash_config.yaml" > "$Temp_Dir/proxy.txt"
sed -n '/^proxies:/,$p' $Temp_Dir/clash_config.yaml > $Temp_Dir/proxy.txt
# 合并形成新的config.yaml并替换配置占位符 # 合并形成新的config.yaml并替换配置占位符
cat $Temp_Dir/templete_config.yaml > $Temp_Dir/config.yaml cat "$Temp_Dir/templete_config.yaml" > "$Temp_Dir/config.yaml"
cat $Temp_Dir/proxy.txt >> $Temp_Dir/config.yaml cat "$Temp_Dir/proxy.txt" >> "$Temp_Dir/config.yaml"
# 替换配置文件中的占位符为环境变量值 # 替换配置文件中的占位符为环境变量值
sed -i "s/CLASH_HTTP_PORT_PLACEHOLDER/${CLASH_HTTP_PORT}/g" $Temp_Dir/config.yaml 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_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_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_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 -i "s/CLASH_ALLOW_LAN_PLACEHOLDER/${CLASH_ALLOW_LAN}/g" "$Temp_Dir/config.yaml"
# 配置 external-controller # 配置 external-controller
if [ "$EXTERNAL_CONTROLLER_ENABLED" = "true" ]; then if [ "$EXTERNAL_CONTROLLER_ENABLED" = "true" ]; then
sed -i "s/EXTERNAL_CONTROLLER_PLACEHOLDER/${EXTERNAL_CONTROLLER}/g" $Temp_Dir/config.yaml sed -i "s/EXTERNAL_CONTROLLER_PLACEHOLDER/${EXTERNAL_CONTROLLER}/g" "$Temp_Dir/config.yaml"
else else
# 如果禁用 external-controller则注释掉该行 # 如果禁用 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_PLACEHOLDER'/# external-controller: disabled/g" "$Temp_Dir/config.yaml"
fi fi
apply_tun_config "$Temp_Dir/config.yaml" apply_tun_config "$Temp_Dir/config.yaml"
apply_mixin_config "$Temp_Dir/config.yaml" "$Server_Dir" apply_mixin_config "$Temp_Dir/config.yaml" "$Server_Dir"
\cp $Temp_Dir/config.yaml $Conf_Dir/ \cp "$Temp_Dir/config.yaml" "$Conf_Dir/"
# Configure Clash Dashboard # Configure Clash Dashboard
Work_Dir=$(cd $(dirname $0); pwd) Work_Dir="$(cd "$(dirname "$0")" && pwd)"
Dashboard_Dir="${Work_Dir}/dashboard/public" Dashboard_Dir="${Work_Dir}/dashboard/public"
if [ "$EXTERNAL_CONTROLLER_ENABLED" = "true" ]; then if [ "$EXTERNAL_CONTROLLER_ENABLED" = "true" ]; then
sed -ri "s@^# external-ui:.*@external-ui: ${Dashboard_Dir}@g" $Conf_Dir/config.yaml sed -ri "s@^# external-ui:.*@external-ui: ${Dashboard_Dir}@g" "$Conf_Dir/config.yaml"
fi fi
sed -r -i '/^secret: /s@(secret: ).*@\1'${Secret}'@g' $Conf_Dir/config.yaml
sed -r -i "/^secret: /s@(secret: ).*@\1${Secret}@g" "$Conf_Dir/config.yaml"
## 启动Clash服务 ## 启动Clash服务
echo -e '\n正在启动Clash服务...' echo -e '\n正在启动Clash服务...'
Text5="服务启动成功!" Text5="服务启动成功!"
Text6="服务启动失败!" Text6="服务启动失败!"
Clash_Bin=$(resolve_clash_bin "$Server_Dir" "$CpuArch")
Clash_Bin="$(resolve_clash_bin "$Server_Dir" "$CpuArch")"
ReturnStatus=$? ReturnStatus=$?
if [ $ReturnStatus -eq 0 ]; then if [ $ReturnStatus -eq 0 ]; then
nohup "$Clash_Bin" -d "$Conf_Dir" &> "$Log_Dir/clash.log" & nohup "$Clash_Bin" -d "$Conf_Dir" &> "$Log_Dir/clash.log" &
PID=$! PID=$!
@ -251,7 +259,8 @@ if [ $ReturnStatus -eq 0 ]; then
echo "$PID" > "$PID_FILE" echo "$PID" > "$PID_FILE"
fi fi
fi fi
if_success $Text5 $Text6 $ReturnStatus
if_success "$Text5" "$Text6" "$ReturnStatus"
# Output Dashboard access address and Secret # Output Dashboard access address and Secret
echo '' echo ''
@ -265,6 +274,7 @@ echo ''
# 添加环境变量 - 使用配置的端口 # 添加环境变量 - 使用配置的端口
Env_File="${CLASH_ENV_FILE:-}" Env_File="${CLASH_ENV_FILE:-}"
if [ "$Env_File" = "off" ] || [ "$Env_File" = "disabled" ]; then if [ "$Env_File" = "off" ] || [ "$Env_File" = "disabled" ]; then
echo -e "\033[33m[WARN] 已关闭环境变量文件生成\033[0m" echo -e "\033[33m[WARN] 已关闭环境变量文件生成\033[0m"
else else
@ -275,11 +285,14 @@ else
Env_File="$Temp_Dir/clash-for-linux.sh" Env_File="$Temp_Dir/clash-for-linux.sh"
fi fi
fi fi
if [ -f /etc/profile.d/clash.sh ]; then if [ -f /etc/profile.d/clash.sh ]; then
echo -e "\033[33m[WARN] 检测到旧版环境变量文件 /etc/profile.d/clash.sh建议确认是否需要清理\033[0m" echo -e "\033[33m[WARN] 检测到旧版环境变量文件 /etc/profile.d/clash.sh建议确认是否需要清理\033[0m"
fi fi
mkdir -p "$(dirname "$Env_File")" mkdir -p "$(dirname "$Env_File")"
cat>"$Env_File"<<EOF
cat >"$Env_File"<<EOF
# 开启系统代理 # 开启系统代理
function proxy_on() { function proxy_on() {
export http_proxy=http://${CLASH_LISTEN_IP}:${CLASH_HTTP_PORT} export http_proxy=http://${CLASH_LISTEN_IP}:${CLASH_HTTP_PORT}
@ -292,7 +305,7 @@ function proxy_on() {
} }
# 关闭系统代理 # 关闭系统代理
function proxy_off(){ function proxy_off() {
unset http_proxy unset http_proxy
unset https_proxy unset https_proxy
unset no_proxy unset no_proxy
@ -303,7 +316,7 @@ function proxy_off(){
} }
EOF EOF
echo -e "请执行以下命令加载环境变量: source ${Env_File}\n" echo -e "请执行以下命令加载环境变量: source ${Env_File}\n"
echo -e "请执行以下命令开启系统代理: proxy_on\n" echo -e "请执行以下命令开启系统代理: proxy_on\n"
echo -e "若要临时关闭系统代理,请执行: proxy_off\n" echo -e "若要临时关闭系统代理,请执行: proxy_off\n"
fi fi