rclone/vendor/storj.io/common/rpc/rpcstatus/status.go
2020-05-12 15:56:50 +00:00

118 lines
2.5 KiB
Go

// Copyright (C) 2020 Storj Labs, Inc.
// See LICENSE for copying information.
// Package rpcstatus contains status code definitions for rpc.
package rpcstatus
import (
"context"
"fmt"
"github.com/zeebo/errs"
"storj.io/common/internal/grpchook"
"storj.io/drpc/drpcerr"
)
// StatusCode is an enumeration of rpc status codes.
type StatusCode uint64
// These constants are all the rpc error codes. It is important that
// their numerical values do not change.
const (
Unknown StatusCode = iota
OK
Canceled
InvalidArgument
DeadlineExceeded
NotFound
AlreadyExists
PermissionDenied
ResourceExhausted
FailedPrecondition
Aborted
OutOfRange
Unimplemented
Internal
Unavailable
DataLoss
Unauthenticated
)
// Code returns the status code associated with the error.
func Code(err error) StatusCode {
// special case: if the error is context canceled or deadline exceeded, the code
// must be those. additionally, grpc returns OK for a nil error, so we will, too.
switch err {
case nil:
return OK
case context.Canceled:
return Canceled
case context.DeadlineExceeded:
return DeadlineExceeded
default:
if code := StatusCode(drpcerr.Code(err)); code != Unknown {
return code
}
// If we have grpc attached try to get grpc code if possible.
if grpchook.HookedConvertToStatusCode != nil {
if code, ok := grpchook.HookedConvertToStatusCode(err); ok {
return StatusCode(code)
}
}
return Unknown
}
}
// Wrap wraps the error with the provided status code.
func Wrap(code StatusCode, err error) error {
if err == nil {
return nil
}
// Should we also handle grpc error status codes.
if grpchook.HookedErrorWrap != nil {
return grpchook.HookedErrorWrap(grpchook.StatusCode(code), err)
}
ce := &codeErr{
code: code,
}
if ee, ok := err.(errsError); ok {
ce.errsError = ee
} else {
ce.errsError = errs.Wrap(err).(errsError)
}
return ce
}
// Error wraps the message with a status code into an error.
func Error(code StatusCode, msg string) error {
return Wrap(code, errs.New("%s", msg))
}
// Errorf : Error :: fmt.Sprintf : fmt.Sprint
func Errorf(code StatusCode, format string, a ...interface{}) error {
return Wrap(code, errs.New(format, a...))
}
type errsError interface {
error
fmt.Formatter
Name() (string, bool)
}
// codeErr implements error that can work both in grpc and drpc.
type codeErr struct {
errsError
code StatusCode
}
func (c *codeErr) Unwrap() error { return c.errsError }
func (c *codeErr) Cause() error { return c.errsError }
func (c *codeErr) Code() uint64 { return uint64(c.code) }