system_dialer.go 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. package internet
  2. import (
  3. "context"
  4. "syscall"
  5. "time"
  6. "github.com/xtls/xray-core/common/net"
  7. "github.com/xtls/xray-core/common/session"
  8. "github.com/xtls/xray-core/features/dns"
  9. "github.com/xtls/xray-core/features/outbound"
  10. )
  11. var effectiveSystemDialer SystemDialer = &DefaultSystemDialer{}
  12. type SystemDialer interface {
  13. Dial(ctx context.Context, source net.Address, destination net.Destination, sockopt *SocketConfig) (net.Conn, error)
  14. }
  15. type DefaultSystemDialer struct {
  16. controllers []controller
  17. dns dns.Client
  18. obm outbound.Manager
  19. }
  20. func resolveSrcAddr(network net.Network, src net.Address) net.Addr {
  21. if src == nil || src == net.AnyIP {
  22. return nil
  23. }
  24. if network == net.Network_TCP {
  25. return &net.TCPAddr{
  26. IP: src.IP(),
  27. Port: 0,
  28. }
  29. }
  30. return &net.UDPAddr{
  31. IP: src.IP(),
  32. Port: 0,
  33. }
  34. }
  35. func hasBindAddr(sockopt *SocketConfig) bool {
  36. return sockopt != nil && len(sockopt.BindAddress) > 0 && sockopt.BindPort > 0
  37. }
  38. func (d *DefaultSystemDialer) Dial(ctx context.Context, src net.Address, dest net.Destination, sockopt *SocketConfig) (net.Conn, error) {
  39. newError("dialing to " + dest.String()).AtDebug().WriteToLog()
  40. if dest.Network == net.Network_UDP && !hasBindAddr(sockopt) {
  41. srcAddr := resolveSrcAddr(net.Network_UDP, src)
  42. if srcAddr == nil {
  43. srcAddr = &net.UDPAddr{
  44. IP: []byte{0, 0, 0, 0},
  45. Port: 0,
  46. }
  47. }
  48. packetConn, err := ListenSystemPacket(ctx, srcAddr, sockopt)
  49. if err != nil {
  50. return nil, err
  51. }
  52. destAddr, err := net.ResolveUDPAddr("udp", dest.NetAddr())
  53. if err != nil {
  54. return nil, err
  55. }
  56. return &PacketConnWrapper{
  57. Conn: packetConn,
  58. Dest: destAddr,
  59. }, nil
  60. }
  61. dialer := &net.Dialer{
  62. Timeout: time.Second * 16,
  63. LocalAddr: resolveSrcAddr(dest.Network, src),
  64. }
  65. if sockopt != nil || len(d.controllers) > 0 {
  66. dialer.Control = func(network, address string, c syscall.RawConn) error {
  67. return c.Control(func(fd uintptr) {
  68. if sockopt != nil {
  69. if err := applyOutboundSocketOptions(network, address, fd, sockopt); err != nil {
  70. newError("failed to apply socket options").Base(err).WriteToLog(session.ExportIDToError(ctx))
  71. }
  72. if dest.Network == net.Network_UDP && hasBindAddr(sockopt) {
  73. if err := bindAddr(fd, sockopt.BindAddress, sockopt.BindPort); err != nil {
  74. newError("failed to bind source address to ", sockopt.BindAddress).Base(err).WriteToLog(session.ExportIDToError(ctx))
  75. }
  76. }
  77. }
  78. for _, ctl := range d.controllers {
  79. if err := ctl(network, address, fd); err != nil {
  80. newError("failed to apply external controller").Base(err).WriteToLog(session.ExportIDToError(ctx))
  81. }
  82. }
  83. })
  84. }
  85. }
  86. return dialer.DialContext(ctx, dest.Network.SystemString(), dest.NetAddr())
  87. }
  88. type PacketConnWrapper struct {
  89. Conn net.PacketConn
  90. Dest net.Addr
  91. }
  92. func (c *PacketConnWrapper) Close() error {
  93. return c.Conn.Close()
  94. }
  95. func (c *PacketConnWrapper) LocalAddr() net.Addr {
  96. return c.Conn.LocalAddr()
  97. }
  98. func (c *PacketConnWrapper) RemoteAddr() net.Addr {
  99. return c.Dest
  100. }
  101. func (c *PacketConnWrapper) Write(p []byte) (int, error) {
  102. return c.Conn.WriteTo(p, c.Dest)
  103. }
  104. func (c *PacketConnWrapper) Read(p []byte) (int, error) {
  105. n, _, err := c.Conn.ReadFrom(p)
  106. return n, err
  107. }
  108. func (c *PacketConnWrapper) WriteTo(p []byte, d net.Addr) (int, error) {
  109. return c.Conn.WriteTo(p, d)
  110. }
  111. func (c *PacketConnWrapper) ReadFrom(p []byte) (int, net.Addr, error) {
  112. return c.Conn.ReadFrom(p)
  113. }
  114. func (c *PacketConnWrapper) SetDeadline(t time.Time) error {
  115. return c.Conn.SetDeadline(t)
  116. }
  117. func (c *PacketConnWrapper) SetReadDeadline(t time.Time) error {
  118. return c.Conn.SetReadDeadline(t)
  119. }
  120. func (c *PacketConnWrapper) SetWriteDeadline(t time.Time) error {
  121. return c.Conn.SetWriteDeadline(t)
  122. }
  123. type SystemDialerAdapter interface {
  124. Dial(network string, address string) (net.Conn, error)
  125. }
  126. type SimpleSystemDialer struct {
  127. adapter SystemDialerAdapter
  128. }
  129. func WithAdapter(dialer SystemDialerAdapter) SystemDialer {
  130. return &SimpleSystemDialer{
  131. adapter: dialer,
  132. }
  133. }
  134. func (v *SimpleSystemDialer) Dial(ctx context.Context, src net.Address, dest net.Destination, sockopt *SocketConfig) (net.Conn, error) {
  135. return v.adapter.Dial(dest.Network.SystemString(), dest.NetAddr())
  136. }
  137. // UseAlternativeSystemDialer replaces the current system dialer with a given one.
  138. // Caller must ensure there is no race condition.
  139. //
  140. // xray:api:stable
  141. func UseAlternativeSystemDialer(dialer SystemDialer) {
  142. if dialer == nil {
  143. dialer = &DefaultSystemDialer{}
  144. }
  145. effectiveSystemDialer = dialer
  146. }
  147. // RegisterDialerController adds a controller to the effective system dialer.
  148. // The controller can be used to operate on file descriptors before they are put into use.
  149. // It only works when effective dialer is the default dialer.
  150. //
  151. // xray:api:beta
  152. func RegisterDialerController(ctl func(network, address string, fd uintptr) error) error {
  153. if ctl == nil {
  154. return newError("nil listener controller")
  155. }
  156. dialer, ok := effectiveSystemDialer.(*DefaultSystemDialer)
  157. if !ok {
  158. return newError("RegisterListenerController not supported in custom dialer")
  159. }
  160. dialer.controllers = append(dialer.controllers, ctl)
  161. return nil
  162. }