package main

import (
	"fmt"
	"net"
	"log"
	"flag"
	"math"
	"github.com/google/gopacket/pcap"
	"github.com/google/gopacket/layers"
	"github.com/google/gopacket"
	manuf "github.com/timest/gomanuf"
	"context"
	"time"
	"sync"
	"runtime"
	"bytes"
)

type info struct {
	Mac net.HardwareAddr
	Manuf string
}

type send struct {
	ips []net.IP
	ipNet *net.IPNet
	iface net.Interface
}

var ( 
	infoSet = make(map[string]info)
	ch = make(chan bool)
)

func getIPSet(ipNet *net.IPNet) (ipSet []net.IP) {
	var ipStringSet []string
	for ip := ipNet.IP.Mask(ipNet.Mask); ipNet.Contains(ip); nextIP(ip) {
		if ip[len(ip) - 1] & 0xff == 0 {
			continue	
		}
		ipStringSet = append(ipStringSet, ip.String())
	}
	for _, ipString := range ipStringSet {
		ip := net.ParseIP(ipString)
		if ip != nil {
			ipSet = append(ipSet, ip)
		} 
	}
	return
}

func nextIP(ip net.IP) {
	for i := len(ip) - 1; i >= 0; i-- {
		ip[i]++
		if ip[i] > 0 {
			break
		}
	}
}

func (sendInfo *send)sendArpPacket(handle *pcap.Handle, ip net.IP) {
	localHaddr := sendInfo.iface.HardwareAddr
	srcIP := sendInfo.ipNet.IP.To4()
	dstIP := ip.To4()
	//fmt.Println(srcIP.String())
	//fmt.Println(dstIP.String())

	if srcIP == nil || dstIP == nil {
        log.Fatal("source address or destination address is empty!")
    }

	eth := &layers.Ethernet {
		SrcMAC: localHaddr, 
		DstMAC: net.HardwareAddr{0xff, 0xff, 0xff, 0xff, 0xff, 0xff}, 
		EthernetType: layers.EthernetTypeARP,	
	}

	a := &layers.ARP {
		AddrType: layers.LinkTypeEthernet, 
		Protocol: layers.EthernetTypeIPv4,
		HwAddressSize: uint8(6), 
		ProtAddressSize: uint8(4),
		Operation: uint16(1),
		SourceHwAddress: localHaddr, 
		SourceProtAddress: srcIP,
		DstHwAddress: net.HardwareAddr{0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, 
		DstProtAddress: dstIP,	
	}

	buffer := gopacket.NewSerializeBuffer()
	var opt gopacket.SerializeOptions
	gopacket.SerializeLayers(buffer, opt, eth, a)
	outgoingPacket := buffer.Bytes()

	err := handle.WritePacketData(outgoingPacket)
	if err != nil {
		log.Fatal("Failed to send package")	
	}
}

func listenArpPacket(handle *pcap.Handle, ctx context.Context, iface net.Interface) {
	ps := gopacket.NewPacketSource(handle, handle.LinkType())

	for {
		select {
		case <-ctx.Done():
			return
		case p := <-ps.Packets():
			arpLayer := p.Layer(layers.LayerTypeARP)
			if arpLayer != nil {
				arp, _ := arpLayer.(*layers.ARP)
				if arpLayer == nil {
					continue
				}
				if arp.Operation != layers.ARPReply || bytes.Equal([]byte(iface.HardwareAddr), arp.SourceHwAddress) {
				// This is a packet I sent.
					continue
				}
				if arp.Operation == layers.ARPReply {
					mac := net.HardwareAddr(arp.SourceHwAddress)
					m := manuf.Search(mac.String())
					if _, ok := infoSet[net.IP(arp.SourceProtAddress).String()]; !ok {
						ch<-true
						infoSet[net.IP(arp.SourceProtAddress).String()] = info{mac, m}
						ch<-false
					}
				}
			}	
		}
	}
}

func main() {
	runtime.GOMAXPROCS(runtime.NumCPU())
	ifaceName := flag.String("i", "", "NetWork interface name")
	flag.Parse()
	usageTime := time.Now()
	
	var ifaceSet []net.Interface
	var err error

	if *ifaceName == "" {
		ifaceSet, err = net.Interfaces()
	} else {
		iface, err := net.InterfaceByName(*ifaceName)
		if err == nil {
			ifaceSet = append(ifaceSet, *iface)
		}
	}
	if err != nil {
		log.Fatal(err)
	}	
	
	sendInfo := &send{}
	
Loop:
	for _, it := range ifaceSet {
		if it.Flags & net.FlagUp == 0 {
			continue
		} else if it.Flags & net.FlagLoopback != 0 {
			continue
		}
		addrs, _ := it.Addrs()
		for _, addr := range addrs {
			if ip, ok := addr.(*net.IPNet); ok {
				if ip.IP.To4() != nil {
					sendInfo.ips = getIPSet(ip)
					sendInfo.ipNet = ip
					sendInfo.iface = it
					break Loop
				}
			}	
		}	
	}

	handle, err := pcap.OpenLive(sendInfo.iface.Name, 65536, true, pcap.BlockForever)
	if err != nil {
		log.Fatal("pcap open fail, err:", err)
	}
	defer handle.Close()

	ctx, cancel := context.WithCancel(context.Background())
	go listenArpPacket(handle, ctx, sendInfo.iface)

	maskSize, _ := sendInfo.ipNet.Mask.Size()

	fmt.Println("************【information of interface】************")
	fmt.Printf("	interfaceName : %v\n", sendInfo.iface.Name)
	fmt.Printf(" 	interfaceIP   : %v/%d\n", sendInfo.ipNet.IP, maskSize)
	fmt.Printf("*	interfaceMAC  : %v %10v\n", sendInfo.iface.HardwareAddr, "*")
	fmt.Println("****************************************************\n")

	//发送ARP包

	interval := 1
	processNum := 300
	wg := &sync.WaitGroup{}

	if len(sendInfo.ips) <= processNum {
		processNum = len(sendInfo.ips)
	} else {
		interval = int(math.Ceil(float64(len(sendInfo.ips)) / float64(processNum)))
	}

	go func() {
		for i := 0; i < len(sendInfo.ips); i += interval {
			length := i + interval
			if length >= len(sendInfo.ips) {
				length = len(sendInfo.ips)
			}
			wg.Add(1)
			go func(ips []net.IP) {
				defer wg.Done()
				for _, ip := range ips {
					sendInfo.sendArpPacket(handle, ip)
				}
			}(sendInfo.ips[i:length])
		}
		wg.Wait()
	}()
	
	t := time.NewTicker(10 * time.Second)
	for {
		select {
		case <-t.C:
			for ip, data := range infoSet {
				fmt.Printf("%-15s %-20s %-40s\n", ip, data.Mac, data.Manuf)
			}
			cancel()
			fmt.Printf("\nUsage time: %v\n\n", time.Since(usageTime))
			return
		case timeStamp := <-ch:
			if timeStamp {
				t.Stop()
			} else {
				t = time.NewTicker(1 * time.Second)
			}
		}
	}
}