client.go 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. package v2raygrpc
  2. import (
  3. "context"
  4. "net"
  5. "sync"
  6. "time"
  7. "github.com/sagernet/sing-box/adapter"
  8. "github.com/sagernet/sing-box/common/tls"
  9. "github.com/sagernet/sing-box/option"
  10. "github.com/sagernet/sing/common"
  11. M "github.com/sagernet/sing/common/metadata"
  12. N "github.com/sagernet/sing/common/network"
  13. "golang.org/x/net/http2"
  14. "google.golang.org/grpc"
  15. "google.golang.org/grpc/backoff"
  16. "google.golang.org/grpc/connectivity"
  17. "google.golang.org/grpc/credentials/insecure"
  18. "google.golang.org/grpc/keepalive"
  19. )
  20. var _ adapter.V2RayClientTransport = (*Client)(nil)
  21. type Client struct {
  22. ctx context.Context
  23. dialer N.Dialer
  24. serverAddr string
  25. serviceName string
  26. dialOptions []grpc.DialOption
  27. conn *grpc.ClientConn
  28. connAccess sync.Mutex
  29. }
  30. func NewClient(ctx context.Context, dialer N.Dialer, serverAddr M.Socksaddr, options option.V2RayGRPCOptions, tlsConfig tls.Config) (adapter.V2RayClientTransport, error) {
  31. var dialOptions []grpc.DialOption
  32. if tlsConfig != nil {
  33. tlsConfig.SetNextProtos([]string{http2.NextProtoTLS})
  34. dialOptions = append(dialOptions, grpc.WithTransportCredentials(NewTLSTransportCredentials(tlsConfig)))
  35. } else {
  36. dialOptions = append(dialOptions, grpc.WithTransportCredentials(insecure.NewCredentials()))
  37. }
  38. if options.IdleTimeout > 0 {
  39. dialOptions = append(dialOptions, grpc.WithKeepaliveParams(keepalive.ClientParameters{
  40. Time: time.Duration(options.IdleTimeout),
  41. Timeout: time.Duration(options.PingTimeout),
  42. PermitWithoutStream: options.PermitWithoutStream,
  43. }))
  44. }
  45. dialOptions = append(dialOptions, grpc.WithConnectParams(grpc.ConnectParams{
  46. Backoff: backoff.Config{
  47. BaseDelay: 500 * time.Millisecond,
  48. Multiplier: 1.5,
  49. Jitter: 0.2,
  50. MaxDelay: 19 * time.Second,
  51. },
  52. MinConnectTimeout: 5 * time.Second,
  53. }))
  54. dialOptions = append(dialOptions, grpc.WithContextDialer(func(ctx context.Context, server string) (net.Conn, error) {
  55. return dialer.DialContext(ctx, N.NetworkTCP, M.ParseSocksaddr(server))
  56. }))
  57. dialOptions = append(dialOptions, grpc.WithReturnConnectionError())
  58. return &Client{
  59. ctx: ctx,
  60. dialer: dialer,
  61. serverAddr: serverAddr.String(),
  62. serviceName: options.ServiceName,
  63. dialOptions: dialOptions,
  64. }, nil
  65. }
  66. func (c *Client) Close() error {
  67. return common.Close(
  68. common.PtrOrNil(c.conn),
  69. )
  70. }
  71. func (c *Client) connect() (*grpc.ClientConn, error) {
  72. conn := c.conn
  73. if conn != nil && conn.GetState() != connectivity.Shutdown {
  74. return conn, nil
  75. }
  76. c.connAccess.Lock()
  77. defer c.connAccess.Unlock()
  78. conn = c.conn
  79. if conn != nil && conn.GetState() != connectivity.Shutdown {
  80. return conn, nil
  81. }
  82. conn, err := grpc.DialContext(c.ctx, c.serverAddr, c.dialOptions...)
  83. if err != nil {
  84. return nil, err
  85. }
  86. c.conn = conn
  87. return conn, nil
  88. }
  89. func (c *Client) DialContext(ctx context.Context) (net.Conn, error) {
  90. clientConn, err := c.connect()
  91. if err != nil {
  92. return nil, err
  93. }
  94. client := NewGunServiceClient(clientConn).(GunServiceCustomNameClient)
  95. ctx, cancel := common.ContextWithCancelCause(ctx)
  96. stream, err := client.TunCustomName(ctx, c.serviceName)
  97. if err != nil {
  98. cancel(err)
  99. return nil, err
  100. }
  101. return NewGRPCConn(stream, cancel), nil
  102. }