mirror of
https://github.com/LukeHagar/libopenapi.git
synced 2025-12-06 04:20:11 +00:00
188 lines
5.2 KiB
Go
188 lines
5.2 KiB
Go
// Copyright 2022 Princess B33f Heavy Industries / Dave Shanley
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package v3
|
|
|
|
import (
|
|
"sort"
|
|
|
|
"github.com/pb33f/libopenapi/datamodel/high"
|
|
"github.com/pb33f/libopenapi/datamodel/low"
|
|
v3low "github.com/pb33f/libopenapi/datamodel/low/v3"
|
|
"github.com/pb33f/libopenapi/orderedmap"
|
|
"github.com/pb33f/libopenapi/utils"
|
|
"gopkg.in/yaml.v3"
|
|
)
|
|
|
|
// Paths represents a high-level OpenAPI 3+ Paths object, that is backed by a low-level one.
|
|
//
|
|
// Holds the relative paths to the individual endpoints and their operations. The path is appended to the URL from the
|
|
// Server Object in order to construct the full URL. The Paths MAY be empty, due to Access Control List (ACL)
|
|
// constraints.
|
|
// - https://spec.openapis.org/oas/v3.1.0#paths-object
|
|
type Paths struct {
|
|
PathItems orderedmap.Map[string, *PathItem] `json:"-" yaml:"-"`
|
|
Extensions map[string]any `json:"-" yaml:"-"`
|
|
low *v3low.Paths
|
|
}
|
|
|
|
// NewPaths creates a new high-level instance of Paths from a low-level one.
|
|
func NewPaths(paths *v3low.Paths) *Paths {
|
|
p := new(Paths)
|
|
p.low = paths
|
|
p.Extensions = high.ExtractExtensions(paths.Extensions)
|
|
items := orderedmap.New[string, *PathItem]()
|
|
|
|
type pathItemResult struct {
|
|
key string
|
|
value *PathItem
|
|
}
|
|
|
|
translateFunc := func(pair orderedmap.Pair[low.KeyReference[string], low.ValueReference[*v3low.PathItem]]) (pathItemResult, error) {
|
|
return pathItemResult{key: pair.Key().Value, value: NewPathItem(pair.Value().Value)}, nil
|
|
}
|
|
resultFunc := func(value pathItemResult) error {
|
|
items.Set(value.key, value.value)
|
|
return nil
|
|
}
|
|
_ = orderedmap.TranslateMapParallel[low.KeyReference[string], low.ValueReference[*v3low.PathItem], pathItemResult](
|
|
paths.PathItems, translateFunc, resultFunc,
|
|
)
|
|
p.PathItems = items
|
|
return p
|
|
}
|
|
|
|
// GoLow returns the low-level Paths instance used to create the high-level one.
|
|
func (p *Paths) GoLow() *v3low.Paths {
|
|
return p.low
|
|
}
|
|
|
|
// GoLowUntyped will return the low-level Paths instance that was used to create the high-level one, with no type
|
|
func (p *Paths) GoLowUntyped() any {
|
|
return p.low
|
|
}
|
|
|
|
// Render will return a YAML representation of the Paths object as a byte slice.
|
|
func (p *Paths) Render() ([]byte, error) {
|
|
return yaml.Marshal(p)
|
|
}
|
|
|
|
func (p *Paths) RenderInline() ([]byte, error) {
|
|
d, _ := p.MarshalYAMLInline()
|
|
return yaml.Marshal(d)
|
|
}
|
|
|
|
// MarshalYAML will create a ready to render YAML representation of the Paths object.
|
|
func (p *Paths) MarshalYAML() (interface{}, error) {
|
|
// map keys correctly.
|
|
m := utils.CreateEmptyMapNode()
|
|
type pathItem struct {
|
|
pi *PathItem
|
|
path string
|
|
line int
|
|
rendered *yaml.Node
|
|
}
|
|
var mapped []*pathItem
|
|
|
|
for pair := orderedmap.First(p.PathItems); pair != nil; pair = pair.Next() {
|
|
k := pair.Key()
|
|
pi := pair.Value()
|
|
ln := 9999 // default to a high value to weight new content to the bottom.
|
|
if p.low != nil {
|
|
lpi := p.low.FindPath(k)
|
|
if lpi != nil {
|
|
ln = lpi.ValueNode.Line
|
|
}
|
|
}
|
|
mapped = append(mapped, &pathItem{pi, k, ln, nil})
|
|
}
|
|
|
|
nb := high.NewNodeBuilder(p, p.low)
|
|
extNode := nb.Render()
|
|
if extNode != nil && extNode.Content != nil {
|
|
var label string
|
|
for u := range extNode.Content {
|
|
if u%2 == 0 {
|
|
label = extNode.Content[u].Value
|
|
continue
|
|
}
|
|
mapped = append(mapped, &pathItem{nil, label,
|
|
extNode.Content[u].Line, extNode.Content[u]})
|
|
}
|
|
}
|
|
|
|
sort.Slice(mapped, func(i, j int) bool {
|
|
return mapped[i].line < mapped[j].line
|
|
})
|
|
for j := range mapped {
|
|
if mapped[j].pi != nil {
|
|
rendered, _ := mapped[j].pi.MarshalYAML()
|
|
m.Content = append(m.Content, utils.CreateStringNode(mapped[j].path))
|
|
m.Content = append(m.Content, rendered.(*yaml.Node))
|
|
}
|
|
if mapped[j].rendered != nil {
|
|
m.Content = append(m.Content, utils.CreateStringNode(mapped[j].path))
|
|
m.Content = append(m.Content, mapped[j].rendered)
|
|
}
|
|
}
|
|
|
|
return m, nil
|
|
}
|
|
|
|
func (p *Paths) MarshalYAMLInline() (interface{}, error) {
|
|
// map keys correctly.
|
|
m := utils.CreateEmptyMapNode()
|
|
type pathItem struct {
|
|
pi *PathItem
|
|
path string
|
|
line int
|
|
rendered *yaml.Node
|
|
}
|
|
var mapped []*pathItem
|
|
|
|
for pair := orderedmap.First(p.PathItems); pair != nil; pair = pair.Next() {
|
|
k := pair.Key()
|
|
pi := pair.Value()
|
|
ln := 9999 // default to a high value to weight new content to the bottom.
|
|
if p.low != nil {
|
|
lpi := p.low.FindPath(k)
|
|
if lpi != nil {
|
|
ln = lpi.ValueNode.Line
|
|
}
|
|
}
|
|
mapped = append(mapped, &pathItem{pi, k, ln, nil})
|
|
}
|
|
|
|
nb := high.NewNodeBuilder(p, p.low)
|
|
nb.Resolve = true
|
|
extNode := nb.Render()
|
|
if extNode != nil && extNode.Content != nil {
|
|
var label string
|
|
for u := range extNode.Content {
|
|
if u%2 == 0 {
|
|
label = extNode.Content[u].Value
|
|
continue
|
|
}
|
|
mapped = append(mapped, &pathItem{nil, label,
|
|
extNode.Content[u].Line, extNode.Content[u]})
|
|
}
|
|
}
|
|
|
|
sort.Slice(mapped, func(i, j int) bool {
|
|
return mapped[i].line < mapped[j].line
|
|
})
|
|
for j := range mapped {
|
|
if mapped[j].pi != nil {
|
|
rendered, _ := mapped[j].pi.MarshalYAMLInline()
|
|
m.Content = append(m.Content, utils.CreateStringNode(mapped[j].path))
|
|
m.Content = append(m.Content, rendered.(*yaml.Node))
|
|
}
|
|
if mapped[j].rendered != nil {
|
|
m.Content = append(m.Content, utils.CreateStringNode(mapped[j].path))
|
|
m.Content = append(m.Content, mapped[j].rendered)
|
|
}
|
|
}
|
|
|
|
return m, nil
|
|
}
|