dial.go 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. package grpc
  2. import (
  3. "context"
  4. gonet "net"
  5. "sync"
  6. "time"
  7. "google.golang.org/grpc"
  8. "google.golang.org/grpc/backoff"
  9. "google.golang.org/grpc/connectivity"
  10. "google.golang.org/grpc/credentials"
  11. "google.golang.org/grpc/keepalive"
  12. "github.com/xtls/xray-core/common"
  13. "github.com/xtls/xray-core/common/net"
  14. "github.com/xtls/xray-core/common/session"
  15. "github.com/xtls/xray-core/transport/internet"
  16. "github.com/xtls/xray-core/transport/internet/grpc/encoding"
  17. "github.com/xtls/xray-core/transport/internet/stat"
  18. "github.com/xtls/xray-core/transport/internet/tls"
  19. )
  20. func Dial(ctx context.Context, dest net.Destination, streamSettings *internet.MemoryStreamConfig) (stat.Connection, error) {
  21. newError("creating connection to ", dest).WriteToLog(session.ExportIDToError(ctx))
  22. conn, err := dialgRPC(ctx, dest, streamSettings)
  23. if err != nil {
  24. return nil, newError("failed to dial gRPC").Base(err)
  25. }
  26. return stat.Connection(conn), nil
  27. }
  28. func init() {
  29. common.Must(internet.RegisterTransportDialer(protocolName, Dial))
  30. }
  31. type dialerConf struct {
  32. net.Destination
  33. *internet.MemoryStreamConfig
  34. }
  35. type dialerCanceller func()
  36. var (
  37. globalDialerMap map[dialerConf]*grpc.ClientConn
  38. globalDialerAccess sync.Mutex
  39. )
  40. func dialgRPC(ctx context.Context, dest net.Destination, streamSettings *internet.MemoryStreamConfig) (net.Conn, error) {
  41. grpcSettings := streamSettings.ProtocolSettings.(*Config)
  42. conn, canceller, err := getGrpcClient(ctx, dest, streamSettings)
  43. if err != nil {
  44. return nil, newError("Cannot dial gRPC").Base(err)
  45. }
  46. client := encoding.NewGRPCServiceClient(conn)
  47. if grpcSettings.MultiMode {
  48. newError("using gRPC multi mode").AtDebug().WriteToLog()
  49. grpcService, err := client.(encoding.GRPCServiceClientX).TunMultiCustomName(ctx, grpcSettings.getNormalizedName())
  50. if err != nil {
  51. canceller()
  52. return nil, newError("Cannot dial gRPC").Base(err)
  53. }
  54. return encoding.NewMultiHunkConn(grpcService, nil), nil
  55. }
  56. grpcService, err := client.(encoding.GRPCServiceClientX).TunCustomName(ctx, grpcSettings.getNormalizedName())
  57. if err != nil {
  58. canceller()
  59. return nil, newError("Cannot dial gRPC").Base(err)
  60. }
  61. return encoding.NewHunkConn(grpcService, nil), nil
  62. }
  63. func getGrpcClient(ctx context.Context, dest net.Destination, streamSettings *internet.MemoryStreamConfig) (*grpc.ClientConn, dialerCanceller, error) {
  64. globalDialerAccess.Lock()
  65. defer globalDialerAccess.Unlock()
  66. if globalDialerMap == nil {
  67. globalDialerMap = make(map[dialerConf]*grpc.ClientConn)
  68. }
  69. tlsConfig := tls.ConfigFromStreamSettings(streamSettings)
  70. sockopt := streamSettings.SocketSettings
  71. grpcSettings := streamSettings.ProtocolSettings.(*Config)
  72. canceller := func() {
  73. globalDialerAccess.Lock()
  74. defer globalDialerAccess.Unlock()
  75. delete(globalDialerMap, dialerConf{dest, streamSettings})
  76. }
  77. if client, found := globalDialerMap[dialerConf{dest, streamSettings}]; found && client.GetState() != connectivity.Shutdown {
  78. return client, canceller, nil
  79. }
  80. var dialOptions = []grpc.DialOption{
  81. grpc.WithConnectParams(grpc.ConnectParams{
  82. Backoff: backoff.Config{
  83. BaseDelay: 500 * time.Millisecond,
  84. Multiplier: 1.5,
  85. Jitter: 0.2,
  86. MaxDelay: 19 * time.Second,
  87. },
  88. MinConnectTimeout: 5 * time.Second,
  89. }),
  90. grpc.WithContextDialer(func(gctx context.Context, s string) (gonet.Conn, error) {
  91. gctx = session.ContextWithID(gctx, session.IDFromContext(ctx))
  92. gctx = session.ContextWithOutbound(gctx, session.OutboundFromContext(ctx))
  93. rawHost, rawPort, err := net.SplitHostPort(s)
  94. select {
  95. case <-gctx.Done():
  96. return nil, gctx.Err()
  97. default:
  98. }
  99. if err != nil {
  100. return nil, err
  101. }
  102. if len(rawPort) == 0 {
  103. rawPort = "443"
  104. }
  105. port, err := net.PortFromString(rawPort)
  106. if err != nil {
  107. return nil, err
  108. }
  109. address := net.ParseAddress(rawHost)
  110. return internet.DialSystem(gctx, net.TCPDestination(address, port), sockopt)
  111. }),
  112. }
  113. if tlsConfig != nil {
  114. dialOptions = append(dialOptions, grpc.WithTransportCredentials(credentials.NewTLS(tlsConfig.GetTLSConfig())))
  115. } else {
  116. dialOptions = append(dialOptions, grpc.WithInsecure())
  117. }
  118. if grpcSettings.IdleTimeout > 0 || grpcSettings.HealthCheckTimeout > 0 || grpcSettings.PermitWithoutStream {
  119. dialOptions = append(dialOptions, grpc.WithKeepaliveParams(keepalive.ClientParameters{
  120. Time: time.Second * time.Duration(grpcSettings.IdleTimeout),
  121. Timeout: time.Second * time.Duration(grpcSettings.HealthCheckTimeout),
  122. PermitWithoutStream: grpcSettings.PermitWithoutStream,
  123. }))
  124. }
  125. var grpcDestHost string
  126. if dest.Address.Family().IsDomain() {
  127. grpcDestHost = dest.Address.Domain()
  128. } else {
  129. grpcDestHost = dest.Address.IP().String()
  130. }
  131. conn, err := grpc.Dial(
  132. gonet.JoinHostPort(grpcDestHost, dest.Port.String()),
  133. dialOptions...,
  134. )
  135. globalDialerMap[dialerConf{dest, streamSettings}] = conn
  136. return conn, canceller, err
  137. }