Skip to content

Commit c25f5ee

Browse files
authored
Merge pull request kubernetes#128407 from ndixita/pod-level-resources
[PodLevelResources] Pod Level Resources Feature Alpha
2 parents 45260fd + b30e6c8 commit c25f5ee

File tree

126 files changed

+6364
-1299
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

126 files changed

+6364
-1299
lines changed

api/openapi-spec/swagger.json

+4
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

api/openapi-spec/v3/api__v1_openapi.json

+8
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

api/openapi-spec/v3/apis__apps__v1_openapi.json

+8
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

api/openapi-spec/v3/apis__batch__v1_openapi.json

+8
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

pkg/api/pod/testing/make.go

+6
Original file line numberDiff line numberDiff line change
@@ -72,6 +72,12 @@ func SetResourceVersion(rv string) Tweak {
7272
}
7373
}
7474

75+
func SetPodResources(resources *api.ResourceRequirements) Tweak {
76+
return func(pod *api.Pod) {
77+
pod.Spec.Resources = resources
78+
}
79+
}
80+
7581
func SetContainers(containers ...api.Container) Tweak {
7682
return func(pod *api.Pod) {
7783
pod.Spec.Containers = containers

pkg/api/pod/util.go

+32
Original file line numberDiff line numberDiff line change
@@ -384,6 +384,7 @@ func GetValidationOptionsFromPodSpecAndMeta(podSpec, oldPodSpec *api.PodSpec, po
384384
AllowNamespacedSysctlsForHostNetAndHostIPC: false,
385385
AllowNonLocalProjectedTokenPath: false,
386386
AllowPodLifecycleSleepActionZeroValue: utilfeature.DefaultFeatureGate.Enabled(features.PodLifecycleSleepActionAllowZero),
387+
PodLevelResourcesEnabled: utilfeature.DefaultFeatureGate.Enabled(features.PodLevelResources),
387388
}
388389

389390
// If old spec uses relaxed validation or enabled the RelaxedEnvironmentVariableValidation feature gate,
@@ -621,6 +622,7 @@ func dropDisabledFields(
621622
}
622623
}
623624

625+
dropDisabledPodLevelResources(podSpec, oldPodSpec)
624626
dropDisabledProcMountField(podSpec, oldPodSpec)
625627

626628
dropDisabledNodeInclusionPolicyFields(podSpec, oldPodSpec)
@@ -674,6 +676,14 @@ func dropDisabledFields(
674676
dropSELinuxChangePolicy(podSpec, oldPodSpec)
675677
}
676678

679+
func dropDisabledPodLevelResources(podSpec, oldPodSpec *api.PodSpec) {
680+
// If the feature is disabled and not in use, drop Resources at the pod-level
681+
// from PodSpec.
682+
if !utilfeature.DefaultFeatureGate.Enabled(features.PodLevelResources) && !podLevelResourcesInUse(oldPodSpec) {
683+
podSpec.Resources = nil
684+
}
685+
}
686+
677687
func dropPodLifecycleSleepAction(podSpec, oldPodSpec *api.PodSpec) {
678688
if utilfeature.DefaultFeatureGate.Enabled(features.PodLifecycleSleepAction) || podLifecycleSleepActionInUse(oldPodSpec) {
679689
return
@@ -1050,6 +1060,28 @@ func supplementalGroupsPolicyInUse(podSpec *api.PodSpec) bool {
10501060
return false
10511061
}
10521062

1063+
// podLevelResourcesInUse returns true if pod-spec is non-nil and Resources field at
1064+
// pod-level has non-empty Requests or Limits.
1065+
func podLevelResourcesInUse(podSpec *api.PodSpec) bool {
1066+
if podSpec == nil {
1067+
return false
1068+
}
1069+
1070+
if podSpec.Resources == nil {
1071+
return false
1072+
}
1073+
1074+
if len(podSpec.Resources.Requests) > 0 {
1075+
return true
1076+
}
1077+
1078+
if len(podSpec.Resources.Limits) > 0 {
1079+
return true
1080+
}
1081+
1082+
return false
1083+
}
1084+
10531085
// inPlacePodVerticalScalingInUse returns true if pod spec is non-nil and ResizePolicy is set
10541086
func inPlacePodVerticalScalingInUse(podSpec *api.PodSpec) bool {
10551087
if podSpec == nil {

pkg/api/pod/util_test.go

+143
Original file line numberDiff line numberDiff line change
@@ -2703,6 +2703,149 @@ func TestDropInPlacePodVerticalScaling(t *testing.T) {
27032703
}
27042704
}
27052705

2706+
func TestDropPodLevelResources(t *testing.T) {
2707+
containers := []api.Container{
2708+
{
2709+
Name: "c1",
2710+
Image: "image",
2711+
Resources: api.ResourceRequirements{
2712+
Requests: api.ResourceList{api.ResourceCPU: resource.MustParse("100m")},
2713+
Limits: api.ResourceList{api.ResourceCPU: resource.MustParse("200m")},
2714+
},
2715+
},
2716+
}
2717+
podWithPodLevelResources := func() *api.Pod {
2718+
return &api.Pod{
2719+
Spec: api.PodSpec{
2720+
Resources: &api.ResourceRequirements{
2721+
Requests: api.ResourceList{
2722+
api.ResourceCPU: resource.MustParse("100m"),
2723+
api.ResourceMemory: resource.MustParse("50Gi"),
2724+
},
2725+
Limits: api.ResourceList{
2726+
api.ResourceCPU: resource.MustParse("100m"),
2727+
api.ResourceMemory: resource.MustParse("50Gi"),
2728+
},
2729+
},
2730+
Containers: containers,
2731+
},
2732+
}
2733+
}
2734+
2735+
podWithoutPodLevelResources := func() *api.Pod {
2736+
return &api.Pod{
2737+
Spec: api.PodSpec{
2738+
Containers: containers,
2739+
},
2740+
}
2741+
}
2742+
2743+
podInfo := []struct {
2744+
description string
2745+
hasPodLevelResources bool
2746+
pod func() *api.Pod
2747+
}{
2748+
{
2749+
description: "has pod-level resources",
2750+
hasPodLevelResources: true,
2751+
pod: podWithPodLevelResources,
2752+
},
2753+
{
2754+
description: "does not have pod-level resources",
2755+
hasPodLevelResources: false,
2756+
pod: podWithoutPodLevelResources,
2757+
},
2758+
{
2759+
description: "is nil",
2760+
hasPodLevelResources: false,
2761+
pod: func() *api.Pod { return nil },
2762+
},
2763+
{
2764+
description: "is empty struct",
2765+
hasPodLevelResources: false,
2766+
// refactor to generalize and use podWithPodLevelResources()
2767+
pod: func() *api.Pod {
2768+
return &api.Pod{
2769+
Spec: api.PodSpec{
2770+
Resources: &api.ResourceRequirements{},
2771+
Containers: containers,
2772+
},
2773+
}
2774+
},
2775+
},
2776+
{
2777+
description: "is empty Requests list",
2778+
hasPodLevelResources: false,
2779+
pod: func() *api.Pod {
2780+
return &api.Pod{
2781+
Spec: api.PodSpec{Resources: &api.ResourceRequirements{
2782+
Requests: api.ResourceList{},
2783+
}}}
2784+
},
2785+
},
2786+
{
2787+
description: "is empty Limits list",
2788+
hasPodLevelResources: false,
2789+
pod: func() *api.Pod {
2790+
return &api.Pod{
2791+
Spec: api.PodSpec{Resources: &api.ResourceRequirements{
2792+
Limits: api.ResourceList{},
2793+
}}}
2794+
},
2795+
},
2796+
}
2797+
2798+
for _, enabled := range []bool{true, false} {
2799+
for _, oldPodInfo := range podInfo {
2800+
for _, newPodInfo := range podInfo {
2801+
oldPodHasPodLevelResources, oldPod := oldPodInfo.hasPodLevelResources, oldPodInfo.pod()
2802+
newPodHasPodLevelResources, newPod := newPodInfo.hasPodLevelResources, newPodInfo.pod()
2803+
if newPod == nil {
2804+
continue
2805+
}
2806+
2807+
t.Run(fmt.Sprintf("feature enabled=%v, old pod %v, new pod %v", enabled, oldPodInfo.description, newPodInfo.description), func(t *testing.T) {
2808+
featuregatetesting.SetFeatureGateDuringTest(t, utilfeature.DefaultFeatureGate, features.PodLevelResources, enabled)
2809+
2810+
var oldPodSpec *api.PodSpec
2811+
if oldPod != nil {
2812+
oldPodSpec = &oldPod.Spec
2813+
}
2814+
2815+
dropDisabledFields(&newPod.Spec, nil, oldPodSpec, nil)
2816+
2817+
// old pod should never be changed
2818+
if !reflect.DeepEqual(oldPod, oldPodInfo.pod()) {
2819+
t.Errorf("old pod changed: %v", cmp.Diff(oldPod, oldPodInfo.pod()))
2820+
}
2821+
2822+
switch {
2823+
case enabled || oldPodHasPodLevelResources:
2824+
// new pod shouldn't change if feature enabled or if old pod has
2825+
// any pod level resources
2826+
if !reflect.DeepEqual(newPod, newPodInfo.pod()) {
2827+
t.Errorf("new pod changed: %v", cmp.Diff(newPod, newPodInfo.pod()))
2828+
}
2829+
case newPodHasPodLevelResources:
2830+
// new pod should be changed
2831+
if reflect.DeepEqual(newPod, newPodInfo.pod()) {
2832+
t.Errorf("new pod was not changed")
2833+
}
2834+
// new pod should not have any pod-level resources
2835+
if !reflect.DeepEqual(newPod, podWithoutPodLevelResources()) {
2836+
t.Errorf("new pod has pod-level resources: %v", cmp.Diff(newPod, podWithoutPodLevelResources()))
2837+
}
2838+
default:
2839+
if newPod.Spec.Resources != nil {
2840+
t.Errorf("expected nil, got: %v", newPod.Spec.Resources)
2841+
}
2842+
}
2843+
})
2844+
}
2845+
}
2846+
}
2847+
}
2848+
27062849
func TestDropSidecarContainers(t *testing.T) {
27072850
containerRestartPolicyAlways := api.ContainerRestartPolicyAlways
27082851

pkg/apis/apps/v1/zz_generated.defaults.go

+16
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

pkg/apis/apps/v1beta1/zz_generated.defaults.go

+8
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

pkg/apis/apps/v1beta2/zz_generated.defaults.go

+16
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)