mirror of
https://github.com/MHSanaei/3x-ui.git
synced 2026-06-07 21:54:10 +00:00
revert: undo install.sh/deploy.sh changes; i18n: add copy-clients translations for all languages
This commit is contained in:
parent
38ff07a050
commit
d327f7f9d1
13 changed files with 297 additions and 216 deletions
16
deploy.sh
16
deploy.sh
|
|
@ -1,16 +0,0 @@
|
||||||
#!/bin/bash
|
|
||||||
set -e
|
|
||||||
cd /opt/3x-uiRsNest
|
|
||||||
|
|
||||||
echo "=== Сборка бэкенда ==="
|
|
||||||
go build -o x-ui main.go
|
|
||||||
|
|
||||||
echo "=== Остановка x-ui ==="
|
|
||||||
systemctl stop x-ui
|
|
||||||
|
|
||||||
echo "=== Замена бинарника ==="
|
|
||||||
cp x-ui /usr/local/x-ui/x-ui
|
|
||||||
|
|
||||||
echo "=== Запуск x-ui ==="
|
|
||||||
systemctl start x-ui
|
|
||||||
systemctl status x-ui
|
|
||||||
325
install.sh
325
install.sh
|
|
@ -11,12 +11,6 @@ cur_dir=$(pwd)
|
||||||
xui_folder="${XUI_MAIN_FOLDER:=/usr/local/x-ui}"
|
xui_folder="${XUI_MAIN_FOLDER:=/usr/local/x-ui}"
|
||||||
xui_service="${XUI_SERVICE:=/etc/systemd/system}"
|
xui_service="${XUI_SERVICE:=/etc/systemd/system}"
|
||||||
|
|
||||||
# Source repository for building x-ui from source (our fork)
|
|
||||||
XUI_REPO_URL="${XUI_REPO_URL:-https://github.com/RsNest/3x-uiRsNest.git}"
|
|
||||||
XUI_REPO_BRANCH="${XUI_REPO_BRANCH:-feature/copy-clients}"
|
|
||||||
XRAY_VERSION="${XRAY_VERSION:-v26.4.17}"
|
|
||||||
GO_VERSION="${GO_VERSION:-1.26.2}"
|
|
||||||
|
|
||||||
# check root
|
# check root
|
||||||
[[ $EUID -ne 0 ]] && echo -e "${red}Fatal error: ${plain} Please run this script with root privilege \n " && exit 1
|
[[ $EUID -ne 0 ]] && echo -e "${red}Fatal error: ${plain} Please run this script with root privilege \n " && exit 1
|
||||||
|
|
||||||
|
|
@ -82,29 +76,29 @@ is_port_in_use() {
|
||||||
install_base() {
|
install_base() {
|
||||||
case "${release}" in
|
case "${release}" in
|
||||||
ubuntu | debian | armbian)
|
ubuntu | debian | armbian)
|
||||||
apt-get update && apt-get install -y -q cron curl tar tzdata socat ca-certificates openssl git build-essential unzip
|
apt-get update && apt-get install -y -q cron curl tar tzdata socat ca-certificates openssl
|
||||||
;;
|
;;
|
||||||
fedora | amzn | virtuozzo | rhel | almalinux | rocky | ol)
|
fedora | amzn | virtuozzo | rhel | almalinux | rocky | ol)
|
||||||
dnf -y update && dnf install -y -q cronie curl tar tzdata socat ca-certificates openssl git gcc make unzip
|
dnf -y update && dnf install -y -q cronie curl tar tzdata socat ca-certificates openssl
|
||||||
;;
|
;;
|
||||||
centos)
|
centos)
|
||||||
if [[ "${VERSION_ID}" =~ ^7 ]]; then
|
if [[ "${VERSION_ID}" =~ ^7 ]]; then
|
||||||
yum -y update && yum install -y cronie curl tar tzdata socat ca-certificates openssl git gcc make unzip
|
yum -y update && yum install -y cronie curl tar tzdata socat ca-certificates openssl
|
||||||
else
|
else
|
||||||
dnf -y update && dnf install -y -q cronie curl tar tzdata socat ca-certificates openssl git gcc make unzip
|
dnf -y update && dnf install -y -q cronie curl tar tzdata socat ca-certificates openssl
|
||||||
fi
|
fi
|
||||||
;;
|
;;
|
||||||
arch | manjaro | parch)
|
arch | manjaro | parch)
|
||||||
pacman -Syu && pacman -Syu --noconfirm cronie curl tar tzdata socat ca-certificates openssl git base-devel unzip
|
pacman -Syu && pacman -Syu --noconfirm cronie curl tar tzdata socat ca-certificates openssl
|
||||||
;;
|
;;
|
||||||
opensuse-tumbleweed | opensuse-leap)
|
opensuse-tumbleweed | opensuse-leap)
|
||||||
zypper refresh && zypper -q install -y cron curl tar timezone socat ca-certificates openssl git gcc make unzip
|
zypper refresh && zypper -q install -y cron curl tar timezone socat ca-certificates openssl
|
||||||
;;
|
;;
|
||||||
alpine)
|
alpine)
|
||||||
apk update && apk add dcron curl tar tzdata socat ca-certificates openssl git build-base unzip
|
apk update && apk add dcron curl tar tzdata socat ca-certificates openssl
|
||||||
;;
|
;;
|
||||||
*)
|
*)
|
||||||
apt-get update && apt-get install -y -q cron curl tar tzdata socat ca-certificates openssl git build-essential unzip
|
apt-get update && apt-get install -y -q cron curl tar tzdata socat ca-certificates openssl
|
||||||
;;
|
;;
|
||||||
esac
|
esac
|
||||||
}
|
}
|
||||||
|
|
@ -788,181 +782,79 @@ config_after_install() {
|
||||||
${xui_folder}/x-ui migrate
|
${xui_folder}/x-ui migrate
|
||||||
}
|
}
|
||||||
|
|
||||||
install_go() {
|
install_x-ui() {
|
||||||
# Install Go toolchain from go.dev if missing or too old
|
cd ${xui_folder%/x-ui}/
|
||||||
local required_major=1
|
|
||||||
local required_minor=26
|
|
||||||
|
|
||||||
if command -v go >/dev/null 2>&1; then
|
# Download resources
|
||||||
local existing_version
|
if [ $# == 0 ]; then
|
||||||
existing_version=$(go version 2>/dev/null | awk '{print $3}' | sed 's/^go//')
|
tag_version=$(curl -Ls "https://api.github.com/repos/MHSanaei/3x-ui/releases/latest" | grep '"tag_name":' | sed -E 's/.*"([^"]+)".*/\1/')
|
||||||
local existing_major existing_minor
|
if [[ ! -n "$tag_version" ]]; then
|
||||||
existing_major=$(echo "${existing_version}" | cut -d. -f1)
|
echo -e "${yellow}Trying to fetch version with IPv4...${plain}"
|
||||||
existing_minor=$(echo "${existing_version}" | cut -d. -f2)
|
tag_version=$(curl -4 -Ls "https://api.github.com/repos/MHSanaei/3x-ui/releases/latest" | grep '"tag_name":' | sed -E 's/.*"([^"]+)".*/\1/')
|
||||||
if [[ "${existing_major}" =~ ^[0-9]+$ ]] && [[ "${existing_minor}" =~ ^[0-9]+$ ]]; then
|
if [[ ! -n "$tag_version" ]]; then
|
||||||
if (( existing_major > required_major )) || \
|
echo -e "${red}Failed to fetch x-ui version, it may be due to GitHub API restrictions, please try it later${plain}"
|
||||||
(( existing_major == required_major && existing_minor >= required_minor )); then
|
exit 1
|
||||||
echo -e "${green}Go ${existing_version} is already installed${plain}"
|
|
||||||
return 0
|
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
echo -e "${yellow}Installed Go ${existing_version} is too old (need >= ${required_major}.${required_minor}), installing ${GO_VERSION}...${plain}"
|
echo -e "Got x-ui latest version: ${tag_version}, beginning the installation..."
|
||||||
|
curl -4fLRo ${xui_folder}-linux-$(arch).tar.gz https://github.com/MHSanaei/3x-ui/releases/download/${tag_version}/x-ui-linux-$(arch).tar.gz
|
||||||
|
if [[ $? -ne 0 ]]; then
|
||||||
|
echo -e "${red}Downloading x-ui failed, please be sure that your server can access GitHub ${plain}"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
else
|
else
|
||||||
echo -e "${yellow}Go toolchain not found, installing ${GO_VERSION}...${plain}"
|
tag_version=$1
|
||||||
|
tag_version_numeric=${tag_version#v}
|
||||||
|
min_version="2.3.5"
|
||||||
|
|
||||||
|
if [[ "$(printf '%s\n' "$min_version" "$tag_version_numeric" | sort -V | head -n1)" != "$min_version" ]]; then
|
||||||
|
echo -e "${red}Please use a newer version (at least v2.3.5). Exiting installation.${plain}"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
url="https://github.com/MHSanaei/3x-ui/releases/download/${tag_version}/x-ui-linux-$(arch).tar.gz"
|
||||||
|
echo -e "Beginning to install x-ui $1"
|
||||||
|
curl -4fLRo ${xui_folder}-linux-$(arch).tar.gz ${url}
|
||||||
|
if [[ $? -ne 0 ]]; then
|
||||||
|
echo -e "${red}Download x-ui $1 failed, please check if the version exists ${plain}"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
fi
|
fi
|
||||||
|
curl -4fLRo /usr/bin/x-ui-temp https://raw.githubusercontent.com/MHSanaei/3x-ui/main/x-ui.sh
|
||||||
local go_arch
|
|
||||||
case "$(arch)" in
|
|
||||||
amd64) go_arch="amd64" ;;
|
|
||||||
386) go_arch="386" ;;
|
|
||||||
arm64) go_arch="arm64" ;;
|
|
||||||
armv6 | armv7 | armv5) go_arch="armv6l" ;;
|
|
||||||
s390x) go_arch="s390x" ;;
|
|
||||||
*) echo -e "${red}Unsupported architecture for Go: $(arch)${plain}"; exit 1 ;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
local go_tarball="go${GO_VERSION}.linux-${go_arch}.tar.gz"
|
|
||||||
echo -e "${green}Downloading Go ${GO_VERSION} (${go_arch}) from go.dev...${plain}"
|
|
||||||
curl -fLRo "/tmp/${go_tarball}" "https://go.dev/dl/${go_tarball}"
|
|
||||||
if [[ $? -ne 0 ]]; then
|
if [[ $? -ne 0 ]]; then
|
||||||
echo -e "${red}Failed to download Go${plain}"
|
echo -e "${red}Failed to download x-ui.sh${plain}"
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
rm -rf /usr/local/go
|
# Stop x-ui service and remove old resources
|
||||||
tar -C /usr/local -xzf "/tmp/${go_tarball}"
|
|
||||||
rm -f "/tmp/${go_tarball}"
|
|
||||||
|
|
||||||
export PATH="/usr/local/go/bin:${PATH}"
|
|
||||||
if ! grep -q '/usr/local/go/bin' /etc/profile 2>/dev/null; then
|
|
||||||
echo 'export PATH="/usr/local/go/bin:${PATH}"' >> /etc/profile
|
|
||||||
fi
|
|
||||||
|
|
||||||
if ! command -v go >/dev/null 2>&1; then
|
|
||||||
echo -e "${red}Go installation failed${plain}"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
echo -e "${green}$(go version) installed${plain}"
|
|
||||||
}
|
|
||||||
|
|
||||||
fetch_xray_assets() {
|
|
||||||
# Downloads Xray-core binary and geo data files into current bin/ dir.
|
|
||||||
# Uses naming conventions compatible with install.sh arch() output.
|
|
||||||
local arch_name="$1"
|
|
||||||
local xray_arch=""
|
|
||||||
case "${arch_name}" in
|
|
||||||
amd64) xray_arch="64" ;;
|
|
||||||
386) xray_arch="32" ;;
|
|
||||||
arm64) xray_arch="arm64-v8a" ;;
|
|
||||||
armv7) xray_arch="arm32-v7a" ;;
|
|
||||||
armv6) xray_arch="arm32-v6" ;;
|
|
||||||
armv5) xray_arch="arm32-v6" ;;
|
|
||||||
s390x) xray_arch="s390x" ;;
|
|
||||||
*) echo -e "${red}Unsupported architecture for Xray: ${arch_name}${plain}"; return 1 ;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
mkdir -p bin
|
|
||||||
cd bin || return 1
|
|
||||||
|
|
||||||
echo -e "${green}Downloading Xray-core ${XRAY_VERSION} (${xray_arch})...${plain}"
|
|
||||||
curl -fLRO "https://github.com/XTLS/Xray-core/releases/download/${XRAY_VERSION}/Xray-linux-${xray_arch}.zip"
|
|
||||||
if [[ $? -ne 0 ]]; then
|
|
||||||
echo -e "${red}Failed to download Xray-core${plain}"
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
unzip -o "Xray-linux-${xray_arch}.zip" >/dev/null
|
|
||||||
rm -f "Xray-linux-${xray_arch}.zip" geoip.dat geosite.dat
|
|
||||||
mv -f xray "xray-linux-${arch_name}"
|
|
||||||
chmod +x "xray-linux-${arch_name}"
|
|
||||||
|
|
||||||
echo -e "${green}Downloading geo data files...${plain}"
|
|
||||||
curl -fLRO https://github.com/Loyalsoldier/v2ray-rules-dat/releases/latest/download/geoip.dat
|
|
||||||
curl -fLRO https://github.com/Loyalsoldier/v2ray-rules-dat/releases/latest/download/geosite.dat
|
|
||||||
curl -fLRo geoip_IR.dat https://github.com/chocolate4u/Iran-v2ray-rules/releases/latest/download/geoip.dat
|
|
||||||
curl -fLRo geosite_IR.dat https://github.com/chocolate4u/Iran-v2ray-rules/releases/latest/download/geosite.dat
|
|
||||||
curl -fLRo geoip_RU.dat https://github.com/runetfreedom/russia-v2ray-rules-dat/releases/latest/download/geoip.dat
|
|
||||||
curl -fLRo geosite_RU.dat https://github.com/runetfreedom/russia-v2ray-rules-dat/releases/latest/download/geosite.dat
|
|
||||||
|
|
||||||
cd ..
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
install_x-ui() {
|
|
||||||
# Build x-ui from our fork instead of downloading upstream release.
|
|
||||||
install_go
|
|
||||||
|
|
||||||
local build_dir
|
|
||||||
build_dir=$(mktemp -d /tmp/3x-ui-build.XXXXXX)
|
|
||||||
trap "rm -rf '${build_dir}'" EXIT
|
|
||||||
|
|
||||||
echo -e "${green}Cloning ${XUI_REPO_URL} (branch: ${XUI_REPO_BRANCH})...${plain}"
|
|
||||||
git clone --depth=1 --branch "${XUI_REPO_BRANCH}" "${XUI_REPO_URL}" "${build_dir}/src"
|
|
||||||
if [[ $? -ne 0 ]]; then
|
|
||||||
echo -e "${red}Failed to clone repository${plain}"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
cd "${build_dir}/src" || exit 1
|
|
||||||
|
|
||||||
local tag_version
|
|
||||||
tag_version="$(git describe --tags --always 2>/dev/null)"
|
|
||||||
[[ -z "${tag_version}" ]] && tag_version="${XUI_REPO_BRANCH}"
|
|
||||||
|
|
||||||
echo -e "${green}Building x-ui from source (this may take several minutes)...${plain}"
|
|
||||||
export CGO_ENABLED=1
|
|
||||||
export CGO_CFLAGS="-D_LARGEFILE64_SOURCE"
|
|
||||||
export PATH="/usr/local/go/bin:${PATH}"
|
|
||||||
go build -ldflags "-w -s" -o x-ui main.go
|
|
||||||
if [[ $? -ne 0 ]]; then
|
|
||||||
echo -e "${red}go build failed${plain}"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
echo -e "${green}Build succeeded${plain}"
|
|
||||||
|
|
||||||
local arch_name
|
|
||||||
arch_name=$(arch)
|
|
||||||
|
|
||||||
fetch_xray_assets "${arch_name}"
|
|
||||||
if [[ $? -ne 0 ]]; then
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Stop running x-ui (if any) and clean previous install
|
|
||||||
if [[ -e ${xui_folder}/ ]]; then
|
if [[ -e ${xui_folder}/ ]]; then
|
||||||
if [[ $release == "alpine" ]]; then
|
if [[ $release == "alpine" ]]; then
|
||||||
rc-service x-ui stop 2>/dev/null
|
rc-service x-ui stop
|
||||||
else
|
else
|
||||||
systemctl stop x-ui 2>/dev/null
|
systemctl stop x-ui
|
||||||
fi
|
fi
|
||||||
rm -rf "${xui_folder}/"
|
rm ${xui_folder}/ -rf
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Install built artifacts
|
# Extract resources and set permissions
|
||||||
mkdir -p "${xui_folder}"
|
tar zxvf x-ui-linux-$(arch).tar.gz
|
||||||
cp -f x-ui "${xui_folder}/x-ui"
|
rm x-ui-linux-$(arch).tar.gz -f
|
||||||
cp -f x-ui.sh "${xui_folder}/x-ui.sh"
|
|
||||||
cp -rf bin "${xui_folder}/"
|
|
||||||
[ -f x-ui.service ] && cp -f x-ui.service "${xui_folder}/"
|
|
||||||
[ -f x-ui.service.debian ] && cp -f x-ui.service.debian "${xui_folder}/"
|
|
||||||
[ -f x-ui.service.arch ] && cp -f x-ui.service.arch "${xui_folder}/"
|
|
||||||
[ -f x-ui.service.rhel ] && cp -f x-ui.service.rhel "${xui_folder}/"
|
|
||||||
[ -f x-ui.rc ] && cp -f x-ui.rc "${xui_folder}/"
|
|
||||||
|
|
||||||
chmod +x "${xui_folder}/x-ui" "${xui_folder}/x-ui.sh"
|
cd x-ui
|
||||||
chmod +x "${xui_folder}/bin/xray-linux-${arch_name}"
|
chmod +x x-ui
|
||||||
|
chmod +x x-ui.sh
|
||||||
|
|
||||||
# Keep backward-compat naming for 32-bit ARM variants
|
# Check the system's architecture and rename the file accordingly
|
||||||
if [[ "${arch_name}" == "armv5" || "${arch_name}" == "armv6" || "${arch_name}" == "armv7" ]]; then
|
if [[ $(arch) == "armv5" || $(arch) == "armv6" || $(arch) == "armv7" ]]; then
|
||||||
mv -f "${xui_folder}/bin/xray-linux-${arch_name}" "${xui_folder}/bin/xray-linux-arm"
|
mv bin/xray-linux-$(arch) bin/xray-linux-arm
|
||||||
chmod +x "${xui_folder}/bin/xray-linux-arm"
|
chmod +x bin/xray-linux-arm
|
||||||
fi
|
fi
|
||||||
|
chmod +x x-ui bin/xray-linux-$(arch)
|
||||||
|
|
||||||
# Install x-ui CLI wrapper
|
# Update x-ui cli and se set permission
|
||||||
cp -f x-ui.sh /usr/bin/x-ui
|
mv -f /usr/bin/x-ui-temp /usr/bin/x-ui
|
||||||
chmod +x /usr/bin/x-ui
|
chmod +x /usr/bin/x-ui
|
||||||
mkdir -p /var/log/x-ui
|
mkdir -p /var/log/x-ui
|
||||||
|
|
||||||
cd "${xui_folder}" || exit 1
|
|
||||||
config_after_install
|
config_after_install
|
||||||
|
|
||||||
# Etckeeper compatibility
|
# Etckeeper compatibility
|
||||||
|
|
@ -979,40 +871,92 @@ install_x-ui() {
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Install service unit from locally-built repo files
|
|
||||||
if [[ $release == "alpine" ]]; then
|
if [[ $release == "alpine" ]]; then
|
||||||
if [ ! -f "${xui_folder}/x-ui.rc" ]; then
|
curl -4fLRo /etc/init.d/x-ui https://raw.githubusercontent.com/MHSanaei/3x-ui/main/x-ui.rc
|
||||||
echo -e "${red}x-ui.rc not found in build output${plain}"
|
if [[ $? -ne 0 ]]; then
|
||||||
|
echo -e "${red}Failed to download x-ui.rc${plain}"
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
cp -f "${xui_folder}/x-ui.rc" /etc/init.d/x-ui
|
|
||||||
chmod +x /etc/init.d/x-ui
|
chmod +x /etc/init.d/x-ui
|
||||||
rc-update add x-ui
|
rc-update add x-ui
|
||||||
rc-service x-ui start
|
rc-service x-ui start
|
||||||
else
|
else
|
||||||
local service_src=""
|
# Install systemd service file
|
||||||
if [ -f "${xui_folder}/x-ui.service" ]; then
|
service_installed=false
|
||||||
service_src="${xui_folder}/x-ui.service"
|
|
||||||
else
|
if [ -f "x-ui.service" ]; then
|
||||||
|
echo -e "${green}Found x-ui.service in extracted files, installing...${plain}"
|
||||||
|
cp -f x-ui.service ${xui_service}/ >/dev/null 2>&1
|
||||||
|
if [[ $? -eq 0 ]]; then
|
||||||
|
service_installed=true
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ "$service_installed" = false ]; then
|
||||||
case "${release}" in
|
case "${release}" in
|
||||||
ubuntu | debian | armbian) service_src="${xui_folder}/x-ui.service.debian" ;;
|
ubuntu | debian | armbian)
|
||||||
arch | manjaro | parch) service_src="${xui_folder}/x-ui.service.arch" ;;
|
if [ -f "x-ui.service.debian" ]; then
|
||||||
*) service_src="${xui_folder}/x-ui.service.rhel" ;;
|
echo -e "${green}Found x-ui.service.debian in extracted files, installing...${plain}"
|
||||||
|
cp -f x-ui.service.debian ${xui_service}/x-ui.service >/dev/null 2>&1
|
||||||
|
if [[ $? -eq 0 ]]; then
|
||||||
|
service_installed=true
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
arch | manjaro | parch)
|
||||||
|
if [ -f "x-ui.service.arch" ]; then
|
||||||
|
echo -e "${green}Found x-ui.service.arch in extracted files, installing...${plain}"
|
||||||
|
cp -f x-ui.service.arch ${xui_service}/x-ui.service >/dev/null 2>&1
|
||||||
|
if [[ $? -eq 0 ]]; then
|
||||||
|
service_installed=true
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
if [ -f "x-ui.service.rhel" ]; then
|
||||||
|
echo -e "${green}Found x-ui.service.rhel in extracted files, installing...${plain}"
|
||||||
|
cp -f x-ui.service.rhel ${xui_service}/x-ui.service >/dev/null 2>&1
|
||||||
|
if [[ $? -eq 0 ]]; then
|
||||||
|
service_installed=true
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
;;
|
||||||
esac
|
esac
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [ ! -f "${service_src}" ]; then
|
# If service file not found in tar.gz, download from GitHub
|
||||||
echo -e "${red}Service file not found in build output: ${service_src}${plain}"
|
if [ "$service_installed" = false ]; then
|
||||||
exit 1
|
echo -e "${yellow}Service files not found in tar.gz, downloading from GitHub...${plain}"
|
||||||
|
case "${release}" in
|
||||||
|
ubuntu | debian | armbian)
|
||||||
|
curl -4fLRo ${xui_service}/x-ui.service https://raw.githubusercontent.com/MHSanaei/3x-ui/main/x-ui.service.debian >/dev/null 2>&1
|
||||||
|
;;
|
||||||
|
arch | manjaro | parch)
|
||||||
|
curl -4fLRo ${xui_service}/x-ui.service https://raw.githubusercontent.com/MHSanaei/3x-ui/main/x-ui.service.arch >/dev/null 2>&1
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
curl -4fLRo ${xui_service}/x-ui.service https://raw.githubusercontent.com/MHSanaei/3x-ui/main/x-ui.service.rhel >/dev/null 2>&1
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
if [[ $? -ne 0 ]]; then
|
||||||
|
echo -e "${red}Failed to install x-ui.service from GitHub${plain}"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
service_installed=true
|
||||||
fi
|
fi
|
||||||
|
|
||||||
echo -e "${green}Installing systemd unit from ${service_src}...${plain}"
|
if [ "$service_installed" = true ]; then
|
||||||
cp -f "${service_src}" "${xui_service}/x-ui.service"
|
echo -e "${green}Setting up systemd unit...${plain}"
|
||||||
chown root:root "${xui_service}/x-ui.service" >/dev/null 2>&1
|
chown root:root ${xui_service}/x-ui.service >/dev/null 2>&1
|
||||||
chmod 644 "${xui_service}/x-ui.service" >/dev/null 2>&1
|
chmod 644 ${xui_service}/x-ui.service >/dev/null 2>&1
|
||||||
systemctl daemon-reload
|
systemctl daemon-reload
|
||||||
systemctl enable x-ui
|
systemctl enable x-ui
|
||||||
systemctl start x-ui
|
systemctl start x-ui
|
||||||
|
else
|
||||||
|
echo -e "${red}Failed to install x-ui.service file${plain}"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
echo -e "${green}x-ui ${tag_version}${plain} installation finished, it is running now..."
|
echo -e "${green}x-ui ${tag_version}${plain} installation finished, it is running now..."
|
||||||
|
|
@ -1038,6 +982,5 @@ install_x-ui() {
|
||||||
}
|
}
|
||||||
|
|
||||||
echo -e "${green}Running...${plain}"
|
echo -e "${green}Running...${plain}"
|
||||||
echo -e "${green}Source: ${XUI_REPO_URL} (branch: ${XUI_REPO_BRANCH})${plain}"
|
|
||||||
install_base
|
install_base
|
||||||
install_x-ui
|
install_x-ui $1
|
||||||
|
|
|
||||||
|
|
@ -298,6 +298,20 @@
|
||||||
"submitEdit" = "احفظ التعديلات"
|
"submitEdit" = "احفظ التعديلات"
|
||||||
"clientCount" = "عدد العملاء"
|
"clientCount" = "عدد العملاء"
|
||||||
"bulk" = "إضافة بالجملة"
|
"bulk" = "إضافة بالجملة"
|
||||||
|
"copyFromInbound" = "نسخ العملاء من الـ Inbound"
|
||||||
|
"copyToInbound" = "نسخ العملاء إلى"
|
||||||
|
"copySelected" = "نسخ المحدد"
|
||||||
|
"copySource" = "المصدر"
|
||||||
|
"copyEmailPreview" = "معاينة البريد الإلكتروني الناتج"
|
||||||
|
"copySelectSourceFirst" = "الرجاء اختيار الـ Inbound المصدر أولاً."
|
||||||
|
"copyResult" = "نتيجة النسخ"
|
||||||
|
"copyResultSuccess" = "تم النسخ بنجاح"
|
||||||
|
"copyResultNone" = "لا يوجد شيء للنسخ: لم يتم اختيار أي عميل أو أن المصدر فارغ"
|
||||||
|
"copyResultErrors" = "أخطاء النسخ"
|
||||||
|
"copyFlowLabel" = "Flow للعملاء الجدد (VLESS)"
|
||||||
|
"copyFlowHint" = "يُطبَّق على جميع العملاء المنسوخين. اتركه فارغاً لتخطيه."
|
||||||
|
"selectAll" = "تحديد الكل"
|
||||||
|
"clearAll" = "مسح الكل"
|
||||||
"method" = "طريقة"
|
"method" = "طريقة"
|
||||||
"first" = "أول واحد"
|
"first" = "أول واحد"
|
||||||
"last" = "آخر واحد"
|
"last" = "آخر واحد"
|
||||||
|
|
|
||||||
|
|
@ -298,6 +298,20 @@
|
||||||
"submitEdit" = "Guardar Cambios"
|
"submitEdit" = "Guardar Cambios"
|
||||||
"clientCount" = "Número de Clientes"
|
"clientCount" = "Número de Clientes"
|
||||||
"bulk" = "Agregar en Lote"
|
"bulk" = "Agregar en Lote"
|
||||||
|
"copyFromInbound" = "Copiar clientes desde entrada"
|
||||||
|
"copyToInbound" = "Copiar clientes a"
|
||||||
|
"copySelected" = "Copiar seleccionados"
|
||||||
|
"copySource" = "Origen"
|
||||||
|
"copyEmailPreview" = "Vista previa del email resultante"
|
||||||
|
"copySelectSourceFirst" = "Seleccione primero una entrada de origen."
|
||||||
|
"copyResult" = "Resultado de la copia"
|
||||||
|
"copyResultSuccess" = "Copiado correctamente"
|
||||||
|
"copyResultNone" = "Nada que copiar: ningún cliente seleccionado o el origen está vacío"
|
||||||
|
"copyResultErrors" = "Errores al copiar"
|
||||||
|
"copyFlowLabel" = "Flow para nuevos clientes (VLESS)"
|
||||||
|
"copyFlowHint" = "Se aplica a todos los clientes copiados. Déjelo vacío para omitir."
|
||||||
|
"selectAll" = "Seleccionar todo"
|
||||||
|
"clearAll" = "Limpiar todo"
|
||||||
"method" = "Método"
|
"method" = "Método"
|
||||||
"first" = "Primero"
|
"first" = "Primero"
|
||||||
"last" = "Último"
|
"last" = "Último"
|
||||||
|
|
|
||||||
|
|
@ -298,6 +298,20 @@
|
||||||
"submitEdit" = "ذخیره تغییرات"
|
"submitEdit" = "ذخیره تغییرات"
|
||||||
"clientCount" = "تعداد کاربران"
|
"clientCount" = "تعداد کاربران"
|
||||||
"bulk" = "انبوهسازی"
|
"bulk" = "انبوهسازی"
|
||||||
|
"copyFromInbound" = "کپی کاربران از اینباند"
|
||||||
|
"copyToInbound" = "کپی کاربران به"
|
||||||
|
"copySelected" = "کپی انتخابشدهها"
|
||||||
|
"copySource" = "منبع"
|
||||||
|
"copyEmailPreview" = "پیشنمایش ایمیل نهایی"
|
||||||
|
"copySelectSourceFirst" = "ابتدا یک اینباند منبع انتخاب کنید."
|
||||||
|
"copyResult" = "نتیجه کپی"
|
||||||
|
"copyResultSuccess" = "با موفقیت کپی شد"
|
||||||
|
"copyResultNone" = "چیزی برای کپی نیست: هیچ کاربری انتخاب نشده یا منبع خالی است"
|
||||||
|
"copyResultErrors" = "خطاهای کپی"
|
||||||
|
"copyFlowLabel" = "Flow برای کاربران جدید (VLESS)"
|
||||||
|
"copyFlowHint" = "برای همه کاربران کپیشده اعمال میشود. برای نادیده گرفتن، خالی بگذارید."
|
||||||
|
"selectAll" = "انتخاب همه"
|
||||||
|
"clearAll" = "پاک کردن همه"
|
||||||
"method" = "روش"
|
"method" = "روش"
|
||||||
"first" = "از"
|
"first" = "از"
|
||||||
"last" = "تا"
|
"last" = "تا"
|
||||||
|
|
|
||||||
|
|
@ -298,6 +298,20 @@
|
||||||
"submitEdit" = "Simpan Perubahan"
|
"submitEdit" = "Simpan Perubahan"
|
||||||
"clientCount" = "Jumlah Klien"
|
"clientCount" = "Jumlah Klien"
|
||||||
"bulk" = "Tambahkan Massal"
|
"bulk" = "Tambahkan Massal"
|
||||||
|
"copyFromInbound" = "Salin klien dari inbound"
|
||||||
|
"copyToInbound" = "Salin klien ke"
|
||||||
|
"copySelected" = "Salin yang dipilih"
|
||||||
|
"copySource" = "Sumber"
|
||||||
|
"copyEmailPreview" = "Pratinjau email hasil"
|
||||||
|
"copySelectSourceFirst" = "Silakan pilih inbound sumber terlebih dahulu."
|
||||||
|
"copyResult" = "Hasil penyalinan"
|
||||||
|
"copyResultSuccess" = "Berhasil disalin"
|
||||||
|
"copyResultNone" = "Tidak ada yang disalin: tidak ada klien yang dipilih atau sumber kosong"
|
||||||
|
"copyResultErrors" = "Kesalahan penyalinan"
|
||||||
|
"copyFlowLabel" = "Flow untuk klien baru (VLESS)"
|
||||||
|
"copyFlowHint" = "Diterapkan ke semua klien yang disalin. Biarkan kosong untuk melewati."
|
||||||
|
"selectAll" = "Pilih semua"
|
||||||
|
"clearAll" = "Hapus semua"
|
||||||
"method" = "Metode"
|
"method" = "Metode"
|
||||||
"first" = "Pertama"
|
"first" = "Pertama"
|
||||||
"last" = "Terakhir"
|
"last" = "Terakhir"
|
||||||
|
|
|
||||||
|
|
@ -298,6 +298,20 @@
|
||||||
"submitEdit" = "変更を保存"
|
"submitEdit" = "変更を保存"
|
||||||
"clientCount" = "クライアント数"
|
"clientCount" = "クライアント数"
|
||||||
"bulk" = "一括作成"
|
"bulk" = "一括作成"
|
||||||
|
"copyFromInbound" = "インバウンドからクライアントをコピー"
|
||||||
|
"copyToInbound" = "クライアントのコピー先"
|
||||||
|
"copySelected" = "選択項目をコピー"
|
||||||
|
"copySource" = "ソース"
|
||||||
|
"copyEmailPreview" = "結果メールのプレビュー"
|
||||||
|
"copySelectSourceFirst" = "先にソースインバウンドを選択してください。"
|
||||||
|
"copyResult" = "コピー結果"
|
||||||
|
"copyResultSuccess" = "正常にコピーされました"
|
||||||
|
"copyResultNone" = "コピーする項目がありません: クライアントが選択されていないかソースが空です"
|
||||||
|
"copyResultErrors" = "コピーエラー"
|
||||||
|
"copyFlowLabel" = "新規クライアントの Flow (VLESS)"
|
||||||
|
"copyFlowHint" = "すべてのコピー対象クライアントに適用されます。空のままにするとスキップします。"
|
||||||
|
"selectAll" = "すべて選択"
|
||||||
|
"clearAll" = "すべて解除"
|
||||||
"method" = "方法"
|
"method" = "方法"
|
||||||
"first" = "最初"
|
"first" = "最初"
|
||||||
"last" = "最後"
|
"last" = "最後"
|
||||||
|
|
|
||||||
|
|
@ -298,6 +298,20 @@
|
||||||
"submitEdit" = "Salvar Alterações"
|
"submitEdit" = "Salvar Alterações"
|
||||||
"clientCount" = "Número de Clientes"
|
"clientCount" = "Número de Clientes"
|
||||||
"bulk" = "Adicionar Vários"
|
"bulk" = "Adicionar Vários"
|
||||||
|
"copyFromInbound" = "Copiar clientes da entrada"
|
||||||
|
"copyToInbound" = "Copiar clientes para"
|
||||||
|
"copySelected" = "Copiar selecionados"
|
||||||
|
"copySource" = "Origem"
|
||||||
|
"copyEmailPreview" = "Prévia do email resultante"
|
||||||
|
"copySelectSourceFirst" = "Selecione primeiro uma entrada de origem."
|
||||||
|
"copyResult" = "Resultado da cópia"
|
||||||
|
"copyResultSuccess" = "Copiado com sucesso"
|
||||||
|
"copyResultNone" = "Nada a copiar: nenhum cliente selecionado ou origem vazia"
|
||||||
|
"copyResultErrors" = "Erros ao copiar"
|
||||||
|
"copyFlowLabel" = "Flow para novos clientes (VLESS)"
|
||||||
|
"copyFlowHint" = "Aplicado a todos os clientes copiados. Deixe em branco para ignorar."
|
||||||
|
"selectAll" = "Selecionar tudo"
|
||||||
|
"clearAll" = "Limpar tudo"
|
||||||
"method" = "Método"
|
"method" = "Método"
|
||||||
"first" = "Primeiro"
|
"first" = "Primeiro"
|
||||||
"last" = "Último"
|
"last" = "Último"
|
||||||
|
|
|
||||||
|
|
@ -298,6 +298,20 @@
|
||||||
"submitEdit" = "Değişiklikleri Kaydet"
|
"submitEdit" = "Değişiklikleri Kaydet"
|
||||||
"clientCount" = "Müşteri Sayısı"
|
"clientCount" = "Müşteri Sayısı"
|
||||||
"bulk" = "Toplu Ekle"
|
"bulk" = "Toplu Ekle"
|
||||||
|
"copyFromInbound" = "Gelen bağlantıdan istemcileri kopyala"
|
||||||
|
"copyToInbound" = "İstemcileri şuraya kopyala"
|
||||||
|
"copySelected" = "Seçilenleri kopyala"
|
||||||
|
"copySource" = "Kaynak"
|
||||||
|
"copyEmailPreview" = "Sonuç e-posta önizlemesi"
|
||||||
|
"copySelectSourceFirst" = "Önce bir kaynak gelen bağlantı seçin."
|
||||||
|
"copyResult" = "Kopyalama sonucu"
|
||||||
|
"copyResultSuccess" = "Başarıyla kopyalandı"
|
||||||
|
"copyResultNone" = "Kopyalanacak bir şey yok: istemci seçilmedi veya kaynak boş"
|
||||||
|
"copyResultErrors" = "Kopyalama hataları"
|
||||||
|
"copyFlowLabel" = "Yeni istemciler için Flow (VLESS)"
|
||||||
|
"copyFlowHint" = "Kopyalanan tüm istemcilere uygulanır. Boş bırakırsanız atlanır."
|
||||||
|
"selectAll" = "Tümünü seç"
|
||||||
|
"clearAll" = "Tümünü temizle"
|
||||||
"method" = "Yöntem"
|
"method" = "Yöntem"
|
||||||
"first" = "İlk"
|
"first" = "İlk"
|
||||||
"last" = "Son"
|
"last" = "Son"
|
||||||
|
|
|
||||||
|
|
@ -298,6 +298,20 @@
|
||||||
"submitEdit" = "Зберегти зміни"
|
"submitEdit" = "Зберегти зміни"
|
||||||
"clientCount" = "Кількість клієнтів"
|
"clientCount" = "Кількість клієнтів"
|
||||||
"bulk" = "Додати групу"
|
"bulk" = "Додати групу"
|
||||||
|
"copyFromInbound" = "Скопіювати клієнтів з інбаунда"
|
||||||
|
"copyToInbound" = "Скопіювати клієнтів у"
|
||||||
|
"copySelected" = "Скопіювати вибраних"
|
||||||
|
"copySource" = "Джерело"
|
||||||
|
"copyEmailPreview" = "Попередній перегляд підсумкових email"
|
||||||
|
"copySelectSourceFirst" = "Спочатку виберіть джерело."
|
||||||
|
"copyResult" = "Результат копіювання"
|
||||||
|
"copyResultSuccess" = "Успішно скопійовано"
|
||||||
|
"copyResultNone" = "Нічого копіювати: жодного клієнта не вибрано або список джерела порожній"
|
||||||
|
"copyResultErrors" = "Помилки під час копіювання"
|
||||||
|
"copyFlowLabel" = "Flow для нових клієнтів (VLESS)"
|
||||||
|
"copyFlowHint" = "Застосується до всіх скопійованих клієнтів. Залиште порожнім, щоб не задавати."
|
||||||
|
"selectAll" = "Вибрати всіх"
|
||||||
|
"clearAll" = "Зняти все"
|
||||||
"method" = "Метод"
|
"method" = "Метод"
|
||||||
"first" = "Перший"
|
"first" = "Перший"
|
||||||
"last" = "Останній"
|
"last" = "Останній"
|
||||||
|
|
|
||||||
|
|
@ -298,6 +298,20 @@
|
||||||
"submitEdit" = "Lưu thay đổi"
|
"submitEdit" = "Lưu thay đổi"
|
||||||
"clientCount" = "Số lượng người dùng"
|
"clientCount" = "Số lượng người dùng"
|
||||||
"bulk" = "Thêm hàng loạt"
|
"bulk" = "Thêm hàng loạt"
|
||||||
|
"copyFromInbound" = "Sao chép người dùng từ Inbound"
|
||||||
|
"copyToInbound" = "Sao chép người dùng đến"
|
||||||
|
"copySelected" = "Sao chép đã chọn"
|
||||||
|
"copySource" = "Nguồn"
|
||||||
|
"copyEmailPreview" = "Xem trước email kết quả"
|
||||||
|
"copySelectSourceFirst" = "Vui lòng chọn Inbound nguồn trước."
|
||||||
|
"copyResult" = "Kết quả sao chép"
|
||||||
|
"copyResultSuccess" = "Đã sao chép thành công"
|
||||||
|
"copyResultNone" = "Không có gì để sao chép: chưa chọn người dùng hoặc nguồn trống"
|
||||||
|
"copyResultErrors" = "Lỗi sao chép"
|
||||||
|
"copyFlowLabel" = "Flow cho người dùng mới (VLESS)"
|
||||||
|
"copyFlowHint" = "Áp dụng cho tất cả người dùng được sao chép. Để trống để bỏ qua."
|
||||||
|
"selectAll" = "Chọn tất cả"
|
||||||
|
"clearAll" = "Bỏ chọn tất cả"
|
||||||
"method" = "Phương pháp"
|
"method" = "Phương pháp"
|
||||||
"first" = "Đầu tiên"
|
"first" = "Đầu tiên"
|
||||||
"last" = "Cuối cùng"
|
"last" = "Cuối cùng"
|
||||||
|
|
|
||||||
|
|
@ -298,6 +298,20 @@
|
||||||
"submitEdit" = "保存修改"
|
"submitEdit" = "保存修改"
|
||||||
"clientCount" = "客户端数量"
|
"clientCount" = "客户端数量"
|
||||||
"bulk" = "批量创建"
|
"bulk" = "批量创建"
|
||||||
|
"copyFromInbound" = "从入站复制客户端"
|
||||||
|
"copyToInbound" = "复制客户端到"
|
||||||
|
"copySelected" = "复制所选"
|
||||||
|
"copySource" = "来源"
|
||||||
|
"copyEmailPreview" = "最终邮箱预览"
|
||||||
|
"copySelectSourceFirst" = "请先选择来源入站。"
|
||||||
|
"copyResult" = "复制结果"
|
||||||
|
"copyResultSuccess" = "复制成功"
|
||||||
|
"copyResultNone" = "没有可复制的内容:未选择客户端或来源为空"
|
||||||
|
"copyResultErrors" = "复制错误"
|
||||||
|
"copyFlowLabel" = "新客户端的 Flow (VLESS)"
|
||||||
|
"copyFlowHint" = "应用于所有复制的客户端。留空则跳过。"
|
||||||
|
"selectAll" = "全选"
|
||||||
|
"clearAll" = "全不选"
|
||||||
"method" = "方法"
|
"method" = "方法"
|
||||||
"first" = "置顶"
|
"first" = "置顶"
|
||||||
"last" = "置底"
|
"last" = "置底"
|
||||||
|
|
|
||||||
|
|
@ -298,6 +298,20 @@
|
||||||
"submitEdit" = "儲存修改"
|
"submitEdit" = "儲存修改"
|
||||||
"clientCount" = "客戶端數量"
|
"clientCount" = "客戶端數量"
|
||||||
"bulk" = "批量建立"
|
"bulk" = "批量建立"
|
||||||
|
"copyFromInbound" = "從入站複製用戶端"
|
||||||
|
"copyToInbound" = "複製用戶端到"
|
||||||
|
"copySelected" = "複製所選"
|
||||||
|
"copySource" = "來源"
|
||||||
|
"copyEmailPreview" = "最終郵箱預覽"
|
||||||
|
"copySelectSourceFirst" = "請先選擇來源入站。"
|
||||||
|
"copyResult" = "複製結果"
|
||||||
|
"copyResultSuccess" = "複製成功"
|
||||||
|
"copyResultNone" = "沒有可複製的內容:未選擇用戶端或來源為空"
|
||||||
|
"copyResultErrors" = "複製錯誤"
|
||||||
|
"copyFlowLabel" = "新用戶端的 Flow (VLESS)"
|
||||||
|
"copyFlowHint" = "套用於所有複製的用戶端。留空則略過。"
|
||||||
|
"selectAll" = "全選"
|
||||||
|
"clearAll" = "全不選"
|
||||||
"method" = "方法"
|
"method" = "方法"
|
||||||
"first" = "置頂"
|
"first" = "置頂"
|
||||||
"last" = "置底"
|
"last" = "置底"
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue