 365770fcbd
			
		
	
	365770fcbd
	
	
	
		
			
			This change introduces some behavior changes that should not impact users that
are following the proper stream protocol. Specifically, one of the following
conditions must be satisfied:
1. The user calls Close on the ClientConn.
2. The user cancels the context provided to NewClientStream, or its deadline
    expires. (Note that it if the context is no longer needed before the deadline
    expires, it is still recommended to call cancel to prevent bloat.) It is always
    recommended to cancel contexts when they are no longer needed, and to
    never use the background context directly, so all users should always be
    doing this.
3. The user calls RecvMsg (or Recv in generated code) until a non-nil error is
    returned.
4. The user receives any error from Header or SendMsg (or Send in generated
    code) besides io.EOF.  If none of the above happen, this will leak a goroutine
    and a context, and grpc will not call the optionally-configured stats handler
    with a stats.End message.
Before this change, if a user created a stream and the server ended the stream,
the stats handler would be invoked with a stats.End containing the final status
of the stream. Subsequent calls to RecvMsg would then trigger the stats handler
with InPayloads, which may be unexpected by stats handlers.
		
	
		
			
				
	
	
		
			101 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			101 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // +build go1.7
 | |
| 
 | |
| /*
 | |
|  *
 | |
|  * Copyright 2016 gRPC authors.
 | |
|  *
 | |
|  * 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
 | |
|  *
 | |
|  *     http://www.apache.org/licenses/LICENSE-2.0
 | |
|  *
 | |
|  * 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.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| package grpc
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"net"
 | |
| 	"net/http"
 | |
| 	"os"
 | |
| 
 | |
| 	netctx "golang.org/x/net/context"
 | |
| 	"google.golang.org/grpc/codes"
 | |
| 	"google.golang.org/grpc/status"
 | |
| 	"google.golang.org/grpc/transport"
 | |
| )
 | |
| 
 | |
| // dialContext connects to the address on the named network.
 | |
| func dialContext(ctx context.Context, network, address string) (net.Conn, error) {
 | |
| 	return (&net.Dialer{}).DialContext(ctx, network, address)
 | |
| }
 | |
| 
 | |
| func sendHTTPRequest(ctx context.Context, req *http.Request, conn net.Conn) error {
 | |
| 	req = req.WithContext(ctx)
 | |
| 	if err := req.Write(conn); err != nil {
 | |
| 		return fmt.Errorf("failed to write the HTTP request: %v", err)
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // toRPCErr converts an error into an error from the status package.
 | |
| func toRPCErr(err error) error {
 | |
| 	if err == nil || err == io.EOF {
 | |
| 		return err
 | |
| 	}
 | |
| 	if _, ok := status.FromError(err); ok {
 | |
| 		return err
 | |
| 	}
 | |
| 	switch e := err.(type) {
 | |
| 	case transport.StreamError:
 | |
| 		return status.Error(e.Code, e.Desc)
 | |
| 	case transport.ConnectionError:
 | |
| 		return status.Error(codes.Unavailable, e.Desc)
 | |
| 	default:
 | |
| 		switch err {
 | |
| 		case context.DeadlineExceeded, netctx.DeadlineExceeded:
 | |
| 			return status.Error(codes.DeadlineExceeded, err.Error())
 | |
| 		case context.Canceled, netctx.Canceled:
 | |
| 			return status.Error(codes.Canceled, err.Error())
 | |
| 		}
 | |
| 	}
 | |
| 	return status.Error(codes.Unknown, err.Error())
 | |
| }
 | |
| 
 | |
| // convertCode converts a standard Go error into its canonical code. Note that
 | |
| // this is only used to translate the error returned by the server applications.
 | |
| func convertCode(err error) codes.Code {
 | |
| 	switch err {
 | |
| 	case nil:
 | |
| 		return codes.OK
 | |
| 	case io.EOF:
 | |
| 		return codes.OutOfRange
 | |
| 	case io.ErrClosedPipe, io.ErrNoProgress, io.ErrShortBuffer, io.ErrShortWrite, io.ErrUnexpectedEOF:
 | |
| 		return codes.FailedPrecondition
 | |
| 	case os.ErrInvalid:
 | |
| 		return codes.InvalidArgument
 | |
| 	case context.Canceled, netctx.Canceled:
 | |
| 		return codes.Canceled
 | |
| 	case context.DeadlineExceeded, netctx.DeadlineExceeded:
 | |
| 		return codes.DeadlineExceeded
 | |
| 	}
 | |
| 	switch {
 | |
| 	case os.IsExist(err):
 | |
| 		return codes.AlreadyExists
 | |
| 	case os.IsNotExist(err):
 | |
| 		return codes.NotFound
 | |
| 	case os.IsPermission(err):
 | |
| 		return codes.PermissionDenied
 | |
| 	}
 | |
| 	return codes.Unknown
 | |
| }
 |