2019-01-12 04:58:27 +00:00
|
|
|
/*
|
|
|
|
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 (
|
|
|
|
"fmt"
|
|
|
|
|
2019-12-12 01:27:03 +00:00
|
|
|
"k8s.io/apimachinery/pkg/api/errors"
|
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
2019-01-12 04:58:27 +00:00
|
|
|
"k8s.io/apimachinery/pkg/util/validation/field"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
ErrCodeKeyNotFound int = iota + 1
|
|
|
|
ErrCodeKeyExists
|
|
|
|
ErrCodeResourceVersionConflicts
|
|
|
|
ErrCodeInvalidObj
|
|
|
|
ErrCodeUnreachable
|
|
|
|
)
|
|
|
|
|
|
|
|
var errCodeToMessage = map[int]string{
|
|
|
|
ErrCodeKeyNotFound: "key not found",
|
|
|
|
ErrCodeKeyExists: "key exists",
|
|
|
|
ErrCodeResourceVersionConflicts: "resource version conflicts",
|
|
|
|
ErrCodeInvalidObj: "invalid object",
|
|
|
|
ErrCodeUnreachable: "server unreachable",
|
|
|
|
}
|
|
|
|
|
|
|
|
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 NewInvalidObjError(key, msg string) *StorageError {
|
|
|
|
return &StorageError{
|
|
|
|
Code: ErrCodeInvalidObj,
|
|
|
|
Key: key,
|
|
|
|
AdditionalErrorMsg: msg,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type StorageError struct {
|
|
|
|
Code int
|
|
|
|
Key string
|
|
|
|
ResourceVersion int64
|
|
|
|
AdditionalErrorMsg string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *StorageError) Error() string {
|
|
|
|
return fmt.Sprintf("StorageError: %s, Code: %d, Key: %s, ResourceVersion: %d, AdditionalErrorMsg: %s",
|
|
|
|
errCodeToMessage[e.Code], e.Code, e.Key, e.ResourceVersion, e.AdditionalErrorMsg)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsNotFound returns true if and only if err is "key" not found error.
|
|
|
|
func IsNotFound(err error) bool {
|
|
|
|
return isErrCode(err, ErrCodeKeyNotFound)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsNodeExist returns true if and only if err is an node already exist error.
|
|
|
|
func IsNodeExist(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)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsInvalidObj returns true if and only if err is invalid error
|
|
|
|
func IsInvalidObj(err error) bool {
|
|
|
|
return isErrCode(err, ErrCodeInvalidObj)
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e InternalError) Error() string {
|
|
|
|
return e.Reason
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsInternalError returns true if and only if err is an InternalError.
|
|
|
|
func IsInternalError(err error) bool {
|
|
|
|
_, ok := err.(InternalError)
|
|
|
|
return ok
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewInternalError(reason string) InternalError {
|
|
|
|
return InternalError{reason}
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewInternalErrorf(format string, a ...interface{}) InternalError {
|
|
|
|
return InternalError{fmt.Sprintf(format, a...)}
|
|
|
|
}
|
2019-12-12 01:27:03 +00:00
|
|
|
|
|
|
|
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 := errors.NewTimeoutError(fmt.Sprintf("Too large resource version: %d, current: %d", minimumResourceVersion, currentRevision), retrySeconds)
|
|
|
|
err.ErrStatus.Details.Causes = []metav1.StatusCause{{Message: tooLargeResourceVersionCauseMsg}}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsTooLargeResourceVersion returns true if the error is a TooLargeResourceVersion error.
|
|
|
|
func IsTooLargeResourceVersion(err error) bool {
|
|
|
|
if !errors.IsTimeout(err) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
switch t := err.(type) {
|
|
|
|
case errors.APIStatus:
|
|
|
|
if d := t.Status().Details; d != nil {
|
|
|
|
for _, cause := range d.Causes {
|
|
|
|
if cause.Message == tooLargeResourceVersionCauseMsg {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|