| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140 | package router//go:generate go run $GOPATH/src/v2ray.com/core/common/errors/errorgen/main.go -pkg router -path App,Routerimport (	"context"	"v2ray.com/core/app"	"v2ray.com/core/app/dns"	"v2ray.com/core/app/log"	"v2ray.com/core/common"	"v2ray.com/core/common/net"	"v2ray.com/core/proxy")var (	ErrNoRuleApplicable = newError("No rule applicable"))type Router struct {	domainStrategy Config_DomainStrategy	rules          []Rule	dnsServer      dns.Server}func NewRouter(ctx context.Context, config *Config) (*Router, error) {	space := app.SpaceFromContext(ctx)	if space == nil {		return nil, newError("no space in context")	}	r := &Router{		domainStrategy: config.DomainStrategy,		rules:          make([]Rule, len(config.Rule)),	}	space.On(app.SpaceInitializing, func(interface{}) error {		for idx, rule := range config.Rule {			r.rules[idx].Tag = rule.Tag			cond, err := rule.BuildCondition()			if err != nil {				return err			}			r.rules[idx].Condition = cond		}		r.dnsServer = dns.FromSpace(space)		if r.dnsServer == nil {			return newError("DNS is not found in the space")		}		return nil	})	return r, nil}type ipResolver struct {	ip        []net.Address	domain    string	resolved  bool	dnsServer dns.Server}func (r *ipResolver) Resolve() []net.Address {	if r.resolved {		return r.ip	}	log.Trace(newError("looking for IP for domain: ", r.domain))	r.resolved = true	ips := r.dnsServer.Get(r.domain)	if len(ips) == 0 {		return nil	}	r.ip = make([]net.Address, len(ips))	for i, ip := range ips {		r.ip[i] = net.IPAddress(ip)	}	return r.ip}func (r *Router) TakeDetour(ctx context.Context) (string, error) {	resolver := &ipResolver{		dnsServer: r.dnsServer,	}	if r.domainStrategy == Config_IpOnDemand {		if dest, ok := proxy.TargetFromContext(ctx); ok && dest.Address.Family().IsDomain() {			resolver.domain = dest.Address.Domain()			ctx = proxy.ContextWithResolveIPs(ctx, resolver)		}	}	for _, rule := range r.rules {		if rule.Apply(ctx) {			return rule.Tag, nil		}	}	dest, ok := proxy.TargetFromContext(ctx)	if !ok {		return "", ErrNoRuleApplicable	}	if r.domainStrategy == Config_IpIfNonMatch && dest.Address.Family().IsDomain() {		resolver.domain = dest.Address.Domain()		ips := resolver.Resolve()		if len(ips) > 0 {			ctx = proxy.ContextWithResolveIPs(ctx, resolver)			for _, rule := range r.rules {				if rule.Apply(ctx) {					return rule.Tag, nil				}			}		}	}	return "", ErrNoRuleApplicable}func (*Router) Interface() interface{} {	return (*Router)(nil)}func (*Router) Start() error {	return nil}func (*Router) Close() {}func FromSpace(space app.Space) *Router {	app := space.GetApplication((*Router)(nil))	if app == nil {		return nil	}	return app.(*Router)}func init() {	common.Must(common.RegisterConfig((*Config)(nil), func(ctx context.Context, config interface{}) (interface{}, error) {		return NewRouter(ctx, config.(*Config))	}))}
 |