251 lines
6.9 KiB
Go
251 lines
6.9 KiB
Go
/*
|
|
Copyright 2015 The Kubernetes 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 storage
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
|
|
apierrors "k8s.io/apimachinery/pkg/api/errors"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/apimachinery/pkg/util/validation/field"
|
|
)
|
|
|
|
var (
|
|
ErrResourceVersionSetOnCreate = errors.New("resourceVersion should not be set on objects to be created")
|
|
ErrStorageNotReady = errors.New("storage not ready")
|
|
)
|
|
|
|
const (
|
|
ErrCodeKeyNotFound int = iota + 1
|
|
ErrCodeKeyExists
|
|
ErrCodeResourceVersionConflicts
|
|
ErrCodeInvalidObj
|
|
ErrCodeUnreachable
|
|
ErrCodeTimeout
|
|
ErrCodeCorruptObj
|
|
)
|
|
|
|
var errCodeToMessage = map[int]string{
|
|
ErrCodeKeyNotFound: "key not found",
|
|
ErrCodeKeyExists: "key exists",
|
|
ErrCodeResourceVersionConflicts: "resource version conflicts",
|
|
ErrCodeInvalidObj: "invalid object",
|
|
ErrCodeUnreachable: "server unreachable",
|
|
ErrCodeTimeout: "request timeout",
|
|
ErrCodeCorruptObj: "corrupt object",
|
|
}
|
|
|
|
func NewKeyNotFoundError(key string, rv int64) *StorageError {
|
|
return &StorageError{
|
|
Code: ErrCodeKeyNotFound,
|
|
Key: key,
|
|
ResourceVersion: rv,
|
|
}
|
|
}
|
|
|
|
func NewKeyExistsError(key string, rv int64) *StorageError {
|
|
return &StorageError{
|
|
Code: ErrCodeKeyExists,
|
|
Key: key,
|
|
ResourceVersion: rv,
|
|
}
|
|
}
|
|
|
|
func NewResourceVersionConflictsError(key string, rv int64) *StorageError {
|
|
return &StorageError{
|
|
Code: ErrCodeResourceVersionConflicts,
|
|
Key: key,
|
|
ResourceVersion: rv,
|
|
}
|
|
}
|
|
|
|
func NewUnreachableError(key string, rv int64) *StorageError {
|
|
return &StorageError{
|
|
Code: ErrCodeUnreachable,
|
|
Key: key,
|
|
ResourceVersion: rv,
|
|
}
|
|
}
|
|
|
|
func NewTimeoutError(key, msg string) *StorageError {
|
|
return &StorageError{
|
|
Code: ErrCodeTimeout,
|
|
Key: key,
|
|
err: errors.New(msg),
|
|
}
|
|
}
|
|
|
|
func NewInvalidObjError(key, msg string) *StorageError {
|
|
return &StorageError{
|
|
Code: ErrCodeInvalidObj,
|
|
Key: key,
|
|
err: errors.New(msg),
|
|
}
|
|
}
|
|
|
|
// NewCorruptObjError returns a new StorageError, it represents a corrupt object:
|
|
// a) object data retrieved from the storage failed to transform with the given err.
|
|
// b) the given object failed to decode with the given err
|
|
func NewCorruptObjError(key string, err error) *StorageError {
|
|
return &StorageError{
|
|
Code: ErrCodeCorruptObj,
|
|
Key: key,
|
|
err: err,
|
|
}
|
|
}
|
|
|
|
type StorageError struct {
|
|
Code int
|
|
Key string
|
|
ResourceVersion int64
|
|
|
|
// inner error
|
|
err error
|
|
}
|
|
|
|
func (e *StorageError) Unwrap() error { return e.err }
|
|
|
|
func (e *StorageError) Error() string {
|
|
return fmt.Sprintf("StorageError: %s, Code: %d, Key: %s, ResourceVersion: %d, AdditionalErrorMsg: %v",
|
|
errCodeToMessage[e.Code], e.Code, e.Key, e.ResourceVersion, e.err)
|
|
}
|
|
|
|
// IsNotFound returns true if and only if err is "key" not found error.
|
|
func IsNotFound(err error) bool {
|
|
return isErrCode(err, ErrCodeKeyNotFound)
|
|
}
|
|
|
|
// IsExist returns true if and only if err is "key" already exists error.
|
|
func IsExist(err error) bool {
|
|
return isErrCode(err, ErrCodeKeyExists)
|
|
}
|
|
|
|
// IsUnreachable returns true if and only if err indicates the server could not be reached.
|
|
func IsUnreachable(err error) bool {
|
|
return isErrCode(err, ErrCodeUnreachable)
|
|
}
|
|
|
|
// IsConflict returns true if and only if err is a write conflict.
|
|
func IsConflict(err error) bool {
|
|
return isErrCode(err, ErrCodeResourceVersionConflicts)
|
|
}
|
|
|
|
// IsRequestTimeout returns true if and only if err indicates that the request has timed out.
|
|
func IsRequestTimeout(err error) bool {
|
|
return isErrCode(err, ErrCodeTimeout)
|
|
}
|
|
|
|
// IsInvalidObj returns true if and only if err is invalid error
|
|
func IsInvalidObj(err error) bool {
|
|
return isErrCode(err, ErrCodeInvalidObj)
|
|
}
|
|
|
|
// IsCorruptObject returns true if and only if:
|
|
// a) the given object data retrieved from the storage is not transformable, or
|
|
// b) the given object failed to decode properly
|
|
func IsCorruptObject(err error) bool {
|
|
if err == nil {
|
|
return false
|
|
}
|
|
var storageErr *StorageError
|
|
if !errors.As(err, &storageErr) {
|
|
return false
|
|
}
|
|
|
|
return storageErr.Code == ErrCodeCorruptObj
|
|
}
|
|
|
|
func isErrCode(err error, code int) bool {
|
|
if err == nil {
|
|
return false
|
|
}
|
|
if e, ok := err.(*StorageError); ok {
|
|
return e.Code == code
|
|
}
|
|
return false
|
|
}
|
|
|
|
// InvalidError is generated when an error caused by invalid API object occurs
|
|
// in the storage package.
|
|
type InvalidError struct {
|
|
Errs field.ErrorList
|
|
}
|
|
|
|
func (e InvalidError) Error() string {
|
|
return e.Errs.ToAggregate().Error()
|
|
}
|
|
|
|
// IsInvalidError returns true if and only if err is an InvalidError.
|
|
func IsInvalidError(err error) bool {
|
|
_, ok := err.(InvalidError)
|
|
return ok
|
|
}
|
|
|
|
func NewInvalidError(errors field.ErrorList) InvalidError {
|
|
return InvalidError{errors}
|
|
}
|
|
|
|
// InternalError is generated when an error occurs in the storage package, i.e.,
|
|
// not from the underlying storage backend (e.g., etcd).
|
|
type InternalError struct {
|
|
Reason string
|
|
|
|
// retain the inner error to maintain the error tree, so as to enable us
|
|
// to do proper error checking, but we also need to be backward compatible.
|
|
err error
|
|
}
|
|
|
|
func (e InternalError) Error() string {
|
|
return e.Reason
|
|
}
|
|
|
|
func (e InternalError) Unwrap() error { return e.err }
|
|
|
|
// IsInternalError returns true if and only if err is an InternalError.
|
|
func IsInternalError(err error) bool {
|
|
_, ok := err.(InternalError)
|
|
return ok
|
|
}
|
|
|
|
func NewInternalError(err error) InternalError {
|
|
return InternalError{Reason: err.Error(), err: err}
|
|
}
|
|
|
|
var tooLargeResourceVersionCauseMsg = "Too large resource version"
|
|
|
|
// NewTooLargeResourceVersionError returns a timeout error with the given retrySeconds for a request for
|
|
// a minimum resource version that is larger than the largest currently available resource version for a requested resource.
|
|
func NewTooLargeResourceVersionError(minimumResourceVersion, currentRevision uint64, retrySeconds int) error {
|
|
err := apierrors.NewTimeoutError(fmt.Sprintf("Too large resource version: %d, current: %d", minimumResourceVersion, currentRevision), retrySeconds)
|
|
err.ErrStatus.Details.Causes = []metav1.StatusCause{
|
|
{
|
|
Type: metav1.CauseTypeResourceVersionTooLarge,
|
|
Message: tooLargeResourceVersionCauseMsg,
|
|
},
|
|
}
|
|
return err
|
|
}
|
|
|
|
// IsTooLargeResourceVersion returns true if the error is a TooLargeResourceVersion error.
|
|
func IsTooLargeResourceVersion(err error) bool {
|
|
if !apierrors.IsTimeout(err) {
|
|
return false
|
|
}
|
|
return apierrors.HasStatusCause(err, metav1.CauseTypeResourceVersionTooLarge)
|
|
}
|