router_dns.go 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119
  1. package route
  2. import (
  3. "context"
  4. "net/netip"
  5. "strings"
  6. "github.com/sagernet/sing-box/adapter"
  7. C "github.com/sagernet/sing-box/constant"
  8. "github.com/sagernet/sing-box/log"
  9. "github.com/sagernet/sing-dns"
  10. E "github.com/sagernet/sing/common/exceptions"
  11. F "github.com/sagernet/sing/common/format"
  12. "golang.org/x/net/dns/dnsmessage"
  13. )
  14. func (r *Router) Exchange(ctx context.Context, message *dnsmessage.Message) (*dnsmessage.Message, error) {
  15. if len(message.Questions) > 0 {
  16. r.dnsLogger.DebugContext(ctx, "exchange ", formatDNSQuestion(message.Questions[0]))
  17. }
  18. ctx, metadata := adapter.AppendContext(ctx)
  19. if len(message.Questions) > 0 {
  20. switch message.Questions[0].Type {
  21. case dnsmessage.TypeA:
  22. metadata.IPVersion = 4
  23. case dnsmessage.TypeAAAA:
  24. metadata.IPVersion = 6
  25. }
  26. }
  27. ctx, transport := r.matchDNS(ctx)
  28. ctx, cancel := context.WithTimeout(ctx, C.DNSTimeout)
  29. defer cancel()
  30. response, err := r.dnsClient.Exchange(ctx, transport, message)
  31. if err != nil && len(message.Questions) > 0 {
  32. r.dnsLogger.ErrorContext(ctx, E.Cause(err, "exchange failed for ", message.Questions[0].Name.String()))
  33. }
  34. if len(message.Questions) > 0 && response != nil {
  35. LogDNSAnswers(r.dnsLogger, ctx, message.Questions[0].Name.String(), response.Answers)
  36. }
  37. return response, err
  38. }
  39. func (r *Router) Lookup(ctx context.Context, domain string, strategy dns.DomainStrategy) ([]netip.Addr, error) {
  40. r.dnsLogger.DebugContext(ctx, "lookup domain ", domain)
  41. ctx, transport := r.matchDNS(ctx)
  42. ctx, cancel := context.WithTimeout(ctx, C.DNSTimeout)
  43. defer cancel()
  44. addrs, err := r.dnsClient.Lookup(ctx, transport, domain, strategy)
  45. if len(addrs) > 0 {
  46. r.dnsLogger.InfoContext(ctx, "lookup succeed for ", domain, ": ", strings.Join(F.MapToString(addrs), " "))
  47. } else {
  48. r.dnsLogger.ErrorContext(ctx, E.Cause(err, "lookup failed for ", domain))
  49. }
  50. return addrs, err
  51. }
  52. func (r *Router) LookupDefault(ctx context.Context, domain string) ([]netip.Addr, error) {
  53. return r.Lookup(ctx, domain, r.defaultDomainStrategy)
  54. }
  55. func LogDNSAnswers(logger log.ContextLogger, ctx context.Context, domain string, answers []dnsmessage.Resource) {
  56. for _, rawAnswer := range answers {
  57. var content string
  58. switch answer := rawAnswer.Body.(type) {
  59. case *dnsmessage.AResource:
  60. content = netip.AddrFrom4(answer.A).String()
  61. case *dnsmessage.NSResource:
  62. content = answer.NS.String()
  63. case *dnsmessage.CNAMEResource:
  64. content = answer.CNAME.String()
  65. case *dnsmessage.SOAResource:
  66. content = answer.MBox.String()
  67. case *dnsmessage.PTRResource:
  68. content = answer.PTR.String()
  69. case *dnsmessage.MXResource:
  70. content = answer.MX.String()
  71. case *dnsmessage.TXTResource:
  72. content = strings.Join(answer.TXT, " ")
  73. case *dnsmessage.AAAAResource:
  74. content = netip.AddrFrom16(answer.AAAA).String()
  75. case *dnsmessage.SRVResource:
  76. content = answer.Target.String()
  77. case *dnsmessage.UnknownResource:
  78. content = answer.Type.String()
  79. default:
  80. continue
  81. }
  82. rType := formatDNSType(rawAnswer.Header.Type)
  83. if rType == "" {
  84. logger.InfoContext(ctx, "exchanged ", domain, " ", rType)
  85. } else {
  86. logger.InfoContext(ctx, "exchanged ", domain, " ", rType, " ", content)
  87. }
  88. }
  89. }
  90. func formatDNSQuestion(question dnsmessage.Question) string {
  91. var qType string
  92. qType = question.Type.String()
  93. if len(qType) > 4 {
  94. qType = qType[4:]
  95. }
  96. var qClass string
  97. qClass = question.Class.String()
  98. if len(qClass) > 5 {
  99. qClass = qClass[5:]
  100. }
  101. return string(question.Name.Data[:question.Name.Length-1]) + " " + qType + " " + qClass
  102. }
  103. func formatDNSType(qType dnsmessage.Type) string {
  104. qTypeName := qType.String()
  105. if len(qTypeName) > 4 {
  106. return qTypeName[4:]
  107. } else {
  108. return F.ToString("unknown (type ", qTypeName, ")")
  109. }
  110. }