go-zero/zrpc/internal/client.go

127 lines
3.2 KiB
Go
Raw Normal View History

2020-07-29 18:06:57 +08:00
package internal
2020-07-26 17:09:05 +08:00
import (
"context"
"errors"
2020-07-26 17:09:05 +08:00
"fmt"
"strings"
2020-07-26 17:09:05 +08:00
"time"
2020-09-18 11:41:52 +08:00
"github.com/tal-tech/go-zero/zrpc/internal/balancer/p2c"
"github.com/tal-tech/go-zero/zrpc/internal/clientinterceptors"
"github.com/tal-tech/go-zero/zrpc/internal/resolver"
2020-07-26 17:09:05 +08:00
"google.golang.org/grpc"
)
const (
dialTimeout = time.Second * 3
separator = '/'
)
2020-07-26 17:09:05 +08:00
2020-08-18 18:36:44 +08:00
func init() {
resolver.RegisterResolver()
}
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 {
Timeout time.Duration
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 {
conn *grpc.ClientConn
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.
func NewClient(target string, opts ...ClientOption) (Client, error) {
2020-09-29 16:09:11 +08:00
var cli client
2020-10-20 17:01:53 +08:00
opts = append([]ClientOption{WithDialOption(grpc.WithBalancerName(p2c.Name))}, 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 {
var cliOpts ClientOptions
2020-07-26 17:09:05 +08:00
for _, opt := range opts {
opt(&cliOpts)
2020-07-26 17:09:05 +08:00
}
options := []grpc.DialOption{
grpc.WithInsecure(),
grpc.WithBlock(),
WithUnaryClientInterceptors(
clientinterceptors.TracingInterceptor,
2020-07-26 17:09:05 +08:00
clientinterceptors.DurationInterceptor,
2020-09-27 17:15:15 +08:00
clientinterceptors.PrometheusInterceptor,
2021-08-04 18:45:05 +08:00
clientinterceptors.BreakerInterceptor,
clientinterceptors.TimeoutInterceptor(cliOpts.Timeout),
clientinterceptors.OpenTracingInterceptor(),
),
WithStreamClientInterceptors(
clientinterceptors.StreamOpenTracingInterceptor(),
2020-07-26 17:09:05 +08:00
),
}
return append(options, cliOpts.DialOptions...)
2020-07-26 17:09:05 +08:00
}
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 {
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",
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-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
}
2021-03-01 23:52:44 +08:00
// WithUnaryClientInterceptor returns a func to customize a ClientOptions with given interceptor.
func WithUnaryClientInterceptor(interceptor grpc.UnaryClientInterceptor) ClientOption {
return func(options *ClientOptions) {
options.DialOptions = append(options.DialOptions, WithUnaryClientInterceptors(interceptor))
}
}