mirror of
				https://github.com/MHSanaei/3x-ui.git
				synced 2025-10-26 10:04:41 +00:00 
			
		
		
		
	Compare commits
	
		
			3 commits
		
	
	
		
			104526aab2
			...
			b2b0024648
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | b2b0024648 | ||
|   | 5822758b7c | ||
|   | 49430b3991 | 
					 5 changed files with 106 additions and 58 deletions
				
			
		
							
								
								
									
										6
									
								
								.github/workflows/docker.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/docker.yml
									
									
									
									
										vendored
									
									
								
							|  | @ -1,7 +1,9 @@ | |||
| name: Release 3X-UI for Docker | ||||
| 
 | ||||
| permissions: | ||||
|   contents: read | ||||
|   packages: write | ||||
| 
 | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|   push: | ||||
|  | @ -27,7 +29,7 @@ jobs: | |||
|           tags: | | ||||
|             type=ref,event=branch | ||||
|             type=ref,event=tag | ||||
|           type=pep440,pattern={{version}} | ||||
|             type=semver,pattern={{version}} | ||||
| 
 | ||||
|       - name: Set up QEMU | ||||
|         uses: docker/setup-qemu-action@v3 | ||||
|  | @ -47,7 +49,7 @@ jobs: | |||
|         uses: docker/login-action@v3 | ||||
|         with: | ||||
|           registry: ghcr.io | ||||
|         username: ${{ github.repository_owner }} | ||||
|           username: ${{ github.actor }} | ||||
|           password: ${{ secrets.GITHUB_TOKEN }} | ||||
| 
 | ||||
|       - name: Build and push Docker image | ||||
|  |  | |||
							
								
								
									
										50
									
								
								sub/sub.go
									
									
									
									
									
								
							
							
						
						
									
										50
									
								
								sub/sub.go
									
									
									
									
									
								
							|  | @ -98,8 +98,14 @@ func (s *Server) initRouter() (*gin.Engine, error) { | |||
| 	} | ||||
| 
 | ||||
| 	// Set base_path based on LinksPath for template rendering
 | ||||
| 	// Ensure LinksPath ends with "/" for proper asset URL generation
 | ||||
| 	basePath := LinksPath | ||||
| 	if basePath != "/" && !strings.HasSuffix(basePath, "/") { | ||||
| 		basePath += "/" | ||||
| 	} | ||||
| 	logger.Debug("sub: Setting base_path to:", basePath) | ||||
| 	engine.Use(func(c *gin.Context) { | ||||
| 		c.Set("base_path", LinksPath) | ||||
| 		c.Set("base_path", basePath) | ||||
| 	}) | ||||
| 
 | ||||
| 	Encrypt, err := s.settingService.GetSubEncrypt() | ||||
|  | @ -179,22 +185,48 @@ func (s *Server) initRouter() (*gin.Engine, error) { | |||
| 		linksPathForAssets = strings.TrimRight(LinksPath, "/") + "/assets" | ||||
| 	} | ||||
| 
 | ||||
| 	// Mount assets in multiple paths to handle different URL patterns
 | ||||
