dial.go 3.9 KB

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