Skip to content

Commit c961ac4

Browse files
committed
add subresources for custom resources
1 parent c61e25e commit c961ac4

File tree

28 files changed

+2428
-215
lines changed

28 files changed

+2428
-215
lines changed

cmd/kube-controller-manager/app/autoscaling.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@ import (
2424
apimeta "k8s.io/apimachinery/pkg/api/meta"
2525
"k8s.io/apimachinery/pkg/runtime/schema"
2626
"k8s.io/client-go/discovery"
27-
discocache "k8s.io/client-go/discovery/cached" // Saturday Night Fever
27+
discocache "k8s.io/client-go/discovery/cached"
2828
"k8s.io/client-go/dynamic"
2929
"k8s.io/client-go/scale"
3030
"k8s.io/kubernetes/pkg/controller/podautoscaler"

hack/.golint_failures

-1
Original file line numberDiff line numberDiff line change
@@ -477,7 +477,6 @@ staging/src/k8s.io/apiextensions-apiserver/pkg/cmd/server
477477
staging/src/k8s.io/apiextensions-apiserver/pkg/controller/finalizer
478478
staging/src/k8s.io/apiextensions-apiserver/pkg/controller/status
479479
staging/src/k8s.io/apiextensions-apiserver/pkg/features
480-
staging/src/k8s.io/apiextensions-apiserver/pkg/registry/customresource
481480
staging/src/k8s.io/apiextensions-apiserver/pkg/registry/customresourcedefinition
482481
staging/src/k8s.io/apiextensions-apiserver/test/integration/testserver
483482
staging/src/k8s.io/apimachinery/pkg/api/meta

pkg/features/kube_features.go

+2-1
Original file line numberDiff line numberDiff line change
@@ -290,7 +290,8 @@ var defaultKubernetesFeatureGates = map[utilfeature.Feature]utilfeature.FeatureS
290290

291291
// inherited features from apiextensions-apiserver, relisted here to get a conflict if it is changed
292292
// unintentionally on either side:
293-
apiextensionsfeatures.CustomResourceValidation: {Default: true, PreRelease: utilfeature.Beta},
293+
apiextensionsfeatures.CustomResourceValidation: {Default: true, PreRelease: utilfeature.Beta},
294+
apiextensionsfeatures.CustomResourceSubResources: {Default: false, PreRelease: utilfeature.Alpha},
294295

295296
// features that enable backwards compatibility but are scheduled to be removed
296297
ServiceProxyAllowExternalIPs: {Default: false, PreRelease: utilfeature.Deprecated},

staging/src/k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/types.go

+40-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,9 @@ limitations under the License.
1616

1717
package apiextensions
1818

19-
import metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
19+
import (
20+
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
21+
)
2022

2123
// CustomResourceDefinitionSpec describes how a user wants their resource to appear
2224
type CustomResourceDefinitionSpec struct {
@@ -30,6 +32,8 @@ type CustomResourceDefinitionSpec struct {
3032
Scope ResourceScope
3133
// Validation describes the validation methods for CustomResources
3234
Validation *CustomResourceValidation
35+
// SubResources describes the subresources for CustomResources
36+
SubResources *CustomResourceSubResources
3337
}
3438

3539
// CustomResourceDefinitionNames indicates the names to serve this CustomResourceDefinition
@@ -146,3 +150,38 @@ type CustomResourceValidation struct {
146150
// OpenAPIV3Schema is the OpenAPI v3 schema to be validated against.
147151
OpenAPIV3Schema *JSONSchemaProps
148152
}
153+
154+
// CustomResourceSubResources defines the status and scale subresources for CustomResources.
155+
type CustomResourceSubResources struct {
156+
// Status denotes the status subresource for CustomResources
157+
Status *CustomResourceSubResourceStatus
158+
// Scale denotes the scale subresource for CustomResources
159+
Scale *CustomResourceSubResourceScale
160+
}
161+
162+
// CustomResourceSubResourceStatus defines how to serve the status subresource for CustomResources.
163+
// Status is represented by the `.status` JSON path inside of a CustomResource. When set,
164+
// * exposes a /status subresource for the custom resource
165+
// * PUT requests to the /status subresource take a custom resource object, and ignore changes to anything except the status stanza
166+
// * PUT/POST/PATCH requests to the custom resource ignore changes to the status stanza
167+
type CustomResourceSubResourceStatus struct{}
168+
169+
// CustomResourceSubResourceScale defines how to serve the scale subresource for CustomResources.
170+
type CustomResourceSubResourceScale struct {
171+
// SpecReplicasPath defines the JSON path inside of a CustomResource that corresponds to Scale.Spec.Replicas.
172+
// Only JSON paths without the array notation are allowed.
173+
// Must be a JSON Path under .spec.
174+
SpecReplicasPath string
175+
// StatusReplicasPath defines the JSON path inside of a CustomResource that corresponds to Scale.Status.Replicas.
176+
// Only JSON paths without the array notation are allowed.
177+
// Must be a JSON Path under .status.
178+
StatusReplicasPath string
179+
// LabelSelectorPath defines the JSON path inside of a CustomResource that corresponds to Scale.Status.Selector.
180+
// Only JSON paths without the array notation are allowed.
181+
LabelSelectorPath string
182+
// ScaleGroupVersion denotes the GroupVersion in the form "group/version"
183+
// of the Scale object sent as the payload for /scale.
184+
// It allows transition to future versions easily.
185+
// Today only autoscaling/v1 is allowed.
186+
ScaleGroupVersion string
187+
}

staging/src/k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1beta1/types.go

+44-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,9 @@ limitations under the License.
1616

1717
package v1beta1
1818

19-
import metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
19+
import (
20+
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
21+
)
2022

2123
// CustomResourceDefinitionSpec describes how a user wants their resource to appear
2224
type CustomResourceDefinitionSpec struct {
@@ -31,6 +33,11 @@ type CustomResourceDefinitionSpec struct {
3133
// Validation describes the validation methods for CustomResources
3234
// +optional
3335
Validation *CustomResourceValidation `json:"validation,omitempty" protobuf:"bytes,5,opt,name=validation"`
36+
// SubResources describes the subresources for CustomResources
37+
// This field is alpha-level and should only be sent to servers that enable
38+
// subresources via the CustomResourceSubResources feature gate.
39+
// +optional
40+
SubResources *CustomResourceSubResources `json:"subResources,omitempty" protobuf:"bytes,6,opt,name=subResources"`
3441
}
3542

3643
// CustomResourceDefinitionNames indicates the names to serve this CustomResourceDefinition
@@ -147,3 +154,39 @@ type CustomResourceValidation struct {
147154
// OpenAPIV3Schema is the OpenAPI v3 schema to be validated against.
148155
OpenAPIV3Schema *JSONSchemaProps `json:"openAPIV3Schema,omitempty" protobuf:"bytes,1,opt,name=openAPIV3Schema"`
149156
}
157+
158+
// CustomResourceSubResources defines the status and scale subresources for CustomResources.
159+
type CustomResourceSubResources struct {
160+
// Status denotes the status subresource for CustomResources
161+
Status *CustomResourceSubResourceStatus `json:"status,omitempty" protobuf:"bytes,1,opt,name=status"`
162+
// Scale denotes the scale subresource for CustomResources
163+
Scale *CustomResourceSubResourceScale `json:"scale,omitempty" protobuf:"bytes,2,opt,name=scale"`
164+
}
165+
166+
// CustomResourceSubResourceStatus defines how to serve the status subresource for CustomResources.
167+
// Status is represented by the `.status` JSON path inside of a CustomResource. When set,
168+
// * exposes a /status subresource for the custom resource
169+
// * PUT requests to the /status subresource take a custom resource object, and ignore changes to anything except the status stanza
170+
// * PUT/POST/PATCH requests to the custom resource ignore changes to the status stanza
171+
type CustomResourceSubResourceStatus struct{}
172+
173+
// CustomResourceSubResourceScale defines how to serve the scale subresource for CustomResources.
174+
type CustomResourceSubResourceScale struct {
175+
// SpecReplicasPath defines the JSON path inside of a CustomResource that corresponds to Scale.Spec.Replicas.
176+
// Only JSON paths without the array notation are allowed.
177+
// Must be a JSON Path under .spec.
178+
SpecReplicasPath string `json:"specReplicasPath" protobuf:"bytes,1,name=specReplicasPath"`
179+
// StatusReplicasPath defines the JSON path inside of a CustomResource that corresponds to Scale.Status.Replicas.
180+
// Only JSON paths without the array notation are allowed.
181+
// Must be a JSON Path under .status.
182+
StatusReplicasPath string `json:"statusReplicasPath" protobuf:"bytes,2,opt,name=statusReplicasPath"`
183+
// LabelSelectorPath defines the JSON path inside of a CustomResource that corresponds to Scale.Status.Selector.
184+
// Only JSON paths without the array notation are allowed.
185+
// +optional
186+
LabelSelectorPath string `json:"labelSelectorPath,omitempty" protobuf:"bytes,3,opt,name=labelSelectorPath"`
187+
// ScaleGroupVersion denotes the GroupVersion in the form "group/version"
188+
// of the Scale object sent as the payload for /scale.
189+
// It allows transition to future versions easily.
190+
// Today only autoscaling/v1 is allowed.
191+
ScaleGroupVersion string `json:"scaleGroupVersion" protobuf:"bytes,4,name=scaleGroupVersion"`
192+
}

staging/src/k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/validation/validation.go

+67-3
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@ package validation
1818

1919
import (
2020
"fmt"
21+
"reflect"
2122
"strings"
2223

2324
genericvalidation "k8s.io/apimachinery/pkg/api/validation"
@@ -107,7 +108,13 @@ func ValidateCustomResourceDefinitionSpec(spec *apiextensions.CustomResourceDefi
107108
if utilfeature.DefaultFeatureGate.Enabled(apiextensionsfeatures.CustomResourceValidation) {
108109
allErrs = append(allErrs, ValidateCustomResourceDefinitionValidation(spec.Validation, fldPath.Child("validation"))...)
109110
} else if spec.Validation != nil {
110-
allErrs = append(allErrs, field.Forbidden(fldPath.Child("validation"), "disabled by feature-gate"))
111+
allErrs = append(allErrs, field.Forbidden(fldPath.Child("validation"), "disabled by feature-gate CustomResourceValidation"))
112+
}
113+
114+
if utilfeature.DefaultFeatureGate.Enabled(apiextensionsfeatures.CustomResourceSubResources) {
115+
allErrs = append(allErrs, ValidateCustomResourceDefinitionSubResources(spec.SubResources, fldPath.Child("subResources"))...)
116+
} else if spec.SubResources != nil {
117+
allErrs = append(allErrs, field.Forbidden(fldPath.Child("subResources"), "disabled by feature-gate CustomResourceSubresources"))
111118
}
112119

113120
return allErrs
@@ -182,9 +189,27 @@ func ValidateCustomResourceDefinitionValidation(customResourceValidation *apiext
182189
return allErrs
183190
}
184191

185-
if customResourceValidation.OpenAPIV3Schema != nil {
192+
if schema := customResourceValidation.OpenAPIV3Schema; schema != nil {
193+
// if subresources are enabled, only properties is allowed inside the root schema
194+
if utilfeature.DefaultFeatureGate.Enabled(apiextensionsfeatures.CustomResourceSubResources) {
195+
v := reflect.ValueOf(schema).Elem()
196+
fieldsPresent := 0
197+
198+
for i := 0; i < v.NumField(); i++ {
199+
field := v.Field(i).Interface()
200+
if !reflect.DeepEqual(field, reflect.Zero(reflect.TypeOf(field)).Interface()) {
201+
fieldsPresent++
202+
}
203+
}
204+
205+
if fieldsPresent > 1 || (fieldsPresent == 1 && v.FieldByName("Properties").IsNil()) {
206+
allErrs = append(allErrs, field.Invalid(fldPath.Child("openAPIV3Schema"), *schema, fmt.Sprintf("if subresources for custom resources are enabled, only properties can be used at the root of the schema")))
207+
return allErrs
208+
}
209+
}
210+
186211
openAPIV3Schema := &specStandardValidatorV3{}
187-
allErrs = append(allErrs, ValidateCustomResourceDefinitionOpenAPISchema(customResourceValidation.OpenAPIV3Schema, fldPath.Child("openAPIV3Schema"), openAPIV3Schema)...)
212+
allErrs = append(allErrs, ValidateCustomResourceDefinitionOpenAPISchema(schema, fldPath.Child("openAPIV3Schema"), openAPIV3Schema)...)
188213
}
189214

190215
// if validation passed otherwise, make sure we can actually construct a schema validator from this custom resource validation.
@@ -326,3 +351,42 @@ func (v *specStandardValidatorV3) validate(schema *apiextensions.JSONSchemaProps
326351

327352
return allErrs
328353
}
354+
355+
// ValidateCustomResourceDefinitionSubResources statically validates
356+
func ValidateCustomResourceDefinitionSubResources(subResources *apiextensions.CustomResourceSubResources, fldPath *field.Path) field.ErrorList {
357+
allErrs := field.ErrorList{}
358+
359+
if subResources == nil {
360+
return allErrs
361+
}
362+
363+
if subResources.Scale != nil {
364+
if len(subResources.Scale.SpecReplicasPath) == 0 {
365+
allErrs = append(allErrs, field.Invalid(fldPath.Child("scale.specReplicasPath"), subResources.Scale.SpecReplicasPath, "specReplicasPath cannot be empty"))
366+
}
367+
368+
// should be constrained json path
369+
specReplicasPath := strings.TrimPrefix(subResources.Scale.SpecReplicasPath, ".")
370+
splitSpecReplicasPath := strings.Split(specReplicasPath, ".")
371+
if len(splitSpecReplicasPath) <= 1 || splitSpecReplicasPath[0] != "spec" {
372+
allErrs = append(allErrs, field.Invalid(fldPath.Child("scale.specReplicasPath"), subResources.Scale.SpecReplicasPath, "specReplicasPath should be a json path under .spec"))
373+
}
374+
375+
if len(subResources.Scale.StatusReplicasPath) == 0 {
376+
allErrs = append(allErrs, field.Invalid(fldPath.Child("scale.statusReplicasPath"), subResources.Scale.StatusReplicasPath, "statusReplicasPath cannot be empty"))
377+
}
378+
379+
// should be constrained json path
380+
statusReplicasPath := strings.TrimPrefix(subResources.Scale.StatusReplicasPath, ".")
381+
splitStatusReplicasPath := strings.Split(statusReplicasPath, ".")
382+
if len(splitStatusReplicasPath) <= 1 || splitStatusReplicasPath[0] != "status" {
383+
allErrs = append(allErrs, field.Invalid(fldPath.Child("scale.statusReplicasPath"), subResources.Scale.StatusReplicasPath, "statusReplicasPath should be a json path under .status"))
384+
}
385+
386+
if subResources.Scale.ScaleGroupVersion != "autoscaling/v1" {
387+
allErrs = append(allErrs, field.Invalid(fldPath.Child("scale.scaleGroupVersion"), subResources.Scale.ScaleGroupVersion, "scaleGroupVersion must be autoscaling/v1"))
388+
}
389+
}
390+
391+
return allErrs
392+
}

staging/src/k8s.io/apiextensions-apiserver/pkg/apiserver/customresource_discovery_controller.go

+21
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@ import (
2222

2323
"github.com/golang/glog"
2424

25+
autoscaling "k8s.io/api/autoscaling/v1"
2526
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
2627
"k8s.io/apimachinery/pkg/labels"
2728
"k8s.io/apimachinery/pkg/runtime/schema"
@@ -117,6 +118,26 @@ func (c *DiscoveryController) sync(version schema.GroupVersion) error {
117118
Verbs: verbs,
118119
ShortNames: crd.Status.AcceptedNames.ShortNames,
119120
})
121+
122+
if crd.Spec.SubResources != nil && crd.Spec.SubResources.Status != nil {
123+
apiResourcesForDiscovery = append(apiResourcesForDiscovery, metav1.APIResource{
124+
Name: crd.Status.AcceptedNames.Plural + "/status",
125+
Namespaced: crd.Spec.Scope == apiextensions.NamespaceScoped,
126+
Kind: crd.Status.AcceptedNames.Kind,
127+
Verbs: metav1.Verbs([]string{"get", "patch", "update"}),
128+
})
129+
}
130+
131+
if crd.Spec.SubResources != nil && crd.Spec.SubResources.Scale != nil {
132+
apiResourcesForDiscovery = append(apiResourcesForDiscovery, metav1.APIResource{
133+
Group: autoscaling.GroupName, // TODO: use crd.Spec.SubResources.Scale.ScaleGroupVersion
134+
Version: "v1",
135+
Kind: "Scale",
136+
Name: crd.Status.AcceptedNames.Plural + "/scale",
137+
Namespaced: crd.Spec.Scope == apiextensions.NamespaceScoped,
138+
Verbs: metav1.Verbs([]string{"get", "patch", "update"}),
139+
})
140+
}
120141
}
121142

122143
if !foundGroup {

0 commit comments

Comments
 (0)