mirror of
				https://github.com/MHSanaei/3x-ui.git
				synced 2025-10-26 18:14:50 +00:00 
			
		
		
		
	Compare commits
	
		
			10 commits
		
	
	
		
			f4d64d54ef
			...
			72eb2c9026
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 72eb2c9026 | ||
|   | 1d76a90ae5 | ||
|   | 3bad8a985c | ||
|   | f27172b38a | ||
|   | 172f2ddaa7 | ||
|   | d69af328dc | ||
|   | ee0e3093ba | ||
|   | 8620b55bd6 | ||
|   | a44697e2c6 | ||
|   | 85c328bb93 | 
					 5 changed files with 89 additions and 110 deletions
				
			
		
							
								
								
									
										24
									
								
								install.sh
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								install.sh
									
									
									
									
									
								
							|  | @ -149,11 +149,15 @@ install_x-ui() { | |||
|     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 "${red}Failed to fetch x-ui version, it may be due to GitHub API restrictions, please try it later${plain}" | ||||
|             exit 1 | ||||
|             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 -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 | ||||
|         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 | ||||
|  | @ -170,13 +174,17 @@ install_x-ui() { | |||
| 
 | ||||
|         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 -N -O /usr/local/x-ui-linux-$(arch).tar.gz ${url} | ||||
|         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 -O /usr/bin/x-ui-temp https://raw.githubusercontent.com/MHSanaei/3x-ui/main/x-ui.sh | ||||
|     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 | ||||
|  | @ -209,7 +217,11 @@ install_x-ui() { | |||
|     config_after_install | ||||
| 
 | ||||
|     if [[ $release == "alpine" ]]; then | ||||
|         wget -O /etc/init.d/x-ui https://raw.githubusercontent.com/MHSanaei/3x-ui/main/x-ui.rc | ||||
|         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 | ||||
|  |  | |||
							
								
								
									
										82
									
								
								update.sh
									
									
									
									
									
								
							
							
						
						
									
										82
									
								
								update.sh
									
									
									
									
									
								
							|  | @ -63,24 +63,12 @@ arch() { | |||
| 	armv6* | armv6) echo 'armv6' ;; | ||||
| 	armv5* | armv5) echo 'armv5' ;; | ||||
| 	s390x) echo 's390x' ;; | ||||
| 	*) rm -f ${cur_dir}/update.sh && _fail "Unsupported CPU architecture!";; | ||||
| 	*) rm -f ${cur_dir}/update.sh && _fail "Unsupported CPU architecture!" ;; | ||||
| 	esac | ||||
| } | ||||
| 
 | ||||
| echo "Arch: $(arch)" | ||||
| 
 | ||||
| check_glibc_version() { | ||||
| 	glibc_version=$(ldd --version | head -n1 | awk '{print $NF}') | ||||
| 
 | ||||
| 	required_version="2.32" | ||||
| 	if [[ "$(printf '%s\n' "$required_version" "$glibc_version" | sort -V | head -n1)" != "$required_version" ]]; then | ||||
| 		echo -e "${red}GLIBC version $glibc_version is too old! Required: 2.32 or higher${plain}" | ||||
| 		_fail "Please upgrade to a newer version of your operating system to get a higher GLIBC version." | ||||
| 	fi | ||||
| 	echo "GLIBC version: $glibc_version (meets requirement of 2.32+)" | ||||
| } | ||||
| check_glibc_version | ||||
| 
 | ||||
