From 47fc3ebda4b370bd7ff9fdfc6b461bf5bb6ffbd9 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E4=B8=96=E7=95=8C?= Date: Sat, 29 Mar 2025 19:51:21 +0800 Subject: [PATCH] Add duplicate tag check --- option/options.go | 52 ++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 51 insertions(+), 1 deletion(-) diff --git a/option/options.go b/option/options.go index 94c97719..85ff81eb 100644 --- a/option/options.go +++ b/option/options.go @@ -4,6 +4,7 @@ import ( "bytes" "context" + E "github.com/sagernet/sing/common/exceptions" "github.com/sagernet/sing/common/json" ) @@ -30,7 +31,7 @@ func (o *Options) UnmarshalJSONContext(ctx context.Context, content []byte) erro return err } o.RawMessage = content - return nil + return checkOptions(o) } type LogOptions struct { @@ -42,3 +43,52 @@ type LogOptions struct { } type StubOptions struct{} + +func checkOptions(options *Options) error { + err := checkInbounds(options.Inbounds) + if err != nil { + return err + } + err = checkOutbounds(options.Outbounds, options.Endpoints) + if err != nil { + return err + } + return nil +} + +func checkInbounds(inbounds []Inbound) error { + seen := make(map[string]bool) + for _, inbound := range inbounds { + if inbound.Tag == "" { + continue + } + if seen[inbound.Tag] { + return E.New("duplicate inbound tag: ", inbound.Tag) + } + seen[inbound.Tag] = true + } + return nil +} + +func checkOutbounds(outbounds []Outbound, endpoints []Endpoint) error { + seen := make(map[string]bool) + for _, outbound := range outbounds { + if outbound.Tag == "" { + continue + } + if seen[outbound.Tag] { + return E.New("duplicate outbound/endpoint tag: ", outbound.Tag) + } + seen[outbound.Tag] = true + } + for _, endpoint := range endpoints { + if endpoint.Tag == "" { + continue + } + if seen[endpoint.Tag] { + return E.New("duplicate outbound/endpoint tag: ", endpoint.Tag) + } + seen[endpoint.Tag] = true + } + return nil +}