dial.go 5.4 KB

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