dae/config/config.go

159 lines
5.1 KiB
Go
Raw Normal View History

2023-01-28 00:50:21 +07:00
/*
* SPDX-License-Identifier: AGPL-3.0-only
2023-03-14 14:01:55 +07:00
* Copyright (c) 2022-2023, daeuniverse Organization <dae@v2raya.org>
2023-01-28 00:50:21 +07:00
*/
2023-01-27 01:10:27 +07:00
package config
2023-01-28 00:50:21 +07:00
import (
"fmt"
2023-03-14 14:01:55 +07:00
"github.com/daeuniverse/dae/pkg/config_parser"
2023-01-28 00:50:21 +07:00
"reflect"
"time"
)
type Global struct {
TproxyPort uint16 `mapstructure:"tproxy_port" default:"12345"`
LogLevel string `mapstructure:"log_level" default:"info"`
// We use DirectTcpCheckUrl to check (tcp)*(ipv4/ipv6) connectivity for direct.
//DirectTcpCheckUrl string `mapstructure:"direct_tcp_check_url" default:"http://www.qualcomm.cn/generate_204"`
2023-02-25 01:38:21 +07:00
TcpCheckUrl string `mapstructure:"tcp_check_url" default:"http://keep-alv.google.com/generate_204"`
UdpCheckDns string `mapstructure:"udp_check_dns" default:"dns.google:53"`
CheckInterval time.Duration `mapstructure:"check_interval" default:"30s"`
CheckTolerance time.Duration `mapstructure:"check_tolerance" default:"0"`
// Deprecated:
DnsUpstream string `mapstructure:"dns_upstream" default:"<empty>"`
LanInterface []string `mapstructure:"lan_interface"`
// Deprecated:
LanNatDirect bool `mapstructure:"lan_nat_direct" default:"false"`
WanInterface []string `mapstructure:"wan_interface"`
AllowInsecure bool `mapstructure:"allow_insecure" default:"false"`
DialMode string `mapstructure:"dial_mode" default:"domain"`
2023-01-28 00:50:21 +07:00
}
2023-02-25 01:38:21 +07:00
type FunctionOrString interface{}
func FunctionOrStringToFunction(fs FunctionOrString) (f *config_parser.Function) {
switch fs := fs.(type) {
case string:
return &config_parser.Function{Name: fs}
case *config_parser.Function:
return fs
default:
panic(fmt.Sprintf("unknown type of 'fallback' in section routing: %T", fs))
}
2023-01-27 01:10:27 +07:00
}
2023-01-28 00:50:21 +07:00
type FunctionListOrString interface{}
2023-03-01 10:36:08 +07:00
func FunctionListOrStringToFunctionList(fs FunctionListOrString) (f []*config_parser.Function) {
switch fs := fs.(type) {
case string:
return []*config_parser.Function{{Name: fs}}
case []*config_parser.Function:
return fs
default:
panic(fmt.Sprintf("unknown type of 'fallback' in section routing: %T", fs))
}
}
2023-02-25 01:38:21 +07:00
type Group struct {
Name string `mapstructure:"_"`
2023-02-12 10:17:10 +07:00
Filter []*config_parser.Function `mapstructure:"filter"`
Policy FunctionListOrString `mapstructure:"policy" required:""`
2023-01-27 01:10:27 +07:00
}
2023-01-28 00:50:21 +07:00
2023-02-25 01:38:21 +07:00
type DnsRequestRouting struct {
Rules []*config_parser.RoutingRule `mapstructure:"_"`
Fallback FunctionOrString `mapstructure:"fallback" required:""`
}
type DnsResponseRouting struct {
Rules []*config_parser.RoutingRule `mapstructure:"_"`
Fallback FunctionOrString `mapstructure:"fallback" required:""`
}
type DnsRouting struct {
Request DnsRequestRouting `mapstructure:"request"`
Response DnsResponseRouting `mapstructure:"response"`
}
type KeyableString string
2023-02-25 01:38:21 +07:00
type Dns struct {
Upstream []KeyableString `mapstructure:"upstream"`
Routing DnsRouting `mapstructure:"routing"`
2023-02-25 01:38:21 +07:00
}
2023-01-28 00:50:21 +07:00
type Routing struct {
Rules []*config_parser.RoutingRule `mapstructure:"_"`
2023-03-03 09:59:31 +07:00
Fallback FunctionOrString `mapstructure:"fallback" default:"direct"`
2023-01-27 01:10:27 +07:00
}
2023-01-28 00:50:21 +07:00
type Config struct {
Global Global `mapstructure:"global" required:"" desc:"GlobalDesc"`
Subscription []KeyableString `mapstructure:"subscription"`
Node []KeyableString `mapstructure:"node"`
2023-03-03 09:59:31 +07:00
Group []Group `mapstructure:"group" desc:"GroupDesc"`
Routing Routing `mapstructure:"routing" required:""`
Dns Dns `mapstructure:"dns" desc:"DnsDesc"`
2023-01-28 00:50:21 +07:00
}
2023-01-27 01:10:27 +07:00
2023-02-09 22:17:49 +07:00
// New params from sections. This func assumes merging (section "include") and deduplication for section names has been executed.
func New(sections []*config_parser.Section) (conf *Config, err error) {
2023-01-28 00:50:21 +07:00
// Set up name to section for further use.
type Section struct {
Val *config_parser.Section
Parsed bool
}
nameToSection := make(map[string]*Section)
for _, section := range sections {
nameToSection[section.Name] = &Section{Val: section}
}
2023-01-27 01:10:27 +07:00
conf = &Config{}
2023-01-28 00:50:21 +07:00
// Use specified parser to parse corresponding section.
_val := reflect.ValueOf(conf)
2023-01-28 00:50:21 +07:00
val := _val.Elem()
typ := val.Type()
for i := 0; i < val.NumField(); i++ {
field := val.Field(i)
structField := typ.Field(i)
2023-01-27 01:10:27 +07:00
2023-01-28 00:50:21 +07:00
// Find corresponding section from sections.
sectionName, ok := structField.Tag.Lookup("mapstructure")
if !ok {
return nil, fmt.Errorf("no mapstructure is specified in field %v", structField.Name)
}
section, ok := nameToSection[sectionName]
if !ok {
2023-02-25 01:38:21 +07:00
if _, required := structField.Tag.Lookup("required"); required {
return nil, fmt.Errorf("section %v is required but not provided", sectionName)
} else {
continue
}
2023-01-28 00:50:21 +07:00
}
// Parse section and unmarshal to field.
2023-02-25 01:38:21 +07:00
if err := SectionParser(field.Addr(), section.Val); err != nil {
2023-01-28 00:50:21 +07:00
return nil, fmt.Errorf("failed to parse \"%v\": %w", sectionName, err)
}
section.Parsed = true
}
2023-02-09 22:17:49 +07:00
// Report unknown. Not "unused" because we assume section name deduplication has been executed before this func.
2023-01-28 00:50:21 +07:00
for name, section := range nameToSection {
2023-02-09 22:17:49 +07:00
if section.Val.Name == "include" {
continue
}
2023-01-28 00:50:21 +07:00
if !section.Parsed {
return nil, fmt.Errorf("unknown section: %v", name)
}
}
// Apply config patches.
for _, patch := range patches {
if err = patch(conf); err != nil {
return nil, err
}
}
return conf, nil
2023-01-28 00:50:21 +07:00
}