mirror of
https://github.com/nitezs/sub2sing-box.git
synced 2024-12-23 14:54:42 -05:00
142 lines
3.1 KiB
Go
Executable File
142 lines
3.1 KiB
Go
Executable File
package parser
|
|
|
|
import (
|
|
"fmt"
|
|
"net/url"
|
|
"strings"
|
|
|
|
"github.com/nitezs/sub2sing-box/constant"
|
|
"github.com/nitezs/sub2sing-box/model"
|
|
"github.com/nitezs/sub2sing-box/util"
|
|
"github.com/sagernet/sing-box/option"
|
|
)
|
|
|
|
func ParseShadowsocks(proxy string) (model.Outbound, error) {
|
|
if !strings.HasPrefix(proxy, constant.ShadowsocksPrefix) {
|
|
return model.Outbound{}, &ParseError{Type: ErrInvalidPrefix, Raw: proxy}
|
|
}
|
|
if !strings.Contains(proxy, "@") {
|
|
s := strings.SplitN(proxy, "#", 2)
|
|
d, err := util.DecodeBase64(strings.TrimPrefix(s[0], "ss://"))
|
|
if err != nil {
|
|
return model.Outbound{}, &ParseError{
|
|
Type: ErrInvalidStruct,
|
|
Message: "url parse error",
|
|
Raw: proxy,
|
|
}
|
|
}
|
|
if len(s) == 2 {
|
|
proxy = "ss://" + d + "#" + s[1]
|
|
} else {
|
|
proxy = "ss://" + d
|
|
}
|
|
}
|
|
link, err := url.Parse(proxy)
|
|
if err != nil {
|
|
return model.Outbound{}, &ParseError{
|
|
Type: ErrInvalidStruct,
|
|
Message: "url parse error",
|
|
Raw: proxy,
|
|
}
|
|
}
|
|
|
|
server := link.Hostname()
|
|
if server == "" {
|
|
return model.Outbound{}, &ParseError{
|
|
Type: ErrInvalidStruct,
|
|
Message: "missing server host",
|
|
Raw: proxy,
|
|
}
|
|
}
|
|
|
|
portStr := link.Port()
|
|
if portStr == "" {
|
|
return model.Outbound{}, &ParseError{
|
|
Type: ErrInvalidStruct,
|
|
Message: "missing server port",
|
|
Raw: proxy,
|
|
}
|
|
}
|
|
port, err := ParsePort(portStr)
|
|
if err != nil {
|
|
return model.Outbound{}, &ParseError{
|
|
Type: ErrInvalidStruct,
|
|
Raw: proxy,
|
|
}
|
|
}
|
|
|
|
method := link.User.Username()
|
|
password, _ := link.User.Password()
|
|
|
|
if password == "" {
|
|
user, err := util.DecodeBase64(method)
|
|
if err == nil {
|
|
methodAndPass := strings.SplitN(user, ":", 2)
|
|
if len(methodAndPass) == 2 {
|
|
method = methodAndPass[0]
|
|
password = methodAndPass[1]
|
|
}
|
|
}
|
|
}
|
|
if isLikelyBase64(password) {
|
|
password, err = util.DecodeBase64(password)
|
|
if err != nil {
|
|
return model.Outbound{}, &ParseError{
|
|
Type: ErrInvalidStruct,
|
|
Message: "password decode error",
|
|
Raw: proxy,
|
|
}
|
|
}
|
|
}
|
|
|
|
query := link.Query()
|
|
pluginStr := query.Get("plugin")
|
|
plugin := ""
|
|
options := ""
|
|
if pluginStr != "" {
|
|
arr := strings.SplitN(pluginStr, ";", 2)
|
|
if len(arr) == 2 {
|
|
plugin = arr[0]
|
|
options = arr[1]
|
|
}
|
|
}
|
|
|
|
remarks := link.Fragment
|
|
if remarks == "" {
|
|
remarks = fmt.Sprintf("%s:%s", server, portStr)
|
|
}
|
|
remarks = strings.TrimSpace(remarks)
|
|
|
|
result := model.Outbound{
|
|
Outbound: option.Outbound{
|
|
Type: "shadowsocks",
|
|
Tag: remarks,
|
|
ShadowsocksOptions: option.ShadowsocksOutboundOptions{
|
|
ServerOptions: option.ServerOptions{
|
|
Server: server,
|
|
ServerPort: port,
|
|
},
|
|
Method: method,
|
|
Password: password,
|
|
Plugin: plugin,
|
|
PluginOptions: options,
|
|
},
|
|
},
|
|
}
|
|
return result, nil
|
|
}
|
|
|
|
func isLikelyBase64(s string) bool {
|
|
if len(s)%4 == 0 && strings.HasSuffix(s, "=") && !strings.Contains(strings.TrimSuffix(s, "="), "=") {
|
|
s = strings.TrimSuffix(s, "=")
|
|
chars := "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"
|
|
for _, c := range s {
|
|
if !strings.ContainsRune(chars, c) {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
return false
|
|
}
|