open-vault/vendor/google.golang.org/grpc/rpc_util.go

498 lines
14 KiB
Go
Raw Normal View History

2016-08-19 15:03:53 +00:00
/*
*
2017-06-16 15:14:18 +00:00
* Copyright 2014 gRPC authors.
2016-08-19 15:03:53 +00:00
*
2017-06-16 15:14:18 +00:00
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
2016-08-19 15:03:53 +00:00
*
2017-06-16 15:14:18 +00:00
* http://www.apache.org/licenses/LICENSE-2.0
2016-08-19 15:03:53 +00:00
*
2017-06-16 15:14:18 +00:00
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
2016-08-19 15:03:53 +00:00
*
*/
package grpc
import (
"bytes"
"compress/gzip"
"encoding/binary"
"io"
"io/ioutil"
"math"
2017-05-24 13:34:59 +00:00
"sync"
2016-11-16 23:22:54 +00:00
"time"
2016-08-19 15:03:53 +00:00
"golang.org/x/net/context"
"google.golang.org/grpc/codes"
2017-05-24 13:34:59 +00:00
"google.golang.org/grpc/credentials"
2017-11-07 16:57:05 +00:00
"google.golang.org/grpc/encoding"
2016-08-19 15:03:53 +00:00
"google.golang.org/grpc/metadata"
"google.golang.org/grpc/peer"
2016-11-16 23:22:54 +00:00
"google.golang.org/grpc/stats"
2017-04-17 15:17:06 +00:00
"google.golang.org/grpc/status"
2016-08-19 15:03:53 +00:00
"google.golang.org/grpc/transport"
)
// Compressor defines the interface gRPC uses to compress a message.
type Compressor interface {
// Do compresses p into w.
Do(w io.Writer, p []byte) error
// Type returns the compression algorithm the Compressor uses.
Type() string
}
2017-05-24 13:34:59 +00:00
type gzipCompressor struct {
pool sync.Pool
2016-08-19 15:03:53 +00:00
}
2017-05-24 13:34:59 +00:00
// NewGZIPCompressor creates a Compressor based on GZIP.
func NewGZIPCompressor() Compressor {
return &gzipCompressor{
pool: sync.Pool{
New: func() interface{} {
return gzip.NewWriter(ioutil.Discard)
},
},
}
2016-08-19 15:03:53 +00:00
}
func (c *gzipCompressor) Do(w io.Writer, p []byte) error {
2017-05-24 13:34:59 +00:00
z := c.pool.Get().(*gzip.Writer)
2017-09-05 22:06:47 +00:00
defer c.pool.Put(z)
2017-05-24 13:34:59 +00:00
z.Reset(w)
2016-08-19 15:03:53 +00:00
if _, err := z.Write(p); err != nil {
return err
}
return z.Close()
}
func (c *gzipCompressor) Type() string {
return "gzip"
}
// Decompressor defines the interface gRPC uses to decompress a message.
type Decompressor interface {
// Do reads the data from r and uncompress them.
Do(r io.Reader) ([]byte, error)
// Type returns the compression algorithm the Decompressor uses.
Type() string
}
type gzipDecompressor struct {
2017-05-24 13:34:59 +00:00
pool sync.Pool
2016-08-19 15:03:53 +00:00
}
// NewGZIPDecompressor creates a Decompressor based on GZIP.
func NewGZIPDecompressor() Decompressor {
return &gzipDecompressor{}
}
func (d *gzipDecompressor) Do(r io.Reader) ([]byte, error) {
2017-05-24 13:34:59 +00:00
var z *gzip.Reader
switch maybeZ := d.pool.Get().(type) {
case nil:
newZ, err := gzip.NewReader(r)
if err != nil {
return nil, err
}
z = newZ
case *gzip.Reader:
z = maybeZ
if err := z.Reset(r); err != nil {
d.pool.Put(z)
return nil, err
}
2016-08-19 15:03:53 +00:00
}
2017-05-24 13:34:59 +00:00
defer func() {
z.Close()
d.pool.Put(z)
}()
2016-08-19 15:03:53 +00:00
return ioutil.ReadAll(z)
}
func (d *gzipDecompressor) Type() string {
return "gzip"
}
// callInfo contains all related configuration and information about an RPC.
type callInfo struct {
2017-11-07 16:57:05 +00:00
compressorType string
2017-05-24 13:34:59 +00:00
failFast bool
headerMD metadata.MD
trailerMD metadata.MD
peer *peer.Peer
traceInfo traceInfo // in trace.go
maxReceiveMessageSize *int
maxSendMessageSize *int
creds credentials.PerRPCCredentials
2016-08-19 15:03:53 +00:00
}
2017-09-05 22:06:47 +00:00
func defaultCallInfo() *callInfo {
return &callInfo{failFast: true}
}
2016-08-19 15:03:53 +00:00
// CallOption configures a Call before it starts or extracts information from
// a Call after it completes.
type CallOption interface {
// before is called before the call is sent to any server. If before
// returns a non-nil error, the RPC fails with that error.
before(*callInfo) error
// after is called after the call has completed. after cannot return an
// error, so any failures should be reported via output parameters.
after(*callInfo)
}
2017-05-24 13:34:59 +00:00
// EmptyCallOption does not alter the Call configuration.
// It can be embedded in another structure to carry satellite data for use
// by interceptors.
type EmptyCallOption struct{}
func (EmptyCallOption) before(*callInfo) error { return nil }
func (EmptyCallOption) after(*callInfo) {}
2016-08-19 15:03:53 +00:00
type beforeCall func(c *callInfo) error
func (o beforeCall) before(c *callInfo) error { return o(c) }
func (o beforeCall) after(c *callInfo) {}
type afterCall func(c *callInfo)
func (o afterCall) before(c *callInfo) error { return nil }
func (o afterCall) after(c *callInfo) { o(c) }
// Header returns a CallOptions that retrieves the header metadata
// for a unary RPC.
func Header(md *metadata.MD) CallOption {
return afterCall(func(c *callInfo) {
*md = c.headerMD
})
}
// Trailer returns a CallOptions that retrieves the trailer metadata
// for a unary RPC.
func Trailer(md *metadata.MD) CallOption {
return afterCall(func(c *callInfo) {
*md = c.trailerMD
})
}
// Peer returns a CallOption that retrieves peer information for a
// unary RPC.
func Peer(peer *peer.Peer) CallOption {
return afterCall(func(c *callInfo) {
2017-04-17 15:17:06 +00:00
if c.peer != nil {
*peer = *c.peer
}
})
}
2016-08-19 15:03:53 +00:00
// FailFast configures the action to take when an RPC is attempted on broken
2017-11-07 16:57:05 +00:00
// connections or unreachable servers. If failFast is true, the RPC will fail
2016-08-19 15:03:53 +00:00
// immediately. Otherwise, the RPC client will block the call until a
2017-11-07 16:57:05 +00:00
// connection is available (or the call is canceled or times out) and will
// retry the call if it fails due to a transient error. gRPC will not retry if
// data was written to the wire unless the server indicates it did not process
// the data. Please refer to
// https://github.com/grpc/grpc/blob/master/doc/wait-for-ready.md.
//
// By default, RPCs are "Fail Fast".
2016-08-19 15:03:53 +00:00
func FailFast(failFast bool) CallOption {
return beforeCall(func(c *callInfo) error {
c.failFast = failFast
return nil
})
}
2017-05-24 13:34:59 +00:00
// MaxCallRecvMsgSize returns a CallOption which sets the maximum message size the client can receive.
func MaxCallRecvMsgSize(s int) CallOption {
return beforeCall(func(o *callInfo) error {
o.maxReceiveMessageSize = &s
return nil
})
}
// MaxCallSendMsgSize returns a CallOption which sets the maximum message size the client can send.
func MaxCallSendMsgSize(s int) CallOption {
return beforeCall(func(o *callInfo) error {
o.maxSendMessageSize = &s
return nil
})
}
// PerRPCCredentials returns a CallOption that sets credentials.PerRPCCredentials
// for a call.
func PerRPCCredentials(creds credentials.PerRPCCredentials) CallOption {
return beforeCall(func(c *callInfo) error {
c.creds = creds
return nil
})
}
2016-08-19 15:03:53 +00:00
// The format of the payload: compressed or not?
type payloadFormat uint8
const (
compressionNone payloadFormat = iota // no compression
compressionMade
)
// parser reads complete gRPC messages from the underlying reader.
type parser struct {
// r is the underlying reader.
// See the comment on recvMsg for the permissible
// error types.
r io.Reader
// The header of a gRPC message. Find more detail
2017-07-18 14:15:54 +00:00
// at https://grpc.io/docs/guides/wire.html.
2016-08-19 15:03:53 +00:00
header [5]byte
}
// recvMsg reads a complete gRPC message from the stream.
//
// It returns the message and its payload (compression/encoding)
// format. The caller owns the returned msg memory.
//
// If there is an error, possible values are:
// * io.EOF, when no messages remain
// * io.ErrUnexpectedEOF
// * of type transport.ConnectionError
// * of type transport.StreamError
// No other error values or types must be returned, which also means
// that the underlying io.Reader must not return an incompatible
// error.
2017-05-24 13:34:59 +00:00
func (p *parser) recvMsg(maxReceiveMessageSize int) (pf payloadFormat, msg []byte, err error) {
2017-05-25 01:07:45 +00:00
if _, err := p.r.Read(p.header[:]); err != nil {
2016-08-19 15:03:53 +00:00
return 0, nil, err
}
pf = payloadFormat(p.header[0])
length := binary.BigEndian.Uint32(p.header[1:])
if length == 0 {
return pf, nil, nil
}
2017-10-27 19:06:04 +00:00
if int64(length) > int64(maxInt) {
return 0, nil, Errorf(codes.ResourceExhausted, "grpc: received message larger than max length allowed on current machine (%d vs. %d)", length, maxInt)
}
if int(length) > maxReceiveMessageSize {
2017-05-24 13:34:59 +00:00
return 0, nil, Errorf(codes.ResourceExhausted, "grpc: received message larger than max (%d vs. %d)", length, maxReceiveMessageSize)
2016-08-19 15:03:53 +00:00
}
// TODO(bradfitz,zhaoq): garbage. reuse buffer after proto decoding instead
// of making it for each message:
msg = make([]byte, int(length))
2017-05-25 01:07:45 +00:00
if _, err := p.r.Read(msg); err != nil {
2016-08-19 15:03:53 +00:00
if err == io.EOF {
err = io.ErrUnexpectedEOF
}
return 0, nil, err
}
return pf, msg, nil
}
2017-09-05 22:06:47 +00:00
// encode serializes msg and returns a buffer of message header and a buffer of msg.
// If msg is nil, it generates the message header and an empty msg buffer.
2017-11-07 16:57:05 +00:00
// TODO(ddyihai): eliminate extra Compressor parameter.
func encode(c Codec, msg interface{}, cp Compressor, outPayload *stats.OutPayload, compressor encoding.Compressor) ([]byte, []byte, error) {
var (
b []byte
cbuf *bytes.Buffer
)
2017-09-05 22:06:47 +00:00
const (
payloadLen = 1
sizeLen = 4
2016-11-16 23:22:54 +00:00
)
2016-08-19 15:03:53 +00:00
if msg != nil {
var err error
b, err = c.Marshal(msg)
if err != nil {
2017-09-05 22:06:47 +00:00
return nil, nil, Errorf(codes.Internal, "grpc: error while marshaling: %v", err.Error())
2016-08-19 15:03:53 +00:00
}
2016-11-16 23:22:54 +00:00
if outPayload != nil {
outPayload.Payload = msg
// TODO truncate large payload.
outPayload.Data = b
outPayload.Length = len(b)
}
2017-11-07 16:57:05 +00:00
if compressor != nil || cp != nil {
cbuf = new(bytes.Buffer)
// Has compressor, check Compressor is set by UseCompressor first.
if compressor != nil {
z, _ := compressor.Compress(cbuf)
if _, err := z.Write(b); err != nil {
return nil, nil, Errorf(codes.Internal, "grpc: error while compressing: %v", err.Error())
}
z.Close()
} else {
// If Compressor is not set by UseCompressor, use default Compressor
if err := cp.Do(cbuf, b); err != nil {
return nil, nil, Errorf(codes.Internal, "grpc: error while compressing: %v", err.Error())
}
2016-08-19 15:03:53 +00:00
}
b = cbuf.Bytes()
}
}
2017-09-05 22:06:47 +00:00
if uint(len(b)) > math.MaxUint32 {
return nil, nil, Errorf(codes.ResourceExhausted, "grpc: message too large (%d bytes)", len(b))
}
2016-08-19 15:03:53 +00:00
2017-09-05 22:06:47 +00:00
bufHeader := make([]byte, payloadLen+sizeLen)
2017-11-07 16:57:05 +00:00
if compressor != nil || cp != nil {
2017-09-05 22:06:47 +00:00
bufHeader[0] = byte(compressionMade)
2017-11-07 16:57:05 +00:00
} else {
bufHeader[0] = byte(compressionNone)
2016-08-19 15:03:53 +00:00
}
2017-11-07 16:57:05 +00:00
2016-08-19 15:03:53 +00:00
// Write length of b into buf
2017-09-05 22:06:47 +00:00
binary.BigEndian.PutUint32(bufHeader[payloadLen:], uint32(len(b)))
2016-11-16 23:22:54 +00:00
if outPayload != nil {
2017-09-05 22:06:47 +00:00
outPayload.WireLength = payloadLen + sizeLen + len(b)
2016-11-16 23:22:54 +00:00
}
2017-09-05 22:06:47 +00:00
return bufHeader, b, nil
2016-08-19 15:03:53 +00:00
}
func checkRecvPayload(pf payloadFormat, recvCompress string, dc Decompressor) error {
switch pf {
case compressionNone:
case compressionMade:
2017-11-07 16:57:05 +00:00
if (dc == nil || recvCompress != dc.Type()) && encoding.GetCompressor(recvCompress) == nil {
2016-09-02 22:05:09 +00:00
return Errorf(codes.Unimplemented, "grpc: Decompressor is not installed for grpc-encoding %q", recvCompress)
2016-08-19 15:03:53 +00:00
}
default:
2016-09-02 22:05:09 +00:00
return Errorf(codes.Internal, "grpc: received unexpected payload format %d", pf)
2016-08-19 15:03:53 +00:00
}
return nil
}
2017-11-07 16:57:05 +00:00
// TODO(ddyihai): eliminate extra Compressor parameter.
func recv(p *parser, c Codec, s *transport.Stream, dc Decompressor, m interface{}, maxReceiveMessageSize int,
inPayload *stats.InPayload, compressor encoding.Compressor) error {
2017-05-24 13:34:59 +00:00
pf, d, err := p.recvMsg(maxReceiveMessageSize)
2016-08-19 15:03:53 +00:00
if err != nil {
return err
}
2016-11-16 23:22:54 +00:00
if inPayload != nil {
inPayload.WireLength = len(d)
}
2016-08-19 15:03:53 +00:00
if err := checkRecvPayload(pf, s.RecvCompress(), dc); err != nil {
return err
}
if pf == compressionMade {
2017-11-07 16:57:05 +00:00
// To match legacy behavior, if the decompressor is set by WithDecompressor or RPCDecompressor,
// use this decompressor as the default.
if dc != nil {
d, err = dc.Do(bytes.NewReader(d))
if err != nil {
return Errorf(codes.Internal, "grpc: failed to decompress the received message %v", err)
}
} else {
dcReader, err := compressor.Decompress(bytes.NewReader(d))
if err != nil {
return Errorf(codes.Internal, "grpc: failed to decompress the received message %v", err)
}
d, err = ioutil.ReadAll(dcReader)
if err != nil {
return Errorf(codes.Internal, "grpc: failed to decompress the received message %v", err)
}
2016-08-19 15:03:53 +00:00
}
}
2017-05-24 13:34:59 +00:00
if len(d) > maxReceiveMessageSize {
2016-08-19 15:03:53 +00:00
// TODO: Revisit the error code. Currently keep it consistent with java
// implementation.
2017-05-24 13:34:59 +00:00
return Errorf(codes.ResourceExhausted, "grpc: received message larger than max (%d vs. %d)", len(d), maxReceiveMessageSize)
2016-08-19 15:03:53 +00:00
}
if err := c.Unmarshal(d, m); err != nil {
return Errorf(codes.Internal, "grpc: failed to unmarshal the received message %v", err)
}
2016-11-16 23:22:54 +00:00
if inPayload != nil {
inPayload.RecvTime = time.Now()
inPayload.Payload = m
// TODO truncate large payload.
inPayload.Data = d
inPayload.Length = len(d)
}
2016-08-19 15:03:53 +00:00
return nil
}
2017-05-24 13:34:59 +00:00
type rpcInfo struct {
2017-09-05 22:06:47 +00:00
failfast bool
2017-05-24 13:34:59 +00:00
bytesSent bool
bytesReceived bool
}
type rpcInfoContextKey struct{}
2017-09-05 22:06:47 +00:00
func newContextWithRPCInfo(ctx context.Context, failfast bool) context.Context {
return context.WithValue(ctx, rpcInfoContextKey{}, &rpcInfo{failfast: failfast})
2017-05-24 13:34:59 +00:00
}
func rpcInfoFromContext(ctx context.Context) (s *rpcInfo, ok bool) {
s, ok = ctx.Value(rpcInfoContextKey{}).(*rpcInfo)
return
}
func updateRPCInfoInContext(ctx context.Context, s rpcInfo) {
if ss, ok := rpcInfoFromContext(ctx); ok {
2017-09-05 22:06:47 +00:00
ss.bytesReceived = s.bytesReceived
ss.bytesSent = s.bytesSent
2017-05-24 13:34:59 +00:00
}
return
}
2016-08-19 15:03:53 +00:00
// Code returns the error code for err if it was produced by the rpc system.
// Otherwise, it returns codes.Unknown.
2017-04-17 15:17:06 +00:00
//
// Deprecated; use status.FromError and Code method instead.
2016-08-19 15:03:53 +00:00
func Code(err error) codes.Code {
2017-04-17 15:17:06 +00:00
if s, ok := status.FromError(err); ok {
return s.Code()
2016-08-19 15:03:53 +00:00
}
return codes.Unknown
}
// ErrorDesc returns the error description of err if it was produced by the rpc system.
// Otherwise, it returns err.Error() or empty string when err is nil.
2017-04-17 15:17:06 +00:00
//
// Deprecated; use status.FromError and Message method instead.
2016-08-19 15:03:53 +00:00
func ErrorDesc(err error) string {
2017-04-17 15:17:06 +00:00
if s, ok := status.FromError(err); ok {
return s.Message()
2016-08-19 15:03:53 +00:00
}
return err.Error()
}
// Errorf returns an error containing an error code and a description;
// Errorf returns nil if c is OK.
2017-04-17 15:17:06 +00:00
//
// Deprecated; use status.Errorf instead.
2016-08-19 15:03:53 +00:00
func Errorf(c codes.Code, format string, a ...interface{}) error {
2017-04-17 15:17:06 +00:00
return status.Errorf(c, format, a...)
2016-08-19 15:03:53 +00:00
}
2017-10-27 19:06:04 +00:00
// The SupportPackageIsVersion variables are referenced from generated protocol
// buffer files to ensure compatibility with the gRPC version used. The latest
// support package version is 5.
2016-08-19 15:03:53 +00:00
//
2017-10-27 19:06:04 +00:00
// Older versions are kept for compatibility. They may be removed if
// compatibility cannot be maintained.
//
// These constants should not be referenced from any other code.
const (
SupportPackageIsVersion3 = true
SupportPackageIsVersion4 = true
SupportPackageIsVersion5 = true
)
2017-03-31 00:03:13 +00:00
// Version is the current grpc version.
2017-10-27 19:06:04 +00:00
const Version = "1.8.0-dev"
2017-05-24 13:34:59 +00:00
const grpcUA = "grpc-go/" + Version