2017-01-26 15:51:44 +00:00
/ *
*
2017-09-30 16:47:47 +00:00
* Copyright 2014 gRPC authors .
2017-01-26 15:51:44 +00:00
*
2017-09-30 16:47:47 +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
2017-01-26 15:51:44 +00:00
*
2017-09-30 16:47:47 +00:00
* http : //www.apache.org/licenses/LICENSE-2.0
2017-01-26 15:51:44 +00:00
*
2017-09-30 16:47:47 +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 .
2017-01-26 15:51:44 +00:00
*
* /
package grpc
import (
"bytes"
"compress/gzip"
"encoding/binary"
"io"
"io/ioutil"
"math"
2017-09-30 16:47:47 +00:00
"sync"
2017-01-26 15:51:44 +00:00
"time"
"golang.org/x/net/context"
"google.golang.org/grpc/codes"
2017-09-30 16:47:47 +00:00
"google.golang.org/grpc/credentials"
2018-01-03 19:19:49 +00:00
"google.golang.org/grpc/encoding"
2017-01-26 15:51:44 +00:00
"google.golang.org/grpc/metadata"
2017-09-30 16:47:47 +00:00
"google.golang.org/grpc/peer"
2017-01-26 15:51:44 +00:00
"google.golang.org/grpc/stats"
2017-09-30 16:47:47 +00:00
"google.golang.org/grpc/status"
2017-01-26 15:51:44 +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-09-30 16:47:47 +00:00
type gzipCompressor struct {
pool sync . Pool
2017-01-26 15:51:44 +00:00
}
2017-09-30 16:47:47 +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 )
} ,
} ,
}
2017-01-26 15:51:44 +00:00
}
func ( c * gzipCompressor ) Do ( w io . Writer , p [ ] byte ) error {
2017-09-30 16:47:47 +00:00
z := c . pool . Get ( ) . ( * gzip . Writer )
defer c . pool . Put ( z )
z . Reset ( w )
2017-01-26 15:51:44 +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-09-30 16:47:47 +00:00
pool sync . Pool
2017-01-26 15:51:44 +00:00
}
// NewGZIPDecompressor creates a Decompressor based on GZIP.
func NewGZIPDecompressor ( ) Decompressor {
return & gzipDecompressor { }
}
func ( d * gzipDecompressor ) Do ( r io . Reader ) ( [ ] byte , error ) {
2017-09-30 16:47:47 +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
}
2017-01-26 15:51:44 +00:00
}
2017-09-30 16:47:47 +00:00
defer func ( ) {
z . Close ( )
d . pool . Put ( z )
} ( )
2017-01-26 15:51:44 +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 {
2018-01-03 19:19:49 +00:00
compressorType string
2017-09-30 16:47:47 +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
2017-01-26 15:51:44 +00:00
}
2018-01-03 19:19:49 +00:00
func defaultCallInfo ( ) * callInfo {
return & callInfo { failFast : true }
}
2017-01-26 15:51:44 +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-09-30 16:47:47 +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 ) { }
2017-01-26 15:51:44 +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
} )
}
2017-09-30 16:47:47 +00:00
// Peer returns a CallOption that retrieves peer information for a
// unary RPC.
func Peer ( peer * peer . Peer ) CallOption {
return afterCall ( func ( c * callInfo ) {
if c . peer != nil {
* peer = * c . peer
}
} )
}
2017-01-26 15:51:44 +00:00
// FailFast configures the action to take when an RPC is attempted on broken
2018-01-03 19:19:49 +00:00
// connections or unreachable servers. If failFast is true, the RPC will fail
2017-01-26 15:51:44 +00:00
// immediately. Otherwise, the RPC client will block the call until a
2018-01-03 19:19:49 +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
2017-09-30 16:47:47 +00:00
// https://github.com/grpc/grpc/blob/master/doc/wait-for-ready.md.
2018-01-03 19:19:49 +00:00
//
// By default, RPCs are "Fail Fast".
2017-01-26 15:51:44 +00:00
func FailFast ( failFast bool ) CallOption {
return beforeCall ( func ( c * callInfo ) error {
c . failFast = failFast
return nil
} )
}
2017-09-30 16:47:47 +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
} )
}
2018-01-03 19:19:49 +00:00
// UseCompressor returns a CallOption which sets the compressor used when
// sending the request. If WithCompressor is also set, UseCompressor has
// higher priority.
//
// This API is EXPERIMENTAL.
func UseCompressor ( name string ) CallOption {
return beforeCall ( func ( c * callInfo ) error {
c . compressorType = name
return nil
} )
}
2017-01-26 15:51:44 +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-09-30 16:47:47 +00:00
// at https://grpc.io/docs/guides/wire.html.
2017-01-26 15:51:44 +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-09-30 16:47:47 +00:00
func ( p * parser ) recvMsg ( maxReceiveMessageSize int ) ( pf payloadFormat , msg [ ] byte , err error ) {
if _ , err := p . r . Read ( p . header [ : ] ) ; err != nil {
2017-01-26 15:51:44 +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
}
2018-01-03 19:19:49 +00:00
if int64 ( length ) > int64 ( maxInt ) {
return 0 , nil , status . Errorf ( codes . ResourceExhausted , "grpc: received message larger than max length allowed on current machine (%d vs. %d)" , length , maxInt )
}
if int ( length ) > maxReceiveMessageSize {
return 0 , nil , status . Errorf ( codes . ResourceExhausted , "grpc: received message larger than max (%d vs. %d)" , length , maxReceiveMessageSize )
2017-01-26 15:51:44 +00:00
}
// TODO(bradfitz,zhaoq): garbage. reuse buffer after proto decoding instead
// of making it for each message:
msg = make ( [ ] byte , int ( length ) )
2017-09-30 16:47:47 +00:00
if _ , err := p . r . Read ( msg ) ; err != nil {
2017-01-26 15:51:44 +00:00
if err == io . EOF {
err = io . ErrUnexpectedEOF
}
return 0 , nil , err
}
return pf , msg , nil
}
2017-09-30 16:47: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.
2018-01-03 19:19:49 +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-30 16:47:47 +00:00
const (
payloadLen = 1
sizeLen = 4
2017-01-26 15:51:44 +00:00
)
if msg != nil {
var err error
b , err = c . Marshal ( msg )
if err != nil {
2018-01-03 19:19:49 +00:00
return nil , nil , status . Errorf ( codes . Internal , "grpc: error while marshaling: %v" , err . Error ( ) )
2017-01-26 15:51:44 +00:00
}
if outPayload != nil {
outPayload . Payload = msg
// TODO truncate large payload.
outPayload . Data = b
outPayload . Length = len ( b )
}
2018-01-03 19:19:49 +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 , status . 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 , status . Errorf ( codes . Internal , "grpc: error while compressing: %v" , err . Error ( ) )
}
2017-01-26 15:51:44 +00:00
}
b = cbuf . Bytes ( )
}
}
2017-09-30 16:47:47 +00:00
if uint ( len ( b ) ) > math . MaxUint32 {
2018-01-03 19:19:49 +00:00
return nil , nil , status . Errorf ( codes . ResourceExhausted , "grpc: message too large (%d bytes)" , len ( b ) )
2017-09-30 16:47:47 +00:00
}
2017-01-26 15:51:44 +00:00
2017-09-30 16:47:47 +00:00
bufHeader := make ( [ ] byte , payloadLen + sizeLen )
2018-01-03 19:19:49 +00:00
if compressor != nil || cp != nil {
2017-09-30 16:47:47 +00:00
bufHeader [ 0 ] = byte ( compressionMade )
2018-01-03 19:19:49 +00:00
} else {
bufHeader [ 0 ] = byte ( compressionNone )
2017-01-26 15:51:44 +00:00
}
2018-01-03 19:19:49 +00:00
2017-01-26 15:51:44 +00:00
// Write length of b into buf
2017-09-30 16:47:47 +00:00
binary . BigEndian . PutUint32 ( bufHeader [ payloadLen : ] , uint32 ( len ( b ) ) )
2017-01-26 15:51:44 +00:00
if outPayload != nil {
2017-09-30 16:47:47 +00:00
outPayload . WireLength = payloadLen + sizeLen + len ( b )
2017-01-26 15:51:44 +00:00
}
2017-09-30 16:47:47 +00:00
return bufHeader , b , nil
2017-01-26 15:51:44 +00:00
}
2018-01-03 19:19:49 +00:00
func checkRecvPayload ( pf payloadFormat , recvCompress string , haveCompressor bool ) * status . Status {
2017-01-26 15:51:44 +00:00
switch pf {
case compressionNone :
case compressionMade :
2018-01-03 19:19:49 +00:00
if recvCompress == "" || recvCompress == encoding . Identity {
return status . New ( codes . Internal , "grpc: compressed flag set with identity or empty encoding" )
}
if ! haveCompressor {
return status . Newf ( codes . Unimplemented , "grpc: Decompressor is not installed for grpc-encoding %q" , recvCompress )
2017-01-26 15:51:44 +00:00
}
default :
2018-01-03 19:19:49 +00:00
return status . Newf ( codes . Internal , "grpc: received unexpected payload format %d" , pf )
2017-01-26 15:51:44 +00:00
}
return nil
}
2018-01-03 19:19:49 +00:00
// For the two compressor parameters, both should not be set, but if they are,
// dc takes precedence over compressor.
// TODO(dfawley): wrap the old compressor/decompressor using the new API?
func recv ( p * parser , c Codec , s * transport . Stream , dc Decompressor , m interface { } , maxReceiveMessageSize int , inPayload * stats . InPayload , compressor encoding . Compressor ) error {
2017-09-30 16:47:47 +00:00
pf , d , err := p . recvMsg ( maxReceiveMessageSize )
2017-01-26 15:51:44 +00:00
if err != nil {
return err
}
if inPayload != nil {
inPayload . WireLength = len ( d )
}
2018-01-03 19:19:49 +00:00
if st := checkRecvPayload ( pf , s . RecvCompress ( ) , compressor != nil || dc != nil ) ; st != nil {
return st . Err ( )
2017-01-26 15:51:44 +00:00
}
2018-01-03 19:19:49 +00:00
2017-01-26 15:51:44 +00:00
if pf == compressionMade {
2018-01-03 19:19:49 +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 status . Errorf ( codes . Internal , "grpc: failed to decompress the received message %v" , err )
}
} else {
dcReader , err := compressor . Decompress ( bytes . NewReader ( d ) )
if err != nil {
return status . Errorf ( codes . Internal , "grpc: failed to decompress the received message %v" , err )
}
d , err = ioutil . ReadAll ( dcReader )
if err != nil {
return status . Errorf ( codes . Internal , "grpc: failed to decompress the received message %v" , err )
}
2017-01-26 15:51:44 +00:00
}
}
2017-09-30 16:47:47 +00:00
if len ( d ) > maxReceiveMessageSize {
2017-01-26 15:51:44 +00:00
// TODO: Revisit the error code. Currently keep it consistent with java
// implementation.
2018-01-03 19:19:49 +00:00
return status . Errorf ( codes . ResourceExhausted , "grpc: received message larger than max (%d vs. %d)" , len ( d ) , maxReceiveMessageSize )
2017-01-26 15:51:44 +00:00
}
if err := c . Unmarshal ( d , m ) ; err != nil {
2018-01-03 19:19:49 +00:00
return status . Errorf ( codes . Internal , "grpc: failed to unmarshal the received message %v" , err )
2017-01-26 15:51:44 +00:00
}
if inPayload != nil {
inPayload . RecvTime = time . Now ( )
inPayload . Payload = m
// TODO truncate large payload.
inPayload . Data = d
inPayload . Length = len ( d )
}
return nil
}
2017-09-30 16:47:47 +00:00
type rpcInfo struct {
2018-01-03 19:19:49 +00:00
failfast bool
2017-09-30 16:47:47 +00:00
}
type rpcInfoContextKey struct { }
2018-01-03 19:19:49 +00:00
func newContextWithRPCInfo ( ctx context . Context , failfast bool ) context . Context {
return context . WithValue ( ctx , rpcInfoContextKey { } , & rpcInfo { failfast : failfast } )
2017-09-30 16:47:47 +00:00
}
func rpcInfoFromContext ( ctx context . Context ) ( s * rpcInfo , ok bool ) {
s , ok = ctx . Value ( rpcInfoContextKey { } ) . ( * rpcInfo )
return
2017-01-26 15:51:44 +00:00
}
// Code returns the error code for err if it was produced by the rpc system.
// Otherwise, it returns codes.Unknown.
2017-09-30 16:47:47 +00:00
//
2018-01-03 19:19:49 +00:00
// Deprecated: use status.FromError and Code method instead.
2017-01-26 15:51:44 +00:00
func Code ( err error ) codes . Code {
2017-09-30 16:47:47 +00:00
if s , ok := status . FromError ( err ) ; ok {
return s . Code ( )
2017-01-26 15:51:44 +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-09-30 16:47:47 +00:00
//
2018-01-03 19:19:49 +00:00
// Deprecated: use status.FromError and Message method instead.
2017-01-26 15:51:44 +00:00
func ErrorDesc ( err error ) string {
2017-09-30 16:47:47 +00:00
if s , ok := status . FromError ( err ) ; ok {
return s . Message ( )
2017-01-26 15:51:44 +00:00
}
return err . Error ( )
}
// Errorf returns an error containing an error code and a description;
// Errorf returns nil if c is OK.
2017-09-30 16:47:47 +00:00
//
2018-01-03 19:19:49 +00:00
// Deprecated: use status.Errorf instead.
2017-01-26 15:51:44 +00:00
func Errorf ( c codes . Code , format string , a ... interface { } ) error {
2017-09-30 16:47:47 +00:00
return status . Errorf ( c , format , a ... )
2017-01-26 15:51:44 +00:00
}
2018-01-03 19:19:49 +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.
//
// Older versions are kept for compatibility. They may be removed if
// compatibility cannot be maintained.
2017-01-26 15:51:44 +00:00
//
2018-01-03 19:19:49 +00:00
// These constants should not be referenced from any other code.
const (
SupportPackageIsVersion3 = true
SupportPackageIsVersion4 = true
SupportPackageIsVersion5 = true
)
2017-09-30 16:47:47 +00:00
// Version is the current grpc version.
2018-01-03 19:19:49 +00:00
const Version = "1.9.0"
2017-09-30 16:47:47 +00:00
const grpcUA = "grpc-go/" + Version