dns.go 1.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657
  1. package sniff
  2. import (
  3. "context"
  4. "encoding/binary"
  5. "io"
  6. "os"
  7. "github.com/sagernet/sing-box/adapter"
  8. C "github.com/sagernet/sing-box/constant"
  9. "github.com/sagernet/sing/common/buf"
  10. E "github.com/sagernet/sing/common/exceptions"
  11. mDNS "github.com/miekg/dns"
  12. )
  13. func StreamDomainNameQuery(readCtx context.Context, metadata *adapter.InboundContext, reader io.Reader) error {
  14. var length uint16
  15. err := binary.Read(reader, binary.BigEndian, &length)
  16. if err != nil {
  17. return E.Cause1(ErrNeedMoreData, err)
  18. }
  19. if length < 12 {
  20. return os.ErrInvalid
  21. }
  22. buffer := buf.NewSize(int(length))
  23. defer buffer.Release()
  24. var n int
  25. n, err = buffer.ReadFullFrom(reader, buffer.FreeLen())
  26. packet := buffer.Bytes()
  27. if n > 2 && packet[2]&0x80 != 0 { // QR
  28. return os.ErrInvalid
  29. }
  30. if n > 5 && packet[4] == 0 && packet[5] == 0 { // QDCOUNT
  31. return os.ErrInvalid
  32. }
  33. for i := 6; i < 10; i++ {
  34. // ANCOUNT, NSCOUNT
  35. if n > i && packet[i] != 0 {
  36. return os.ErrInvalid
  37. }
  38. }
  39. if err != nil {
  40. return E.Cause1(ErrNeedMoreData, err)
  41. }
  42. return DomainNameQuery(readCtx, metadata, packet)
  43. }
  44. func DomainNameQuery(ctx context.Context, metadata *adapter.InboundContext, packet []byte) error {
  45. var msg mDNS.Msg
  46. err := msg.Unpack(packet)
  47. if err != nil || msg.Response || len(msg.Question) == 0 || len(msg.Answer) > 0 || len(msg.Ns) > 0 {
  48. return err
  49. }
  50. metadata.Protocol = C.ProtocolDNS
  51. return nil
  52. }