217 lines
5.8 KiB
Go
217 lines
5.8 KiB
Go
|
// Copyright 2018 Google Inc.
|
||
|
//
|
||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
// you may not use this file except in compliance with the License.
|
||
|
// You may obtain a copy of the License at
|
||
|
//
|
||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||
|
//
|
||
|
// Unless required by applicable law or agreed to in writing, software
|
||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
// See the License for the specific language governing permissions and
|
||
|
// limitations under the License.
|
||
|
|
||
|
// Package pcapreplayer provides a net.PacketConn which replays a pcap file, and
|
||
|
// optionally records a new golden file.
|
||
|
package pcapreplayer
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"net"
|
||
|
"os"
|
||
|
"path/filepath"
|
||
|
"time"
|
||
|
|
||
|
"github.com/d2g/dhcp4client"
|
||
|
"github.com/google/gopacket"
|
||
|
"github.com/google/gopacket/layers"
|
||
|
"github.com/google/gopacket/pcapgo"
|
||
|
)
|
||
|
|
||
|
func pcapopen(input, output string) (*pcapgo.Reader, *pcapgo.Writer, error) {
|
||
|
f, err := os.Open(input)
|
||
|
if err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
pcapr, err := pcapgo.NewReader(f)
|
||
|
if err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
|
||
|
var pcapw *pcapgo.Writer
|
||
|
if output != "" {
|
||
|
if err := os.MkdirAll(filepath.Dir(output), 0755); err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
of, err := os.Create(output)
|
||
|
if err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
pcapw = pcapgo.NewWriter(of)
|
||
|
if err := pcapw.WriteFileHeader(1600, layers.LinkTypeEthernet); err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
}
|
||
|
return pcapr, pcapw, nil
|
||
|
}
|
||
|
|
||
|
func readFrom(r *pcapgo.Reader, buf []byte) (int, net.IP, error) {
|
||
|
data, _, err := r.ReadPacketData()
|
||
|
if err != nil {
|
||
|
return 0, nil, err
|
||
|
}
|
||
|
pkt := gopacket.NewPacket(data, layers.LayerTypeEthernet, gopacket.DecodeOptions{})
|
||
|
// TODO: get source IP
|
||
|
udp := pkt.Layer(layers.LayerTypeUDP)
|
||
|
if udp == nil {
|
||
|
return 0, nil, fmt.Errorf("pcap contained unexpected non-UDP packet")
|
||
|
}
|
||
|
|
||
|
//log.Printf("ReadFrom(): %x, %v, pkt = %+v", udp.LayerPayload(), err, pkt)
|
||
|
copy(buf, udp.LayerPayload())
|
||
|
return len(udp.LayerPayload()), net.ParseIP("192.168.23.1"), err
|
||
|
}
|
||
|
|
||
|
func mustParseMAC(s string) net.HardwareAddr {
|
||
|
hw, err := net.ParseMAC(s)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
return hw
|
||
|
}
|
||
|
|
||
|
type layer interface {
|
||
|
gopacket.NetworkLayer
|
||
|
|
||
|
// Cannot embed gopacket.SerializableLayer because it includes a conflicting
|
||
|
// definition of LayerType():
|
||
|
SerializeTo(b gopacket.SerializeBuffer, opts gopacket.SerializeOptions) error
|
||
|
}
|
||
|
|
||
|
func pcapwrite(w *pcapgo.Writer, ip layer, udp *layers.UDP, b []byte) error {
|
||
|
buf := gopacket.NewSerializeBuffer()
|
||
|
udp.SetNetworkLayerForChecksum(ip)
|
||
|
var ethernetType layers.EthernetType
|
||
|
switch ip.LayerType() {
|
||
|
case layers.LayerTypeIPv4:
|
||
|
ethernetType = layers.EthernetTypeIPv4
|
||
|
case layers.LayerTypeIPv6:
|
||
|
ethernetType = layers.EthernetTypeIPv6
|
||
|
}
|
||
|
gopacket.SerializeLayers(buf,
|
||
|
gopacket.SerializeOptions{
|
||
|
FixLengths: true,
|
||
|
ComputeChecksums: true,
|
||
|
},
|
||
|
&layers.Ethernet{
|
||
|
SrcMAC: mustParseMAC("00:00:5E:00:53:00"), // RFC7042
|
||
|
DstMAC: mustParseMAC("33:33:00:01:00:02"), // IPv6mcast_01:00:02
|
||
|
EthernetType: ethernetType,
|
||
|
},
|
||
|
ip,
|
||
|
udp,
|
||
|
gopacket.Payload(b),
|
||
|
)
|
||
|
got := buf.Bytes()
|
||
|
|
||
|
ci := gopacket.CaptureInfo{
|
||
|
CaptureLength: len(got),
|
||
|
Length: len(got),
|
||
|
}
|
||
|
if err := w.WritePacket(ci, got); err != nil {
|
||
|
return fmt.Errorf("pcap.WritePacket(): %v", err)
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// packetConn is a net.PacketConn which replays a pcap file.
|
||
|
type packetConn struct {
|
||
|
pcapr *pcapgo.Reader
|
||
|
pcapw *pcapgo.Writer
|
||
|
}
|
||
|
|
||
|
// NewPCAP returns a net.PacketConn which replays packets from pcap file input,
|
||
|
// writing packets to pcap file output (if non-empty).
|
||
|
//
|
||
|
// See https://en.wikipedia.org/wiki/Pcap for details on pcap.
|
||
|
func NewPacketConn(input, output string) (net.PacketConn, error) {
|
||
|
pcapr, pcapw, err := pcapopen(input, output)
|
||
|
return &packetConn{pcapr, pcapw}, err
|
||
|
}
|
||
|
|
||
|
func (r *packetConn) LocalAddr() net.Addr { return nil }
|
||
|
func (r *packetConn) Close() error { return nil }
|
||
|
func (r *packetConn) SetDeadline(t time.Time) error { return nil }
|
||
|
func (r *packetConn) SetReadDeadline(t time.Time) error { return nil }
|
||
|
func (r *packetConn) SetWriteDeadline(t time.Time) error { return nil }
|
||
|
|
||
|
func (r *packetConn) WriteTo(b []byte, addr net.Addr) (n int, err error) {
|
||
|
if r.pcapw == nil {
|
||
|
return len(b), nil
|
||
|
}
|
||
|
|
||
|
return len(b), pcapwrite(r.pcapw,
|
||
|
&layers.IPv6{
|
||
|
Version: 6,
|
||
|
TrafficClass: 0,
|
||
|
NextHeader: layers.IPProtocolUDP,
|
||
|
HopLimit: 1,
|
||
|
SrcIP: net.ParseIP("fe80::200:5eff:fe00:5300"),
|
||
|
DstIP: net.ParseIP("ff02::1:2"),
|
||
|
},
|
||
|
&layers.UDP{
|
||
|
SrcPort: 546,
|
||
|
DstPort: 547,
|
||
|
},
|
||
|
b)
|
||
|
}
|
||
|
|
||
|
func (r *packetConn) ReadFrom(buf []byte) (int, net.Addr, error) {
|
||
|
l, ip, err := readFrom(r.pcapr, buf)
|
||
|
return l, &net.IPAddr{IP: ip}, err
|
||
|
}
|
||
|
|
||
|
// dhcp4conn is a dhcp4client.ConnectionInt which replays a pcap file.
|
||
|
type dhcp4conn struct {
|
||
|
pcapr *pcapgo.Reader
|
||
|
pcapw *pcapgo.Writer
|
||
|
}
|
||
|
|
||
|
// NewDHCP4Conn returns a dhcp4client.ConnectionInt which replays packets from
|
||
|
// pcap file input, writing packets to pcap file output (if non-empty).
|
||
|
//
|
||
|
// See https://en.wikipedia.org/wiki/Pcap for details on pcap.
|
||
|
func NewDHCP4Conn(input, output string) (dhcp4client.ConnectionInt, error) {
|
||
|
pcapr, pcapw, err := pcapopen(input, output)
|
||
|
return &dhcp4conn{pcapr, pcapw}, err
|
||
|
}
|
||
|
|
||
|
func (r *dhcp4conn) Close() error { return nil }
|
||
|
func (r *dhcp4conn) SetReadTimeout(t time.Duration) error { return nil }
|
||
|
|
||
|
func (r *dhcp4conn) Write(b []byte) error {
|
||
|
if r.pcapw == nil {
|
||
|
return nil
|
||
|
}
|
||
|
return pcapwrite(r.pcapw,
|
||
|
&layers.IPv4{
|
||
|
Version: 4,
|
||
|
TTL: 255,
|
||
|
Protocol: layers.IPProtocolUDP,
|
||
|
SrcIP: net.ParseIP("0.0.0.0"),
|
||
|
DstIP: net.ParseIP("255.255.255.255"),
|
||
|
},
|
||
|
&layers.UDP{
|
||
|
SrcPort: 68,
|
||
|
DstPort: 67,
|
||
|
},
|
||
|
b)
|
||
|
}
|
||
|
|
||
|
func (r *dhcp4conn) ReadFrom() ([]byte, net.IP, error) {
|
||
|
buf := make([]byte, 9000)
|
||
|
_, ip, err := readFrom(r.pcapr, buf)
|
||
|
return buf, ip, err
|
||
|
}
|