dns.go 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. package inbound
  2. import (
  3. "context"
  4. "encoding/binary"
  5. "io"
  6. "net"
  7. "net/netip"
  8. "github.com/sagernet/sing-box/adapter"
  9. C "github.com/sagernet/sing-box/constant"
  10. "github.com/sagernet/sing-box/log"
  11. "github.com/sagernet/sing-box/option"
  12. "github.com/sagernet/sing/common"
  13. "github.com/sagernet/sing/common/buf"
  14. N "github.com/sagernet/sing/common/network"
  15. "github.com/sagernet/sing/common/udpnat"
  16. "golang.org/x/net/dns/dnsmessage"
  17. )
  18. type DNS struct {
  19. myInboundAdapter
  20. udpNat *udpnat.Service[netip.AddrPort]
  21. }
  22. func NewDNS(ctx context.Context, router adapter.Router, logger log.ContextLogger, tag string, options option.DNSInboundOptions) *DNS {
  23. dns := &DNS{
  24. myInboundAdapter: myInboundAdapter{
  25. protocol: C.TypeTProxy,
  26. network: options.Network.Build(),
  27. ctx: ctx,
  28. router: router,
  29. logger: logger,
  30. tag: tag,
  31. listenOptions: options.ListenOptions,
  32. },
  33. }
  34. dns.connHandler = dns
  35. dns.packetHandler = dns
  36. dns.udpNat = udpnat.New[netip.AddrPort](10, adapter.NewUpstreamContextHandler(nil, dns.newPacketConnection, dns))
  37. dns.packetUpstream = dns.udpNat
  38. return dns
  39. }
  40. func (d *DNS) NewConnection(ctx context.Context, conn net.Conn, metadata adapter.InboundContext) error {
  41. return NewDNSConnection(ctx, d.router, d.logger, conn, metadata)
  42. }
  43. func (d *DNS) NewPacket(ctx context.Context, conn N.PacketConn, buffer *buf.Buffer, metadata adapter.InboundContext) error {
  44. d.udpNat.NewContextPacket(ctx, metadata.Source.AddrPort(), buffer, adapter.UpstreamMetadata(metadata), func(natConn N.PacketConn) (context.Context, N.PacketWriter) {
  45. return adapter.WithContext(log.ContextWithNewID(ctx), &metadata), natConn
  46. })
  47. return nil
  48. }
  49. func (d *DNS) newPacketConnection(ctx context.Context, conn N.PacketConn, metadata adapter.InboundContext) error {
  50. return NewDNSPacketConnection(ctx, d.router, d.logger, conn, metadata)
  51. }
  52. func NewDNSConnection(ctx context.Context, router adapter.Router, logger log.ContextLogger, conn net.Conn, metadata adapter.InboundContext) error {
  53. ctx = adapter.WithContext(ctx, &metadata)
  54. _buffer := buf.StackNewSize(1024)
  55. defer common.KeepAlive(_buffer)
  56. buffer := common.Dup(_buffer)
  57. defer buffer.Release()
  58. for {
  59. var queryLength uint16
  60. err := binary.Read(conn, binary.BigEndian, &queryLength)
  61. if err != nil {
  62. return err
  63. }
  64. if queryLength > 1024 {
  65. return io.ErrShortBuffer
  66. }
  67. buffer.FullReset()
  68. _, err = buffer.ReadFullFrom(conn, int(queryLength))
  69. if err != nil {
  70. return err
  71. }
  72. var message dnsmessage.Message
  73. err = message.Unpack(buffer.Bytes())
  74. if err != nil {
  75. return err
  76. }
  77. if len(message.Questions) > 0 {
  78. question := message.Questions[0]
  79. metadata.Domain = string(question.Name.Data[:question.Name.Length-1])
  80. logger.DebugContext(ctx, "inbound dns query ", formatDNSQuestion(question), " from ", metadata.Source)
  81. }
  82. go func() error {
  83. response, err := router.Exchange(ctx, &message)
  84. if err != nil {
  85. return err
  86. }
  87. _responseBuffer := buf.StackNewSize(1024)
  88. defer common.KeepAlive(_responseBuffer)
  89. responseBuffer := common.Dup(_responseBuffer)
  90. defer responseBuffer.Release()
  91. responseBuffer.Resize(2, 0)
  92. n, err := response.AppendPack(responseBuffer.Index(0))
  93. if err != nil {
  94. return err
  95. }
  96. responseBuffer.Truncate(len(n))
  97. binary.BigEndian.PutUint16(responseBuffer.ExtendHeader(2), uint16(len(n)))
  98. _, err = conn.Write(responseBuffer.Bytes())
  99. return err
  100. }()
  101. }
  102. }
  103. func NewDNSPacketConnection(ctx context.Context, router adapter.Router, logger log.ContextLogger, conn N.PacketConn, metadata adapter.InboundContext) error {
  104. ctx = adapter.WithContext(ctx, &metadata)
  105. _buffer := buf.StackNewSize(1024)
  106. defer common.KeepAlive(_buffer)
  107. buffer := common.Dup(_buffer)
  108. defer buffer.Release()
  109. for {
  110. buffer.FullReset()
  111. destination, err := conn.ReadPacket(buffer)
  112. if err != nil {
  113. return err
  114. }
  115. var message dnsmessage.Message
  116. err = message.Unpack(buffer.Bytes())
  117. if err != nil {
  118. return err
  119. }
  120. if len(message.Questions) > 0 {
  121. question := message.Questions[0]
  122. metadata.Domain = string(question.Name.Data[:question.Name.Length-1])
  123. logger.DebugContext(ctx, "inbound dns query ", formatDNSQuestion(question), " from ", metadata.Source)
  124. }
  125. go func() error {
  126. response, err := router.Exchange(ctx, &message)
  127. if err != nil {
  128. return err
  129. }
  130. _responseBuffer := buf.StackNewSize(1024)
  131. defer common.KeepAlive(_responseBuffer)
  132. responseBuffer := common.Dup(_responseBuffer)
  133. defer responseBuffer.Release()
  134. n, err := response.AppendPack(responseBuffer.Index(0))
  135. if err != nil {
  136. return err
  137. }
  138. responseBuffer.Truncate(len(n))
  139. err = conn.WritePacket(responseBuffer, destination)
  140. return err
  141. }()
  142. }
  143. }
  144. func formatDNSQuestion(question dnsmessage.Question) string {
  145. return string(question.Name.Data[:question.Name.Length-1]) + " " + question.Type.String()[4:] + " " + question.Class.String()[5:]
  146. }