|
|
|
@ -7,14 +7,11 @@ import (
|
|
|
|
|
|
|
|
|
|
// Payload represents a single UDP payload.
|
|
|
|
|
type Payload struct { |
|
|
|
|
payload *buf.Buffer |
|
|
|
|
source net.Destination |
|
|
|
|
originalDest net.Destination |
|
|
|
|
Content *buf.Buffer |
|
|
|
|
Source net.Destination |
|
|
|
|
OriginalDestination net.Destination |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// PayloadHandler is function to handle Payload.
|
|
|
|
|
type PayloadHandler func(payload *buf.Buffer, source net.Destination, originalDest net.Destination) |
|
|
|
|
|
|
|
|
|
type HubOption func(h *Hub) |
|
|
|
|
|
|
|
|
|
func HubCapacity(cap int) HubOption { |
|
|
|
@ -31,12 +28,12 @@ func HubReceiveOriginalDestination(r bool) HubOption {
|
|
|
|
|
|
|
|
|
|
type Hub struct { |
|
|
|
|
conn *net.UDPConn |
|
|
|
|
callback PayloadHandler |
|
|
|
|
cache chan *Payload |
|
|
|
|
capacity int |
|
|
|
|
recvOrigDest bool |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
func ListenUDP(address net.Address, port net.Port, callback PayloadHandler, options ...HubOption) (*Hub, error) { |
|
|
|
|
func ListenUDP(address net.Address, port net.Port, options ...HubOption) (*Hub, error) { |
|
|
|
|
udpConn, err := net.ListenUDP("udp", &net.UDPAddr{ |
|
|
|
|
IP: address.IP(), |
|
|
|
|
Port: int(port), |
|
|
|
@ -48,13 +45,14 @@ func ListenUDP(address net.Address, port net.Port, callback PayloadHandler, opti
|
|
|
|
|
hub := &Hub{ |
|
|
|
|
conn: udpConn, |
|
|
|
|
capacity: 256, |
|
|
|
|
callback: callback, |
|
|
|
|
recvOrigDest: false, |
|
|
|
|
} |
|
|
|
|
for _, opt := range options { |
|
|
|
|
opt(hub) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
hub.cache = make(chan *Payload, hub.capacity) |
|
|
|
|
|
|
|
|
|
if hub.recvOrigDest { |
|
|
|
|
rawConn, err := udpConn.SyscallConn() |
|
|
|
|
if err != nil { |
|
|
|
@ -70,10 +68,7 @@ func ListenUDP(address net.Address, port net.Port, callback PayloadHandler, opti
|
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
c := make(chan *Payload, hub.capacity) |
|
|
|
|
|
|
|
|
|
go hub.start(c) |
|
|
|
|
go hub.process(c) |
|
|
|
|
go hub.start() |
|
|
|
|
return hub, nil |
|
|
|
|
} |
|
|
|
|
|
|
|
|
@ -90,13 +85,8 @@ func (h *Hub) WriteTo(payload []byte, dest net.Destination) (int, error) {
|
|
|
|
|
}) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
func (h *Hub) process(c <-chan *Payload) { |
|
|
|
|
for p := range c { |
|
|
|
|
h.callback(p.payload, p.source, p.originalDest) |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
func (h *Hub) start(c chan<- *Payload) { |
|
|
|
|
func (h *Hub) start() { |
|
|
|
|
c := h.cache |
|
|
|
|
defer close(c) |
|
|
|
|
|
|
|
|
|
oobBytes := make([]byte, 256) |
|
|
|
@ -119,13 +109,13 @@ func (h *Hub) start(c chan<- *Payload) {
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
payload := &Payload{ |
|
|
|
|
payload: buffer, |
|
|
|
|
Content: buffer, |
|
|
|
|
Source: net.UDPDestination(net.IPAddress(addr.IP), net.Port(addr.Port)), |
|
|
|
|
} |
|
|
|
|
payload.source = net.UDPDestination(net.IPAddress(addr.IP), net.Port(addr.Port)) |
|
|
|
|
if h.recvOrigDest && noob > 0 { |
|
|
|
|
payload.originalDest = RetrieveOriginalDest(oobBytes[:noob]) |
|
|
|
|
if payload.originalDest.IsValid() { |
|
|
|
|
newError("UDP original destination: ", payload.originalDest).AtDebug().WriteToLog() |
|
|
|
|
payload.OriginalDestination = RetrieveOriginalDest(oobBytes[:noob]) |
|
|
|
|
if payload.OriginalDestination.IsValid() { |
|
|
|
|
newError("UDP original destination: ", payload.OriginalDestination).AtDebug().WriteToLog() |
|
|
|
|
} else { |
|
|
|
|
newError("failed to read UDP original destination").WriteToLog() |
|
|
|
|
} |
|
|
|
@ -143,3 +133,7 @@ func (h *Hub) start(c chan<- *Payload) {
|
|
|
|
|
func (h *Hub) Addr() net.Addr { |
|
|
|
|
return h.conn.LocalAddr() |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
func (h *Hub) Receive() <-chan *Payload { |
|
|
|
|
return h.cache |
|
|
|
|
} |
|
|
|
|