// Package grpc provides a gRPC client for micro framework package grpc import ( "context" "crypto/tls" "fmt" "net" "os" "reflect" "strconv" "strings" "sync" "time" "go.unistack.org/micro/v3/broker" "go.unistack.org/micro/v3/client" "go.unistack.org/micro/v3/codec" "go.unistack.org/micro/v3/errors" "go.unistack.org/micro/v3/metadata" "go.unistack.org/micro/v3/options" "go.unistack.org/micro/v3/selector" "go.unistack.org/micro/v3/semconv" "go.unistack.org/micro/v3/tracer" "google.golang.org/grpc" "google.golang.org/grpc/codes" "google.golang.org/grpc/credentials" "google.golang.org/grpc/credentials/insecure" "google.golang.org/grpc/encoding" gmetadata "google.golang.org/grpc/metadata" "google.golang.org/grpc/status" ) const ( DefaultContentType = "application/grpc" ) type grpcClient struct { funcPublish client.FuncPublish funcBatchPublish client.FuncBatchPublish funcCall client.FuncCall funcStream client.FuncStream pool *ConnPool opts client.Options sync.RWMutex init bool } // secure returns the dial option for whether its a secure or insecure connection func (g *grpcClient) secure(addr string) grpc.DialOption { // first we check if theres'a tls config if g.opts.TLSConfig != nil { creds := credentials.NewTLS(g.opts.TLSConfig) // return tls config if it exists return grpc.WithTransportCredentials(creds) } // default config tlsConfig := &tls.Config{MinVersion: tls.VersionTLS12} defaultCreds := grpc.WithTransportCredentials(credentials.NewTLS(tlsConfig)) // check if the address is prepended with https if strings.HasPrefix(addr, "https://") { return defaultCreds } // if no port is specified or port is 443 default to tls _, port, err := net.SplitHostPort(addr) // assuming with no port its going to be secured if port == "443" { return defaultCreds } else if err != nil && strings.Contains(err.Error(), "missing port in address") { return defaultCreds } // other fallback to insecure return grpc.WithTransportCredentials(insecure.NewCredentials()) } func (g *grpcClient) call(ctx context.Context, addr string, req client.Request, rsp interface{}, opts client.CallOptions) error { var header map[string]string if md, ok := metadata.FromOutgoingContext(ctx); ok { header = make(map[string]string, len(md)) for k, v := range md { header[strings.ToLower(k)] = v } } else { header = make(map[string]string, 2) } if opts.RequestMetadata != nil { for k, v := range opts.RequestMetadata { header[k] = v } } // set timeout in nanoseconds header["Grpc-Timeout"] = fmt.Sprintf("%dn", opts.RequestTimeout) header["timeout"] = fmt.Sprintf("%dn", opts.RequestTimeout) header["content-type"] = req.ContentType() md := gmetadata.New(header) ctx = gmetadata.NewOutgoingContext(ctx, md) cf, err := g.newCodec(req.ContentType()) if err != nil { return errors.InternalServerError("go.micro.client", "%+v", err) } maxRecvMsgSize := g.maxRecvMsgSizeValue() maxSendMsgSize := g.maxSendMsgSizeValue() cfgService := g.serviceConfig() var grr error var dialCtx context.Context var cancel context.CancelFunc if opts.DialTimeout >= 0 { dialCtx, cancel = context.WithTimeout(ctx, opts.DialTimeout) } else { dialCtx, cancel = context.WithCancel(ctx) } defer cancel() grpcDialOptions := []grpc.DialOption{ g.secure(addr), grpc.WithDefaultCallOptions( grpc.MaxCallRecvMsgSize(maxRecvMsgSize), grpc.MaxCallSendMsgSize(maxSendMsgSize), ), grpc.WithDefaultServiceConfig(cfgService), } if opts := g.getGrpcDialOptions(g.opts.Context); opts != nil { grpcDialOptions = append(grpcDialOptions, opts...) } if opts := g.getGrpcDialOptions(opts.Context); opts != nil { grpcDialOptions = append(grpcDialOptions, opts...) } contextDialer := g.opts.ContextDialer if opts.ContextDialer != nil { contextDialer = opts.ContextDialer } if contextDialer != nil { grpcDialOptions = append(grpcDialOptions, grpc.WithContextDialer(contextDialer)) } cc, err := g.pool.Get(dialCtx, addr, grpcDialOptions...) if err != nil { return errors.InternalServerError("go.micro.client", "Error sending request: %v", err) } defer func() { // defer execution of release g.pool.Put(cc, grr) }() ch := make(chan error, 1) var gmd gmetadata.MD grpcCallOptions := []grpc.CallOption{ grpc.CallContentSubtype((&wrapMicroCodec{cf}).Name()), } if opts := g.getGrpcCallOptions(opts.Context); opts != nil { grpcCallOptions = append(grpcCallOptions, opts...) } if opts.ResponseMetadata != nil { gmd = gmetadata.MD{} grpcCallOptions = append(grpcCallOptions, grpc.Header(&gmd)) } go func() { err := cc.Invoke(ctx, methodToGRPC(req.Service(), req.Endpoint()), req.Body(), rsp, grpcCallOptions...) ch <- microError(err) }() select { case err := <-ch: grr = err case <-ctx.Done(): grr = errors.Timeout("go.micro.client", "%v", ctx.Err()) } if opts.ResponseMetadata != nil { *opts.ResponseMetadata = metadata.New(gmd.Len()) for k, v := range gmd { opts.ResponseMetadata.Set(k, strings.Join(v, ",")) } } return grr } func (g *grpcClient) stream(ctx context.Context, addr string, req client.Request, rsp interface{}, opts client.CallOptions) error { var header map[string]string if md, ok := metadata.FromOutgoingContext(ctx); ok { header = make(map[string]string, len(md)) for k, v := range md { header[k] = v } } else { header = make(map[string]string) } // set timeout in nanoseconds if opts.StreamTimeout > time.Duration(0) { header["Grpc-Timeout"] = fmt.Sprintf("%dn", opts.StreamTimeout) header["timeout"] = fmt.Sprintf("%dn", opts.StreamTimeout) } // set the content type for the request header["content-type"] = req.ContentType() md := gmetadata.New(header) ctx = gmetadata.NewOutgoingContext(ctx, md) cf, err := g.newCodec(req.ContentType()) if err != nil { return errors.InternalServerError("go.micro.client", "%+v", err) } var dialCtx context.Context var cancel context.CancelFunc if opts.DialTimeout >= 0 { dialCtx, cancel = context.WithTimeout(ctx, opts.DialTimeout) } else { dialCtx, cancel = context.WithCancel(ctx) } defer cancel() wc := &wrapMicroCodec{cf} maxRecvMsgSize := g.maxRecvMsgSizeValue() maxSendMsgSize := g.maxSendMsgSizeValue() cfgService := g.serviceConfig() grpcDialOptions := []grpc.DialOption{ g.secure(addr), grpc.WithDefaultCallOptions( grpc.MaxCallRecvMsgSize(maxRecvMsgSize), grpc.MaxCallSendMsgSize(maxSendMsgSize), ), grpc.WithDefaultServiceConfig(cfgService), } if opts := g.getGrpcDialOptions(opts.Context); opts != nil { grpcDialOptions = append(grpcDialOptions, opts...) } contextDialer := g.opts.ContextDialer if opts.ContextDialer != nil { contextDialer = opts.ContextDialer } if contextDialer != nil { grpcDialOptions = append(grpcDialOptions, grpc.WithContextDialer(contextDialer)) } cc, err := g.pool.Get(dialCtx, addr, grpcDialOptions...) if err != nil { return errors.InternalServerError("go.micro.client", "Error sending request: %v", err) } desc := &grpc.StreamDesc{ StreamName: req.Service() + req.Endpoint(), ClientStreams: true, ServerStreams: true, } grpcCallOptions := []grpc.CallOption{ // grpc.ForceCodec(wc), grpc.CallContentSubtype(wc.Name()), } if opts := g.getGrpcCallOptions(opts.Context); opts != nil { grpcCallOptions = append(grpcCallOptions, opts...) } var gmd gmetadata.MD if opts.ResponseMetadata != nil { gmd = gmetadata.MD{} grpcCallOptions = append(grpcCallOptions, grpc.Header(&gmd)) } // create a new cancelling context newCtx, cancel := context.WithCancel(ctx) st, err := cc.NewStream(newCtx, desc, methodToGRPC(req.Service(), req.Endpoint()), grpcCallOptions...) if err != nil { // we need to cleanup as we dialled and created a context // cancel the context cancel() // release the connection g.pool.Put(cc, err) // now return the error return errors.InternalServerError("go.micro.client", "Error creating stream: %v", err) } // set request codec if r, ok := req.(*grpcRequest); ok { r.codec = cf } // setup the stream response stream := &grpcStream{ ClientStream: st, context: ctx, request: req, response: &response{ conn: cc, stream: st, codec: cf, }, conn: cc, close: func(err error) { // cancel the context if an error occurred if err != nil { cancel() } // defer execution of release g.pool.Put(cc, err) }, } // set the stream as the response val := reflect.ValueOf(rsp).Elem() val.Set(reflect.ValueOf(stream).Elem()) return nil } func (g *grpcClient) poolMaxStreams() int { if g.opts.Context == nil { return DefaultPoolMaxStreams } v := g.opts.Context.Value(poolMaxStreams{}) if v == nil { return DefaultPoolMaxStreams } return v.(int) } func (g *grpcClient) poolMaxIdle() int { if g.opts.Context == nil { return DefaultPoolMaxIdle } v := g.opts.Context.Value(poolMaxIdle{}) if v == nil { return DefaultPoolMaxIdle } return v.(int) } func (g *grpcClient) maxRecvMsgSizeValue() int { if g.opts.Context == nil { return DefaultMaxRecvMsgSize } v := g.opts.Context.Value(maxRecvMsgSizeKey{}) if v == nil { return DefaultMaxRecvMsgSize } return v.(int) } func (g *grpcClient) maxSendMsgSizeValue() int { if g.opts.Context == nil { return DefaultMaxSendMsgSize } v := g.opts.Context.Value(maxSendMsgSizeKey{}) if v == nil { return DefaultMaxSendMsgSize } return v.(int) } func (g *grpcClient) newCodec(ct string) (codec.Codec, error) { g.RLock() defer g.RUnlock() if idx := strings.IndexRune(ct, ';'); idx >= 0 { ct = ct[:idx] } if c, ok := g.opts.Codecs[ct]; ok { return c, nil } return nil, codec.ErrUnknownContentType } func (g *grpcClient) serviceConfig() string { if g.opts.Context == nil { return DefaultServiceConfig } v := g.opts.Context.Value(serviceConfigKey{}) if v == nil { return DefaultServiceConfig } return v.(string) } func (g *grpcClient) Init(opts ...client.Option) error { if len(opts) == 0 && g.init { return nil } size := g.opts.PoolSize ttl := g.opts.PoolTTL for _, o := range opts { o(&g.opts) } // update pool configuration if the options changed if size != g.opts.PoolSize || ttl != g.opts.PoolTTL { g.pool.Lock() g.pool.size = g.opts.PoolSize g.pool.ttl = int64(g.opts.PoolTTL.Seconds()) g.pool.Unlock() } g.funcCall = g.fnCall g.funcStream = g.fnStream g.funcPublish = g.fnPublish g.funcBatchPublish = g.fnBatchPublish g.opts.Hooks.EachPrev(func(hook options.Hook) { switch h := hook.(type) { case client.HookCall: g.funcCall = h(g.funcCall) case client.HookStream: g.funcStream = h(g.funcStream) case client.HookPublish: g.funcPublish = h(g.funcPublish) case client.HookBatchPublish: g.funcBatchPublish = h(g.funcBatchPublish) } }) return nil } func (g *grpcClient) Options() client.Options { return g.opts } func (g *grpcClient) NewMessage(topic string, msg interface{}, opts ...client.MessageOption) client.Message { return newGRPCEvent(topic, msg, g.opts.ContentType, opts...) } func (g *grpcClient) NewRequest(service, method string, req interface{}, reqOpts ...client.RequestOption) client.Request { return newGRPCRequest(service, method, req, g.opts.ContentType, reqOpts...) } func (g *grpcClient) Call(ctx context.Context, req client.Request, rsp interface{}, opts ...client.CallOption) error { if req == nil { return errors.InternalServerError("go.micro.client", "req is nil") } else if rsp == nil { return errors.InternalServerError("go.micro.client", "rsp is nil") } ts := time.Now() g.opts.Meter.Counter(semconv.ClientRequestInflight, "endpoint", req.Endpoint()).Inc() var sp tracer.Span ctx, sp = g.opts.Tracer.Start(ctx, req.Endpoint()+" rpc-client", tracer.WithSpanKind(tracer.SpanKindClient), tracer.WithSpanLabels("endpoint", req.Endpoint()), ) err := g.funcCall(ctx, req, rsp, opts...) g.opts.Meter.Counter(semconv.ClientRequestInflight, "endpoint", req.Endpoint()).Dec() te := time.Since(ts) g.opts.Meter.Summary(semconv.ClientRequestLatencyMicroseconds, "endpoint", req.Endpoint()).Update(te.Seconds()) g.opts.Meter.Histogram(semconv.ClientRequestDurationSeconds, "endpoint", req.Endpoint()).Update(te.Seconds()) if me := errors.FromError(err); me == nil { sp.Finish() g.opts.Meter.Counter(semconv.ClientRequestTotal, "endpoint", req.Endpoint(), "status", "success", "code", strconv.Itoa(int(200))).Inc() } else { sp.SetStatus(tracer.SpanStatusError, err.Error()) g.opts.Meter.Counter(semconv.ClientRequestTotal, "endpoint", req.Endpoint(), "status", "failure", "code", strconv.Itoa(int(me.Code))).Inc() } return err } func (g *grpcClient) fnCall(ctx context.Context, req client.Request, rsp interface{}, opts ...client.CallOption) error { // make a copy of call opts callOpts := g.opts.CallOptions for _, opt := range opts { opt(&callOpts) } // check if we already have a deadline d, ok := ctx.Deadline() if !ok { // no deadline so we create a new one var cancel context.CancelFunc ctx, cancel = context.WithTimeout(ctx, callOpts.RequestTimeout) defer cancel() } else { // got a deadline so no need to setup context // but we need to set the timeout we pass along opt := client.WithRequestTimeout(time.Until(d)) opt(&callOpts) } // should we noop right here? select { case <-ctx.Done(): return errors.New("go.micro.client", fmt.Sprintf("%v", ctx.Err()), 408) default: } // make copy of call method gcall := g.call // use the router passed as a call option, or fallback to the rpc clients router if callOpts.Router == nil { callOpts.Router = g.opts.Router } if callOpts.Selector == nil { callOpts.Selector = g.opts.Selector } // inject proxy address // TODO: don't even bother using Lookup/Select in this case if len(g.opts.Proxy) > 0 { callOpts.Address = []string{g.opts.Proxy} } var next selector.Next call := func(i int) error { // call backoff first. Someone may want an initial start delay t, err := callOpts.Backoff(ctx, req, i) if err != nil { return errors.InternalServerError("go.micro.client", "%+v", err) } // only sleep if greater than 0 if t.Seconds() > 0 { time.Sleep(t) } if next == nil { var routes []string // lookup the route to send the reques to // TODO apply any filtering here routes, err = g.opts.Lookup(ctx, req, callOpts) if err != nil { return errors.InternalServerError("go.micro.client", "%+v", err) } // balance the list of nodes next, err = callOpts.Selector.Select(routes) if err != nil { return err } } // get the next node node := next() // make the call err = gcall(ctx, node, req, rsp, callOpts) // record the result of the call to inform future routing decisions if verr := g.opts.Selector.Record(node, err); verr != nil { return verr } // try and transform the error to a go-micro error if verr, ok := err.(*errors.Error); ok { return verr } return err } ch := make(chan error, callOpts.Retries+1) var gerr error for i := 0; i <= callOpts.Retries; i++ { go func(i int) { ch <- call(i) }(i) select { case <-ctx.Done(): return errors.New("go.micro.client", fmt.Sprintf("%v", ctx.Err()), 408) case err := <-ch: // if the call succeeded lets bail early if err == nil { return nil } retry, rerr := callOpts.Retry(ctx, req, i, err) if rerr != nil { return rerr } if !retry { return err } gerr = err } } return gerr } func (g *grpcClient) Stream(ctx context.Context, req client.Request, opts ...client.CallOption) (client.Stream, error) { ts := time.Now() g.opts.Meter.Counter(semconv.ClientRequestInflight, "endpoint", req.Endpoint()).Inc() var sp tracer.Span ctx, sp = g.opts.Tracer.Start(ctx, req.Endpoint()+" rpc-client", tracer.WithSpanKind(tracer.SpanKindClient), tracer.WithSpanLabels("endpoint", req.Endpoint()), ) stream, err := g.funcStream(ctx, req, opts...) g.opts.Meter.Counter(semconv.ClientRequestInflight, "endpoint", req.Endpoint()).Dec() te := time.Since(ts) g.opts.Meter.Summary(semconv.ClientRequestLatencyMicroseconds, "endpoint", req.Endpoint()).Update(te.Seconds()) g.opts.Meter.Histogram(semconv.ClientRequestDurationSeconds, "endpoint", req.Endpoint()).Update(te.Seconds()) if me := status.Convert(err); me == nil { sp.Finish() g.opts.Meter.Counter(semconv.ClientRequestTotal, "endpoint", req.Endpoint(), "status", "success", "code", strconv.Itoa(int(codes.OK))).Inc() } else { sp.SetStatus(tracer.SpanStatusError, err.Error()) g.opts.Meter.Counter(semconv.ClientRequestTotal, "endpoint", req.Endpoint(), "status", "failure", "code", strconv.Itoa(int(me.Code()))).Inc() } return stream, err } func (g *grpcClient) fnStream(ctx context.Context, req client.Request, opts ...client.CallOption) (client.Stream, error) { // make a copy of call opts callOpts := g.opts.CallOptions for _, opt := range opts { opt(&callOpts) } // #200 - streams shouldn't have a request timeout set on the context // should we noop right here? select { case <-ctx.Done(): return nil, errors.New("go.micro.client", fmt.Sprintf("%v", ctx.Err()), 408) default: } // make a copy of stream gstream := g.stream // use the router passed as a call option, or fallback to the rpc clients router if callOpts.Router == nil { callOpts.Router = g.opts.Router } if callOpts.Selector == nil { callOpts.Selector = g.opts.Selector } // inject proxy address // TODO: don't even bother using Lookup/Select in this case if len(g.opts.Proxy) > 0 { callOpts.Address = []string{g.opts.Proxy} } var next selector.Next call := func(i int) (client.Stream, error) { // call backoff first. Someone may want an initial start delay t, err := callOpts.Backoff(ctx, req, i) if err != nil { return nil, errors.InternalServerError("go.micro.client", "%+v", err) } // only sleep if greater than 0 if t.Seconds() > 0 { time.Sleep(t) } if next == nil { var routes []string // lookup the route to send the reques to // TODO apply any filtering here routes, err = g.opts.Lookup(ctx, req, callOpts) if err != nil { return nil, errors.InternalServerError("go.micro.client", "%+v", err) } // balance the list of nodes next, err = callOpts.Selector.Select(routes) if err != nil { return nil, err } } // get the next node node := next() // make the call stream := &grpcStream{} err = gstream(ctx, node, req, stream, callOpts) // record the result of the call to inform future routing decisions if verr := g.opts.Selector.Record(node, err); verr != nil { return nil, verr } // try and transform the error to a go-micro error if verr, ok := err.(*errors.Error); ok { return nil, verr } if rerr := g.opts.Selector.Record(node, err); rerr != nil { return nil, rerr } return stream, err } type response struct { stream client.Stream err error } ch := make(chan response, callOpts.Retries+1) var grr error for i := 0; i <= callOpts.Retries; i++ { go func(i int) { s, err := call(i) ch <- response{s, err} }(i) select { case <-ctx.Done(): return nil, errors.New("go.micro.client", fmt.Sprintf("%v", ctx.Err()), 408) case rsp := <-ch: // if the call succeeded lets bail early if rsp.err == nil { return rsp.stream, nil } retry, rerr := callOpts.Retry(ctx, req, i, grr) if rerr != nil { return nil, rerr } if !retry { return nil, rsp.err } grr = rsp.err } } return nil, grr } func (g *grpcClient) BatchPublish(ctx context.Context, ps []client.Message, opts ...client.PublishOption) error { return g.funcBatchPublish(ctx, ps, opts...) } func (g *grpcClient) fnBatchPublish(ctx context.Context, ps []client.Message, opts ...client.PublishOption) error { return g.publish(ctx, ps, opts...) } func (g *grpcClient) Publish(ctx context.Context, p client.Message, opts ...client.PublishOption) error { return g.funcPublish(ctx, p, opts...) } func (g *grpcClient) fnPublish(ctx context.Context, p client.Message, opts ...client.PublishOption) error { return g.publish(ctx, []client.Message{p}, opts...) } func (g *grpcClient) publish(ctx context.Context, ps []client.Message, opts ...client.PublishOption) error { var body []byte options := client.NewPublishOptions(opts...) // get proxy exchange := "" if v, ok := os.LookupEnv("MICRO_PROXY"); ok { exchange = v } // get the exchange if len(options.Exchange) > 0 { exchange = options.Exchange } msgs := make([]*broker.Message, 0, len(ps)) omd, ok := metadata.FromOutgoingContext(ctx) if !ok { omd = metadata.New(2) } for _, p := range ps { md := metadata.Copy(omd) topic := p.Topic() if len(exchange) > 0 { topic = exchange } md.Set(metadata.HeaderTopic, topic) iter := p.Metadata().Iterator() var k, v string for iter.Next(&k, &v) { md.Set(k, v) } md[metadata.HeaderContentType] = p.ContentType() // passed in raw data if d, ok := p.Payload().(*codec.Frame); ok { body = d.Data } else { // use codec for payload cf, err := g.newCodec(p.ContentType()) if err != nil { return errors.InternalServerError("go.micro.client", "%+v", err) } // set the body b, err := cf.Marshal(p.Payload()) if err != nil { return errors.InternalServerError("go.micro.client", "%+v", err) } body = b } msgs = append(msgs, &broker.Message{Header: md, Body: body}) } return g.opts.Broker.BatchPublish(ctx, msgs, broker.PublishContext(options.Context), broker.PublishBodyOnly(options.BodyOnly), ) } func (g *grpcClient) String() string { return "grpc" } func (g *grpcClient) Name() string { return g.opts.Name } func (g *grpcClient) getGrpcDialOptions(ctx context.Context) []grpc.DialOption { var opts []grpc.DialOption if g.opts.CallOptions.Context != nil { if v := g.opts.CallOptions.Context.Value(grpcDialOptions{}); v != nil { if vopts, ok := v.([]grpc.DialOption); ok { opts = append(opts, vopts...) } } } if ctx != nil { if v := ctx.Value(grpcDialOptions{}); v != nil { if vopts, ok := v.([]grpc.DialOption); ok { opts = append(opts, vopts...) } } } return opts } func (g *grpcClient) getGrpcCallOptions(ctx context.Context) []grpc.CallOption { var opts []grpc.CallOption if g.opts.CallOptions.Context != nil { if v := g.opts.CallOptions.Context.Value(grpcCallOptions{}); v != nil { if vopts, ok := v.([]grpc.CallOption); ok { opts = append(opts, vopts...) } } } if ctx != nil { if v := ctx.Value(grpcCallOptions{}); v != nil { if vopts, ok := v.([]grpc.CallOption); ok { opts = append(opts, vopts...) } } } return opts } func NewClient(opts ...client.Option) client.Client { options := client.NewOptions(opts...) // default content type for grpc if options.ContentType == "" { options.ContentType = DefaultContentType } c := &grpcClient{ opts: options, } c.pool = NewConnPool(options.PoolSize, options.PoolTTL, c.poolMaxIdle(), c.poolMaxStreams()) if c.opts.Context != nil { if codecs, ok := c.opts.Context.Value(codecsKey{}).(map[string]encoding.Codec); ok && codecs != nil { for k, v := range codecs { c.opts.Codecs[k] = &wrapGrpcCodec{v} } } } for _, k := range options.Codecs { encoding.RegisterCodec(&wrapMicroCodec{k}) } c.funcCall = c.fnCall c.funcStream = c.fnStream c.funcPublish = c.fnPublish c.funcBatchPublish = c.fnBatchPublish return c }