dns.go 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102
  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. dnsOutbound "github.com/sagernet/sing-box/protocol/dns"
  10. "github.com/sagernet/sing-dns"
  11. "github.com/sagernet/sing-tun"
  12. "github.com/sagernet/sing/common/buf"
  13. E "github.com/sagernet/sing/common/exceptions"
  14. M "github.com/sagernet/sing/common/metadata"
  15. N "github.com/sagernet/sing/common/network"
  16. "github.com/sagernet/sing/common/udpnat2"
  17. mDNS "github.com/miekg/dns"
  18. )
  19. func (r *Router) hijackDNSStream(ctx context.Context, conn net.Conn, metadata adapter.InboundContext) error {
  20. metadata.Destination = M.Socksaddr{}
  21. for {
  22. conn.SetReadDeadline(time.Now().Add(C.DNSTimeout))
  23. err := dnsOutbound.HandleStreamDNSRequest(ctx, r, conn, metadata)
  24. if err != nil {
  25. return err
  26. }
  27. }
  28. }
  29. func (r *Router) hijackDNSPacket(ctx context.Context, conn N.PacketConn, packetBuffers []*N.PacketBuffer, metadata adapter.InboundContext, onClose N.CloseHandlerFunc) {
  30. if natConn, isNatConn := conn.(udpnat.Conn); isNatConn {
  31. metadata.Destination = M.Socksaddr{}
  32. for _, packet := range packetBuffers {
  33. buffer := packet.Buffer
  34. destination := packet.Destination
  35. N.PutPacketBuffer(packet)
  36. go ExchangeDNSPacket(ctx, r, natConn, buffer, metadata, destination)
  37. }
  38. natConn.SetHandler(&dnsHijacker{
  39. router: r,
  40. conn: conn,
  41. ctx: ctx,
  42. metadata: metadata,
  43. onClose: onClose,
  44. })
  45. return
  46. }
  47. err := dnsOutbound.NewDNSPacketConnection(ctx, r, conn, packetBuffers, metadata)
  48. N.CloseOnHandshakeFailure(conn, onClose, err)
  49. if err != nil && !E.IsClosedOrCanceled(err) {
  50. r.dnsLogger.ErrorContext(ctx, E.Cause(err, "process packet connection"))
  51. }
  52. }
  53. func ExchangeDNSPacket(ctx context.Context, router *Router, 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. router.dnsLogger.ErrorContext(ctx, E.Cause(err, "process packet connection"))
  57. }
  58. }
  59. func exchangeDNSPacket(ctx context.Context, router *Router, 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)
  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 *Router
  79. conn N.PacketConn
  80. ctx context.Context
  81. metadata adapter.InboundContext
  82. onClose N.CloseHandlerFunc
  83. }
  84. func (h *dnsHijacker) NewPacketEx(buffer *buf.Buffer, destination M.Socksaddr) {
  85. go ExchangeDNSPacket(h.ctx, h.router, h.conn, buffer, h.metadata, destination)
  86. }
  87. func (h *dnsHijacker) Close() error {
  88. if h.onClose != nil {
  89. h.onClose(nil)
  90. }
  91. return nil
  92. }