// 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" but only if // needed. // // The following parts of primary are never modified by merging: // - 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, 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..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 } func mergeConsumes(primary *spec.Swagger, m *spec.Swagger) (skipped []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 } func mergeProduces(primary *spec.Swagger, m *spec.Swagger) (skipped []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 } 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) (skipped []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 } 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) } }