mirror of
https://github.com/LukeHagar/libopenapi.git
synced 2025-12-06 04:20:11 +00:00
379 lines
15 KiB
Go
379 lines
15 KiB
Go
// Copyright 2022 Princess B33f Heavy Industries / Dave Shanley
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
// Package libopenapi is a library containing tools for reading and in and manipulating Swagger (OpenAPI 2) and OpenAPI 3+
|
|
// specifications into strongly typed documents. These documents have two APIs, a high level (porcelain) and a
|
|
// low level (plumbing).
|
|
//
|
|
// Every single type has a 'GoLow()' method that drops down from the high API to the low API. Once in the low API,
|
|
// the entire original document data is available, including all comments, line and column numbers for keys and values.
|
|
//
|
|
// There are two steps to creating a using Document. First, create a new Document using the NewDocument() method
|
|
// and pass in a specification []byte array that contains the OpenAPI Specification. It doesn't matter if YAML or JSON
|
|
// are used.
|
|
package libopenapi
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
|
|
"github.com/pb33f/libopenapi/index"
|
|
|
|
"github.com/pb33f/libopenapi/datamodel"
|
|
v2high "github.com/pb33f/libopenapi/datamodel/high/v2"
|
|
v3high "github.com/pb33f/libopenapi/datamodel/high/v3"
|
|
v2low "github.com/pb33f/libopenapi/datamodel/low/v2"
|
|
v3low "github.com/pb33f/libopenapi/datamodel/low/v3"
|
|
"github.com/pb33f/libopenapi/utils"
|
|
what_changed "github.com/pb33f/libopenapi/what-changed"
|
|
"github.com/pb33f/libopenapi/what-changed/model"
|
|
"gopkg.in/yaml.v3"
|
|
)
|
|
|
|
// Document Represents an OpenAPI specification that can then be rendered into a model or serialized back into
|
|
// a string document after being manipulated.
|
|
type Document interface {
|
|
// GetVersion will return the exact version of the OpenAPI specification set for the document.
|
|
GetVersion() string
|
|
|
|
// GetRolodex will return the Rolodex instance that was used to load the document.
|
|
GetRolodex() *index.Rolodex
|
|
|
|
// GetSpecInfo will return the *datamodel.SpecInfo instance that contains all specification information.
|
|
GetSpecInfo() *datamodel.SpecInfo
|
|
|
|
// SetConfiguration will set the configuration for the document. This allows for finer grained control over
|
|
// allowing remote or local references, as well as a BaseURL to allow for relative file references.
|
|
SetConfiguration(configuration *datamodel.DocumentConfiguration)
|
|
|
|
// GetConfiguration will return the configuration for the document. This allows for finer grained control over
|
|
// allowing remote or local references, as well as a BaseURL to allow for relative file references.
|
|
GetConfiguration() *datamodel.DocumentConfiguration
|
|
|
|
// BuildV2Model will build out a Swagger (version 2) model from the specification used to create the document
|
|
// If there are any issues, then no model will be returned, instead a slice of errors will explain all the
|
|
// problems that occurred. This method will only support version 2 specifications and will throw an error for
|
|
// any other types.
|
|
BuildV2Model() (*DocumentModel[v2high.Swagger], []error)
|
|
|
|
// BuildV3Model will build out an OpenAPI (version 3+) model from the specification used to create the document
|
|
// If there are any issues, then no model will be returned, instead a slice of errors will explain all the
|
|
// problems that occurred. This method will only support version 3 specifications and will throw an error for
|
|
// any other types.
|
|
BuildV3Model() (*DocumentModel[v3high.Document], []error)
|
|
|
|
// RenderAndReload will render the high level model as it currently exists (including any mutations, additions
|
|
// and removals to and from any object in the tree). It will then reload the low level model with the new bytes
|
|
// extracted from the model that was re-rendered. This is useful if you want to make changes to the high level model
|
|
// and then 'reload' the model into memory, so that line numbers and column numbers are correct and all update
|
|
// according to the changes made.
|
|
//
|
|
// The method returns the raw YAML bytes that were rendered, and any errors that occurred during rebuilding of the model.
|
|
// This is a destructive operation, and will re-build the entire model from scratch using the new bytes, so any
|
|
// references to the old model will be lost. The second return is the new Document that was created, and the third
|
|
// return is any errors hit trying to re-render.
|
|
//
|
|
// **IMPORTANT** This method only supports OpenAPI Documents. The Swagger model will not support mutations correctly
|
|
// and will not update when called. This choice has been made because we don't want to continue supporting Swagger,
|
|
// it's too old, so it should be motivation to upgrade to OpenAPI 3.
|
|
RenderAndReload() ([]byte, Document, *DocumentModel[v3high.Document], []error)
|
|
|
|
// Render will render the high level model as it currently exists (including any mutations, additions
|
|
// and removals to and from any object in the tree). Unlike RenderAndReload, Render will simply print the state
|
|
// of the model as it currently exists, and will not re-load the model into memory. It means that the low-level and
|
|
// the high-level models will be out of sync, and the index will only be useful for the original document.
|
|
//
|
|
// Why use this instead of RenderAndReload?
|
|
//
|
|
// The simple answer is that RenderAndReload is a destructive operation, and will re-build the entire model from
|
|
// scratch using the new bytes, which is desirable if you want to make changes to the high level model and then
|
|
// 'reload' the model into memory, so that line numbers and column numbers are correct and the index is accurate.
|
|
// However, if you don't care about the low-level model, and you're not using the index, and you just want to
|
|
// print the state of the model as it currently exists, then Render() is the method to use.
|
|
// **IMPORTANT** This method only supports OpenAPI Documents.
|
|
Render() ([]byte, error)
|
|
|
|
// Serialize will re-render a Document back into a []byte slice. If any modifications have been made to the
|
|
// underlying data model using low level APIs, then those changes will be reflected in the serialized output.
|
|
//
|
|
// It's important to know that this should not be used if the resolver has been used on a specification to
|
|
// for anything other than checking for circular references. If the resolver is used to resolve the spec, then this
|
|
// method may spin out forever if the specification backing the model has circular references.
|
|
// Deprecated: This method is deprecated and will be removed in a future release. Use RenderAndReload() instead.
|
|
// This method does not support mutations correctly.
|
|
Serialize() ([]byte, error)
|
|
}
|
|
|
|
type document struct {
|
|
rolodex *index.Rolodex
|
|
version string
|
|
info *datamodel.SpecInfo
|
|
config *datamodel.DocumentConfiguration
|
|
highOpenAPI3Model *DocumentModel[v3high.Document]
|
|
highSwaggerModel *DocumentModel[v2high.Swagger]
|
|
}
|
|
|
|
// DocumentModel represents either a Swagger document (version 2) or an OpenAPI document (version 3) that is
|
|
// built from a parent Document.
|
|
type DocumentModel[T v2high.Swagger | v3high.Document] struct {
|
|
Model T
|
|
Index *index.SpecIndex // index created from the document.
|
|
}
|
|
|
|
// NewDocument will create a new OpenAPI instance from an OpenAPI specification []byte array. If anything goes
|
|
// wrong when parsing, reading or processing the OpenAPI specification, there will be no document returned, instead
|
|
// a slice of errors will be returned that explain everything that failed.
|
|
//
|
|
// After creating a Document, the option to build a model becomes available, in either V2 or V3 flavors. The models
|
|
// are about 70% different between Swagger and OpenAPI 3, which is why two different models are available.
|
|
//
|
|
// This function will automatically follow (meaning load) any file or remote references that are found anywhere.
|
|
// Which means recursively also, like a spider, it will follow every reference found, local or remote.
|
|
//
|
|
// If this isn't the behavior you want, or that you feel this is a potential security risk,
|
|
// then you can use the NewDocumentWithConfiguration() function instead, which allows you to set a configuration that
|
|
// will allow you to control if file or remote references are allowed.
|
|
func NewDocument(specByteArray []byte) (Document, error) {
|
|
return NewDocumentWithTypeCheck(specByteArray, false)
|
|
}
|
|
|
|
func NewDocumentWithTypeCheck(specByteArray []byte, bypassCheck bool) (Document, error) {
|
|
info, err := datamodel.ExtractSpecInfoWithDocumentCheck(specByteArray, bypassCheck)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
d := new(document)
|
|
d.version = info.Version
|
|
d.info = info
|
|
return d, nil
|
|
}
|
|
|
|
// NewDocumentWithConfiguration is the same as NewDocument, except it's a convenience function that calls NewDocument
|
|
// under the hood and then calls SetConfiguration() on the returned Document.
|
|
func NewDocumentWithConfiguration(specByteArray []byte, configuration *datamodel.DocumentConfiguration) (Document, error) {
|
|
var d Document
|
|
var err error
|
|
if configuration != nil && configuration.BypassDocumentCheck {
|
|
d, err = NewDocumentWithTypeCheck(specByteArray, true)
|
|
} else {
|
|
d, err = NewDocument(specByteArray)
|
|
}
|
|
|
|
if d != nil {
|
|
d.SetConfiguration(configuration)
|
|
}
|
|
return d, err
|
|
}
|
|
|
|
func (d *document) GetRolodex() *index.Rolodex {
|
|
return d.rolodex
|
|
}
|
|
|
|
func (d *document) GetVersion() string {
|
|
return d.version
|
|
}
|
|
|
|
func (d *document) GetSpecInfo() *datamodel.SpecInfo {
|
|
return d.info
|
|
}
|
|
|
|
func (d *document) GetConfiguration() *datamodel.DocumentConfiguration {
|
|
return d.config
|
|
}
|
|
|
|
func (d *document) SetConfiguration(configuration *datamodel.DocumentConfiguration) {
|
|
d.config = configuration
|
|
}
|
|
|
|
func (d *document) Serialize() ([]byte, error) {
|
|
if d.info == nil {
|
|
return nil, fmt.Errorf("unable to serialize, document has not yet been initialized")
|
|
}
|
|
if d.info.SpecFileType == datamodel.YAMLFileType {
|
|
return yaml.Marshal(d.info.RootNode)
|
|
} else {
|
|
yamlData, _ := yaml.Marshal(d.info.RootNode)
|
|
return utils.ConvertYAMLtoJSON(yamlData)
|
|
}
|
|
}
|
|
|
|
func (d *document) RenderAndReload() ([]byte, Document, *DocumentModel[v3high.Document], []error) {
|
|
|
|
newBytes, rerr := d.Render()
|
|
if rerr != nil {
|
|
return nil, nil, nil, []error{rerr}
|
|
}
|
|
|
|
var errs []error
|
|
|
|
newDoc, err := NewDocumentWithConfiguration(newBytes, d.config)
|
|
errs = append(errs, err)
|
|
|
|
// build the model.
|
|
m, buildErrs := newDoc.BuildV3Model()
|
|
if buildErrs != nil {
|
|
return newBytes, newDoc, m, errs
|
|
}
|
|
// this document is now dead, long live the new document!
|
|
return newBytes, newDoc, m, nil
|
|
}
|
|
|
|
func (d *document) Render() ([]byte, error) {
|
|
if d.highSwaggerModel != nil && d.highOpenAPI3Model == nil {
|
|
return nil, errors.New("this method only supports OpenAPI 3 documents, not Swagger")
|
|
}
|
|
|
|
var newBytes []byte
|
|
var jsonErr error
|
|
if d.info.SpecFileType == datamodel.JSONFileType {
|
|
jsonIndent := " "
|
|
i := d.info.OriginalIndentation
|
|
if i > 2 {
|
|
for l := 0; l < i-2; l++ {
|
|
jsonIndent += " "
|
|
}
|
|
}
|
|
newBytes, jsonErr = d.highOpenAPI3Model.Model.RenderJSON(jsonIndent)
|
|
}
|
|
if d.info.SpecFileType == datamodel.YAMLFileType {
|
|
newBytes = d.highOpenAPI3Model.Model.RenderWithIndention(d.info.OriginalIndentation)
|
|
}
|
|
return newBytes, jsonErr
|
|
}
|
|
|
|
func (d *document) BuildV2Model() (*DocumentModel[v2high.Swagger], []error) {
|
|
if d.highSwaggerModel != nil {
|
|
return d.highSwaggerModel, nil
|
|
}
|
|
var errs []error
|
|
if d.info == nil {
|
|
errs = append(errs, fmt.Errorf("unable to build swagger document, no specification has been loaded"))
|
|
return nil, errs
|
|
}
|
|
if d.info.SpecFormat != datamodel.OAS2 {
|
|
errs = append(errs, fmt.Errorf("unable to build swagger document, "+
|
|
"supplied spec is a different version (%v). Try 'BuildV3Model()'", d.info.SpecFormat))
|
|
return nil, errs
|
|
}
|
|
|
|
var lowDoc *v2low.Swagger
|
|
if d.config == nil {
|
|
d.config = datamodel.NewDocumentConfiguration()
|
|
}
|
|
|
|
var docErr error
|
|
lowDoc, docErr = v2low.CreateDocumentFromConfig(d.info, d.config)
|
|
d.rolodex = lowDoc.Rolodex
|
|
|
|
if docErr != nil {
|
|
errs = append(errs, utils.UnwrapErrors(docErr)...)
|
|
}
|
|
|
|
// Do not short-circuit on circular reference errors, so the client
|
|
// has the option of ignoring them.
|
|
for _, err := range errs {
|
|
var refErr *index.ResolvingError
|
|
if errors.As(err, &refErr) {
|
|
if refErr.CircularReference == nil {
|
|
return nil, errs
|
|
}
|
|
}
|
|
}
|
|
highDoc := v2high.NewSwaggerDocument(lowDoc)
|
|
|
|
d.highSwaggerModel = &DocumentModel[v2high.Swagger]{
|
|
Model: *highDoc,
|
|
Index: lowDoc.Index,
|
|
}
|
|
return d.highSwaggerModel, errs
|
|
}
|
|
|
|
func (d *document) BuildV3Model() (*DocumentModel[v3high.Document], []error) {
|
|
if d.highOpenAPI3Model != nil {
|
|
return d.highOpenAPI3Model, nil
|
|
}
|
|
var errs []error
|
|
if d.info == nil {
|
|
errs = append(errs, fmt.Errorf("unable to build document, no specification has been loaded"))
|
|
return nil, errs
|
|
}
|
|
if d.info.SpecFormat != datamodel.OAS3 {
|
|
errs = append(errs, fmt.Errorf("unable to build openapi document, "+
|
|
"supplied spec is a different version (%v). Try 'BuildV2Model()'", d.info.SpecFormat))
|
|
return nil, errs
|
|
}
|
|
|
|
var lowDoc *v3low.Document
|
|
if d.config == nil {
|
|
d.config = &datamodel.DocumentConfiguration{
|
|
AllowFileReferences: false,
|
|
AllowRemoteReferences: false,
|
|
}
|
|
}
|
|
|
|
var docErr error
|
|
lowDoc, docErr = v3low.CreateDocumentFromConfig(d.info, d.config)
|
|
d.rolodex = lowDoc.Rolodex
|
|
|
|
if docErr != nil {
|
|
errs = append(errs, utils.UnwrapErrors(docErr)...)
|
|
}
|
|
|
|
// Do not short-circuit on circular reference errors, so the client
|
|
// has the option of ignoring them.
|
|
for _, err := range utils.UnwrapErrors(docErr) {
|
|
var refErr *index.ResolvingError
|
|
if errors.As(err, &refErr) {
|
|
if refErr.CircularReference == nil {
|
|
return nil, errs
|
|
}
|
|
}
|
|
}
|
|
|
|
highDoc := v3high.NewDocument(lowDoc)
|
|
highDoc.Rolodex = lowDoc.Index.GetRolodex()
|
|
|
|
d.highOpenAPI3Model = &DocumentModel[v3high.Document]{
|
|
Model: *highDoc,
|
|
Index: lowDoc.Index,
|
|
}
|
|
return d.highOpenAPI3Model, errs
|
|
}
|
|
|
|
// CompareDocuments will accept a left and right Document implementing struct, build a model for the correct
|
|
// version and then compare model documents for changes.
|
|
//
|
|
// If there are any errors when building the models, those errors are returned with a nil pointer for the
|
|
// model.DocumentChanges. If there are any changes found however between either Document, then a pointer to
|
|
// model.DocumentChanges is returned containing every single change, broken down, model by model.
|
|
func CompareDocuments(original, updated Document) (*model.DocumentChanges, []error) {
|
|
var errs []error
|
|
if original.GetSpecInfo().SpecType == utils.OpenApi3 && updated.GetSpecInfo().SpecType == utils.OpenApi3 {
|
|
v3ModelLeft, oErrs := original.BuildV3Model()
|
|
if len(oErrs) > 0 {
|
|
errs = oErrs
|
|
}
|
|
v3ModelRight, uErrs := updated.BuildV3Model()
|
|
if len(uErrs) > 0 {
|
|
errs = append(errs, uErrs...)
|
|
}
|
|
if v3ModelLeft != nil && v3ModelRight != nil {
|
|
return what_changed.CompareOpenAPIDocuments(v3ModelLeft.Model.GoLow(), v3ModelRight.Model.GoLow()), errs
|
|
} else {
|
|
return nil, errs
|
|
}
|
|
}
|
|
if original.GetSpecInfo().SpecType == utils.OpenApi2 && updated.GetSpecInfo().SpecType == utils.OpenApi2 {
|
|
v2ModelLeft, oErrs := original.BuildV2Model()
|
|
if len(oErrs) > 0 {
|
|
errs = oErrs
|
|
}
|
|
v2ModelRight, uErrs := updated.BuildV2Model()
|
|
if len(uErrs) > 0 {
|
|
errs = append(errs, uErrs...)
|
|
}
|
|
return what_changed.CompareSwaggerDocuments(v2ModelLeft.Model.GoLow(), v2ModelRight.Model.GoLow()), errs
|
|
}
|
|
return nil, []error{fmt.Errorf("unable to compare documents, one or both documents are not of the same version")}
|
|
}
|