mirror of
https://github.com/k3s-io/kubernetes.git
synced 2025-08-02 16:29:21 +00:00
apiextensions: add structural schema -> go-openapi schema conversion
This commit is contained in:
parent
9c3af43c84
commit
5d6c25854e
@ -0,0 +1,154 @@
|
|||||||
|
/*
|
||||||
|
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 schema
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/go-openapi/spec"
|
||||||
|
)
|
||||||
|
|
||||||
|
// ToGoOpenAPI converts a structural schema to go-openapi schema. It is faithful and roundtrippable
|
||||||
|
// with the exception of `nullable:true` for empty type (`type:""`).
|
||||||
|
//
|
||||||
|
// WARNING: Do not use the returned schema to perform CRD validation until this restriction is solved.
|
||||||
|
//
|
||||||
|
// Nullable:true is mapped to `type:[<structural-type>,"null"]`
|
||||||
|
// if the structural type is non-empty, and nullable is dropped if the structural type is empty.
|
||||||
|
func (s *Structural) ToGoOpenAPI() *spec.Schema {
|
||||||
|
if s == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
ret := &spec.Schema{}
|
||||||
|
|
||||||
|
if s.Items != nil {
|
||||||
|
ret.Items = &spec.SchemaOrArray{Schema: s.Items.ToGoOpenAPI()}
|
||||||
|
}
|
||||||
|
if s.Properties != nil {
|
||||||
|
ret.Properties = make(map[string]spec.Schema, len(s.Properties))
|
||||||
|
for k, v := range s.Properties {
|
||||||
|
ret.Properties[k] = *v.ToGoOpenAPI()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
s.Generic.toGoOpenAPI(ret)
|
||||||
|
s.Extensions.toGoOpenAPI(ret)
|
||||||
|
s.ValueValidation.toGoOpenAPI(ret)
|
||||||
|
|
||||||
|
return ret
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *Generic) toGoOpenAPI(ret *spec.Schema) {
|
||||||
|
if g == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(g.Type) != 0 {
|
||||||
|
ret.Type = spec.StringOrArray{g.Type}
|
||||||
|
if g.Nullable {
|
||||||
|
// go-openapi does not support nullable, but multiple type values.
|
||||||
|
// Only when type is already non-empty, adding null to the types is correct though.
|
||||||
|
// If you add null as only type, you enforce null, in contrast to nullable being
|
||||||
|
// ineffective if no type is provided in a schema.
|
||||||
|
ret.Type = append(ret.Type, "null")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if g.AdditionalProperties != nil {
|
||||||
|
ret.AdditionalProperties = &spec.SchemaOrBool{
|
||||||
|
Allows: g.AdditionalProperties.Bool,
|
||||||
|
Schema: g.AdditionalProperties.Structural.ToGoOpenAPI(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ret.Description = g.Description
|
||||||
|
ret.Title = g.Title
|
||||||
|
ret.Default = g.Default.Object
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *Extensions) toGoOpenAPI(ret *spec.Schema) {
|
||||||
|
if x == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if x.XPreserveUnknownFields {
|
||||||
|
ret.VendorExtensible.AddExtension("x-kubernetes-preserve-unknown-fields", true)
|
||||||
|
}
|
||||||
|
if x.XEmbeddedResource {
|
||||||
|
ret.VendorExtensible.AddExtension("x-kubernetes-embedded-resource", true)
|
||||||
|
}
|
||||||
|
if x.XIntOrString {
|
||||||
|
ret.VendorExtensible.AddExtension("x-kubernetes-int-or-string", true)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *ValueValidation) toGoOpenAPI(ret *spec.Schema) {
|
||||||
|
if v == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
ret.Format = v.Format
|
||||||
|
ret.Maximum = v.Maximum
|
||||||
|
ret.ExclusiveMaximum = v.ExclusiveMaximum
|
||||||
|
ret.Minimum = v.Minimum
|
||||||
|
ret.ExclusiveMinimum = v.ExclusiveMinimum
|
||||||
|
ret.MaxLength = v.MaxLength
|
||||||
|
ret.MinLength = v.MinLength
|
||||||
|
ret.Pattern = v.Pattern
|
||||||
|
ret.MaxItems = v.MaxItems
|
||||||
|
ret.MinItems = v.MinItems
|
||||||
|
ret.UniqueItems = v.UniqueItems
|
||||||
|
ret.MultipleOf = v.MultipleOf
|
||||||
|
if v.Enum != nil {
|
||||||
|
ret.Enum = make([]interface{}, 0, len(v.Enum))
|
||||||
|
for i := range v.Enum {
|
||||||
|
ret.Enum = append(ret.Enum, v.Enum[i].Object)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ret.MaxProperties = v.MaxProperties
|
||||||
|
ret.MinProperties = v.MinProperties
|
||||||
|
ret.Required = v.Required
|
||||||
|
for i := range v.AllOf {
|
||||||
|
ret.AllOf = append(ret.AllOf, *v.AllOf[i].toGoOpenAPI())
|
||||||
|
}
|
||||||
|
for i := range v.AnyOf {
|
||||||
|
ret.AnyOf = append(ret.AnyOf, *v.AnyOf[i].toGoOpenAPI())
|
||||||
|
}
|
||||||
|
for i := range v.OneOf {
|
||||||
|
ret.OneOf = append(ret.OneOf, *v.OneOf[i].toGoOpenAPI())
|
||||||
|
}
|
||||||
|
ret.Not = v.Not.toGoOpenAPI()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (vv *NestedValueValidation) toGoOpenAPI() *spec.Schema {
|
||||||
|
if vv == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
ret := &spec.Schema{}
|
||||||
|
|
||||||
|
vv.ValueValidation.toGoOpenAPI(ret)
|
||||||
|
if vv.Items != nil {
|
||||||
|
ret.Items = &spec.SchemaOrArray{Schema: vv.Items.toGoOpenAPI()}
|
||||||
|
}
|
||||||
|
if vv.Properties != nil {
|
||||||
|
ret.Properties = make(map[string]spec.Schema, len(vv.Properties))
|
||||||
|
for k, v := range vv.Properties {
|
||||||
|
ret.Properties[k] = *v.toGoOpenAPI()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
vv.ForbiddenGenerics.toGoOpenAPI(ret) // normally empty. Exception: int-or-string
|
||||||
|
vv.ForbiddenExtensions.toGoOpenAPI(ret) // shouldn't do anything
|
||||||
|
|
||||||
|
return ret
|
||||||
|
}
|
@ -0,0 +1,116 @@
|
|||||||
|
/*
|
||||||
|
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 schema
|
||||||
|
|
||||||
|
import (
|
||||||
|
"math/rand"
|
||||||
|
"reflect"
|
||||||
|
"regexp"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
fuzz "github.com/google/gofuzz"
|
||||||
|
|
||||||
|
"k8s.io/apiextensions-apiserver/pkg/apis/apiextensions"
|
||||||
|
apiextensionsv1beta1 "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1beta1"
|
||||||
|
"k8s.io/apimachinery/pkg/util/diff"
|
||||||
|
"k8s.io/apimachinery/pkg/util/json"
|
||||||
|
)
|
||||||
|
|
||||||
|
var nullTypeRE = regexp.MustCompile(`"type":\["([^"]*)","null"]`)
|
||||||
|
|
||||||
|
func TestStructuralRoundtrip(t *testing.T) {
|
||||||
|
f := fuzz.New()
|
||||||
|
seed := time.Now().UnixNano()
|
||||||
|
t.Logf("seed = %v", seed)
|
||||||
|
//seed = int64(1549012506261785182)
|
||||||
|
f.RandSource(rand.New(rand.NewSource(seed)))
|
||||||
|
f.Funcs(
|
||||||
|
func(s *JSON, c fuzz.Continue) {
|
||||||
|
switch c.Intn(6) {
|
||||||
|
case 0:
|
||||||
|
s.Object = float64(42.0)
|
||||||
|
case 1:
|
||||||
|
s.Object = map[string]interface{}{"foo": "bar"}
|
||||||
|
case 2:
|
||||||
|
s.Object = ""
|
||||||
|
case 3:
|
||||||
|
s.Object = []string{}
|
||||||
|
case 4:
|
||||||
|
s.Object = map[string]interface{}{}
|
||||||
|
case 5:
|
||||||
|
s.Object = nil
|
||||||
|
}
|
||||||
|
},
|
||||||
|
func(g *Generic, c fuzz.Continue) {
|
||||||
|
c.FuzzNoCustom(g)
|
||||||
|
|
||||||
|
// TODO: make nullable in case of empty type survive go-openapi JSON -> API schema roundtrip
|
||||||
|
// go-openapi does not support nullable. Adding it to a type slice produces OpenAPI v3
|
||||||
|
// incompatible JSON which we cannot unmarshal (without string-replace magic to transform
|
||||||
|
// null types back into nullable). If type is empty, nullable:true is not preserved
|
||||||
|
// at all.
|
||||||
|
if len(g.Type) == 0 {
|
||||||
|
g.Nullable = false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
)
|
||||||
|
f.MaxDepth(3)
|
||||||
|
f.NilChance(0.5)
|
||||||
|
|
||||||
|
for i := 0; i < 10000; i++ {
|
||||||
|
orig := &Structural{}
|
||||||
|
f.Fuzz(orig)
|
||||||
|
|
||||||
|
// normalize Structural.ValueValidation to zero values if it was nil before
|
||||||
|
normalizer := Visitor{
|
||||||
|
Structural: func(s *Structural) bool {
|
||||||
|
if s.ValueValidation == nil {
|
||||||
|
s.ValueValidation = &ValueValidation{}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
},
|
||||||
|
}
|
||||||
|
normalizer.Visit(orig)
|
||||||
|
|
||||||
|
goOpenAPI := orig.ToGoOpenAPI()
|
||||||
|
bs, err := json.Marshal(goOpenAPI)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
str := nullTypeRE.ReplaceAllString(string(bs), `"type":"$1","nullable":true`) // unfold nullable type:[<type>,"null"] -> type:<type>,nullable:true
|
||||||
|
v1beta1Schema := &apiextensionsv1beta1.JSONSchemaProps{}
|
||||||
|
err = json.Unmarshal([]byte(str), v1beta1Schema)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
internalSchema := &apiextensions.JSONSchemaProps{}
|
||||||
|
err = apiextensionsv1beta1.Convert_v1beta1_JSONSchemaProps_To_apiextensions_JSONSchemaProps(v1beta1Schema, internalSchema, nil)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
s, err := NewStructural(internalSchema)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !reflect.DeepEqual(orig, s) {
|
||||||
|
t.Fatalf("original and result differ: %v", diff.ObjectDiff(orig, s))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -29,6 +29,7 @@ func NewSchemaValidator(customResourceValidation *apiextensions.CustomResourceVa
|
|||||||
// Convert CRD schema to openapi schema
|
// Convert CRD schema to openapi schema
|
||||||
openapiSchema := &spec.Schema{}
|
openapiSchema := &spec.Schema{}
|
||||||
if customResourceValidation != nil {
|
if customResourceValidation != nil {
|
||||||
|
// WARNING: do not replace this with Structural.ToGoOpenAPI until it supports nullable.
|
||||||
if err := ConvertJSONSchemaProps(customResourceValidation.OpenAPIV3Schema, openapiSchema); err != nil {
|
if err := ConvertJSONSchemaProps(customResourceValidation.OpenAPIV3Schema, openapiSchema); err != nil {
|
||||||
return nil, nil, err
|
return nil, nil, err
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user