You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
boundary/internal/servers/controller/handlers/errors.go

98 lines
3.4 KiB

package handlers
import (
"context"
"fmt"
"io"
"net/http"
"sort"
"github.com/grpc-ecosystem/grpc-gateway/runtime"
"github.com/hashicorp/go-hclog"
pb "github.com/hashicorp/watchtower/internal/gen/controller/api"
"google.golang.org/genproto/googleapis/rpc/errdetails"
"google.golang.org/grpc/codes"
"google.golang.org/grpc/status"
)
// NotFoundError returns an ApiError indicating a resource couldn't be found.
func NotFoundErrorf(msg string, a ...interface{}) error {
return status.Errorf(codes.NotFound, msg, a...)
}
func InvalidArgumentErrorf(msg string, fields map[string]string) error {
st := status.New(codes.InvalidArgument, msg)
br := &errdetails.BadRequest{}
for k, v := range fields {
br.FieldViolations = append(br.FieldViolations, &errdetails.BadRequest_FieldViolation{Field: k, Description: v})
}
sort.Slice(br.FieldViolations, func(i, j int) bool {
return br.FieldViolations[i].GetField() < br.FieldViolations[j].GetField()
})
st, err := st.WithDetails(br)
if err != nil {
hclog.Default().Error("failure building status with details", "details", br, "error", err)
return status.Error(codes.Internal, "Failed to build InvalidArgument error.")
}
return st.Err()
}
func statusErrorToApiError(s *status.Status) *pb.Error {
apiErr := &pb.Error{}
apiErr.Status = int32(runtime.HTTPStatusFromCode(s.Code()))
if s.Code() == codes.Unimplemented {
// Instead of returning a 501 we always want to return a 405 when a method isn't implemented.
apiErr.Status = http.StatusMethodNotAllowed
}
apiErr.Message = s.Message()
// TODO(ICU-193): Decouple from the status codes and instead use codes defined specifically for our API.
apiErr.Code = s.Code().String()
for _, ed := range s.Details() {
switch ed.(type) {
case *errdetails.BadRequest:
br := ed.(*errdetails.BadRequest)
for _, fv := range br.GetFieldViolations() {
if apiErr.Details == nil {
apiErr.Details = &pb.ErrorDetails{}
}
apiErr.Details.RequestFields = append(apiErr.Details.RequestFields, &pb.FieldError{Name: fv.GetField(), Description: fv.GetDescription()})
}
}
}
return apiErr
}
// TODO(ICU-194): Remove all information from internal errors.
func ErrorHandler(logger hclog.Logger) runtime.ProtoErrorHandlerFunc {
const errorFallback = `{"error": "failed to marshal error message"}`
return func(ctx context.Context, _ *runtime.ServeMux, marshaler runtime.Marshaler, w http.ResponseWriter, r *http.Request, inErr error) {
if inErr == runtime.ErrUnknownURI {
// grpc gateway uses this error when the path was not matched, but the error uses codes.Unimplemented which doesn't match the intention.
// Overwrite the error to match our expected behavior.
inErr = status.Error(codes.NotFound, http.StatusText(http.StatusNotFound))
}
s, ok := status.FromError(inErr)
if !ok {
s = status.New(codes.Unknown, inErr.Error())
}
apiErr := statusErrorToApiError(s)
buf, merr := marshaler.Marshal(apiErr)
if merr != nil {
logger.Error("failed to marshal error response", "response", fmt.Sprintf("%#v", apiErr), "error", merr)
w.WriteHeader(http.StatusInternalServerError)
if _, err := io.WriteString(w, errorFallback); err != nil {
logger.Error("failed to write response", "error", err)
}
return
}
w.Header().Set("Content-Type", marshaler.ContentType())
w.WriteHeader(int(apiErr.GetStatus()))
if _, err := w.Write(buf); err != nil {
logger.Error("failed to send response chunk", "error", err)
return
}
}
}