package option import ( "bytes" "context" E "github.com/sagernet/sing/common/exceptions" "github.com/sagernet/sing/common/json" ) type _Options struct { RawMessage json.RawMessage `json:"-"` Schema string `json:"$schema,omitempty"` Log *LogOptions `json:"log,omitempty"` DNS *DNSOptions `json:"dns,omitempty"` NTP *NTPOptions `json:"ntp,omitempty"` Certificate *CertificateOptions `json:"certificate,omitempty"` Endpoints []Endpoint `json:"endpoints,omitempty"` Inbounds []Inbound `json:"inbounds,omitempty"` Outbounds []Outbound `json:"outbounds,omitempty"` Route *RouteOptions `json:"route,omitempty"` Experimental *ExperimentalOptions `json:"experimental,omitempty"` } type Options _Options func (o *Options) UnmarshalJSONContext(ctx context.Context, content []byte) error { decoder := json.NewDecoderContext(ctx, bytes.NewReader(content)) decoder.DisallowUnknownFields() err := decoder.Decode((*_Options)(o)) if err != nil { return err } o.RawMessage = content return checkOptions(o) } type LogOptions struct { Disabled bool `json:"disabled,omitempty"` Level string `json:"level,omitempty"` Output string `json:"output,omitempty"` Timestamp bool `json:"timestamp,omitempty"` DisableColor bool `json:"-"` } 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 }