default.go 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198
  1. package dialer
  2. import (
  3. "context"
  4. "net"
  5. "net/netip"
  6. "time"
  7. "github.com/sagernet/sing-box/adapter"
  8. "github.com/sagernet/sing-box/common/conntrack"
  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. )
  16. var _ WireGuardListener = (*DefaultDialer)(nil)
  17. type DefaultDialer struct {
  18. dialer4 tcpDialer
  19. dialer6 tcpDialer
  20. udpDialer4 net.Dialer
  21. udpDialer6 net.Dialer
  22. udpListener net.ListenConfig
  23. udpAddr4 string
  24. udpAddr6 string
  25. isWireGuardListener bool
  26. }
  27. func NewDefault(networkManager adapter.NetworkManager, options option.DialerOptions) (*DefaultDialer, error) {
  28. var dialer net.Dialer
  29. var listener net.ListenConfig
  30. if options.BindInterface != "" {
  31. var interfaceFinder control.InterfaceFinder
  32. if networkManager != nil {
  33. interfaceFinder = networkManager.InterfaceFinder()
  34. } else {
  35. interfaceFinder = control.NewDefaultInterfaceFinder()
  36. }
  37. bindFunc := control.BindToInterface(interfaceFinder, options.BindInterface, -1)
  38. dialer.Control = control.Append(dialer.Control, bindFunc)
  39. listener.Control = control.Append(listener.Control, bindFunc)
  40. } else if networkManager != nil && networkManager.AutoDetectInterface() {
  41. bindFunc := networkManager.AutoDetectInterfaceFunc()
  42. dialer.Control = control.Append(dialer.Control, bindFunc)
  43. listener.Control = control.Append(listener.Control, bindFunc)
  44. } else if networkManager != nil && networkManager.DefaultInterface() != "" {
  45. bindFunc := control.BindToInterface(networkManager.InterfaceFinder(), networkManager.DefaultInterface(), -1)
  46. dialer.Control = control.Append(dialer.Control, bindFunc)
  47. listener.Control = control.Append(listener.Control, bindFunc)
  48. }
  49. var autoRedirectOutputMark uint32
  50. if networkManager != nil {
  51. autoRedirectOutputMark = networkManager.AutoRedirectOutputMark()
  52. }
  53. if autoRedirectOutputMark > 0 {
  54. dialer.Control = control.Append(dialer.Control, control.RoutingMark(autoRedirectOutputMark))
  55. listener.Control = control.Append(listener.Control, control.RoutingMark(autoRedirectOutputMark))
  56. }
  57. if options.RoutingMark > 0 {
  58. dialer.Control = control.Append(dialer.Control, control.RoutingMark(options.RoutingMark))
  59. listener.Control = control.Append(listener.Control, control.RoutingMark(options.RoutingMark))
  60. if autoRedirectOutputMark > 0 {
  61. return nil, E.New("`auto_redirect` with `route_[_exclude]_address_set is conflict with `routing_mark`")
  62. }
  63. } else if networkManager != nil && networkManager.DefaultMark() > 0 {
  64. dialer.Control = control.Append(dialer.Control, control.RoutingMark(networkManager.DefaultMark()))
  65. listener.Control = control.Append(listener.Control, control.RoutingMark(networkManager.DefaultMark()))
  66. if autoRedirectOutputMark > 0 {
  67. return nil, E.New("`auto_redirect` with `route_[_exclude]_address_set is conflict with `default_mark`")
  68. }
  69. }
  70. if options.ReuseAddr {
  71. listener.Control = control.Append(listener.Control, control.ReuseAddr())
  72. }
  73. if options.ProtectPath != "" {
  74. dialer.Control = control.Append(dialer.Control, control.ProtectPath(options.ProtectPath))
  75. listener.Control = control.Append(listener.Control, control.ProtectPath(options.ProtectPath))
  76. }
  77. if options.ConnectTimeout != 0 {
  78. dialer.Timeout = time.Duration(options.ConnectTimeout)
  79. } else {
  80. dialer.Timeout = C.TCPConnectTimeout
  81. }
  82. // TODO: Add an option to customize the keep alive period
  83. dialer.KeepAlive = C.TCPKeepAliveInitial
  84. dialer.Control = control.Append(dialer.Control, control.SetKeepAlivePeriod(C.TCPKeepAliveInitial, C.TCPKeepAliveInterval))
  85. var udpFragment bool
  86. if options.UDPFragment != nil {
  87. udpFragment = *options.UDPFragment
  88. } else {
  89. udpFragment = options.UDPFragmentDefault
  90. }
  91. if !udpFragment {
  92. dialer.Control = control.Append(dialer.Control, control.DisableUDPFragment())
  93. listener.Control = control.Append(listener.Control, control.DisableUDPFragment())
  94. }
  95. var (
  96. dialer4 = dialer
  97. udpDialer4 = dialer
  98. udpAddr4 string
  99. )
  100. if options.Inet4BindAddress != nil {
  101. bindAddr := options.Inet4BindAddress.Build(netip.IPv4Unspecified())
  102. dialer4.LocalAddr = &net.TCPAddr{IP: bindAddr.AsSlice()}
  103. udpDialer4.LocalAddr = &net.UDPAddr{IP: bindAddr.AsSlice()}
  104. udpAddr4 = M.SocksaddrFrom(bindAddr, 0).String()
  105. }
  106. var (
  107. dialer6 = dialer
  108. udpDialer6 = dialer
  109. udpAddr6 string
  110. )
  111. if options.Inet6BindAddress != nil {
  112. bindAddr := options.Inet6BindAddress.Build(netip.IPv6Unspecified())
  113. dialer6.LocalAddr = &net.TCPAddr{IP: bindAddr.AsSlice()}
  114. udpDialer6.LocalAddr = &net.UDPAddr{IP: bindAddr.AsSlice()}
  115. udpAddr6 = M.SocksaddrFrom(bindAddr, 0).String()
  116. }
  117. if options.TCPMultiPath {
  118. if !go121Available {
  119. return nil, E.New("MultiPath TCP requires go1.21, please recompile your binary.")
  120. }
  121. setMultiPathTCP(&dialer4)
  122. }
  123. if options.IsWireGuardListener {
  124. for _, controlFn := range WgControlFns {
  125. listener.Control = control.Append(listener.Control, controlFn)
  126. }
  127. }
  128. tcpDialer4, err := newTCPDialer(dialer4, options.TCPFastOpen)
  129. if err != nil {
  130. return nil, err
  131. }
  132. tcpDialer6, err := newTCPDialer(dialer6, options.TCPFastOpen)
  133. if err != nil {
  134. return nil, err
  135. }
  136. return &DefaultDialer{
  137. tcpDialer4,
  138. tcpDialer6,
  139. udpDialer4,
  140. udpDialer6,
  141. listener,
  142. udpAddr4,
  143. udpAddr6,
  144. options.IsWireGuardListener,
  145. }, nil
  146. }
  147. func (d *DefaultDialer) DialContext(ctx context.Context, network string, address M.Socksaddr) (net.Conn, error) {
  148. if !address.IsValid() {
  149. return nil, E.New("invalid address")
  150. }
  151. switch N.NetworkName(network) {
  152. case N.NetworkUDP:
  153. if !address.IsIPv6() {
  154. return trackConn(d.udpDialer4.DialContext(ctx, network, address.String()))
  155. } else {
  156. return trackConn(d.udpDialer6.DialContext(ctx, network, address.String()))
  157. }
  158. }
  159. if !address.IsIPv6() {
  160. return trackConn(DialSlowContext(&d.dialer4, ctx, network, address))
  161. } else {
  162. return trackConn(DialSlowContext(&d.dialer6, ctx, network, address))
  163. }
  164. }
  165. func (d *DefaultDialer) ListenPacket(ctx context.Context, destination M.Socksaddr) (net.PacketConn, error) {
  166. if destination.IsIPv6() {
  167. return trackPacketConn(d.udpListener.ListenPacket(ctx, N.NetworkUDP, d.udpAddr6))
  168. } else if destination.IsIPv4() && !destination.Addr.IsUnspecified() {
  169. return trackPacketConn(d.udpListener.ListenPacket(ctx, N.NetworkUDP+"4", d.udpAddr4))
  170. } else {
  171. return trackPacketConn(d.udpListener.ListenPacket(ctx, N.NetworkUDP, d.udpAddr4))
  172. }
  173. }
  174. func (d *DefaultDialer) ListenPacketCompat(network, address string) (net.PacketConn, error) {
  175. return d.udpListener.ListenPacket(context.Background(), network, address)
  176. }
  177. func trackConn(conn net.Conn, err error) (net.Conn, error) {
  178. if !conntrack.Enabled || err != nil {
  179. return conn, err
  180. }
  181. return conntrack.NewConn(conn)
  182. }
  183. func trackPacketConn(conn net.PacketConn, err error) (net.PacketConn, error) {
  184. if !conntrack.Enabled || err != nil {
  185. return conn, err
  186. }
  187. return conntrack.NewPacketConn(conn)
  188. }