mirror of
https://github.com/apernet/OpenGFW.git
synced 2024-12-23 09:29:22 +08:00
137 lines
2.9 KiB
Go
137 lines
2.9 KiB
Go
package io
|
|
|
|
import (
|
|
"context"
|
|
"hash/crc32"
|
|
"io"
|
|
"net"
|
|
"os"
|
|
"sort"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/google/gopacket"
|
|
"github.com/google/gopacket/pcapgo"
|
|
)
|
|
|
|
var _ PacketIO = (*pcapPacketIO)(nil)
|
|
|
|
type pcapPacketIO struct {
|
|
pcapFile io.ReadCloser
|
|
pcap *pcapgo.Reader
|
|
timeOffset *time.Duration
|
|
ioCancel context.CancelFunc
|
|
config PcapPacketIOConfig
|
|
|
|
dialer *net.Dialer
|
|
}
|
|
|
|
type PcapPacketIOConfig struct {
|
|
PcapFile string
|
|
Realtime bool
|
|
}
|
|
|
|
func NewPcapPacketIO(config PcapPacketIOConfig) (PacketIO, error) {
|
|
pcapFile, err := os.Open(config.PcapFile)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
handle, err := pcapgo.NewReader(pcapFile)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &pcapPacketIO{
|
|
pcapFile: pcapFile,
|
|
pcap: handle,
|
|
timeOffset: nil,
|
|
ioCancel: nil,
|
|
config: config,
|
|
dialer: &net.Dialer{},
|
|
}, nil
|
|
}
|
|
|
|
func (p *pcapPacketIO) Register(ctx context.Context, cb PacketCallback) error {
|
|
go func() {
|
|
packetSource := gopacket.NewPacketSource(p.pcap, p.pcap.LinkType())
|
|
for packet := range packetSource.Packets() {
|
|
p.wait(packet)
|
|
|
|
networkLayer := packet.NetworkLayer()
|
|
if networkLayer != nil {
|
|
src, dst := networkLayer.NetworkFlow().Endpoints()
|
|
endpoints := []string{src.String(), dst.String()}
|
|
sort.Strings(endpoints)
|
|
id := crc32.Checksum([]byte(strings.Join(endpoints, ",")), crc32.IEEETable)
|
|
|
|
cb(&pcapPacket{
|
|
streamID: id,
|
|
timestamp: packet.Metadata().Timestamp,
|
|
data: packet.LinkLayer().LayerPayload(),
|
|
}, nil)
|
|
}
|
|
}
|
|
// Give the workers a chance to finish everything
|
|
time.Sleep(time.Second)
|
|
// Stop the engine when all packets are finished
|
|
p.ioCancel()
|
|
}()
|
|
|
|
return nil
|
|
}
|
|
|
|
// A normal dialer is sufficient as pcap IO does not mess up with the networking
|
|
func (p *pcapPacketIO) ProtectedDialContext(ctx context.Context, network, address string) (net.Conn, error) {
|
|
return p.dialer.DialContext(ctx, network, address)
|
|
}
|
|
|
|
func (p *pcapPacketIO) SetVerdict(pkt Packet, v Verdict, newPacket []byte) error {
|
|
return nil
|
|
}
|
|
|
|
func (p *pcapPacketIO) SetCancelFunc(cancelFunc context.CancelFunc) error {
|
|
p.ioCancel = cancelFunc
|
|
return nil
|
|
}
|
|
|
|
func (p *pcapPacketIO) Close() error {
|
|
return p.pcapFile.Close()
|
|
}
|
|
|
|
// Intentionally slow down the replay
|
|
// In realtime mode, this is to match the timestamps in the capture
|
|
func (p *pcapPacketIO) wait(packet gopacket.Packet) {
|
|
if !p.config.Realtime {
|
|
return
|
|
}
|
|
|
|
if p.timeOffset == nil {
|
|
offset := time.Since(packet.Metadata().Timestamp)
|
|
p.timeOffset = &offset
|
|
} else {
|
|
t := time.Until(packet.Metadata().Timestamp.Add(*p.timeOffset))
|
|
time.Sleep(t)
|
|
}
|
|
}
|
|
|
|
var _ Packet = (*pcapPacket)(nil)
|
|
|
|
type pcapPacket struct {
|
|
streamID uint32
|
|
timestamp time.Time
|
|
data []byte
|
|
}
|
|
|
|
func (p *pcapPacket) StreamID() uint32 {
|
|
return p.streamID
|
|
}
|
|
|
|
func (p *pcapPacket) Timestamp() time.Time {
|
|
return p.timestamp
|
|
}
|
|
|
|
func (p *pcapPacket) Data() []byte {
|
|
return p.data
|
|
}
|