3
0

tun_disabled.go 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133
  1. package overlay
  2. import (
  3. "fmt"
  4. "io"
  5. "net/netip"
  6. "strings"
  7. "github.com/rcrowley/go-metrics"
  8. "github.com/sirupsen/logrus"
  9. "github.com/slackhq/nebula/iputil"
  10. "github.com/slackhq/nebula/routing"
  11. )
  12. type disabledTun struct {
  13. read chan []byte
  14. vpnNetworks []netip.Prefix
  15. // Track these metrics since we don't have the tun device to do it for us
  16. tx metrics.Counter
  17. rx metrics.Counter
  18. l *logrus.Logger
  19. }
  20. func newDisabledTun(vpnNetworks []netip.Prefix, queueLen int, metricsEnabled bool, l *logrus.Logger) *disabledTun {
  21. tun := &disabledTun{
  22. vpnNetworks: vpnNetworks,
  23. read: make(chan []byte, queueLen),
  24. l: l,
  25. }
  26. if metricsEnabled {
  27. tun.tx = metrics.GetOrRegisterCounter("messages.tx.message", nil)
  28. tun.rx = metrics.GetOrRegisterCounter("messages.rx.message", nil)
  29. } else {
  30. tun.tx = &metrics.NilCounter{}
  31. tun.rx = &metrics.NilCounter{}
  32. }
  33. return tun
  34. }
  35. func (*disabledTun) Activate() error {
  36. return nil
  37. }
  38. func (*disabledTun) RoutesFor(addr netip.Addr) routing.Gateways {
  39. return routing.Gateways{}
  40. }
  41. func (t *disabledTun) Networks() []netip.Prefix {
  42. return t.vpnNetworks
  43. }
  44. func (*disabledTun) Name() string {
  45. return "disabled"
  46. }
  47. func (t *disabledTun) Read(b []byte) (int, error) {
  48. r, ok := <-t.read
  49. if !ok {
  50. return 0, io.EOF
  51. }
  52. if len(r) > len(b) {
  53. return 0, fmt.Errorf("packet larger than mtu: %d > %d bytes", len(r), len(b))
  54. }
  55. t.tx.Inc(1)
  56. if t.l.Level >= logrus.DebugLevel {
  57. t.l.WithField("raw", prettyPacket(r)).Debugf("Write payload")
  58. }
  59. return copy(b, r), nil
  60. }
  61. func (t *disabledTun) handleICMPEchoRequest(b []byte) bool {
  62. out := make([]byte, len(b))
  63. out = iputil.CreateICMPEchoResponse(b, out)
  64. if out == nil {
  65. return false
  66. }
  67. // attempt to write it, but don't block
  68. select {
  69. case t.read <- out:
  70. default:
  71. t.l.Debugf("tun_disabled: dropped ICMP Echo Reply response")
  72. }
  73. return true
  74. }
  75. func (t *disabledTun) Write(b []byte) (int, error) {
  76. t.rx.Inc(1)
  77. // Check for ICMP Echo Request before spending time doing the full parsing
  78. if t.handleICMPEchoRequest(b) {
  79. if t.l.Level >= logrus.DebugLevel {
  80. t.l.WithField("raw", prettyPacket(b)).Debugf("Disabled tun responded to ICMP Echo Request")
  81. }
  82. } else if t.l.Level >= logrus.DebugLevel {
  83. t.l.WithField("raw", prettyPacket(b)).Debugf("Disabled tun received unexpected payload")
  84. }
  85. return len(b), nil
  86. }
  87. func (t *disabledTun) NewMultiQueueReader() (io.ReadWriteCloser, error) {
  88. return t, nil
  89. }
  90. func (t *disabledTun) Close() error {
  91. if t.read != nil {
  92. close(t.read)
  93. t.read = nil
  94. }
  95. return nil
  96. }
  97. type prettyPacket []byte
  98. func (p prettyPacket) String() string {
  99. var s strings.Builder
  100. for i, b := range p {
  101. if i > 0 && i%8 == 0 {
  102. s.WriteString(" ")
  103. }
  104. s.WriteString(fmt.Sprintf("%02x ", b))
  105. }
  106. return s.String()
  107. }