ipnet_test.go 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133
  1. package net_test
  2. import (
  3. "net"
  4. "os"
  5. "path/filepath"
  6. "testing"
  7. proto "github.com/golang/protobuf/proto"
  8. "v2ray.com/core/app/router"
  9. "v2ray.com/core/common/platform"
  10. "v2ray.com/ext/sysio"
  11. "v2ray.com/core/common"
  12. . "v2ray.com/core/common/net"
  13. . "v2ray.com/ext/assert"
  14. )
  15. func parseCIDR(str string) *net.IPNet {
  16. _, ipNet, err := net.ParseCIDR(str)
  17. common.Must(err)
  18. return ipNet
  19. }
  20. func TestIPNet(t *testing.T) {
  21. assert := With(t)
  22. ipNet := NewIPNetTable()
  23. ipNet.Add(parseCIDR(("0.0.0.0/8")))
  24. ipNet.Add(parseCIDR(("10.0.0.0/8")))
  25. ipNet.Add(parseCIDR(("100.64.0.0/10")))
  26. ipNet.Add(parseCIDR(("127.0.0.0/8")))
  27. ipNet.Add(parseCIDR(("169.254.0.0/16")))
  28. ipNet.Add(parseCIDR(("172.16.0.0/12")))
  29. ipNet.Add(parseCIDR(("192.0.0.0/24")))
  30. ipNet.Add(parseCIDR(("192.0.2.0/24")))
  31. ipNet.Add(parseCIDR(("192.168.0.0/16")))
  32. ipNet.Add(parseCIDR(("198.18.0.0/15")))
  33. ipNet.Add(parseCIDR(("198.51.100.0/24")))
  34. ipNet.Add(parseCIDR(("203.0.113.0/24")))
  35. ipNet.Add(parseCIDR(("8.8.8.8/32")))
  36. ipNet.AddIP(net.ParseIP("91.108.4.0"), 16)
  37. assert(ipNet.Contains(ParseIP("192.168.1.1")), IsTrue)
  38. assert(ipNet.Contains(ParseIP("192.0.0.0")), IsTrue)
  39. assert(ipNet.Contains(ParseIP("192.0.1.0")), IsFalse)
  40. assert(ipNet.Contains(ParseIP("0.1.0.0")), IsTrue)
  41. assert(ipNet.Contains(ParseIP("1.0.0.1")), IsFalse)
  42. assert(ipNet.Contains(ParseIP("8.8.8.7")), IsFalse)
  43. assert(ipNet.Contains(ParseIP("8.8.8.8")), IsTrue)
  44. assert(ipNet.Contains(ParseIP("2001:cdba::3257:9652")), IsFalse)
  45. assert(ipNet.Contains(ParseIP("91.108.255.254")), IsTrue)
  46. }
  47. func TestGeoIPCN(t *testing.T) {
  48. assert := With(t)
  49. common.Must(sysio.CopyFile(platform.GetAssetLocation("geoip.dat"), filepath.Join(os.Getenv("GOPATH"), "src", "v2ray.com", "core", "release", "config", "geoip.dat")))
  50. ips, err := loadGeoIP("CN")
  51. common.Must(err)
  52. ipNet := NewIPNetTable()
  53. for _, ip := range ips {
  54. ipNet.AddIP(ip.Ip, byte(ip.Prefix))
  55. }
  56. assert(ipNet.Contains([]byte{8, 8, 8, 8}), IsFalse)
  57. }
  58. func loadGeoIP(country string) ([]*router.CIDR, error) {
  59. geoipBytes, err := sysio.ReadAsset("geoip.dat")
  60. if err != nil {
  61. return nil, err
  62. }
  63. var geoipList router.GeoIPList
  64. if err := proto.Unmarshal(geoipBytes, &geoipList); err != nil {
  65. return nil, err
  66. }
  67. for _, geoip := range geoipList.Entry {
  68. if geoip.CountryCode == country {
  69. return geoip.Cidr, nil
  70. }
  71. }
  72. panic("country not found: " + country)
  73. }
  74. func BenchmarkIPNetQuery(b *testing.B) {
  75. common.Must(sysio.CopyFile(platform.GetAssetLocation("geoip.dat"), filepath.Join(os.Getenv("GOPATH"), "src", "v2ray.com", "core", "release", "config", "geoip.dat")))
  76. ips, err := loadGeoIP("CN")
  77. common.Must(err)
  78. ipNet := NewIPNetTable()
  79. for _, ip := range ips {
  80. ipNet.AddIP(ip.Ip, byte(ip.Prefix))
  81. }
  82. b.ResetTimer()
  83. for i := 0; i < b.N; i++ {
  84. ipNet.Contains([]byte{8, 8, 8, 8})
  85. }
  86. }
  87. func BenchmarkCIDRQuery(b *testing.B) {
  88. common.Must(sysio.CopyFile(platform.GetAssetLocation("geoip.dat"), filepath.Join(os.Getenv("GOPATH"), "src", "v2ray.com", "core", "release", "config", "geoip.dat")))
  89. ips, err := loadGeoIP("CN")
  90. common.Must(err)
  91. ipNet := make([]*net.IPNet, 0, 1024)
  92. for _, ip := range ips {
  93. if len(ip.Ip) != 4 {
  94. continue
  95. }
  96. ipNet = append(ipNet, &net.IPNet{
  97. IP: net.IP(ip.Ip),
  98. Mask: net.CIDRMask(int(ip.Prefix), 32),
  99. })
  100. }
  101. b.ResetTimer()
  102. for i := 0; i < b.N; i++ {
  103. for _, n := range ipNet {
  104. if n.Contains([]byte{8, 8, 8, 8}) {
  105. break
  106. }
  107. }
  108. }
  109. }