Merge pull request #99 from wnlen/codex/resolve-platform-adaptation-issues

Add systemd installer, non-root service user, and multi-arch subconverter resolution
This commit is contained in:
wnlen
2026-01-13 23:46:56 +08:00
committed by GitHub
8 changed files with 222 additions and 17 deletions

View File

@ -159,11 +159,10 @@ $ proxy_off
## systemd 服务
将仓库中的 `systemd/clash-for-linux.service` 复制到 `/etc/systemd/system/`,并根据实际路径修改 `WorkingDirectory` 与脚本路径
推荐使用自动安装脚本生成 systemd 单元(自动识别安装路径、创建低权限用户并修正目录权限)
```bash
$ sudo cp systemd/clash-for-linux.service /etc/systemd/system/
$ sudo vim /etc/systemd/system/clash-for-linux.service
$ sudo bash scripts/install_systemd.sh
```
启用并启动服务:
@ -179,6 +178,41 @@ $ sudo systemctl enable --now clash-for-linux.service
$ sudo systemctl stop clash-for-linux.service
```
> 如需自定义运行用户,可在执行脚本前设置 `CLASH_SERVICE_USER`(可选 `CLASH_SERVICE_GROUP`)。
> 默认使用 `clash` 用户运行服务systemd 环境文件输出到 `temp/clash-for-linux.sh`。
如果需要手动安装,可参考 `systemd/clash-for-linux.service` 模板并替换安装路径。
<br>
## subconverter 多架构支持
`subconverter` 用于将订阅内容转换为标准 clash 配置。默认会尝试以下位置:
- `tools/subconverter/subconverter`
- `tools/subconverter/subconverter-<arch>`
- `tools/subconverter/bin/subconverter-<arch>`
其中 `<arch>` 取值为:
- `linux-amd64`
- `linux-arm64`
- `linux-armv7`
你也可以设置:
- `SUBCONVERTER_PATH`:指定自定义 `subconverter` 可执行文件路径。
- `SUBCONVERTER_AUTO_DOWNLOAD=true`:启用自动下载(需 `curl`/`wget`)。
- `SUBCONVERTER_DOWNLOAD_URL_TEMPLATE`:下载模板,使用 `{arch}` 占位符,如:
```bash
export SUBCONVERTER_AUTO_DOWNLOAD=true
export SUBCONVERTER_DOWNLOAD_URL_TEMPLATE='https://example.com/subconverter_{arch}.tar.gz'
```
`subconverter` 不可用时会自动跳过转换,并提示警告。
<br>

View File

@ -21,7 +21,11 @@ else
echo "$decoded_content" > ${Server_Dir}/temp/clash_config.yaml
else
echo "解码后的内容不符合clash标准尝试将其转换为标准格式"
${Server_Dir}/tools/subconverter/subconverter -g &>> ${Server_Dir}/logs/subconverter.log
if [ -z "$SUBCONVERTER_BIN" ]; then
echo "subconverter 未配置,无法执行转换"
exit 1
fi
"${SUBCONVERTER_BIN}" -g &>> ${Server_Dir}/logs/subconverter.log
converted_file=${Server_Dir}/temp/clash_config.yaml
# 判断转换后的内容是否符合clash配置文件标准
if awk '/^proxies:/{p=1} /^proxy-groups:/{g=1} /^rules:/{r=1} p&&g&&r{exit} END{if(p&&g&&r) exit 0; else exit 1}' $converted_file; then

52
scripts/install_systemd.sh Executable file
View File

@ -0,0 +1,52 @@
#!/bin/bash
set -euo pipefail
Server_Dir=$(cd "$(dirname "${BASH_SOURCE[0]}")/.." && pwd)
Service_Name="clash-for-linux"
Service_User="${CLASH_SERVICE_USER:-clash}"
Service_Group="${CLASH_SERVICE_GROUP:-$Service_User}"
Unit_Path="/etc/systemd/system/${Service_Name}.service"
if [ "$(id -u)" -ne 0 ]; then
echo -e "\033[31m[ERROR] 需要 root 权限来安装 systemd 单元\033[0m"
exit 1
fi
if ! getent group "$Service_Group" >/dev/null 2>&1; then
groupadd --system "$Service_Group"
fi
if ! id "$Service_User" >/dev/null 2>&1; then
useradd --system --no-create-home --shell /usr/sbin/nologin --gid "$Service_Group" "$Service_User"
fi
install -d -m 0755 "$Server_Dir/conf" "$Server_Dir/logs" "$Server_Dir/temp"
chown -R "$Service_User:$Service_Group" "$Server_Dir/conf" "$Server_Dir/logs" "$Server_Dir/temp"
cat >"$Unit_Path"<<EOF
[Unit]
Description=Clash for Linux
After=network.target
[Service]
Type=simple
WorkingDirectory=$Server_Dir
ExecStart=/bin/bash $Server_Dir/start.sh
ExecStop=/bin/bash $Server_Dir/shutdown.sh
Restart=on-failure
RestartSec=5
User=$Service_User
Group=$Service_Group
PIDFile=$Server_Dir/temp/clash.pid
Environment=CLASH_ENV_FILE=$Server_Dir/temp/clash-for-linux.sh
[Install]
WantedBy=multi-user.target
EOF
systemctl daemon-reload
echo -e "\033[32m[OK] 已生成 systemd 单元: ${Unit_Path}\033[0m"
echo -e "可执行以下命令启动服务:"
echo -e " sudo systemctl enable --now ${Service_Name}.service"

82
scripts/resolve_subconverter.sh Executable file
View File

@ -0,0 +1,82 @@
#!/bin/bash
Subconverter_Bin=""
Subconverter_Ready=false
Subconverter_Dir="${Server_Dir}/tools/subconverter"
Default_Bin="${Subconverter_Dir}/subconverter"
resolve_subconverter_arch() {
local raw_arch="$1"
case "$raw_arch" in
x86_64|amd64)
echo "linux-amd64"
;;
aarch64|arm64)
echo "linux-arm64"
;;
armv7*|armv7l)
echo "linux-armv7"
;;
*)
echo ""
;;
esac
}
try_subconverter_bin() {
local candidate="$1"
if [ -n "$candidate" ] && [ -x "$candidate" ]; then
Subconverter_Bin="$candidate"
Subconverter_Ready=true
return 0
fi
return 1
}
if [ -n "$SUBCONVERTER_PATH" ]; then
try_subconverter_bin "$SUBCONVERTER_PATH" && return 0
else
try_subconverter_bin "$Default_Bin" && return 0
fi
Detected_Arch="${CpuArch:-$(uname -m 2>/dev/null)}"
Resolved_Arch=$(resolve_subconverter_arch "$Detected_Arch")
if [ -n "$Resolved_Arch" ]; then
try_subconverter_bin "${Subconverter_Dir}/subconverter-${Resolved_Arch}" && return 0
try_subconverter_bin "${Subconverter_Dir}/bin/subconverter-${Resolved_Arch}" && return 0
try_subconverter_bin "${Subconverter_Dir}/${Resolved_Arch}/subconverter" && return 0
fi
if [ "${SUBCONVERTER_AUTO_DOWNLOAD:-false}" = "true" ] && [ -n "$Resolved_Arch" ]; then
Download_Template="${SUBCONVERTER_DOWNLOAD_URL_TEMPLATE:-}"
if [ -z "$Download_Template" ]; then
echo -e "\033[33m[WARN] 未设置 SUBCONVERTER_DOWNLOAD_URL_TEMPLATE跳过 subconverter 自动下载\033[0m"
return 0
fi
Download_Url="${Download_Template//\{arch\}/${Resolved_Arch}}"
Download_Archive="${Server_Dir}/temp/subconverter-${Resolved_Arch}.tar.gz"
Extract_Dir="${Server_Dir}/temp/subconverter-${Resolved_Arch}"
mkdir -p "${Extract_Dir}"
if command -v curl >/dev/null 2>&1; then
curl -L -sS -o "${Download_Archive}" "${Download_Url}"
elif command -v wget >/dev/null 2>&1; then
wget -q -O "${Download_Archive}" "${Download_Url}"
else
echo -e "\033[33m[WARN] 未找到 curl 或 wget无法自动下载 subconverter\033[0m"
return 0
fi
if [ -f "${Download_Archive}" ]; then
tar -xzf "${Download_Archive}" -C "${Extract_Dir}" 2>/dev/null
Downloaded_Bin=$(find "${Extract_Dir}" -maxdepth 3 -type f -name "subconverter" -print -quit)
if [ -n "${Downloaded_Bin}" ]; then
mv "${Downloaded_Bin}" "${Subconverter_Dir}/subconverter-${Resolved_Arch}"
chmod +x "${Subconverter_Dir}/subconverter-${Resolved_Arch}"
try_subconverter_bin "${Subconverter_Dir}/subconverter-${Resolved_Arch}" && return 0
fi
fi
fi

View File

@ -36,6 +36,18 @@ else
fi
# 清除环境变量
> /etc/profile.d/clash-for-linux.sh
Env_File="${CLASH_ENV_FILE:-}"
if [ "$Env_File" != "off" ] && [ "$Env_File" != "disabled" ]; then
if [ -z "$Env_File" ]; then
if [ -w /etc/profile.d ]; then
Env_File="/etc/profile.d/clash-for-linux.sh"
else
Env_File="$Temp_Dir/clash-for-linux.sh"
fi
fi
if [ -f "$Env_File" ]; then
> "$Env_File"
fi
fi
echo -e "\n服务关闭成功请执行以下命令关闭系统代理proxy_off\n"

View File

@ -12,9 +12,11 @@ export Server_Dir=$(cd $(dirname "${BASH_SOURCE[0]}") && pwd)
source $Server_Dir/.env
# 给二进制启动程序、脚本等添加可执行权限
chmod +x $Server_Dir/bin/*
chmod +x $Server_Dir/scripts/*
chmod +x $Server_Dir/tools/subconverter/subconverter
chmod +x $Server_Dir/bin/* 2>/dev/null
chmod +x $Server_Dir/scripts/* 2>/dev/null
if [ -f "$Server_Dir/tools/subconverter/subconverter" ]; then
chmod +x $Server_Dir/tools/subconverter/subconverter 2>/dev/null
fi
@ -175,8 +177,10 @@ if_success $Text3 $Text4 $ReturnStatus
## 判断订阅内容是否符合clash配置文件标准尝试转换需 subconverter 可执行文件支持)
if [ -x "$Server_Dir/tools/subconverter/subconverter" ]; then
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
else
@ -262,11 +266,23 @@ else
fi
echo ''
# 添加环境变量(root权限) - 使用配置的端口
if [ -f /etc/profile.d/clash.sh ]; then
# 添加环境变量 - 使用配置的端口
Env_File="${CLASH_ENV_FILE:-}"
if [ "$Env_File" = "off" ] || [ "$Env_File" = "disabled" ]; then
echo -e "\033[33m[WARN] 已关闭环境变量文件生成\033[0m"
else
if [ -z "$Env_File" ]; then
if [ -w /etc/profile.d ]; then
Env_File="/etc/profile.d/clash-for-linux.sh"
else
Env_File="$Temp_Dir/clash-for-linux.sh"
fi
fi
if [ -f /etc/profile.d/clash.sh ]; then
echo -e "\033[33m[WARN] 检测到旧版环境变量文件 /etc/profile.d/clash.sh建议确认是否需要清理\033[0m"
fi
cat>/etc/profile.d/clash-for-linux.sh<<EOF
fi
mkdir -p "$(dirname "$Env_File")"
cat>"$Env_File"<<EOF
# 开启系统代理
function proxy_on() {
export http_proxy=http://${CLASH_LISTEN_IP}:${CLASH_HTTP_PORT}
@ -290,6 +306,7 @@ function proxy_off(){
}
EOF
echo -e "请执行以下命令加载环境变量: source /etc/profile.d/clash-for-linux.sh\n"
echo -e "请执行以下命令加载环境变量: source ${Env_File}\n"
echo -e "请执行以下命令开启系统代理: proxy_on\n"
echo -e "若要临时关闭系统代理,请执行: proxy_off\n"
fi

View File

@ -9,8 +9,10 @@ ExecStart=/bin/bash /opt/clash-for-linux/start.sh
ExecStop=/bin/bash /opt/clash-for-linux/shutdown.sh
Restart=on-failure
RestartSec=5
User=root
User=clash
Group=clash
PIDFile=/opt/clash-for-linux/temp/clash.pid
Environment=CLASH_ENV_FILE=/opt/clash-for-linux/temp/clash-for-linux.sh
[Install]
WantedBy=multi-user.target

View File

@ -152,8 +152,10 @@ if_success $Text3 $Text4 $ReturnStatus
\cp -a $Temp_Dir/clash.yaml $Temp_Dir/clash_config.yaml
## 判断订阅内容是否符合clash配置文件标准尝试转换需 subconverter 可执行文件支持)
if [ -x "$Server_Dir/tools/subconverter/subconverter" ]; then
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
else