dhcp_shared.go 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197
  1. package dhcp
  2. import (
  3. "context"
  4. "math/rand"
  5. "strings"
  6. "time"
  7. C "github.com/sagernet/sing-box/constant"
  8. "github.com/sagernet/sing-box/dns"
  9. "github.com/sagernet/sing/common/buf"
  10. E "github.com/sagernet/sing/common/exceptions"
  11. M "github.com/sagernet/sing/common/metadata"
  12. N "github.com/sagernet/sing/common/network"
  13. mDNS "github.com/miekg/dns"
  14. )
  15. func (t *Transport) exchangeSingleRequest(ctx context.Context, servers []M.Socksaddr, message *mDNS.Msg, domain string) (*mDNS.Msg, error) {
  16. var lastErr error
  17. for _, fqdn := range t.nameList(domain) {
  18. response, err := t.tryOneName(ctx, servers, fqdn, message)
  19. if err != nil {
  20. lastErr = err
  21. continue
  22. }
  23. return response, nil
  24. }
  25. return nil, lastErr
  26. }
  27. func (t *Transport) exchangeParallel(ctx context.Context, servers []M.Socksaddr, message *mDNS.Msg, domain string) (*mDNS.Msg, error) {
  28. returned := make(chan struct{})
  29. defer close(returned)
  30. type queryResult struct {
  31. response *mDNS.Msg
  32. err error
  33. }
  34. results := make(chan queryResult)
  35. startRacer := func(ctx context.Context, fqdn string) {
  36. response, err := t.tryOneName(ctx, servers, fqdn, message)
  37. if err == nil {
  38. if response.Rcode != mDNS.RcodeSuccess {
  39. err = dns.RcodeError(response.Rcode)
  40. } else if len(dns.MessageToAddresses(response)) == 0 {
  41. err = E.New(fqdn, ": empty result")
  42. }
  43. }
  44. select {
  45. case results <- queryResult{response, err}:
  46. case <-returned:
  47. }
  48. }
  49. queryCtx, queryCancel := context.WithCancel(ctx)
  50. defer queryCancel()
  51. var nameCount int
  52. for _, fqdn := range t.nameList(domain) {
  53. nameCount++
  54. go startRacer(queryCtx, fqdn)
  55. }
  56. var errors []error
  57. for {
  58. select {
  59. case <-ctx.Done():
  60. return nil, ctx.Err()
  61. case result := <-results:
  62. if result.err == nil {
  63. return result.response, nil
  64. }
  65. errors = append(errors, result.err)
  66. if len(errors) == nameCount {
  67. return nil, E.Errors(errors...)
  68. }
  69. }
  70. }
  71. }
  72. func (t *Transport) tryOneName(ctx context.Context, servers []M.Socksaddr, fqdn string, message *mDNS.Msg) (*mDNS.Msg, error) {
  73. sLen := len(servers)
  74. var lastErr error
  75. for i := 0; i < t.attempts; i++ {
  76. for j := 0; j < sLen; j++ {
  77. server := servers[j]
  78. question := message.Question[0]
  79. question.Name = fqdn
  80. response, err := t.exchangeOne(ctx, server, question, C.DNSTimeout, false, true)
  81. if err != nil {
  82. lastErr = err
  83. continue
  84. }
  85. return response, nil
  86. }
  87. }
  88. return nil, E.Cause(lastErr, fqdn)
  89. }
  90. func (t *Transport) exchangeOne(ctx context.Context, server M.Socksaddr, question mDNS.Question, timeout time.Duration, useTCP, ad bool) (*mDNS.Msg, error) {
  91. if server.Port == 0 {
  92. server.Port = 53
  93. }
  94. var networks []string
  95. if useTCP {
  96. networks = []string{N.NetworkTCP}
  97. } else {
  98. networks = []string{N.NetworkUDP, N.NetworkTCP}
  99. }
  100. request := &mDNS.Msg{
  101. MsgHdr: mDNS.MsgHdr{
  102. Id: uint16(rand.Uint32()),
  103. RecursionDesired: true,
  104. AuthenticatedData: ad,
  105. },
  106. Question: []mDNS.Question{question},
  107. Compress: true,
  108. }
  109. request.SetEdns0(buf.UDPBufferSize, false)
  110. buffer := buf.Get(buf.UDPBufferSize)
  111. defer buf.Put(buffer)
  112. for _, network := range networks {
  113. ctx, cancel := context.WithDeadline(ctx, time.Now().Add(timeout))
  114. defer cancel()
  115. conn, err := t.dialer.DialContext(ctx, network, server)
  116. if err != nil {
  117. return nil, err
  118. }
  119. defer conn.Close()
  120. if deadline, loaded := ctx.Deadline(); loaded && !deadline.IsZero() {
  121. conn.SetDeadline(deadline)
  122. }
  123. rawMessage, err := request.PackBuffer(buffer)
  124. if err != nil {
  125. return nil, E.Cause(err, "pack request")
  126. }
  127. _, err = conn.Write(rawMessage)
  128. if err != nil {
  129. return nil, E.Cause(err, "write request")
  130. }
  131. n, err := conn.Read(buffer)
  132. if err != nil {
  133. return nil, E.Cause(err, "read response")
  134. }
  135. var response mDNS.Msg
  136. err = response.Unpack(buffer[:n])
  137. if err != nil {
  138. return nil, E.Cause(err, "unpack response")
  139. }
  140. if response.Truncated && network == N.NetworkUDP {
  141. continue
  142. }
  143. return &response, nil
  144. }
  145. panic("unexpected")
  146. }
  147. func (t *Transport) nameList(name string) []string {
  148. l := len(name)
  149. rooted := l > 0 && name[l-1] == '.'
  150. if l > 254 || l == 254 && !rooted {
  151. return nil
  152. }
  153. if rooted {
  154. if avoidDNS(name) {
  155. return nil
  156. }
  157. return []string{name}
  158. }
  159. hasNdots := strings.Count(name, ".") >= t.ndots
  160. name += "."
  161. // l++
  162. names := make([]string, 0, 1+len(t.search))
  163. if hasNdots && !avoidDNS(name) {
  164. names = append(names, name)
  165. }
  166. for _, suffix := range t.search {
  167. fqdn := name + suffix
  168. if !avoidDNS(fqdn) && len(fqdn) <= 254 {
  169. names = append(names, fqdn)
  170. }
  171. }
  172. if !hasNdots && !avoidDNS(name) {
  173. names = append(names, name)
  174. }
  175. return names
  176. }
  177. func avoidDNS(name string) bool {
  178. if name == "" {
  179. return true
  180. }
  181. if name[len(name)-1] == '.' {
  182. name = name[:len(name)-1]
  183. }
  184. return strings.HasSuffix(name, ".onion")
  185. }