mirror of
				https://github.com/bestnite/sub2sing-box.git
				synced 2025-10-25 08:41:01 +00:00 
			
		
		
		
	init
This commit is contained in:
		
							
								
								
									
										83
									
								
								parser/hysteria.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										83
									
								
								parser/hysteria.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,83 @@ | ||||
| package parser | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"net/url" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"sub2sing-box/model" | ||||
| ) | ||||
|  | ||||
| //hysteria://host:port?protocol=udp&auth=123456&peer=sni.domain&insecure=1&upmbps=100&downmbps=100&alpn=hysteria&obfs=xplus&obfsParam=123456#remarks | ||||
| // | ||||
| //- host: hostname or IP address of the server to connect to (required) | ||||
| //- port: port of the server to connect to (required) | ||||
| //- protocol: protocol to use ("udp", "wechat-video", "faketcp") (optional, default: "udp") | ||||
| //- auth: authentication payload (string) (optional) | ||||
| //- peer: SNI for TLS (optional) | ||||
| //- insecure: ignore certificate errors (optional) | ||||
| //- upmbps: upstream bandwidth in Mbps (required) | ||||
| //- downmbps: downstream bandwidth in Mbps (required) | ||||
| //- alpn: QUIC ALPN (optional) | ||||
| //- obfs: Obfuscation mode (optional, empty or "xplus") | ||||
| //- obfsParam: Obfuscation password (optional) | ||||
| //- remarks: remarks (optional) | ||||
|  | ||||
| func ParseHysteria(proxy string) (model.Proxy, error) { | ||||
| 	if !strings.HasPrefix(proxy, "hysteria://") { | ||||
| 		return model.Proxy{}, errors.New("invalid hysteria Url") | ||||
| 	} | ||||
| 	parts := strings.SplitN(strings.TrimPrefix(proxy, "hysteria://"), "?", 2) | ||||
| 	serverInfo := strings.SplitN(parts[0], ":", 2) | ||||
| 	if len(serverInfo) != 2 { | ||||
| 		return model.Proxy{}, errors.New("invalid hysteria Url") | ||||
| 	} | ||||
| 	params, err := url.ParseQuery(parts[1]) | ||||
| 	if err != nil { | ||||
| 		return model.Proxy{}, errors.New("invalid hysteria Url") | ||||
| 	} | ||||
| 	host := serverInfo[0] | ||||
| 	port, err := strconv.Atoi(serverInfo[1]) | ||||
| 	if err != nil { | ||||
| 		return model.Proxy{}, errors.New("invalid hysteria Url") | ||||
| 	} | ||||
| 	protocol := params.Get("protocol") | ||||
| 	auth := params.Get("auth") | ||||
| 	// peer := params.Get("peer") | ||||
| 	insecure := params.Get("insecure") | ||||
| 	upmbps := params.Get("upmbps") | ||||
| 	downmbps := params.Get("downmbps") | ||||
| 	alpn := params.Get("alpn") | ||||
| 	obfs := params.Get("obfs") | ||||
| 	// obfsParam := params.Get("obfsParam") | ||||
| 	remarks := "" | ||||
| 	if strings.Contains(parts[1], "#") { | ||||
| 		r := strings.Split(parts[1], "#") | ||||
| 		remarks = r[len(r)-1] | ||||
| 	} else { | ||||
| 		remarks = serverInfo[0] + ":" + serverInfo[1] | ||||
| 	} | ||||
| 	insecureBool, err := strconv.ParseBool(insecure) | ||||
| 	if err != nil { | ||||
| 		return model.Proxy{}, errors.New("invalid hysteria Url") | ||||
| 	} | ||||
| 	result := model.Proxy{ | ||||
| 		Type: "hysteria", | ||||
| 		Hysteria: model.Hysteria{ | ||||
| 			Tag:        remarks, | ||||
| 			Server:     host, | ||||
| 			ServerPort: uint16(port), | ||||
| 			Up:         upmbps, | ||||
| 			Down:       downmbps, | ||||
| 			Auth:       []byte(auth), | ||||
| 			Obfs:       obfs, | ||||
| 			Network:    protocol, | ||||
| 			TLS: &model.OutboundTLSOptions{ | ||||
| 				Enabled:  true, | ||||
| 				Insecure: insecureBool, | ||||
| 				ALPN:     strings.Split(alpn, ","), | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
| 	return result, nil | ||||
| } | ||||
							
								
								
									
										62
									
								
								parser/hysteria2.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								parser/hysteria2.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,62 @@ | ||||
| package parser | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"net/url" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"sub2sing-box/model" | ||||
| ) | ||||
|  | ||||
| // hysteria2://letmein@example.com/?insecure=1&obfs=salamander&obfs-password=gawrgura&pinSHA256=deadbeef&sni=real.example.com | ||||
|  | ||||
| func ParseHysteria2(proxy string) (model.Proxy, error) { | ||||
| 	if !strings.HasPrefix(proxy, "hysteria2://") && !strings.HasPrefix(proxy, "hy2://") { | ||||
| 		return model.Proxy{}, errors.New("invalid hysteria2 Url") | ||||
| 	} | ||||
| 	parts := strings.SplitN(strings.TrimPrefix(proxy, "hysteria2://"), "@", 2) | ||||
| 	serverInfo := strings.SplitN(parts[1], "/?", 2) | ||||
| 	serverAndPort := strings.SplitN(serverInfo[0], ":", 2) | ||||
| 	if len(serverAndPort) == 1 { | ||||
| 		serverAndPort = append(serverAndPort, "443") | ||||
| 	} else if len(serverAndPort) != 2 { | ||||
| 		return model.Proxy{}, errors.New("invalid hysteria2 Url") | ||||
| 	} | ||||
| 	params, err := url.ParseQuery(serverInfo[1]) | ||||
| 	if err != nil { | ||||
| 		return model.Proxy{}, errors.New("invalid hysteria2 Url") | ||||
| 	} | ||||
| 	port, err := strconv.Atoi(serverAndPort[1]) | ||||
| 	if err != nil { | ||||
| 		return model.Proxy{}, errors.New("invalid hysteria2 Url") | ||||
| 	} | ||||
| 	remarks := params.Get("name") | ||||
| 	certificate := make([]string, 0) | ||||
| 	if params.Get("pinSHA256") != "" { | ||||
| 		certificate = append(certificate, params.Get("pinSHA256")) | ||||
| 	} | ||||
| 	server := serverAndPort[0] | ||||
| 	password := parts[0] | ||||
| 	network := params.Get("network") | ||||
| 	result := model.Proxy{ | ||||
| 		Type: "hysteria2", | ||||
| 		Hysteria2: model.Hysteria2{ | ||||
| 			Tag:        remarks, | ||||
| 			Server:     server, | ||||
| 			ServerPort: uint16(port), | ||||
| 			Password:   password, | ||||
| 			Obfs: &model.Hysteria2Obfs{ | ||||
| 				Type:     params.Get("obfs"), | ||||
| 				Password: params.Get("obfs-password"), | ||||
| 			}, | ||||
| 			TLS: &model.OutboundTLSOptions{ | ||||
| 				Enabled:     params.Get("pinSHA256") != "", | ||||
| 				Insecure:    params.Get("insecure") == "1", | ||||
| 				ServerName:  params.Get("sni"), | ||||
| 				Certificate: certificate, | ||||
| 			}, | ||||
| 			Network: network, | ||||
| 		}, | ||||
| 	} | ||||
| 	return result, nil | ||||
| } | ||||
							
								
								
									
										64
									
								
								parser/shadowsocks.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								parser/shadowsocks.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,64 @@ | ||||
| package parser | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"net/url" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"sub2sing-box/model" | ||||
| 	. "sub2sing-box/util" | ||||
| ) | ||||
|  | ||||
| func ParseShadowsocks(proxy string) (model.Proxy, error) { | ||||
| 	if !strings.HasPrefix(proxy, "ss://") { | ||||
| 		return model.Proxy{}, errors.New("invalid ss Url") | ||||
| 	} | ||||
| 	parts := strings.SplitN(strings.TrimPrefix(proxy, "ss://"), "@", 2) | ||||
| 	if len(parts) != 2 { | ||||
| 		return model.Proxy{}, errors.New("invalid ss Url") | ||||
| 	} | ||||
| 	if !strings.Contains(parts[0], ":") { | ||||
| 		decoded, err := DecodeBase64(parts[0]) | ||||
| 		if err != nil { | ||||
| 			return model.Proxy{}, errors.New("invalid ss Url" + err.Error()) | ||||
| 		} | ||||
| 		parts[0] = decoded | ||||
| 	} | ||||
| 	credentials := strings.SplitN(parts[0], ":", 2) | ||||
| 	if len(credentials) != 2 { | ||||
| 		return model.Proxy{}, errors.New("invalid ss Url") | ||||
| 	} | ||||
| 	serverInfo := strings.SplitN(parts[1], "#", 2) | ||||
| 	serverAndPort := strings.SplitN(serverInfo[0], ":", 2) | ||||
| 	if len(serverAndPort) != 2 { | ||||
| 		return model.Proxy{}, errors.New("invalid ss Url") | ||||
| 	} | ||||
| 	port, err := strconv.Atoi(strings.TrimSpace(serverAndPort[1])) | ||||
| 	if err != nil { | ||||
| 		return model.Proxy{}, errors.New("invalid ss Url" + err.Error()) | ||||
| 	} | ||||
| 	remarks := "" | ||||
| 	if len(serverInfo) == 2 { | ||||
| 		unescape, err := url.QueryUnescape(serverInfo[1]) | ||||
| 		if err != nil { | ||||
| 			return model.Proxy{}, errors.New("invalid ss Url" + err.Error()) | ||||
| 		} | ||||
| 		remarks = strings.TrimSpace(unescape) | ||||
| 	} else { | ||||
| 		remarks = strings.TrimSpace(serverAndPort[0]) | ||||
| 	} | ||||
| 	method := credentials[0] | ||||
| 	password := credentials[1] | ||||
| 	server := strings.TrimSpace(serverAndPort[0]) | ||||
| 	result := model.Proxy{ | ||||
| 		Type: "shadowsocks", | ||||
| 		Shadowsocks: model.Shadowsocks{ | ||||
| 			Tag:        remarks, | ||||
| 			Method:     method, | ||||
| 			Password:   password, | ||||
| 			Server:     server, | ||||
| 			ServerPort: uint16(port), | ||||
| 		}, | ||||
| 	} | ||||
| 	return result, nil | ||||
| } | ||||
							
								
								
									
										56
									
								
								parser/trojan.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								parser/trojan.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,56 @@ | ||||
| package parser | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"net/url" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"sub2sing-box/model" | ||||
| ) | ||||
|  | ||||
| func ParseTrojan(proxy string) (model.Proxy, error) { | ||||
| 	if !strings.HasPrefix(proxy, "trojan://") { | ||||
| 		return model.Proxy{}, fmt.Errorf("invalid trojan Url") | ||||
| 	} | ||||
| 	parts := strings.SplitN(strings.TrimPrefix(proxy, "trojan://"), "@", 2) | ||||
| 	if len(parts) != 2 { | ||||
| 		return model.Proxy{}, fmt.Errorf("invalid trojan Url") | ||||
| 	} | ||||
| 	serverInfo := strings.SplitN(parts[1], "#", 2) | ||||
| 	serverAndPortAndParams := strings.SplitN(serverInfo[0], "?", 2) | ||||
| 	serverAndPort := strings.SplitN(serverAndPortAndParams[0], ":", 2) | ||||
| 	params, err := url.ParseQuery(serverAndPortAndParams[1]) | ||||
| 	if err != nil { | ||||
| 		return model.Proxy{}, err | ||||
| 	} | ||||
| 	if len(serverAndPort) != 2 { | ||||
| 		return model.Proxy{}, fmt.Errorf("invalid trojan") | ||||
| 	} | ||||
| 	port, err := strconv.Atoi(strings.TrimSpace(serverAndPort[1])) | ||||
| 	if err != nil { | ||||
| 		return model.Proxy{}, err | ||||
| 	} | ||||
| 	remarks := "" | ||||
| 	if len(serverInfo) == 2 { | ||||
| 		remarks, _ = url.QueryUnescape(strings.TrimSpace(serverInfo[1])) | ||||
| 	} else { | ||||
| 		remarks = serverAndPort[0] | ||||
| 	} | ||||
| 	server := strings.TrimSpace(serverAndPort[0]) | ||||
| 	password := strings.TrimSpace(parts[0]) | ||||
| 	result := model.Proxy{ | ||||
| 		Type: "trojan", | ||||
| 		Trojan: model.Trojan{ | ||||
| 			Tag:        remarks, | ||||
| 			Server:     server, | ||||
| 			ServerPort: uint16(port), | ||||
| 			TLS: &model.OutboundTLSOptions{ | ||||
| 				Enabled:    true, | ||||
| 				ServerName: params.Get("sni"), | ||||
| 			}, | ||||
| 			Password: password, | ||||
| 			Network:  "tcp", | ||||
| 		}, | ||||
| 	} | ||||
| 	return result, nil | ||||
| } | ||||
							
								
								
									
										89
									
								
								parser/vless.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										89
									
								
								parser/vless.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,89 @@ | ||||
