dns.go 3.1 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495
  1. package route
  2. import (
  3. "context"
  4. "errors"
  5. "net"
  6. "time"
  7. "github.com/sagernet/sing-box/adapter"
  8. C "github.com/sagernet/sing-box/constant"
  9. "github.com/sagernet/sing-box/dns"
  10. dnsOutbound "github.com/sagernet/sing-box/protocol/dns"
  11. "github.com/sagernet/sing-tun"
  12. "github.com/sagernet/sing/common/buf"
  13. E "github.com/sagernet/sing/common/exceptions"
  14. "github.com/sagernet/sing/common/logger"
  15. M "github.com/sagernet/sing/common/metadata"
  16. N "github.com/sagernet/sing/common/network"
  17. "github.com/sagernet/sing/common/udpnat2"
  18. mDNS "github.com/miekg/dns"
  19. )
  20. func (r *Router) hijackDNSStream(ctx context.Context, conn net.Conn, metadata adapter.InboundContext) error {
  21. metadata.Destination = M.Socksaddr{}
  22. for {
  23. conn.SetReadDeadline(time.Now().Add(C.DNSTimeout))
  24. err := dnsOutbound.HandleStreamDNSRequest(ctx, r.dns, conn, metadata)
  25. if err != nil {
  26. return err
  27. }
  28. }
  29. }
  30. func (r *Router) hijackDNSPacket(ctx context.Context, conn N.PacketConn, packetBuffers []*N.PacketBuffer, metadata adapter.InboundContext) {
  31. if natConn, isNatConn := conn.(udpnat.Conn); isNatConn {
  32. metadata.Destination = M.Socksaddr{}
  33. for _, packet := range packetBuffers {
  34. buffer := packet.Buffer
  35. destination := packet.Destination
  36. N.PutPacketBuffer(packet)
  37. go ExchangeDNSPacket(ctx, r.dns, r.logger, natConn, buffer, metadata, destination)
  38. }
  39. natConn.SetHandler(&dnsHijacker{
  40. router: r.dns,
  41. logger: r.logger,
  42. conn: conn,
  43. ctx: ctx,
  44. metadata: metadata,
  45. })
  46. return
  47. }
  48. err := dnsOutbound.NewDNSPacketConnection(ctx, r.dns, conn, packetBuffers, metadata)
  49. if err != nil && !E.IsClosedOrCanceled(err) {
  50. r.logger.ErrorContext(ctx, E.Cause(err, "process DNS packet connection"))
  51. }
  52. }
  53. func ExchangeDNSPacket(ctx context.Context, router adapter.DNSRouter, logger logger.ContextLogger, conn N.PacketConn, buffer *buf.Buffer, metadata adapter.InboundContext, destination M.Socksaddr) {
  54. err := exchangeDNSPacket(ctx, router, conn, buffer, metadata, destination)
  55. if err != nil && !errors.Is(err, tun.ErrDrop) && !E.IsClosedOrCanceled(err) {
  56. logger.ErrorContext(ctx, E.Cause(err, "process DNS packet connection"))
  57. }
  58. }
  59. func exchangeDNSPacket(ctx context.Context, router adapter.DNSRouter, conn N.PacketConn, buffer *buf.Buffer, metadata adapter.InboundContext, destination M.Socksaddr) error {
  60. var message mDNS.Msg
  61. err := message.Unpack(buffer.Bytes())
  62. buffer.Release()
  63. if err != nil {
  64. return E.Cause(err, "unpack request")
  65. }
  66. response, err := router.Exchange(adapter.WithContext(ctx, &metadata), &message, adapter.DNSQueryOptions{})
  67. if err != nil {
  68. return err
  69. }
  70. responseBuffer, err := dns.TruncateDNSMessage(&message, response, 1024)
  71. if err != nil {
  72. return err
  73. }
  74. err = conn.WritePacket(responseBuffer, destination)
  75. return err
  76. }
  77. type dnsHijacker struct {
  78. router adapter.DNSRouter
  79. logger logger.ContextLogger
  80. conn N.PacketConn
  81. ctx context.Context
  82. metadata adapter.InboundContext
  83. }
  84. func (h *dnsHijacker) NewPacketEx(buffer *buf.Buffer, destination M.Socksaddr) {
  85. go ExchangeDNSPacket(h.ctx, h.router, h.logger, h.conn, buffer, h.metadata, destination)
  86. }