2024-03-13 13:30:45 +08:00
|
|
|
|
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"
|
2023-09-15 00:13:45 +08:00
|
|
|
|
"net/url"
|
2023-09-13 13:47:22 +08:00
|
|
|
|
"regexp"
|
2023-09-21 09:08:02 +08:00
|
|
|
|
"sort"
|
2023-09-22 23:43:26 +08:00
|
|
|
|
"strconv"
|
2023-09-12 18:40:24 +08:00
|
|
|
|
"strings"
|
2023-09-21 17:37:37 +08:00
|
|
|
|
"sub2clash/logger"
|
2023-09-13 00:46:17 +08:00
|
|
|
|
"sub2clash/model"
|
|
|
|
|
"sub2clash/parser"
|
|
|
|
|
"sub2clash/utils"
|
2023-09-13 13:47:22 +08:00
|
|
|
|
"sub2clash/validator"
|
2023-11-03 02:35:30 +08:00
|
|
|
|
|
|
|
|
|
"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,
|
|
|
|
|
) {
|
2023-09-12 18:40:24 +08:00
|
|
|
|
// 定义变量
|
2023-09-17 13:21:01 +08:00
|
|
|
|
var temp = &model.Subscription{}
|
|
|
|
|
var sub = &model.Subscription{}
|
2023-09-13 13:47:22 +08:00
|
|
|
|
var err error
|
|
|
|
|
var templateBytes []byte
|
2023-09-12 18:40:24 +08:00
|
|
|
|
// 加载模板
|
2023-09-15 08:26:48 +08:00
|
|
|
|
if query.Template != "" {
|
|
|
|
|
template = query.Template
|
|
|
|
|
}
|
2024-02-15 15:35:07 +08:00
|
|
|
|
if strings.HasPrefix(template, "http") {
|
|
|
|
|
templateBytes, err = utils.LoadSubscription(template, query.Refresh)
|
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 {
|
2024-02-15 15:35:07 +08:00
|
|
|
|
unescape, err := url.QueryUnescape(template)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, errors.New("加载模板失败: " + err.Error())
|
|
|
|
|
}
|
|
|
|
|
templateBytes, err = utils.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
|
|
|
|
}
|
|
|
|
|
// 解析模板
|
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())
|
|
|
|
|
}
|
2023-09-21 09:08:02 +08:00
|
|
|
|
var proxyList []model.Proxy
|
2023-09-12 18:40:24 +08:00
|
|
|
|
// 加载订阅
|
2023-09-13 13:47:22 +08:00
|
|
|
|
for i := range query.Subs {
|
|
|
|
|
data, err := utils.LoadSubscription(query.Subs[i], query.Refresh)
|
2023-09-25 23:58:13 +08:00
|
|
|
|
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())
|
|
|
|
|
}
|
2023-09-13 00:46:17 +08:00
|
|
|
|
// 解析订阅
|
|
|
|
|
err = yaml.Unmarshal(data, &sub)
|
2024-03-10 13:26:35 +08:00
|
|
|
|
var newProxies []model.Proxy
|
2023-09-13 00:46:17 +08:00
|
|
|
|
if err != nil {
|
2023-10-31 15:14:29 +08:00
|
|
|
|
reg, _ := regexp.Compile("(ssr|ss|vmess|trojan|vless|hysteria)://")
|
2023-09-13 13:47:22 +08:00
|
|
|
|
if reg.Match(data) {
|
2023-09-21 09:08:02 +08:00
|
|
|
|
p := utils.ParseProxy(strings.Split(string(data), "\n")...)
|
2023-09-25 23:58:13 +08:00
|
|
|
|
newProxies = p
|
2023-09-13 13:47:22 +08:00
|
|
|
|
} else {
|
|
|
|
|
// 如果无法直接解析,尝试Base64解码
|
|
|
|
|
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())
|
|
|
|
|
}
|
2023-09-21 09:08:02 +08:00
|
|
|
|
p := utils.ParseProxy(strings.Split(base64, "\n")...)
|
2023-09-25 23:58:13 +08:00
|
|
|
|
newProxies = p
|
2023-09-13 00:46:17 +08:00
|
|
|
|
}
|
|
|
|
|
} else {
|
2023-09-25 23:58:13 +08:00
|
|
|
|
newProxies = sub.Proxies
|
|
|
|
|
}
|
|
|
|
|
if subName != "" {
|
|
|
|
|
for i := range newProxies {
|
|
|
|
|
newProxies[i].SubName = subName
|
|
|
|
|
}
|
2023-09-13 00:46:17 +08:00
|
|
|
|
}
|
2023-09-25 23:58:13 +08:00
|
|
|
|
proxyList = append(proxyList, newProxies...)
|
2023-09-13 13:47:22 +08:00
|
|
|
|
}
|
2023-09-23 00:58:57 +08:00
|
|
|
|
// 添加自定义节点
|
|
|
|
|
if len(query.Proxies) != 0 {
|
|
|
|
|
proxyList = append(proxyList, utils.ParseProxy(query.Proxies...)...)
|
|
|
|
|
}
|
2023-09-25 23:58:13 +08:00
|
|
|
|
// 给节点添加订阅名称
|
|
|
|
|
for i := range proxyList {
|
|
|
|
|
if proxyList[i].SubName != "" {
|
|
|
|
|
proxyList[i].Name = strings.TrimSpace(proxyList[i].SubName) + " " + strings.TrimSpace(proxyList[i].Name)
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-09-23 01:31:04 +08:00
|
|
|
|
// 去掉配置相同的节点
|
2023-09-22 23:43:26 +08:00
|
|
|
|
proxies := make(map[string]*model.Proxy)
|
2023-09-23 01:11:58 +08:00
|
|
|
|
newProxies := make([]model.Proxy, 0, len(proxyList))
|
2023-09-22 23:43:26 +08:00
|
|
|
|
for i := range proxyList {
|
2024-04-17 21:52:03 +08:00
|
|
|
|
key := proxyList[i].Server + strconv.Itoa(proxyList[i].Port) + proxyList[i].Type + proxyList[i].UUID + proxyList[i].Password
|
2023-09-23 01:11:58 +08:00
|
|
|
|
if _, exist := proxies[key]; !exist {
|
|
|
|
|
proxies[key] = &proxyList[i]
|
|
|
|
|
newProxies = append(newProxies, proxyList[i])
|
2023-09-22 23:43:26 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-09-23 01:11:58 +08:00
|
|
|
|
proxyList = newProxies
|
2023-09-23 00:58:57 +08:00
|
|
|
|
// 删除节点
|
2023-09-22 23:43:26 +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())
|
|
|
|
|
}
|
|
|
|
|
// 删除匹配到的节点
|
2023-09-22 23:43:26 +08:00
|
|
|
|
if removeReg.MatchString(proxyList[i].Name) {
|
|
|
|
|
continue // 如果匹配到要删除的元素,跳过该元素,不添加到新切片中
|
|
|
|
|
}
|
|
|
|
|
newProxyList = append(newProxyList, proxyList[i]) // 将要保留的元素添加到新切片中
|
|
|
|
|
}
|
|
|
|
|
proxyList = newProxyList
|
|
|
|
|
}
|
2023-09-23 00:58:57 +08:00
|
|
|
|
// 重命名
|
|
|
|
|
if len(query.ReplaceKeys) != 0 {
|
|
|
|
|
// 创建重命名正则表达式
|
|
|
|
|
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 {
|
|
|
|
|
// 重命名匹配到的节点
|
|
|
|
|
for j, v := range replaceRegs {
|
|
|
|
|
if v.MatchString(proxyList[i].Name) {
|
|
|
|
|
proxyList[i].Name = v.ReplaceAllString(
|
|
|
|
|
proxyList[i].Name, query.ReplaceTo[j],
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-09-23 01:31:04 +08:00
|
|
|
|
// 重名检测
|
|
|
|
|
names := make(map[string]int)
|
|
|
|
|
for i := range proxyList {
|
|
|
|
|
if _, exist := names[proxyList[i].Name]; exist {
|
2024-03-09 15:58:16 +08:00
|
|
|
|
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])
|
2024-03-09 15:58:16 +08:00
|
|
|
|
} else {
|
|
|
|
|
names[proxyList[i].Name] = 0
|
2023-09-23 01:31:04 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-09-23 00:58:57 +08:00
|
|
|
|
// trim
|
|
|
|
|
for i := range proxyList {
|
|
|
|
|
proxyList[i].Name = strings.TrimSpace(proxyList[i].Name)
|
|
|
|
|
}
|
2023-09-21 09:08:02 +08:00
|
|
|
|
// 将新增节点都添加到临时变量 t 中,防止策略组排序错乱
|
|
|
|
|
var t = &model.Subscription{}
|
|
|
|
|
utils.AddProxy(t, query.AutoTest, query.Lazy, clashType, proxyList...)
|
|
|
|
|
// 排序策略组
|
|
|
|
|
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-03-10 13:26:35 +08:00
|
|
|
|
MergeSubAndTemplate(temp, t, query.IgnoreCountryGrooup)
|
2023-09-13 13:47:22 +08:00
|
|
|
|
// 处理自定义规则
|
|
|
|
|
for _, v := range query.Rules {
|
|
|
|
|
if v.Prepend {
|
|
|
|
|
utils.PrependRules(temp, v.Rule)
|
|
|
|
|
} else {
|
|
|
|
|
utils.AppendRules(temp, v.Rule)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// 处理自定义 ruleProvider
|
|
|
|
|
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 {
|
|
|
|
|
utils.PrependRuleProvider(
|
2023-09-15 00:13:45 +08:00
|
|
|
|
temp, v.Name, v.Group, provider,
|
2023-09-13 13:47:22 +08:00
|
|
|
|
)
|
|
|
|
|
} else {
|
|
|
|
|
utils.AppenddRuleProvider(
|
2023-09-15 00:13:45 +08:00
|
|
|
|
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
|
|
|
|
|
}
|
2023-09-15 00:13:45 +08:00
|
|
|
|
|
2024-03-10 13:26:35 +08:00
|
|
|
|
func MergeSubAndTemplate(temp *model.Subscription, sub *model.Subscription, igcg bool) {
|
2023-09-15 00:13:45 +08:00
|
|
|
|
// 只合并节点、策略组
|
|
|
|
|
// 统计所有国家策略组名称
|
2023-09-15 00:31:57 +08:00
|
|
|
|
var countryGroupNames []string
|
2023-09-15 00:13:45 +08:00
|
|
|
|
for _, proxyGroup := range sub.ProxyGroups {
|
|
|
|
|
if proxyGroup.IsCountryGrop {
|
2023-09-15 00:31:57 +08:00
|
|
|
|
countryGroupNames = append(
|
|
|
|
|
countryGroupNames, proxyGroup.Name,
|
2023-09-15 00:13:45 +08:00
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-09-22 23:43:26 +08:00
|
|
|
|
var proxyNames []string
|
|
|
|
|
for _, proxy := range sub.Proxies {
|
|
|
|
|
proxyNames = append(proxyNames, proxy.Name)
|
|
|
|
|
}
|
2023-09-15 00:13:45 +08:00
|
|
|
|
// 将订阅中的节点添加到模板中
|
|
|
|
|
temp.Proxies = append(temp.Proxies, sub.Proxies...)
|
|
|
|
|
// 将订阅中的策略组添加到模板中
|
|
|
|
|
for i := range temp.ProxyGroups {
|
2023-09-22 23:43:26 +08:00
|
|
|
|
if temp.ProxyGroups[i].IsCountryGrop {
|
|
|
|
|
continue
|
2023-09-15 08:51:22 +08:00
|
|
|
|
}
|
2023-09-29 15:58:38 +08:00
|
|
|
|
newProxies := make([]string, 0)
|
2023-09-25 23:58:13 +08:00
|
|
|
|
countryGroupMap := make(map[string]model.ProxyGroup)
|
|
|
|
|
for _, v := range sub.ProxyGroups {
|
|
|
|
|
if v.IsCountryGrop {
|
|
|
|
|
countryGroupMap[v.Name] = v
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-09-22 23:43:26 +08:00
|
|
|
|
for j := range temp.ProxyGroups[i].Proxies {
|
2023-09-25 23:58:13 +08:00
|
|
|
|
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...)
|
|
|
|
|
}
|
2023-09-25 23:58:13 +08:00
|
|
|
|
default:
|
2024-04-17 21:52:03 +08:00
|
|
|
|
if !igcg {
|
|
|
|
|
if len(key) == 2 {
|
|
|
|
|
newProxies = append(
|
|
|
|
|
newProxies, countryGroupMap[utils.GetContryName(key)].Proxies...,
|
|
|
|
|
)
|
|
|
|
|
}
|
2023-09-25 23:58:13 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-09-22 23:43:26 +08:00
|
|
|
|
} else {
|
|
|
|
|
newProxies = append(newProxies, temp.ProxyGroups[i].Proxies[j])
|
|
|
|
|
}
|
2023-09-15 08:51:22 +08:00
|
|
|
|
}
|
2023-09-22 23:43:26 +08:00
|
|
|
|
temp.ProxyGroups[i].Proxies = newProxies
|
2023-09-15 00:13:45 +08:00
|
|
|
|
}
|
2024-03-10 13:26:35 +08:00
|
|
|
|
if !igcg {
|
|
|
|
|
temp.ProxyGroups = append(temp.ProxyGroups, sub.ProxyGroups...)
|
|
|
|
|
}
|
2023-09-15 00:13:45 +08:00
|
|
|
|
}
|