| package parser | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"net/url" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"sub2sing-box/model" | ||||
| ) | ||||
|  | ||||
| func ParseVless(proxy string) (model.Proxy, error) { | ||||
| 	if !strings.HasPrefix(proxy, "vless://") { | ||||
| 		return model.Proxy{}, fmt.Errorf("invalid vless Url") | ||||
| 	} | ||||
| 	parts := strings.SplitN(strings.TrimPrefix(proxy, "vless://"), "@", 2) | ||||
| 	if len(parts) != 2 { | ||||
| 		return model.Proxy{}, fmt.Errorf("invalid vless Url") | ||||
| 	} | ||||
| 	serverInfo := strings.SplitN(parts[1], "#", 2) | ||||
| 	serverAndPortAndParams := strings.SplitN(serverInfo[0], "?", 2) | ||||
| 	serverAndPort := strings.SplitN(serverAndPortAndParams[0], ":", 2) | ||||
| 	params, err := url.ParseQuery(serverAndPortAndParams[1]) | ||||
| 	if err != nil { | ||||
| 		return model.Proxy{}, err | ||||
| 	} | ||||
| 	if len(serverAndPort) != 2 { | ||||
| 		return model.Proxy{}, fmt.Errorf("invalid vless") | ||||
| 	} | ||||
| 	port, err := strconv.Atoi(strings.TrimSpace(serverAndPort[1])) | ||||
| 	if err != nil { | ||||
| 		return model.Proxy{}, err | ||||
| 	} | ||||
| 	remarks := "" | ||||
| 	if len(serverInfo) == 2 { | ||||
| 		if strings.Contains(serverInfo[1], "|") { | ||||
| 			remarks = strings.SplitN(serverInfo[1], "|", 2)[1] | ||||
| 		} else { | ||||
| 			remarks, err = url.QueryUnescape(serverInfo[1]) | ||||
| 			if err != nil { | ||||
| 				return model.Proxy{}, err | ||||
| 			} | ||||
| 		} | ||||
| 	} else { | ||||
| 		remarks, err = url.QueryUnescape(serverAndPort[0]) | ||||
| 		if err != nil { | ||||
| 			return model.Proxy{}, err | ||||
| 		} | ||||
| 	} | ||||
| 	server := strings.TrimSpace(serverAndPort[0]) | ||||
| 	uuid := strings.TrimSpace(parts[0]) | ||||
| 	network := params.Get("type") | ||||
| 	result := model.Proxy{ | ||||
| 		Type: "vless", | ||||
| 		VLESS: model.VLESS{ | ||||
| 			Tag:        remarks, | ||||
| 			Server:     server, | ||||
| 			ServerPort: uint16(port), | ||||
| 			UUID:       uuid, | ||||
| 			Network:    network, | ||||
| 			TLS: &model.OutboundTLSOptions{ | ||||
| 				Enabled:    params.Get("security") == "reality", | ||||
| 				ServerName: params.Get("sni"), | ||||
| 				UTLS: &model.OutboundUTLSOptions{ | ||||
| 					Enabled:     params.Get("fp") != "", | ||||
| 					Fingerprint: params.Get("fp"), | ||||
| 				}, | ||||
| 				Reality: &model.OutboundRealityOptions{ | ||||
| 					Enabled:   params.Get("pbk") != "", | ||||
| 					PublicKey: params.Get("pbk"), | ||||
| 					ShortID:   params.Get("sid"), | ||||
| 				}, | ||||
| 				ALPN: strings.Split(params.Get("alpn"), ","), | ||||
| 			}, | ||||
| 			Transport: &model.V2RayTransportOptions{ | ||||
| 				WebsocketOptions: model.V2RayWebsocketOptions{ | ||||
| 					Path: params.Get("path"), | ||||
| 					Headers: map[string]string{ | ||||
| 						"Host": params.Get("host"), | ||||
| 					}, | ||||
| 				}, | ||||
| 				GRPCOptions: model.V2RayGRPCOptions{ | ||||
| 					ServiceName: params.Get("serviceName"), | ||||
| 				}, | ||||
| 			}, | ||||
| 			Flow: params.Get("flow"), | ||||
| 		}, | ||||
| 	} | ||||
| 	return result, nil | ||||
| } | ||||
							
								
								
									
										133
									
								
								parser/vmess.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										133
									
								
								parser/vmess.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,133 @@ | ||||
