mirror of
				https://github.com/optim-enterprises-bv/kubernetes.git
				synced 2025-11-04 12:18:16 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			426 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			426 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// Copyright 2015 go-swagger maintainers
 | 
						|
//
 | 
						|
// 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 analysis
 | 
						|
 | 
						|
import (
 | 
						|
	"fmt"
 | 
						|
	"reflect"
 | 
						|
 | 
						|
	"github.com/go-openapi/spec"
 | 
						|
)
 | 
						|
 | 
						|
// Mixin modifies the primary swagger spec by adding the paths and
 | 
						|
// definitions from the mixin specs. Top level parameters and
 | 
						|
// responses from the mixins are also carried over. Operation id
 | 
						|
// collisions are avoided by appending "Mixin<N>" but only if
 | 
						|
// needed.
 | 
						|
//
 | 
						|
// The following parts of primary are subject to merge, filling empty details
 | 
						|
//   - Info
 | 
						|
//   - BasePath
 | 
						|
//   - Host
 | 
						|
//   - ExternalDocs
 | 
						|
//
 | 
						|
// Consider calling FixEmptyResponseDescriptions() on the modified primary
 | 
						|
// if you read them from storage and they are valid to start with.
 | 
						|
//
 | 
						|
// Entries in "paths", "definitions", "parameters" and "responses" are
 | 
						|
// added to the primary in the order of the given mixins. If the entry
 | 
						|
// already exists in primary it is skipped with a warning message.
 | 
						|
//
 | 
						|
// The count of skipped entries (from collisions) is returned so any
 | 
						|
// deviation from the number expected can flag a warning in your build
 | 
						|
// scripts. Carefully review the collisions before accepting them;
 | 
						|
// consider renaming things if possible.
 | 
						|
//
 | 
						|
// No key normalization takes place (paths, type defs,
 | 
						|
// etc). Ensure they are canonical if your downstream tools do
 | 
						|
// key normalization of any form.
 | 
						|
//
 | 
						|
// Merging schemes (http, https), and consumers/producers do not account for
 | 
						|
// collisions.
 | 
						|
func Mixin(primary *spec.Swagger, mixins ...*spec.Swagger) []string {
 | 
						|
	skipped := make([]string, 0, len(mixins))
 | 
						|
	opIds := getOpIds(primary)
 | 
						|
	initPrimary(primary)
 | 
						|
 | 
						|
	for i, m := range mixins {
 | 
						|
		skipped = append(skipped, mergeSwaggerProps(primary, m)...)
 | 
						|
 | 
						|
		skipped = append(skipped, mergeConsumes(primary, m)...)
 | 
						|
 | 
						|
		skipped = append(skipped, mergeProduces(primary, m)...)
 | 
						|
 | 
						|
		skipped = append(skipped, mergeTags(primary, m)...)
 | 
						|
 | 
						|
		skipped = append(skipped, mergeSchemes(primary, m)...)
 | 
						|
 | 
						|
		skipped = append(skipped, mergeSecurityDefinitions(primary, m)...)
 | 
						|
 | 
						|
		skipped = append(skipped, mergeSecurityRequirements(primary, m)...)
 | 
						|
 | 
						|
		skipped = append(skipped, mergeDefinitions(primary, m)...)
 | 
						|
 | 
						|
		// merging paths requires a map of operationIDs to work with
 | 
						|
		skipped = append(skipped, mergePaths(primary, m, opIds, i)...)
 | 
						|
 | 
						|
		skipped = append(skipped, mergeParameters(primary, m)...)
 | 
						|
 | 
						|
		skipped = append(skipped, mergeResponses(primary, m)...)
 | 
						|
	}
 | 
						|
	return skipped
 | 
						|
}
 | 
						|
 | 
						|
// getOpIds extracts all the paths.<path>.operationIds from the given
 | 
						|
// spec and returns them as the keys in a map with 'true' values.
 | 
						|
