dialer.go 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. package dialer
  2. import (
  3. "context"
  4. "net"
  5. "net/netip"
  6. "time"
  7. "github.com/sagernet/sing-box/adapter"
  8. C "github.com/sagernet/sing-box/constant"
  9. "github.com/sagernet/sing-box/experimental/deprecated"
  10. "github.com/sagernet/sing-box/option"
  11. E "github.com/sagernet/sing/common/exceptions"
  12. M "github.com/sagernet/sing/common/metadata"
  13. N "github.com/sagernet/sing/common/network"
  14. "github.com/sagernet/sing/service"
  15. )
  16. type Options struct {
  17. Context context.Context
  18. Options option.DialerOptions
  19. RemoteIsDomain bool
  20. DirectResolver bool
  21. ResolverOnDetour bool
  22. NewDialer bool
  23. LegacyDNSDialer bool
  24. DirectOutbound bool
  25. }
  26. // TODO: merge with NewWithOptions
  27. func New(ctx context.Context, options option.DialerOptions, remoteIsDomain bool) (N.Dialer, error) {
  28. return NewWithOptions(Options{
  29. Context: ctx,
  30. Options: options,
  31. RemoteIsDomain: remoteIsDomain,
  32. })
  33. }
  34. func NewWithOptions(options Options) (N.Dialer, error) {
  35. dialOptions := options.Options
  36. var (
  37. dialer N.Dialer
  38. err error
  39. )
  40. if dialOptions.Detour != "" {
  41. outboundManager := service.FromContext[adapter.OutboundManager](options.Context)
  42. if outboundManager == nil {
  43. return nil, E.New("missing outbound manager")
  44. }
  45. dialer = NewDetour(outboundManager, dialOptions.Detour, options.LegacyDNSDialer)
  46. } else {
  47. dialer, err = NewDefault(options.Context, dialOptions)
  48. if err != nil {
  49. return nil, err
  50. }
  51. }
  52. if options.RemoteIsDomain && (dialOptions.Detour == "" || options.ResolverOnDetour || dialOptions.DomainResolver != nil && dialOptions.DomainResolver.Server != "") {
  53. networkManager := service.FromContext[adapter.NetworkManager](options.Context)
  54. dnsTransport := service.FromContext[adapter.DNSTransportManager](options.Context)
  55. var defaultOptions adapter.NetworkOptions
  56. if networkManager != nil {
  57. defaultOptions = networkManager.DefaultOptions()
  58. }
  59. var (
  60. server string
  61. dnsQueryOptions adapter.DNSQueryOptions
  62. resolveFallbackDelay time.Duration
  63. )
  64. if dialOptions.DomainResolver != nil && dialOptions.DomainResolver.Server != "" {
  65. var transport adapter.DNSTransport
  66. if !options.DirectResolver {
  67. var loaded bool
  68. transport, loaded = dnsTransport.Transport(dialOptions.DomainResolver.Server)
  69. if !loaded {
  70. return nil, E.New("domain resolver not found: " + dialOptions.DomainResolver.Server)
  71. }
  72. }
  73. var strategy C.DomainStrategy
  74. if dialOptions.DomainResolver.Strategy != option.DomainStrategy(C.DomainStrategyAsIS) {
  75. strategy = C.DomainStrategy(dialOptions.DomainResolver.Strategy)
  76. } else if
  77. //nolint:staticcheck
  78. dialOptions.DomainStrategy != option.DomainStrategy(C.DomainStrategyAsIS) {
  79. //nolint:staticcheck
  80. strategy = C.DomainStrategy(dialOptions.DomainStrategy)
  81. deprecated.Report(options.Context, deprecated.OptionLegacyDomainStrategyOptions)
  82. }
  83. server = dialOptions.DomainResolver.Server
  84. dnsQueryOptions = adapter.DNSQueryOptions{
  85. Transport: transport,
  86. Strategy: strategy,
  87. Timeout: time.Duration(dialOptions.DomainResolver.Timeout),
  88. DisableCache: dialOptions.DomainResolver.DisableCache,
  89. RewriteTTL: dialOptions.DomainResolver.RewriteTTL,
  90. ClientSubnet: dialOptions.DomainResolver.ClientSubnet.Build(netip.Prefix{}),
  91. }
  92. resolveFallbackDelay = time.Duration(dialOptions.FallbackDelay)
  93. } else if options.DirectResolver {
  94. return nil, E.New("missing domain resolver for domain server address")
  95. } else {
  96. if defaultOptions.DomainResolver != "" {
  97. dnsQueryOptions = defaultOptions.DomainResolveOptions
  98. transport, loaded := dnsTransport.Transport(defaultOptions.DomainResolver)
  99. if !loaded {
  100. return nil, E.New("default domain resolver not found: " + defaultOptions.DomainResolver)
  101. }
  102. dnsQueryOptions.Transport = transport
  103. resolveFallbackDelay = time.Duration(dialOptions.FallbackDelay)
  104. } else {
  105. transports := dnsTransport.Transports()
  106. if len(transports) < 2 {
  107. dnsQueryOptions.Transport = dnsTransport.Default()
  108. } else if options.NewDialer {
  109. return nil, E.New("missing domain resolver for domain server address")
  110. } else if !options.DirectOutbound {
  111. deprecated.Report(options.Context, deprecated.OptionMissingDomainResolver)
  112. }
  113. }
  114. if
  115. //nolint:staticcheck
  116. dialOptions.DomainStrategy != option.DomainStrategy(C.DomainStrategyAsIS) {
  117. //nolint:staticcheck
  118. dnsQueryOptions.Strategy = C.DomainStrategy(dialOptions.DomainStrategy)
  119. deprecated.Report(options.Context, deprecated.OptionLegacyDomainStrategyOptions)
  120. }
  121. }
  122. dialer = NewResolveDialer(
  123. options.Context,
  124. dialer,
  125. dialOptions.Detour == "" && !dialOptions.TCPFastOpen,
  126. server,
  127. dnsQueryOptions,
  128. resolveFallbackDelay,
  129. )
  130. }
  131. return dialer, nil
  132. }
  133. type ParallelInterfaceDialer interface {
  134. N.Dialer
  135. DialParallelInterface(ctx context.Context, network string, destination M.Socksaddr, strategy *C.NetworkStrategy, interfaceType []C.InterfaceType, fallbackInterfaceType []C.InterfaceType, fallbackDelay time.Duration) (net.Conn, error)
  136. ListenSerialInterfacePacket(ctx context.Context, destination M.Socksaddr, strategy *C.NetworkStrategy, interfaceType []C.InterfaceType, fallbackInterfaceType []C.InterfaceType, fallbackDelay time.Duration) (net.PacketConn, error)
  137. }
  138. type ParallelNetworkDialer interface {
  139. DialParallelNetwork(ctx context.Context, network string, destination M.Socksaddr, destinationAddresses []netip.Addr, strategy *C.NetworkStrategy, interfaceType []C.InterfaceType, fallbackInterfaceType []C.InterfaceType, fallbackDelay time.Duration) (net.Conn, error)
  140. ListenSerialNetworkPacket(ctx context.Context, destination M.Socksaddr, destinationAddresses []netip.Addr, strategy *C.NetworkStrategy, interfaceType []C.InterfaceType, fallbackInterfaceType []C.InterfaceType, fallbackDelay time.Duration) (net.PacketConn, netip.Addr, error)
  141. }