2021-10-21 15:24:11 -06:00
|
|
|
//go:build e2e_testing
|
2021-03-29 13:29:20 -06:00
|
|
|
// +build e2e_testing
|
|
|
|
|
2021-11-03 19:54:04 -06:00
|
|
|
package udp
|
2021-03-29 13:29:20 -06:00
|
|
|
|
|
|
|
import (
|
2021-04-14 12:50:09 -06:00
|
|
|
"fmt"
|
2021-03-29 13:29:20 -06:00
|
|
|
"net"
|
|
|
|
|
|
|
|
"github.com/sirupsen/logrus"
|
2021-11-03 19:54:04 -06:00
|
|
|
"github.com/slackhq/nebula/config"
|
|
|
|
"github.com/slackhq/nebula/firewall"
|
|
|
|
"github.com/slackhq/nebula/header"
|
2021-03-29 13:29:20 -06:00
|
|
|
)
|
|
|
|
|
2021-11-03 19:54:04 -06:00
|
|
|
type Packet struct {
|
2021-03-29 13:29:20 -06:00
|
|
|
ToIp net.IP
|
|
|
|
ToPort uint16
|
|
|
|
FromIp net.IP
|
|
|
|
FromPort uint16
|
|
|
|
Data []byte
|
|
|
|
}
|
|
|
|
|
2021-11-03 19:54:04 -06:00
|
|
|
func (u *Packet) Copy() *Packet {
|
|
|
|
n := &Packet{
|
2021-03-31 12:36:10 -06:00
|
|
|
ToIp: make(net.IP, len(u.ToIp)),
|
|
|
|
ToPort: u.ToPort,
|
|
|
|
FromIp: make(net.IP, len(u.FromIp)),
|
|
|
|
FromPort: u.FromPort,
|
|
|
|
Data: make([]byte, len(u.Data)),
|
|
|
|
}
|
|
|
|
|
|
|
|
copy(n.ToIp, u.ToIp)
|
|
|
|
copy(n.FromIp, u.FromIp)
|
|
|
|
copy(n.Data, u.Data)
|
|
|
|
return n
|
|
|
|
}
|
|
|
|
|
2021-11-03 19:54:04 -06:00
|
|
|
type Conn struct {
|
|
|
|
Addr *Addr
|
2021-03-29 13:29:20 -06:00
|
|
|
|
2021-11-03 19:54:04 -06:00
|
|
|
RxPackets chan *Packet // Packets to receive into nebula
|
|
|
|
TxPackets chan *Packet // Packets transmitted outside by nebula
|
2021-03-29 13:29:20 -06:00
|
|
|
|
|
|
|
l *logrus.Logger
|
|
|
|
}
|
|
|
|
|
2021-11-03 19:54:04 -06:00
|
|
|
func NewListener(l *logrus.Logger, ip string, port int, _ bool, _ int) (*Conn, error) {
|
|
|
|
return &Conn{
|
|
|
|
Addr: &Addr{net.ParseIP(ip), uint16(port)},
|
2022-06-27 11:33:29 -06:00
|
|
|
RxPackets: make(chan *Packet, 10),
|
|
|
|
TxPackets: make(chan *Packet, 10),
|
2021-03-29 13:29:20 -06:00
|
|
|
l: l,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Send will place a UdpPacket onto the receive queue for nebula to consume
|
|
|
|
// this is an encrypted packet or a handshake message in most cases
|
|
|
|
// packets were transmitted from another nebula node, you can send them with Tun.Send
|
2021-11-03 19:54:04 -06:00
|
|
|
func (u *Conn) Send(packet *Packet) {
|
|
|
|
h := &header.H{}
|
2021-04-14 12:50:09 -06:00
|
|
|
if err := h.Parse(packet.Data); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2023-03-30 10:09:20 -06:00
|
|
|
if u.l.Level >= logrus.DebugLevel {
|
2022-09-01 08:44:58 -06:00
|
|
|
u.l.WithField("header", h).
|
|
|
|
WithField("udpAddr", fmt.Sprintf("%v:%v", packet.FromIp, packet.FromPort)).
|
|
|
|
WithField("dataLen", len(packet.Data)).
|
2023-03-13 11:35:14 -06:00
|
|
|
Debug("UDP receiving injected packet")
|
2022-09-01 08:44:58 -06:00
|
|
|
}
|
2021-11-03 19:54:04 -06:00
|
|
|
u.RxPackets <- packet
|
2021-03-29 13:29:20 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// Get will pull a UdpPacket from the transmit queue
|
|
|
|
// nebula meant to send this message on the network, it will be encrypted
|
|
|
|
// packets were ingested from the tun side (in most cases), you can send them with Tun.Send
|
2021-11-03 19:54:04 -06:00
|
|
|
func (u *Conn) Get(block bool) *Packet {
|
2021-03-29 13:29:20 -06:00
|
|
|
if block {
|
2021-11-03 19:54:04 -06:00
|
|
|
return <-u.TxPackets
|
2021-03-29 13:29:20 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
select {
|
2021-11-03 19:54:04 -06:00
|
|
|
case p := <-u.TxPackets:
|
2021-03-29 13:29:20 -06:00
|
|
|
return p
|
|
|
|
default:
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//********************************************************************************************************************//
|
|
|
|
// Below this is boilerplate implementation to make nebula actually work
|
|
|
|
//********************************************************************************************************************//
|
|
|
|
|
2021-11-03 19:54:04 -06:00
|
|
|
func (u *Conn) WriteTo(b []byte, addr *Addr) error {
|
|
|
|
p := &Packet{
|
2021-03-29 13:29:20 -06:00
|
|
|
Data: make([]byte, len(b), len(b)),
|
|
|
|
FromIp: make([]byte, 16),
|
2021-11-03 19:54:04 -06:00
|
|
|
FromPort: u.Addr.Port,
|
2021-03-29 13:29:20 -06:00
|
|
|
ToIp: make([]byte, 16),
|
|
|
|
ToPort: addr.Port,
|
|
|
|
}
|
|
|
|
|
|
|
|
copy(p.Data, b)
|
2021-03-31 09:26:35 -06:00
|
|
|
copy(p.ToIp, addr.IP.To16())
|
2021-11-03 19:54:04 -06:00
|
|
|
copy(p.FromIp, u.Addr.IP.To16())
|
2021-03-29 13:29:20 -06:00
|
|
|
|
2021-11-03 19:54:04 -06:00
|
|
|
u.TxPackets <- p
|
2021-03-29 13:29:20 -06:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-11-03 19:54:04 -06:00
|
|
|
func (u *Conn) ListenOut(r EncReader, lhf LightHouseHandlerFunc, cache *firewall.ConntrackCacheTicker, q int) {
|
|
|
|
plaintext := make([]byte, MTU)
|
|
|
|
h := &header.H{}
|
|
|
|
fwPacket := &firewall.Packet{}
|
|
|
|
ua := &Addr{IP: make([]byte, 16)}
|
2021-03-29 13:29:20 -06:00
|
|
|
nb := make([]byte, 12, 12)
|
|
|
|
|
|
|
|
for {
|
2022-09-01 08:44:58 -06:00
|
|
|
p, ok := <-u.RxPackets
|
|
|
|
if !ok {
|
|
|
|
return
|
|
|
|
}
|
2021-03-29 13:29:20 -06:00
|
|
|
ua.Port = p.FromPort
|
|
|
|
copy(ua.IP, p.FromIp.To16())
|
2022-06-21 12:35:23 -06:00
|
|
|
r(ua, nil, plaintext[:0], p.Data, h, fwPacket, lhf, nb, q, cache.Get(u.l))
|
2021-03-29 13:29:20 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-03 19:54:04 -06:00
|
|
|
func (u *Conn) ReloadConfig(*config.C) {}
|
2021-03-29 13:29:20 -06:00
|
|
|
|
2021-11-03 19:54:04 -06:00
|
|
|
func NewUDPStatsEmitter(_ []*Conn) func() {
|
2021-03-29 13:29:20 -06:00
|
|
|
// No UDP stats for non-linux
|
|
|
|
return func() {}
|
|
|
|
}
|
|
|
|
|
2021-11-03 19:54:04 -06:00
|
|
|
func (u *Conn) LocalAddr() (*Addr, error) {
|
|
|
|
return u.Addr, nil
|
2021-03-29 13:29:20 -06:00
|
|
|
}
|
|
|
|
|
2021-11-03 19:54:04 -06:00
|
|
|
func (u *Conn) Rebind() error {
|
2021-03-29 13:29:20 -06:00
|
|
|
return nil
|
|
|
|
}
|