mirror of
https://github.com/wnlen/clash-for-linux.git
synced 2026-02-04 10:11:28 +08:00
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:
40
README.md
40
README.md
@ -159,11 +159,10 @@ $ proxy_off
|
|||||||
|
|
||||||
## systemd 服务
|
## systemd 服务
|
||||||
|
|
||||||
将仓库中的 `systemd/clash-for-linux.service` 复制到 `/etc/systemd/system/`,并根据实际路径修改 `WorkingDirectory` 与脚本路径:
|
推荐使用自动安装脚本生成 systemd 单元(自动识别安装路径、创建低权限用户并修正目录权限):
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
$ sudo cp systemd/clash-for-linux.service /etc/systemd/system/
|
$ sudo bash scripts/install_systemd.sh
|
||||||
$ sudo vim /etc/systemd/system/clash-for-linux.service
|
|
||||||
```
|
```
|
||||||
|
|
||||||
启用并启动服务:
|
启用并启动服务:
|
||||||
@ -179,6 +178,41 @@ $ sudo systemctl enable --now clash-for-linux.service
|
|||||||
$ sudo systemctl stop 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>
|
<br>
|
||||||
|
|
||||||
|
|||||||
@ -21,7 +21,11 @@ else
|
|||||||
echo "$decoded_content" > ${Server_Dir}/temp/clash_config.yaml
|
echo "$decoded_content" > ${Server_Dir}/temp/clash_config.yaml
|
||||||
else
|
else
|
||||||
echo "解码后的内容不符合clash标准,尝试将其转换为标准格式"
|
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
|
converted_file=${Server_Dir}/temp/clash_config.yaml
|
||||||
# 判断转换后的内容是否符合clash配置文件标准
|
# 判断转换后的内容是否符合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
|
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
52
scripts/install_systemd.sh
Executable 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
82
scripts/resolve_subconverter.sh
Executable 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
|
||||||
14
shutdown.sh
14
shutdown.sh
@ -36,6 +36,18 @@ else
|
|||||||
fi
|
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"
|
echo -e "\n服务关闭成功,请执行以下命令关闭系统代理:proxy_off\n"
|
||||||
|
|||||||
31
start.sh
31
start.sh
@ -12,9 +12,11 @@ export Server_Dir=$(cd $(dirname "${BASH_SOURCE[0]}") && pwd)
|
|||||||
source $Server_Dir/.env
|
source $Server_Dir/.env
|
||||||
|
|
||||||
# 给二进制启动程序、脚本等添加可执行权限
|
# 给二进制启动程序、脚本等添加可执行权限
|
||||||
chmod +x $Server_Dir/bin/*
|
chmod +x $Server_Dir/bin/* 2>/dev/null
|
||||||
chmod +x $Server_Dir/scripts/*
|
chmod +x $Server_Dir/scripts/* 2>/dev/null
|
||||||
chmod +x $Server_Dir/tools/subconverter/subconverter
|
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 可执行文件支持)
|
## 判断订阅内容是否符合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配置文件标准:'
|
echo -e '\n判断订阅内容是否符合clash配置文件标准:'
|
||||||
|
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
|
||||||
@ -262,11 +266,23 @@ else
|
|||||||
fi
|
fi
|
||||||
echo ''
|
echo ''
|
||||||
|
|
||||||
# 添加环境变量(root权限) - 使用配置的端口
|
# 添加环境变量 - 使用配置的端口
|
||||||
|
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
|
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
|
||||||
cat>/etc/profile.d/clash-for-linux.sh<<EOF
|
mkdir -p "$(dirname "$Env_File")"
|
||||||
|
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}
|
||||||
@ -290,6 +306,7 @@ function proxy_off(){
|
|||||||
}
|
}
|
||||||
EOF
|
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_on\n"
|
||||||
echo -e "若要临时关闭系统代理,请执行: proxy_off\n"
|
echo -e "若要临时关闭系统代理,请执行: proxy_off\n"
|
||||||
|
fi
|
||||||
|
|||||||
@ -9,8 +9,10 @@ ExecStart=/bin/bash /opt/clash-for-linux/start.sh
|
|||||||
ExecStop=/bin/bash /opt/clash-for-linux/shutdown.sh
|
ExecStop=/bin/bash /opt/clash-for-linux/shutdown.sh
|
||||||
Restart=on-failure
|
Restart=on-failure
|
||||||
RestartSec=5
|
RestartSec=5
|
||||||
User=root
|
User=clash
|
||||||
|
Group=clash
|
||||||
PIDFile=/opt/clash-for-linux/temp/clash.pid
|
PIDFile=/opt/clash-for-linux/temp/clash.pid
|
||||||
|
Environment=CLASH_ENV_FILE=/opt/clash-for-linux/temp/clash-for-linux.sh
|
||||||
|
|
||||||
[Install]
|
[Install]
|
||||||
WantedBy=multi-user.target
|
WantedBy=multi-user.target
|
||||||
|
|||||||
@ -152,8 +152,10 @@ if_success $Text3 $Text4 $ReturnStatus
|
|||||||
\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 可执行文件支持)
|
||||||
if [ -x "$Server_Dir/tools/subconverter/subconverter" ]; then
|
source $Server_Dir/scripts/resolve_subconverter.sh
|
||||||
|
if [ "$Subconverter_Ready" = "true" ]; then
|
||||||
echo -e '\n判断订阅内容是否符合clash配置文件标准:'
|
echo -e '\n判断订阅内容是否符合clash配置文件标准:'
|
||||||
|
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
|
||||||
|
|||||||
Reference in New Issue
Block a user