mirror of
				https://github.com/MHSanaei/3x-ui.git
				synced 2025-10-24 17:14:40 +00:00 
			
		
		
		
	 64a5a9f1bc
			
		
	
	
		64a5a9f1bc
		
			
		
	
	
	
	
		
			
			* [refactor] api controller * [fix] access log path better to not hardcode the access log path, maybe some ppl dont want to use the default ./access.log * [fix] set select options from logs paths in xray settings * [update] .gitignore * [lint] all .go files * [update] use status code for jsonMsg and 401 to unauthorize * [update] handle response status code via axios * [fix] set correct value if log paths is set to 'none' we also use the default value for the paths if its set to none * [fix] iplimit - only warning access log if f2b is installed
		
			
				
	
	
		
			218 lines
		
	
	
	
		
			4.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			218 lines
		
	
	
	
		
			4.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package service
 | |
| 
 | |
| import (
 | |
| 	"encoding/json"
 | |
| 	"errors"
 | |
| 	"sync"
 | |
| 
 | |
| 	"x-ui/logger"
 | |
| 	"x-ui/xray"
 | |
| 
 | |
| 	"go.uber.org/atomic"
 | |
| )
 | |
| 
 | |
| var (
 | |
| 	p                 *xray.Process
 | |
| 	lock              sync.Mutex
 | |
| 	isNeedXrayRestart atomic.Bool
 | |
| 	result            string
 | |
| )
 | |
| 
 | |
| type XrayService struct {
 | |
| 	inboundService InboundService
 | |
| 	settingService SettingService
 | |
| 	xrayAPI        xray.XrayAPI
 | |
| }
 | |
| 
 | |
| func (s *XrayService) IsXrayRunning() bool {
 | |
| 	return p != nil && p.IsRunning()
 | |
| }
 | |
| 
 | |
| func (s *XrayService) GetXrayErr() error {
 | |
| 	if p == nil {
 | |
| 		return nil
 | |
| 	}
 | |
| 	return p.GetErr()
 | |
| }
 | |
| 
 | |
| func (s *XrayService) GetXrayResult() string {
 | |
| 	if result != "" {
 | |
| 		return result
 | |
| 	}
 | |
| 	if s.IsXrayRunning() {
 | |
| 		return ""
 | |
| 	}
 | |
| 	if p == nil {
 | |
| 		return ""
 | |
| 	}
 | |
| 	result = p.GetResult()
 | |
| 	return result
 | |
| }
 | |
| 
 | |
| func (s *XrayService) GetXrayVersion() string {
 | |
| 	if p == nil {
 | |
| 		return "Unknown"
 | |
| 	}
 | |
| 	return p.GetVersion()
 | |
| }
 | |
| 
 | |
| func RemoveIndex(s []interface{}, index int) []interface{} {
 | |
| 	return append(s[:index], s[index+1:]...)
 | |
| }
 | |
| 
 | |
