Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Internal] Log a warning when declaring inline entities #994

Merged
merged 3 commits into from
Jul 23, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
139 changes: 74 additions & 65 deletions openapi/code/package.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ package code
import (
"context"
"fmt"
"log"
"regexp"
"sort"
"strings"
Expand Down Expand Up @@ -119,38 +120,9 @@ func (pkg *Package) ImportedPackages() (res []string) {
// processedEntities keeps track of the entities that are being generated to avoid infinite recursion.
func (pkg *Package) schemaToEntity(s *openapi.Schema, path []string, hasName bool, processedEntities map[string]*Entity) *Entity {
if s.IsRef() {
pair := strings.Split(s.Component(), ".")
if len(pair) == 2 && pair[0] != pkg.Name {
schemaPackage := pair[0]
schemaType := pair[1]
if pkg.extImports == nil {
pkg.extImports = map[string]*Entity{}
}
known, ok := pkg.extImports[s.Component()]
if ok {
return known
}
// referred entity is declared in another package
pkg.extImports[s.Component()] = &Entity{
Named: Named{
Name: schemaType,
},
Package: &Package{
Named: Named{
Name: schemaPackage,
},
},
}
return pkg.extImports[s.Component()]
}
// if schema is src, load it to this package
src := pkg.Components.Schemas.Resolve(s)
if src == nil {
return nil
}
component := pkg.localComponent(&s.Node)
return pkg.definedEntity(component, *src, processedEntities)
return pkg.refEntity(s, processedEntities)
}

e := &Entity{
Named: Named{
Description: s.Description,
Expand All @@ -162,40 +134,80 @@ func (pkg *Package) schemaToEntity(s *openapi.Schema, path []string, hasName boo
if s.JsonPath != "" {
processedEntities[s.JsonPath] = e
}
// pull embedded types up, if they can be defined at package level
if s.IsDefinable() && !hasName {
renaudhartert-db marked this conversation as resolved.
Show resolved Hide resolved
// TODO: log message or panic when overrides a type
e.Named.Name = strings.Join(path, "")

// Some entities are declared anonymously as part of another entity (e.g.
// an object in an object). This is not a recommended pattern but we need
// to handle it. We do that by declaring the entity as if it was explicitly
// defined, using its path as name.
if (s.IsObject() || s.IsEnum()) && !hasName {
fieldType := "enum"
if s.IsObject() {
fieldType = "object"
}
name := strings.Join(path, "")
log.Printf("[WARN] Found anonymous %s %q. Please update your OpenAPI schema so that this entity is explicitly defined in components.", fieldType, name)
e.Named.Name = name
pkg.define(e)
}

e.fields = map[string]*Field{}
e.IsAny = s.IsAny
e.IsComputed = s.IsComputed
e.RequiredOrder = s.Required
// enum
if len(s.Enum) != 0 {

switch {
case len(s.Enum) != 0:
return pkg.makeEnum(e, s, path)
}
// object
if len(s.Properties) != 0 {
case len(s.Properties) != 0:
return pkg.makeObject(e, s, path, processedEntities)
}
// array
if s.ArrayValue != nil {
case s.ArrayValue != nil:
e.ArrayValue = pkg.schemaToEntity(s.ArrayValue, append(path, "Item"), false, processedEntities)
return e
}
// map
if s.MapValue != nil {
case s.MapValue != nil:
e.MapValue = pkg.schemaToEntity(s.MapValue, path, hasName, processedEntities)
return e
default:
e.IsBool = s.Type == "boolean" || s.Type == "bool"
e.IsString = s.Type == "string"
e.IsInt64 = s.Type == "integer" && s.Format == "int64"
e.IsFloat64 = s.Type == "number" && s.Format == "double"
e.IsInt = s.Type == "integer" || s.Type == "int"
return e
}
e.IsBool = s.Type == "boolean" || s.Type == "bool"
e.IsString = s.Type == "string"
e.IsInt64 = s.Type == "integer" && s.Format == "int64"
e.IsFloat64 = s.Type == "number" && s.Format == "double"
e.IsInt = s.Type == "integer" || s.Type == "int"
return e
}

func (pkg *Package) refEntity(s *openapi.Schema, processedEntities map[string]*Entity) *Entity {
pair := strings.Split(s.Component(), ".")
if len(pair) == 2 && pair[0] != pkg.Name {
schemaPackage := pair[0]
schemaType := pair[1]
if pkg.extImports == nil {
pkg.extImports = map[string]*Entity{}
}
known, ok := pkg.extImports[s.Component()]
if ok {
return known
}
// referred entity is declared in another package
pkg.extImports[s.Component()] = &Entity{
Named: Named{
Name: schemaType,
},
Package: &Package{
Named: Named{
Name: schemaPackage,
},
},
}
return pkg.extImports[s.Component()]
}
// if schema is src, load it to this package
src := pkg.Components.Schemas.Resolve(s)
if src == nil {
return nil
}
component := pkg.localComponent(&s.Node)
return pkg.definedEntity(component, *src, processedEntities)
}

// makeObject converts OpenAPI Schema into type representation
Expand Down Expand Up @@ -255,23 +267,19 @@ func (pkg *Package) localComponent(n *openapi.Node) string {
// processedEntities keeps track of the entities that are being generated to avoid infinite recursion.
func (pkg *Package) definedEntity(name string, s *openapi.Schema, processedEntities map[string]*Entity) *Entity {
if s == nil {
entity := &Entity{
Named: Named{
Name: name,
Description: "",
},
return pkg.define(&Entity{
Named: Named{Name: name, Description: ""},
fields: map[string]*Field{},
}
return pkg.define(entity)
})
}

// Return existing entity if it has already been generated.
if entity, ok := processedEntities[s.JsonPath]; ok {
// Return existing entity if it's already being generated.
return entity
}

e := pkg.schemaToEntity(s, []string{name}, true, processedEntities)
if e == nil {
// gets here when responses are objects with no properties
if e == nil { // happens when responses have no properties
return nil
}
if e.ArrayValue != nil {
Expand All @@ -280,14 +288,13 @@ func (pkg *Package) definedEntity(name string, s *openapi.Schema, processedEntit
if e.Name == "" {
e.Named = Named{name, s.Description}
}

return pkg.define(e)
}

func (pkg *Package) define(entity *Entity) *Entity {
k := entity.PascalName()
_, defined := pkg.types[k]
if defined {
//panic(fmt.Sprintf("%s is already defined", entity.Name))
if _, ok := pkg.types[k]; ok {
return entity
}
if entity.Package == nil {
Expand Down Expand Up @@ -354,13 +361,15 @@ func (pkg *Package) getService(tag *openapi.Tag) *Service {
// Load takes OpenAPI specification and loads a service model
func (pkg *Package) Load(ctx context.Context, spec *openapi.Specification, tag openapi.Tag) error {
svc := pkg.getService(&tag)

for k, v := range spec.Components.Schemas {
split := strings.Split(k, ".")
if split[0] != pkg.Name {
continue
}
pkg.definedEntity(split[1], *v, map[string]*Entity{})
}

// Fill in subservice information
if tag.ParentService != "" {
parentTag, err := spec.GetTagByServiceName(tag.ParentService)
Expand Down
6 changes: 0 additions & 6 deletions openapi/model.go
Original file line number Diff line number Diff line change
Expand Up @@ -272,12 +272,6 @@ func (s *Schema) IsObject() bool {
return len(s.Properties) != 0
}

// IsDefinable states that type could be translated into a valid top-level type
// in Go, Python, Java, Scala, and JavaScript
func (s *Schema) IsDefinable() bool {
return s.IsObject() || s.IsEnum()
}

func (s *Schema) IsMap() bool {
return s.MapValue != nil
}
Expand Down
Loading