mirror of
https://gitclone.com/github.com/MetaCubeX/Clash.Meta
synced 2024-11-15 21:51:23 +08:00
91 lines
2.6 KiB
Go
91 lines
2.6 KiB
Go
|
package provider
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"github.com/Dreamacro/clash/adapter/provider"
|
||
|
"github.com/Dreamacro/clash/common/structure"
|
||
|
C "github.com/Dreamacro/clash/constant"
|
||
|
P "github.com/Dreamacro/clash/constant/provider"
|
||
|
RC "github.com/Dreamacro/clash/rule/common"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
type ruleProviderSchema struct {
|
||
|
Type string `provider:"type"`
|
||
|
Behavior string `provider:"behavior"`
|
||
|
Path string `provider:"path"`
|
||
|
URL string `provider:"url,omitempty"`
|
||
|
Interval int `provider:"interval,omitempty"`
|
||
|
}
|
||
|
|
||
|
func ParseRuleProvider(name string, mapping map[string]interface{}) (P.RuleProvider, error) {
|
||
|
schema := &ruleProviderSchema{}
|
||
|
decoder := structure.NewDecoder(structure.Option{TagName: "provider", WeaklyTypedInput: true})
|
||
|
if err := decoder.Decode(mapping, schema); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
var behavior P.RuleType
|
||
|
|
||
|
switch schema.Behavior {
|
||
|
case "domain":
|
||
|
behavior = P.Domain
|
||
|
case "ipcidr":
|
||
|
behavior = P.IPCIDR
|
||
|
case "classical":
|
||
|
behavior = P.Classical
|
||
|
default:
|
||
|
return nil, fmt.Errorf("unsupported behavior type: %s", schema.Behavior)
|
||
|
}
|
||
|
|
||
|
path := C.Path.Resolve(schema.Path)
|
||
|
var vehicle P.Vehicle
|
||
|
switch schema.Type {
|
||
|
case "file":
|
||
|
vehicle = provider.NewFileVehicle(path)
|
||
|
case "http":
|
||
|
vehicle = provider.NewHTTPVehicle(schema.URL, path)
|
||
|
default:
|
||
|
return nil, fmt.Errorf("unsupported vehicle type: %s", schema.Type)
|
||
|
}
|
||
|
|
||
|
return NewRuleSetProvider(name, behavior, time.Duration(uint(schema.Interval))*time.Second, vehicle), nil
|
||
|
}
|
||
|
|
||
|
func parseRule(tp, payload, target string, params []string) (C.Rule, error) {
|
||
|
var (
|
||
|
parseErr error
|
||
|
parsed C.Rule
|
||
|
)
|
||
|
|
||
|
ruleExtra := &C.RuleExtra{
|
||
|
Network: RC.FindNetwork(params),
|
||
|
SourceIPs: RC.FindSourceIPs(params),
|
||
|
}
|
||
|
|
||
|
switch tp {
|
||
|
case "DOMAIN":
|
||
|
parsed = RC.NewDomain(payload, target, ruleExtra)
|
||
|
case "DOMAIN-SUFFIX":
|
||
|
parsed = RC.NewDomainSuffix(payload, target, ruleExtra)
|
||
|
case "DOMAIN-KEYWORD":
|
||
|
parsed = RC.NewDomainKeyword(payload, target, ruleExtra)
|
||
|
case "GEOSITE":
|
||
|
parsed, parseErr = RC.NewGEOSITE(payload, target, ruleExtra)
|
||
|
case "IP-CIDR", "IP-CIDR6":
|
||
|
noResolve := RC.HasNoResolve(params)
|
||
|
parsed, parseErr = RC.NewIPCIDR(payload, target, ruleExtra, RC.WithIPCIDRNoResolve(noResolve))
|
||
|
case "SRC-IP-CIDR":
|
||
|
parsed, parseErr = RC.NewIPCIDR(payload, target, ruleExtra, RC.WithIPCIDRSourceIP(true), RC.WithIPCIDRNoResolve(true))
|
||
|
case "SRC-PORT":
|
||
|
parsed, parseErr = RC.NewPort(payload, target, true, ruleExtra)
|
||
|
case "DST-PORT":
|
||
|
parsed, parseErr = RC.NewPort(payload, target, false, ruleExtra)
|
||
|
case "PROCESS-NAME":
|
||
|
parsed, parseErr = RC.NewProcess(payload, target, ruleExtra)
|
||
|
default:
|
||
|
parseErr = fmt.Errorf("unsupported rule type %s", tp)
|
||
|
}
|
||
|
|
||
|
return parsed, parseErr
|
||
|
}
|