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