mirror of https://github.com/v2ray/v2ray-core
				
				
				
			
		
			
				
	
	
		
			194 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			Go
		
	
	
			
		
		
	
	
			194 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			Go
		
	
	
// +build !confonly
 | 
						|
 | 
						|
package router
 | 
						|
 | 
						|
import (
 | 
						|
	"encoding/binary"
 | 
						|
	"sort"
 | 
						|
 | 
						|
	"v2ray.com/core/common/net"
 | 
						|
)
 | 
						|
 | 
						|
type ipv6 struct {
 | 
						|
	a uint64
 | 
						|
	b uint64
 | 
						|
}
 | 
						|
 | 
						|
type GeoIPMatcher struct {
 | 
						|
	countryCode string
 | 
						|
	ip4         []uint32
 | 
						|
	prefix4     []uint8
 | 
						|
	ip6         []ipv6
 | 
						|
	prefix6     []uint8
 | 
						|
}
 | 
						|
 | 
						|
func normalize4(ip uint32, prefix uint8) uint32 {
 | 
						|
	return (ip >> (32 - prefix)) << (32 - prefix)
 | 
						|
}
 | 
						|
 | 
						|
func normalize6(ip ipv6, prefix uint8) ipv6 {
 | 
						|
	if prefix <= 64 {
 | 
						|
		ip.a = (ip.a >> (64 - prefix)) << (64 - prefix)
 | 
						|
		ip.b = 0
 | 
						|
	} else {
 | 
						|
		ip.b = (ip.b >> (128 - prefix)) << (128 - prefix)
 | 
						|
	}
 | 
						|
	return ip
 | 
						|
}
 | 
						|
 | 
						|
