mirror of
				https://github.com/superseriousbusiness/gotosocial.git
				synced 2025-11-04 00:02:25 -06:00 
			
		
		
		
	
		
			
				
	
	
		
			515 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			515 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, skippedInfo, skippedDocs []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 {
 | 
						|
		skippedInfo = mergeInfo(primary.Info, m.Info)
 | 
						|
		skipped = append(skipped, skippedInfo...)
 | 
						|
	}
 | 
						|
 | 
						|
	if primary.ExternalDocs == nil {
 | 
						|
		primary.ExternalDocs = m.ExternalDocs
 | 
						|
	} else if m != nil {
 | 
						|
		skippedDocs = mergeExternalDocs(primary.ExternalDocs, m.ExternalDocs)
 | 
						|
		skipped = append(skipped, skippedDocs...)
 | 
						|
	}
 | 
						|
 | 
						|
	return skipped
 | 
						|
}
 | 
						|
 | 
						|
//nolint:unparam
 | 
						|
func mergeExternalDocs(primary *spec.ExternalDocumentation, m *spec.ExternalDocumentation) []string {
 | 
						|
	if primary.Description == "" {
 | 
						|
		primary.Description = m.Description
 | 
						|
	}
 | 
						|
 | 
						|
	if primary.URL == "" {
 | 
						|
		primary.URL = m.URL
 | 
						|
	}
 | 
						|
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func mergeInfo(primary *spec.Info, m *spec.Info) []string {
 | 
						|
	var sk, skipped []string
 | 
						|
 | 
						|
	primary.Extensions, sk = mergeExtensions(primary.Extensions, m.Extensions)
 | 
						|
	skipped = append(skipped, sk...)
 | 
						|
 | 
						|
	if primary.Description == "" {
 | 
						|
		primary.Description = m.Description
 | 
						|
	}
 | 
						|
 | 
						|
	if primary.Title == "" {
 | 
						|
		primary.Description = m.Description
 | 
						|
	}
 | 
						|
 | 
						|
	if primary.TermsOfService == "" {
 | 
						|
		primary.TermsOfService = m.TermsOfService
 | 
						|
	}
 | 
						|
 | 
						|
	if primary.Version == "" {
 | 
						|
		primary.Version = m.Version
 | 
						|
	}
 | 
						|
 | 
						|
	if primary.Contact == nil {
 | 
						|
		primary.Contact = m.Contact
 | 
						|
	} else if m.Contact != nil {
 | 
						|
		var csk []string
 | 
						|
		primary.Contact.Extensions, csk = mergeExtensions(primary.Contact.Extensions, m.Contact.Extensions)
 | 
						|
		skipped = append(skipped, csk...)
 | 
						|
 | 
						|
		if primary.Contact.Name == "" {
 | 
						|
			primary.Contact.Name = m.Contact.Name
 | 
						|
		}
 | 
						|
 | 
						|
		if primary.Contact.URL == "" {
 | 
						|
			primary.Contact.URL = m.Contact.URL
 | 
						|
		}
 | 
						|
 | 
						|
		if primary.Contact.Email == "" {
 | 
						|
			primary.Contact.Email = m.Contact.Email
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if primary.License == nil {
 | 
						|
		primary.License = m.License
 | 
						|
	} else if m.License != nil {
 | 
						|
		var lsk []string
 | 
						|
		primary.License.Extensions, lsk = mergeExtensions(primary.License.Extensions, m.License.Extensions)
 | 
						|
		skipped = append(skipped, lsk...)
 | 
						|
 | 
						|
		if primary.License.Name == "" {
 | 
						|
			primary.License.Name = m.License.Name
 | 
						|
		}
 | 
						|
 | 
						|
		if primary.License.URL == "" {
 | 
						|
			primary.License.URL = m.License.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)
 | 
						|
	}
 | 
						|
}
 |