mirror of https://github.com/v2ray/v2ray-core
				
				
				
			
		
			
				
	
	
		
			119 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Go
		
	
	
			
		
		
	
	
			119 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Go
		
	
	
// +build !confonly
 | 
						|
 | 
						|
package dns
 | 
						|
 | 
						|
import (
 | 
						|
	"v2ray.com/core/common"
 | 
						|
	"v2ray.com/core/common/net"
 | 
						|
	"v2ray.com/core/common/strmatcher"
 | 
						|
	"v2ray.com/core/features"
 | 
						|
)
 | 
						|
 | 
						|
// StaticHosts represents static domain-ip mapping in DNS server.
 | 
						|
type StaticHosts struct {
 | 
						|
	ips      [][]net.Address
 | 
						|
	matchers *strmatcher.MatcherGroup
 | 
						|
}
 | 
						|
 | 
						|
var typeMap = map[DomainMatchingType]strmatcher.Type{
 | 
						|
	DomainMatchingType_Full:      strmatcher.Full,
 | 
						|
	DomainMatchingType_Subdomain: strmatcher.Domain,
 | 
						|
	DomainMatchingType_Keyword:   strmatcher.Substr,
 | 
						|
	DomainMatchingType_Regex:     strmatcher.Regex,
 | 
						|
}
 | 
						|
 | 
						|
func toStrMatcher(t DomainMatchingType, domain string) (strmatcher.Matcher, error) {
 | 
						|
	strMType, f := typeMap[t]
 | 
						|
	if !f {
 | 
						|
		return nil, newError("unknown mapping type", t).AtWarning()
 | 
						|
	}
 | 
						|
	matcher, err := strMType.New(domain)
 | 
						|
	if err != nil {
 | 
						|
		return nil, newError("failed to create str matcher").Base(err)
 | 
						|
	}
 | 
						|
	return matcher, nil
 | 
						|
}
 | 
						|
 | 
						|
// NewStaticHosts creates a new StaticHosts instance.
 | 
						|
func NewStaticHosts(hosts []*Config_HostMapping, legacy map[string]*net.IPOrDomain) (*StaticHosts, error) {
 | 
						|
	g := new(strmatcher.MatcherGroup)
 | 
						|
	sh := &StaticHosts{
 | 
						|
		ips:      make([][]net.Address, len(hosts)+len(legacy)+16),
 | 
						|
		matchers: g,
 | 
						|
	}
 | 
						|
 | 
						|
	if legacy != nil {
 | 
						|
		features.PrintDeprecatedFeatureWarning("simple host mapping")
 | 
						|
 | 
						|
		for domain, ip := range legacy {
 | 
						|
			matcher, err := strmatcher.Full.New(domain)
 | 
						|
			common.Must(err)
 | 
						|
			id := g.Add(matcher)
 | 
						|
 | 
						|
			address := ip.AsAddress()
 | 
						|
			if address.Family().IsDomain() {
 | 
						|
				return nil, newError("invalid domain address in static hosts: ", address.Domain()).AtWarning()
 | 
						|
			}
 | 
						|
 | 
						|
			sh.ips[id] = []net.Address{address}
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	for _, mapping := range hosts {
 | 
						|
		matcher, err := toStrMatcher(mapping.Type, mapping.Domain)
 | 
						|
		if err != nil {
 | 
						|
			return nil, newError("failed to create domain matcher").Base(err)
 | 
						|
		}
 | 
						|
		id := g.Add(matcher)
 | 
						|
		ips := make([]net.Address, 0, len(mapping.Ip)+1)
 | 
						|
		if len(mapping.Ip) > 0 {
 | 
						|
			for _, ip := range mapping.Ip {
 | 
						|
				addr := net.IPAddress(ip)
 | 
						|
				if addr == nil {
 | 
						|
					return nil, newError("invalid IP address in static hosts: ", ip).AtWarning()
 | 
						|
				}
 | 
						|
				ips = append(ips, addr)
 | 
						|
			}
 | 
						|
		} else if len(mapping.ProxiedDomain) > 0 {
 | 
						|
			ips = append(ips, net.DomainAddress(mapping.ProxiedDomain))
 | 
						|
		} else {
 | 
						|
			return nil, newError("neither IP address nor proxied domain specified for domain: ", mapping.Domain).AtWarning()
 | 
						|
		}
 | 
						|
 | 
						|
		// Special handling for localhost IPv6. This is a dirty workaround as JSON config supports only single IP mapping.
 | 
						|
		if len(ips) == 1 && ips[0] == net.LocalHostIP {
 | 
						|
			ips = append(ips, net.LocalHostIPv6)
 | 
						|
		}
 | 
						|
 | 
						|
		sh.ips[id] = ips
 | 
						|
	}
 | 
						|
 | 
						|
	return sh, nil
 | 
						|
}
 | 
						|
 | 
						|
func filterIP(ips []net.Address, option IPOption) []net.Address {
 | 
						|
	filtered := make([]net.Address, 0, len(ips))
 | 
						|
	for _, ip := range ips {
 | 
						|
		if (ip.Family().IsIPv4() && option.IPv4Enable) || (ip.Family().IsIPv6() && option.IPv6Enable) {
 | 
						|
			filtered = append(filtered, ip)
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if len(filtered) == 0 {
 | 
						|
		return nil
 | 
						|
	}
 | 
						|
	return filtered
 | 
						|
}
 | 
						|
 | 
						|
// LookupIP returns IP address for the given domain, if exists in this StaticHosts.
 | 
						|
func (h *StaticHosts) LookupIP(domain string, option IPOption) []net.Address {
 | 
						|
	id := h.matchers.Match(domain)
 | 
						|
	if id == 0 {
 | 
						|
		return nil
 | 
						|
	}
 | 
						|
	ips := h.ips[id]
 | 
						|
	if len(ips) == 1 && ips[0].Family().IsDomain() {
 | 
						|
		return ips
 | 
						|
	}
 | 
						|
	return filterIP(ips, option)
 | 
						|
}
 |