default.go 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. package dialer
  2. import (
  3. "context"
  4. "net"
  5. "time"
  6. "github.com/sagernet/sing-box/adapter"
  7. "github.com/sagernet/sing-box/common/dialer/conntrack"
  8. "github.com/sagernet/sing-box/common/warning"
  9. C "github.com/sagernet/sing-box/constant"
  10. "github.com/sagernet/sing-box/option"
  11. "github.com/sagernet/sing/common/control"
  12. E "github.com/sagernet/sing/common/exceptions"
  13. M "github.com/sagernet/sing/common/metadata"
  14. N "github.com/sagernet/sing/common/network"
  15. "github.com/sagernet/tfo-go"
  16. )
  17. var warnBindInterfaceOnUnsupportedPlatform = warning.New(
  18. func() bool {
  19. return !(C.IsLinux || C.IsWindows || C.IsDarwin)
  20. },
  21. "outbound option `bind_interface` is only supported on Linux and Windows",
  22. )
  23. var warnRoutingMarkOnUnsupportedPlatform = warning.New(
  24. func() bool {
  25. return !C.IsLinux
  26. },
  27. "outbound option `routing_mark` is only supported on Linux",
  28. )
  29. var warnReuseAdderOnUnsupportedPlatform = warning.New(
  30. func() bool {
  31. return !(C.IsDarwin || C.IsDragonfly || C.IsFreebsd || C.IsLinux || C.IsNetbsd || C.IsOpenbsd || C.IsSolaris || C.IsWindows)
  32. },
  33. "outbound option `reuse_addr` is unsupported on current platform",
  34. )
  35. var warnProtectPathOnNonAndroid = warning.New(
  36. func() bool {
  37. return !C.IsAndroid
  38. },
  39. "outbound option `protect_path` is only supported on Android",
  40. )
  41. var warnTFOOnUnsupportedPlatform = warning.New(
  42. func() bool {
  43. return !(C.IsDarwin || C.IsFreebsd || C.IsLinux || C.IsWindows)
  44. },
  45. "outbound option `tcp_fast_open` is unsupported on current platform",
  46. )
  47. type DefaultDialer struct {
  48. dialer4 tfo.Dialer
  49. dialer6 tfo.Dialer
  50. udpDialer4 net.Dialer
  51. udpDialer6 net.Dialer
  52. udpListener net.ListenConfig
  53. udpAddr4 string
  54. udpAddr6 string
  55. }
  56. func NewDefault(router adapter.Router, options option.DialerOptions) *DefaultDialer {
  57. var dialer net.Dialer
  58. var listener net.ListenConfig
  59. if options.BindInterface != "" {
  60. warnBindInterfaceOnUnsupportedPlatform.Check()
  61. bindFunc := control.BindToInterface(router.InterfaceFinder(), options.BindInterface, -1)
  62. dialer.Control = control.Append(dialer.Control, bindFunc)
  63. listener.Control = control.Append(listener.Control, bindFunc)
  64. } else if router.AutoDetectInterface() {
  65. bindFunc := router.AutoDetectInterfaceFunc()
  66. dialer.Control = control.Append(dialer.Control, bindFunc)
  67. listener.Control = control.Append(listener.Control, bindFunc)
  68. } else if router.DefaultInterface() != "" {
  69. bindFunc := control.BindToInterface(router.InterfaceFinder(), router.DefaultInterface(), -1)
  70. dialer.Control = control.Append(dialer.Control, bindFunc)
  71. listener.Control = control.Append(listener.Control, bindFunc)
  72. }
  73. if options.RoutingMark != 0 {
  74. warnRoutingMarkOnUnsupportedPlatform.Check()
  75. dialer.Control = control.Append(dialer.Control, control.RoutingMark(options.RoutingMark))
  76. listener.Control = control.Append(listener.Control, control.RoutingMark(options.RoutingMark))
  77. } else if router.DefaultMark() != 0 {
  78. dialer.Control = control.Append(dialer.Control, control.RoutingMark(router.DefaultMark()))
  79. listener.Control = control.Append(listener.Control, control.RoutingMark(router.DefaultMark()))
  80. }
  81. if options.ReuseAddr {
  82. warnReuseAdderOnUnsupportedPlatform.Check()
  83. listener.Control = control.Append(listener.Control, control.ReuseAddr())
  84. }
  85. if options.ProtectPath != "" {
  86. warnProtectPathOnNonAndroid.Check()
  87. dialer.Control = control.Append(dialer.Control, control.ProtectPath(options.ProtectPath))
  88. listener.Control = control.Append(listener.Control, control.ProtectPath(options.ProtectPath))
  89. }
  90. if options.ConnectTimeout != 0 {
  91. dialer.Timeout = time.Duration(options.ConnectTimeout)
  92. } else {
  93. dialer.Timeout = C.TCPTimeout
  94. }
  95. if options.TCPFastOpen {
  96. warnTFOOnUnsupportedPlatform.Check()
  97. }
  98. var udpFragment bool
  99. if options.UDPFragment != nil {
  100. udpFragment = *options.UDPFragment
  101. } else {
  102. udpFragment = options.UDPFragmentDefault
  103. }
  104. if !udpFragment {
  105. dialer.Control = control.Append(dialer.Control, control.DisableUDPFragment())
  106. listener.Control = control.Append(listener.Control, control.DisableUDPFragment())
  107. }
  108. var (
  109. dialer4 = dialer
  110. udpDialer4 = dialer
  111. udpAddr4 string
  112. )
  113. if options.Inet4BindAddress != nil {
  114. bindAddr := options.Inet4BindAddress.Build()
  115. dialer4.LocalAddr = &net.TCPAddr{IP: bindAddr.AsSlice()}
  116. udpDialer4.LocalAddr = &net.UDPAddr{IP: bindAddr.AsSlice()}
  117. udpAddr4 = M.SocksaddrFrom(bindAddr, 0).String()
  118. }
  119. var (
  120. dialer6 = dialer
  121. udpDialer6 = dialer
  122. udpAddr6 string
  123. )
  124. if options.Inet6BindAddress != nil {
  125. bindAddr := options.Inet6BindAddress.Build()
  126. dialer6.LocalAddr = &net.TCPAddr{IP: bindAddr.AsSlice()}
  127. udpDialer6.LocalAddr = &net.UDPAddr{IP: bindAddr.AsSlice()}
  128. udpAddr6 = M.SocksaddrFrom(bindAddr, 0).String()
  129. }
  130. return &DefaultDialer{
  131. tfo.Dialer{Dialer: dialer4, DisableTFO: !options.TCPFastOpen},
  132. tfo.Dialer{Dialer: dialer6, DisableTFO: !options.TCPFastOpen},
  133. udpDialer4,
  134. udpDialer6,
  135. listener,
  136. udpAddr4,
  137. udpAddr6,
  138. }
  139. }
  140. func (d *DefaultDialer) DialContext(ctx context.Context, network string, address M.Socksaddr) (net.Conn, error) {
  141. if !address.IsValid() {
  142. return nil, E.New("invalid address")
  143. }
  144. switch N.NetworkName(network) {
  145. case N.NetworkUDP:
  146. if !address.IsIPv6() {
  147. return d.udpDialer4.DialContext(ctx, network, address.String())
  148. } else {
  149. return d.udpDialer6.DialContext(ctx, network, address.String())
  150. }
  151. }
  152. if !address.IsIPv6() {
  153. return trackConn(DialSlowContext(&d.dialer4, ctx, network, address))
  154. } else {
  155. return trackConn(DialSlowContext(&d.dialer6, ctx, network, address))
  156. }
  157. }
  158. func (d *DefaultDialer) ListenPacket(ctx context.Context, destination M.Socksaddr) (net.PacketConn, error) {
  159. if !destination.IsIPv6() {
  160. return trackPacketConn(d.udpListener.ListenPacket(ctx, N.NetworkUDP, d.udpAddr4))
  161. } else {
  162. return trackPacketConn(d.udpListener.ListenPacket(ctx, N.NetworkUDP, d.udpAddr6))
  163. }
  164. }
  165. func trackConn(conn net.Conn, err error) (net.Conn, error) {
  166. if !conntrack.Enabled || err != nil {
  167. return conn, err
  168. }
  169. return conntrack.NewConn(conn)
  170. }
  171. func trackPacketConn(conn net.PacketConn, err error) (net.PacketConn, error) {
  172. if !conntrack.Enabled || err != nil {
  173. return conn, err
  174. }
  175. return conntrack.NewPacketConn(conn)
  176. }