dns.go 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269
  1. package outbound
  2. import (
  3. "context"
  4. "encoding/binary"
  5. "net"
  6. "os"
  7. "github.com/sagernet/sing-box/adapter"
  8. C "github.com/sagernet/sing-box/constant"
  9. "github.com/sagernet/sing-dns"
  10. "github.com/sagernet/sing/common"
  11. "github.com/sagernet/sing/common/buf"
  12. "github.com/sagernet/sing/common/bufio"
  13. "github.com/sagernet/sing/common/canceler"
  14. M "github.com/sagernet/sing/common/metadata"
  15. N "github.com/sagernet/sing/common/network"
  16. "github.com/sagernet/sing/common/task"
  17. mDNS "github.com/miekg/dns"
  18. )
  19. var _ adapter.Outbound = (*DNS)(nil)
  20. type DNS struct {
  21. myOutboundAdapter
  22. }
  23. func NewDNS(router adapter.Router, tag string) *DNS {
  24. return &DNS{
  25. myOutboundAdapter{
  26. protocol: C.TypeDNS,
  27. network: []string{N.NetworkTCP, N.NetworkUDP},
  28. router: router,
  29. tag: tag,
  30. },
  31. }
  32. }
  33. func (d *DNS) DialContext(ctx context.Context, network string, destination M.Socksaddr) (net.Conn, error) {
  34. return nil, os.ErrInvalid
  35. }
  36. func (d *DNS) ListenPacket(ctx context.Context, destination M.Socksaddr) (net.PacketConn, error) {
  37. return nil, os.ErrInvalid
  38. }
  39. func (d *DNS) NewConnection(ctx context.Context, conn net.Conn, metadata adapter.InboundContext) error {
  40. defer conn.Close()
  41. ctx = adapter.WithContext(ctx, &metadata)
  42. for {
  43. err := d.handleConnection(ctx, conn, metadata)
  44. if err != nil {
  45. return err
  46. }
  47. }
  48. }
  49. func (d *DNS) handleConnection(ctx context.Context, conn net.Conn, metadata adapter.InboundContext) error {
  50. var queryLength uint16
  51. err := binary.Read(conn, binary.BigEndian, &queryLength)
  52. if err != nil {
  53. return err
  54. }
  55. if queryLength == 0 {
  56. return dns.RCodeFormatError
  57. }
  58. _buffer := buf.StackNewSize(int(queryLength))
  59. defer common.KeepAlive(_buffer)
  60. buffer := common.Dup(_buffer)
  61. defer buffer.Release()
  62. _, err = buffer.ReadFullFrom(conn, int(queryLength))
  63. if err != nil {
  64. return err
  65. }
  66. var message mDNS.Msg
  67. err = message.Unpack(buffer.Bytes())
  68. if err != nil {
  69. return err
  70. }
  71. metadataInQuery := metadata
  72. go func() error {
  73. response, err := d.router.Exchange(adapter.WithContext(ctx, &metadataInQuery), &message)
  74. if err != nil {
  75. return err
  76. }
  77. _responseBuffer := buf.StackNewPacket()
  78. defer common.KeepAlive(_responseBuffer)
  79. responseBuffer := common.Dup(_responseBuffer)
  80. defer responseBuffer.Release()
  81. responseBuffer.Resize(2, 0)
  82. n, err := response.PackBuffer(responseBuffer.FreeBytes())
  83. if err != nil {
  84. return err
  85. }
  86. responseBuffer.Truncate(len(n))
  87. binary.BigEndian.PutUint16(responseBuffer.ExtendHeader(2), uint16(len(n)))
  88. _, err = conn.Write(responseBuffer.Bytes())
  89. return err
  90. }()
  91. return nil
  92. }
  93. func (d *DNS) NewPacketConnection(ctx context.Context, conn N.PacketConn, metadata adapter.InboundContext) error {
  94. var reader N.PacketReader = conn
  95. var counters []N.CountFunc
  96. var cachedPackets []*N.PacketBuffer
  97. for {
  98. reader, counters = N.UnwrapCountPacketReader(reader, counters)
  99. if cachedReader, isCached := reader.(N.CachedPacketReader); isCached {
  100. packet := cachedReader.ReadCachedPacket()
  101. if packet != nil {
  102. cachedPackets = append(cachedPackets, packet)
  103. continue
  104. }
  105. }
  106. if readWaiter, created := bufio.CreatePacketReadWaiter(reader); created {
  107. return d.newPacketConnection(ctx, conn, readWaiter, counters, cachedPackets, metadata)
  108. }
  109. break
  110. }
  111. ctx = adapter.WithContext(ctx, &metadata)
  112. fastClose, cancel := common.ContextWithCancelCause(ctx)
  113. timeout := canceler.New(fastClose, cancel, C.DNSTimeout)
  114. var group task.Group
  115. group.Append0(func(ctx context.Context) error {
  116. for {
  117. var message mDNS.Msg
  118. var destination M.Socksaddr
  119. var err error
  120. if len(cachedPackets) > 0 {
  121. packet := cachedPackets[0]
  122. cachedPackets = cachedPackets[1:]
  123. for _, counter := range counters {
  124. counter(int64(packet.Buffer.Len()))
  125. }
  126. err = message.Unpack(packet.Buffer.Bytes())
  127. packet.Buffer.Release()
  128. if err != nil {
  129. cancel(err)
  130. return err
  131. }
  132. destination = packet.Destination
  133. } else {
  134. buffer := buf.NewPacket()
  135. destination, err = conn.ReadPacket(buffer)
  136. if err != nil {
  137. buffer.Release()
  138. cancel(err)
  139. return err
  140. }
  141. for _, counter := range counters {
  142. counter(int64(buffer.Len()))
  143. }
  144. err = message.Unpack(buffer.Bytes())
  145. buffer.Release()
  146. if err != nil {
  147. cancel(err)
  148. return err
  149. }
  150. timeout.Update()
  151. }
  152. metadataInQuery := metadata
  153. go func() error {
  154. response, err := d.router.Exchange(adapter.WithContext(ctx, &metadataInQuery), &message)
  155. if err != nil {
  156. cancel(err)
  157. return err
  158. }
  159. timeout.Update()
  160. responseBuffer := buf.NewPacket()
  161. n, err := response.PackBuffer(responseBuffer.FreeBytes())
  162. if err != nil {
  163. cancel(err)
  164. responseBuffer.Release()
  165. return err
  166. }
  167. responseBuffer.Truncate(len(n))
  168. err = conn.WritePacket(responseBuffer, destination)
  169. if err != nil {
  170. cancel(err)
  171. }
  172. return err
  173. }()
  174. }
  175. })
  176. group.Cleanup(func() {
  177. conn.Close()
  178. })
  179. return group.Run(fastClose)
  180. }
  181. func (d *DNS) newPacketConnection(ctx context.Context, conn N.PacketConn, readWaiter N.PacketReadWaiter, readCounters []N.CountFunc, cached []*N.PacketBuffer, metadata adapter.InboundContext) error {
  182. ctx = adapter.WithContext(ctx, &metadata)
  183. fastClose, cancel := common.ContextWithCancelCause(ctx)
  184. timeout := canceler.New(fastClose, cancel, C.DNSTimeout)
  185. var group task.Group
  186. group.Append0(func(ctx context.Context) error {
  187. var buffer *buf.Buffer
  188. readWaiter.InitializeReadWaiter(func() *buf.Buffer {
  189. buffer = buf.NewSize(dns.FixedPacketSize)
  190. buffer.FullReset()
  191. return buffer
  192. })
  193. defer readWaiter.InitializeReadWaiter(nil)
  194. for {
  195. var message mDNS.Msg
  196. var destination M.Socksaddr
  197. var err error
  198. if len(cached) > 0 {
  199. packet := cached[0]
  200. cached = cached[1:]
  201. for _, counter := range readCounters {
  202. counter(int64(packet.Buffer.Len()))
  203. }
  204. err = message.Unpack(packet.Buffer.Bytes())
  205. packet.Buffer.Release()
  206. if err != nil {
  207. cancel(err)
  208. return err
  209. }
  210. destination = packet.Destination
  211. } else {
  212. destination, err = readWaiter.WaitReadPacket()
  213. if err != nil {
  214. buffer.Release()
  215. cancel(err)
  216. return err
  217. }
  218. for _, counter := range readCounters {
  219. counter(int64(buffer.Len()))
  220. }
  221. err = message.Unpack(buffer.Bytes())
  222. buffer.Release()
  223. if err != nil {
  224. cancel(err)
  225. return err
  226. }
  227. timeout.Update()
  228. }
  229. metadataInQuery := metadata
  230. go func() error {
  231. response, err := d.router.Exchange(adapter.WithContext(ctx, &metadataInQuery), &message)
  232. if err != nil {
  233. cancel(err)
  234. return err
  235. }
  236. timeout.Update()
  237. responseBuffer := buf.NewPacket()
  238. n, err := response.PackBuffer(responseBuffer.FreeBytes())
  239. if err != nil {
  240. cancel(err)
  241. responseBuffer.Release()
  242. return err
  243. }
  244. responseBuffer.Truncate(len(n))
  245. err = conn.WritePacket(responseBuffer, destination)
  246. if err != nil {
  247. cancel(err)
  248. }
  249. return err
  250. }()
  251. }
  252. })
  253. group.Cleanup(func() {
  254. conn.Close()
  255. })
  256. return group.Run(fastClose)
  257. }