udp_generic.go 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. //go:build (!linux || android) && !e2e_testing && !darwin
  2. // +build !linux android
  3. // +build !e2e_testing
  4. // +build !darwin
  5. // udp_generic implements the nebula UDP interface in pure Go stdlib. This
  6. // means it can be used on platforms like Darwin and Windows.
  7. package udp
  8. import (
  9. "context"
  10. "errors"
  11. "fmt"
  12. "net"
  13. "net/netip"
  14. "time"
  15. "github.com/sirupsen/logrus"
  16. "github.com/slackhq/nebula/config"
  17. "github.com/slackhq/nebula/firewall"
  18. "github.com/slackhq/nebula/header"
  19. )
  20. type GenericConn struct {
  21. *net.UDPConn
  22. l *logrus.Logger
  23. }
  24. var _ Conn = &GenericConn{}
  25. func NewGenericListener(l *logrus.Logger, ip netip.Addr, port int, multi bool, batch int) (Conn, error) {
  26. lc := NewListenConfig(multi)
  27. pc, err := lc.ListenPacket(context.TODO(), "udp", net.JoinHostPort(ip.String(), fmt.Sprintf("%v", port)))
  28. if err != nil {
  29. return nil, err
  30. }
  31. if uc, ok := pc.(*net.UDPConn); ok {
  32. return &GenericConn{UDPConn: uc, l: l}, nil
  33. }
  34. return nil, fmt.Errorf("Unexpected PacketConn: %T %#v", pc, pc)
  35. }
  36. func (u *GenericConn) WriteTo(b []byte, addr netip.AddrPort) error {
  37. _, err := u.UDPConn.WriteToUDPAddrPort(b, addr)
  38. return err
  39. }
  40. func (u *GenericConn) LocalAddr() (netip.AddrPort, error) {
  41. a := u.UDPConn.LocalAddr()
  42. switch v := a.(type) {
  43. case *net.UDPAddr:
  44. addr, ok := netip.AddrFromSlice(v.IP)
  45. if !ok {
  46. return netip.AddrPort{}, fmt.Errorf("LocalAddr returned invalid IP address: %s", v.IP)
  47. }
  48. return netip.AddrPortFrom(addr, uint16(v.Port)), nil
  49. default:
  50. return netip.AddrPort{}, fmt.Errorf("LocalAddr returned: %#v", a)
  51. }
  52. }
  53. func (u *GenericConn) ReloadConfig(c *config.C) {
  54. // TODO
  55. }
  56. func NewUDPStatsEmitter(udpConns []Conn) func() {
  57. // No UDP stats for non-linux
  58. return func() {}
  59. }
  60. type rawMessage struct {
  61. Len uint32
  62. }
  63. func (u *GenericConn) ListenOut(r EncReader, lhf LightHouseHandlerFunc, cache *firewall.ConntrackCacheTicker, q int) {
  64. plaintext := make([]byte, MTU)
  65. buffer := make([]byte, MTU)
  66. h := &header.H{}
  67. fwPacket := &firewall.Packet{}
  68. nb := make([]byte, 12, 12)
  69. var lastRecvErr time.Time
  70. for {
  71. // Just read one packet at a time
  72. n, rua, err := u.ReadFromUDPAddrPort(buffer)
  73. if err != nil {
  74. if errors.Is(err, net.ErrClosed) {
  75. u.l.WithError(err).Debug("udp socket is closed, exiting read loop")
  76. return
  77. }
  78. // Dampen unexpected message warns to once per minute
  79. if lastRecvErr.IsZero() || time.Since(lastRecvErr) > time.Minute {
  80. lastRecvErr = time.Now()
  81. u.l.WithError(err).Warn("unexpected udp socket receive error")
  82. }
  83. continue
  84. }
  85. r(
  86. netip.AddrPortFrom(rua.Addr().Unmap(), rua.Port()),
  87. plaintext[:0],
  88. buffer[:n],
  89. h,
  90. fwPacket,
  91. lhf,
  92. nb,
  93. q,
  94. cache.Get(u.l),
  95. )
  96. }
  97. }