| package parser | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"net/url" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"sub2sing-box/model" | ||||
| 	. "sub2sing-box/util" | ||||
| ) | ||||
|  | ||||
| func ParseVmess(proxy string) (model.Proxy, error) { | ||||
| 	if !strings.HasPrefix(proxy, "vmess://") { | ||||
| 		return model.Proxy{}, errors.New("invalid vmess url") | ||||
| 	} | ||||
| 	base64, err := DecodeBase64(strings.TrimPrefix(proxy, "vmess://")) | ||||
| 	if err != nil { | ||||
| 		return model.Proxy{}, errors.New("invalid vmess url" + err.Error()) | ||||
| 	} | ||||
| 	var vmess model.VmessJson | ||||
| 	err = json.Unmarshal([]byte(base64), &vmess) | ||||
| 	if err != nil { | ||||
| 		return model.Proxy{}, errors.New("invalid vmess url" + err.Error()) | ||||
| 	} | ||||
| 	port := 0 | ||||
| 	switch vmess.Port.(type) { | ||||
| 	case string: | ||||
| 		port, err = strconv.Atoi(vmess.Port.(string)) | ||||
| 		if err != nil { | ||||
| 			return model.Proxy{}, errors.New("invalid vmess url" + err.Error()) | ||||
| 		} | ||||
| 	case float64: | ||||
| 		port = int(vmess.Port.(float64)) | ||||
| 	} | ||||
| 	aid := 0 | ||||
| 	switch vmess.Aid.(type) { | ||||
| 	case string: | ||||
| 		aid, err = strconv.Atoi(vmess.Aid.(string)) | ||||
| 		if err != nil { | ||||
| 			return model.Proxy{}, errors.New("invalid vmess url" + err.Error()) | ||||
| 		} | ||||
| 	case float64: | ||||
| 		aid = int(vmess.Aid.(float64)) | ||||
| 	} | ||||
| 	if vmess.Scy == "" { | ||||
| 		vmess.Scy = "auto" | ||||
| 	} | ||||
|  | ||||
| 	name, err := url.QueryUnescape(vmess.Ps) | ||||
| 	if err != nil { | ||||
| 		name = vmess.Ps | ||||
| 	} | ||||
|  | ||||
| 	result := model.Proxy{ | ||||
| 		Type: "vmess", | ||||
| 		VMess: model.VMess{ | ||||
| 			Tag:        name, | ||||
| 			Server:     vmess.Add, | ||||
| 			ServerPort: uint16(port), | ||||
| 			UUID:       vmess.Id, | ||||
| 			AlterId:    aid, | ||||
| 			Security:   vmess.Scy, | ||||
| 			Network:    vmess.Net, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	if vmess.Tls == "tls" { | ||||
| 		tls := model.OutboundTLSOptions{ | ||||
| 			Enabled: true, | ||||
| 			UTLS: &model.OutboundUTLSOptions{ | ||||
| 				Fingerprint: vmess.Fp, | ||||
| 			}, | ||||
| 			ALPN: strings.Split(vmess.Alpn, ","), | ||||
| 		} | ||||
| 		result.VMess.TLS = &tls | ||||
| 	} | ||||
|  | ||||
| 	if vmess.Net == "ws" { | ||||
| 		if vmess.Path == "" { | ||||
| 			vmess.Path = "/" | ||||
| 		} | ||||
| 		if vmess.Host == "" { | ||||
| 			vmess.Host = vmess.Add | ||||
| 		} | ||||
| 		ws := model.V2RayWebsocketOptions{ | ||||
| 			Path: vmess.Path, | ||||
| 			Headers: map[string]string{ | ||||
| 				"Host": vmess.Host, | ||||
| 			}, | ||||
| 		} | ||||
| 		transport := model.V2RayTransportOptions{ | ||||
| 			Type:             "ws", | ||||
| 			WebsocketOptions: ws, | ||||
| 		} | ||||
| 		result.VMess.Transport = &transport | ||||
| 	} | ||||
|  | ||||
| 	if vmess.Net == "quic" { | ||||
| 		quic := model.V2RayQUICOptions{} | ||||
| 		transport := model.V2RayTransportOptions{ | ||||
| 			Type:        "quic", | ||||
| 			QUICOptions: quic, | ||||
| 		} | ||||
| 		result.VMess.Transport = &transport | ||||
| 	} | ||||
|  | ||||
| 	if vmess.Net == "grpc" { | ||||
| 		grpc := model.V2RayGRPCOptions{ | ||||
| 			ServiceName:         vmess.Path, | ||||
| 			PermitWithoutStream: true, | ||||
| 		} | ||||
| 		transport := model.V2RayTransportOptions{ | ||||
| 			Type:        "grpc", | ||||
| 			GRPCOptions: grpc, | ||||
| 		} | ||||
| 		result.VMess.Transport = &transport | ||||
| 	} | ||||
|  | ||||
| 	if vmess.Net == "h2" { | ||||
| 		httpOps := model.V2RayHTTPOptions{ | ||||
| 			Host: strings.Split(vmess.Host, ","), | ||||
| 			Path: vmess.Path, | ||||
| 		} | ||||
| 		transport := model.V2RayTransportOptions{ | ||||
| 			Type:        "http", | ||||
| 			HTTPOptions: httpOps, | ||||
| 		} | ||||
| 		result.VMess.Transport = &transport | ||||
| 	} | ||||
|  | ||||
| 	return result, nil | ||||
| } | ||||
		Reference in New Issue
	
	Block a user