dialer.go 5.0 KB

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