client.go 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706
  1. package dns
  2. import (
  3. "context"
  4. "errors"
  5. "net"
  6. "net/netip"
  7. "strings"
  8. "time"
  9. "github.com/sagernet/sing-box/adapter"
  10. "github.com/sagernet/sing-box/common/compatible"
  11. C "github.com/sagernet/sing-box/constant"
  12. "github.com/sagernet/sing/common"
  13. E "github.com/sagernet/sing/common/exceptions"
  14. "github.com/sagernet/sing/common/logger"
  15. M "github.com/sagernet/sing/common/metadata"
  16. "github.com/sagernet/sing/common/task"
  17. "github.com/sagernet/sing/contrab/freelru"
  18. "github.com/sagernet/sing/contrab/maphash"
  19. "github.com/miekg/dns"
  20. )
  21. var (
  22. ErrNoRawSupport = E.New("no raw query support by current transport")
  23. ErrNotCached = E.New("not cached")
  24. ErrResponseRejected = E.New("response rejected")
  25. ErrResponseRejectedCached = E.Extend(ErrResponseRejected, "cached")
  26. )
  27. var _ adapter.DNSClient = (*Client)(nil)
  28. type Client struct {
  29. timeout time.Duration
  30. disableCache bool
  31. disableExpire bool
  32. independentCache bool
  33. clientSubnet netip.Prefix
  34. rdrc adapter.RDRCStore
  35. initRDRCFunc func() adapter.RDRCStore
  36. logger logger.ContextLogger
  37. cache freelru.Cache[dns.Question, *dns.Msg]
  38. cacheLock compatible.Map[dns.Question, chan struct{}]
  39. transportCache freelru.Cache[transportCacheKey, *dns.Msg]
  40. transportCacheLock compatible.Map[dns.Question, chan struct{}]
  41. }
  42. type ClientOptions struct {
  43. Timeout time.Duration
  44. DisableCache bool
  45. DisableExpire bool
  46. IndependentCache bool
  47. CacheCapacity uint32
  48. ClientSubnet netip.Prefix
  49. RDRC func() adapter.RDRCStore
  50. Logger logger.ContextLogger
  51. }
  52. func NewClient(options ClientOptions) *Client {
  53. client := &Client{
  54. timeout: options.Timeout,
  55. disableCache: options.DisableCache,
  56. disableExpire: options.DisableExpire,
  57. independentCache: options.IndependentCache,
  58. clientSubnet: options.ClientSubnet,
  59. initRDRCFunc: options.RDRC,
  60. logger: options.Logger,
  61. }
  62. if client.timeout == 0 {
  63. client.timeout = C.DNSTimeout
  64. }
  65. cacheCapacity := options.CacheCapacity
  66. if cacheCapacity < 1024 {
  67. cacheCapacity = 1024
  68. }
  69. if !client.disableCache {
  70. if !client.independentCache {
  71. client.cache = common.Must1(freelru.NewSharded[dns.Question, *dns.Msg](cacheCapacity, maphash.NewHasher[dns.Question]().Hash32))
  72. } else {
  73. client.transportCache = common.Must1(freelru.NewSharded[transportCacheKey, *dns.Msg](cacheCapacity, maphash.NewHasher[transportCacheKey]().Hash32))
  74. }
  75. }
  76. return client
  77. }
  78. type transportCacheKey struct {
  79. dns.Question
  80. transportTag string
  81. }
  82. func (c *Client) Start() {
  83. if c.initRDRCFunc != nil {
  84. c.rdrc = c.initRDRCFunc()
  85. }
  86. }
  87. func extractNegativeTTL(response *dns.Msg) (uint32, bool) {
  88. for _, record := range response.Ns {
  89. if soa, isSOA := record.(*dns.SOA); isSOA {
  90. soaTTL := soa.Header().Ttl
  91. soaMinimum := soa.Minttl
  92. if soaTTL < soaMinimum {
  93. return soaTTL, true
  94. }
  95. return soaMinimum, true
  96. }
  97. }
  98. return 0, false
  99. }
  100. func (c *Client) Exchange(ctx context.Context, transport adapter.DNSTransport, message *dns.Msg, options adapter.DNSQueryOptions, responseChecker func(responseAddrs []netip.Addr) bool) (*dns.Msg, error) {
  101. if len(message.Question) == 0 {
  102. if c.logger != nil {
  103. c.logger.WarnContext(ctx, "bad question size: ", len(message.Question))
  104. }
  105. return FixedResponseStatus(message, dns.RcodeFormatError), nil
  106. }
  107. question := message.Question[0]
  108. if question.Qtype == dns.TypeA && options.Strategy == C.DomainStrategyIPv6Only || question.Qtype == dns.TypeAAAA && options.Strategy == C.DomainStrategyIPv4Only {
  109. if c.logger != nil {
  110. c.logger.DebugContext(ctx, "strategy rejected")
  111. }
  112. return FixedResponseStatus(message, dns.RcodeSuccess), nil
  113. }
  114. clientSubnet := options.ClientSubnet
  115. if !clientSubnet.IsValid() {
  116. clientSubnet = c.clientSubnet
  117. }
  118. if clientSubnet.IsValid() {
  119. message = SetClientSubnet(message, clientSubnet)
  120. }
  121. isSimpleRequest := len(message.Question) == 1 &&
  122. len(message.Ns) == 0 &&
  123. (len(message.Extra) == 0 || len(message.Extra) == 1 &&
  124. message.Extra[0].Header().Rrtype == dns.TypeOPT &&
  125. message.Extra[0].Header().Class > 0 &&
  126. message.Extra[0].Header().Ttl == 0 &&
  127. len(message.Extra[0].(*dns.OPT).Option) == 0) &&
  128. !options.ClientSubnet.IsValid()
  129. disableCache := !isSimpleRequest || c.disableCache || options.DisableCache
  130. if !disableCache {
  131. if c.cache != nil {
  132. cond, loaded := c.cacheLock.LoadOrStore(question, make(chan struct{}))
  133. if loaded {
  134. select {
  135. case <-cond:
  136. case <-ctx.Done():
  137. return nil, ctx.Err()
  138. }
  139. } else {
  140. defer func() {
  141. c.cacheLock.Delete(question)
  142. close(cond)
  143. }()
  144. }
  145. } else if c.transportCache != nil {
  146. cond, loaded := c.transportCacheLock.LoadOrStore(question, make(chan struct{}))
  147. if loaded {
  148. select {
  149. case <-cond:
  150. case <-ctx.Done():
  151. return nil, ctx.Err()
  152. }
  153. } else {
  154. defer func() {
  155. c.transportCacheLock.Delete(question)
  156. close(cond)
  157. }()
  158. }
  159. }
  160. response, ttl := c.loadResponse(question, transport)
  161. if response != nil {
  162. logCachedResponse(c.logger, ctx, response, ttl)
  163. response.Id = message.Id
  164. return response, nil
  165. }
  166. }
  167. messageId := message.Id
  168. contextTransport, clientSubnetLoaded := transportTagFromContext(ctx)
  169. if clientSubnetLoaded && transport.Tag() == contextTransport {
  170. return nil, E.New("DNS query loopback in transport[", contextTransport, "]")
  171. }
  172. ctx = contextWithTransportTag(ctx, transport.Tag())
  173. if !disableCache && responseChecker != nil && c.rdrc != nil {
  174. rejected := c.rdrc.LoadRDRC(transport.Tag(), question.Name, question.Qtype)
  175. if rejected {
  176. return nil, ErrResponseRejectedCached
  177. }
  178. }
  179. ctx, cancel := context.WithTimeout(ctx, c.timeout)
  180. response, err := transport.Exchange(ctx, message)
  181. cancel()
  182. if err != nil {
  183. var rcodeError RcodeError
  184. if errors.As(err, &rcodeError) {
  185. response = FixedResponseStatus(message, int(rcodeError))
  186. } else {
  187. return nil, err
  188. }
  189. }
  190. /*if question.Qtype == dns.TypeA || question.Qtype == dns.TypeAAAA {
  191. validResponse := response
  192. loop:
  193. for {
  194. var (
  195. addresses int
  196. queryCNAME string
  197. )
  198. for _, rawRR := range validResponse.Answer {
  199. switch rr := rawRR.(type) {
  200. case *dns.A:
  201. break loop
  202. case *dns.AAAA:
  203. break loop
  204. case *dns.CNAME:
  205. queryCNAME = rr.Target
  206. }
  207. }
  208. if queryCNAME == "" {
  209. break
  210. }
  211. exMessage := *message
  212. exMessage.Question = []dns.Question{{
  213. Name: queryCNAME,
  214. Qtype: question.Qtype,
  215. }}
  216. validResponse, err = c.Exchange(ctx, transport, &exMessage, options, responseChecker)
  217. if err != nil {
  218. return nil, err
  219. }
  220. }
  221. if validResponse != response {
  222. response.Answer = append(response.Answer, validResponse.Answer...)
  223. }
  224. }*/
  225. disableCache = disableCache || (response.Rcode != dns.RcodeSuccess && response.Rcode != dns.RcodeNameError)
  226. if responseChecker != nil {
  227. var rejected bool
  228. // TODO: add accept_any rule and support to check response instead of addresses
  229. if response.Rcode != dns.RcodeSuccess && response.Rcode != dns.RcodeNameError {
  230. rejected = true
  231. } else if len(response.Answer) == 0 {
  232. rejected = !responseChecker(nil)
  233. } else {
  234. rejected = !responseChecker(MessageToAddresses(response))
  235. }
  236. if rejected {
  237. if !disableCache && c.rdrc != nil {
  238. c.rdrc.SaveRDRCAsync(transport.Tag(), question.Name, question.Qtype, c.logger)
  239. }
  240. logRejectedResponse(c.logger, ctx, response)
  241. return response, ErrResponseRejected
  242. }
  243. }
  244. if question.Qtype == dns.TypeHTTPS {
  245. if options.Strategy == C.DomainStrategyIPv4Only || options.Strategy == C.DomainStrategyIPv6Only {
  246. for _, rr := range response.Answer {
  247. https, isHTTPS := rr.(*dns.HTTPS)
  248. if !isHTTPS {
  249. continue
  250. }
  251. content := https.SVCB
  252. content.Value = common.Filter(content.Value, func(it dns.SVCBKeyValue) bool {
  253. if options.Strategy == C.DomainStrategyIPv4Only {
  254. return it.Key() != dns.SVCB_IPV6HINT
  255. } else {
  256. return it.Key() != dns.SVCB_IPV4HINT
  257. }
  258. })
  259. https.SVCB = content
  260. }
  261. }
  262. }
  263. var timeToLive uint32
  264. if len(response.Answer) == 0 {
  265. if soaTTL, hasSOA := extractNegativeTTL(response); hasSOA {
  266. timeToLive = soaTTL
  267. }
  268. }
  269. if timeToLive == 0 {
  270. for _, recordList := range [][]dns.RR{response.Answer, response.Ns, response.Extra} {
  271. for _, record := range recordList {
  272. if record.Header().Rrtype == dns.TypeOPT {
  273. continue
  274. }
  275. if timeToLive == 0 || record.Header().Ttl > 0 && record.Header().Ttl < timeToLive {
  276. timeToLive = record.Header().Ttl
  277. }
  278. }
  279. }
  280. }
  281. if options.RewriteTTL != nil {
  282. timeToLive = *options.RewriteTTL
  283. }
  284. for _, recordList := range [][]dns.RR{response.Answer, response.Ns, response.Extra} {
  285. for _, record := range recordList {
  286. if record.Header().Rrtype == dns.TypeOPT {
  287. continue
  288. }
  289. record.Header().Ttl = timeToLive
  290. }
  291. }
  292. if !disableCache {
  293. c.storeCache(transport, question, response, timeToLive)
  294. }
  295. response.Id = messageId
  296. requestEDNSOpt := message.IsEdns0()
  297. responseEDNSOpt := response.IsEdns0()
  298. if responseEDNSOpt != nil && (requestEDNSOpt == nil || requestEDNSOpt.Version() < responseEDNSOpt.Version()) {
  299. response.Extra = common.Filter(response.Extra, func(it dns.RR) bool {
  300. return it.Header().Rrtype != dns.TypeOPT
  301. })
  302. if requestEDNSOpt != nil {
  303. response.SetEdns0(responseEDNSOpt.UDPSize(), responseEDNSOpt.Do())
  304. }
  305. }
  306. logExchangedResponse(c.logger, ctx, response, timeToLive)
  307. return response, nil
  308. }
  309. func (c *Client) Lookup(ctx context.Context, transport adapter.DNSTransport, domain string, options adapter.DNSQueryOptions, responseChecker func(responseAddrs []netip.Addr) bool) ([]netip.Addr, error) {
  310. domain = FqdnToDomain(domain)
  311. dnsName := dns.Fqdn(domain)
  312. var strategy C.DomainStrategy
  313. if options.LookupStrategy != C.DomainStrategyAsIS {
  314. strategy = options.LookupStrategy
  315. } else {
  316. strategy = options.Strategy
  317. }
  318. lookupOptions := options
  319. if options.LookupStrategy != C.DomainStrategyAsIS {
  320. lookupOptions.Strategy = strategy
  321. }
  322. if strategy == C.DomainStrategyIPv4Only {
  323. return c.lookupToExchange(ctx, transport, dnsName, dns.TypeA, lookupOptions, responseChecker)
  324. } else if strategy == C.DomainStrategyIPv6Only {
  325. return c.lookupToExchange(ctx, transport, dnsName, dns.TypeAAAA, lookupOptions, responseChecker)
  326. }
  327. var response4 []netip.Addr
  328. var response6 []netip.Addr
  329. var group task.Group
  330. group.Append("exchange4", func(ctx context.Context) error {
  331. response, err := c.lookupToExchange(ctx, transport, dnsName, dns.TypeA, lookupOptions, responseChecker)
  332. if err != nil {
  333. return err
  334. }
  335. response4 = response
  336. return nil
  337. })
  338. group.Append("exchange6", func(ctx context.Context) error {
  339. response, err := c.lookupToExchange(ctx, transport, dnsName, dns.TypeAAAA, lookupOptions, responseChecker)
  340. if err != nil {
  341. return err
  342. }
  343. response6 = response
  344. return nil
  345. })
  346. err := group.Run(ctx)
  347. if len(response4) == 0 && len(response6) == 0 {
  348. return nil, err
  349. }
  350. return sortAddresses(response4, response6, strategy), nil
  351. }
  352. func (c *Client) ClearCache() {
  353. if c.cache != nil {
  354. c.cache.Purge()
  355. } else if c.transportCache != nil {
  356. c.transportCache.Purge()
  357. }
  358. }
  359. func sortAddresses(response4 []netip.Addr, response6 []netip.Addr, strategy C.DomainStrategy) []netip.Addr {
  360. if strategy == C.DomainStrategyPreferIPv6 {
  361. return append(response6, response4...)
  362. } else {
  363. return append(response4, response6...)
  364. }
  365. }
  366. func (c *Client) storeCache(transport adapter.DNSTransport, question dns.Question, message *dns.Msg, timeToLive uint32) {
  367. if timeToLive == 0 {
  368. return
  369. }
  370. if c.disableExpire {
  371. if !c.independentCache {
  372. c.cache.Add(question, message.Copy())
  373. } else {
  374. c.transportCache.Add(transportCacheKey{
  375. Question: question,
  376. transportTag: transport.Tag(),
  377. }, message.Copy())
  378. }
  379. } else {
  380. if !c.independentCache {
  381. c.cache.AddWithLifetime(question, message.Copy(), time.Second*time.Duration(timeToLive))
  382. } else {
  383. c.transportCache.AddWithLifetime(transportCacheKey{
  384. Question: question,
  385. transportTag: transport.Tag(),
  386. }, message.Copy(), time.Second*time.Duration(timeToLive))
  387. }
  388. }
  389. }
  390. func (c *Client) lookupToExchange(ctx context.Context, transport adapter.DNSTransport, name string, qType uint16, options adapter.DNSQueryOptions, responseChecker func(responseAddrs []netip.Addr) bool) ([]netip.Addr, error) {
  391. question := dns.Question{
  392. Name: name,
  393. Qtype: qType,
  394. Qclass: dns.ClassINET,
  395. }
  396. disableCache := c.disableCache || options.DisableCache
  397. if !disableCache {
  398. cachedAddresses, err := c.questionCache(question, transport)
  399. if err != ErrNotCached {
  400. return cachedAddresses, err
  401. }
  402. }
  403. message := dns.Msg{
  404. MsgHdr: dns.MsgHdr{
  405. RecursionDesired: true,
  406. },
  407. Question: []dns.Question{question},
  408. }
  409. response, err := c.Exchange(ctx, transport, &message, options, responseChecker)
  410. if err != nil {
  411. return nil, err
  412. }
  413. if response.Rcode != dns.RcodeSuccess {
  414. return nil, RcodeError(response.Rcode)
  415. }
  416. return MessageToAddresses(response), nil
  417. }
  418. func (c *Client) questionCache(question dns.Question, transport adapter.DNSTransport) ([]netip.Addr, error) {
  419. response, _ := c.loadResponse(question, transport)
  420. if response == nil {
  421. return nil, ErrNotCached
  422. }
  423. if response.Rcode != dns.RcodeSuccess {
  424. return nil, RcodeError(response.Rcode)
  425. }
  426. return MessageToAddresses(response), nil
  427. }
  428. func (c *Client) loadResponse(question dns.Question, transport adapter.DNSTransport) (*dns.Msg, int) {
  429. var (
  430. response *dns.Msg
  431. loaded bool
  432. )
  433. if c.disableExpire {
  434. if !c.independentCache {
  435. response, loaded = c.cache.Get(question)
  436. } else {
  437. response, loaded = c.transportCache.Get(transportCacheKey{
  438. Question: question,
  439. transportTag: transport.Tag(),
  440. })
  441. }
  442. if !loaded {
  443. return nil, 0
  444. }
  445. return response.Copy(), 0
  446. } else {
  447. var expireAt time.Time
  448. if !c.independentCache {
  449. response, expireAt, loaded = c.cache.GetWithLifetime(question)
  450. } else {
  451. response, expireAt, loaded = c.transportCache.GetWithLifetime(transportCacheKey{
  452. Question: question,
  453. transportTag: transport.Tag(),
  454. })
  455. }
  456. if !loaded {
  457. return nil, 0
  458. }
  459. timeNow := time.Now()
  460. if timeNow.After(expireAt) {
  461. if !c.independentCache {
  462. c.cache.Remove(question)
  463. } else {
  464. c.transportCache.Remove(transportCacheKey{
  465. Question: question,
  466. transportTag: transport.Tag(),
  467. })
  468. }
  469. return nil, 0
  470. }
  471. var originTTL int
  472. for _, recordList := range [][]dns.RR{response.Answer, response.Ns, response.Extra} {
  473. for _, record := range recordList {
  474. if record.Header().Rrtype == dns.TypeOPT {
  475. continue
  476. }
  477. if originTTL == 0 || record.Header().Ttl > 0 && int(record.Header().Ttl) < originTTL {
  478. originTTL = int(record.Header().Ttl)
  479. }
  480. }
  481. }
  482. nowTTL := int(expireAt.Sub(timeNow).Seconds())
  483. if nowTTL < 0 {
  484. nowTTL = 0
  485. }
  486. response = response.Copy()
  487. if originTTL > 0 {
  488. duration := uint32(originTTL - nowTTL)
  489. for _, recordList := range [][]dns.RR{response.Answer, response.Ns, response.Extra} {
  490. for _, record := range recordList {
  491. if record.Header().Rrtype == dns.TypeOPT {
  492. continue
  493. }
  494. record.Header().Ttl = record.Header().Ttl - duration
  495. }
  496. }
  497. } else {
  498. for _, recordList := range [][]dns.RR{response.Answer, response.Ns, response.Extra} {
  499. for _, record := range recordList {
  500. if record.Header().Rrtype == dns.TypeOPT {
  501. continue
  502. }
  503. record.Header().Ttl = uint32(nowTTL)
  504. }
  505. }
  506. }
  507. return response, nowTTL
  508. }
  509. }
  510. func MessageToAddresses(response *dns.Msg) []netip.Addr {
  511. if response == nil || response.Rcode != dns.RcodeSuccess {
  512. return nil
  513. }
  514. addresses := make([]netip.Addr, 0, len(response.Answer))
  515. for _, rawAnswer := range response.Answer {
  516. switch answer := rawAnswer.(type) {
  517. case *dns.A:
  518. addresses = append(addresses, M.AddrFromIP(answer.A))
  519. case *dns.AAAA:
  520. addresses = append(addresses, M.AddrFromIP(answer.AAAA))
  521. case *dns.HTTPS:
  522. for _, value := range answer.SVCB.Value {
  523. if value.Key() == dns.SVCB_IPV4HINT || value.Key() == dns.SVCB_IPV6HINT {
  524. addresses = append(addresses, common.Map(strings.Split(value.String(), ","), M.ParseAddr)...)
  525. }
  526. }
  527. }
  528. }
  529. return addresses
  530. }
  531. func wrapError(err error) error {
  532. switch dnsErr := err.(type) {
  533. case *net.DNSError:
  534. if dnsErr.IsNotFound {
  535. return RcodeNameError
  536. }
  537. case *net.AddrError:
  538. return RcodeNameError
  539. }
  540. return err
  541. }
  542. type transportKey struct{}
  543. func contextWithTransportTag(ctx context.Context, transportTag string) context.Context {
  544. return context.WithValue(ctx, transportKey{}, transportTag)
  545. }
  546. func transportTagFromContext(ctx context.Context) (string, bool) {
  547. value, loaded := ctx.Value(transportKey{}).(string)
  548. return value, loaded
  549. }
  550. func FixedResponseStatus(message *dns.Msg, rcode int) *dns.Msg {
  551. return &dns.Msg{
  552. MsgHdr: dns.MsgHdr{
  553. Id: message.Id,
  554. Response: true,
  555. Authoritative: true,
  556. RecursionDesired: true,
  557. RecursionAvailable: true,
  558. Rcode: rcode,
  559. },
  560. Question: message.Question,
  561. }
  562. }
  563. func FixedResponse(id uint16, question dns.Question, addresses []netip.Addr, timeToLive uint32) *dns.Msg {
  564. response := dns.Msg{
  565. MsgHdr: dns.MsgHdr{
  566. Id: id,
  567. Response: true,
  568. Authoritative: true,
  569. RecursionDesired: true,
  570. RecursionAvailable: true,
  571. Rcode: dns.RcodeSuccess,
  572. },
  573. Question: []dns.Question{question},
  574. }
  575. for _, address := range addresses {
  576. if address.Is4() && question.Qtype == dns.TypeA {
  577. response.Answer = append(response.Answer, &dns.A{
  578. Hdr: dns.RR_Header{
  579. Name: question.Name,
  580. Rrtype: dns.TypeA,
  581. Class: dns.ClassINET,
  582. Ttl: timeToLive,
  583. },
  584. A: address.AsSlice(),
  585. })
  586. } else if address.Is6() && question.Qtype == dns.TypeAAAA {
  587. response.Answer = append(response.Answer, &dns.AAAA{
  588. Hdr: dns.RR_Header{
  589. Name: question.Name,
  590. Rrtype: dns.TypeAAAA,
  591. Class: dns.ClassINET,
  592. Ttl: timeToLive,
  593. },
  594. AAAA: address.AsSlice(),
  595. })
  596. }
  597. }
  598. return &response
  599. }
  600. func FixedResponseCNAME(id uint16, question dns.Question, record string, timeToLive uint32) *dns.Msg {
  601. response := dns.Msg{
  602. MsgHdr: dns.MsgHdr{
  603. Id: id,
  604. Response: true,
  605. Authoritative: true,
  606. RecursionDesired: true,
  607. RecursionAvailable: true,
  608. Rcode: dns.RcodeSuccess,
  609. },
  610. Question: []dns.Question{question},
  611. Answer: []dns.RR{
  612. &dns.CNAME{
  613. Hdr: dns.RR_Header{
  614. Name: question.Name,
  615. Rrtype: dns.TypeCNAME,
  616. Class: dns.ClassINET,
  617. Ttl: timeToLive,
  618. },
  619. Target: record,
  620. },
  621. },
  622. }
  623. return &response
  624. }
  625. func FixedResponseTXT(id uint16, question dns.Question, records []string, timeToLive uint32) *dns.Msg {
  626. response := dns.Msg{
  627. MsgHdr: dns.MsgHdr{
  628. Id: id,
  629. Response: true,
  630. Authoritative: true,
  631. RecursionDesired: true,
  632. RecursionAvailable: true,
  633. Rcode: dns.RcodeSuccess,
  634. },
  635. Question: []dns.Question{question},
  636. Answer: []dns.RR{
  637. &dns.TXT{
  638. Hdr: dns.RR_Header{
  639. Name: question.Name,
  640. Rrtype: dns.TypeA,
  641. Class: dns.ClassINET,
  642. Ttl: timeToLive,
  643. },
  644. Txt: records,
  645. },
  646. },
  647. }
  648. return &response
  649. }
  650. func FixedResponseMX(id uint16, question dns.Question, records []*net.MX, timeToLive uint32) *dns.Msg {
  651. response := dns.Msg{
  652. MsgHdr: dns.MsgHdr{
  653. Id: id,
  654. Response: true,
  655. Authoritative: true,
  656. RecursionDesired: true,
  657. RecursionAvailable: true,
  658. Rcode: dns.RcodeSuccess,
  659. },
  660. Question: []dns.Question{question},
  661. }
  662. for _, record := range records {
  663. response.Answer = append(response.Answer, &dns.MX{
  664. Hdr: dns.RR_Header{
  665. Name: question.Name,
  666. Rrtype: dns.TypeA,
  667. Class: dns.ClassINET,
  668. Ttl: timeToLive,
  669. },
  670. Preference: record.Pref,
  671. Mx: record.Host,
  672. })
  673. }
  674. return &response
  675. }