Append subscription groups to other groups

This commit is contained in:
世界 2024-11-18 13:42:43 +08:00
parent 774257d003
commit 646e6b13ac
No known key found for this signature in database
GPG Key ID: CD109927C34A63C4
2 changed files with 113 additions and 80 deletions

View File

@ -203,5 +203,5 @@ func closeMonitor(ctx context.Context) {
return return
default: default:
} }
log.Fatal("sing-box did not close!") log.Fatal("serenity did not close!")
} }

View File

@ -3,6 +3,7 @@ package template
import ( import (
"bytes" "bytes"
"regexp" "regexp"
"sort"
"text/template" "text/template"
M "github.com/sagernet/serenity/common/metadata" M "github.com/sagernet/serenity/common/metadata"
@ -129,7 +130,21 @@ func (t *Template) renderOutbounds(metadata M.Metadata, options *boxOption.Optio
subscriptionGroups = make(map[string][]boxOption.Outbound) subscriptionGroups = make(map[string][]boxOption.Outbound)
) )
for _, extraGroup := range t.groups { for _, extraGroup := range t.groups {
myFilter := func(outboundTag string) bool { if extraGroup.Target != option.ExtraGroupTargetSubscription {
continue
}
tmpl := template.New("tag")
if extraGroup.TagPerSubscription != "" {
_, err := tmpl.Parse(extraGroup.TagPerSubscription)
if err != nil {
return E.Cause(err, "parse `tag_per_subscription`: ", extraGroup.TagPerSubscription)
}
} else {
common.Must1(tmpl.Parse("{{ .tag }} ({{ .subscription_name }})"))
}
var outboundTags []string
for _, it := range subscriptions {
subscriptionTags := common.Filter(common.Map(it.Servers, outboundToString), func(outboundTag string) bool {
if len(extraGroup.filter) > 0 { if len(extraGroup.filter) > 0 {
if !common.Any(extraGroup.filter, func(it *regexp.Regexp) bool { if !common.Any(extraGroup.filter, func(it *regexp.Regexp) bool {
return it.MatchString(outboundTag) return it.MatchString(outboundTag)
@ -145,49 +160,7 @@ func (t *Template) renderOutbounds(metadata M.Metadata, options *boxOption.Optio
} }
} }
return true return true
} })
if extraGroup.Target != option.ExtraGroupTargetSubscription {
extraTags := common.Filter(common.FlatMap(subscriptions, func(it *subscription.Subscription) []string {
return common.Map(it.Servers, outboundToString)
}), myFilter)
if len(extraTags) == 0 {
continue
}
groupOutbound := boxOption.Outbound{
Tag: extraGroup.Tag,
Type: extraGroup.Type,
}
switch extraGroup.Type {
case C.TypeSelector:
selectorOptions := common.PtrValueOrDefault(extraGroup.CustomSelector)
groupOutbound.Options = &selectorOptions
selectorOptions.Outbounds = append(selectorOptions.Outbounds, extraTags...)
case C.TypeURLTest:
urltestOptions := common.PtrValueOrDefault(extraGroup.CustomURLTest)
groupOutbound.Options = &urltestOptions
urltestOptions.Outbounds = append(urltestOptions.Outbounds, extraTags...)
}
if extraGroup.Target == option.ExtraGroupTargetDefault {
defaultGroups = append(defaultGroups, groupOutbound)
} else {
globalGroups = append(globalGroups, groupOutbound)
}
} else {
tmpl := template.New("tag")
if extraGroup.TagPerSubscription != "" {
_, err := tmpl.Parse(extraGroup.TagPerSubscription)
if err != nil {
return E.Cause(err, "parse `tag_per_subscription`: ", extraGroup.TagPerSubscription)
}
} else {
common.Must1(tmpl.Parse("{{ .tag }} ({{ .subscription_name }})"))
}
var outboundTags []string
for _, it := range subscriptions {
subscriptionTags := common.Filter(common.Map(it.Servers, outboundToString), myFilter)
if len(subscriptionTags) == 0 {
continue
}
var tagPerSubscription string var tagPerSubscription string
if len(outboundTags) == 0 && len(subscriptions) == 1 { if len(outboundTags) == 0 && len(subscriptions) == 1 {
tagPerSubscription = extraGroup.Tag tagPerSubscription = extraGroup.Tag
@ -210,15 +183,76 @@ func (t *Template) renderOutbounds(metadata M.Metadata, options *boxOption.Optio
case C.TypeSelector: case C.TypeSelector:
selectorOptions := common.PtrValueOrDefault(extraGroup.CustomSelector) selectorOptions := common.PtrValueOrDefault(extraGroup.CustomSelector)
groupOutboundPerSubscription.Options = &selectorOptions groupOutboundPerSubscription.Options = &selectorOptions
selectorOptions.Outbounds = append(selectorOptions.Outbounds, subscriptionTags...) selectorOptions.Outbounds = common.Uniq(append(selectorOptions.Outbounds, subscriptionTags...))
if len(selectorOptions.Outbounds) == 0 {
continue
}
case C.TypeURLTest: case C.TypeURLTest:
urltestOptions := common.PtrValueOrDefault(extraGroup.CustomURLTest) urltestOptions := common.PtrValueOrDefault(extraGroup.CustomURLTest)
groupOutboundPerSubscription.Options = &urltestOptions groupOutboundPerSubscription.Options = &urltestOptions
urltestOptions.Outbounds = append(urltestOptions.Outbounds, subscriptionTags...) urltestOptions.Outbounds = common.Uniq(append(urltestOptions.Outbounds, subscriptionTags...))
if len(urltestOptions.Outbounds) == 0 {
continue
}
} }
subscriptionGroups[it.Name] = append(subscriptionGroups[it.Name], groupOutboundPerSubscription) subscriptionGroups[it.Name] = append(subscriptionGroups[it.Name], groupOutboundPerSubscription)
} }
} }
for _, extraGroup := range t.groups {
if extraGroup.Target == option.ExtraGroupTargetSubscription {
continue
}
extraTags := groupTags
for _, group := range subscriptionGroups {
extraTags = append(extraTags, common.Map(group, outboundToString)...)
}
sort.Strings(extraTags)
if len(extraTags) == 0 || extraGroup.filter != nil || extraGroup.exclude != nil {
extraTags = append(extraTags, common.Filter(common.FlatMap(subscriptions, func(it *subscription.Subscription) []string {
return common.Map(it.Servers, outboundToString)
}), func(outboundTag string) bool {
if len(extraGroup.filter) > 0 {
if !common.Any(extraGroup.filter, func(it *regexp.Regexp) bool {
return it.MatchString(outboundTag)
}) {
return false
}
}
if len(extraGroup.exclude) > 0 {
if common.Any(extraGroup.exclude, func(it *regexp.Regexp) bool {
return it.MatchString(outboundTag)
}) {
return false
}
}
return true
})...)
}
groupOutbound := boxOption.Outbound{
Tag: extraGroup.Tag,
Type: extraGroup.Type,
}
switch extraGroup.Type {
case C.TypeSelector:
selectorOptions := common.PtrValueOrDefault(extraGroup.CustomSelector)
groupOutbound.Options = &selectorOptions
selectorOptions.Outbounds = common.Uniq(append(selectorOptions.Outbounds, extraTags...))
if len(selectorOptions.Outbounds) == 0 {
continue
}
case C.TypeURLTest:
urltestOptions := common.PtrValueOrDefault(extraGroup.CustomURLTest)
groupOutbound.Options = &urltestOptions
urltestOptions.Outbounds = common.Uniq(append(urltestOptions.Outbounds, extraTags...))
if len(urltestOptions.Outbounds) == 0 {
continue
}
}
if extraGroup.Target == option.ExtraGroupTargetDefault {
defaultGroups = append(defaultGroups, groupOutbound)
} else {
globalGroups = append(globalGroups, groupOutbound)
}
} }
options.Outbounds = append(options.Outbounds, allGroups...) options.Outbounds = append(options.Outbounds, allGroups...)
@ -238,7 +272,6 @@ func (t *Template) renderOutbounds(metadata M.Metadata, options *boxOption.Optio
} }
options.Outbounds = groupJoin(options.Outbounds, defaultTag, false, groupTags...) options.Outbounds = groupJoin(options.Outbounds, defaultTag, false, groupTags...)
options.Outbounds = groupJoin(options.Outbounds, defaultTag, false, globalOutboundTags...) options.Outbounds = groupJoin(options.Outbounds, defaultTag, false, globalOutboundTags...)
options.Outbounds = append(options.Outbounds, allGroupOutbounds...) options.Outbounds = append(options.Outbounds, allGroupOutbounds...)
return nil return nil
} }