mirror of
https://github.com/SagerNet/sing-box.git
synced 2025-01-25 10:16:55 +00:00
195 lines
4.7 KiB
Go
195 lines
4.7 KiB
Go
package rule
|
|
|
|
import (
|
|
"context"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/sagernet/fswatch"
|
|
"github.com/sagernet/sing-box/adapter"
|
|
"github.com/sagernet/sing-box/common/srs"
|
|
C "github.com/sagernet/sing-box/constant"
|
|
"github.com/sagernet/sing-box/option"
|
|
"github.com/sagernet/sing/common"
|
|
"github.com/sagernet/sing/common/atomic"
|
|
E "github.com/sagernet/sing/common/exceptions"
|
|
F "github.com/sagernet/sing/common/format"
|
|
"github.com/sagernet/sing/common/json"
|
|
"github.com/sagernet/sing/common/logger"
|
|
"github.com/sagernet/sing/common/x/list"
|
|
"github.com/sagernet/sing/service/filemanager"
|
|
|
|
"go4.org/netipx"
|
|
)
|
|
|
|
var _ adapter.RuleSet = (*LocalRuleSet)(nil)
|
|
|
|
type LocalRuleSet struct {
|
|
ctx context.Context
|
|
logger logger.Logger
|
|
tag string
|
|
rules []adapter.HeadlessRule
|
|
metadata adapter.RuleSetMetadata
|
|
fileFormat string
|
|
watcher *fswatch.Watcher
|
|
refs atomic.Int32
|
|
}
|
|
|
|
func NewLocalRuleSet(ctx context.Context, logger logger.Logger, options option.RuleSet) (*LocalRuleSet, error) {
|
|
ruleSet := &LocalRuleSet{
|
|
ctx: ctx,
|
|
logger: logger,
|
|
tag: options.Tag,
|
|
fileFormat: options.Format,
|
|
}
|
|
if options.Type == C.RuleSetTypeInline {
|
|
if len(options.InlineOptions.Rules) == 0 {
|
|
return nil, E.New("empty inline rule-set")
|
|
}
|
|
err := ruleSet.reloadRules(options.InlineOptions.Rules)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
} else {
|
|
err := ruleSet.reloadFile(filemanager.BasePath(ctx, options.LocalOptions.Path))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
if options.Type == C.RuleSetTypeLocal {
|
|
filePath, _ := filepath.Abs(options.LocalOptions.Path)
|
|
watcher, err := fswatch.NewWatcher(fswatch.Options{
|
|
Path: []string{filePath},
|
|
Callback: func(path string) {
|
|
uErr := ruleSet.reloadFile(path)
|
|
if uErr != nil {
|
|
logger.Error(E.Cause(uErr, "reload rule-set ", options.Tag))
|
|
}
|
|
},
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
ruleSet.watcher = watcher
|
|
}
|
|
return ruleSet, nil
|
|
}
|
|
|
|
func (s *LocalRuleSet) Name() string {
|
|
return s.tag
|
|
}
|
|
|
|
func (s *LocalRuleSet) String() string {
|
|
return strings.Join(F.MapToString(s.rules), " ")
|
|
}
|
|
|
|
func (s *LocalRuleSet) StartContext(ctx context.Context, startContext *adapter.HTTPStartContext) error {
|
|
if s.watcher != nil {
|
|
err := s.watcher.Start()
|
|
if err != nil {
|
|
s.logger.Error(E.Cause(err, "watch rule-set file"))
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (s *LocalRuleSet) reloadFile(path string) error {
|
|
var ruleSet option.PlainRuleSetCompat
|
|
switch s.fileFormat {
|
|
case C.RuleSetFormatSource, "":
|
|
content, err := os.ReadFile(path)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
ruleSet, err = json.UnmarshalExtended[option.PlainRuleSetCompat](content)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
case C.RuleSetFormatBinary:
|
|
setFile, err := os.Open(path)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
ruleSet, err = srs.Read(setFile, false)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
default:
|
|
return E.New("unknown rule-set format: ", s.fileFormat)
|
|
}
|
|
plainRuleSet, err := ruleSet.Upgrade()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return s.reloadRules(plainRuleSet.Rules)
|
|
}
|
|
|
|
func (s *LocalRuleSet) reloadRules(headlessRules []option.HeadlessRule) error {
|
|
rules := make([]adapter.HeadlessRule, len(headlessRules))
|
|
var err error
|
|
for i, ruleOptions := range headlessRules {
|
|
rules[i], err = NewHeadlessRule(s.ctx, ruleOptions)
|
|
if err != nil {
|
|
return E.Cause(err, "parse rule_set.rules.[", i, "]")
|
|
}
|
|
}
|
|
var metadata adapter.RuleSetMetadata
|
|
metadata.ContainsProcessRule = hasHeadlessRule(headlessRules, isProcessHeadlessRule)
|
|
metadata.ContainsWIFIRule = hasHeadlessRule(headlessRules, isWIFIHeadlessRule)
|
|
metadata.ContainsIPCIDRRule = hasHeadlessRule(headlessRules, isIPCIDRHeadlessRule)
|
|
s.rules = rules
|
|
s.metadata = metadata
|
|
return nil
|
|
}
|
|
|
|
func (s *LocalRuleSet) PostStart() error {
|
|
return nil
|
|
}
|
|
|
|
func (s *LocalRuleSet) Metadata() adapter.RuleSetMetadata {
|
|
return s.metadata
|
|
}
|
|
|
|
func (s *LocalRuleSet) ExtractIPSet() []*netipx.IPSet {
|
|
return common.FlatMap(s.rules, extractIPSetFromRule)
|
|
}
|
|
|
|
func (s *LocalRuleSet) IncRef() {
|
|
s.refs.Add(1)
|
|
}
|
|
|
|
func (s *LocalRuleSet) DecRef() {
|
|
if s.refs.Add(-1) < 0 {
|
|
panic("rule-set: negative refs")
|
|
}
|
|
}
|
|
|
|
func (s *LocalRuleSet) Cleanup() {
|
|
if s.refs.Load() == 0 {
|
|
s.rules = nil
|
|
}
|
|
}
|
|
|
|
func (s *LocalRuleSet) RegisterCallback(callback adapter.RuleSetUpdateCallback) *list.Element[adapter.RuleSetUpdateCallback] {
|
|
return nil
|
|
}
|
|
|
|
func (s *LocalRuleSet) UnregisterCallback(element *list.Element[adapter.RuleSetUpdateCallback]) {
|
|
}
|
|
|
|
func (s *LocalRuleSet) Close() error {
|
|
s.rules = nil
|
|
return common.Close(common.PtrOrNil(s.watcher))
|
|
}
|
|
|
|
func (s *LocalRuleSet) Match(metadata *adapter.InboundContext) bool {
|
|
for _, rule := range s.rules {
|
|
if rule.Match(metadata) {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|