12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879 |
- /*
- Copyright 2019 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 (
- "k8s.io/api/core/v1"
- "k8s.io/apimachinery/pkg/fields"
- "k8s.io/apimachinery/pkg/labels"
- v1helper "k8s.io/kubernetes/pkg/apis/core/v1/helper"
- )
- // PodMatchesNodeSelectorAndAffinityTerms checks whether the pod is schedulable onto nodes according to
- // the requirements in both NodeAffinity and nodeSelector.
- func PodMatchesNodeSelectorAndAffinityTerms(pod *v1.Pod, node *v1.Node) bool {
- // Check if node.Labels match pod.Spec.NodeSelector.
- if len(pod.Spec.NodeSelector) > 0 {
- selector := labels.SelectorFromSet(pod.Spec.NodeSelector)
- if !selector.Matches(labels.Set(node.Labels)) {
- return false
- }
- }
- // 1. nil NodeSelector matches all nodes (i.e. does not filter out any nodes)
- // 2. nil []NodeSelectorTerm (equivalent to non-nil empty NodeSelector) matches no nodes
- // 3. zero-length non-nil []NodeSelectorTerm matches no nodes also, just for simplicity
- // 4. nil []NodeSelectorRequirement (equivalent to non-nil empty NodeSelectorTerm) matches no nodes
- // 5. zero-length non-nil []NodeSelectorRequirement matches no nodes also, just for simplicity
- // 6. non-nil empty NodeSelectorRequirement is not allowed
- nodeAffinityMatches := true
- affinity := pod.Spec.Affinity
- if affinity != nil && affinity.NodeAffinity != nil {
- nodeAffinity := affinity.NodeAffinity
- // if no required NodeAffinity requirements, will do no-op, means select all nodes.
- // TODO: Replace next line with subsequent commented-out line when implement RequiredDuringSchedulingRequiredDuringExecution.
- if nodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution == nil {
- // if nodeAffinity.RequiredDuringSchedulingRequiredDuringExecution == nil && nodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution == nil {
- return true
- }
- // Match node selector for requiredDuringSchedulingRequiredDuringExecution.
- // TODO: Uncomment this block when implement RequiredDuringSchedulingRequiredDuringExecution.
- // if nodeAffinity.RequiredDuringSchedulingRequiredDuringExecution != nil {
- // nodeSelectorTerms := nodeAffinity.RequiredDuringSchedulingRequiredDuringExecution.NodeSelectorTerms
- // klog.V(10).Infof("Match for RequiredDuringSchedulingRequiredDuringExecution node selector terms %+v", nodeSelectorTerms)
- // nodeAffinityMatches = nodeMatchesNodeSelectorTerms(node, nodeSelectorTerms)
- // }
- // Match node selector for requiredDuringSchedulingIgnoredDuringExecution.
- if nodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution != nil {
- nodeSelectorTerms := nodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms
- nodeAffinityMatches = nodeAffinityMatches && nodeMatchesNodeSelectorTerms(node, nodeSelectorTerms)
- }
- }
- return nodeAffinityMatches
- }
- // nodeMatchesNodeSelectorTerms checks if a node's labels satisfy a list of node selector terms,
- // terms are ORed, and an empty list of terms will match nothing.
- func nodeMatchesNodeSelectorTerms(node *v1.Node, nodeSelectorTerms []v1.NodeSelectorTerm) bool {
- return v1helper.MatchNodeSelectorTerms(nodeSelectorTerms, node.Labels, fields.Set{
- "metadata.name": node.Name,
- })
- }
|