128 lines
3.9 KiB
Go
128 lines
3.9 KiB
Go
|
package runtime
|
||
|
|
||
|
import (
|
||
|
"io"
|
||
|
"net/http"
|
||
|
|
||
|
"github.com/golang/protobuf/proto"
|
||
|
"golang.org/x/net/context"
|
||
|
"google.golang.org/grpc/codes"
|
||
|
"google.golang.org/grpc/grpclog"
|
||
|
"google.golang.org/grpc/status"
|
||
|
)
|
||
|
|
||
|
// HTTPStatusFromCode converts a gRPC error code into the corresponding HTTP response status.
|
||
|
func HTTPStatusFromCode(code codes.Code) int {
|
||
|
switch code {
|
||
|
case codes.OK:
|
||
|
return http.StatusOK
|
||
|
case codes.Canceled:
|
||
|
return http.StatusRequestTimeout
|
||
|
case codes.Unknown:
|
||
|
return http.StatusInternalServerError
|
||
|
case codes.InvalidArgument:
|
||
|
return http.StatusBadRequest
|
||
|
case codes.DeadlineExceeded:
|
||
|
return http.StatusRequestTimeout
|
||
|
case codes.NotFound:
|
||
|
return http.StatusNotFound
|
||
|
case codes.AlreadyExists:
|
||
|
return http.StatusConflict
|
||
|
case codes.PermissionDenied:
|
||
|
return http.StatusForbidden
|
||
|
case codes.Unauthenticated:
|
||
|
return http.StatusUnauthorized
|
||
|
case codes.ResourceExhausted:
|
||
|
return http.StatusForbidden
|
||
|
case codes.FailedPrecondition:
|
||
|
return http.StatusPreconditionFailed
|
||
|
case codes.Aborted:
|
||
|
return http.StatusConflict
|
||
|
case codes.OutOfRange:
|
||
|
return http.StatusBadRequest
|
||
|
case codes.Unimplemented:
|
||
|
return http.StatusNotImplemented
|
||
|
case codes.Internal:
|
||
|
return http.StatusInternalServerError
|
||
|
case codes.Unavailable:
|
||
|
return http.StatusServiceUnavailable
|
||
|
case codes.DataLoss:
|
||
|
return http.StatusInternalServerError
|
||
|
}
|
||
|
|
||
|
grpclog.Printf("Unknown gRPC error code: %v", code)
|
||
|
return http.StatusInternalServerError
|
||
|
}
|
||
|
|
||
|
var (
|
||
|
// HTTPError replies to the request with the error.
|
||
|
// You can set a custom function to this variable to customize error format.
|
||
|
HTTPError = DefaultHTTPError
|
||
|
// OtherErrorHandler handles the following error used by the gateway: StatusMethodNotAllowed StatusNotFound and StatusBadRequest
|
||
|
OtherErrorHandler = DefaultOtherErrorHandler
|
||
|
)
|
||
|
|
||
|
type errorBody struct {
|
||
|
Error string `protobuf:"bytes,1,name=error" json:"error"`
|
||
|
Code int32 `protobuf:"varint,2,name=code" json:"code"`
|
||
|
}
|
||
|
|
||
|
//Make this also conform to proto.Message for builtin JSONPb Marshaler
|
||
|
func (e *errorBody) Reset() { *e = errorBody{} }
|
||
|
func (e *errorBody) String() string { return proto.CompactTextString(e) }
|
||
|
func (*errorBody) ProtoMessage() {}
|
||
|
|
||
|
// DefaultHTTPError is the default implementation of HTTPError.
|
||
|
// If "err" is an error from gRPC system, the function replies with the status code mapped by HTTPStatusFromCode.
|
||
|
// If otherwise, it replies with http.StatusInternalServerError.
|
||
|
//
|
||
|
// The response body returned by this function is a JSON object,
|
||
|
// which contains a member whose key is "error" and whose value is err.Error().
|
||
|
func DefaultHTTPError(ctx context.Context, mux *ServeMux, marshaler Marshaler, w http.ResponseWriter, _ *http.Request, err error) {
|
||
|
const fallback = `{"error": "failed to marshal error message"}`
|
||
|
|
||
|
w.Header().Del("Trailer")
|
||
|
w.Header().Set("Content-Type", marshaler.ContentType())
|
||
|
|
||
|
s, ok := status.FromError(err)
|
||
|
if !ok {
|
||
|
s = status.New(codes.Unknown, err.Error())
|
||
|
}
|
||
|
|
||
|
body := &errorBody{
|
||
|
Error: s.Message(),
|
||
|
Code: int32(s.Code()),
|
||
|
}
|
||
|
|
||
|
buf, merr := marshaler.Marshal(body)
|
||
|
if merr != nil {
|
||
|
grpclog.Printf("Failed to marshal error message %q: %v", body, merr)
|
||
|
w.WriteHeader(http.StatusInternalServerError)
|
||
|
if _, err := io.WriteString(w, fallback); err != nil {
|
||
|
grpclog.Printf("Failed to write response: %v", err)
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
md, ok := ServerMetadataFromContext(ctx)
|
||
|
if !ok {
|
||
|
grpclog.Printf("Failed to extract ServerMetadata from context")
|
||
|
}
|
||
|
|
||
|
handleForwardResponseServerMetadata(w, mux, md)
|
||
|
handleForwardResponseTrailerHeader(w, md)
|
||
|
st := HTTPStatusFromCode(s.Code())
|
||
|
w.WriteHeader(st)
|
||
|
if _, err := w.Write(buf); err != nil {
|
||
|
grpclog.Printf("Failed to write response: %v", err)
|
||
|
}
|
||
|
|
||
|
handleForwardResponseTrailer(w, md)
|
||
|
}
|
||
|
|
||
|
// DefaultOtherErrorHandler is the default implementation of OtherErrorHandler.
|
||
|
// It simply writes a string representation of the given error into "w".
|
||
|
func DefaultOtherErrorHandler(w http.ResponseWriter, _ *http.Request, msg string, code int) {
|
||
|
http.Error(w, msg, code)
|
||
|
}
|