1
0
mirror of https://github.com/nitezs/sub2clash.git synced 2024-12-24 11:14:41 -05:00
sub2clash/validator/sub.go

158 lines
4.8 KiB
Go
Raw Normal View History

2023-09-12 06:40:24 -04:00
package validator
2023-09-13 01:47:22 -04:00
import (
2023-09-17 03:52:37 -04:00
"crypto/sha256"
"encoding/hex"
2023-09-13 01:47:22 -04:00
"errors"
"net/url"
"regexp"
"strings"
"github.com/gin-gonic/gin"
2023-09-13 01:47:22 -04:00
)
2023-09-17 03:52:37 -04:00
type SubValidator struct {
Sub string `form:"sub" binding:""`
Subs []string `form:"-" binding:""`
Proxy string `form:"proxy" binding:""`
Proxies []string `form:"-" binding:""`
Refresh bool `form:"refresh,default=false" binding:""`
Template string `form:"template" binding:""`
RuleProvider string `form:"ruleProvider" binding:""`
RuleProviders []RuleProviderStruct `form:"-" binding:""`
Rule string `form:"rule" binding:""`
Rules []RuleStruct `form:"-" binding:""`
AutoTest bool `form:"autoTest,default=false" binding:""`
Lazy bool `form:"lazy,default=false" binding:""`
Sort string `form:"sort" binding:""`
Remove string `form:"remove" binding:""`
Replace string `form:"replace" binding:""`
ReplaceKeys []string `form:"-" binding:""`
ReplaceTo []string `form:"-" binding:""`
NodeListMode bool `form:"nodeList,default=false" binding:""`
IgnoreCountryGrooup bool `form:"ignoreCountryGroup,default=false" binding:""`
2023-09-13 01:47:22 -04:00
}
type RuleProviderStruct struct {
Behavior string
Url string
Group string
Prepend bool
Name string
2023-09-13 01:47:22 -04:00
}
type RuleStruct struct {
Rule string
Prepend bool
}
2023-09-17 03:52:37 -04:00
func ParseQuery(c *gin.Context) (SubValidator, error) {
var query SubValidator
2023-09-13 01:47:22 -04:00
if err := c.ShouldBind(&query); err != nil {
2023-09-17 03:52:37 -04:00
return SubValidator{}, errors.New("参数错误: " + err.Error())
2023-09-13 01:47:22 -04:00
}
if query.Sub == "" && query.Proxy == "" {
2023-09-17 03:52:37 -04:00
return SubValidator{}, errors.New("参数错误: sub 和 proxy 不能同时为空")
2023-09-13 01:47:22 -04:00
}
if query.Sub != "" {
query.Subs = strings.Split(query.Sub, ",")
for i := range query.Subs {
2023-09-17 03:52:37 -04:00
if !strings.HasPrefix(query.Subs[i], "http") {
return SubValidator{}, errors.New("参数错误: sub 格式错误")
}
2023-09-13 01:47:22 -04:00
if _, err := url.ParseRequestURI(query.Subs[i]); err != nil {
2023-09-17 03:52:37 -04:00
return SubValidator{}, errors.New("参数错误: " + err.Error())
2023-09-13 01:47:22 -04:00
}
}
} else {
query.Subs = nil
}
if query.Proxy != "" {
query.Proxies = strings.Split(query.Proxy, ",")
} else {
query.Proxies = nil
}
if query.Template != "" {
if strings.HasPrefix(query.Template, "http") {
uri, err := url.ParseRequestURI(query.Template)
if err != nil {
2023-09-17 03:52:37 -04:00
return SubValidator{}, err
2023-09-14 20:26:48 -04:00
}
query.Template = uri.String()
2023-09-13 01:47:22 -04:00
}
}
if query.RuleProvider != "" {
reg := regexp.MustCompile(`\[(.*?)\]`)
ruleProviders := reg.FindAllStringSubmatch(query.RuleProvider, -1)
for i := range ruleProviders {
length := len(ruleProviders)
parts := strings.Split(ruleProviders[length-i-1][1], ",")
if len(parts) < 4 {
2023-09-17 03:52:37 -04:00
return SubValidator{}, errors.New("参数错误: ruleProvider 格式错误")
2023-09-13 01:47:22 -04:00
}
u := parts[1]
uri, err := url.ParseRequestURI(u)
if err != nil {
2023-09-17 03:52:37 -04:00
return SubValidator{}, errors.New("参数错误: " + err.Error())
2023-09-13 01:47:22 -04:00
}
2023-09-14 20:26:48 -04:00
u = uri.String()
if len(parts) == 4 {
2023-09-17 03:52:37 -04:00
hash := sha256.Sum224([]byte(u))
parts = append(parts, hex.EncodeToString(hash[:]))
}
2023-09-13 01:47:22 -04:00
query.RuleProviders = append(
query.RuleProviders, RuleProviderStruct{
Behavior: parts[0],
Url: u,
Group: parts[2],
Prepend: parts[3] == "true",
Name: parts[4],
2023-09-13 01:47:22 -04:00
},
)
}
// 校验 Rule-Provider 是否有重名
names := make(map[string]bool)
for _, ruleProvider := range query.RuleProviders {
if _, ok := names[ruleProvider.Name]; ok {
2023-09-17 03:52:37 -04:00
return SubValidator{}, errors.New("参数错误: Rule-Provider 名称重复")
}
names[ruleProvider.Name] = true
}
2023-09-13 01:47:22 -04:00
} else {
query.RuleProviders = nil
}
if query.Rule != "" {
reg := regexp.MustCompile(`\[(.*?)\]`)
rules := reg.FindAllStringSubmatch(query.Rule, -1)
for i := range rules {
length := len(rules)
r := rules[length-1-i][1]
strings.LastIndex(r, ",")
parts := [2]string{}
parts[0] = r[:strings.LastIndex(r, ",")]
parts[1] = r[strings.LastIndex(r, ",")+1:]
query.Rules = append(
query.Rules, RuleStruct{
Rule: parts[0],
Prepend: parts[1] == "true",
},
)
}
} else {
query.Rules = nil
}
if strings.TrimSpace(query.Replace) != "" {
2023-09-22 12:58:57 -04:00
reg := regexp.MustCompile(`\[<(.*?)>,<(.*?)>\]`)
replaces := reg.FindAllStringSubmatch(query.Replace, -1)
for i := range replaces {
length := len(replaces[i])
if length != 3 {
return SubValidator{}, errors.New("参数错误: replace 格式错误")
}
query.ReplaceKeys = append(query.ReplaceKeys, replaces[i][1])
query.ReplaceTo = append(query.ReplaceTo, replaces[i][2])
}
}
2023-09-13 01:47:22 -04:00
return query, nil
2023-09-12 06:40:24 -04:00
}