| 	var assetsFS http.FileSystem | ||||
| 	if _, err := os.Stat("web/assets"); err == nil { | ||||
| 		engine.StaticFS("/assets", http.FS(os.DirFS("web/assets"))) | ||||
| 		if linksPathForAssets != "/assets" { | ||||
| 			engine.StaticFS(linksPathForAssets, http.FS(os.DirFS("web/assets"))) | ||||
| 		} | ||||
| 		assetsFS = http.FS(os.DirFS("web/assets")) | ||||
| 	} else { | ||||
| 		if subFS, err := fs.Sub(webpkg.EmbeddedAssets(), "assets"); err == nil { | ||||
| 			engine.StaticFS("/assets", http.FS(subFS)) | ||||
| 			if linksPathForAssets != "/assets" { | ||||
| 				engine.StaticFS(linksPathForAssets, http.FS(subFS)) | ||||
| 			} | ||||
| 			assetsFS = http.FS(subFS) | ||||
| 		} else { | ||||
| 			logger.Error("sub: failed to mount embedded assets:", err) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if assetsFS != nil { | ||||
| 		engine.StaticFS("/assets", assetsFS) | ||||
| 		if linksPathForAssets != "/assets" { | ||||
| 			engine.StaticFS(linksPathForAssets, assetsFS) | ||||
| 		} | ||||
| 
 | ||||
| 		// Add middleware to handle dynamic asset paths with subid
 | ||||
| 		if LinksPath != "/" { | ||||
| 			engine.Use(func(c *gin.Context) { | ||||
| 				path := c.Request.URL.Path | ||||
| 				// Check if this is an asset request with subid pattern: /sub/path/{subid}/assets/...
 | ||||
| 				pathPrefix := strings.TrimRight(LinksPath, "/") + "/" | ||||
| 				if strings.HasPrefix(path, pathPrefix) && strings.Contains(path, "/assets/") { | ||||
| 					// Extract the asset path after /assets/
 | ||||
| 					assetsIndex := strings.Index(path, "/assets/") | ||||
| 					if assetsIndex != -1 { | ||||
| 						assetPath := path[assetsIndex+8:] // +8 to skip "/assets/"
 | ||||
| 						if assetPath != "" { | ||||
| 							// Serve the asset file
 | ||||
| 							c.FileFromFS(assetPath, assetsFS) | ||||
| 							c.Abort() | ||||
| 							return | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 				c.Next() | ||||
| 			}) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	g := engine.Group("/") | ||||
| 
 | ||||
| 	s.sub = NewSUBController( | ||||
|  |  | |||
|  | @ -87,7 +87,20 @@ func (a *SUBController) subs(c *gin.Context) { | |||
| 			if !a.jsonEnabled { | ||||
| 				subJsonURL = "" | ||||
| 			} | ||||
| 			page := a.subService.BuildPageData(subId, hostHeader, traffic, lastOnline, subs, subURL, subJsonURL) | ||||
| 			// Get base_path from context (set by middleware)
 | ||||
| 			basePath, exists := c.Get("base_path") | ||||
| 			if !exists { | ||||
| 				basePath = "/" | ||||
| 			} | ||||
| 			// Add subId to base_path for asset URLs
 | ||||
| 			basePathStr := basePath.(string) | ||||
| 			if basePathStr == "/" { | ||||
| 				basePathStr = "/" + subId + "/" | ||||
| 			} else { | ||||
| 				// Remove trailing slash if exists, add subId, then add trailing slash
 | ||||
| 				basePathStr = strings.TrimRight(basePathStr, "/") + "/" + subId + "/" | ||||
| 			} | ||||
| 			page := a.subService.BuildPageData(subId, hostHeader, traffic, lastOnline, subs, subURL, subJsonURL, basePathStr) | ||||
| 			c.HTML(200, "subpage.html", gin.H{ | ||||
| 				"title":        "subscription.title", | ||||
| 				"cur_ver":      config.GetVersion(), | ||||
|  |  | |||
|  | @ -1148,7 +1148,7 @@ func (s *SubService) joinPathWithID(basePath, subId string) string { | |||
| 
 | ||||
| // BuildPageData parses header and prepares the template view model.
 | ||||
| // BuildPageData constructs page data for rendering the subscription information page.
 | ||||
| func (s *SubService) BuildPageData(subId string, hostHeader string, traffic xray.ClientTraffic, lastOnline int64, subs []string, subURL, subJsonURL string) PageData { | ||||
| func (s *SubService) BuildPageData(subId string, hostHeader string, traffic xray.ClientTraffic, lastOnline int64, subs []string, subURL, subJsonURL string, basePath string) PageData { | ||||
| 	download := common.FormatTraffic(traffic.Down) | ||||
| 	upload := common.FormatTraffic(traffic.Up) | ||||
| 	total := "∞" | ||||
|  | @ -1167,7 +1167,7 @@ func (s *SubService) BuildPageData(subId string, hostHeader string, traffic xray | |||
| 
 | ||||
| 	return PageData{ | ||||
| 		Host:         hostHeader, | ||||
| 		BasePath:     "/", // kept as "/"; templates now use context base_path injected from router
 | ||||
| 		BasePath:     basePath, | ||||
| 		SId:          subId, | ||||
| 		Download:     download, | ||||
| 		Upload:       upload, | ||||
|  |  | |||
|  | @ -35,8 +35,8 @@ | |||
|                   <a-space direction="vertical" :size="10"> | ||||
|                     <a-theme-switch-login></a-theme-switch-login> | ||||
|                     <span>{{ i18n "pages.settings.language" }}</span> | ||||
|                     <a-select ref="selectLang" class="w-100" v-model="lang" | ||||
|                       @change="LanguageManager.setLanguage(lang)" :dropdown-class-name="themeSwitcher.currentTheme"> | ||||
|                     <a-select ref="selectLang" class="w-100" v-model="lang" @change="LanguageManager.setLanguage(lang)" | ||||
|                       :dropdown-class-name="themeSwitcher.currentTheme"> | ||||
|                       <a-select-option :value="l.value" label="English" v-for="l in LanguageManager.supportedLanguages"> | ||||
|                         <span role="img" aria-label="l.name" v-text="l.icon"></span> | ||||
|                           <span v-text="l.name"></span> | ||||
|  | @ -68,7 +68,7 @@ | |||
|                       </a-input> | ||||
|                     </a-form-item> | ||||
|                     <a-form-item> | ||||
|                       <a-input-password autocomplete="password" name="password" v-model.trim="user.password" | ||||
|                       <a-input-password autocomplete="current-password" name="password" v-model.trim="user.password" | ||||
|                         placeholder='{{ i18n "password" }}' required> | ||||
|                         <a-icon slot="prefix" type="lock" class="fs-1rem"></a-icon> | ||||
|                       </a-input-password> | ||||
|  | @ -81,7 +81,8 @@ | |||
|                     </a-form-item> | ||||
|                     <a-form-item> | ||||
|                       <a-row justify="center" class="centered"> | ||||
|                         <div class="wave-btn-bg wave-btn-bg-cl h-50px mt-1rem" :style="loadingStates.spinning ? 'width: 52px' : 'display: inline-block'"> | ||||
|                         <div class="wave-btn-bg wave-btn-bg-cl h-50px mt-1rem" | ||||
|                           :style="loadingStates.spinning ? 'width: 52px' : 'display: inline-block'"> | ||||
|                           <a-button class="ant-btn-primary-login" type="primary" :loading="loadingStates.spinning" | ||||
|                             :icon="loadingStates.spinning ? 'poweroff' : undefined" html-type="submit"> | ||||
|                             [[ loadingStates.spinning ? '' : '{{ i18n "login" }}' ]] | ||||
|  |  | |||
		Loading…
	
		Reference in a new issue