| install_base() { | ||||
| 	echo -e "${green}Updating and install dependency packages...${plain}" | ||||
| 	case "${release}" in | ||||
|  | @ -125,30 +113,22 @@ update_x-ui() { | |||
| 	fi | ||||
| 
 | ||||
| 	echo -e "${green}Downloading new x-ui version...${plain}" | ||||
| 	if [ $# == 0 ]; then | ||||
| 		tag_version=$(${curl_bin} -Ls "https://api.github.com/repos/MHSanaei/3x-ui/releases/latest" 2>/dev/null | grep '"tag_name":' | sed -E 's/.*"([^"]+)".*/\1/') | ||||
| 
 | ||||
| 	tag_version=$(${curl_bin} -Ls "https://api.github.com/repos/MHSanaei/3x-ui/releases/latest" 2>/dev/null | 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_bin} -4 -Ls "https://api.github.com/repos/MHSanaei/3x-ui/releases/latest" | grep '"tag_name":' | sed -E 's/.*"([^"]+)".*/\1/') | ||||
| 		if [[ ! -n "$tag_version" ]]; then | ||||
| 			_fail "ERROR: Failed to fetch x-ui version, it may be due to GitHub API restrictions, please try it later" | ||||
| 		fi | ||||
| 		echo -e "Got x-ui latest version: ${tag_version}, beginning the installation..." | ||||
| 		${wget_bin} -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 2>/dev/null | ||||
| 	fi | ||||
| 	echo -e "Got x-ui latest version: ${tag_version}, beginning the installation..." | ||||
| 	${wget_bin} -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 2>/dev/null | ||||
| 	if [[ $? -ne 0 ]]; then | ||||
| 		echo -e "${yellow}Trying to fetch version with IPv4...${plain}" | ||||
| 		${wget_bin} --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 2>/dev/null | ||||
| 		if [[ $? -ne 0 ]]; then | ||||
| 			_fail "ERROR: Downloading x-ui failed, please be sure that your server can access GitHub" | ||||
| 		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 | ||||
| 			_fail "ERROR: Please use a newer version (at least v2.3.5). Exiting installation." | ||||
| 		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_bin} -N -O /usr/local/x-ui-linux-$(arch).tar.gz ${url} >/dev/null 2>&1 | ||||
| 		if [[ $? -ne 0 ]]; then | ||||
| 			_fail "ERROR: Download x-ui $1 failed, please check if the version exists." | ||||
| 			_fail "ERROR: Failed to downloading x-ui, please be sure that your server can access GitHub" | ||||
| 		fi | ||||
| 	fi | ||||
| 
 | ||||
|  | @ -207,13 +187,17 @@ update_x-ui() { | |||
| 
 | ||||
| 	echo -e "${green}Downloading and installing x-ui.sh script...${plain}" | ||||
| 	${wget_bin} -O /usr/bin/x-ui https://raw.githubusercontent.com/MHSanaei/3x-ui/main/x-ui.sh >/dev/null 2>&1 | ||||
| 	if [ $? -eq 0 ]; then | ||||
| 		chmod +x /usr/local/x-ui/x-ui.sh >/dev/null 2>&1 | ||||
| 		chmod +x /usr/bin/x-ui >/dev/null 2>&1 | ||||
| 	else | ||||
| 		_fail "ERROR: Failed to download x-ui.sh script." | ||||
| 	if [[ $? -ne 0 ]]; then | ||||
| 		echo -e "${yellow}Trying to fetch x-ui with IPv4...${plain}" | ||||
| 		${wget_bin} --inet4-only -O /usr/bin/x-ui https://raw.githubusercontent.com/MHSanaei/3x-ui/main/x-ui.sh >/dev/null 2>&1 | ||||
| 		if [[ $? -ne 0 ]]; then | ||||
| 			_fail "ERROR: Failed to downloading x-ui.sh script, please be sure that your server can access GitHub" | ||||
| 		fi | ||||
| 	fi | ||||
| 
 | ||||
| 	chmod +x /usr/local/x-ui/x-ui.sh >/dev/null 2>&1 | ||||
| 	chmod +x /usr/bin/x-ui >/dev/null 2>&1 | ||||
| 
 | ||||
| 	echo -e "${green}Changing owner...${plain}" | ||||
| 	chown -R root:root /usr/local/x-ui >/dev/null 2>&1 | ||||
| 
 | ||||
|  | @ -223,16 +207,18 @@ update_x-ui() { | |||
| 	fi | ||||
| 
 | ||||
| 	if [[ $release == "alpine" ]]; then | ||||
| 		echo -e "${green}Downloading and installing startup unit...${plain}" | ||||
| 		echo -e "${green}Downloading and installing startup unit x-ui.rc...${plain}" | ||||
| 		${wget_bin} -O /etc/init.d/x-ui https://raw.githubusercontent.com/MHSanaei/3x-ui/main/x-ui.rc >/dev/null 2>&1 | ||||
| 		if [ $? -eq 0 ]; then | ||||
| 			chmod +x /etc/init.d/x-ui >/dev/null 2>&1 | ||||
| 			chown root:root /etc/init.d/x-ui >/dev/null 2>&1 | ||||
| 			rc-update add x-ui >/dev/null 2>&1 | ||||
| 			rc-service x-ui start >/dev/null 2>&1 | ||||
| 		else | ||||
| 			_fail "ERROR: Failed to download startup unit." | ||||
| 		if [[ $? -ne 0 ]]; then | ||||
| 			${wget_bin} --inet4-only -O /etc/init.d/x-ui https://raw.githubusercontent.com/MHSanaei/3x-ui/main/x-ui.rc >/dev/null 2>&1 | ||||
| 			if [[ $? -ne 0 ]]; then | ||||
| 				_fail "ERROR: Failed to downloading startup unit x-ui.rc, please be sure that your server can access GitHub" | ||||
| 			fi | ||||
| 		fi | ||||
| 		chmod +x /etc/init.d/x-ui >/dev/null 2>&1 | ||||
| 		chown root:root /etc/init.d/x-ui >/dev/null 2>&1 | ||||
| 		rc-update add x-ui >/dev/null 2>&1 | ||||
| 		rc-service x-ui start >/dev/null 2>&1 | ||||
| 	else | ||||
| 		echo -e "${green}Installing systemd unit...${plain}" | ||||
| 		cp -f x-ui.service /etc/systemd/system/ >/dev/null 2>&1 | ||||
|  | @ -248,7 +234,7 @@ update_x-ui() { | |||
| 	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                             │ | ||||
|  | @ -268,4 +254,4 @@ update_x-ui() { | |||
| 
 | ||||
| echo -e "${green}Running...${plain}" | ||||
| install_base | ||||
| update_x-ui $1 | ||||
| update_x-ui $1 | ||||
|  |  | |||
|  | @ -108,17 +108,11 @@ | |||
|     el: '#app', | ||||
|     data: { | ||||
|       themeSwitcher, | ||||
|       loadingStates: { | ||||
|         fetched: false, | ||||
|         spinning: false | ||||
|       }, | ||||
|       user: { | ||||
|         username: "", | ||||
|         password: "", | ||||
|         twoFactorCode: "" | ||||
|       }, | ||||
|       loadingStates: { fetched: false, spinning: false }, | ||||
|       user: { username: "", password: "", twoFactorCode: "" }, | ||||
|       twoFactorEnable: false, | ||||
|       lang: "" | ||||
|       lang: "", | ||||
|       animationStarted: false | ||||
|     }, | ||||
|     async mounted() { | ||||
|       this.lang = LanguageManager.getLanguage(); | ||||
|  | @ -127,65 +121,52 @@ | |||
|     methods: { | ||||
|       async login() { | ||||
|         this.loadingStates.spinning = true; | ||||
| 
 | ||||
|         const msg = await HttpUtil.post('/login', this.user); | ||||
| 
 | ||||
|         if (msg.success) { | ||||
|           location.href = basePath + 'panel/'; | ||||
|         } | ||||
| 
 | ||||
|         this.loadingStates.spinning = false; | ||||
|       }, | ||||
|       async getTwoFactorEnable() { | ||||
|         const msg = await HttpUtil.post('/getTwoFactorEnable'); | ||||
| 
 | ||||
|         if (msg.success) { | ||||
|           this.twoFactorEnable = msg.obj; | ||||
|           this.loadingStates.fetched = true; | ||||
| 
 | ||||
|           this.$nextTick(() => { | ||||
|             if (!this.animationStarted) { | ||||
|               this.animationStarted = true; | ||||
|               this.initHeadline(); | ||||
|             } | ||||
|           }); | ||||
|           return msg.obj; | ||||
|         } | ||||
|       }, | ||||
|       initHeadline() { | ||||
|         const animationDelay = 2000; | ||||
|         const headlines = this.$el.querySelectorAll('.headline'); | ||||
|         headlines.forEach((headline) => { | ||||
|           const first = headline.querySelector('.is-visible'); | ||||
|           if (!first) return; | ||||
|           setTimeout(() => this.hideWord(first, animationDelay), animationDelay); | ||||
|         }); | ||||
|       }, | ||||
|       hideWord(word, delay) { | ||||
|         const nextWord = this.takeNext(word); | ||||
|         this.switchWord(word, nextWord); | ||||
|         setTimeout(() => this.hideWord(nextWord, delay), delay); | ||||
|       }, | ||||
|       takeNext(word) { | ||||
|         return word.nextElementSibling || word.parentElement.firstElementChild; | ||||
|       }, | ||||
|       switchWord(oldWord, newWord) { | ||||
|         oldWord.classList.remove('is-visible'); | ||||
|         oldWord.classList.add('is-hidden'); | ||||
|         newWord.classList.remove('is-hidden'); | ||||
|         newWord.classList.add('is-visible'); | ||||
|       } | ||||
|     }, | ||||
|   }); | ||||
| 
 | ||||
|   document.addEventListener("DOMContentLoaded", function () { | ||||
|     var animationDelay = 2000; | ||||
|     initHeadline(); | ||||
| 
 | ||||
|     function initHeadline() { | ||||
|       animateHeadline(document.querySelectorAll('.headline')); | ||||
|     } | ||||
| 
 | ||||
|     function animateHeadline(headlines) { | ||||
|       var duration = animationDelay; | ||||
|       headlines.forEach(function (headline) { | ||||
|         setTimeout(function () { | ||||
|           hideWord(headline.querySelector('.is-visible')); | ||||
|         }, duration); | ||||
|       }); | ||||
|     } | ||||
| 
 | ||||
|     function hideWord(word) { | ||||
|       var nextWord = takeNext(word); | ||||
|       switchWord(word, nextWord); | ||||
|       setTimeout(function () { | ||||
|         hideWord(nextWord); | ||||
|       }, animationDelay); | ||||
|     } | ||||
| 
 | ||||
|     function takeNext(word) { | ||||
|       return word.nextElementSibling ? word.nextElementSibling : word.parentElement.firstElementChild; | ||||
|     } | ||||
| 
 | ||||
|     function switchWord(oldWord, newWord) { | ||||
|       oldWord.classList.remove('is-visible'); | ||||
|       oldWord.classList.add('is-hidden'); | ||||
|       newWord.classList.remove('is-hidden'); | ||||
|       newWord.classList.add('is-visible'); | ||||
|     } | ||||
|   }); | ||||
| 
 | ||||
|   const pm_input_selector = 'input.ant-input, textarea.ant-input'; | ||||
|   const pm_strip_props = [ | ||||
|     'background', | ||||
|  | @ -261,4 +242,4 @@ | |||
|     pm_init(); | ||||
|   } | ||||
| </script> | ||||
| {{ template "page/body_end" .}} | ||||
| {{ template "page/body_end" .}} | ||||
|  |  | |||
|  | @ -648,7 +648,7 @@ | |||
| "ips" = "🔢 IP-адреса:\r\n{{ .IPs }}\r\n" | ||||
| "serverUpTime" = "⏳ Время работы сервера: {{ .UpTime }} {{ .Unit }}\r\n" | ||||
| "serverLoad" = "📈 Нагрузка сервера: {{ .Load1 }}, {{ .Load2 }}, {{ .Load3 }}\r\n" | ||||
| "serverMemory" = "📋 Диск сервера: {{ .Current }}/{{ .Total }}\r\n" | ||||
| "serverMemory" = "📋 ОЗУ сервера: {{ .Current }}/{{ .Total }}\r\n" | ||||
| "tcpCount" = "🔹 Количество TCP-соединений: {{ .Count }}\r\n" | ||||
| "udpCount" = "🔸 Количество UDP-соединений: {{ .Count }}\r\n" | ||||
| "traffic" = "🚦 Трафик: {{ .Total }} (↑{{ .Upload }},↓{{ .Download }})\r\n" | ||||
|  |  | |||
							
								
								
									
										4
									
								
								x-ui.sh
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								x-ui.sh
									
									
									
									
									
								
							|  | @ -85,7 +85,7 @@ install() { | |||
| } | ||||
| 
 | ||||
| update() { | ||||
|     confirm "This function will forcefully reinstall the latest version, and the data will not be lost. Do you want to continue?" "y" | ||||
|     confirm "This function will update all x-ui components to the latest version, and the data will not be lost. Do you want to continue?" "y" | ||||
|     if [[ $? != 0 ]]; then | ||||
|         LOGE "Cancelled" | ||||
|         if [[ $# == 0 ]]; then | ||||
|  | @ -93,7 +93,7 @@ update() { | |||
|         fi | ||||
|         return 0 | ||||
|     fi | ||||
|     bash <(curl -Ls https://raw.githubusercontent.com/MHSanaei/3x-ui/main/install.sh) | ||||
|     bash <(curl -Ls https://raw.githubusercontent.com/MHSanaei/3x-ui/main/update.sh) | ||||
|     if [[ $? == 0 ]]; then | ||||
|         LOGI "Update is complete, Panel has automatically restarted " | ||||
|         before_show_menu | ||||
|  |  | |||
		Loading…
	
		Reference in a new issue