mirror of
				https://github.com/MHSanaei/3x-ui.git
				synced 2025-10-25 17:44:40 +00:00 
			
		
		
		
	Optimize XrayAPI functionality and structure
This commit is contained in:
		
							parent
							
								
									767ee4ec2b
								
							
						
					
					
						commit
						5afb8d85fc
					
				
					 1 changed files with 91 additions and 81 deletions
				
			
		
							
								
								
									
										142
									
								
								xray/api.go
									
									
									
									
									
								
							
							
						
						
									
										142
									
								
								xray/api.go
									
									
									
									
									
								
							|  | @ -31,24 +31,27 @@ type XrayAPI struct { | ||||||
| 	isConnected          bool | 	isConnected          bool | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (x *XrayAPI) Init(apiPort int) (err error) { | func (x *XrayAPI) Init(apiPort int) error { | ||||||
| 	if apiPort == 0 { | 	if apiPort <= 0 { | ||||||
| 		return common.NewError("xray api port wrong:", apiPort) | 		return fmt.Errorf("invalid Xray API port: %d", apiPort) | ||||||
| 	} | 	} | ||||||
| 	conn, err := grpc.NewClient(fmt.Sprintf("127.0.0.1:%v", apiPort), grpc.WithTransportCredentials(insecure.NewCredentials())) | 
 | ||||||
|  | 	addr := fmt.Sprintf("127.0.0.1:%d", apiPort) | ||||||
|  | 	conn, err := grpc.NewClient(addr, grpc.WithTransportCredentials(insecure.NewCredentials())) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return fmt.Errorf("failed to connect to Xray API: %w", err) | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	x.grpcClient = conn | 	x.grpcClient = conn | ||||||
| 	x.isConnected = true | 	x.isConnected = true | ||||||
| 
 | 
 | ||||||
| 	hsClient := command.NewHandlerServiceClient(x.grpcClient) | 	hsClient := command.NewHandlerServiceClient(conn) | ||||||
| 	ssClient := statsService.NewStatsServiceClient(x.grpcClient) | 	ssClient := statsService.NewStatsServiceClient(conn) | ||||||
| 
 | 
 | ||||||
| 	x.HandlerServiceClient = &hsClient | 	x.HandlerServiceClient = &hsClient | ||||||
| 	x.StatsServiceClient = &ssClient | 	x.StatsServiceClient = &ssClient | ||||||
| 
 | 
 | ||||||
| 	return | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (x *XrayAPI) Close() { | func (x *XrayAPI) Close() { | ||||||
|  | @ -149,94 +152,101 @@ func (x *XrayAPI) AddUser(Protocol string, inboundTag string, user map[string]in | ||||||
| 	return err | 	return err | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (x *XrayAPI) RemoveUser(inboundTag string, email string) error { | func (x *XrayAPI) RemoveUser(inboundTag, email string) error { | ||||||
| 	client := *x.HandlerServiceClient | 	ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) | ||||||
| 	_, err := client.AlterInbound(context.Background(), &command.AlterInboundRequest{ | 	defer cancel() | ||||||
|  | 
 | ||||||
|  | 	op := &command.RemoveUserOperation{Email: email} | ||||||
|  | 	req := &command.AlterInboundRequest{ | ||||||
| 		Tag:       inboundTag, | 		Tag:       inboundTag, | ||||||
| 		Operation: serial.ToTypedMessage(&command.RemoveUserOperation{ | 		Operation: serial.ToTypedMessage(op), | ||||||
| 			Email: email, | 	} | ||||||
| 		}), | 
 | ||||||
| 	}) | 	_, err := (*x.HandlerServiceClient).AlterInbound(ctx, req) | ||||||
| 	return err | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("failed to remove user: %w", err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (x *XrayAPI) GetTraffic(reset bool) ([]*Traffic, []*ClientTraffic, error) { | func (x *XrayAPI) GetTraffic(reset bool) ([]*Traffic, []*ClientTraffic, error) { | ||||||
| 	if x.grpcClient == nil { | 	if x.grpcClient == nil { | ||||||
| 		return nil, nil, common.NewError("xray api is not initialized") | 		return nil, nil, common.NewError("xray api is not initialized") | ||||||
| 	} | 	} | ||||||
| 	trafficRegex := regexp.MustCompile("(inbound|outbound)>>>([^>]+)>>>traffic>>>(downlink|uplink)") |  | ||||||
| 	ClientTrafficRegex := regexp.MustCompile("(user)>>>([^>]+)>>>traffic>>>(downlink|uplink)") |  | ||||||
| 
 | 
 | ||||||
| 	client := *x.StatsServiceClient | 	trafficRegex := regexp.MustCompile(`(inbound|outbound)>>>([^>]+)>>>traffic>>>(downlink|uplink)`) | ||||||
|  | 	clientTrafficRegex := regexp.MustCompile(`user>>>([^>]+)>>>traffic>>>(downlink|uplink)`) | ||||||
|  | 
 | ||||||
| 	ctx, cancel := context.WithTimeout(context.Background(), time.Second*10) | 	ctx, cancel := context.WithTimeout(context.Background(), time.Second*10) | ||||||
| 	defer cancel() | 	defer cancel() | ||||||
| 	request := &statsService.QueryStatsRequest{ | 
 | ||||||
| 		Reset_: reset, | 	resp, err := (*x.StatsServiceClient).QueryStats(ctx, &statsService.QueryStatsRequest{Reset_: reset}) | ||||||
| 	} |  | ||||||
| 	resp, err := client.QueryStats(ctx, request) |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, nil, err | 		return nil, nil, err | ||||||
| 	} | 	} | ||||||
| 	tagTrafficMap := map[string]*Traffic{} |  | ||||||
| 	emailTrafficMap := map[string]*ClientTraffic{} |  | ||||||
| 
 | 
 | ||||||
| 	clientTraffics := make([]*ClientTraffic, 0) | 	tagTrafficMap := make(map[string]*Traffic) | ||||||
| 	traffics := make([]*Traffic, 0) | 	emailTrafficMap := make(map[string]*ClientTraffic) | ||||||
|  | 
 | ||||||
| 	for _, stat := range resp.GetStat() { | 	for _, stat := range resp.GetStat() { | ||||||
| 		matchs := trafficRegex.FindStringSubmatch(stat.Name) | 		if matches := trafficRegex.FindStringSubmatch(stat.Name); len(matches) == 4 { | ||||||
| 		if len(matchs) < 3 { | 			processTraffic(matches, stat.Value, tagTrafficMap) | ||||||
| 
 | 		} else if matches := clientTrafficRegex.FindStringSubmatch(stat.Name); len(matches) == 3 { | ||||||
| 			matchs := ClientTrafficRegex.FindStringSubmatch(stat.Name) | 			processClientTraffic(matches, stat.Value, emailTrafficMap) | ||||||
| 			if len(matchs) < 3 { |  | ||||||
| 				continue |  | ||||||
| 			} else { |  | ||||||
| 
 |  | ||||||
| 				isUser := matchs[1] == "user" |  | ||||||
| 				email := matchs[2] |  | ||||||
| 				isDown := matchs[3] == "downlink" |  | ||||||
| 				if !isUser { |  | ||||||
| 					continue |  | ||||||
| 		} | 		} | ||||||
| 				traffic, ok := emailTrafficMap[email] |  | ||||||
| 				if !ok { |  | ||||||
| 					traffic = &ClientTraffic{ |  | ||||||
| 						Email: email, |  | ||||||
| 					} |  | ||||||
| 					emailTrafficMap[email] = traffic |  | ||||||
| 					clientTraffics = append(clientTraffics, traffic) |  | ||||||
| 				} |  | ||||||
| 				if isDown { |  | ||||||
| 					traffic.Down = stat.Value |  | ||||||
| 				} else { |  | ||||||
| 					traffic.Up = stat.Value |  | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	return mapToSlice(tagTrafficMap), mapToSlice(emailTrafficMap), nil | ||||||
| } | } | ||||||
| 			continue | 
 | ||||||
| 		} | func processTraffic(matches []string, value int64, trafficMap map[string]*Traffic) { | ||||||
| 		isInbound := matchs[1] == "inbound" | 	isInbound := matches[1] == "inbound" | ||||||
| 		isOutbound := matchs[1] == "outbound" | 	tag := matches[2] | ||||||
| 		tag := matchs[2] | 	isDown := matches[3] == "downlink" | ||||||
| 		isDown := matchs[3] == "downlink" | 
 | ||||||
| 	if tag == "api" { | 	if tag == "api" { | ||||||
| 			continue | 		return | ||||||
| 	} | 	} | ||||||
| 		traffic, ok := tagTrafficMap[tag] | 
 | ||||||
|  | 	traffic, ok := trafficMap[tag] | ||||||
| 	if !ok { | 	if !ok { | ||||||
| 		traffic = &Traffic{ | 		traffic = &Traffic{ | ||||||
| 			IsInbound:  isInbound, | 			IsInbound:  isInbound, | ||||||
| 				IsOutbound: isOutbound, | 			IsOutbound: !isInbound, | ||||||
| 			Tag:        tag, | 			Tag:        tag, | ||||||
| 		} | 		} | ||||||
| 			tagTrafficMap[tag] = traffic | 		trafficMap[tag] = traffic | ||||||
| 			traffics = append(traffics, traffic) |  | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	if isDown { | 	if isDown { | ||||||
| 			traffic.Down = stat.Value | 		traffic.Down = value | ||||||
| 	} else { | 	} else { | ||||||
| 			traffic.Up = stat.Value | 		traffic.Up = value | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 	return traffics, clientTraffics, nil | func processClientTraffic(matches []string, value int64, clientTrafficMap map[string]*ClientTraffic) { | ||||||
|  | 	email := matches[1] | ||||||
|  | 	isDown := matches[2] == "downlink" | ||||||
|  | 
 | ||||||
|  | 	traffic, ok := clientTrafficMap[email] | ||||||
|  | 	if !ok { | ||||||
|  | 		traffic = &ClientTraffic{Email: email} | ||||||
|  | 		clientTrafficMap[email] = traffic | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if isDown { | ||||||
|  | 		traffic.Down = value | ||||||
|  | 	} else { | ||||||
|  | 		traffic.Up = value | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func mapToSlice[T any](m map[string]*T) []*T { | ||||||
|  | 	result := make([]*T, 0, len(m)) | ||||||
|  | 	for _, v := range m { | ||||||
|  | 		result = append(result, v) | ||||||
|  | 	} | ||||||
|  | 	return result | ||||||
| } | } | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue
	
	 mhsanaei
						mhsanaei