mirror of
				https://github.com/MHSanaei/3x-ui.git
				synced 2025-10-26 18:14:50 +00:00 
			
		
		
		
	Compare commits
	
		
			4 commits
		
	
	
		
			e6ac973bb9
			...
			ea849d6305
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | ea849d6305 | ||
|   | 02bff4db6c | ||
|   | 8ff4e1ff31 | ||
|   | 26c6438ec2 | 
					 8 changed files with 68 additions and 22 deletions
				
			
		|  | @ -142,7 +142,10 @@ | |||
|       }, | ||||
|       npvtunUrl() { | ||||
|         return this.app.subUrl;  | ||||
|       } | ||||
|       }, | ||||
| 	  happUrl() { | ||||
| 		return `happ://add/${encodeURIComponent(this.app.subUrl)}`; | ||||
| 	  } | ||||
|     }, | ||||
|     methods: { | ||||
|       renderLink, | ||||
|  |  | |||
|  | @ -28,7 +28,7 @@ | |||
|     </a-form-item> | ||||
| 
 | ||||
|     <a-form-item label='{{ i18n "pages.inbounds.port" }}'> | ||||
|         <a-input-number v-model.number="inbound.port" :min="1" :max="65531"></a-input-number> | ||||
|         <a-input-number v-model.number="inbound.port" :min="1" :max="65535"></a-input-number> | ||||
|     </a-form-item> | ||||
| 
 | ||||
|     <a-form-item> | ||||
|  | @ -51,8 +51,9 @@ | |||
|                     <span>{{ i18n "pages.inbounds.periodicTrafficResetDesc" }}</span> | ||||
|                     <br v-if="dbInbound.lastTrafficResetTime && dbInbound.lastTrafficResetTime > 0"> | ||||
|                     <span v-if="dbInbound.lastTrafficResetTime && dbInbound.lastTrafficResetTime > 0"> | ||||
|                         <strong>{{ i18n "pages.inbounds.lastReset" }}:</strong>  | ||||
|                         <span v-if="datepicker == 'gregorian'">[[ moment(dbInbound.lastTrafficResetTime).format('YYYY-MM-DD HH:mm:ss') ]]</span> | ||||
|                         <strong>{{ i18n "pages.inbounds.lastReset" }}:</strong> | ||||
|                         <span v-if="datepicker == 'gregorian'">[[ | ||||
|                             moment(dbInbound.lastTrafficResetTime).format('YYYY-MM-DD HH:mm:ss') ]]</span> | ||||
|                         <span v-else>[[ DateUtil.convertToJalalian(moment(dbInbound.lastTrafficResetTime)) ]]</span> | ||||
|                     </span> | ||||
|                 </template> | ||||
|  | @ -145,4 +146,4 @@ | |||
|     </a-collapse-panel> | ||||
| </a-collapse> | ||||
| 
 | ||||
| {{end}} | ||||
| {{end}} | ||||
|  | @ -3,12 +3,14 @@ | |||
|   <a-divider :style="{ margin: '5px 0 0' }"></a-divider> | ||||
|   <a-form-item label="External Proxy"> | ||||
|     <a-switch v-model="externalProxy"></a-switch> | ||||
|     <a-button icon="plus" v-if="externalProxy" type="primary" :style="{ marginLeft: '10px' }" size="small" @click="inbound.stream.externalProxy.push({forceTls: 'same', dest: '', port: 443, remark: ''})"></a-button> | ||||
|     <a-button icon="plus" v-if="externalProxy" type="primary" :style="{ marginLeft: '10px' }" size="small" | ||||
|       @click="inbound.stream.externalProxy.push({forceTls: 'same', dest: '', port: 443, remark: ''})"></a-button> | ||||
|   </a-form-item> | ||||
|   <a-input-group :style="{ margin: '8px 0' }" compact v-for="(row, index) in inbound.stream.externalProxy"> | ||||
|     <template> | ||||
|       <a-tooltip title="Force TLS"> | ||||
|         <a-select v-model="row.forceTls" :style="{ width: '20%', margin: '0px' }" :dropdown-class-name="themeSwitcher.currentTheme"> | ||||
|         <a-select v-model="row.forceTls" :style="{ width: '20%', margin: '0px' }" | ||||
|           :dropdown-class-name="themeSwitcher.currentTheme"> | ||||
|           <a-select-option value="same">{{ i18n "pages.inbounds.same" }}</a-select-option> | ||||
|           <a-select-option value="none">{{ i18n "none" }}</a-select-option> | ||||
|           <a-select-option value="tls">TLS</a-select-option> | ||||
|  | @ -17,7 +19,7 @@ | |||
|     </template> | ||||
|     <a-input :style="{ width: '30%' }" v-model.trim="row.dest" placeholder='{{ i18n "host" }}'></a-input> | ||||
|     <a-tooltip title='{{ i18n "pages.inbounds.port" }}'> | ||||
|       <a-input-number :style="{ width: '15%' }" v-model.number="row.port" min="1" max="65531"></a-input-number> | ||||
|       <a-input-number :style="{ width: '15%' }" v-model.number="row.port" min="1" max="65535"></a-input-number> | ||||
|     </a-tooltip> | ||||
|     <a-input :style="{ width: '30%', top: '0' }" v-model.trim="row.remark" placeholder='{{ i18n "remark" }}'> | ||||
|       <template slot="addonAfter"> | ||||
|  | @ -26,4 +28,4 @@ | |||
|     </a-input> | ||||
|   </a-input-group> | ||||
| </a-form> | ||||
| {{end}} | ||||
| {{end}} | ||||
|  | @ -7,12 +7,13 @@ | |||
|       <a-input v-model.trim="dnsModal.dnsServer.address"></a-input> | ||||
|     </a-form-item> | ||||
|     <a-form-item label='{{ i18n "pages.inbounds.port" }}'> | ||||
|       <a-input-number v-model.number="dnsModal.dnsServer.port" :min="1" :max="65531"></a-input-number> | ||||
|       <a-input-number v-model.number="dnsModal.dnsServer.port" :min="1" :max="65535"></a-input-number> | ||||
|     </a-form-item> | ||||
|     <a-form-item label='{{ i18n "pages.xray.dns.strategy" }}'> | ||||
|       <a-select v-model="dnsModal.dnsServer.queryStrategy" :style="{ width: '100%' }" | ||||
|         :dropdown-class-name="themeSwitcher.currentTheme"> | ||||
|         <a-select-option :value="l" :label="l" v-for="l in ['UseSystem', 'UseIP', 'UseIPv4', 'UseIPv6']"> [[ l ]] </a-select-option> | ||||
|         <a-select-option :value="l" :label="l" v-for="l in ['UseSystem', 'UseIP', 'UseIPv4', 'UseIPv6']"> [[ l ]] | ||||
|         </a-select-option> | ||||
|       </a-select> | ||||
|     </a-form-item> | ||||
|     <a-divider :style="{ margin: '5px 0' }"></a-divider> | ||||
|  | @ -75,7 +76,7 @@ | |||
|     isEdit: false, | ||||
|     confirm: null, | ||||
|     dnsServer: { ...defaultDnsObject }, | ||||
|     ok() {       | ||||
|     ok() { | ||||
|       ObjectUtil.execute(dnsModal.confirm, { ...dnsModal.dnsServer }); | ||||
|     }, | ||||
|     show({ | ||||
|  | @ -106,7 +107,7 @@ | |||
|         } | ||||
|       } else { | ||||
|         this.dnsServer = { ...defaultDnsObject }; | ||||
|        | ||||
| 
 | ||||
|         this.dnsServer.domains = []; | ||||
|         this.dnsServer.expectIPs = []; | ||||
|         this.dnsServer.unexpectedIPs = []; | ||||
|  |  | |||
|  | @ -39,7 +39,7 @@ | |||
|             <template #title>{{ i18n "pages.settings.panelPort"}}</template> | ||||
|             <template #description>{{ i18n "pages.settings.panelPortDesc"}}</template> | ||||
|             <template #control> | ||||
|                 <a-input-number :min="1" :min="65531" v-model="allSetting.webPort" :style="{ width: '100%' }"></a-input> | ||||
|                 <a-input-number :min="1" :min="65535" v-model="allSetting.webPort" :style="{ width: '100%' }"></a-input> | ||||
|             </template> | ||||
|         </a-setting-list-item> | ||||
|         <a-setting-list-item paddings="small"> | ||||
|  | @ -137,7 +137,8 @@ | |||
|             <template #title>{{ i18n "pages.settings.datepicker"}}</template> | ||||
|             <template #description>{{ i18n "pages.settings.datepickerDescription"}}</template> | ||||
|             <template #control> | ||||
|                 <a-select :style="{ width: '100%' }" :dropdown-class-name="themeSwitcher.currentTheme" v-model="datepicker"> | ||||
|                 <a-select :style="{ width: '100%' }" :dropdown-class-name="themeSwitcher.currentTheme" | ||||
|                     v-model="datepicker"> | ||||
|                     <a-select-option v-for="item in datepickerList" :value="item.value"> | ||||
|                         <span v-text="item.name"></span> | ||||
|                     </a-select-option> | ||||
|  |  | |||
|  | @ -40,7 +40,7 @@ | |||
|             <template #title>{{ i18n "pages.settings.subPort"}}</template> | ||||
|             <template #description>{{ i18n "pages.settings.subPortDesc"}}</template> | ||||
|             <template #control> | ||||
|                 <a-input-number v-model="allSetting.subPort" :min="1" :min="65531" | ||||
|                 <a-input-number v-model="allSetting.subPort" :min="1" :min="65535" | ||||
|                     :style="{ width: '100%' }"></a-input-number> | ||||
|             </template> | ||||
|         </a-setting-list-item> | ||||
|  | @ -48,13 +48,10 @@ | |||
|             <template #title>{{ i18n "pages.settings.subPath"}}</template> | ||||
|             <template #description>{{ i18n "pages.settings.subPathDesc"}}</template> | ||||
|             <template #control> | ||||
|                 <a-input | ||||
|                     type="text" | ||||
|                     v-model="allSetting.subPath" | ||||
|                 <a-input type="text" v-model="allSetting.subPath" | ||||
|                     @input="allSetting.subPath = ((typeof $event === 'string' ? $event : ($event && $event.target ? $event.target.value : '')) || '').replace(/[:*]/g, '')" | ||||
|                     @blur="allSetting.subPath = (p => { p = p || '/'; if (!p.startsWith('/')) p='/' + p; if (!p.endsWith('/')) p += '/'; return p.replace(/\/+/g,'/'); })(allSetting.subPath)" | ||||
|                     placeholder="/sub/" | ||||
|                 ></a-input> | ||||
|                     placeholder="/sub/"></a-input> | ||||
|             </template> | ||||
|         </a-setting-list-item> | ||||
|         <a-setting-list-item paddings="small"> | ||||
|  | @ -108,4 +105,4 @@ | |||
|         </a-setting-list-item> | ||||
|     </a-collapse-panel> | ||||
| </a-collapse> | ||||
| {{end}} | ||||
| {{end}} | ||||
|  | @ -218,6 +218,8 @@ | |||
|                                             <a-menu-item key="android-npvtunnel" | ||||
|                                                 @click="copy(app.subUrl)">NPV | ||||
|                                                 Tunnel</a-menu-item> | ||||
| 											<a-menu-item key="android-happ" | ||||
|                                                 @click="open('happ://add/' + encodeURIComponent(app.subUrl))">Happ</a-menu-item>	 | ||||
|                                         </a-menu> | ||||
|                                     </a-dropdown> | ||||
|                                 </a-col> | ||||
|  | @ -244,6 +246,8 @@ | |||
|                                                 @click="copy(npvtunUrl)">NPV | ||||
|                                                 Tunnel | ||||
|                                             </a-menu-item> | ||||
| 											<a-menu-item key="ios-happ" | ||||
|                                                 @click="open(happUrl)">Happ</a-menu-item> | ||||
|                                         </a-menu> | ||||
|                                     </a-dropdown> | ||||
|                                 </a-col> | ||||
|  |  | |||
|  | @ -38,6 +38,25 @@ func (s *InboundService) GetInbounds(userId int) ([]*model.Inbound, error) { | |||
| 	if err != nil && err != gorm.ErrRecordNotFound { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	// Enrich client stats with UUID/SubId from inbound settings
 | ||||
| 	for _, inbound := range inbounds { | ||||
| 		clients, _ := s.GetClients(inbound) | ||||
| 		if len(clients) == 0 || len(inbound.ClientStats) == 0 { | ||||
| 			continue | ||||
| 		} | ||||
| 		// Build a map email -> client
 | ||||
| 		cMap := make(map[string]model.Client, len(clients)) | ||||
| 		for _, c := range clients { | ||||
| 			cMap[strings.ToLower(c.Email)] = c | ||||
| 		} | ||||
| 		for i := range inbound.ClientStats { | ||||
| 			email := strings.ToLower(inbound.ClientStats[i].Email) | ||||
| 			if c, ok := cMap[email]; ok { | ||||
| 				inbound.ClientStats[i].UUID = c.ID | ||||
| 				inbound.ClientStats[i].SubId = c.SubID | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return inbounds, nil | ||||
| } | ||||
| 
 | ||||
|  | @ -50,6 +69,24 @@ func (s *InboundService) GetAllInbounds() ([]*model.Inbound, error) { | |||
| 	if err != nil && err != gorm.ErrRecordNotFound { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	// Enrich client stats with UUID/SubId from inbound settings
 | ||||
| 	for _, inbound := range inbounds { | ||||
| 		clients, _ := s.GetClients(inbound) | ||||
| 		if len(clients) == 0 || len(inbound.ClientStats) == 0 { | ||||
| 			continue | ||||
| 		} | ||||
| 		cMap := make(map[string]model.Client, len(clients)) | ||||
| 		for _, c := range clients { | ||||
| 			cMap[strings.ToLower(c.Email)] = c | ||||
| 		} | ||||
| 		for i := range inbound.ClientStats { | ||||
| 			email := strings.ToLower(inbound.ClientStats[i].Email) | ||||
| 			if c, ok := cMap[email]; ok { | ||||
| 				inbound.ClientStats[i].UUID = c.ID | ||||
| 				inbound.ClientStats[i].SubId = c.SubID | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return inbounds, nil | ||||
| } | ||||
| 
 | ||||
|  |  | |||
		Loading…
	
		Reference in a new issue