2023-02-25 01:38:21 +07:00
|
|
|
/*
|
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
2024-01-04 16:28:16 +07:00
|
|
|
* Copyright (c) 2022-2024, daeuniverse Organization <dae@v2raya.org>
|
2023-02-25 01:38:21 +07:00
|
|
|
*/
|
|
|
|
|
|
|
|
package dns
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2023-04-23 12:27:29 +07:00
|
|
|
"strconv"
|
|
|
|
|
2023-03-14 14:01:55 +07:00
|
|
|
"github.com/daeuniverse/dae/common/consts"
|
|
|
|
"github.com/daeuniverse/dae/component/routing"
|
|
|
|
"github.com/daeuniverse/dae/component/routing/domain_matcher"
|
|
|
|
"github.com/daeuniverse/dae/config"
|
|
|
|
"github.com/daeuniverse/dae/pkg/config_parser"
|
2023-03-24 23:35:45 +07:00
|
|
|
"github.com/sirupsen/logrus"
|
2023-02-25 01:38:21 +07:00
|
|
|
)
|
|
|
|
|
|
|
|
type RequestMatcherBuilder struct {
|
2023-03-24 23:35:45 +07:00
|
|
|
log *logrus.Logger
|
2023-02-25 01:38:21 +07:00
|
|
|
upstreamName2Id map[string]uint8
|
|
|
|
simulatedDomainSet []routing.DomainSet
|
|
|
|
fallback *routing.Outbound
|
|
|
|
rules []requestMatchSet
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewRequestMatcherBuilder(log *logrus.Logger, rules []*config_parser.RoutingRule, upstreamName2Id map[string]uint8, fallback config.FunctionOrString) (b *RequestMatcherBuilder, err error) {
|
2023-03-24 23:35:45 +07:00
|
|
|
b = &RequestMatcherBuilder{log: log, upstreamName2Id: upstreamName2Id}
|
2023-02-25 01:38:21 +07:00
|
|
|
rulesBuilder := routing.NewRulesBuilder(log)
|
|
|
|
rulesBuilder.RegisterFunctionParser(consts.Function_QName, routing.PlainParserFactory(b.addQName))
|
|
|
|
rulesBuilder.RegisterFunctionParser(consts.Function_QType, TypeParserFactory(b.addQType))
|
|
|
|
if err = rulesBuilder.Apply(rules); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err = b.addFallback(fallback); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return b, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *RequestMatcherBuilder) upstreamToId(upstream string) (upstreamId consts.DnsRequestOutboundIndex, err error) {
|
|
|
|
switch upstream {
|
2023-03-31 14:46:53 +07:00
|
|
|
case consts.DnsRequestOutboundIndex_Reject.String():
|
|
|
|
upstreamId = consts.DnsRequestOutboundIndex_Reject
|
2023-02-25 01:38:21 +07:00
|
|
|
case consts.DnsRequestOutboundIndex_AsIs.String():
|
|
|
|
upstreamId = consts.DnsRequestOutboundIndex_AsIs
|
|
|
|
case consts.DnsRequestOutboundIndex_LogicalAnd.String():
|
|
|
|
upstreamId = consts.DnsRequestOutboundIndex_LogicalAnd
|
|
|
|
case consts.DnsRequestOutboundIndex_LogicalOr.String():
|
|
|
|
upstreamId = consts.DnsRequestOutboundIndex_LogicalOr
|
|
|
|
default:
|
|
|
|
_upstreamId, ok := b.upstreamName2Id[upstream]
|
|
|
|
if !ok {
|
|
|
|
return 0, fmt.Errorf("upstream %v not found; please define it in section \"dns.upstream\"", strconv.Quote(upstream))
|
|
|
|
}
|
|
|
|
upstreamId = consts.DnsRequestOutboundIndex(_upstreamId)
|
|
|
|
}
|
|
|
|
return upstreamId, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *RequestMatcherBuilder) addQName(f *config_parser.Function, key string, values []string, upstream *routing.Outbound) (err error) {
|
|
|
|
switch consts.RoutingDomainKey(key) {
|
|
|
|
case consts.RoutingDomainKey_Regex,
|
|
|
|
consts.RoutingDomainKey_Full,
|
|
|
|
consts.RoutingDomainKey_Keyword,
|
|
|
|
consts.RoutingDomainKey_Suffix:
|
|
|
|
default:
|
|
|
|
return fmt.Errorf("addQName: unsupported key: %v", key)
|
|
|
|
}
|
|
|
|
b.simulatedDomainSet = append(b.simulatedDomainSet, routing.DomainSet{
|
|
|
|
Key: consts.RoutingDomainKey(key),
|
2023-05-27 12:51:19 +07:00
|
|
|
RuleIndex: len(b.rules),
|
2023-02-25 01:38:21 +07:00
|
|
|
Domains: values,
|
|
|
|
})
|
|
|
|
upstreamId, err := b.upstreamToId(upstream.Name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b.rules = append(b.rules, requestMatchSet{
|
|
|
|
Type: consts.MatchType_DomainSet,
|
|
|
|
Not: f.Not,
|
|
|
|
Upstream: uint8(upstreamId),
|
|
|
|
})
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-07-09 15:02:17 +07:00
|
|
|
func (b *RequestMatcherBuilder) addQType(f *config_parser.Function, values []uint16, upstream *routing.Outbound) (err error) {
|
2023-02-25 01:38:21 +07:00
|
|
|
for i, value := range values {
|
|
|
|
upstreamName := consts.OutboundLogicalOr.String()
|
|
|
|
if i == len(values)-1 {
|
|
|
|
upstreamName = upstream.Name
|
|
|
|
}
|
|
|
|
upstreamId, err := b.upstreamToId(upstreamName)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b.rules = append(b.rules, requestMatchSet{
|
|
|
|
Type: consts.MatchType_QType,
|
|
|
|
Value: uint16(value),
|
|
|
|
Not: f.Not,
|
|
|
|
Upstream: uint8(upstreamId),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *RequestMatcherBuilder) addFallback(fallbackOutbound config.FunctionOrString) (err error) {
|
|
|
|
upstream, err := routing.ParseOutbound(config.FunctionOrStringToFunction(fallbackOutbound))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-04-02 10:07:53 +07:00
|
|
|
if upstream.Must {
|
|
|
|
return fmt.Errorf("unsupported param: must")
|
|
|
|
}
|
|
|
|
if upstream.Mark != 0 {
|
|
|
|
return fmt.Errorf("unsupported param: mark")
|
|
|
|
}
|
2023-02-25 01:38:21 +07:00
|
|
|
upstreamId, err := b.upstreamToId(upstream.Name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b.rules = append(b.rules, requestMatchSet{
|
|
|
|
Type: consts.MatchType_Fallback,
|
|
|
|
Upstream: uint8(upstreamId),
|
|
|
|
})
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *RequestMatcherBuilder) Build() (matcher *RequestMatcher, err error) {
|
|
|
|
var m RequestMatcher
|
|
|
|
// Build domainMatcher
|
2023-03-24 23:35:45 +07:00
|
|
|
m.domainMatcher = domain_matcher.NewAhocorasickSlimtrie(b.log, consts.MaxMatchSetLen)
|
2023-02-25 01:38:21 +07:00
|
|
|
for _, domains := range b.simulatedDomainSet {
|
|
|
|
m.domainMatcher.AddSet(domains.RuleIndex, domains.Domains, domains.Key)
|
|
|
|
}
|
|
|
|
if err = m.domainMatcher.Build(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write routings.
|
|
|
|
// Fallback rule MUST be the last.
|
|
|
|
if b.rules[len(b.rules)-1].Type != consts.MatchType_Fallback {
|
|
|
|
return nil, fmt.Errorf("fallback rule MUST be the last")
|
|
|
|
}
|
|
|
|
m.matches = b.rules
|
|
|
|
|
|
|
|
return &m, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type RequestMatcher struct {
|
|
|
|
domainMatcher routing.DomainMatcher // All domain matchSets use one DomainMatcher.
|
|
|
|
|
|
|
|
matches []requestMatchSet
|
|
|
|
}
|
|
|
|
|
|
|
|
type requestMatchSet struct {
|
|
|
|
Value uint16
|
|
|
|
Not bool
|
|
|
|
Type consts.MatchType
|
|
|
|
Upstream uint8
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *RequestMatcher) Match(
|
|
|
|
qName string,
|
2023-07-09 15:02:17 +07:00
|
|
|
qType uint16,
|
2023-02-25 01:38:21 +07:00
|
|
|
) (upstreamIndex consts.DnsRequestOutboundIndex, err error) {
|
|
|
|
var domainMatchBitmap []uint32
|
|
|
|
if qName != "" {
|
|
|
|
domainMatchBitmap = m.domainMatcher.MatchDomainBitmap(qName)
|
|
|
|
}
|
|
|
|
|
|
|
|
goodSubrule := false
|
|
|
|
badRule := false
|
|
|
|
for i, match := range m.matches {
|
|
|
|
if badRule || goodSubrule {
|
|
|
|
goto beforeNextLoop
|
|
|
|
}
|
|
|
|
switch match.Type {
|
|
|
|
case consts.MatchType_DomainSet:
|
|
|
|
if domainMatchBitmap != nil && (domainMatchBitmap[i/32]>>(i%32))&1 > 0 {
|
|
|
|
goodSubrule = true
|
|
|
|
}
|
|
|
|
case consts.MatchType_QType:
|
2023-07-09 15:02:17 +07:00
|
|
|
if qType == match.Value {
|
2023-02-25 01:38:21 +07:00
|
|
|
goodSubrule = true
|
|
|
|
}
|
|
|
|
case consts.MatchType_Fallback:
|
|
|
|
goodSubrule = true
|
|
|
|
default:
|
|
|
|
return 0, fmt.Errorf("unknown match type: %v", match.Type)
|
|
|
|
}
|
|
|
|
beforeNextLoop:
|
|
|
|
upstream := consts.DnsRequestOutboundIndex(match.Upstream)
|
|
|
|
if upstream != consts.DnsRequestOutboundIndex_LogicalOr {
|
|
|
|
// This match_set reaches the end of subrule.
|
|
|
|
// We are now at end of rule, or next match_set belongs to another
|
|
|
|
// subrule.
|
|
|
|
|
|
|
|
if goodSubrule == match.Not {
|
|
|
|
// This subrule does not hit.
|
|
|
|
badRule = true
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reset goodSubrule.
|
|
|
|
goodSubrule = false
|
|
|
|
}
|
|
|
|
|
|
|
|
if upstream&consts.DnsRequestOutboundIndex_LogicalMask !=
|
|
|
|
consts.DnsRequestOutboundIndex_LogicalMask {
|
|
|
|
// Tail of a rule (line).
|
|
|
|
// Decide whether to hit.
|
|
|
|
if !badRule {
|
|
|
|
return upstream, nil
|
|
|
|
}
|
|
|
|
badRule = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0, fmt.Errorf("no match set hit")
|
|
|
|
}
|