dial.go 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108
  1. // +build !confonly
  2. package grpc
  3. import (
  4. "context"
  5. gonet "net"
  6. "sync"
  7. "time"
  8. "google.golang.org/grpc"
  9. "google.golang.org/grpc/backoff"
  10. "google.golang.org/grpc/connectivity"
  11. "google.golang.org/grpc/credentials"
  12. "github.com/v2fly/v2ray-core/v4/common"
  13. "github.com/v2fly/v2ray-core/v4/common/net"
  14. "github.com/v2fly/v2ray-core/v4/common/session"
  15. "github.com/v2fly/v2ray-core/v4/transport/internet"
  16. "github.com/v2fly/v2ray-core/v4/transport/internet/grpc/encoding"
  17. "github.com/v2fly/v2ray-core/v4/transport/internet/tls"
  18. )
  19. func Dial(ctx context.Context, dest net.Destination, streamSettings *internet.MemoryStreamConfig) (internet.Connection, error) {
  20. newError("creating connection to ", dest).WriteToLog(session.ExportIDToError(ctx))
  21. conn, err := dialgRPC(ctx, dest, streamSettings)
  22. if err != nil {
  23. return nil, newError("failed to dial Grpc").Base(err)
  24. }
  25. return internet.Connection(conn), nil
  26. }
  27. func init() {
  28. common.Must(internet.RegisterTransportDialer(protocolName, Dial))
  29. }
  30. var (
  31. globalDialerMap map[net.Destination]*grpc.ClientConn
  32. globalDialerAccess sync.Mutex
  33. )
  34. func dialgRPC(ctx context.Context, dest net.Destination, streamSettings *internet.MemoryStreamConfig) (net.Conn, error) {
  35. grpcSettings := streamSettings.ProtocolSettings.(*Config)
  36. config := tls.ConfigFromStreamSettings(streamSettings)
  37. dialOption := grpc.WithInsecure()
  38. if config != nil {
  39. dialOption = grpc.WithTransportCredentials(credentials.NewTLS(config.GetTLSConfig()))
  40. }
  41. conn, err := getGrpcClient(ctx, dest, dialOption)
  42. if err != nil {
  43. return nil, newError("Cannot dial grpc").Base(err)
  44. }
  45. client := encoding.NewGunServiceClient(conn)
  46. gunService, err := client.(encoding.GunServiceClientX).TunCustomName(ctx, grpcSettings.ServiceName)
  47. if err != nil {
  48. return nil, newError("Cannot dial grpc").Base(err)
  49. }
  50. return encoding.NewGunConn(gunService, nil), nil
  51. }
  52. func getGrpcClient(ctx context.Context, dest net.Destination, dialOption grpc.DialOption) (*grpc.ClientConn, error) {
  53. globalDialerAccess.Lock()
  54. defer globalDialerAccess.Unlock()
  55. if globalDialerMap == nil {
  56. globalDialerMap = make(map[net.Destination]*grpc.ClientConn)
  57. }
  58. // TODO Should support chain proxy to the same destination
  59. if client, found := globalDialerMap[dest]; found && client.GetState() != connectivity.Shutdown {
  60. return client, nil
  61. }
  62. conn, err := grpc.Dial(
  63. dest.Address.String()+":"+dest.Port.String(),
  64. dialOption,
  65. grpc.WithConnectParams(grpc.ConnectParams{
  66. Backoff: backoff.Config{
  67. BaseDelay: 500 * time.Millisecond,
  68. Multiplier: 1.5,
  69. Jitter: 0.2,
  70. MaxDelay: 19 * time.Second,
  71. },
  72. MinConnectTimeout: 5 * time.Second,
  73. }),
  74. grpc.WithContextDialer(func(ctxGrpc context.Context, s string) (gonet.Conn, error) {
  75. rawHost, rawPort, err := net.SplitHostPort(s)
  76. if err != nil {
  77. return nil, err
  78. }
  79. if len(rawPort) == 0 {
  80. rawPort = "443"
  81. }
  82. port, err := net.PortFromString(rawPort)
  83. if err != nil {
  84. return nil, err
  85. }
  86. address := net.ParseAddress(rawHost)
  87. return internet.DialSystem(ctx, net.TCPDestination(address, port), nil)
  88. }),
  89. )
  90. globalDialerMap[dest] = conn
  91. return conn, err
  92. }