client.go 3.3 KB

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