2020-07-29 18:06:57 +08:00
|
|
|
package internal
|
2020-07-26 17:09:05 +08:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2020-09-23 22:07:26 +08:00
|
|
|
"errors"
|
2020-07-26 17:09:05 +08:00
|
|
|
"fmt"
|
2020-09-23 22:07:26 +08:00
|
|
|
"strings"
|
2020-07-26 17:09:05 +08:00
|
|
|
"time"
|
|
|
|
|
2022-01-04 15:51:32 +08:00
|
|
|
"github.com/zeromicro/go-zero/zrpc/internal/balancer/p2c"
|
|
|
|
"github.com/zeromicro/go-zero/zrpc/internal/clientinterceptors"
|
|
|
|
"github.com/zeromicro/go-zero/zrpc/resolver"
|
2020-07-26 17:09:05 +08:00
|
|
|
"google.golang.org/grpc"
|
2021-10-30 22:07:15 +08:00
|
|
|
"google.golang.org/grpc/credentials"
|
2022-04-18 14:15:09 +08:00
|
|
|
"google.golang.org/grpc/credentials/insecure"
|
2020-07-26 17:09:05 +08:00
|
|
|
)
|
|
|
|
|
2020-09-23 22:07:26 +08:00
|
|
|
const (
|
|
|
|
dialTimeout = time.Second * 3
|
|
|
|
separator = '/'
|
|
|
|
)
|
2020-07-26 17:09:05 +08:00
|
|
|
|
2020-08-18 18:36:44 +08:00
|
|
|
func init() {
|
2021-12-01 20:22:15 +08:00
|
|
|
resolver.Register()
|
2020-08-18 18:36:44 +08:00
|
|
|
}
|
|
|
|
|
2020-07-26 17:09:05 +08:00
|
|
|
type (
|
2021-03-01 23:52:44 +08:00
|
|
|
// Client interface wraps the Conn method.
|
|
|
|
Client interface {
|
|
|
|
Conn() *grpc.ClientConn
|
|
|
|
}
|
|
|
|
|
|
|
|
// A ClientOptions is a client options.
|
2020-07-26 17:09:05 +08:00
|
|
|
ClientOptions struct {
|
2021-11-02 19:02:02 +08:00
|
|
|
NonBlock bool
|
2020-07-26 17:09:05 +08:00
|
|
|
Timeout time.Duration
|
2021-10-30 23:15:39 +08:00
|
|
|
Secure bool
|
2020-07-26 17:09:05 +08:00
|
|
|
DialOptions []grpc.DialOption
|
|
|
|
}
|
|
|
|
|
2021-03-01 23:52:44 +08:00
|
|
|
// ClientOption defines the method to customize a ClientOptions.
|
2020-07-26 17:09:05 +08:00
|
|
|
ClientOption func(options *ClientOptions)
|
2020-08-18 18:36:44 +08:00
|
|
|
|
|
|
|
client struct {
|
2023-01-08 19:34:05 +08:00
|
|
|
conn *grpc.ClientConn
|
|
|
|
middlewares ClientMiddlewaresConf
|
2020-08-18 18:36:44 +08:00
|
|
|
}
|
2020-07-26 17:09:05 +08:00
|
|
|
)
|
|
|
|
|
2021-03-01 23:52:44 +08:00
|
|
|
// NewClient returns a Client.
|
2023-05-06 12:39:19 +08:00
|
|
|
func NewClient(target string, middlewares ClientMiddlewaresConf, opts ...ClientOption) (Client, error) {
|
2023-01-08 19:34:05 +08:00
|
|
|
cli := client{
|
|
|
|
middlewares: middlewares,
|
|
|
|
}
|
2022-04-24 22:42:40 +08:00
|
|
|
|
|
|
|
svcCfg := fmt.Sprintf(`{"loadBalancingPolicy":"%s"}`, p2c.Name)
|
|
|
|
balancerOpt := WithDialOption(grpc.WithDefaultServiceConfig(svcCfg))
|
|
|
|
opts = append([]ClientOption{balancerOpt}, opts...)
|
2020-09-29 16:09:11 +08:00
|
|
|
if err := cli.dial(target, opts...); err != nil {
|
2020-08-18 18:36:44 +08:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2020-09-29 16:09:11 +08:00
|
|
|
return &cli, nil
|
2020-08-18 18:36:44 +08:00
|
|
|
}
|
|
|
|
|
2020-09-29 16:09:11 +08:00
|
|
|
func (c *client) Conn() *grpc.ClientConn {
|
|
|
|
return c.conn
|
2020-07-26 17:09:05 +08:00
|
|
|
}
|
|
|
|
|
2020-09-29 16:09:11 +08:00
|
|
|
func (c *client) buildDialOptions(opts ...ClientOption) []grpc.DialOption {
|
2020-10-20 17:57:41 +08:00
|
|
|
var cliOpts ClientOptions
|
2020-07-26 17:09:05 +08:00
|
|
|
for _, opt := range opts {
|
2020-10-20 17:57:41 +08:00
|
|
|
opt(&cliOpts)
|
2020-07-26 17:09:05 +08:00
|
|
|
}
|
|
|
|
|
2021-10-30 23:15:39 +08:00
|
|
|
var options []grpc.DialOption
|
|
|
|
if !cliOpts.Secure {
|
2023-01-09 00:55:13 +08:00
|
|
|
options = append([]grpc.DialOption(nil),
|
|
|
|
grpc.WithTransportCredentials(insecure.NewCredentials()))
|
2021-10-30 23:15:39 +08:00
|
|
|
}
|
|
|
|
|
2021-11-02 19:02:02 +08:00
|
|
|
if !cliOpts.NonBlock {
|
|
|
|
options = append(options, grpc.WithBlock())
|
|
|
|
}
|
|
|
|
|
2021-10-30 23:15:39 +08:00
|
|
|
options = append(options,
|
2023-01-09 00:55:13 +08:00
|
|
|
grpc.WithChainUnaryInterceptor(c.buildUnaryInterceptors(cliOpts.Timeout)...),
|
|
|
|
grpc.WithChainStreamInterceptor(c.buildStreamInterceptors()...),
|
2021-10-30 23:15:39 +08:00
|
|
|
)
|
2020-07-26 17:09:05 +08:00
|
|
|
|
2020-10-20 17:57:41 +08:00
|
|
|
return append(options, cliOpts.DialOptions...)
|
2020-07-26 17:09:05 +08:00
|
|
|
}
|
|
|
|
|
2023-01-08 19:34:05 +08:00
|
|
|
func (c *client) buildStreamInterceptors() []grpc.StreamClientInterceptor {
|
|
|
|
var interceptors []grpc.StreamClientInterceptor
|
|
|
|
|
|
|
|
if c.middlewares.Trace {
|
|
|
|
interceptors = append(interceptors, clientinterceptors.StreamTracingInterceptor)
|
|
|
|
}
|
|
|
|
|
|
|
|
return interceptors
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *client) buildUnaryInterceptors(timeout time.Duration) []grpc.UnaryClientInterceptor {
|
|
|
|
var interceptors []grpc.UnaryClientInterceptor
|
|
|
|
|
|
|
|
if c.middlewares.Trace {
|
|
|
|
interceptors = append(interceptors, clientinterceptors.UnaryTracingInterceptor)
|
|
|
|
}
|
|
|
|
if c.middlewares.Duration {
|
|
|
|
interceptors = append(interceptors, clientinterceptors.DurationInterceptor)
|
|
|
|
}
|
|
|
|
if c.middlewares.Prometheus {
|
|
|
|
interceptors = append(interceptors, clientinterceptors.PrometheusInterceptor)
|
|
|
|
}
|
|
|
|
if c.middlewares.Breaker {
|
|
|
|
interceptors = append(interceptors, clientinterceptors.BreakerInterceptor)
|
|
|
|
}
|
|
|
|
if c.middlewares.Timeout {
|
|
|
|
interceptors = append(interceptors, clientinterceptors.TimeoutInterceptor(timeout))
|
|
|
|
}
|
|
|
|
|
|
|
|
return interceptors
|
|
|
|
}
|
|
|
|
|
2020-09-29 16:09:11 +08:00
|
|
|
func (c *client) dial(server string, opts ...ClientOption) error {
|
|
|
|
options := c.buildDialOptions(opts...)
|
2020-07-26 17:09:05 +08:00
|
|
|
timeCtx, cancel := context.WithTimeout(context.Background(), dialTimeout)
|
|
|
|
defer cancel()
|
|
|
|
conn, err := grpc.DialContext(timeCtx, server, options...)
|
|
|
|
if err != nil {
|
2020-09-23 22:07:26 +08:00
|
|
|
service := server
|
|
|
|
if errors.Is(err, context.DeadlineExceeded) {
|
|
|
|
pos := strings.LastIndexByte(server, separator)
|
|
|
|
// len(server) - 1 is the index of last char
|
|
|
|
if 0 < pos && pos < len(server)-1 {
|
|
|
|
service = server[pos+1:]
|
|
|
|
}
|
|
|
|
}
|
2021-05-28 23:39:07 +08:00
|
|
|
return fmt.Errorf("rpc dial: %s, error: %s, make sure rpc service %q is already started",
|
2020-09-23 22:07:26 +08:00
|
|
|
server, err.Error(), service)
|
2020-07-26 17:09:05 +08:00
|
|
|
}
|
|
|
|
|
2020-09-29 16:09:11 +08:00
|
|
|
c.conn = conn
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-03-01 23:52:44 +08:00
|
|
|
// WithDialOption returns a func to customize a ClientOptions with given dial option.
|
2020-09-29 16:09:11 +08:00
|
|
|
func WithDialOption(opt grpc.DialOption) ClientOption {
|
|
|
|
return func(options *ClientOptions) {
|
|
|
|
options.DialOptions = append(options.DialOptions, opt)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-02 19:02:02 +08:00
|
|
|
// WithNonBlock sets the dialing to be nonblock.
|
|
|
|
func WithNonBlock() ClientOption {
|
|
|
|
return func(options *ClientOptions) {
|
|
|
|
options.NonBlock = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-14 19:58:17 +08:00
|
|
|
// WithStreamClientInterceptor returns a func to customize a ClientOptions with given interceptor.
|
|
|
|
func WithStreamClientInterceptor(interceptor grpc.StreamClientInterceptor) ClientOption {
|
|
|
|
return func(options *ClientOptions) {
|
2023-01-09 00:55:13 +08:00
|
|
|
options.DialOptions = append(options.DialOptions,
|
|
|
|
grpc.WithChainStreamInterceptor(interceptor))
|
2022-05-14 19:58:17 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-01 23:52:44 +08:00
|
|
|
// WithTimeout returns a func to customize a ClientOptions with given timeout.
|
2020-09-29 16:09:11 +08:00
|
|
|
func WithTimeout(timeout time.Duration) ClientOption {
|
|
|
|
return func(options *ClientOptions) {
|
|
|
|
options.Timeout = timeout
|
|
|
|
}
|
2020-07-26 17:09:05 +08:00
|
|
|
}
|
2020-10-20 17:57:41 +08:00
|
|
|
|
2021-11-02 20:42:22 +08:00
|
|
|
// WithTransportCredentials return a func to make the gRPC calls secured with given credentials.
|
|
|
|
func WithTransportCredentials(creds credentials.TransportCredentials) ClientOption {
|
2020-10-20 17:57:41 +08:00
|
|
|
return func(options *ClientOptions) {
|
2021-10-30 23:15:39 +08:00
|
|
|
options.Secure = true
|
2021-11-02 20:42:22 +08:00
|
|
|
options.DialOptions = append(options.DialOptions, grpc.WithTransportCredentials(creds))
|
2021-10-30 22:07:15 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-02 20:42:22 +08:00
|
|
|
// WithUnaryClientInterceptor returns a func to customize a ClientOptions with given interceptor.
|
|
|
|
func WithUnaryClientInterceptor(interceptor grpc.UnaryClientInterceptor) ClientOption {
|
2021-10-30 22:07:15 +08:00
|
|
|
return func(options *ClientOptions) {
|
2023-01-09 00:55:13 +08:00
|
|
|
options.DialOptions = append(options.DialOptions,
|
|
|
|
grpc.WithChainUnaryInterceptor(interceptor))
|
2021-10-30 22:07:15 +08:00
|
|
|
}
|
|
|
|
}
|