| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153 | package dns//go:generate errorgenimport (	"context"	"sync"	"time"	"v2ray.com/core"	"v2ray.com/core/common"	"v2ray.com/core/common/net"	"v2ray.com/core/common/strmatcher")type Server struct {	sync.Mutex	hosts          *StaticHosts	servers        []NameServerInterface	clientIP       net.IP	domainMatcher  strmatcher.IndexMatcher	domainIndexMap map[uint32]uint32}func New(ctx context.Context, config *Config) (*Server, error) {	server := &Server{		servers: make([]NameServerInterface, 0, len(config.NameServers)+len(config.NameServer)),	}	if len(config.ClientIp) > 0 {		if len(config.ClientIp) != 4 && len(config.ClientIp) != 16 {			return nil, newError("unexpected IP length", len(config.ClientIp))		}		server.clientIP = net.IP(config.ClientIp)	}	hosts, err := NewStaticHosts(config.StaticHosts, config.Hosts)	if err != nil {		return nil, newError("failed to create hosts").Base(err)	}	server.hosts = hosts	v := core.MustFromContext(ctx)	if err := v.RegisterFeature((*core.DNSClient)(nil), server); err != nil {		return nil, newError("unable to register DNSClient.").Base(err)	}	addNameServer := func(endpoint *net.Endpoint) int {		address := endpoint.Address.AsAddress()		if address.Family().IsDomain() && address.Domain() == "localhost" {			server.servers = append(server.servers, NewLocalNameServer())		} else {			dest := endpoint.AsDestination()			if dest.Network == net.Network_Unknown {				dest.Network = net.Network_UDP			}			if dest.Network == net.Network_UDP {				server.servers = append(server.servers, NewClassicNameServer(dest, v.Dispatcher(), server.clientIP))			}		}		return len(server.servers) - 1	}	if len(config.NameServers) > 0 {		core.PrintDeprecatedFeatureWarning("simple DNS server")	}	for _, destPB := range config.NameServers {		addNameServer(destPB)	}	if len(config.NameServer) > 0 {		domainMatcher := &strmatcher.MatcherGroup{}		domainIndexMap := make(map[uint32]uint32)		for _, ns := range config.NameServer {			idx := addNameServer(ns.Address)			for _, domain := range ns.PrioritizedDomain {				matcher, err := toStrMatcher(domain.Type, domain.Domain)				if err != nil {					return nil, newError("failed to create proritized domain").Base(err).AtWarning()				}				midx := domainMatcher.Add(matcher)				domainIndexMap[midx] = uint32(idx)			}		}		server.domainMatcher = domainMatcher		server.domainIndexMap = domainIndexMap	}	if len(config.NameServers) == 0 {		server.servers = append(server.servers, NewLocalNameServer())	}	return server, nil}// Start implements common.Runnable.func (s *Server) Start() error {	return nil}// Close implements common.Closable.func (s *Server) Close() error {	return nil}func (s *Server) queryIPTimeout(server NameServerInterface, domain string) ([]net.IP, error) {	ctx, cancel := context.WithTimeout(context.Background(), time.Second*4)	ips, err := server.QueryIP(ctx, domain)	cancel()	return ips, err}func (s *Server) LookupIP(domain string) ([]net.IP, error) {	if ip := s.hosts.LookupIP(domain); len(ip) > 0 {		return ip, nil	}	var lastErr error	if s.domainMatcher != nil {		idx := s.domainMatcher.Match(domain)		if idx > 0 {			ns := s.servers[idx]			ips, err := s.queryIPTimeout(ns, domain)			if len(ips) > 0 {				return ips, nil			}			if err != nil {				lastErr = err			}		}	}	for _, server := range s.servers {		ips, err := s.queryIPTimeout(server, domain)		if len(ips) > 0 {			return ips, nil		}		if err != nil {			lastErr = err		}	}	return nil, newError("returning nil for domain ", domain).Base(lastErr)}func init() {	common.Must(common.RegisterConfig((*Config)(nil), func(ctx context.Context, config interface{}) (interface{}, error) {		return New(ctx, config.(*Config))	}))}
 |