/* Copyright 2014 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 helper import ( "encoding/json" "fmt" "strconv" "strings" "k8s.io/apimachinery/pkg/api/resource" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/conversion" "k8s.io/apimachinery/pkg/fields" "k8s.io/apimachinery/pkg/labels" "k8s.io/apimachinery/pkg/util/sets" "k8s.io/apimachinery/pkg/util/validation" "k8s.io/kubernetes/pkg/apis/core" ) // IsHugePageResourceName returns true if the resource name has the huge page // resource prefix. func IsHugePageResourceName(name core.ResourceName) bool { return strings.HasPrefix(string(name), core.ResourceHugePagesPrefix) } // IsHugePageResourceValueDivisible returns true if the resource value of storage is // integer multiple of page size. func IsHugePageResourceValueDivisible(name core.ResourceName, quantity resource.Quantity) bool { pageSize, err := HugePageSizeFromResourceName(name) if err != nil { return false } if pageSize.Sign() <= 0 || pageSize.MilliValue()%int64(1000) != int64(0) { return false } return quantity.Value()%pageSize.Value() == 0 } // IsQuotaHugePageResourceName returns true if the resource name has the quota // related huge page resource prefix. func IsQuotaHugePageResourceName(name core.ResourceName) bool { return strings.HasPrefix(string(name), core.ResourceHugePagesPrefix) || strings.HasPrefix(string(name), core.ResourceRequestsHugePagesPrefix) } // HugePageResourceName returns a ResourceName with the canonical hugepage // prefix prepended for the specified page size. The page size is converted // to its canonical representation. func HugePageResourceName(pageSize resource.Quantity) core.ResourceName { return core.ResourceName(fmt.Sprintf("%s%s", core.ResourceHugePagesPrefix, pageSize.String())) } // HugePageSizeFromResourceName returns the page size for the specified huge page // resource name. If the specified input is not a valid huge page resource name // an error is returned. func HugePageSizeFromResourceName(name core.ResourceName) (resource.Quantity, error) { if !IsHugePageResourceName(name) { return resource.Quantity{}, fmt.Errorf("resource name: %s is an invalid hugepage name", name) } pageSize := strings.TrimPrefix(string(name), core.ResourceHugePagesPrefix) return resource.ParseQuantity(pageSize) } // NonConvertibleFields iterates over the provided map and filters out all but // any keys with the "non-convertible.kubernetes.io" prefix. func NonConvertibleFields(annotations map[string]string) map[string]string { nonConvertibleKeys := map[string]string{} for key, value := range annotations { if strings.HasPrefix(key, core.NonConvertibleAnnotationPrefix) { nonConvertibleKeys[key] = value } } return nonConvertibleKeys } // Semantic can do semantic deep equality checks for core objects. // Example: apiequality.Semantic.DeepEqual(aPod, aPodWithNonNilButEmptyMaps) == true var Semantic = conversion.EqualitiesOrDie( func(a, b resource.Quantity) bool { // Ignore formatting, only care that numeric value stayed the same. // TODO: if we decide it's important, it should be safe to start comparing the format. // // Uninitialized quantities are equivalent to 0 quantities. return a.Cmp(b) == 0 }, func(a, b metav1.MicroTime) bool { return a.UTC() == b.UTC() }, func(a, b metav1.Time) bool { return a.UTC() == b.UTC() }, func(a, b labels.Selector) bool { return a.String() == b.String() }, func(a, b fields.Selector) bool { return a.String() == b.String() }, ) var standardResourceQuotaScopes = sets.New( core.ResourceQuotaScopeTerminating, core.ResourceQuotaScopeNotTerminating, core.ResourceQuotaScopeBestEffort, core.ResourceQuotaScopeNotBestEffort, core.ResourceQuotaScopePriorityClass, ) // IsStandardResourceQuotaScope returns true if the scope is a standard value func IsStandardResourceQuotaScope(scope core.ResourceQuotaScope) bool { return standardResourceQuotaScopes.Has(scope) || scope == core.ResourceQuotaScopeCrossNamespacePodAffinity } var podObjectCountQuotaResources = sets.New( core.ResourcePods, ) var podComputeQuotaResources = sets.New( core.ResourceCPU, core.ResourceMemory, core.ResourceLimitsCPU, core.ResourceLimitsMemory, core.ResourceRequestsCPU, core.ResourceRequestsMemory, ) // IsResourceQuotaScopeValidForResource returns true if the resource applies to the specified scope func IsResourceQuotaScopeValidForResource(scope core.ResourceQuotaScope, resource core.ResourceName) bool { switch scope { case core.ResourceQuotaScopeTerminating, core.ResourceQuotaScopeNotTerminating, core.ResourceQuotaScopeNotBestEffort, core.ResourceQuotaScopePriorityClass, core.ResourceQuotaScopeCrossNamespacePodAffinity: return podObjectCountQuotaResources.Has(resource) || podComputeQuotaResources.Has(resource) case core.ResourceQuotaScopeBestEffort: return podObjectCountQuotaResources.Has(resource) default: return true } } var standardContainerResources = sets.New( core.ResourceCPU, core.ResourceMemory, core.ResourceEphemeralStorage, ) // IsStandardContainerResourceName returns true if the container can make a resource request // for the specified resource func IsStandardContainerResourceName(name core.ResourceName) bool { return standardContainerResources.Has(name) || IsHugePageResourceName(name) } // IsExtendedResourceName returns true if: // 1. the resource name is not in the default namespace; // 2. resource name does not have "requests." prefix, // to avoid confusion with the convention in quota // 3. it satisfies the rules in IsQualifiedName() after converted into quota resource name func IsExtendedResourceName(name core.ResourceName) bool { if IsNativeResource(name) || strings.HasPrefix(string(name), core.DefaultResourceRequestsPrefix) { return false } // Ensure it satisfies the rules in IsQualifiedName() after converted into quota resource name nameForQuota := fmt.Sprintf("%s%s", core.DefaultResourceRequestsPrefix, string(name)) if errs := validation.IsQualifiedName(nameForQuota); len(errs) != 0 { return false } return true } // IsNativeResource returns true if the resource name is in the // *kubernetes.io/ namespace. Partially-qualified (unprefixed) names are // implicitly in the kubernetes.io/ namespace. func IsNativeResource(name core.ResourceName) bool { return !strings.Contains(string(name), "/") || strings.Contains(string(name), core.ResourceDefaultNamespacePrefix) } // IsOvercommitAllowed returns true if the resource is in the default // namespace and is not hugepages. func IsOvercommitAllowed(name core.ResourceName) bool { return IsNativeResource(name) && !IsHugePageResourceName(name) } var standardLimitRangeTypes = sets.New( core.LimitTypePod, core.LimitTypeContainer, core.LimitTypePersistentVolumeClaim, ) // IsStandardLimitRangeType returns true if the type is Pod or Container func IsStandardLimitRangeType(value core.LimitType) bool { return standardLimitRangeTypes.Has(value) } var standardQuotaResources = sets.New( core.ResourceCPU, core.ResourceMemory, core.ResourceEphemeralStorage, core.ResourceRequestsCPU, core.ResourceRequestsMemory, core.ResourceRequestsStorage, core.ResourceRequestsEphemeralStorage, core.ResourceLimitsCPU, core.ResourceLimitsMemory, core.ResourceLimitsEphemeralStorage, core.ResourcePods, core.ResourceQuotas, core.ResourceServices, core.ResourceReplicationControllers, core.ResourceSecrets, core.ResourcePersistentVolumeClaims, core.ResourceConfigMaps, core.ResourceServicesNodePorts, core.ResourceServicesLoadBalancers, ) // IsStandardQuotaResourceName returns true if the resource is known to // the quota tracking system func IsStandardQuotaResourceName(name core.ResourceName) bool { return standardQuotaResources.Has(name) || IsQuotaHugePageResourceName(name) } var standardResources = sets.New( core.ResourceCPU, core.ResourceMemory, core.ResourceEphemeralStorage, core.ResourceRequestsCPU, core.ResourceRequestsMemory, core.ResourceRequestsEphemeralStorage, core.ResourceLimitsCPU, core.ResourceLimitsMemory, core.ResourceLimitsEphemeralStorage, core.ResourcePods, core.ResourceQuotas, core.ResourceServices, core.ResourceReplicationControllers, core.ResourceSecrets, core.ResourceConfigMaps, core.ResourcePersistentVolumeClaims, core.ResourceStorage, core.ResourceRequestsStorage, core.ResourceServicesNodePorts, core.ResourceServicesLoadBalancers, ) // IsStandardResourceName returns true if the resource is known to the system func IsStandardResourceName(name core.ResourceName) bool { return standardResources.Has(name) || IsQuotaHugePageResourceName(name) } var integerResources = sets.New( core.ResourcePods, core.ResourceQuotas, core.ResourceServices, core.ResourceReplicationControllers, core.ResourceSecrets, core.ResourceConfigMaps, core.ResourcePersistentVolumeClaims, core.ResourceServicesNodePorts, core.ResourceServicesLoadBalancers, ) // IsIntegerResourceName returns true if the resource is measured in integer values func IsIntegerResourceName(name core.ResourceName) bool { return integerResources.Has(name) || IsExtendedResourceName(name) } // IsServiceIPSet aims to check if the service's ClusterIP is set or not // the objective is not to perform validation here func IsServiceIPSet(service *core.Service) bool { // This function assumes that the service is semantically validated // it does not test if the IP is valid, just makes sure that it is set. return len(service.Spec.ClusterIP) > 0 && service.Spec.ClusterIP != core.ClusterIPNone } var standardFinalizers = sets.New( string(core.FinalizerKubernetes), metav1.FinalizerOrphanDependents, metav1.FinalizerDeleteDependents, ) // IsStandardFinalizerName checks if the input string is a standard finalizer name func IsStandardFinalizerName(str string) bool { return standardFinalizers.Has(str) } // GetAccessModesAsString returns a string representation of an array of access modes. // modes, when present, are always in the same order: RWO,ROX,RWX,RWOP. func GetAccessModesAsString(modes []core.PersistentVolumeAccessMode) string { modes = removeDuplicateAccessModes(modes) modesStr := []string{} if ContainsAccessMode(modes, core.ReadWriteOnce) { modesStr = append(modesStr, "RWO") } if ContainsAccessMode(modes, core.ReadOnlyMany) { modesStr = append(modesStr, "ROX") } if ContainsAccessMode(modes, core.ReadWriteMany) { modesStr = append(modesStr, "RWX") } if ContainsAccessMode(modes, core.ReadWriteOncePod) { modesStr = append(modesStr, "RWOP") } return strings.Join(modesStr, ",") } // GetAccessModesFromString returns an array of AccessModes from a string created by GetAccessModesAsString func GetAccessModesFromString(modes string) []core.PersistentVolumeAccessMode { strmodes := strings.Split(modes, ",") accessModes := []core.PersistentVolumeAccessMode{} for _, s := range strmodes { s = strings.Trim(s, " ") switch { case s == "RWO": accessModes = append(accessModes, core.ReadWriteOnce) case s == "ROX": accessModes = append(accessModes, core.ReadOnlyMany) case s == "RWX": accessModes = append(accessModes, core.ReadWriteMany) case s == "RWOP": accessModes = append(accessModes, core.ReadWriteOncePod) } } return accessModes } // removeDuplicateAccessModes returns an array of access modes without any duplicates func removeDuplicateAccessModes(modes []core.PersistentVolumeAccessMode) []core.PersistentVolumeAccessMode { accessModes := []core.PersistentVolumeAccessMode{} for _, m := range modes { if !ContainsAccessMode(accessModes, m) { accessModes = append(accessModes, m) } } return accessModes } func ContainsAccessMode(modes []core.PersistentVolumeAccessMode, mode core.PersistentVolumeAccessMode) bool { for _, m := range modes { if m == mode { return true } } return false } func ClaimContainsAllocatedResources(pvc *core.PersistentVolumeClaim) bool { if pvc == nil { return false } if pvc.Status.AllocatedResources != nil { return true } return false } func ClaimContainsAllocatedResourceStatus(pvc *core.PersistentVolumeClaim) bool { if pvc == nil { return false } if pvc.Status.AllocatedResourceStatuses != nil { return true } return false } // GetTolerationsFromPodAnnotations gets the json serialized tolerations data from Pod.Annotations // and converts it to the []Toleration type in core. func GetTolerationsFromPodAnnotations(annotations map[string]string) ([]core.Toleration, error) { var tolerations []core.Toleration if len(annotations) > 0 && annotations[core.TolerationsAnnotationKey] != "" { err := json.Unmarshal([]byte(annotations[core.TolerationsAnnotationKey]), &tolerations) if err != nil { return tolerations, err } } return tolerations, nil } // AddOrUpdateTolerationInPod tries to add a toleration to the pod's toleration list. // Returns true if something was updated, false otherwise. func AddOrUpdateTolerationInPod(pod *core.Pod, toleration *core.Toleration) bool { podTolerations := pod.Spec.Tolerations var newTolerations []core.Toleration updated := false for i := range podTolerations { if toleration.MatchToleration(&podTolerations[i]) { if Semantic.DeepEqual(toleration, podTolerations[i]) { return false } newTolerations = append(newTolerations, *toleration) updated = true continue } newTolerations = append(newTolerations, podTolerations[i]) } if !updated { newTolerations = append(newTolerations, *toleration) } pod.Spec.Tolerations = newTolerations return true } // GetTaintsFromNodeAnnotations gets the json serialized taints data from Pod.Annotations // and converts it to the []Taint type in core. func GetTaintsFromNodeAnnotations(annotations map[string]string) ([]core.Taint, error) { var taints []core.Taint if len(annotations) > 0 && annotations[core.TaintsAnnotationKey] != "" { err := json.Unmarshal([]byte(annotations[core.TaintsAnnotationKey]), &taints) if err != nil { return []core.Taint{}, err } } return taints, nil } // GetPersistentVolumeClass returns StorageClassName. func GetPersistentVolumeClass(volume *core.PersistentVolume) string { // Use beta annotation first if class, found := volume.Annotations[core.BetaStorageClassAnnotation]; found { return class } return volume.Spec.StorageClassName } // GetPersistentVolumeClaimClass returns StorageClassName. If no storage class was // requested, it returns "". func GetPersistentVolumeClaimClass(claim *core.PersistentVolumeClaim) string { // Use beta annotation first if class, found := claim.Annotations[core.BetaStorageClassAnnotation]; found { return class } if claim.Spec.StorageClassName != nil { return *claim.Spec.StorageClassName } return "" } // PersistentVolumeClaimHasClass returns true if given claim has set StorageClassName field. func PersistentVolumeClaimHasClass(claim *core.PersistentVolumeClaim) bool { // Use beta annotation first if _, found := claim.Annotations[core.BetaStorageClassAnnotation]; found { return true } if claim.Spec.StorageClassName != nil { return true } return false } // GetDeletionCostFromPodAnnotations returns the integer value of pod-deletion-cost. Returns 0 // if not set or the value is invalid. func GetDeletionCostFromPodAnnotations(annotations map[string]string) (int32, error) { if value, exist := annotations[core.PodDeletionCost]; exist { // values that start with plus sign (e.g, "+10") or leading zeros (e.g., "008") are not valid. if !validFirstDigit(value) { return 0, fmt.Errorf("invalid value %q", value) } i, err := strconv.ParseInt(value, 10, 32) if err != nil { // make sure we default to 0 on error. return 0, err } return int32(i), nil } return 0, nil } func validFirstDigit(str string) bool { if len(str) == 0 { return false } return str[0] == '-' || (str[0] == '0' && str == "0") || (str[0] >= '1' && str[0] <= '9') }