Browse Source

rebuild ipranges

mz 4 years ago
parent
commit
fbaa312622
1 changed files with 140 additions and 143 deletions
  1. 140 143
      task/ip.go

+ 140 - 143
task/ip.go

@@ -2,6 +2,7 @@ package task
 
 import (
 	"bufio"
+	"fmt"
 	"log"
 	"math/rand"
 	"net"
@@ -22,200 +23,196 @@ var (
 	IPFile = defaultInputFile
 )
 
-func randipEndWith(num int) uint8 {
+func randIPEndWith(num byte) byte {
 	rand.Seed(time.Now().UnixNano())
-	return uint8(rand.Intn(num))
+	return byte(rand.Intn(int(num)))
 }
 
-func loadIPRanges() []*net.IPAddr {
-	if IPFile == "" {
-		IPFile = defaultInputFile
-	}
-	file, err := os.Open(IPFile)
-	if err != nil {
-		log.Fatal(err)
+type IPRanges struct {
+	ips     []*net.IPAddr
+	mask    string
+	firstIP net.IP
+	ipNet   *net.IPNet
+}
+
+func newIPRanges() *IPRanges {
+	return &IPRanges{
+		ips: make([]*net.IPAddr, 0),
 	}
-	defer file.Close()
-	firstIPs := make([]*net.IPAddr, 0)
-	scanner := bufio.NewScanner(file)
-	for scanner.Scan() {
-		ipString := scanner.Text()
-		// 如果不含有 '/' 则代表不是 IP 段,而是一个单独的 IP,因此需要加上 /32 /128 子网掩码
-		if !strings.Contains(ipString, "/") {
-			mask := "/32"
-			if IPv6 {
-				mask = "/128"
-			}
-			ipString += mask
-		}
+}
+
+func (r *IPRanges) fixIP(ip string) string {
+	// 如果不含有 '/' 则代表不是 IP 段,而是一个单独的 IP,因此需要加上 /32 /128 子网掩码
+	if i := strings.IndexByte(ip, '/'); i < 0 {
+		r.mask = "/32"
 		if IPv6 {
-			//IPv6
-			firstIPs = append(firstIPs, loadIPv6(ipString)...)
-			continue
+			r.mask = "/128"
 		}
-		// IPv4
-		firstIPs = append(firstIPs, loadIPv4(ipString)...)
+		ip += r.mask
+	} else {
+		r.mask = ip[i:]
 	}
-	return firstIPs
+	return ip
 }
 
-func loadIPv4(ipString string) (firstIPs []*net.IPAddr) {
-	firstIP, IPRange, err := net.ParseCIDR(ipString)
-	// fmt.Println(firstIP)
-	// fmt.Println(IPRange)
-	if err != nil {
-		log.Fatal(err)
+func (r *IPRanges) parseCIDR(ip string) {
+	var err error
+	if r.firstIP, r.ipNet, err = net.ParseCIDR(r.fixIP(ip)); err != nil {
+		log.Fatalln("ParseCIDR err", err)
+	}
+}
+
+func (r *IPRanges) appendIPv4(d byte) {
+	r.appendIP(net.IPv4(r.firstIP[12], r.firstIP[13], r.firstIP[14], d))
+}
+
+func (r *IPRanges) appendIP(ip net.IP) {
+	r.ips = append(r.ips, &net.IPAddr{IP: ip})
+}
+
+// 返回第四段 ip 的最小值及可用数目
+func (r *IPRanges) getIPRange() (minIP, hosts byte) {
+	minIP = r.firstIP[15] & r.ipNet.Mask[3] // IP 第四段最小值
+
+	// 根据子网掩码获取主机数量
+	m := net.IPv4Mask(255, 255, 255, 255)
+	for i, v := range r.ipNet.Mask {
+		m[i] ^= v
+	}
+	total, _ := strconv.ParseInt(m.String(), 16, 32) // 总可用 IP 数
+	if total > 255 {                                 // 矫正 第四段 可用 IP 数
+		hosts = 255
+		return
+	}
+	if total == 0 {
+		hosts = 1
+		return
 	}
-	minIP, maxIP, hostNum := getCidrIPRange(ipString) // 获取 IP 最后一段最小值和最大值
-	for IPRange.Contains(firstIP) {
+	hosts = byte(total)
+	return
+}
+
+func (r *IPRanges) chooseIPv4() {
+	minIP, hosts := r.getIPRange()
+	fmt.Println(minIP, hosts)
+	for r.ipNet.Contains(r.firstIP) {
 		if TestAll { // 如果是测速全部 IP
-			for i := minIP; i <= maxIP; i++ { // 遍历 IP 最后一段最小值到最大值
-				firstIP[15] = i
-				firstIPCopy := make([]byte, len(firstIP))
-				copy(firstIPCopy, firstIP)
-				firstIPs = append(firstIPs, &net.IPAddr{IP: firstIPCopy})
+			for i := byte(0); i <= hosts; i++ { // 遍历 IP 最后一段最小值到最大值
+				r.appendIPv4(i + minIP)
 			}
 		} else { // 随机 IP 的最后一段 0.0.0.X
-			firstIP[15] = minIP + randipEndWith(hostNum)
-			firstIPCopy := make([]byte, len(firstIP))
-			copy(firstIPCopy, firstIP)
-			firstIPs = append(firstIPs, &net.IPAddr{IP: firstIPCopy})
-		}
-		firstIP[14]++ // 0.0.(X+1).X
-		if firstIP[14] == 0 {
-			firstIP[13]++ // 0.(X+1).X.X
-			if firstIP[13] == 0 {
-				firstIP[12]++ // (X+1).X.X.X
+			r.appendIPv4(minIP + randIPEndWith(hosts))
+		}
+		r.firstIP[14]++ // 0.0.(X+1).X
+		if r.firstIP[14] == 0 {
+			r.firstIP[13]++ // 0.(X+1).X.X
+			if r.firstIP[13] == 0 {
+				r.firstIP[12]++ // (X+1).X.X.X
 			}
 		}
 	}
-	return
 }
 
-func loadIPv6(ipString string) (firstIPs []*net.IPAddr) {
-	firstIP, IPRange, err := net.ParseCIDR(ipString)
-	// fmt.Println(firstIP)
-	// fmt.Println(IPRange)
-	if err != nil {
-		log.Fatal(err)
-	}
+func (r *IPRanges) chooseIPv6() {
 	var tempIP uint8
-	for IPRange.Contains(firstIP) {
+	for r.ipNet.Contains(r.firstIP) {
 		//fmt.Println(firstIP)
 		//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])
-		if !strings.Contains(ipString, "/128") {
-			firstIP[15] = randipEndWith(255) // 随机 IP 的最后一段
-			firstIP[14] = randipEndWith(255) // 随机 IP 的最后一段
-		}
-		firstIPCopy := make([]byte, len(firstIP))
-		copy(firstIPCopy, firstIP)
-		firstIPs = append(firstIPs, &net.IPAddr{IP: firstIPCopy})
-		tempIP = firstIP[13]
-		firstIP[13] += randipEndWith(255)
-		if firstIP[13] >= tempIP {
+		if r.mask == "/128" {
+			r.firstIP[15] = randIPEndWith(255) // 随机 IP 的最后一段
+			r.firstIP[14] = randIPEndWith(255) // 随机 IP 的最后一段
+		}
+		firstIPCopy := make([]byte, len(r.firstIP))
+		copy(firstIPCopy, r.firstIP)
+		r.appendIP(firstIPCopy)
+		tempIP = r.firstIP[13]
+		r.firstIP[13] += randIPEndWith(255)
+		if r.firstIP[13] >= tempIP {
 			continue
 		}
-		tempIP = firstIP[12]
-		firstIP[12] += randipEndWith(255)
-		if firstIP[12] >= tempIP {
+		tempIP = r.firstIP[12]
+		r.firstIP[12] += randIPEndWith(255)
+		if r.firstIP[12] >= tempIP {
 			continue
 		}
-		tempIP = firstIP[11]
-		firstIP[11] += randipEndWith(255)
-		if firstIP[11] >= tempIP {
+		tempIP = r.firstIP[11]
+		r.firstIP[11] += randIPEndWith(255)
+		if r.firstIP[11] >= tempIP {
 			continue
 		}
-		tempIP = firstIP[10]
-		firstIP[10] += randipEndWith(255)
-		if firstIP[10] >= tempIP {
+		tempIP = r.firstIP[10]
+		r.firstIP[10] += randIPEndWith(255)
+		if r.firstIP[10] >= tempIP {
 			continue
 		}
-		tempIP = firstIP[9]
-		firstIP[9] += randipEndWith(255)
-		if firstIP[9] >= tempIP {
+		tempIP = r.firstIP[9]
+		r.firstIP[9] += randIPEndWith(255)
+		if r.firstIP[9] >= tempIP {
 			continue
 		}
-		tempIP = firstIP[8]
-		firstIP[8] += randipEndWith(255)
-		if firstIP[8] >= tempIP {
+		tempIP = r.firstIP[8]
+		r.firstIP[8] += randIPEndWith(255)
+		if r.firstIP[8] >= tempIP {
 			continue
 		}
-		tempIP = firstIP[7]
-		firstIP[7] += randipEndWith(255)
-		if firstIP[7] >= tempIP {
+		tempIP = r.firstIP[7]
+		r.firstIP[7] += randIPEndWith(255)
+		if r.firstIP[7] >= tempIP {
 			continue
 		}
-		tempIP = firstIP[6]
-		firstIP[6] += randipEndWith(255)
-		if firstIP[6] >= tempIP {
+		tempIP = r.firstIP[6]
+		r.firstIP[6] += randIPEndWith(255)
+		if r.firstIP[6] >= tempIP {
 			continue
 		}
-		tempIP = firstIP[5]
-		firstIP[5] += randipEndWith(255)
-		if firstIP[5] >= tempIP {
+		tempIP = r.firstIP[5]
+		r.firstIP[5] += randIPEndWith(255)
+		if r.firstIP[5] >= tempIP {
 			continue
 		}
-		tempIP = firstIP[4]
-		firstIP[4] += randipEndWith(255)
-		if firstIP[4] >= tempIP {
+		tempIP = r.firstIP[4]
+		r.firstIP[4] += randIPEndWith(255)
+		if r.firstIP[4] >= tempIP {
 			continue
 		}
-		tempIP = firstIP[3]
-		firstIP[3] += randipEndWith(255)
-		if firstIP[3] >= tempIP {
+		tempIP = r.firstIP[3]
+		r.firstIP[3] += randIPEndWith(255)
+		if r.firstIP[3] >= tempIP {
 			continue
 		}
-		tempIP = firstIP[2]
-		firstIP[2] += randipEndWith(255)
-		if firstIP[2] >= tempIP {
+		tempIP = r.firstIP[2]
+		r.firstIP[2] += randIPEndWith(255)
+		if r.firstIP[2] >= tempIP {
 			continue
 		}
-		tempIP = firstIP[1]
-		firstIP[1] += randipEndWith(255)
-		if firstIP[1] >= tempIP {
+		tempIP = r.firstIP[1]
+		r.firstIP[1] += randIPEndWith(255)
+		if r.firstIP[1] >= tempIP {
 			continue
 		}
-		tempIP = firstIP[0]
-		firstIP[0] += randipEndWith(255)
+		tempIP = r.firstIP[0]
+		r.firstIP[0] += randIPEndWith(255)
 	}
-	return
 }
 
-// 根据子网掩码获取主机数量
-func getCIDRHostNum(mask uint8) (subnetNum int) {
-	if mask >= 32 {
-		return 1
+func loadIPRanges() []*net.IPAddr {
+	if IPFile == "" {
+		IPFile = defaultInputFile
 	}
-
-	if mask < 32 {
-		for i := int(32 - mask - 1); i >= 1; i-- {
-			subnetNum += 1 << i
-		}
-		subnetNum += 2
+	file, err := os.Open(IPFile)
+	if err != nil {
+		log.Fatal(err)
 	}
-	if subnetNum > 0xFF {
-		subnetNum = 0xFF
+	defer file.Close()
+	ranges := newIPRanges()
+	scanner := bufio.NewScanner(file)
+	for scanner.Scan() {
+		ranges.parseCIDR(scanner.Text())
+		if IPv6 {
+			ranges.chooseIPv6()
+			continue
+		}
+		ranges.chooseIPv4()
 	}
-	return
-}
-
-// 获取 IP 最后一段最小值和最大值、主机数量
-func getCidrIPRange(cidr string) (minIP, maxIP uint8, ipNum int) {
-	i := strings.IndexByte(cidr, '/')
-	addr := cidr[:i]
-	mask, _ := strconv.ParseUint(cidr[i+1:], 10, 8)
-	i = strings.LastIndexByte(addr, '.')
-	seg4, _ := strconv.ParseUint(addr[i+1:], 10, 8)
-	minIP, maxIP = getIPSegRange(uint8(seg4), uint8(32-mask))
-	ipNum = getCIDRHostNum(uint8(mask))
-	return
-}
-
-// 根据输入的基础IP地址和CIDR掩码计算一个IP片段的区间
-func getIPSegRange(userSegIP, offset uint8) (uint8, uint8) {
-	var ipSegMax uint8 = 0xFF
-	netSegIP := ipSegMax << offset
-	segMinIP := netSegIP & userSegIP
-	segMaxIP := userSegIP&(0xFF<<offset) | ^(0xFF << offset)
-	return segMinIP, segMaxIP
+	return ranges.ips
 }