mirror of https://github.com/k3s-io/k3s
196 lines
4.9 KiB
Go
196 lines
4.9 KiB
Go
|
// Copyright 2019 The Kubernetes Authors.
|
||
|
// SPDX-License-Identifier: Apache-2.0
|
||
|
|
||
|
package accumulator
|
||
|
|
||
|
import (
|
||
|
"encoding/json"
|
||
|
"strings"
|
||
|
|
||
|
"github.com/go-openapi/spec"
|
||
|
"github.com/pkg/errors"
|
||
|
"sigs.k8s.io/kustomize/api/filesys"
|
||
|
"sigs.k8s.io/kustomize/api/ifc"
|
||
|
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
||
|
"sigs.k8s.io/kustomize/api/resid"
|
||
|
"sigs.k8s.io/kustomize/api/types"
|
||
|
"sigs.k8s.io/yaml"
|
||
|
)
|
||
|
|
||
|
// OpenAPIDefinition describes single type.
|
||
|
// Normally these definitions are auto-generated using gen-openapi.
|
||
|
// Same as in k8s.io / kube-openapi / pkg / common.
|
||
|
type OpenAPIDefinition struct {
|
||
|
Schema spec.Schema
|
||
|
Dependencies []string
|
||
|
}
|
||
|
|
||
|
type myProperties map[string]spec.Schema
|
||
|
type nameToApiMap map[string]OpenAPIDefinition
|
||
|
|
||
|
// LoadConfigFromCRDs parse CRD schemas from paths into a TransformerConfig
|
||
|
func LoadConfigFromCRDs(
|
||
|
ldr ifc.Loader, paths []string) (*builtinconfig.TransformerConfig, error) {
|
||
|
tc := builtinconfig.MakeEmptyConfig()
|
||
|
for _, path := range paths {
|
||
|
content, err := ldr.Load(path)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
m, err := makeNameToApiMap(content)
|
||
|
if err != nil {
|
||
|
return nil, errors.Wrapf(err, "unable to parse open API definition from '%s'", path)
|
||
|
}
|
||
|
otherTc, err := makeConfigFromApiMap(m)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
tc, err = tc.Merge(otherTc)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
}
|
||
|
return tc, nil
|
||
|
}
|
||
|
|
||
|
func makeNameToApiMap(content []byte) (result nameToApiMap, err error) {
|
||
|
if content[0] == '{' {
|
||
|
err = json.Unmarshal(content, &result)
|
||
|
} else {
|
||
|
err = yaml.Unmarshal(content, &result)
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func makeConfigFromApiMap(m nameToApiMap) (*builtinconfig.TransformerConfig, error) {
|
||
|
result := builtinconfig.MakeEmptyConfig()
|
||
|
for name, api := range m {
|
||
|
if !looksLikeAk8sType(api.Schema.SchemaProps.Properties) {
|
||
|
continue
|
||
|
}
|
||
|
tc := builtinconfig.MakeEmptyConfig()
|
||
|
err := loadCrdIntoConfig(
|
||
|
tc, makeGvkFromTypeName(name), m, name, []string{})
|
||
|
if err != nil {
|
||
|
return result, err
|
||
|
}
|
||
|
result, err = result.Merge(tc)
|
||
|
if err != nil {
|
||
|
return result, err
|
||
|
}
|
||
|
}
|
||
|
return result, nil
|
||
|
}
|
||
|
|
||
|
// TODO: Get Group and Version for CRD from the
|
||
|
// openAPI definition once
|
||
|
// "x-kubernetes-group-version-kind" is available in CRD
|
||
|
func makeGvkFromTypeName(n string) resid.Gvk {
|
||
|
names := strings.Split(n, filesys.SelfDir)
|
||
|
kind := names[len(names)-1]
|
||
|
return resid.Gvk{Kind: kind}
|
||
|
}
|
||
|
|
||
|
func looksLikeAk8sType(properties myProperties) bool {
|
||
|
_, ok := properties["kind"]
|
||
|
if !ok {
|
||
|
return false
|
||
|
}
|
||
|
_, ok = properties["apiVersion"]
|
||
|
if !ok {
|
||
|
return false
|
||
|
}
|
||
|
_, ok = properties["metadata"]
|
||
|
return ok
|
||
|
}
|
||
|
|
||
|
const (
|
||
|
// "x-kubernetes-annotation": ""
|
||
|
xAnnotation = "x-kubernetes-annotation"
|
||
|
|
||
|
// "x-kubernetes-label-selector": ""
|
||
|
xLabelSelector = "x-kubernetes-label-selector"
|
||
|
|
||
|
// "x-kubernetes-identity": ""
|
||
|
xIdentity = "x-kubernetes-identity"
|
||
|
|
||
|
// "x-kubernetes-object-ref-api-version": <apiVersion name>
|
||
|
xVersion = "x-kubernetes-object-ref-api-version"
|
||
|
|
||
|
// "x-kubernetes-object-ref-kind": <kind name>
|
||
|
xKind = "x-kubernetes-object-ref-kind"
|
||
|
|
||
|
// "x-kubernetes-object-ref-name-key": "name"
|
||
|
// default is "name"
|
||
|
xNameKey = "x-kubernetes-object-ref-name-key"
|
||
|
)
|
||
|
|
||
|
// loadCrdIntoConfig loads a CRD spec into a TransformerConfig
|
||
|
func loadCrdIntoConfig(
|
||
|
theConfig *builtinconfig.TransformerConfig, theGvk resid.Gvk, theMap nameToApiMap,
|
||
|
typeName string, path []string) (err error) {
|
||
|
api, ok := theMap[typeName]
|
||
|
if !ok {
|
||
|
return nil
|
||
|
}
|
||
|
for propName, property := range api.Schema.SchemaProps.Properties {
|
||
|
_, annotate := property.Extensions.GetString(xAnnotation)
|
||
|
if annotate {
|
||
|
err = theConfig.AddAnnotationFieldSpec(
|
||
|
makeFs(theGvk, append(path, propName)))
|
||
|
if err != nil {
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
_, label := property.Extensions.GetString(xLabelSelector)
|
||
|
if label {
|
||
|
err = theConfig.AddLabelFieldSpec(
|
||
|
makeFs(theGvk, append(path, propName)))
|
||
|
if err != nil {
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
_, identity := property.Extensions.GetString(xIdentity)
|
||
|
if identity {
|
||
|
err = theConfig.AddPrefixFieldSpec(
|
||
|
makeFs(theGvk, append(path, propName)))
|
||
|
if err != nil {
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
version, ok := property.Extensions.GetString(xVersion)
|
||
|
if ok {
|
||
|
kind, ok := property.Extensions.GetString(xKind)
|
||
|
if ok {
|
||
|
nameKey, ok := property.Extensions.GetString(xNameKey)
|
||
|
if !ok {
|
||
|
nameKey = "name"
|
||
|
}
|
||
|
err = theConfig.AddNamereferenceFieldSpec(
|
||
|
builtinconfig.NameBackReferences{
|
||
|
Gvk: resid.Gvk{Kind: kind, Version: version},
|
||
|
Referrers: []types.FieldSpec{
|
||
|
makeFs(theGvk, append(path, propName, nameKey))},
|
||
|
})
|
||
|
if err != nil {
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if property.Ref.GetURL() != nil {
|
||
|
loadCrdIntoConfig(
|
||
|
theConfig, theGvk, theMap,
|
||
|
property.Ref.String(), append(path, propName))
|
||
|
}
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func makeFs(in resid.Gvk, path []string) types.FieldSpec {
|
||
|
return types.FieldSpec{
|
||
|
CreateIfNotPresent: false,
|
||
|
Gvk: in,
|
||
|
Path: strings.Join(path, "/"),
|
||
|
}
|
||
|
}
|