sub2clash/api/handler/default.go

294 lines
7.8 KiB
Go
Raw Permalink Normal View History

package handler
2023-09-12 18:40:24 +08:00
import (
2023-09-17 15:52:37 +08:00
"crypto/sha256"
2023-09-13 13:47:22 +08:00
"encoding/hex"
2023-09-12 18:40:24 +08:00
"errors"
"net/url"
2023-09-13 13:47:22 +08:00
"regexp"
"sort"
"strconv"
2023-09-12 18:40:24 +08:00
"strings"
"github.com/nitezs/sub2clash/common"
"github.com/nitezs/sub2clash/logger"
"github.com/nitezs/sub2clash/model"
"github.com/nitezs/sub2clash/parser"
"github.com/nitezs/sub2clash/validator"
"go.uber.org/zap"
"gopkg.in/yaml.v3"
2023-09-12 18:40:24 +08:00
)
2023-09-17 15:52:37 +08:00
func BuildSub(clashType model.ClashType, query validator.SubValidator, template string) (
2023-09-13 00:46:17 +08:00
*model.Subscription, error,
) {
2024-08-11 23:55:47 +08:00
var temp = &model.Subscription{}
var sub = &model.Subscription{}
2023-09-13 13:47:22 +08:00
var err error
var templateBytes []byte
2024-08-11 23:55:47 +08:00
2023-09-15 08:26:48 +08:00
if query.Template != "" {
template = query.Template
}
if strings.HasPrefix(template, "http") {
templateBytes, err = common.LoadSubscription(template, query.Refresh, query.UserAgent)
2023-09-13 13:47:22 +08:00
if err != nil {
2023-09-21 17:37:37 +08:00
logger.Logger.Debug(
"load template failed", zap.String("template", template), zap.Error(err),
)
2023-09-13 13:47:22 +08:00
return nil, errors.New("加载模板失败: " + err.Error())
}
} else {
unescape, err := url.QueryUnescape(template)
if err != nil {
return nil, errors.New("加载模板失败: " + err.Error())
}
templateBytes, err = common.LoadTemplate(unescape)
2023-09-13 13:47:22 +08:00
if err != nil {
2023-09-21 17:37:37 +08:00
logger.Logger.Debug(
"load template failed", zap.String("template", template), zap.Error(err),
)
2023-09-13 13:47:22 +08:00
return nil, errors.New("加载模板失败: " + err.Error())
}
2023-09-12 18:40:24 +08:00
}
2024-08-11 23:55:47 +08:00
2023-09-13 13:47:22 +08:00
err = yaml.Unmarshal(templateBytes, &temp)
2023-09-12 18:40:24 +08:00
if err != nil {
2023-09-21 17:37:37 +08:00
logger.Logger.Debug("parse template failed", zap.Error(err))
2023-09-12 18:40:24 +08:00
return nil, errors.New("解析模板失败: " + err.Error())
}
var proxyList []model.Proxy
2024-08-11 23:55:47 +08:00
2023-09-13 13:47:22 +08:00
for i := range query.Subs {
data, err := common.LoadSubscription(query.Subs[i], query.Refresh, query.UserAgent)
subName := ""
if strings.Contains(query.Subs[i], "#") {
subName = query.Subs[i][strings.LastIndex(query.Subs[i], "#")+1:]
}
2023-09-12 18:40:24 +08:00
if err != nil {
2023-09-21 17:37:37 +08:00
logger.Logger.Debug(
"load subscription failed", zap.String("url", query.Subs[i]), zap.Error(err),
)
2023-09-12 18:40:24 +08:00
return nil, errors.New("加载订阅失败: " + err.Error())
}
2024-08-11 23:55:47 +08:00
2023-09-13 00:46:17 +08:00
err = yaml.Unmarshal(data, &sub)
var newProxies []model.Proxy
2023-09-13 00:46:17 +08:00
if err != nil {
2024-04-23 14:39:16 +08:00
reg, _ := regexp.Compile("(ssr|ss|vmess|trojan|vless|hysteria|hy2|hysteria2)://")
2023-09-13 13:47:22 +08:00
if reg.Match(data) {
p := common.ParseProxy(strings.Split(string(data), "\n")...)
newProxies = p
2023-09-13 13:47:22 +08:00
} else {
2024-08-11 23:55:47 +08:00
2023-09-13 13:47:22 +08:00
base64, err := parser.DecodeBase64(string(data))
if err != nil {
2023-09-21 17:37:37 +08:00
logger.Logger.Debug(
"parse subscription failed", zap.String("url", query.Subs[i]),
zap.String("data", string(data)),
zap.Error(err),
)
2023-09-13 13:47:22 +08:00
return nil, errors.New("加载订阅失败: " + err.Error())
}
p := common.ParseProxy(strings.Split(base64, "\n")...)
newProxies = p
2023-09-13 00:46:17 +08:00
}
} else {
newProxies = sub.Proxies
}
if subName != "" {
for i := range newProxies {
newProxies[i].SubName = subName
}
2023-09-13 00:46:17 +08:00
}
proxyList = append(proxyList, newProxies...)
2023-09-13 13:47:22 +08:00
}
2024-08-11 23:55:47 +08:00
2023-09-23 00:58:57 +08:00
if len(query.Proxies) != 0 {
proxyList = append(proxyList, common.ParseProxy(query.Proxies...)...)
2023-09-23 00:58:57 +08:00
}
2024-08-11 23:55:47 +08:00
for i := range proxyList {
if proxyList[i].SubName != "" {
proxyList[i].Name = strings.TrimSpace(proxyList[i].SubName) + " " + strings.TrimSpace(proxyList[i].Name)
}
}
2024-08-11 23:55:47 +08:00
proxies := make(map[string]*model.Proxy)
2023-09-23 01:11:58 +08:00
newProxies := make([]model.Proxy, 0, len(proxyList))
for i := range proxyList {
2024-10-08 10:05:13 +08:00
key := proxyList[i].Server + strconv.Itoa(proxyList[i].Port) + proxyList[i].Type + proxyList[i].UUID + proxyList[i].Password
if proxyList[i].Network == "ws" {
key += proxyList[i].WSOpts.Path + proxyList[i].WSOpts.Headers["Host"]
}
2023-09-23 01:11:58 +08:00
if _, exist := proxies[key]; !exist {
proxies[key] = &proxyList[i]
newProxies = append(newProxies, proxyList[i])
}
}
2023-09-23 01:11:58 +08:00
proxyList = newProxies
2024-08-11 23:55:47 +08:00
if strings.TrimSpace(query.Remove) != "" {
newProxyList := make([]model.Proxy, 0, len(proxyList))
for i := range proxyList {
2023-09-23 00:58:57 +08:00
removeReg, err := regexp.Compile(query.Remove)
if err != nil {
logger.Logger.Debug("remove regexp compile failed", zap.Error(err))
return nil, errors.New("remove 参数非法: " + err.Error())
}
2024-08-11 23:55:47 +08:00
if removeReg.MatchString(proxyList[i].Name) {
2024-08-11 23:55:47 +08:00
continue
}
2024-08-11 23:55:47 +08:00
newProxyList = append(newProxyList, proxyList[i])
}
proxyList = newProxyList
}
2024-08-11 23:55:47 +08:00
2023-09-23 00:58:57 +08:00
if len(query.ReplaceKeys) != 0 {
2024-08-11 23:55:47 +08:00
2023-09-23 00:58:57 +08:00
replaceRegs := make([]*regexp.Regexp, 0, len(query.ReplaceKeys))
for _, v := range query.ReplaceKeys {
replaceReg, err := regexp.Compile(v)
if err != nil {
logger.Logger.Debug("replace regexp compile failed", zap.Error(err))
return nil, errors.New("replace 参数非法: " + err.Error())
}
replaceRegs = append(replaceRegs, replaceReg)
}
for i := range proxyList {
2024-08-11 23:55:47 +08:00
2023-09-23 00:58:57 +08:00
for j, v := range replaceRegs {
if v.MatchString(proxyList[i].Name) {
proxyList[i].Name = v.ReplaceAllString(
proxyList[i].Name, query.ReplaceTo[j],
)
}
}
}
}
2024-08-11 23:55:47 +08:00
2023-09-23 01:31:04 +08:00
names := make(map[string]int)
for i := range proxyList {
if _, exist := names[proxyList[i].Name]; exist {
names[proxyList[i].Name] = names[proxyList[i].Name] + 1
2023-09-23 01:31:04 +08:00
proxyList[i].Name = proxyList[i].Name + " " + strconv.Itoa(names[proxyList[i].Name])
} else {
names[proxyList[i].Name] = 0
2023-09-23 01:31:04 +08:00
}
}
2024-08-11 23:55:47 +08:00
2023-09-23 00:58:57 +08:00
for i := range proxyList {
proxyList[i].Name = strings.TrimSpace(proxyList[i].Name)
}
2024-08-11 23:55:47 +08:00
var t = &model.Subscription{}
common.AddProxy(t, query.AutoTest, query.Lazy, clashType, proxyList...)
2024-08-11 23:55:47 +08:00
switch query.Sort {
case "sizeasc":
sort.Sort(model.ProxyGroupsSortBySize(t.ProxyGroups))
case "sizedesc":
sort.Sort(sort.Reverse(model.ProxyGroupsSortBySize(t.ProxyGroups)))
case "nameasc":
sort.Sort(model.ProxyGroupsSortByName(t.ProxyGroups))
case "namedesc":
sort.Sort(sort.Reverse(model.ProxyGroupsSortByName(t.ProxyGroups)))
default:
sort.Sort(model.ProxyGroupsSortByName(t.ProxyGroups))
}
2024-08-11 23:55:47 +08:00
MergeSubAndTemplate(temp, t, query.IgnoreCountryGrooup)
2024-08-11 23:55:47 +08:00
2023-09-13 13:47:22 +08:00
for _, v := range query.Rules {
if v.Prepend {
common.PrependRules(temp, v.Rule)
2023-09-13 13:47:22 +08:00
} else {
common.AppendRules(temp, v.Rule)
2023-09-13 13:47:22 +08:00
}
}
2024-08-11 23:55:47 +08:00
2023-09-13 13:47:22 +08:00
for _, v := range query.RuleProviders {
2023-09-17 15:52:37 +08:00
hash := sha256.Sum224([]byte(v.Url))
2023-09-13 13:47:22 +08:00
name := hex.EncodeToString(hash[:])
provider := model.RuleProvider{
Type: "http",
Behavior: v.Behavior,
Url: v.Url,
Path: "./" + name + ".yaml",
Interval: 3600,
}
if v.Prepend {
common.PrependRuleProvider(
temp, v.Name, v.Group, provider,
2023-09-13 13:47:22 +08:00
)
} else {
common.AppenddRuleProvider(
temp, v.Name, v.Group, provider,
2023-09-13 13:47:22 +08:00
)
}
2023-09-12 18:40:24 +08:00
}
return temp, nil
}
func MergeSubAndTemplate(temp *model.Subscription, sub *model.Subscription, igcg bool) {
2024-08-11 23:55:47 +08:00
2023-09-15 00:31:57 +08:00
var countryGroupNames []string
for _, proxyGroup := range sub.ProxyGroups {
if proxyGroup.IsCountryGrop {
2023-09-15 00:31:57 +08:00
countryGroupNames = append(
countryGroupNames, proxyGroup.Name,
)
}
}
var proxyNames []string
for _, proxy := range sub.Proxies {
proxyNames = append(proxyNames, proxy.Name)
}
2024-08-11 23:55:47 +08:00
temp.Proxies = append(temp.Proxies, sub.Proxies...)
2024-08-11 23:55:47 +08:00
for i := range temp.ProxyGroups {
if temp.ProxyGroups[i].IsCountryGrop {
continue
2023-09-15 08:51:22 +08:00
}
newProxies := make([]string, 0)
countryGroupMap := make(map[string]model.ProxyGroup)
for _, v := range sub.ProxyGroups {
if v.IsCountryGrop {
countryGroupMap[v.Name] = v
}
}
for j := range temp.ProxyGroups[i].Proxies {
reg := regexp.MustCompile("<(.*?)>")
if reg.Match([]byte(temp.ProxyGroups[i].Proxies[j])) {
key := reg.FindStringSubmatch(temp.ProxyGroups[i].Proxies[j])[1]
switch key {
case "all":
newProxies = append(newProxies, proxyNames...)
case "countries":
2024-04-17 21:52:03 +08:00
if !igcg {
newProxies = append(newProxies, countryGroupNames...)
}
default:
2024-04-17 21:52:03 +08:00
if !igcg {
if len(key) == 2 {
newProxies = append(
newProxies, countryGroupMap[common.GetContryName(key)].Proxies...,
2024-04-17 21:52:03 +08:00
)
}
}
}
} else {
newProxies = append(newProxies, temp.ProxyGroups[i].Proxies[j])
}
2023-09-15 08:51:22 +08:00
}
temp.ProxyGroups[i].Proxies = newProxies
}
if !igcg {
temp.ProxyGroups = append(temp.ProxyGroups, sub.ProxyGroups...)
}
}