dae/component/routing/matcher_builder.go

131 lines
3.2 KiB
Go
Raw Normal View History

2023-01-23 18:54: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-23 18:54:21 +07:00
*/
package routing
import (
"fmt"
2023-03-14 14:01:55 +07:00
"github.com/daeuniverse/dae/common/consts"
"github.com/daeuniverse/dae/pkg/config_parser"
2023-04-02 10:07:53 +07:00
"github.com/sirupsen/logrus"
"strconv"
2023-01-23 18:54:21 +07:00
)
type DomainSet struct {
Key consts.RoutingDomainKey
RuleIndex int
Domains []string
}
type Outbound struct {
Name string
Mark uint32
2023-04-02 10:07:53 +07:00
Must bool
}
2023-02-25 01:38:21 +07:00
type RulesBuilder struct {
log *logrus.Logger
parsers map[string]FunctionParser
2023-01-23 18:54:21 +07:00
}
2023-02-25 01:38:21 +07:00
func NewRulesBuilder(log *logrus.Logger) *RulesBuilder {
return &RulesBuilder{
log: log,
parsers: make(map[string]FunctionParser),
2023-01-23 18:54:21 +07:00
}
}
2023-02-25 01:38:21 +07:00
func (b *RulesBuilder) RegisterFunctionParser(funcName string, parser FunctionParser) {
b.parsers[funcName] = parser
}
2023-02-25 01:38:21 +07:00
func (b *RulesBuilder) Apply(rules []*config_parser.RoutingRule) (err error) {
2023-01-23 18:54:21 +07:00
for _, rule := range rules {
b.log.Debugln("[rule]", rule.String(true, false, false))
2023-02-25 01:38:21 +07:00
outbound, err := ParseOutbound(&rule.Outbound)
if err != nil {
return err
}
2023-01-29 10:19:58 +07:00
2023-01-23 18:54:21 +07:00
// rule is like: domain(domain:baidu.com) && port(443) -> proxy
for iFunc, f := range rule.AndFunctions {
// f is like: domain(domain:baidu.com)
2023-02-25 01:38:21 +07:00
functionParser, ok := b.parsers[f.Name]
if !ok {
return fmt.Errorf("unknown function: %v", f.Name)
}
paramValueGroups, keyOrder := groupParamValuesByKey(f.Params)
for jMatchSet, key := range keyOrder {
paramValueGroup := paramValueGroups[key]
// Preprocess the outbound.
overrideOutbound := &Outbound{
2023-02-25 01:38:21 +07:00
Name: consts.OutboundLogicalOr.String(),
Mark: outbound.Mark,
2023-04-02 10:07:53 +07:00
Must: outbound.Must,
}
if jMatchSet == len(keyOrder)-1 {
2023-02-25 01:38:21 +07:00
overrideOutbound.Name = consts.OutboundLogicalAnd.String()
if iFunc == len(rule.AndFunctions)-1 {
overrideOutbound.Name = outbound.Name
}
2023-01-23 18:54:21 +07:00
}
2023-01-29 10:19:58 +07:00
{
// Debug
symNot := ""
if f.Not {
symNot = "!"
}
2023-02-25 01:38:21 +07:00
b.log.Debugf("\t%v%v(%v) -> %v", symNot, f.Name, key, overrideOutbound.Name)
2023-01-29 10:19:58 +07:00
}
2023-02-25 01:38:21 +07:00
if err = functionParser(b.log, f, key, paramValueGroup, overrideOutbound); err != nil {
return fmt.Errorf("failed to parse '%v': %w", f.String(false, false, false), err)
2023-01-23 18:54:21 +07:00
}
}
}
}
return nil
}
2023-02-25 01:38:21 +07:00
func groupParamValuesByKey(params []*config_parser.Param) (keyToValues map[string][]string, keyOrder []string) {
groups := make(map[string][]string)
for _, param := range params {
if _, ok := groups[param.Key]; !ok {
keyOrder = append(keyOrder, param.Key)
}
groups[param.Key] = append(groups[param.Key], param.Val)
}
return groups, keyOrder
}
2023-01-23 18:54:21 +07:00
2023-02-25 01:38:21 +07:00
func ParseOutbound(rawOutbound *config_parser.Function) (outbound *Outbound, err error) {
outbound = &Outbound{
Name: rawOutbound.Name,
Mark: 0,
2023-04-02 10:07:53 +07:00
Must: false,
2023-02-25 01:38:21 +07:00
}
for _, p := range rawOutbound.Params {
switch p.Key {
case consts.OutboundParam_Mark:
var _mark uint64
_mark, err = strconv.ParseUint(p.Val, 0, 32)
if err != nil {
return nil, fmt.Errorf("failed to parse mark: %v", err)
}
outbound.Mark = uint32(_mark)
2023-04-02 10:07:53 +07:00
case "":
if p.Val == "must" {
outbound.Must = true
} else {
return nil, fmt.Errorf("unknown outbound param: %v", p.Val)
}
2023-02-25 01:38:21 +07:00
default:
2023-04-02 10:07:53 +07:00
return nil, fmt.Errorf("unknown outbound param key: %v", p.Key)
2023-02-25 01:38:21 +07:00
}
}
return outbound, nil
2023-01-27 01:10:27 +07:00
}