router_dns.go 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. package route
  2. import (
  3. "context"
  4. "net/netip"
  5. "strings"
  6. "time"
  7. "github.com/sagernet/sing-box/adapter"
  8. C "github.com/sagernet/sing-box/constant"
  9. "github.com/sagernet/sing-box/log"
  10. "github.com/sagernet/sing-dns"
  11. "github.com/sagernet/sing/common/cache"
  12. E "github.com/sagernet/sing/common/exceptions"
  13. F "github.com/sagernet/sing/common/format"
  14. M "github.com/sagernet/sing/common/metadata"
  15. mDNS "github.com/miekg/dns"
  16. )
  17. type DNSReverseMapping struct {
  18. cache *cache.LruCache[netip.Addr, string]
  19. }
  20. func NewDNSReverseMapping() *DNSReverseMapping {
  21. return &DNSReverseMapping{
  22. cache: cache.New[netip.Addr, string](),
  23. }
  24. }
  25. func (m *DNSReverseMapping) Save(address netip.Addr, domain string, ttl int) {
  26. m.cache.StoreWithExpire(address, domain, time.Now().Add(time.Duration(ttl)*time.Second))
  27. }
  28. func (m *DNSReverseMapping) Query(address netip.Addr) (string, bool) {
  29. domain, loaded := m.cache.Load(address)
  30. return domain, loaded
  31. }
  32. func (r *Router) matchDNS(ctx context.Context, allowFakeIP bool) (context.Context, dns.Transport, dns.DomainStrategy) {
  33. metadata := adapter.ContextFrom(ctx)
  34. if metadata == nil {
  35. panic("no context")
  36. }
  37. for i, rule := range r.dnsRules {
  38. metadata.ResetRuleCache()
  39. if rule.Match(metadata) {
  40. detour := rule.Outbound()
  41. transport, loaded := r.transportMap[detour]
  42. if !loaded {
  43. r.dnsLogger.ErrorContext(ctx, "transport not found: ", detour)
  44. continue
  45. }
  46. if _, isFakeIP := transport.(adapter.FakeIPTransport); isFakeIP && !allowFakeIP {
  47. continue
  48. }
  49. r.dnsLogger.DebugContext(ctx, "match[", i, "] ", rule.String(), " => ", detour)
  50. if rule.DisableCache() {
  51. ctx = dns.ContextWithDisableCache(ctx, true)
  52. }
  53. if rewriteTTL := rule.RewriteTTL(); rewriteTTL != nil {
  54. ctx = dns.ContextWithRewriteTTL(ctx, *rewriteTTL)
  55. }
  56. if domainStrategy, dsLoaded := r.transportDomainStrategy[transport]; dsLoaded {
  57. return ctx, transport, domainStrategy
  58. } else {
  59. return ctx, transport, r.defaultDomainStrategy
  60. }
  61. }
  62. }
  63. if domainStrategy, dsLoaded := r.transportDomainStrategy[r.defaultTransport]; dsLoaded {
  64. return ctx, r.defaultTransport, domainStrategy
  65. } else {
  66. return ctx, r.defaultTransport, r.defaultDomainStrategy
  67. }
  68. }
  69. func (r *Router) Exchange(ctx context.Context, message *mDNS.Msg) (*mDNS.Msg, error) {
  70. if len(message.Question) > 0 {
  71. r.dnsLogger.DebugContext(ctx, "exchange ", formatQuestion(message.Question[0].String()))
  72. }
  73. var (
  74. response *mDNS.Msg
  75. cached bool
  76. err error
  77. )
  78. response, cached = r.dnsClient.ExchangeCache(ctx, message)
  79. if !cached {
  80. ctx, metadata := adapter.AppendContext(ctx)
  81. if len(message.Question) > 0 {
  82. metadata.QueryType = message.Question[0].Qtype
  83. switch metadata.QueryType {
  84. case mDNS.TypeA:
  85. metadata.IPVersion = 4
  86. case mDNS.TypeAAAA:
  87. metadata.IPVersion = 6
  88. }
  89. metadata.Domain = fqdnToDomain(message.Question[0].Name)
  90. }
  91. ctx, transport, strategy := r.matchDNS(ctx, true)
  92. ctx, cancel := context.WithTimeout(ctx, C.DNSTimeout)
  93. defer cancel()
  94. response, err = r.dnsClient.Exchange(ctx, transport, message, strategy)
  95. if err != nil && len(message.Question) > 0 {
  96. r.dnsLogger.ErrorContext(ctx, E.Cause(err, "exchange failed for ", formatQuestion(message.Question[0].String())))
  97. }
  98. }
  99. if len(message.Question) > 0 && response != nil {
  100. LogDNSAnswers(r.dnsLogger, ctx, message.Question[0].Name, response.Answer)
  101. }
  102. if r.dnsReverseMapping != nil && len(message.Question) > 0 && response != nil && len(response.Answer) > 0 {
  103. for _, answer := range response.Answer {
  104. switch record := answer.(type) {
  105. case *mDNS.A:
  106. r.dnsReverseMapping.Save(M.AddrFromIP(record.A), fqdnToDomain(record.Hdr.Name), int(record.Hdr.Ttl))
  107. case *mDNS.AAAA:
  108. r.dnsReverseMapping.Save(M.AddrFromIP(record.AAAA), fqdnToDomain(record.Hdr.Name), int(record.Hdr.Ttl))
  109. }
  110. }
  111. }
  112. return response, err
  113. }
  114. func (r *Router) Lookup(ctx context.Context, domain string, strategy dns.DomainStrategy) ([]netip.Addr, error) {
  115. r.dnsLogger.DebugContext(ctx, "lookup domain ", domain)
  116. ctx, metadata := adapter.AppendContext(ctx)
  117. metadata.Domain = domain
  118. ctx, transport, transportStrategy := r.matchDNS(ctx, false)
  119. if strategy == dns.DomainStrategyAsIS {
  120. strategy = transportStrategy
  121. }
  122. ctx, cancel := context.WithTimeout(ctx, C.DNSTimeout)
  123. defer cancel()
  124. addrs, err := r.dnsClient.Lookup(ctx, transport, domain, strategy)
  125. if len(addrs) > 0 {
  126. r.dnsLogger.InfoContext(ctx, "lookup succeed for ", domain, ": ", strings.Join(F.MapToString(addrs), " "))
  127. } else if err != nil {
  128. r.dnsLogger.ErrorContext(ctx, E.Cause(err, "lookup failed for ", domain))
  129. } else {
  130. r.dnsLogger.ErrorContext(ctx, "lookup failed for ", domain, ": empty result")
  131. err = dns.RCodeNameError
  132. }
  133. return addrs, err
  134. }
  135. func (r *Router) LookupDefault(ctx context.Context, domain string) ([]netip.Addr, error) {
  136. return r.Lookup(ctx, domain, dns.DomainStrategyAsIS)
  137. }
  138. func (r *Router) ClearDNSCache() {
  139. r.dnsClient.ClearCache()
  140. if r.platformInterface != nil {
  141. r.platformInterface.ClearDNSCache()
  142. }
  143. }
  144. func LogDNSAnswers(logger log.ContextLogger, ctx context.Context, domain string, answers []mDNS.RR) {
  145. for _, answer := range answers {
  146. logger.InfoContext(ctx, "exchanged ", domain, " ", mDNS.Type(answer.Header().Rrtype).String(), " ", formatQuestion(answer.String()))
  147. }
  148. }
  149. func fqdnToDomain(fqdn string) string {
  150. if mDNS.IsFqdn(fqdn) {
  151. return fqdn[:len(fqdn)-1]
  152. }
  153. return fqdn
  154. }
  155. func formatQuestion(string string) string {
  156. if strings.HasPrefix(string, ";") {
  157. string = string[1:]
  158. }
  159. string = strings.ReplaceAll(string, "\t", " ")
  160. for strings.Contains(string, " ") {
  161. string = strings.ReplaceAll(string, " ", " ")
  162. }
  163. return string
  164. }