mirror of
https://github.com/k3s-io/kubernetes.git
synced 2025-08-01 07:47:56 +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
|
||||
openapiSchema := &spec.Schema{}
|
||||
if customResourceValidation != nil {
|
||||
// WARNING: do not replace this with Structural.ToGoOpenAPI until it supports nullable.
|
||||
if err := ConvertJSONSchemaProps(customResourceValidation.OpenAPIV3Schema, openapiSchema); err != nil {
|
||||
return nil, nil, err
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user