| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259 | #!/bin/bashred='\033[0;31m'green='\033[0;32m'blue='\033[0;34m'yellow='\033[0;33m'plain='\033[0m'cur_dir=$(pwd)# check root[[ $EUID -ne 0 ]] && echo -e "${red}Fatal error: ${plain} Please run this script with root privilege \n " && exit 1# Check OS and set release variableif [[ -f /etc/os-release ]]; then    source /etc/os-release    release=$IDelif [[ -f /usr/lib/os-release ]]; then    source /usr/lib/os-release    release=$IDelse    echo "Failed to check the system OS, please contact the author!" >&2    exit 1fiecho "The OS release is: $release"arch() {    case "$(uname -m)" in    x86_64 | x64 | amd64) echo 'amd64' ;;    i*86 | x86) echo '386' ;;    armv8* | armv8 | arm64 | aarch64) echo 'arm64' ;;    armv7* | armv7 | arm) echo 'armv7' ;;    armv6* | armv6) echo 'armv6' ;;    armv5* | armv5) echo 'armv5' ;;    s390x) echo 's390x' ;;    *) echo -e "${green}Unsupported CPU architecture! ${plain}" && rm -f install.sh && exit 1 ;;    esac}echo "Arch: $(arch)"install_base() {    case "${release}" in    ubuntu | debian | armbian)        apt-get update && apt-get install -y -q wget curl tar tzdata        ;;    centos | rhel | almalinux | rocky | ol)        yum -y update && yum install -y -q wget curl tar tzdata        ;;    fedora | amzn | virtuozzo)        dnf -y update && dnf install -y -q wget curl tar tzdata        ;;    arch | manjaro | parch)        pacman -Syu && pacman -Syu --noconfirm wget curl tar tzdata        ;;    opensuse-tumbleweed | opensuse-leap)        zypper refresh && zypper -q install -y wget curl tar timezone        ;;    alpine)        apk update && apk add wget curl tar tzdata        ;;    *)        apt-get update && apt-get install -y -q wget curl tar tzdata        ;;    esac}gen_random_string() {    local length="$1"    local random_string=$(LC_ALL=C tr -dc 'a-zA-Z0-9' </dev/urandom | fold -w "$length" | head -n 1)    echo "$random_string"}config_after_install() {    local existing_hasDefaultCredential=$(/usr/local/x-ui/x-ui setting -show true | grep -Eo 'hasDefaultCredential: .+' | awk '{print $2}')    local existing_webBasePath=$(/usr/local/x-ui/x-ui setting -show true | grep -Eo 'webBasePath: .+' | awk '{print $2}')    local existing_port=$(/usr/local/x-ui/x-ui setting -show true | grep -Eo 'port: .+' | awk '{print $2}')    local URL_lists=(        "https://api4.ipify.org"		"https://ipv4.icanhazip.com"		"https://v4.api.ipinfo.io/ip"		"https://ipv4.myexternalip.com/raw"		"https://4.ident.me"		"https://check-host.net/ip"    )    local server_ip=""    for ip_address in "${URL_lists[@]}"; do        server_ip=$(curl -s --max-time 3 "${ip_address}" 2>/dev/null | tr -d '[:space:]')        if [[ -n "${server_ip}" ]]; then            break        fi    done    if [[ ${#existing_webBasePath} -lt 4 ]]; then        if [[ "$existing_hasDefaultCredential" == "true" ]]; then            local config_webBasePath=$(gen_random_string 18)            local config_username=$(gen_random_string 10)            local config_password=$(gen_random_string 10)            read -rp "Would you like to customize the Panel Port settings? (If not, a random port will be applied) [y/n]: " config_confirm            if [[ "${config_confirm}" == "y" || "${config_confirm}" == "Y" ]]; then                read -rp "Please set up the panel port: " config_port                echo -e "${yellow}Your Panel Port is: ${config_port}${plain}"            else                local config_port=$(shuf -i 1024-62000 -n 1)                echo -e "${yellow}Generated random port: ${config_port}${plain}"            fi            /usr/local/x-ui/x-ui setting -username "${config_username}" -password "${config_password}" -port "${config_port}" -webBasePath "${config_webBasePath}"            echo -e "This is a fresh installation, generating random login info for security concerns:"            echo -e "###############################################"            echo -e "${green}Username: ${config_username}${plain}"            echo -e "${green}Password: ${config_password}${plain}"            echo -e "${green}Port: ${config_port}${plain}"            echo -e "${green}WebBasePath: ${config_webBasePath}${plain}"            echo -e "${green}Access URL: http://${server_ip}:${config_port}/${config_webBasePath}${plain}"            echo -e "###############################################"        else            local config_webBasePath=$(gen_random_string 18)            echo -e "${yellow}WebBasePath is missing or too short. Generating a new one...${plain}"            /usr/local/x-ui/x-ui setting -webBasePath "${config_webBasePath}"            echo -e "${green}New WebBasePath: ${config_webBasePath}${plain}"            echo -e "${green}Access URL: http://${server_ip}:${existing_port}/${config_webBasePath}${plain}"        fi    else        if [[ "$existing_hasDefaultCredential" == "true" ]]; then            local config_username=$(gen_random_string 10)            local config_password=$(gen_random_string 10)            echo -e "${yellow}Default credentials detected. Security update required...${plain}"            /usr/local/x-ui/x-ui setting -username "${config_username}" -password "${config_password}"            echo -e "Generated new random login credentials:"            echo -e "###############################################"            echo -e "${green}Username: ${config_username}${plain}"            echo -e "${green}Password: ${config_password}${plain}"            echo -e "###############################################"        else            echo -e "${green}Username, Password, and WebBasePath are properly set. Exiting...${plain}"        fi    fi    /usr/local/x-ui/x-ui migrate}install_x-ui() {    cd /usr/local/    # Download resources    if [ $# == 0 ]; then        tag_version=$(curl -Ls "https://api.github.com/repos/MHSanaei/3x-ui/releases/latest" | grep '"tag_name":' | sed -E 's/.*"([^"]+)".*/\1/')        if [[ ! -n "$tag_version" ]]; then            echo -e "${yellow}Trying to fetch version with IPv4...${plain}"            tag_version=$(curl -4 -Ls "https://api.github.com/repos/MHSanaei/3x-ui/releases/latest" | grep '"tag_name":' | sed -E 's/.*"([^"]+)".*/\1/')            if [[ ! -n "$tag_version" ]]; then                echo -e "${red}Failed to fetch x-ui version, it may be due to GitHub API restrictions, please try it later${plain}"                exit 1            fi        fi        echo -e "Got x-ui latest version: ${tag_version}, beginning the installation..."        wget --inet4-only -N -O /usr/local/x-ui-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        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"        wget --inet4-only -N -O /usr/local/x-ui-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    wget --inet4-only -O /usr/bin/x-ui-temp https://raw.githubusercontent.com/MHSanaei/3x-ui/main/x-ui.sh    if [[ $? -ne 0 ]]; then        echo -e "${red}Failed to download x-ui.sh${plain}"        exit 1    fi    # Stop x-ui service and remove old resources    if [[ -e /usr/local/x-ui/ ]]; then        if [[ $release == "alpine" ]]; then            rc-service x-ui stop        else            systemctl stop x-ui        fi        rm /usr/local/x-ui/ -rf    fi    # Extract resources and set permissions    tar zxvf x-ui-linux-$(arch).tar.gz    rm x-ui-linux-$(arch).tar.gz -f        cd x-ui    chmod +x x-ui    chmod +x x-ui.sh    # Check the system's architecture and rename the file accordingly    if [[ $(arch) == "armv5" || $(arch) == "armv6" || $(arch) == "armv7" ]]; then        mv bin/xray-linux-$(arch) bin/xray-linux-arm        chmod +x bin/xray-linux-arm    fi    chmod +x x-ui bin/xray-linux-$(arch)    # Update x-ui cli and se set permission    mv -f /usr/bin/x-ui-temp /usr/bin/x-ui    chmod +x /usr/bin/x-ui    config_after_install    if [[ $release == "alpine" ]]; then        wget --inet4-only -O /etc/init.d/x-ui https://raw.githubusercontent.com/MHSanaei/3x-ui/main/x-ui.rc        if [[ $? -ne 0 ]]; then            echo -e "${red}Failed to download x-ui.rc${plain}"            exit 1        fi        chmod +x /etc/init.d/x-ui        rc-update add x-ui        rc-service x-ui start    else        cp -f x-ui.service /etc/systemd/system/        systemctl daemon-reload        systemctl enable x-ui        systemctl start x-ui    fi    echo -e "${green}x-ui ${tag_version}${plain} installation finished, it is running now..."    echo -e ""    echo -e "┌───────────────────────────────────────────────────────┐│  ${blue}x-ui control menu usages (subcommands):${plain}              ││                                                       ││  ${blue}x-ui${plain}              - Admin Management Script          ││  ${blue}x-ui start${plain}        - Start                            ││  ${blue}x-ui stop${plain}         - Stop                             ││  ${blue}x-ui restart${plain}      - Restart                          ││  ${blue}x-ui status${plain}       - Current Status                   ││  ${blue}x-ui settings${plain}     - Current Settings                 ││  ${blue}x-ui enable${plain}       - Enable Autostart on OS Startup   ││  ${blue}x-ui disable${plain}      - Disable Autostart on OS Startup  ││  ${blue}x-ui log${plain}          - Check logs                       ││  ${blue}x-ui banlog${plain}       - Check Fail2ban ban logs          ││  ${blue}x-ui update${plain}       - Update                           ││  ${blue}x-ui legacy${plain}       - legacy version                   ││  ${blue}x-ui install${plain}      - Install                          ││  ${blue}x-ui uninstall${plain}    - Uninstall                        │└───────────────────────────────────────────────────────┘"}echo -e "${green}Running...${plain}"install_baseinstall_x-ui $1
 |