func (m *GeoIPMatcher) Init(cidrs []*CIDR) error {
 | 
						|
	ip4Count := 0
 | 
						|
	ip6Count := 0
 | 
						|
 | 
						|
	for _, cidr := range cidrs {
 | 
						|
		ip := cidr.Ip
 | 
						|
		switch len(ip) {
 | 
						|
		case 4:
 | 
						|
			ip4Count++
 | 
						|
		case 16:
 | 
						|
			ip6Count++
 | 
						|
		default:
 | 
						|
			return newError("unexpect ip length: ", len(ip))
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	cidrList := CIDRList(cidrs)
 | 
						|
	sort.Sort(&cidrList)
 | 
						|
 | 
						|
	m.ip4 = make([]uint32, 0, ip4Count)
 | 
						|
	m.prefix4 = make([]uint8, 0, ip4Count)
 | 
						|
	m.ip6 = make([]ipv6, 0, ip6Count)
 | 
						|
	m.prefix6 = make([]uint8, 0, ip6Count)
 | 
						|
 | 
						|
	for _, cidr := range cidrs {
 | 
						|
		ip := cidr.Ip
 | 
						|
		prefix := uint8(cidr.Prefix)
 | 
						|
		switch len(ip) {
 | 
						|
		case 4:
 | 
						|
			m.ip4 = append(m.ip4, normalize4(binary.BigEndian.Uint32(ip), prefix))
 | 
						|
			m.prefix4 = append(m.prefix4, prefix)
 | 
						|
		case 16:
 | 
						|
			ip6 := ipv6{
 | 
						|
				a: binary.BigEndian.Uint64(ip[0:8]),
 | 
						|
				b: binary.BigEndian.Uint64(ip[8:16]),
 | 
						|
			}
 | 
						|
			ip6 = normalize6(ip6, prefix)
 | 
						|
 | 
						|
			m.ip6 = append(m.ip6, ip6)
 | 
						|
			m.prefix6 = append(m.prefix6, prefix)
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (m *GeoIPMatcher) match4(ip uint32) bool {
 | 
						|
	if len(m.ip4) == 0 {
 | 
						|
		return false
 | 
						|
	}
 | 
						|
 | 
						|
	if ip < m.ip4[0] {
 | 
						|
		return false
 | 
						|
	}
 | 
						|
 | 
						|
	size := uint32(len(m.ip4))
 | 
						|
	l := uint32(0)
 | 
						|
	r := size
 | 
						|
	for l < r {
 | 
						|
		x := ((l + r) >> 1)
 | 
						|
		if ip < m.ip4[x] {
 | 
						|
			r = x
 | 
						|
			continue
 | 
						|
		}
 | 
						|
 | 
						|
		nip := normalize4(ip, m.prefix4[x])
 | 
						|
		if nip == m.ip4[x] {
 | 
						|
			return true
 | 
						|
		}
 | 
						|
 | 
						|
		l = x + 1
 | 
						|
	}
 | 
						|
 | 
						|
	return l > 0 && normalize4(ip, m.prefix4[l-1]) == m.ip4[l-1]
 | 
						|
}
 | 
						|
 | 
						|
func less6(a ipv6, b ipv6) bool {
 | 
						|
	return a.a < b.a || (a.a == b.a && a.b < b.b)
 | 
						|
}
 | 
						|
 | 
						|
func (m *GeoIPMatcher) match6(ip ipv6) bool {
 | 
						|
	if len(m.ip6) == 0 {
 | 
						|
		return false
 | 
						|
	}
 | 
						|
 | 
						|
	if less6(ip, m.ip6[0]) {
 | 
						|
		return false
 | 
						|
	}
 | 
						|
 | 
						|
	size := uint32(len(m.ip6))
 | 
						|
	l := uint32(0)
 | 
						|
	r := size
 | 
						|
	for l < r {
 | 
						|
		x := (l + r) / 2
 | 
						|
		if less6(ip, m.ip6[x]) {
 | 
						|
			r = x
 | 
						|
			continue
 | 
						|
		}
 | 
						|
 | 
						|
		if normalize6(ip, m.prefix6[x]) == m.ip6[x] {
 | 
						|
			return true
 | 
						|
		}
 | 
						|
 | 
						|
		l = x + 1
 | 
						|
	}
 | 
						|
 | 
						|
	return l > 0 && normalize6(ip, m.prefix6[l-1]) == m.ip6[l-1]
 | 
						|
}
 | 
						|
 | 
						|
// Match returns true if the given ip is included by the GeoIP.
 | 
						|
func (m *GeoIPMatcher) Match(ip net.IP) bool {
 | 
						|
	switch len(ip) {
 | 
						|
	case 4:
 | 
						|
		return m.match4(binary.BigEndian.Uint32(ip))
 | 
						|
	case 16:
 | 
						|
		return m.match6(ipv6{
 | 
						|
			a: binary.BigEndian.Uint64(ip[0:8]),
 | 
						|
			b: binary.BigEndian.Uint64(ip[8:16]),
 | 
						|
		})
 | 
						|
	default:
 | 
						|
		return false
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// GeoIPMatcherContainer is a container for GeoIPMatchers. It keeps unique copies of GeoIPMatcher by country code.
 | 
						|
type GeoIPMatcherContainer struct {
 | 
						|
	matchers []*GeoIPMatcher
 | 
						|
}
 | 
						|
 | 
						|
// Add adds a new GeoIP set into the container.
 | 
						|
// If the country code of GeoIP is not empty, GeoIPMatcherContainer will try to find an existing one, instead of adding a new one.
 | 
						|
func (c *GeoIPMatcherContainer) Add(geoip *GeoIP) (*GeoIPMatcher, error) {
 | 
						|
	if len(geoip.CountryCode) > 0 {
 | 
						|
		for _, m := range c.matchers {
 | 
						|
			if m.countryCode == geoip.CountryCode {
 | 
						|
				return m, nil
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	m := &GeoIPMatcher{
 | 
						|
		countryCode: geoip.CountryCode,
 | 
						|
	}
 | 
						|
	if err := m.Init(geoip.Cidr); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	if len(geoip.CountryCode) > 0 {
 | 
						|
		c.matchers = append(c.matchers, m)
 | 
						|
	}
 | 
						|
	return m, nil
 | 
						|
}
 | 
						|
 | 
						|
var (
 | 
						|
	globalGeoIPContainer GeoIPMatcherContainer
 | 
						|
)
 |