mirror of
https://github.com/LukeHagar/libopenapi.git
synced 2025-12-09 20:47:44 +00:00
Every `Build()` method now requires a `context.Context`. This is so the rolodex knows where to resolve from when locating relative links. Without knowing where we are, there is no way to resolve anything. This new mechanism allows the model to recurse across as many files as required to locate references, without loosing track of where we are in the process. Signed-off-by: quobix <dave@quobix.com>
221 lines
6.3 KiB
Go
221 lines
6.3 KiB
Go
// Copyright 2022 Princess B33f Heavy Industries / Dave Shanley
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package v3
|
|
|
|
import (
|
|
"context"
|
|
"crypto/sha256"
|
|
"fmt"
|
|
"github.com/pb33f/libopenapi/datamodel/low"
|
|
"github.com/pb33f/libopenapi/datamodel/low/base"
|
|
"github.com/pb33f/libopenapi/index"
|
|
"github.com/pb33f/libopenapi/utils"
|
|
"gopkg.in/yaml.v3"
|
|
"sort"
|
|
"strings"
|
|
)
|
|
|
|
// Parameter represents a high-level OpenAPI 3+ Parameter object, that is backed by a low-level one.
|
|
//
|
|
// A unique parameter is defined by a combination of a name and location.
|
|
// - https://spec.openapis.org/oas/v3.1.0#parameter-object
|
|
type Parameter struct {
|
|
Name low.NodeReference[string]
|
|
In low.NodeReference[string]
|
|
Description low.NodeReference[string]
|
|
Required low.NodeReference[bool]
|
|
Deprecated low.NodeReference[bool]
|
|
AllowEmptyValue low.NodeReference[bool]
|
|
Style low.NodeReference[string]
|
|
Explode low.NodeReference[bool]
|
|
AllowReserved low.NodeReference[bool]
|
|
Schema low.NodeReference[*base.SchemaProxy]
|
|
Example low.NodeReference[any]
|
|
Examples low.NodeReference[map[low.KeyReference[string]]low.ValueReference[*base.Example]]
|
|
Content low.NodeReference[map[low.KeyReference[string]]low.ValueReference[*MediaType]]
|
|
Extensions map[low.KeyReference[string]]low.ValueReference[any]
|
|
*low.Reference
|
|
}
|
|
|
|
// FindContent will attempt to locate a MediaType instance using the specified name.
|
|
func (p *Parameter) FindContent(cType string) *low.ValueReference[*MediaType] {
|
|
return low.FindItemInMap[*MediaType](cType, p.Content.Value)
|
|
}
|
|
|
|
// FindExample will attempt to locate a base.Example instance using the specified name.
|
|
func (p *Parameter) FindExample(eType string) *low.ValueReference[*base.Example] {
|
|
return low.FindItemInMap[*base.Example](eType, p.Examples.Value)
|
|
}
|
|
|
|
// FindExtension attempts to locate an extension using the specified name.
|
|
func (p *Parameter) FindExtension(ext string) *low.ValueReference[any] {
|
|
return low.FindItemInMap[any](ext, p.Extensions)
|
|
}
|
|
|
|
// GetExtensions returns all extensions for Parameter.
|
|
func (p *Parameter) GetExtensions() map[low.KeyReference[string]]low.ValueReference[any] {
|
|
return p.Extensions
|
|
}
|
|
|
|
// Build will extract examples, extensions and content/media types.
|
|
func (p *Parameter) Build(ctx context.Context, _, root *yaml.Node, idx *index.SpecIndex) error {
|
|
root = utils.NodeAlias(root)
|
|
utils.CheckForMergeNodes(root)
|
|
p.Reference = new(low.Reference)
|
|
p.Extensions = low.ExtractExtensions(root)
|
|
|
|
// handle example if set.
|
|
_, expLabel, expNode := utils.FindKeyNodeFull(base.ExampleLabel, root.Content)
|
|
if expNode != nil {
|
|
p.Example = low.ExtractExample(expNode, expLabel)
|
|
}
|
|
|
|
// handle schema
|
|
sch, sErr := base.ExtractSchema(ctx, root, idx)
|
|
if sErr != nil {
|
|
return sErr
|
|
}
|
|
if sch != nil {
|
|
p.Schema = *sch
|
|
}
|
|
|
|
// handle examples if set.
|
|
exps, expsL, expsN, eErr := low.ExtractMap[*base.Example](ctx, base.ExamplesLabel, root, idx)
|
|
if eErr != nil {
|
|
return eErr
|
|
}
|
|
if exps != nil {
|
|
p.Examples = low.NodeReference[map[low.KeyReference[string]]low.ValueReference[*base.Example]]{
|
|
Value: exps,
|
|
KeyNode: expsL,
|
|
ValueNode: expsN,
|
|
}
|
|
}
|
|
|
|
// handle content, if set.
|
|
con, cL, cN, cErr := low.ExtractMap[*MediaType](ctx, ContentLabel, root, idx)
|
|
if cErr != nil {
|
|
return cErr
|
|
}
|
|
p.Content = low.NodeReference[map[low.KeyReference[string]]low.ValueReference[*MediaType]]{
|
|
Value: con,
|
|
KeyNode: cL,
|
|
ValueNode: cN,
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Hash will return a consistent SHA256 Hash of the Parameter object
|
|
func (p *Parameter) Hash() [32]byte {
|
|
var f []string
|
|
if p.Name.Value != "" {
|
|
f = append(f, p.Name.Value)
|
|
}
|
|
if p.In.Value != "" {
|
|
f = append(f, p.In.Value)
|
|
}
|
|
if p.Description.Value != "" {
|
|
f = append(f, p.Description.Value)
|
|
}
|
|
f = append(f, fmt.Sprint(p.Required.Value))
|
|
f = append(f, fmt.Sprint(p.Deprecated.Value))
|
|
f = append(f, fmt.Sprint(p.AllowEmptyValue.Value))
|
|
if p.Style.Value != "" {
|
|
f = append(f, fmt.Sprint(p.Style.Value))
|
|
}
|
|
f = append(f, fmt.Sprint(p.Explode.Value))
|
|
f = append(f, fmt.Sprint(p.AllowReserved.Value))
|
|
if p.Schema.Value != nil {
|
|
f = append(f, fmt.Sprintf("%x", p.Schema.Value.Schema().Hash()))
|
|
}
|
|
if p.Example.Value != nil {
|
|
f = append(f, fmt.Sprintf("%x", p.Example.Value))
|
|
}
|
|
|
|
var keys []string
|
|
keys = make([]string, len(p.Examples.Value))
|
|
z := 0
|
|
for k := range p.Examples.Value {
|
|
keys[z] = low.GenerateHashString(p.Examples.Value[k].Value)
|
|
z++
|
|
}
|
|
sort.Strings(keys)
|
|
f = append(f, keys...)
|
|
keys = make([]string, len(p.Content.Value))
|
|
z = 0
|
|
for k := range p.Content.Value {
|
|
keys[z] = low.GenerateHashString(p.Content.Value[k].Value)
|
|
z++
|
|
}
|
|
sort.Strings(keys)
|
|
f = append(f, keys...)
|
|
keys = make([]string, len(p.Extensions))
|
|
z = 0
|
|
for k := range p.Extensions {
|
|
keys[z] = fmt.Sprintf("%s-%x", k.Value, sha256.Sum256([]byte(fmt.Sprint(p.Extensions[k].Value))))
|
|
z++
|
|
}
|
|
sort.Strings(keys)
|
|
f = append(f, keys...)
|
|
|
|
return sha256.Sum256([]byte(strings.Join(f, "|")))
|
|
}
|
|
|
|
// IsParameter compliance methods.
|
|
|
|
func (p *Parameter) GetName() *low.NodeReference[string] {
|
|
return &p.Name
|
|
}
|
|
func (p *Parameter) GetIn() *low.NodeReference[string] {
|
|
return &p.In
|
|
}
|
|
func (p *Parameter) GetDescription() *low.NodeReference[string] {
|
|
return &p.Description
|
|
}
|
|
func (p *Parameter) GetRequired() *low.NodeReference[bool] {
|
|
return &p.Required
|
|
}
|
|
func (p *Parameter) GetDeprecated() *low.NodeReference[bool] {
|
|
return &p.Deprecated
|
|
}
|
|
func (p *Parameter) GetAllowEmptyValue() *low.NodeReference[bool] {
|
|
return &p.AllowEmptyValue
|
|
}
|
|
func (p *Parameter) GetSchema() *low.NodeReference[any] {
|
|
i := low.NodeReference[any]{
|
|
KeyNode: p.Schema.KeyNode,
|
|
ValueNode: p.Schema.ValueNode,
|
|
Value: p.Schema.Value,
|
|
}
|
|
return &i
|
|
}
|
|
func (p *Parameter) GetStyle() *low.NodeReference[string] {
|
|
return &p.Style
|
|
}
|
|
func (p *Parameter) GetAllowReserved() *low.NodeReference[bool] {
|
|
return &p.AllowReserved
|
|
}
|
|
func (p *Parameter) GetExplode() *low.NodeReference[bool] {
|
|
return &p.Explode
|
|
}
|
|
func (p *Parameter) GetExample() *low.NodeReference[any] {
|
|
return &p.Example
|
|
}
|
|
func (p *Parameter) GetExamples() *low.NodeReference[any] {
|
|
i := low.NodeReference[any]{
|
|
KeyNode: p.Examples.KeyNode,
|
|
ValueNode: p.Examples.ValueNode,
|
|
Value: p.Examples.Value,
|
|
}
|
|
return &i
|
|
}
|
|
func (p *Parameter) GetContent() *low.NodeReference[any] {
|
|
c := low.NodeReference[any]{
|
|
KeyNode: p.Content.KeyNode,
|
|
ValueNode: p.Content.ValueNode,
|
|
Value: p.Content.Value,
|
|
}
|
|
return &c
|
|
}
|