
Now that we have most of the services required for use with containerd, it was found that common patterns were used throughout services. By defining a central `errdefs` package, we ensure that services will map errors to and from grpc consistently and cleanly. One can decorate an error with as much context as necessary, using `pkg/errors` and still have the error mapped correctly via grpc. We make a few sacrifices. At this point, the common errors we use across the repository all map directly to grpc error codes. While this seems positively crazy, it actually works out quite well. The error conditions that were specific weren't super necessary and the ones that were necessary now simply have better context information. We lose the ability to add new codes, but this constraint may not be a bad thing. Effectively, as long as one uses the errors defined in `errdefs`, the error class will be mapped correctly across the grpc boundary and everything will be good. If you don't use those definitions, the error maps to "unknown" and the error message is preserved. Signed-off-by: Stephen J Day <stephen.day@docker.com>
55 lines
1.9 KiB
Go
55 lines
1.9 KiB
Go
// Package errdefs defines the common errors used throughout containerd
|
|
// packages.
|
|
//
|
|
// Use with errors.Wrap and error.Wrapf to add context to an error.
|
|
//
|
|
// To detect an error class, use the IsXXX functions to tell whether an error
|
|
// is of a certain type.
|
|
//
|
|
// The functions ToGRPC and FromGRPC can be used to map server-side and
|
|
// client-side errors to the correct types.
|
|
package errdefs
|
|
|
|
import "github.com/pkg/errors"
|
|
|
|
// Definitions of common error types used throughout containerd. All containerd
|
|
// errors returned by most packages will map into one of these errors classes.
|
|
// Packages should return errors of these types when they want to instruct a
|
|
// client to take a particular action.
|
|
//
|
|
// For the most part, we just try to provide local grpc errors. Most conditions
|
|
// map very well to those defined by grpc.
|
|
var (
|
|
ErrUnknown = errors.New("unknown") // used internally to represent a missed mapping.
|
|
ErrInvalidArgument = errors.New("invalid")
|
|
ErrNotFound = errors.New("not found")
|
|
ErrAlreadyExists = errors.New("already exists")
|
|
ErrFailedPrecondition = errors.New("failed precondition")
|
|
ErrUnavailable = errors.New("unavailable")
|
|
)
|
|
|
|
func IsInvalidArgument(err error) bool {
|
|
return errors.Cause(err) == ErrInvalidArgument
|
|
}
|
|
|
|
// IsNotFound returns true if the error is due to a missing object
|
|
func IsNotFound(err error) bool {
|
|
return errors.Cause(err) == ErrNotFound
|
|
}
|
|
|
|
// IsAlreadyExists returns true if the error is due to an already existing
|
|
// metadata item
|
|
func IsAlreadyExists(err error) bool {
|
|
return errors.Cause(err) == ErrAlreadyExists
|
|
}
|
|
|
|
// IsFailedPrecondition returns true if an operation could not proceed to the
|
|
// lack of a particular condition.
|
|
func IsFailedPrecondition(err error) bool {
|
|
return errors.Cause(err) == ErrFailedPrecondition
|
|
}
|
|
|
|
func IsUnavailable(err error) bool {
|
|
return errors.Cause(err) == ErrUnavailable
|
|
}
|