router_dns.go 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  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) matchDNS(ctx context.Context) (context.Context, dns.Transport, dns.DomainStrategy) {
  15. metadata := adapter.ContextFrom(ctx)
  16. if metadata == nil {
  17. panic("no context")
  18. }
  19. for i, rule := range r.dnsRules {
  20. if rule.Match(metadata) {
  21. if rule.DisableCache() {
  22. ctx = dns.ContextWithDisableCache(ctx, true)
  23. }
  24. detour := rule.Outbound()
  25. r.dnsLogger.DebugContext(ctx, "match[", i, "] ", rule.String(), " => ", detour)
  26. if transport, loaded := r.transportMap[detour]; loaded {
  27. if domainStrategy, dsLoaded := r.transportDomainStrategy[transport]; dsLoaded {
  28. return ctx, transport, domainStrategy
  29. } else {
  30. return ctx, transport, r.defaultDomainStrategy
  31. }
  32. }
  33. r.dnsLogger.ErrorContext(ctx, "transport not found: ", detour)
  34. }
  35. }
  36. return ctx, r.defaultTransport, r.defaultDomainStrategy
  37. }
  38. func (r *Router) Exchange(ctx context.Context, message *dnsmessage.Message) (*dnsmessage.Message, error) {
  39. if len(message.Questions) > 0 {
  40. r.dnsLogger.DebugContext(ctx, "exchange ", formatDNSQuestion(message.Questions[0]))
  41. }
  42. ctx, metadata := adapter.AppendContext(ctx)
  43. if len(message.Questions) > 0 {
  44. switch message.Questions[0].Type {
  45. case dnsmessage.TypeA:
  46. metadata.IPVersion = 4
  47. case dnsmessage.TypeAAAA:
  48. metadata.IPVersion = 6
  49. }
  50. }
  51. ctx, transport, strategy := r.matchDNS(ctx)
  52. ctx, cancel := context.WithTimeout(ctx, C.DNSTimeout)
  53. defer cancel()
  54. response, err := r.dnsClient.Exchange(ctx, transport, message, strategy)
  55. if err != nil && len(message.Questions) > 0 {
  56. r.dnsLogger.ErrorContext(ctx, E.Cause(err, "exchange failed for ", message.Questions[0].Name.String()))
  57. }
  58. if len(message.Questions) > 0 && response != nil {
  59. LogDNSAnswers(r.dnsLogger, ctx, message.Questions[0].Name.String(), response.Answers)
  60. }
  61. return response, err
  62. }
  63. func (r *Router) Lookup(ctx context.Context, domain string, strategy dns.DomainStrategy) ([]netip.Addr, error) {
  64. r.dnsLogger.DebugContext(ctx, "lookup domain ", domain)
  65. ctx, transport, transportStrategy := r.matchDNS(ctx)
  66. if strategy == dns.DomainStrategyAsIS {
  67. strategy = transportStrategy
  68. }
  69. ctx, cancel := context.WithTimeout(ctx, C.DNSTimeout)
  70. defer cancel()
  71. addrs, err := r.dnsClient.Lookup(ctx, transport, domain, strategy)
  72. if len(addrs) > 0 {
  73. r.dnsLogger.InfoContext(ctx, "lookup succeed for ", domain, ": ", strings.Join(F.MapToString(addrs), " "))
  74. } else {
  75. r.dnsLogger.ErrorContext(ctx, E.Cause(err, "lookup failed for ", domain))
  76. }
  77. return addrs, err
  78. }
  79. func (r *Router) LookupDefault(ctx context.Context, domain string) ([]netip.Addr, error) {
  80. return r.Lookup(ctx, domain, dns.DomainStrategyAsIS)
  81. }
  82. func LogDNSAnswers(logger log.ContextLogger, ctx context.Context, domain string, answers []dnsmessage.Resource) {
  83. for _, rawAnswer := range answers {
  84. var content string
  85. switch answer := rawAnswer.Body.(type) {
  86. case *dnsmessage.AResource:
  87. content = netip.AddrFrom4(answer.A).String()
  88. case *dnsmessage.NSResource:
  89. content = answer.NS.String()
  90. case *dnsmessage.CNAMEResource:
  91. content = answer.CNAME.String()
  92. case *dnsmessage.SOAResource:
  93. content = answer.MBox.String()
  94. case *dnsmessage.PTRResource:
  95. content = answer.PTR.String()
  96. case *dnsmessage.MXResource:
  97. content = answer.MX.String()
  98. case *dnsmessage.TXTResource:
  99. content = strings.Join(answer.TXT, " ")
  100. case *dnsmessage.AAAAResource:
  101. content = netip.AddrFrom16(answer.AAAA).String()
  102. case *dnsmessage.SRVResource:
  103. content = answer.Target.String()
  104. case *dnsmessage.UnknownResource:
  105. content = answer.Type.String()
  106. default:
  107. continue
  108. }
  109. rType := formatDNSType(rawAnswer.Header.Type)
  110. if rType == "" {
  111. logger.InfoContext(ctx, "exchanged ", domain, " ", rType)
  112. } else {
  113. logger.InfoContext(ctx, "exchanged ", domain, " ", rType, " ", content)
  114. }
  115. }
  116. }
  117. func formatDNSQuestion(question dnsmessage.Question) string {
  118. var qType string
  119. qType = question.Type.String()
  120. if len(qType) > 4 {
  121. qType = qType[4:]
  122. }
  123. var qClass string
  124. qClass = question.Class.String()
  125. if len(qClass) > 5 {
  126. qClass = qClass[5:]
  127. }
  128. return string(question.Name.Data[:question.Name.Length-1]) + " " + qType + " " + qClass
  129. }
  130. func formatDNSType(qType dnsmessage.Type) string {
  131. qTypeName := qType.String()
  132. if len(qTypeName) > 4 {
  133. return qTypeName[4:]
  134. } else {
  135. return F.ToString("unknown (type ", qTypeName, ")")
  136. }
  137. }