func processGraphMessage(g *graph.Graph, m []byte) error { g.Lock() defer g.Unlock() var msg shttp.WSMessage if err := json.Unmarshal(m, &msg); err != nil { return err } if msg.Namespace != "Graph" { return nil } msgType, obj, err := graph.UnmarshalWSMessage(msg) if err != nil { return err } switch msgType { case "NodeUpdated": n := obj.(*graph.Node) node := g.GetNode(n.ID) if node != nil { g.SetMetadata(node, n.Metadata()) } case "NodeDeleted": g.DelNode(obj.(*graph.Node)) case "NodeAdded": n := obj.(*graph.Node) if g.GetNode(n.ID) == nil { g.AddNode(n) } case "EdgeUpdated": e := obj.(*graph.Edge) edge := g.GetEdge(e.ID) if edge != nil { g.SetMetadata(edge, e.Metadata()) } case "EdgeDeleted": g.DelEdge(obj.(*graph.Edge)) case "EdgeAdded": e := obj.(*graph.Edge) if g.GetEdge(e.ID) == nil { g.AddEdge(e) } } return nil }
func InjectPacket(pp *PacketParams, g *graph.Graph) error { srcdata := pp.SrcNode.Metadata() dstdata := pp.DstNode.Metadata() if _, ok := srcdata["IPV4"]; !ok { return errors.New("Source Node doesn't have an IP") } srcIP := getIP(srcdata["IPV4"].(string)) if srcIP == nil { return errors.New("Source Node doesn't have proper IP") } if _, ok := dstdata["IPV4"]; !ok { return errors.New("Destination Node doesn't have an IP") } dstIP := getIP(dstdata["IPV4"].(string)) if dstIP == nil { return errors.New("Destination Node doesn't have proper IP") } srcMAC, err := net.ParseMAC(srcdata["MAC"].(string)) if err != nil || srcMAC == nil { return errors.New("Source Node doesn't have proper MAC") } dstMAC, err := net.ParseMAC(dstdata["MAC"].(string)) if err != nil || dstMAC == nil { return errors.New("Destination Node doesn't have proper MAC") } //create packet buffer := gopacket.NewSerializeBuffer() ipLayer := &layers.IPv4{Version: 4, SrcIP: srcIP, DstIP: dstIP} ethLayer := &layers.Ethernet{EthernetType: layers.EthernetTypeIPv4, SrcMAC: srcMAC, DstMAC: dstMAC} switch pp.Type { case "icmp": ipLayer.Protocol = layers.IPProtocolICMPv4 gopacket.SerializeLayers(buffer, options, ethLayer, ipLayer, &layers.ICMPv4{ TypeCode: layers.CreateICMPv4TypeCode(layers.ICMPv4TypeEchoRequest, 0), }, gopacket.Payload([]byte(pp.Payload)), ) default: return fmt.Errorf("Unsupported traffic type '%s'", pp.Type) } g.RLock() srcNode := g.GetNode(pp.SrcNode.ID) if srcNode == nil { g.RUnlock() return errors.New("Unable to find source node") } nscontext, err := topology.NewNetNSContextByNode(g, srcNode) defer nscontext.Close() g.RUnlock() if err != nil { return err } handle, err := pcap.OpenLive(srcdata["Name"].(string), 1024, false, 2000) if err != nil { return fmt.Errorf("Unable to open the source node: %s", err.Error()) } defer handle.Close() packet := buffer.Bytes() for i := 0; i < pp.Count; i++ { if err := handle.WritePacketData(packet); err != nil { return fmt.Errorf("Write error: %s", err.Error()) } } return nil }