2024-03-11 23:39:58 +08:00
|
|
|
package util
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
2024-03-12 01:34:08 +08:00
|
|
|
"fmt"
|
2024-03-11 23:39:58 +08:00
|
|
|
"os"
|
2024-03-18 22:32:33 +08:00
|
|
|
"path/filepath"
|
2024-03-11 23:39:58 +08:00
|
|
|
"regexp"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"sub2sing-box/internal/model"
|
2024-03-19 17:01:53 +08:00
|
|
|
"sub2sing-box/internal/util"
|
2024-03-11 23:39:58 +08:00
|
|
|
"sub2sing-box/pkg/parser"
|
|
|
|
)
|
|
|
|
|
2024-03-12 01:34:08 +08:00
|
|
|
func Convert(subscriptions []string, proxies []string, template string, delete string, rename map[string]string) (string, error) {
|
|
|
|
result := ""
|
|
|
|
var err error
|
|
|
|
|
|
|
|
proxyList, err := ConvertSubscriptionsToSProxy(subscriptions)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
for _, proxy := range proxies {
|
|
|
|
p, err := ConvertCProxyToSProxy(proxy)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
proxyList = append(proxyList, p)
|
|
|
|
}
|
|
|
|
|
|
|
|
if delete != "" {
|
|
|
|
proxyList, err = DeleteProxy(proxyList, delete)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for k, v := range rename {
|
|
|
|
proxyList, err = RenameProxy(proxyList, k, v)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
keep := make(map[int]bool)
|
|
|
|
set := make(map[string]struct {
|
|
|
|
Proxy model.Proxy
|
|
|
|
Count int
|
|
|
|
})
|
|
|
|
for i, p := range proxyList {
|
|
|
|
if _, exists := set[p.Tag]; !exists {
|
|
|
|
keep[i] = true
|
|
|
|
set[p.Tag] = struct {
|
|
|
|
Proxy model.Proxy
|
|
|
|
Count int
|
|
|
|
}{p, 0}
|
|
|
|
} else {
|
|
|
|
p1, _ := json.Marshal(p)
|
|
|
|
p2, _ := json.Marshal(set[p.Tag])
|
|
|
|
if string(p1) != string(p2) {
|
|
|
|
set[p.Tag] = struct {
|
|
|
|
Proxy model.Proxy
|
|
|
|
Count int
|
|
|
|
}{p, set[p.Tag].Count + 1}
|
|
|
|
keep[i] = true
|
|
|
|
proxyList[i].Tag = fmt.Sprintf("%s %d", p.Tag, set[p.Tag].Count)
|
|
|
|
} else {
|
|
|
|
keep[i] = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
var newProxyList []model.Proxy
|
|
|
|
for i, p := range proxyList {
|
|
|
|
if keep[i] {
|
|
|
|
newProxyList = append(newProxyList, p)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
proxyList = newProxyList
|
|
|
|
|
|
|
|
if template != "" {
|
|
|
|
result, err = MergeTemplate(proxyList, template)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
r, err := json.Marshal(proxyList)
|
|
|
|
result = string(r)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return string(result), nil
|
|
|
|
}
|
|
|
|
|
2024-03-11 23:39:58 +08:00
|
|
|
func MergeTemplate(proxies []model.Proxy, template string) (string, error) {
|
2024-03-19 17:01:53 +08:00
|
|
|
var config model.Config
|
|
|
|
var err error
|
|
|
|
if strings.HasPrefix(template, "http") {
|
|
|
|
data, err := util.Fetch(template, 3)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
err = json.Unmarshal([]byte(data), &config)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if !strings.Contains(template, string(filepath.Separator)) {
|
|
|
|
if _, err := os.Stat(template); os.IsNotExist(err) {
|
|
|
|
template = filepath.Join("templates", template)
|
|
|
|
}
|
2024-03-18 22:32:33 +08:00
|
|
|
}
|
2024-03-19 17:01:53 +08:00
|
|
|
config, err = ReadTemplate(template)
|
2024-03-18 22:32:33 +08:00
|
|
|
}
|
2024-03-11 23:39:58 +08:00
|
|
|
proxyTags := make([]string, 0)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
for _, p := range proxies {
|
|
|
|
proxyTags = append(proxyTags, p.Tag)
|
|
|
|
}
|
|
|
|
ps, err := json.Marshal(&proxies)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
var newOutbounds []model.Outbound
|
|
|
|
err = json.Unmarshal(ps, &newOutbounds)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
for i, outbound := range config.Outbounds {
|
2024-03-18 22:32:33 +08:00
|
|
|
var parsedOutbound []string = make([]string, 0)
|
|
|
|
for _, o := range outbound.Outbounds {
|
|
|
|
if o == "<all-proxy-tags>" {
|
|
|
|
parsedOutbound = append(parsedOutbound, proxyTags...)
|
|
|
|
} else {
|
|
|
|
parsedOutbound = append(parsedOutbound, o)
|
2024-03-11 23:39:58 +08:00
|
|
|
}
|
|
|
|
}
|
2024-03-18 22:32:33 +08:00
|
|
|
config.Outbounds[i].Outbounds = parsedOutbound
|
2024-03-11 23:39:58 +08:00
|
|
|
}
|
|
|
|
config.Outbounds = append(config.Outbounds, newOutbounds...)
|
2024-03-18 22:32:33 +08:00
|
|
|
//TODO: 国家策略组
|
2024-03-11 23:39:58 +08:00
|
|
|
data, err := json.Marshal(config)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return string(data), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func ConvertCProxyToSProxy(proxy string) (model.Proxy, error) {
|
|
|
|
for prefix, parseFunc := range parser.ParserMap {
|
|
|
|
if strings.HasPrefix(proxy, prefix) {
|
|
|
|
proxy, err := parseFunc(proxy)
|
|
|
|
if err != nil {
|
|
|
|
return model.Proxy{}, err
|
|
|
|
}
|
|
|
|
return proxy, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return model.Proxy{}, errors.New("Unknown proxy format")
|
|
|
|
}
|
|
|
|
|
|
|
|
func ConvertCProxyToJson(proxy string) (string, error) {
|
|
|
|
sProxy, err := ConvertCProxyToSProxy(proxy)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
data, err := json.Marshal(&sProxy)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return string(data), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func ConvertSubscriptionsToSProxy(urls []string) ([]model.Proxy, error) {
|
|
|
|
proxyList := make([]model.Proxy, 0)
|
|
|
|
for _, url := range urls {
|
2024-03-19 17:01:53 +08:00
|
|
|
data, err := util.Fetch(url, 3)
|
2024-03-11 23:39:58 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2024-03-19 17:01:53 +08:00
|
|
|
proxy, err := util.DecodeBase64(data)
|
2024-03-11 23:39:58 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
proxies := strings.Split(proxy, "\n")
|
|
|
|
for _, p := range proxies {
|
|
|
|
for prefix, parseFunc := range parser.ParserMap {
|
|
|
|
if strings.HasPrefix(p, prefix) {
|
|
|
|
proxy, err := parseFunc(p)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
proxyList = append(proxyList, proxy)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return proxyList, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func ConvertSubscriptionsToJson(urls []string) (string, error) {
|
|
|
|
proxyList, err := ConvertSubscriptionsToSProxy(urls)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
result, err := json.Marshal(proxyList)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return string(result), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func ReadTemplate(path string) (model.Config, error) {
|
|
|
|
data, err := os.ReadFile(path)
|
|
|
|
if err != nil {
|
|
|
|
return model.Config{}, err
|
|
|
|
}
|
|
|
|
var res model.Config
|
|
|
|
err = json.Unmarshal(data, &res)
|
|
|
|
if err != nil {
|
|
|
|
return model.Config{}, err
|
|
|
|
}
|
|
|
|
return res, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func DeleteProxy(proxies []model.Proxy, regex string) ([]model.Proxy, error) {
|
|
|
|
reg, err := regexp.Compile(regex)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
var newProxies []model.Proxy
|
|
|
|
for _, p := range proxies {
|
|
|
|
if !reg.MatchString(p.Tag) {
|
|
|
|
newProxies = append(newProxies, p)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return newProxies, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func RenameProxy(proxies []model.Proxy, regex string, replaceText string) ([]model.Proxy, error) {
|
|
|
|
reg, err := regexp.Compile(regex)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
for i, p := range proxies {
|
|
|
|
if reg.MatchString(p.Tag) {
|
|
|
|
proxies[i].Tag = reg.ReplaceAllString(p.Tag, replaceText)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return proxies, nil
|
|
|
|
}
|
2024-03-18 22:32:33 +08:00
|
|
|
|
|
|
|
func GetContryName(proxyName string) string {
|
|
|
|
countryMaps := []map[string]string{
|
|
|
|
model.CountryFlag,
|
|
|
|
model.CountryChineseName,
|
|
|
|
model.CountryISO,
|
|
|
|
model.CountryEnglishName,
|
|
|
|
}
|
|
|
|
for _, countryMap := range countryMaps {
|
|
|
|
for k, v := range countryMap {
|
|
|
|
if strings.Contains(proxyName, k) {
|
|
|
|
return v
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return "其他地区"
|
|
|
|
}
|