func getOpIds(s *spec.Swagger) map[string]bool {
 | 
						|
	rv := make(map[string]bool)
 | 
						|
	if s.Paths == nil {
 | 
						|
		return rv
 | 
						|
	}
 | 
						|
	for _, v := range s.Paths.Paths {
 | 
						|
		piops := pathItemOps(v)
 | 
						|
		for _, op := range piops {
 | 
						|
			rv[op.ID] = true
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return rv
 | 
						|
}
 | 
						|
 | 
						|
func pathItemOps(p spec.PathItem) []*spec.Operation {
 | 
						|
	var rv []*spec.Operation
 | 
						|
	rv = appendOp(rv, p.Get)
 | 
						|
	rv = appendOp(rv, p.Put)
 | 
						|
	rv = appendOp(rv, p.Post)
 | 
						|
	rv = appendOp(rv, p.Delete)
 | 
						|
	rv = appendOp(rv, p.Head)
 | 
						|
	rv = appendOp(rv, p.Patch)
 | 
						|
	return rv
 | 
						|
}
 | 
						|
 | 
						|
func appendOp(ops []*spec.Operation, op *spec.Operation) []*spec.Operation {
 | 
						|
	if op == nil {
 | 
						|
		return ops
 | 
						|
	}
 | 
						|
	return append(ops, op)
 | 
						|
}
 | 
						|
 | 
						|
func mergeSecurityDefinitions(primary *spec.Swagger, m *spec.Swagger) (skipped []string) {
 | 
						|
	for k, v := range m.SecurityDefinitions {
 | 
						|
		if _, exists := primary.SecurityDefinitions[k]; exists {
 | 
						|
			warn := fmt.Sprintf(
 | 
						|
				"SecurityDefinitions entry '%v' already exists in primary or higher priority mixin, skipping\n", k)
 | 
						|
			skipped = append(skipped, warn)
 | 
						|
			continue
 | 
						|
		}
 | 
						|
		primary.SecurityDefinitions[k] = v
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func mergeSecurityRequirements(primary *spec.Swagger, m *spec.Swagger) (skipped []string) {
 | 
						|
	for _, v := range m.Security {
 | 
						|
		found := false
 | 
						|
		for _, vv := range primary.Security {
 | 
						|
			if reflect.DeepEqual(v, vv) {
 | 
						|
				found = true
 | 
						|
				break
 | 
						|
			}
 | 
						|
		}
 | 
						|
		if found {
 | 
						|
			warn := fmt.Sprintf(
 | 
						|
				"Security requirement: '%v' already exists in primary or higher priority mixin, skipping\n", v)
 | 
						|
			skipped = append(skipped, warn)
 | 
						|
			continue
 | 
						|
		}
 | 
						|
		primary.Security = append(primary.Security, v)
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func mergeDefinitions(primary *spec.Swagger, m *spec.Swagger) (skipped []string) {
 | 
						|
	for k, v := range m.Definitions {
 | 
						|
		// assume name collisions represent IDENTICAL type. careful.
 | 
						|
		if _, exists := primary.Definitions[k]; exists {
 | 
						|
			warn := fmt.Sprintf(
 | 
						|
				"definitions entry '%v' already exists in primary or higher priority mixin, skipping\n", k)
 | 
						|
			skipped = append(skipped, warn)
 | 
						|
			continue
 | 
						|
		}
 | 
						|
		primary.Definitions[k] = v
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func mergePaths(primary *spec.Swagger, m *spec.Swagger, opIds map[string]bool, mixIndex int) (skipped []string) {
 | 
						|
	if m.Paths != nil {
 | 
						|
		for k, v := range m.Paths.Paths {
 | 
						|
			if _, exists := primary.Paths.Paths[k]; exists {
 | 
						|
				warn := fmt.Sprintf(
 | 
						|
					"paths entry '%v' already exists in primary or higher priority mixin, skipping\n", k)
 | 
						|
				skipped = append(skipped, warn)
 | 
						|
				continue
 | 
						|
			}
 | 
						|
 | 
						|
			// Swagger requires that operationIds be
 | 
						|
			// unique within a spec. If we find a
 | 
						|
			// collision we append "Mixin0" to the
 | 
						|
			// operatoinId we are adding, where 0 is mixin
 | 
						|
			// index.  We assume that operationIds with
 | 
						|
			// all the proivded specs are already unique.
 | 
						|
			piops := pathItemOps(v)
 | 
						|
			for _, piop := range piops {
 | 
						|
				if opIds[piop.ID] {
 | 
						|
					piop.ID = fmt.Sprintf("%v%v%v", piop.ID, "Mixin", mixIndex)
 | 
						|
				}
 | 
						|
				opIds[piop.ID] = true
 | 
						|
			}
 | 
						|
			primary.Paths.Paths[k] = v
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func mergeParameters(primary *spec.Swagger, m *spec.Swagger) (skipped []string) {
 | 
						|
	for k, v := range m.Parameters {
 | 
						|
		// could try to rename on conflict but would
 | 
						|
		// have to fix $refs in the mixin. Complain
 | 
						|
		// for now
 | 
						|
		if _, exists := primary.Parameters[k]; exists {
 | 
						|
			warn := fmt.Sprintf(
 | 
						|
				"top level parameters entry '%v' already exists in primary or higher priority mixin, skipping\n", k)
 | 
						|
			skipped = append(skipped, warn)
 | 
						|
			continue
 | 
						|
		}
 | 
						|
		primary.Parameters[k] = v
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func mergeResponses(primary *spec.Swagger, m *spec.Swagger) (skipped []string) {
 | 
						|
	for k, v := range m.Responses {
 | 
						|
		// could try to rename on conflict but would
 | 
						|
		// have to fix $refs in the mixin. Complain
 | 
						|
		// for now
 | 
						|
		if _, exists := primary.Responses[k]; exists {
 | 
						|
			warn := fmt.Sprintf(
 | 
						|
				"top level responses entry '%v' already exists in primary or higher priority mixin, skipping\n", k)
 | 
						|
			skipped = append(skipped, warn)
 | 
						|
			continue
 | 
						|
		}
 | 
						|
		primary.Responses[k] = v
 | 
						|
	}
 | 
						|
	return skipped
 | 
						|
}
 | 
						|
 | 
						|
func mergeConsumes(primary *spec.Swagger, m *spec.Swagger) []string {
 | 
						|
	for _, v := range m.Consumes {
 | 
						|
		found := false
 | 
						|
		for _, vv := range primary.Consumes {
 | 
						|
			if v == vv {
 | 
						|
				found = true
 | 
						|
				break
 | 
						|
			}
 | 
						|
		}
 | 
						|
		if found {
 | 
						|
			// no warning here: we just skip it
 | 
						|
			continue
 | 
						|
		}
 | 
						|
		primary.Consumes = append(primary.Consumes, v)
 | 
						|
	}
 | 
						|
	return []string{}
 | 
						|
}
 | 
						|
 | 
						|
func mergeProduces(primary *spec.Swagger, m *spec.Swagger) []string {
 | 
						|
	for _, v := range m.Produces {
 | 
						|
		found := false
 | 
						|
		for _, vv := range primary.Produces {
 | 
						|
			if v == vv {
 | 
						|
				found = true
 | 
						|
				break
 | 
						|
			}
 | 
						|
		}
 | 
						|
		if found {
 | 
						|
			// no warning here: we just skip it
 | 
						|
			continue
 | 
						|
		}
 | 
						|
		primary.Produces = append(primary.Produces, v)
 | 
						|
	}
 | 
						|
	return []string{}
 | 
						|
}
 | 
						|
 | 
						|
func mergeTags(primary *spec.Swagger, m *spec.Swagger) (skipped []string) {
 | 
						|
	for _, v := range m.Tags {
 | 
						|
		found := false
 | 
						|
		for _, vv := range primary.Tags {
 | 
						|
			if v.Name == vv.Name {
 | 
						|
				found = true
 | 
						|
				break
 | 
						|
			}
 | 
						|
		}
 | 
						|
		if found {
 | 
						|
			warn := fmt.Sprintf(
 | 
						|
				"top level tags entry with name '%v' already exists in primary or higher priority mixin, skipping\n", v.Name)
 | 
						|
			skipped = append(skipped, warn)
 | 
						|
			continue
 | 
						|
		}
 | 
						|
		primary.Tags = append(primary.Tags, v)
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func mergeSchemes(primary *spec.Swagger, m *spec.Swagger) []string {
 | 
						|
	for _, v := range m.Schemes {
 | 
						|
		found := false
 | 
						|
		for _, vv := range primary.Schemes {
 | 
						|
			if v == vv {
 | 
						|
				found = true
 | 
						|
				break
 | 
						|
			}
 | 
						|
		}
 | 
						|
		if found {
 | 
						|
			// no warning here: we just skip it
 | 
						|
			continue
 | 
						|
		}
 | 
						|
		primary.Schemes = append(primary.Schemes, v)
 | 
						|
	}
 | 
						|
	return []string{}
 | 
						|
}
 | 
						|
 | 
						|
func mergeSwaggerProps(primary *spec.Swagger, m *spec.Swagger) []string {
 | 
						|
	var skipped []string
 | 
						|
	primary.Extensions, skipped = mergeExtensions(primary.Extensions, m.Extensions)
 | 
						|
 | 
						|
	// merging details in swagger top properties
 | 
						|
	if primary.Host == "" {
 | 
						|
		primary.Host = m.Host
 | 
						|
	}
 | 
						|
	if primary.BasePath == "" {
 | 
						|
		primary.BasePath = m.BasePath
 | 
						|
	}
 | 
						|
	if primary.Info == nil {
 | 
						|
		primary.Info = m.Info
 | 
						|
	} else if m.Info != nil {
 | 
						|
		var sk []string
 | 
						|
		primary.Info.Extensions, sk = mergeExtensions(primary.Info.Extensions, m.Info.Extensions)
 | 
						|
		skipped = append(skipped, sk...)
 | 
						|
		if primary.Info.Description == "" {
 | 
						|
			primary.Info.Description = m.Info.Description
 | 
						|
		}
 | 
						|
		if primary.Info.Title == "" {
 | 
						|
			primary.Info.Description = m.Info.Description
 | 
						|
		}
 | 
						|
		if primary.Info.TermsOfService == "" {
 | 
						|
			primary.Info.TermsOfService = m.Info.TermsOfService
 | 
						|
		}
 | 
						|
		if primary.Info.Version == "" {
 | 
						|
			primary.Info.Version = m.Info.Version
 | 
						|
		}
 | 
						|
 | 
						|
		if primary.Info.Contact == nil {
 | 
						|
			primary.Info.Contact = m.Info.Contact
 | 
						|
		} else if m.Info.Contact != nil {
 | 
						|
			if primary.Info.Contact.Name == "" {
 | 
						|
				primary.Info.Contact.Name = m.Info.Contact.Name
 | 
						|
			}
 | 
						|
			if primary.Info.Contact.URL == "" {
 | 
						|
				primary.Info.Contact.URL = m.Info.Contact.URL
 | 
						|
			}
 | 
						|
			if primary.Info.Contact.Email == "" {
 | 
						|
				primary.Info.Contact.Email = m.Info.Contact.Email
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		if primary.Info.License == nil {
 | 
						|
			primary.Info.License = m.Info.License
 | 
						|
		} else if m.Info.License != nil {
 | 
						|
			if primary.Info.License.Name == "" {
 | 
						|
				primary.Info.License.Name = m.Info.License.Name
 | 
						|
			}
 | 
						|
			if primary.Info.License.URL == "" {
 | 
						|
				primary.Info.License.URL = m.Info.License.URL
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
	}
 | 
						|
	if primary.ExternalDocs == nil {
 | 
						|
		primary.ExternalDocs = m.ExternalDocs
 | 
						|
	} else if m.ExternalDocs != nil {
 | 
						|
		if primary.ExternalDocs.Description == "" {
 | 
						|
			primary.ExternalDocs.Description = m.ExternalDocs.Description
 | 
						|
		}
 | 
						|
		if primary.ExternalDocs.URL == "" {
 | 
						|
			primary.ExternalDocs.URL = m.ExternalDocs.URL
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return skipped
 | 
						|
}
 | 
						|
 | 
						|
func mergeExtensions(primary spec.Extensions, m spec.Extensions) (result spec.Extensions, skipped []string) {
 | 
						|
	if primary == nil {
 | 
						|
		result = m
 | 
						|
		return
 | 
						|
	}
 | 
						|
	if m == nil {
 | 
						|
		result = primary
 | 
						|
		return
 | 
						|
	}
 | 
						|
	result = primary
 | 
						|
	for k, v := range m {
 | 
						|
		if _, found := primary[k]; found {
 | 
						|
			skipped = append(skipped, k)
 | 
						|
			continue
 | 
						|
		}
 | 
						|
		primary[k] = v
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func initPrimary(primary *spec.Swagger) {
 | 
						|
	if primary.SecurityDefinitions == nil {
 | 
						|
		primary.SecurityDefinitions = make(map[string]*spec.SecurityScheme)
 | 
						|
	}
 | 
						|
	if primary.Security == nil {
 | 
						|
		primary.Security = make([]map[string][]string, 0, 10)
 | 
						|
	}
 | 
						|
	if primary.Produces == nil {
 | 
						|
		primary.Produces = make([]string, 0, 10)
 | 
						|
	}
 | 
						|
	if primary.Consumes == nil {
 | 
						|
		primary.Consumes = make([]string, 0, 10)
 | 
						|
	}
 | 
						|
	if primary.Tags == nil {
 | 
						|
		primary.Tags = make([]spec.Tag, 0, 10)
 | 
						|
	}
 | 
						|
	if primary.Schemes == nil {
 | 
						|
		primary.Schemes = make([]string, 0, 10)
 | 
						|
	}
 | 
						|
	if primary.Paths == nil {
 | 
						|
		primary.Paths = &spec.Paths{Paths: make(map[string]spec.PathItem)}
 | 
						|
	}
 | 
						|
	if primary.Paths.Paths == nil {
 | 
						|
		primary.Paths.Paths = make(map[string]spec.PathItem)
 | 
						|
	}
 | 
						|
	if primary.Definitions == nil {
 | 
						|
		primary.Definitions = make(spec.Definitions)
 | 
						|
	}
 | 
						|
	if primary.Parameters == nil {
 | 
						|
		primary.Parameters = make(map[string]spec.Parameter)
 | 
						|
	}
 | 
						|
	if primary.Responses == nil {
 | 
						|
		primary.Responses = make(map[string]spec.Response)
 | 
						|
	}
 | 
						|
}
 |