udp_generic.go 2.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293
  1. //go:build (!linux || android) && !e2e_testing
  2. // +build !linux android
  3. // +build !e2e_testing
  4. // udp_generic implements the nebula UDP interface in pure Go stdlib. This
  5. // means it can be used on platforms like Darwin and Windows.
  6. package udp
  7. import (
  8. "context"
  9. "fmt"
  10. "net"
  11. "github.com/sirupsen/logrus"
  12. "github.com/slackhq/nebula/config"
  13. "github.com/slackhq/nebula/firewall"
  14. "github.com/slackhq/nebula/header"
  15. )
  16. type GenericConn struct {
  17. *net.UDPConn
  18. l *logrus.Logger
  19. }
  20. var _ Conn = &GenericConn{}
  21. func NewGenericListener(l *logrus.Logger, ip net.IP, port int, multi bool, batch int) (Conn, error) {
  22. lc := NewListenConfig(multi)
  23. pc, err := lc.ListenPacket(context.TODO(), "udp", net.JoinHostPort(ip.String(), fmt.Sprintf("%v", port)))
  24. if err != nil {
  25. return nil, err
  26. }
  27. if uc, ok := pc.(*net.UDPConn); ok {
  28. return &GenericConn{UDPConn: uc, l: l}, nil
  29. }
  30. return nil, fmt.Errorf("Unexpected PacketConn: %T %#v", pc, pc)
  31. }
  32. func (u *GenericConn) WriteTo(b []byte, addr *Addr) error {
  33. _, err := u.UDPConn.WriteToUDP(b, &net.UDPAddr{IP: addr.IP, Port: int(addr.Port)})
  34. return err
  35. }
  36. func (u *GenericConn) LocalAddr() (*Addr, error) {
  37. a := u.UDPConn.LocalAddr()
  38. switch v := a.(type) {
  39. case *net.UDPAddr:
  40. addr := &Addr{IP: make([]byte, len(v.IP))}
  41. copy(addr.IP, v.IP)
  42. addr.Port = uint16(v.Port)
  43. return addr, nil
  44. default:
  45. return nil, fmt.Errorf("LocalAddr returned: %#v", a)
  46. }
  47. }
  48. func (u *GenericConn) ReloadConfig(c *config.C) {
  49. // TODO
  50. }
  51. func NewUDPStatsEmitter(udpConns []Conn) func() {
  52. // No UDP stats for non-linux
  53. return func() {}
  54. }
  55. type rawMessage struct {
  56. Len uint32
  57. }
  58. func (u *GenericConn) ListenOut(r EncReader, lhf LightHouseHandlerFunc, cache *firewall.ConntrackCacheTicker, q int) {
  59. plaintext := make([]byte, MTU)
  60. buffer := make([]byte, MTU)
  61. h := &header.H{}
  62. fwPacket := &firewall.Packet{}
  63. udpAddr := &Addr{IP: make([]byte, 16)}
  64. nb := make([]byte, 12, 12)
  65. for {
  66. // Just read one packet at a time
  67. n, rua, err := u.ReadFromUDP(buffer)
  68. if err != nil {
  69. u.l.WithError(err).Debug("udp socket is closed, exiting read loop")
  70. return
  71. }
  72. udpAddr.IP = rua.IP
  73. udpAddr.Port = uint16(rua.Port)
  74. r(udpAddr, plaintext[:0], buffer[:n], h, fwPacket, lhf, nb, q, cache.Get(u.l))
  75. }
  76. }