ip.go 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. package task
  2. import (
  3. "bufio"
  4. "log"
  5. "math/rand"
  6. "net"
  7. "os"
  8. "strconv"
  9. "strings"
  10. "time"
  11. )
  12. const defaultInputFile = "ip.txt"
  13. var (
  14. // IPv6 IP version is 6
  15. IPv6 = false
  16. // TestAll test all ip
  17. TestAll = false
  18. // IPFile is the filename of IP Rangs
  19. IPFile = defaultInputFile
  20. )
  21. func InitRandSeed() {
  22. rand.Seed(time.Now().UnixNano())
  23. }
  24. func randIPEndWith(num byte) byte {
  25. return byte(rand.Intn(int(num)))
  26. }
  27. type IPRanges struct {
  28. ips []*net.IPAddr
  29. mask string
  30. firstIP net.IP
  31. ipNet *net.IPNet
  32. }
  33. func newIPRanges() *IPRanges {
  34. return &IPRanges{
  35. ips: make([]*net.IPAddr, 0),
  36. }
  37. }
  38. func (r *IPRanges) fixIP(ip string) string {
  39. // 如果不含有 '/' 则代表不是 IP 段,而是一个单独的 IP,因此需要加上 /32 /128 子网掩码
  40. if i := strings.IndexByte(ip, '/'); i < 0 {
  41. r.mask = "/32"
  42. if IPv6 {
  43. r.mask = "/128"
  44. }
  45. ip += r.mask
  46. } else {
  47. r.mask = ip[i:]
  48. }
  49. return ip
  50. }
  51. func (r *IPRanges) parseCIDR(ip string) {
  52. var err error
  53. if r.firstIP, r.ipNet, err = net.ParseCIDR(r.fixIP(ip)); err != nil {
  54. log.Fatalln("ParseCIDR err", err)
  55. }
  56. }
  57. func (r *IPRanges) appendIPv4(d byte) {
  58. r.appendIP(net.IPv4(r.firstIP[12], r.firstIP[13], r.firstIP[14], d))
  59. }
  60. func (r *IPRanges) appendIP(ip net.IP) {
  61. r.ips = append(r.ips, &net.IPAddr{IP: ip})
  62. }
  63. // 返回第四段 ip 的最小值及可用数目
  64. func (r *IPRanges) getIPRange() (minIP, hosts byte) {
  65. minIP = r.firstIP[15] & r.ipNet.Mask[3] // IP 第四段最小值
  66. // 根据子网掩码获取主机数量
  67. m := net.IPv4Mask(255, 255, 255, 255)
  68. for i, v := range r.ipNet.Mask {
  69. m[i] ^= v
  70. }
  71. total, _ := strconv.ParseInt(m.String(), 16, 32) // 总可用 IP 数
  72. if total > 255 { // 矫正 第四段 可用 IP 数
  73. hosts = 255
  74. return
  75. }
  76. if total == 0 {
  77. hosts = 1
  78. return
  79. }
  80. hosts = byte(total)
  81. return
  82. }
  83. func (r *IPRanges) chooseIPv4() {
  84. minIP, hosts := r.getIPRange()
  85. for r.ipNet.Contains(r.firstIP) {
  86. if TestAll { // 如果是测速全部 IP
  87. for i := 0; i <= int(hosts); i++ { // 遍历 IP 最后一段最小值到最大值
  88. r.appendIPv4(byte(i) + minIP)
  89. }
  90. } else { // 随机 IP 的最后一段 0.0.0.X
  91. r.appendIPv4(minIP + randIPEndWith(hosts))
  92. }
  93. r.firstIP[14]++ // 0.0.(X+1).X
  94. if r.firstIP[14] == 0 {
  95. r.firstIP[13]++ // 0.(X+1).X.X
  96. if r.firstIP[13] == 0 {
  97. r.firstIP[12]++ // (X+1).X.X.X
  98. }
  99. }
  100. }
  101. }
  102. func (r *IPRanges) chooseIPv6() {
  103. var tempIP uint8
  104. for r.ipNet.Contains(r.firstIP) {
  105. //fmt.Println(firstIP)
  106. //fmt.Println(firstIP[0], firstIP[1], firstIP[2], firstIP[3], firstIP[4], firstIP[5], firstIP[6], firstIP[7], firstIP[8], firstIP[9], firstIP[10], firstIP[11], firstIP[12], firstIP[13], firstIP[14], firstIP[15])
  107. if r.mask != "/128" {
  108. r.firstIP[15] = randIPEndWith(255) // 随机 IP 的最后一段
  109. r.firstIP[14] = randIPEndWith(255) // 随机 IP 的最后一段
  110. }
  111. targetIP := make([]byte, len(r.firstIP))
  112. copy(targetIP, r.firstIP)
  113. r.appendIP(targetIP)
  114. for i := 13; i >= 0; i-- {
  115. tempIP = r.firstIP[i]
  116. r.firstIP[i] += randIPEndWith(255)
  117. if r.firstIP[i] >= tempIP {
  118. break
  119. }
  120. }
  121. }
  122. }
  123. func loadIPRanges() []*net.IPAddr {
  124. if IPFile == "" {
  125. IPFile = defaultInputFile
  126. }
  127. file, err := os.Open(IPFile)
  128. if err != nil {
  129. log.Fatal(err)
  130. }
  131. defer file.Close()
  132. ranges := newIPRanges()
  133. scanner := bufio.NewScanner(file)
  134. for scanner.Scan() {
  135. ranges.parseCIDR(scanner.Text())
  136. if IPv6 {
  137. ranges.chooseIPv6()
  138. continue
  139. }
  140. ranges.chooseIPv4()
  141. }
  142. return ranges.ips
  143. }