| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116 | package rulesimport (	"errors"	"time"	"github.com/v2ray/v2ray-core/app"	"github.com/v2ray/v2ray-core/app/dns"	"github.com/v2ray/v2ray-core/app/router"	"github.com/v2ray/v2ray-core/common/collect"	v2net "github.com/v2ray/v2ray-core/common/net")var (	ErrorInvalidRule      = errors.New("Invalid Rule")	ErrorNoRuleApplicable = errors.New("No rule applicable"))type cacheEntry struct {	tag        string	err        error	validUntil time.Time}func newCacheEntry(tag string, err error) *cacheEntry {	this := &cacheEntry{		tag: tag,		err: err,	}	this.Extend()	return this}func (this *cacheEntry) IsValid() bool {	return this.validUntil.Before(time.Now())}func (this *cacheEntry) Extend() {	this.validUntil = time.Now().Add(time.Hour)}type Router struct {	config *RouterRuleConfig	cache  *collect.ValidityMap	space  app.Space}func NewRouter(config *RouterRuleConfig, space app.Space) *Router {	return &Router{		config: config,		cache:  collect.NewValidityMap(3600),		space:  space,	}}// @Privatefunc (this *Router) ResolveIP(dest v2net.Destination) []v2net.Destination {	dnsServer := this.space.GetApp(dns.APP_ID).(dns.Server)	ips := dnsServer.Get(dest.Address().Domain())	if len(ips) == 0 {		return nil	}	dests := make([]v2net.Destination, len(ips))	for idx, ip := range ips {		if dest.IsTCP() {			dests[idx] = v2net.TCPDestination(v2net.IPAddress(ip), dest.Port())		} else {			dests[idx] = v2net.UDPDestination(v2net.IPAddress(ip), dest.Port())		}	}	return dests}func (this *Router) takeDetourWithoutCache(dest v2net.Destination) (string, error) {	for _, rule := range this.config.Rules {		if rule.Apply(dest) {			return rule.Tag, nil		}	}	if this.config.DomainStrategy == UseIPIfNonMatch && dest.Address().IsDomain() {		ipDests := this.ResolveIP(dest)		if ipDests != nil {			for _, ipDest := range ipDests {				for _, rule := range this.config.Rules {					if rule.Apply(ipDest) {						return rule.Tag, nil					}				}			}		}	}	return "", ErrorNoRuleApplicable}func (this *Router) TakeDetour(dest v2net.Destination) (string, error) {	rawEntry := this.cache.Get(dest)	if rawEntry == nil {		tag, err := this.takeDetourWithoutCache(dest)		this.cache.Set(dest, newCacheEntry(tag, err))		return tag, err	}	entry := rawEntry.(*cacheEntry)	return entry.tag, entry.err}type RouterFactory struct {}func (this *RouterFactory) Create(rawConfig interface{}, space app.Space) (router.Router, error) {	return NewRouter(rawConfig.(*RouterRuleConfig), space), nil}func init() {	router.RegisterRouter("rules", &RouterFactory{})}
 |