| 
									
										
										
										
											2023-06-04 21:02:19 +00:00
										 |  |  | package xray | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2023-07-17 23:10:22 +00:00
										 |  |  | 	"encoding/json" | 
					
						
							| 
									
										
										
										
											2023-06-04 21:02:19 +00:00
										 |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"regexp" | 
					
						
							|  |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2024-03-10 21:31:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-17 23:10:22 +00:00
										 |  |  | 	"x-ui/logger" | 
					
						
							| 
									
										
										
										
											2023-06-04 21:02:19 +00:00
										 |  |  | 	"x-ui/util/common" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/xtls/xray-core/app/proxyman/command" | 
					
						
							|  |  |  | 	statsService "github.com/xtls/xray-core/app/stats/command" | 
					
						
							|  |  |  | 	"github.com/xtls/xray-core/common/protocol" | 
					
						
							|  |  |  | 	"github.com/xtls/xray-core/common/serial" | 
					
						
							| 
									
										
										
										
											2023-07-17 23:10:22 +00:00
										 |  |  | 	"github.com/xtls/xray-core/infra/conf" | 
					
						
							| 
									
										
										
										
											2023-06-04 21:02:19 +00:00
										 |  |  | 	"github.com/xtls/xray-core/proxy/shadowsocks" | 
					
						
							| 
									
										
										
										
											2023-07-27 08:04:46 +00:00
										 |  |  | 	"github.com/xtls/xray-core/proxy/shadowsocks_2022" | 
					
						
							| 
									
										
										
										
											2023-06-04 21:02:19 +00:00
										 |  |  | 	"github.com/xtls/xray-core/proxy/trojan" | 
					
						
							|  |  |  | 	"github.com/xtls/xray-core/proxy/vless" | 
					
						
							|  |  |  | 	"github.com/xtls/xray-core/proxy/vmess" | 
					
						
							|  |  |  | 	"google.golang.org/grpc" | 
					
						
							|  |  |  | 	"google.golang.org/grpc/credentials/insecure" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type XrayAPI struct { | 
					
						
							|  |  |  | 	HandlerServiceClient *command.HandlerServiceClient | 
					
						
							|  |  |  | 	StatsServiceClient   *statsService.StatsServiceClient | 
					
						
							|  |  |  | 	grpcClient           *grpc.ClientConn | 
					
						
							|  |  |  | 	isConnected          bool | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (x *XrayAPI) Init(apiPort int) (err error) { | 
					
						
							|  |  |  | 	if apiPort == 0 { | 
					
						
							|  |  |  | 		return common.NewError("xray api port wrong:", apiPort) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	x.grpcClient, err = grpc.Dial(fmt.Sprintf("127.0.0.1:%v", apiPort), grpc.WithTransportCredentials(insecure.NewCredentials())) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	x.isConnected = true | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	hsClient := command.NewHandlerServiceClient(x.grpcClient) | 
					
						
							|  |  |  | 	ssClient := statsService.NewStatsServiceClient(x.grpcClient) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	x.HandlerServiceClient = &hsClient | 
					
						
							|  |  |  | 	x.StatsServiceClient = &ssClient | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (x *XrayAPI) Close() { | 
					
						
							| 
									
										
										
										
											2024-01-02 09:42:07 +00:00
										 |  |  | 	if x.grpcClient != nil { | 
					
						
							|  |  |  | 		x.grpcClient.Close() | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-06-04 21:02:19 +00:00
										 |  |  | 	x.HandlerServiceClient = nil | 
					
						
							|  |  |  | 	x.StatsServiceClient = nil | 
					
						
							|  |  |  | 	x.isConnected = false | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-17 23:10:22 +00:00
										 |  |  | func (x *XrayAPI) AddInbound(inbound []byte) error { | 
					
						
							|  |  |  | 	client := *x.HandlerServiceClient | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	conf := new(conf.InboundDetourConfig) | 
					
						
							|  |  |  | 	err := json.Unmarshal(inbound, conf) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		logger.Debug("Failed to unmarshal inbound:", err) | 
					
						
							| 
									
										
										
										
											2023-07-27 08:28:12 +00:00
										 |  |  | 		return err | 
					
						
							| 
									
										
										
										
											2023-07-17 23:10:22 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	config, err := conf.Build() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		logger.Debug("Failed to build inbound Detur:", err) | 
					
						
							| 
									
										
										
										
											2023-07-27 08:28:12 +00:00
										 |  |  | 		return err | 
					
						
							| 
									
										
										
										
											2023-07-17 23:10:22 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	inboundConfig := command.AddInboundRequest{Inbound: config} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_, err = client.AddInbound(context.Background(), &inboundConfig) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return err | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (x *XrayAPI) DelInbound(tag string) error { | 
					
						
							|  |  |  | 	client := *x.HandlerServiceClient | 
					
						
							|  |  |  | 	_, err := client.RemoveInbound(context.Background(), &command.RemoveInboundRequest{ | 
					
						
							|  |  |  | 		Tag: tag, | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	return err | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-04 21:02:19 +00:00
										 |  |  | func (x *XrayAPI) AddUser(Protocol string, inboundTag string, user map[string]interface{}) error { | 
					
						
							|  |  |  | 	var account *serial.TypedMessage | 
					
						
							|  |  |  | 	switch Protocol { | 
					
						
							|  |  |  | 	case "vmess": | 
					
						
							|  |  |  | 		account = serial.ToTypedMessage(&vmess.Account{ | 
					
						
							| 
									
										
										
										
											2023-06-13 13:56:50 +00:00
										 |  |  | 			Id: user["id"].(string), | 
					
						
							| 
									
										
										
										
											2023-06-04 21:02:19 +00:00
										 |  |  | 		}) | 
					
						
							|  |  |  | 	case "vless": | 
					
						
							|  |  |  | 		account = serial.ToTypedMessage(&vless.Account{ | 
					
						
							|  |  |  | 			Id:   user["id"].(string), | 
					
						
							|  |  |  | 			Flow: user["flow"].(string), | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 	case "trojan": | 
					
						
							|  |  |  | 		account = serial.ToTypedMessage(&trojan.Account{ | 
					
						
							|  |  |  | 			Password: user["password"].(string), | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 	case "shadowsocks": | 
					
						
							| 
									
										
										
										
											2023-07-27 08:04:46 +00:00
										 |  |  | 		var ssCipherType shadowsocks.CipherType | 
					
						
							|  |  |  | 		switch user["cipher"].(string) { | 
					
						
							|  |  |  | 		case "aes-128-gcm": | 
					
						
							|  |  |  | 			ssCipherType = shadowsocks.CipherType_AES_128_GCM | 
					
						
							|  |  |  | 		case "aes-256-gcm": | 
					
						
							|  |  |  | 			ssCipherType = shadowsocks.CipherType_AES_256_GCM | 
					
						
							| 
									
										
										
										
											2023-08-29 19:02:49 +00:00
										 |  |  | 		case "chacha20-poly1305", "chacha20-ietf-poly1305": | 
					
						
							| 
									
										
										
										
											2023-07-27 08:04:46 +00:00
										 |  |  | 			ssCipherType = shadowsocks.CipherType_CHACHA20_POLY1305 | 
					
						
							| 
									
										
										
										
											2023-08-29 19:02:49 +00:00
										 |  |  | 		case "xchacha20-poly1305", "xchacha20-ietf-poly1305": | 
					
						
							| 
									
										
										
										
											2023-07-27 08:04:46 +00:00
										 |  |  | 			ssCipherType = shadowsocks.CipherType_XCHACHA20_POLY1305 | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			ssCipherType = shadowsocks.CipherType_NONE | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if ssCipherType != shadowsocks.CipherType_NONE { | 
					
						
							|  |  |  | 			account = serial.ToTypedMessage(&shadowsocks.Account{ | 
					
						
							|  |  |  | 				Password:   user["password"].(string), | 
					
						
							|  |  |  | 				CipherType: ssCipherType, | 
					
						
							|  |  |  | 			}) | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			account = serial.ToTypedMessage(&shadowsocks_2022.User{ | 
					
						
							|  |  |  | 				Key:   user["password"].(string), | 
					
						
							|  |  |  | 				Email: user["email"].(string), | 
					
						
							|  |  |  | 			}) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2023-06-04 21:02:19 +00:00
										 |  |  | 	default: | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	client := *x.HandlerServiceClient | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_, err := client.AlterInbound(context.Background(), &command.AlterInboundRequest{ | 
					
						
							|  |  |  | 		Tag: inboundTag, | 
					
						
							|  |  |  | 		Operation: serial.ToTypedMessage(&command.AddUserOperation{ | 
					
						
							|  |  |  | 			User: &protocol.User{ | 
					
						
							|  |  |  | 				Email:   user["email"].(string), | 
					
						
							|  |  |  | 				Account: account, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}), | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	return err | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (x *XrayAPI) RemoveUser(inboundTag string, email string) error { | 
					
						
							|  |  |  | 	client := *x.HandlerServiceClient | 
					
						
							|  |  |  | 	_, err := client.AlterInbound(context.Background(), &command.AlterInboundRequest{ | 
					
						
							|  |  |  | 		Tag: inboundTag, | 
					
						
							|  |  |  | 		Operation: serial.ToTypedMessage(&command.RemoveUserOperation{ | 
					
						
							|  |  |  | 			Email: email, | 
					
						
							|  |  |  | 		}), | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	return err | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (x *XrayAPI) GetTraffic(reset bool) ([]*Traffic, []*ClientTraffic, error) { | 
					
						
							|  |  |  | 	if x.grpcClient == nil { | 
					
						
							|  |  |  | 		return nil, nil, common.NewError("xray api is not initialized") | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-03-10 21:31:24 +00:00
										 |  |  | 	trafficRegex := regexp.MustCompile("(inbound|outbound)>>>([^>]+)>>>traffic>>>(downlink|uplink)") | 
					
						
							|  |  |  | 	ClientTrafficRegex := regexp.MustCompile("(user)>>>([^>]+)>>>traffic>>>(downlink|uplink)") | 
					
						
							| 
									
										
										
										
											2023-06-04 21:02:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	client := *x.StatsServiceClient | 
					
						
							|  |  |  | 	ctx, cancel := context.WithTimeout(context.Background(), time.Second*10) | 
					
						
							|  |  |  | 	defer cancel() | 
					
						
							|  |  |  | 	request := &statsService.QueryStatsRequest{ | 
					
						
							|  |  |  | 		Reset_: reset, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	resp, err := client.QueryStats(ctx, request) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	tagTrafficMap := map[string]*Traffic{} | 
					
						
							|  |  |  | 	emailTrafficMap := map[string]*ClientTraffic{} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	clientTraffics := make([]*ClientTraffic, 0) | 
					
						
							|  |  |  | 	traffics := make([]*Traffic, 0) | 
					
						
							|  |  |  | 	for _, stat := range resp.GetStat() { | 
					
						
							|  |  |  | 		matchs := trafficRegex.FindStringSubmatch(stat.Name) | 
					
						
							|  |  |  | 		if len(matchs) < 3 { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			matchs := ClientTrafficRegex.FindStringSubmatch(stat.Name) | 
					
						
							|  |  |  | 			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 | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		isInbound := matchs[1] == "inbound" | 
					
						
							| 
									
										
										
										
											2024-01-29 20:37:20 +00:00
										 |  |  | 		isOutbound := matchs[1] == "outbound" | 
					
						
							| 
									
										
										
										
											2023-06-04 21:02:19 +00:00
										 |  |  | 		tag := matchs[2] | 
					
						
							|  |  |  | 		isDown := matchs[3] == "downlink" | 
					
						
							|  |  |  | 		if tag == "api" { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		traffic, ok := tagTrafficMap[tag] | 
					
						
							|  |  |  | 		if !ok { | 
					
						
							|  |  |  | 			traffic = &Traffic{ | 
					
						
							| 
									
										
										
										
											2024-01-29 20:37:20 +00:00
										 |  |  | 				IsInbound:  isInbound, | 
					
						
							|  |  |  | 				IsOutbound: isOutbound, | 
					
						
							|  |  |  | 				Tag:        tag, | 
					
						
							| 
									
										
										
										
											2023-06-04 21:02:19 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			tagTrafficMap[tag] = traffic | 
					
						
							|  |  |  | 			traffics = append(traffics, traffic) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if isDown { | 
					
						
							|  |  |  | 			traffic.Down = stat.Value | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			traffic.Up = stat.Value | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return traffics, clientTraffics, nil | 
					
						
							|  |  |  | } |