2019-08-30 18:33:25 +00:00
|
|
|
/*
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2019-09-27 21:51:53 +00:00
|
|
|
package builder
|
2019-08-30 18:33:25 +00:00
|
|
|
|
|
|
|
import (
|
2019-09-27 21:51:53 +00:00
|
|
|
"k8s.io/kube-openapi/pkg/aggregator"
|
2021-07-02 08:43:15 +00:00
|
|
|
"k8s.io/kube-openapi/pkg/validation/spec"
|
2019-08-30 18:33:25 +00:00
|
|
|
)
|
|
|
|
|
2019-09-27 21:51:53 +00:00
|
|
|
// MergeSpecs aggregates all OpenAPI specs, reusing the metadata of the first, static spec as the basis.
|
|
|
|
// The static spec has the highest priority, and its paths and definitions won't get overlapped by
|
|
|
|
// user-defined CRDs. None of the input is mutated, but input and output share data structures.
|
|
|
|
func MergeSpecs(staticSpec *spec.Swagger, crdSpecs ...*spec.Swagger) (*spec.Swagger, error) {
|
2019-08-30 18:33:25 +00:00
|
|
|
// create shallow copy of staticSpec, but replace paths and definitions because we modify them.
|
|
|
|
specToReturn := *staticSpec
|
|
|
|
if staticSpec.Definitions != nil {
|
|
|
|
specToReturn.Definitions = spec.Definitions{}
|
|
|
|
for k, s := range staticSpec.Definitions {
|
|
|
|
specToReturn.Definitions[k] = s
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if staticSpec.Paths != nil {
|
|
|
|
specToReturn.Paths = &spec.Paths{
|
|
|
|
Paths: map[string]spec.PathItem{},
|
|
|
|
}
|
|
|
|
for k, p := range staticSpec.Paths.Paths {
|
|
|
|
specToReturn.Paths.Paths[k] = p
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-27 21:51:53 +00:00
|
|
|
crdSpec := &spec.Swagger{}
|
2019-08-30 18:33:25 +00:00
|
|
|
for _, s := range crdSpecs {
|
2019-09-27 21:51:53 +00:00
|
|
|
// merge specs without checking conflicts, since the naming controller prevents
|
|
|
|
// conflicts between user-defined CRDs
|
|
|
|
mergeSpec(crdSpec, s)
|
2019-08-30 18:33:25 +00:00
|
|
|
}
|
|
|
|
|
2019-09-27 21:51:53 +00:00
|
|
|
// The static spec has the highest priority. Resolve conflicts to prevent user-defined
|
|
|
|
// CRDs potentially overlapping the built-in apiextensions API
|
|
|
|
if err := aggregator.MergeSpecsIgnorePathConflict(&specToReturn, crdSpec); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &specToReturn, nil
|
2019-08-30 18:33:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// mergeSpec copies paths and definitions from source to dest, mutating dest, but not source.
|
|
|
|
// We assume that conflicts do not matter.
|
|
|
|
func mergeSpec(dest, source *spec.Swagger) {
|
|
|
|
if source == nil || source.Paths == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if dest.Paths == nil {
|
|
|
|
dest.Paths = &spec.Paths{}
|
|
|
|
}
|
|
|
|
for k, v := range source.Definitions {
|
|
|
|
if dest.Definitions == nil {
|
|
|
|
dest.Definitions = spec.Definitions{}
|
|
|
|
}
|
|
|
|
dest.Definitions[k] = v
|
|
|
|
}
|
|
|
|
for k, v := range source.Paths.Paths {
|
|
|
|
if dest.Paths.Paths == nil {
|
|
|
|
dest.Paths.Paths = map[string]spec.PathItem{}
|
|
|
|
}
|
|
|
|
dest.Paths.Paths[k] = v
|
|
|
|
}
|
|
|
|
}
|