| func (s *XrayService) GetXrayConfig() (*xray.Config, error) {
 | |
| 	templateConfig, err := s.settingService.GetXrayConfigTemplate()
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	xrayConfig := &xray.Config{}
 | |
| 	err = json.Unmarshal([]byte(templateConfig), xrayConfig)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	s.inboundService.AddTraffic(nil, nil)
 | |
| 
 | |
| 	inbounds, err := s.inboundService.GetAllInbounds()
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	for _, inbound := range inbounds {
 | |
| 		if !inbound.Enable {
 | |
| 			continue
 | |
| 		}
 | |
| 		// get settings clients
 | |
| 		settings := map[string]interface{}{}
 | |
| 		json.Unmarshal([]byte(inbound.Settings), &settings)
 | |
| 		clients, ok := settings["clients"].([]interface{})
 | |
| 		if ok {
 | |
| 			// check users active or not
 | |
| 			clientStats := inbound.ClientStats
 | |
| 			for _, clientTraffic := range clientStats {
 | |
| 				indexDecrease := 0
 | |
| 				for index, client := range clients {
 | |
| 					c := client.(map[string]interface{})
 | |
| 					if c["email"] == clientTraffic.Email {
 | |
| 						if !clientTraffic.Enable {
 | |
| 							clients = RemoveIndex(clients, index-indexDecrease)
 | |
| 							indexDecrease++
 | |
| 							logger.Info("Remove Inbound User ", c["email"], " due the expire or traffic limit")
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			// clear client config for additional parameters
 | |
| 			var final_clients []interface{}
 | |
| 			for _, client := range clients {
 | |
| 				c := client.(map[string]interface{})
 | |
| 				if c["enable"] != nil {
 | |
| 					if enable, ok := c["enable"].(bool); ok && !enable {
 | |
| 						continue
 | |
| 					}
 | |
| 				}
 | |
| 				for key := range c {
 | |
| 					if key != "email" && key != "id" && key != "password" && key != "flow" && key != "method" {
 | |
| 						delete(c, key)
 | |
| 					}
 | |
| 					if c["flow"] == "xtls-rprx-vision-udp443" {
 | |
| 						c["flow"] = "xtls-rprx-vision"
 | |
| 					}
 | |
| 				}
 | |
| 				final_clients = append(final_clients, interface{}(c))
 | |
| 			}
 | |
| 
 | |
| 			settings["clients"] = final_clients
 | |
| 			modifiedSettings, err := json.MarshalIndent(settings, "", "  ")
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 
 | |
| 			inbound.Settings = string(modifiedSettings)
 | |
| 		}
 | |
| 
 | |
| 		if len(inbound.StreamSettings) > 0 {
 | |
| 			// Unmarshal stream JSON
 | |
| 			var stream map[string]interface{}
 | |
| 			json.Unmarshal([]byte(inbound.StreamSettings), &stream)
 | |
| 
 | |
| 			// Remove the "settings" field under "tlsSettings" and "realitySettings"
 | |
| 			tlsSettings, ok1 := stream["tlsSettings"].(map[string]interface{})
 | |
| 			realitySettings, ok2 := stream["realitySettings"].(map[string]interface{})
 | |
| 			if ok1 || ok2 {
 | |
| 				if ok1 {
 | |
| 					delete(tlsSettings, "settings")
 | |
| 				} else if ok2 {
 | |
| 					delete(realitySettings, "settings")
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			delete(stream, "externalProxy")
 | |
| 
 | |
| 			newStream, err := json.MarshalIndent(stream, "", "  ")
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 			inbound.StreamSettings = string(newStream)
 | |
| 		}
 | |
| 
 | |
| 		inboundConfig := inbound.GenXrayInboundConfig()
 | |
| 		xrayConfig.InboundConfigs = append(xrayConfig.InboundConfigs, *inboundConfig)
 | |
| 	}
 | |
| 	return xrayConfig, nil
 | |
| }
 | |
| 
 | |
| func (s *XrayService) GetXrayTraffic() ([]*xray.Traffic, []*xray.ClientTraffic, error) {
 | |
| 	if !s.IsXrayRunning() {
 | |
| 		return nil, nil, errors.New("xray is not running")
 | |
| 	}
 | |
| 	s.xrayAPI.Init(p.GetAPIPort())
 | |
| 	defer s.xrayAPI.Close()
 | |
| 	return s.xrayAPI.GetTraffic(true)
 | |
| }
 | |
| 
 | |
| func (s *XrayService) RestartXray(isForce bool) error {
 | |
| 	lock.Lock()
 | |
| 	defer lock.Unlock()
 | |
| 	logger.Debug("restart xray, force:", isForce)
 | |
| 
 | |
| 	xrayConfig, err := s.GetXrayConfig()
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	if s.IsXrayRunning() {
 | |
| 		if !isForce && p.GetConfig().Equals(xrayConfig) {
 | |
| 			logger.Debug("It does not need to restart xray")
 | |
| 			return nil
 | |
| 		}
 | |
| 		p.Stop()
 | |
| 	}
 | |
| 
 | |
| 	p = xray.NewProcess(xrayConfig)
 | |
| 	result = ""
 | |
| 	err = p.Start()
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (s *XrayService) StopXray() error {
 | |
| 	lock.Lock()
 | |
| 	defer lock.Unlock()
 | |
| 	logger.Debug("stop xray")
 | |
| 	if s.IsXrayRunning() {
 | |
| 		return p.Stop()
 | |
| 	}
 | |
| 	return errors.New("xray is not running")
 | |
| }
 | |
| 
 | |
| func (s *XrayService) SetToNeedRestart() {
 | |
| 	isNeedXrayRestart.Store(true)
 | |
| }
 | |
| 
 | |
| func (s *XrayService) IsNeedRestartAndSetFalse() bool {
 | |
| 	return isNeedXrayRestart.CompareAndSwap(true, false)
 | |
| }
 |