dialer.go 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  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. }
  25. // TODO: merge with NewWithOptions
  26. func New(ctx context.Context, options option.DialerOptions, remoteIsDomain bool) (N.Dialer, error) {
  27. return NewWithOptions(Options{
  28. Context: ctx,
  29. Options: options,
  30. RemoteIsDomain: remoteIsDomain,
  31. })
  32. }
  33. func NewWithOptions(options Options) (N.Dialer, error) {
  34. dialOptions := options.Options
  35. var (
  36. dialer N.Dialer
  37. err error
  38. )
  39. if dialOptions.Detour != "" {
  40. outboundManager := service.FromContext[adapter.OutboundManager](options.Context)
  41. if outboundManager == nil {
  42. return nil, E.New("missing outbound manager")
  43. }
  44. dialer = NewDetour(outboundManager, dialOptions.Detour, options.LegacyDNSDialer)
  45. } else {
  46. dialer, err = NewDefault(options.Context, dialOptions)
  47. if err != nil {
  48. return nil, err
  49. }
  50. }
  51. if options.RemoteIsDomain && (dialOptions.Detour == "" || options.ResolverOnDetour || dialOptions.DomainResolver != nil && dialOptions.DomainResolver.Server != "") {
  52. networkManager := service.FromContext[adapter.NetworkManager](options.Context)
  53. dnsTransport := service.FromContext[adapter.DNSTransportManager](options.Context)
  54. var defaultOptions adapter.NetworkOptions
  55. if networkManager != nil {
  56. defaultOptions = networkManager.DefaultOptions()
  57. }
  58. var (
  59. server string
  60. dnsQueryOptions adapter.DNSQueryOptions
  61. resolveFallbackDelay time.Duration
  62. )
  63. if dialOptions.DomainResolver != nil && dialOptions.DomainResolver.Server != "" {
  64. var transport adapter.DNSTransport
  65. if !options.DirectResolver {
  66. var loaded bool
  67. transport, loaded = dnsTransport.Transport(dialOptions.DomainResolver.Server)
  68. if !loaded {
  69. return nil, E.New("domain resolver not found: " + dialOptions.DomainResolver.Server)
  70. }
  71. }
  72. var strategy C.DomainStrategy
  73. if dialOptions.DomainResolver.Strategy != option.DomainStrategy(C.DomainStrategyAsIS) {
  74. strategy = C.DomainStrategy(dialOptions.DomainResolver.Strategy)
  75. } else if
  76. //nolint:staticcheck
  77. dialOptions.DomainStrategy != option.DomainStrategy(C.DomainStrategyAsIS) {
  78. //nolint:staticcheck
  79. strategy = C.DomainStrategy(dialOptions.DomainStrategy)
  80. }
  81. server = dialOptions.DomainResolver.Server
  82. dnsQueryOptions = adapter.DNSQueryOptions{
  83. Transport: transport,
  84. Strategy: strategy,
  85. DisableCache: dialOptions.DomainResolver.DisableCache,
  86. RewriteTTL: dialOptions.DomainResolver.RewriteTTL,
  87. ClientSubnet: dialOptions.DomainResolver.ClientSubnet.Build(netip.Prefix{}),
  88. }
  89. resolveFallbackDelay = time.Duration(dialOptions.FallbackDelay)
  90. } else if options.DirectResolver {
  91. return nil, E.New("missing domain resolver for domain server address")
  92. } else if defaultOptions.DomainResolver != "" {
  93. dnsQueryOptions = defaultOptions.DomainResolveOptions
  94. transport, loaded := dnsTransport.Transport(defaultOptions.DomainResolver)
  95. if !loaded {
  96. return nil, E.New("default domain resolver not found: " + defaultOptions.DomainResolver)
  97. }
  98. dnsQueryOptions.Transport = transport
  99. resolveFallbackDelay = time.Duration(dialOptions.FallbackDelay)
  100. } else if options.NewDialer {
  101. return nil, E.New("missing domain resolver for domain server address")
  102. } else {
  103. transports := dnsTransport.Transports()
  104. if len(transports) < 2 {
  105. dnsQueryOptions.Transport = dnsTransport.Default()
  106. } else {
  107. deprecated.Report(options.Context, deprecated.OptionMissingDomainResolver)
  108. }
  109. }
  110. dialer = NewResolveDialer(
  111. options.Context,
  112. dialer,
  113. dialOptions.Detour == "" && !dialOptions.TCPFastOpen,
  114. server,
  115. dnsQueryOptions,
  116. resolveFallbackDelay,
  117. )
  118. }
  119. return dialer, nil
  120. }
  121. type ParallelInterfaceDialer interface {
  122. N.Dialer
  123. DialParallelInterface(ctx context.Context, network string, destination M.Socksaddr, strategy *C.NetworkStrategy, interfaceType []C.InterfaceType, fallbackInterfaceType []C.InterfaceType, fallbackDelay time.Duration) (net.Conn, error)
  124. ListenSerialInterfacePacket(ctx context.Context, destination M.Socksaddr, strategy *C.NetworkStrategy, interfaceType []C.InterfaceType, fallbackInterfaceType []C.InterfaceType, fallbackDelay time.Duration) (net.PacketConn, error)
  125. }
  126. type ParallelNetworkDialer interface {
  127. 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)
  128. 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)
  129. }