forked from gitea/gitea
1
0
Fork 0
gitea/vendor/github.com/go-swagger/go-swagger/generator/shared.go

1288 lines
37 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 generator
import (
"bytes"
"errors"
"fmt"
"io/ioutil"
"log"
"os"
"path"
"path/filepath"
"reflect"
"regexp"
"sort"
"strings"
"text/template"
"unicode"
swaggererrors "github.com/go-openapi/errors"
"github.com/go-openapi/analysis"
"github.com/go-openapi/loads"
"github.com/go-openapi/spec"
"github.com/go-openapi/strfmt"
"github.com/go-openapi/swag"
"github.com/go-openapi/validate"
"golang.org/x/tools/imports"
)
//go:generate go-bindata -mode 420 -modtime 1482416923 -pkg=generator -ignore=.*\.sw? -ignore=.*\.md ./templates/...
// LanguageOpts to describe a language to the code generator
type LanguageOpts struct {
ReservedWords []string
BaseImportFunc func(string) string `json:"-"`
reservedWordsSet map[string]struct{}
initialized bool
formatFunc func(string, []byte) ([]byte, error)
fileNameFunc func(string) string
}
// Init the language option
func (l *LanguageOpts) Init() {
if !l.initialized {
l.initialized = true
l.reservedWordsSet = make(map[string]struct{})
for _, rw := range l.ReservedWords {
l.reservedWordsSet[rw] = struct{}{}
}
}
}
// MangleName makes sure a reserved word gets a safe name
func (l *LanguageOpts) MangleName(name, suffix string) string {
if _, ok := l.reservedWordsSet[swag.ToFileName(name)]; !ok {
return name
}
return strings.Join([]string{name, suffix}, "_")
}
// MangleVarName makes sure a reserved word gets a safe name
func (l *LanguageOpts) MangleVarName(name string) string {
nm := swag.ToVarName(name)
if _, ok := l.reservedWordsSet[nm]; !ok {
return nm
}
return nm + "Var"
}
// MangleFileName makes sure a file name gets a safe name
func (l *LanguageOpts) MangleFileName(name string) string {
if l.fileNameFunc != nil {
return l.fileNameFunc(name)
}
return swag.ToFileName(name)
}
// ManglePackageName makes sure a package gets a safe name.
// In case of a file system path (e.g. name contains "/" or "\" on Windows), this return only the last element.
func (l *LanguageOpts) ManglePackageName(name, suffix string) string {
if name == "" {
return suffix
}
pth := filepath.ToSlash(filepath.Clean(name)) // preserve path
_, pkg := path.Split(pth) // drop path
return l.MangleName(swag.ToFileName(pkg), suffix)
}
// ManglePackagePath makes sure a full package path gets a safe name.
// Only the last part of the path is altered.
func (l *LanguageOpts) ManglePackagePath(name string, suffix string) string {
if name == "" {
return suffix
}
target := filepath.ToSlash(filepath.Clean(name)) // preserve path
parts := strings.Split(target, "/")
parts[len(parts)-1] = l.ManglePackageName(parts[len(parts)-1], suffix)
return strings.Join(parts, "/")
}
// FormatContent formats a file with a language specific formatter
func (l *LanguageOpts) FormatContent(name string, content []byte) ([]byte, error) {
if l.formatFunc != nil {
return l.formatFunc(name, content)
}
return content, nil
}
func (l *LanguageOpts) baseImport(tgt string) string {
if l.BaseImportFunc != nil {
return l.BaseImportFunc(tgt)
}
return ""
}
var golang = GoLangOpts()
// GoLangOpts for rendering items as golang code
func GoLangOpts() *LanguageOpts {
var goOtherReservedSuffixes = map[string]bool{
// see:
// https://golang.org/src/go/build/syslist.go
// https://golang.org/doc/install/source#environment
// goos
"android": true,
"darwin": true,
"dragonfly": true,
"freebsd": true,
"js": true,
"linux": true,
"nacl": true,
"netbsd": true,
"openbsd": true,
"plan9": true,
"solaris": true,
"windows": true,
"zos": true,
// arch
"386": true,
"amd64": true,
"amd64p32": true,
"arm": true,
"armbe": true,
"arm64": true,
"arm64be": true,
"mips": true,
"mipsle": true,
"mips64": true,
"mips64le": true,
"mips64p32": true,
"mips64p32le": true,
"ppc": true,
"ppc64": true,
"ppc64le": true,
"riscv": true,
"riscv64": true,
"s390": true,
"s390x": true,
"sparc": true,
"sparc64": true,
"wasm": true,
// other reserved suffixes
"test": true,
}
opts := new(LanguageOpts)
opts.ReservedWords = []string{
"break", "default", "func", "interface", "select",
"case", "defer", "go", "map", "struct",
"chan", "else", "goto", "package", "switch",
"const", "fallthrough", "if", "range", "type",
"continue", "for", "import", "return", "var",
}
opts.formatFunc = func(ffn string, content []byte) ([]byte, error) {
opts := new(imports.Options)
opts.TabIndent = true
opts.TabWidth = 2
opts.Fragment = true
opts.Comments = true
return imports.Process(ffn, content, opts)
}
opts.fileNameFunc = func(name string) string {
// whenever a generated file name ends with a suffix
// that is meaningful to go build, adds a "swagger"
// suffix
parts := strings.Split(swag.ToFileName(name), "_")
if goOtherReservedSuffixes[parts[len(parts)-1]] {
// file name ending with a reserved arch or os name
// are appended an innocuous suffix "swagger"
parts = append(parts, "swagger")
}
return strings.Join(parts, "_")
}
opts.BaseImportFunc = func(tgt string) string {
tgt = filepath.Clean(tgt)
// On Windows, filepath.Abs("") behaves differently than on Unix.
// Windows: yields an error, since Abs() does not know the volume.
// UNIX: returns current working directory
if tgt == "" {
tgt = "."
}
tgtAbsPath, err := filepath.Abs(tgt)
if err != nil {
log.Fatalf("could not evaluate base import path with target \"%s\": %v", tgt, err)
}
var tgtAbsPathExtended string
tgtAbsPathExtended, err = filepath.EvalSymlinks(tgtAbsPath)
if err != nil {
log.Fatalf("could not evaluate base import path with target \"%s\" (with symlink resolution): %v", tgtAbsPath, err)
}
gopath := os.Getenv("GOPATH")
if gopath == "" {
gopath = filepath.Join(os.Getenv("HOME"), "go")
}
var pth string
for _, gp := range filepath.SplitList(gopath) {
// EvalSymLinks also calls the Clean
gopathExtended, er := filepath.EvalSymlinks(gp)
if er != nil {
log.Fatalln(er)
}
gopathExtended = filepath.Join(gopathExtended, "src")
gp = filepath.Join(gp, "src")
// At this stage we have expanded and unexpanded target path. GOPATH is fully expanded.
// Expanded means symlink free.
// We compare both types of targetpath<s> with gopath.
// If any one of them coincides with gopath , it is imperative that
// target path lies inside gopath. How?
// - Case 1: Irrespective of symlinks paths coincide. Both non-expanded paths.
// - Case 2: Symlink in target path points to location inside GOPATH. (Expanded Target Path)
// - Case 3: Symlink in target path points to directory outside GOPATH (Unexpanded target path)
// Case 1: - Do nothing case. If non-expanded paths match just generate base import path as if
// there are no symlinks.
// Case 2: - Symlink in target path points to location inside GOPATH. (Expanded Target Path)
// First if will fail. Second if will succeed.
// Case 3: - Symlink in target path points to directory outside GOPATH (Unexpanded target path)
// First if will succeed and break.
//compares non expanded path for both
if ok, relativepath := checkPrefixAndFetchRelativePath(tgtAbsPath, gp); ok {
pth = relativepath
break
}
// Compares non-expanded target path
if ok, relativepath := checkPrefixAndFetchRelativePath(tgtAbsPath, gopathExtended); ok {
pth = relativepath
break
}
// Compares expanded target path.
if ok, relativepath := checkPrefixAndFetchRelativePath(tgtAbsPathExtended, gopathExtended); ok {
pth = relativepath
break
}
}
mod, goModuleAbsPath, err := tryResolveModule(tgtAbsPath)
switch {
case err != nil:
log.Fatalf("Failed to resolve module using go.mod file: %s", err)
case mod != "":
relTgt := relPathToRelGoPath(goModuleAbsPath, tgtAbsPath)
if !strings.HasSuffix(mod, relTgt) {
return mod + relTgt
}
return mod
}
if pth == "" {
log.Fatalln("target must reside inside a location in the $GOPATH/src or be a module")
}
return pth
}
opts.Init()
return opts
}
var moduleRe = regexp.MustCompile(`module[ \t]+([^\s]+)`)
// resolveGoModFile walks up the directory tree starting from 'dir' until it
// finds a go.mod file. If go.mod is found it will return the related file
// object. If no go.mod file is found it will return an error.
func resolveGoModFile(dir string) (*os.File, string, error) {
goModPath := filepath.Join(dir, "go.mod")
f, err := os.Open(goModPath)
if err != nil {
if os.IsNotExist(err) && dir != filepath.Dir(dir) {
return resolveGoModFile(filepath.Dir(dir))
}
return nil, "", err
}
return f, dir, nil
}
// relPathToRelGoPath takes a relative os path and returns the relative go
// package path. For unix nothing will change but for windows \ will be
// converted to /.
func relPathToRelGoPath(modAbsPath, absPath string) string {
if absPath == "." {
return ""
}
path := strings.TrimPrefix(absPath, modAbsPath)
pathItems := strings.Split(path, string(filepath.Separator))
return strings.Join(pathItems, "/")
}
func tryResolveModule(baseTargetPath string) (string, string, error) {
f, goModAbsPath, err := resolveGoModFile(baseTargetPath)
switch {
case os.IsNotExist(err):
return "", "", nil
case err != nil:
return "", "", err
}
src, err := ioutil.ReadAll(f)
if err != nil {
return "", "", err
}
match := moduleRe.FindSubmatch(src)
if len(match) != 2 {
return "", "", nil
}
return string(match[1]), goModAbsPath, nil
}
func findSwaggerSpec(nm string) (string, error) {
specs := []string{"swagger.json", "swagger.yml", "swagger.yaml"}
if nm != "" {
specs = []string{nm}
}
var name string
for _, nn := range specs {
f, err := os.Stat(nn)
if err != nil && !os.IsNotExist(err) {
return "", err
}
if err != nil && os.IsNotExist(err) {
continue
}
if f.IsDir() {
return "", fmt.Errorf("%s is a directory", nn)
}
name = nn
break
}
if name == "" {
return "", errors.New("couldn't find a swagger spec")
}
return name, nil
}
// DefaultSectionOpts for a given opts, this is used when no config file is passed
// and uses the embedded templates when no local override can be found
func DefaultSectionOpts(gen *GenOpts) {
sec := gen.Sections
if len(sec.Models) == 0 {
sec.Models = []TemplateOpts{
{
Name: "definition",
Source: "asset:model",
Target: "{{ joinFilePath .Target (toPackagePath .ModelPackage) }}",
FileName: "{{ (snakize (pascalize .Name)) }}.go",
},
}
}
if len(sec.Operations) == 0 {
if gen.IsClient {
sec.Operations = []TemplateOpts{
{
Name: "parameters",
Source: "asset:clientParameter",
Target: "{{ joinFilePath .Target (toPackagePath .ClientPackage) (toPackagePath .Package) }}",
FileName: "{{ (snakize (pascalize .Name)) }}_parameters.go",
},
{
Name: "responses",
Source: "asset:clientResponse",
Target: "{{ joinFilePath .Target (toPackagePath .ClientPackage) (toPackagePath .Package) }}",
FileName: "{{ (snakize (pascalize .Name)) }}_responses.go",
},
}
} else {
ops := []TemplateOpts{}
if gen.IncludeParameters {
ops = append(ops, TemplateOpts{
Name: "parameters",
Source: "asset:serverParameter",
Target: "{{ if gt (len .Tags) 0 }}{{ joinFilePath .Target (toPackagePath .ServerPackage) (toPackagePath .APIPackage) (toPackagePath .Package) }}{{ else }}{{ joinFilePath .Target (toPackagePath .ServerPackage) (toPackagePath .Package) }}{{ end }}",
FileName: "{{ (snakize (pascalize .Name)) }}_parameters.go",
})
}
if gen.IncludeURLBuilder {
ops = append(ops, TemplateOpts{
Name: "urlbuilder",
Source: "asset:serverUrlbuilder",
Target: "{{ if gt (len .Tags) 0 }}{{ joinFilePath .Target (toPackagePath .ServerPackage) (toPackagePath .APIPackage) (toPackagePath .Package) }}{{ else }}{{ joinFilePath .Target (toPackagePath .ServerPackage) (toPackagePath .Package) }}{{ end }}",
FileName: "{{ (snakize (pascalize .Name)) }}_urlbuilder.go",
})
}
if gen.IncludeResponses {
ops = append(ops, TemplateOpts{
Name: "responses",
Source: "asset:serverResponses",
Target: "{{ if gt (len .Tags) 0 }}{{ joinFilePath .Target (toPackagePath .ServerPackage) (toPackagePath .APIPackage) (toPackagePath .Package) }}{{ else }}{{ joinFilePath .Target (toPackagePath .ServerPackage) (toPackagePath .Package) }}{{ end }}",
FileName: "{{ (snakize (pascalize .Name)) }}_responses.go",
})
}
if gen.IncludeHandler {
ops = append(ops, TemplateOpts{
Name: "handler",
Source: "asset:serverOperation",
Target: "{{ if gt (len .Tags) 0 }}{{ joinFilePath .Target (toPackagePath .ServerPackage) (toPackagePath .APIPackage) (toPackagePath .Package) }}{{ else }}{{ joinFilePath .Target (toPackagePath .ServerPackage) (toPackagePath .Package) }}{{ end }}",
FileName: "{{ (snakize (pascalize .Name)) }}.go",
})
}
sec.Operations = ops
}
}
if len(sec.OperationGroups) == 0 {
if gen.IsClient {
sec.OperationGroups = []TemplateOpts{
{
Name: "client",
Source: "asset:clientClient",
Target: "{{ joinFilePath .Target (toPackagePath .ClientPackage) (toPackagePath .Name)}}",
FileName: "{{ (snakize (pascalize .Name)) }}_client.go",
},
}
} else {
sec.OperationGroups = []TemplateOpts{}
}
}
if len(sec.Application) == 0 {
if gen.IsClient {
sec.Application = []TemplateOpts{
{
Name: "facade",
Source: "asset:clientFacade",
Target: "{{ joinFilePath .Target (toPackagePath .ClientPackage) }}",
FileName: "{{ snakize .Name }}Client.go",
},
}
} else {
sec.Application = []TemplateOpts{
{
Name: "configure",
Source: "asset:serverConfigureapi",
Target: "{{ joinFilePath .Target (toPackagePath .ServerPackage) }}",
FileName: "configure_{{ (snakize (pascalize .Name)) }}.go",
SkipExists: !gen.RegenerateConfigureAPI,
},
{
Name: "main",
Source: "asset:serverMain",
Target: "{{ joinFilePath .Target \"cmd\" (dasherize (pascalize .Name)) }}-server",
FileName: "main.go",
},
{
Name: "embedded_spec",
Source: "asset:swaggerJsonEmbed",
Target: "{{ joinFilePath .Target (toPackagePath .ServerPackage) }}",
FileName: "embedded_spec.go",
},
{
Name: "server",
Source: "asset:serverServer",
Target: "{{ joinFilePath .Target (toPackagePath .ServerPackage) }}",
FileName: "server.go",
},
{
Name: "builder",
Source: "asset:serverBuilder",
Target: "{{ joinFilePath .Target (toPackagePath .ServerPackage) (toPackagePath .APIPackage) }}",
FileName: "{{ snakize (pascalize .Name) }}_api.go",
},
{
Name: "doc",
Source: "asset:serverDoc",
Target: "{{ joinFilePath .Target (toPackagePath .ServerPackage) }}",
FileName: "doc.go",
},
}
}
}
gen.Sections = sec
}
// TemplateOpts allows
type TemplateOpts struct {
Name string `mapstructure:"name"`
Source string `mapstructure:"source"`
Target string `mapstructure:"target"`
FileName string `mapstructure:"file_name"`
SkipExists bool `mapstructure:"skip_exists"`
SkipFormat bool `mapstructure:"skip_format"`
}
// SectionOpts allows for specifying options to customize the templates used for generation
type SectionOpts struct {
Application []TemplateOpts `mapstructure:"application"`
Operations []TemplateOpts `mapstructure:"operations"`
OperationGroups []TemplateOpts `mapstructure:"operation_groups"`
Models []TemplateOpts `mapstructure:"models"`
}
// GenOpts the options for the generator
type GenOpts struct {
IncludeModel bool
IncludeValidator bool
IncludeHandler bool
IncludeParameters bool
IncludeResponses bool
IncludeURLBuilder bool
IncludeMain bool
IncludeSupport bool
ExcludeSpec bool
DumpData bool
ValidateSpec bool
FlattenOpts *analysis.FlattenOpts
IsClient bool
defaultsEnsured bool
PropertiesSpecOrder bool
StrictAdditionalProperties bool
AllowTemplateOverride bool
Spec string
APIPackage string
ModelPackage string
ServerPackage string
ClientPackage string
Principal string
Target string
Sections SectionOpts
LanguageOpts *LanguageOpts
TypeMapping map[string]string
Imports map[string]string
DefaultScheme string
DefaultProduces string
DefaultConsumes string
TemplateDir string
Template string
RegenerateConfigureAPI bool
Operations []string
Models []string
Tags []string
Name string
FlagStrategy string
CompatibilityMode string
ExistingModels string
Copyright string
}
// CheckOpts carries out some global consistency checks on options.
//
// At the moment, these checks simply protect TargetPath() and SpecPath()
// functions. More checks may be added here.
func (g *GenOpts) CheckOpts() error {
if !filepath.IsAbs(g.Target) {
if _, err := filepath.Abs(g.Target); err != nil {
return fmt.Errorf("could not locate target %s: %v", g.Target, err)
}
}
if filepath.IsAbs(g.ServerPackage) {
return fmt.Errorf("you shouldn't specify an absolute path in --server-package: %s", g.ServerPackage)
}
if !filepath.IsAbs(g.Spec) && !strings.HasPrefix(g.Spec, "http://") && !strings.HasPrefix(g.Spec, "https://") {
if _, err := filepath.Abs(g.Spec); err != nil {
return fmt.Errorf("could not locate spec: %s", g.Spec)
}
}
return nil
}
// TargetPath returns the target generation path relative to the server package.
// This method is used by templates, e.g. with {{ .TargetPath }}
//
// Errors cases are prevented by calling CheckOpts beforehand.
//
// Example:
// Target: ${PWD}/tmp
// ServerPackage: abc/efg
//
// Server is generated in ${PWD}/tmp/abc/efg
// relative TargetPath returned: ../../../tmp
//
func (g *GenOpts) TargetPath() string {
var tgt string
if g.Target == "" {
tgt = "." // That's for windows
} else {
tgt = g.Target
}
tgtAbs, _ := filepath.Abs(tgt)
srvPkg := filepath.FromSlash(g.LanguageOpts.ManglePackagePath(g.ServerPackage, "server"))
srvrAbs := filepath.Join(tgtAbs, srvPkg)
tgtRel, _ := filepath.Rel(srvrAbs, filepath.Dir(tgtAbs))
tgtRel = filepath.Join(tgtRel, filepath.Base(tgtAbs))
return tgtRel
}
// SpecPath returns the path to the spec relative to the server package.
// If the spec is remote keep this absolute location.
//
// If spec is not relative to server (e.g. lives on a different drive on windows),
// then the resolved path is absolute.
//
// This method is used by templates, e.g. with {{ .SpecPath }}
//
// Errors cases are prevented by calling CheckOpts beforehand.
func (g *GenOpts) SpecPath() string {
if strings.HasPrefix(g.Spec, "http://") || strings.HasPrefix(g.Spec, "https://") {
return g.Spec
}
// Local specifications
specAbs, _ := filepath.Abs(g.Spec)
var tgt string
if g.Target == "" {
tgt = "." // That's for windows
} else {
tgt = g.Target
}
tgtAbs, _ := filepath.Abs(tgt)
srvPkg := filepath.FromSlash(g.LanguageOpts.ManglePackagePath(g.ServerPackage, "server"))
srvAbs := filepath.Join(tgtAbs, srvPkg)
specRel, err := filepath.Rel(srvAbs, specAbs)
if err != nil {
return specAbs
}
return specRel
}
// EnsureDefaults for these gen opts
func (g *GenOpts) EnsureDefaults() error {
if g.defaultsEnsured {
return nil
}
DefaultSectionOpts(g)
if g.LanguageOpts == nil {
g.LanguageOpts = GoLangOpts()
}
// set defaults for flattening options
g.FlattenOpts = &analysis.FlattenOpts{
Minimal: true,
Verbose: true,
RemoveUnused: false,
Expand: false,
}
g.defaultsEnsured = true
return nil
}
func (g *GenOpts) location(t *TemplateOpts, data interface{}) (string, string, error) {
v := reflect.Indirect(reflect.ValueOf(data))
fld := v.FieldByName("Name")
var name string
if fld.IsValid() {
log.Println("name field", fld.String())
name = fld.String()
}
fldpack := v.FieldByName("Package")
pkg := g.APIPackage
if fldpack.IsValid() {
log.Println("package field", fldpack.String())
pkg = fldpack.String()
}
var tags []string
tagsF := v.FieldByName("Tags")
if tagsF.IsValid() {
tags = tagsF.Interface().([]string)
}
pthTpl, err := template.New(t.Name + "-target").Funcs(FuncMap).Parse(t.Target)
if err != nil {
return "", "", err
}
fNameTpl, err := template.New(t.Name + "-filename").Funcs(FuncMap).Parse(t.FileName)
if err != nil {
return "", "", err
}
d := struct {
Name, Package, APIPackage, ServerPackage, ClientPackage, ModelPackage, Target string
Tags []string
}{
Name: name,
Package: pkg,
APIPackage: g.APIPackage,
ServerPackage: g.ServerPackage,
ClientPackage: g.ClientPackage,
ModelPackage: g.ModelPackage,
Target: g.Target,
Tags: tags,
}
// pretty.Println(data)
var pthBuf bytes.Buffer
if e := pthTpl.Execute(&pthBuf, d); e != nil {
return "", "", e
}
var fNameBuf bytes.Buffer
if e := fNameTpl.Execute(&fNameBuf, d); e != nil {
return "", "", e
}
return pthBuf.String(), fileName(fNameBuf.String()), nil
}
func (g *GenOpts) render(t *TemplateOpts, data interface{}) ([]byte, error) {
var templ *template.Template
if strings.HasPrefix(strings.ToLower(t.Source), "asset:") {
tt, err := templates.Get(strings.TrimPrefix(t.Source, "asset:"))
if err != nil {
return nil, err
}
templ = tt
}
if templ == nil {
// try to load from repository (and enable dependencies)
name := swag.ToJSONName(strings.TrimSuffix(t.Source, ".gotmpl"))
tt, err := templates.Get(name)
if err == nil {
templ = tt
}
}
if templ == nil {
// try to load template from disk, in TemplateDir if specified
// (dependencies resolution is limited to preloaded assets)
var templateFile string
if g.TemplateDir != "" {
templateFile = filepath.Join(g.TemplateDir, t.Source)
} else {
templateFile = t.Source
}
content, err := ioutil.ReadFile(templateFile)
if err != nil {
return nil, fmt.Errorf("error while opening %s template file: %v", templateFile, err)
}
tt, err := template.New(t.Source).Funcs(FuncMap).Parse(string(content))
if err != nil {
return nil, fmt.Errorf("template parsing failed on template %s: %v", t.Name, err)
}
templ = tt
}
if templ == nil {
return nil, fmt.Errorf("template %q not found", t.Source)
}
var tBuf bytes.Buffer
if err := templ.Execute(&tBuf, data); err != nil {
return nil, fmt.Errorf("template execution failed for template %s: %v", t.Name, err)
}
log.Printf("executed template %s", t.Source)
return tBuf.Bytes(), nil
}
// Render template and write generated source code
// generated code is reformatted ("linted"), which gives an
// additional level of checking. If this step fails, the generated
// code is still dumped, for template debugging purposes.
func (g *GenOpts) write(t *TemplateOpts, data interface{}) error {
dir, fname, err := g.location(t, data)
if err != nil {
return fmt.Errorf("failed to resolve template location for template %s: %v", t.Name, err)
}
if t.SkipExists && fileExists(dir, fname) {
debugLog("skipping generation of %s because it already exists and skip_exist directive is set for %s",
filepath.Join(dir, fname), t.Name)
return nil
}
log.Printf("creating generated file %q in %q as %s", fname, dir, t.Name)
content, err := g.render(t, data)
if err != nil {
return fmt.Errorf("failed rendering template data for %s: %v", t.Name, err)
}
if dir != "" {
_, exists := os.Stat(dir)
if os.IsNotExist(exists) {
debugLog("creating directory %q for \"%s\"", dir, t.Name)
// Directory settings consistent with file privileges.
// Environment's umask may alter this setup
if e := os.MkdirAll(dir, 0755); e != nil {
return e
}
}
}
// Conditionally format the code, unless the user wants to skip
formatted := content
var writeerr error
if !t.SkipFormat {
formatted, err = g.LanguageOpts.FormatContent(fname, content)
if err != nil {
log.Printf("source formatting failed on template-generated source (%q for %s). Check that your template produces valid code", filepath.Join(dir, fname), t.Name)
writeerr = ioutil.WriteFile(filepath.Join(dir, fname), content, 0644)
if writeerr != nil {
return fmt.Errorf("failed to write (unformatted) file %q in %q: %v", fname, dir, writeerr)
}
log.Printf("unformatted generated source %q has been dumped for template debugging purposes. DO NOT build on this source!", fname)
return fmt.Errorf("source formatting on generated source %q failed: %v", t.Name, err)
}
}
writeerr = ioutil.WriteFile(filepath.Join(dir, fname), formatted, 0644)
if writeerr != nil {
return fmt.Errorf("failed to write file %q in %q: %v", fname, dir, writeerr)
}
return err
}
func fileName(in string) string {
ext := filepath.Ext(in)
return swag.ToFileName(strings.TrimSuffix(in, ext)) + ext
}
func (g *GenOpts) shouldRenderApp(t *TemplateOpts, app *GenApp) bool {
switch swag.ToFileName(swag.ToGoName(t.Name)) {
case "main":
return g.IncludeMain
case "embedded_spec":
return !g.ExcludeSpec
default:
return true
}
}
func (g *GenOpts) shouldRenderOperations() bool {
return g.IncludeHandler || g.IncludeParameters || g.IncludeResponses
}
func (g *GenOpts) renderApplication(app *GenApp) error {
log.Printf("rendering %d templates for application %s", len(g.Sections.Application), app.Name)
for _, templ := range g.Sections.Application {
if !g.shouldRenderApp(&templ, app) {
continue
}
if err := g.write(&templ, app); err != nil {
return err
}
}
return nil
}
func (g *GenOpts) renderOperationGroup(gg *GenOperationGroup) error {
log.Printf("rendering %d templates for operation group %s", len(g.Sections.OperationGroups), g.Name)
for _, templ := range g.Sections.OperationGroups {
if !g.shouldRenderOperations() {
continue
}
if err := g.write(&templ, gg); err != nil {
return err
}
}
return nil
}
func (g *GenOpts) renderOperation(gg *GenOperation) error {
log.Printf("rendering %d templates for operation %s", len(g.Sections.Operations), g.Name)
for _, templ := range g.Sections.Operations {
if !g.shouldRenderOperations() {
continue
}
if err := g.write(&templ, gg); err != nil {
return err
}
}
return nil
}
func (g *GenOpts) renderDefinition(gg *GenDefinition) error {
log.Printf("rendering %d templates for model %s", len(g.Sections.Models), gg.Name)
for _, templ := range g.Sections.Models {
if !g.IncludeModel {
continue
}
if err := g.write(&templ, gg); err != nil {
return err
}
}
return nil
}
func validateSpec(path string, doc *loads.Document) (err error) {
if doc == nil {
if path, doc, err = loadSpec(path); err != nil {
return err
}
}
result := validate.Spec(doc, strfmt.Default)
if result == nil {
return nil
}
str := fmt.Sprintf("The swagger spec at %q is invalid against swagger specification %s. see errors :\n", path, doc.Version())
for _, desc := range result.(*swaggererrors.CompositeError).Errors {
str += fmt.Sprintf("- %s\n", desc)
}
return errors.New(str)
}
func loadSpec(specFile string) (string, *loads.Document, error) {
// find swagger spec document, verify it exists
specPath := specFile
var err error
if !strings.HasPrefix(specPath, "http") {
specPath, err = findSwaggerSpec(specFile)
if err != nil {
return "", nil, err
}
}
// load swagger spec
specDoc, err := loads.Spec(specPath)
if err != nil {
return "", nil, err
}
return specPath, specDoc, nil
}
func fileExists(target, name string) bool {
_, err := os.Stat(filepath.Join(target, name))
return !os.IsNotExist(err)
}
func gatherModels(specDoc *loads.Document, modelNames []string) (map[string]spec.Schema, error) {
models, mnc := make(map[string]spec.Schema), len(modelNames)
defs := specDoc.Spec().Definitions
if mnc > 0 {
var unknownModels []string
for _, m := range modelNames {
_, ok := defs[m]
if !ok {
unknownModels = append(unknownModels, m)
}
}
if len(unknownModels) != 0 {
return nil, fmt.Errorf("unknown models: %s", strings.Join(unknownModels, ", "))
}
}
for k, v := range defs {
if mnc == 0 {
models[k] = v
}
for _, nm := range modelNames {
if k == nm {
models[k] = v
}
}
}
return models, nil
}
func appNameOrDefault(specDoc *loads.Document, name, defaultName string) string {
if strings.TrimSpace(name) == "" {
if specDoc.Spec().Info != nil && strings.TrimSpace(specDoc.Spec().Info.Title) != "" {
name = specDoc.Spec().Info.Title
} else {
name = defaultName
}
}
return strings.TrimSuffix(strings.TrimSuffix(strings.TrimSuffix(swag.ToGoName(name), "Test"), "API"), "Test")
}
func containsString(names []string, name string) bool {
for _, nm := range names {
if nm == name {
return true
}
}
return false
}
type opRef struct {
Method string
Path string
Key string
ID string
Op *spec.Operation
}
type opRefs []opRef
func (o opRefs) Len() int { return len(o) }
func (o opRefs) Swap(i, j int) { o[i], o[j] = o[j], o[i] }
func (o opRefs) Less(i, j int) bool { return o[i].Key < o[j].Key }
func gatherOperations(specDoc *analysis.Spec, operationIDs []string) map[string]opRef {
var oprefs opRefs
for method, pathItem := range specDoc.Operations() {
for path, operation := range pathItem {
// nm := ensureUniqueName(operation.ID, method, path, operations)
vv := *operation
oprefs = append(oprefs, opRef{
Key: swag.ToGoName(strings.ToLower(method) + " " + path),
Method: method,
Path: path,
ID: vv.ID,
Op: &vv,
})
}
}
sort.Sort(oprefs)
operations := make(map[string]opRef)
for _, opr := range oprefs {
nm := opr.ID
if nm == "" {
nm = opr.Key
}
oo, found := operations[nm]
if found && oo.Method != opr.Method && oo.Path != opr.Path {
nm = opr.Key
}
if len(operationIDs) == 0 || containsString(operationIDs, opr.ID) || containsString(operationIDs, nm) {
opr.ID = nm
opr.Op.ID = nm
operations[nm] = opr
}
}
return operations
}
func pascalize(arg string) string {
runes := []rune(arg)
switch len(runes) {
case 0:
return ""
case 1: // handle special case when we have a single rune that is not handled by swag.ToGoName
switch runes[0] {
case '+', '-', '#', '_': // those cases are handled differently than swag utility
return prefixForName(arg)
}
}
return swag.ToGoName(swag.ToGoName(arg)) // want to remove spaces
}
func prefixForName(arg string) string {
first := []rune(arg)[0]
if len(arg) == 0 || unicode.IsLetter(first) {
return ""
}
switch first {
case '+':
return "Plus"
case '-':
return "Minus"
case '#':
return "HashTag"
// other cases ($,@ etc..) handled by swag.ToGoName
}
return "Nr"
}
func init() {
// this makes the ToGoName func behave with the special
// prefixing rule above
swag.GoNamePrefixFunc = prefixForName
}
func pruneEmpty(in []string) (out []string) {
for _, v := range in {
if v != "" {
out = append(out, v)
}
}
return
}
func trimBOM(in string) string {
return strings.Trim(in, "\xef\xbb\xbf")
}
func validateAndFlattenSpec(opts *GenOpts, specDoc *loads.Document) (*loads.Document, error) {
var err error
// Validate if needed
if opts.ValidateSpec {
log.Printf("validating spec %v", opts.Spec)
if erv := validateSpec(opts.Spec, specDoc); erv != nil {
return specDoc, erv
}
}
// Restore spec to original
opts.Spec, specDoc, err = loadSpec(opts.Spec)
if err != nil {
return nil, err
}
absBasePath := specDoc.SpecFilePath()
if !filepath.IsAbs(absBasePath) {
cwd, _ := os.Getwd()
absBasePath = filepath.Join(cwd, absBasePath)
}
// Some preprocessing is required before codegen
//
// This ensures at least that $ref's in the spec document are canonical,
// i.e all $ref are local to this file and point to some uniquely named definition.
//
// Default option is to ensure minimal flattening of $ref, bundling remote $refs and relocating arbitrary JSON
// pointers as definitions.
// This preprocessing may introduce duplicate names (e.g. remote $ref with same name). In this case, a definition
// suffixed with "OAIGen" is produced.
//
// Full flattening option farther transforms the spec by moving every complex object (e.g. with some properties)
// as a standalone definition.
//
// Eventually, an "expand spec" option is available. It is essentially useful for testing purposes.
//
// NOTE(fredbi): spec expansion may produce some unsupported constructs and is not yet protected against the
// following cases:
// - polymorphic types generation may fail with expansion (expand destructs the reuse intent of the $ref in allOf)
// - name duplicates may occur and result in compilation failures
// The right place to fix these shortcomings is go-openapi/analysis.
opts.FlattenOpts.BasePath = absBasePath // BasePath must be absolute
opts.FlattenOpts.Spec = analysis.New(specDoc.Spec())
var preprocessingOption string
switch {
case opts.FlattenOpts.Expand:
preprocessingOption = "expand"
case opts.FlattenOpts.Minimal:
preprocessingOption = "minimal flattening"
default:
preprocessingOption = "full flattening"
}
log.Printf("preprocessing spec with option: %s", preprocessingOption)
if err = analysis.Flatten(*opts.FlattenOpts); err != nil {
return nil, err
}
// yields the preprocessed spec document
return specDoc, nil
}
// gatherSecuritySchemes produces a sorted representation from a map of spec security schemes
func gatherSecuritySchemes(securitySchemes map[string]spec.SecurityScheme, appName, principal, receiver string) (security GenSecuritySchemes) {
for scheme, req := range securitySchemes {
isOAuth2 := strings.ToLower(req.Type) == "oauth2"
var scopes []string
if isOAuth2 {
for k := range req.Scopes {
scopes = append(scopes, k)
}
}
sort.Strings(scopes)
security = append(security, GenSecurityScheme{
AppName: appName,
ID: scheme,
ReceiverName: receiver,
Name: req.Name,
IsBasicAuth: strings.ToLower(req.Type) == "basic",
IsAPIKeyAuth: strings.ToLower(req.Type) == "apikey",
IsOAuth2: isOAuth2,
Scopes: scopes,
Principal: principal,
Source: req.In,
// from original spec
Description: req.Description,
Type: strings.ToLower(req.Type),
In: req.In,
Flow: req.Flow,
AuthorizationURL: req.AuthorizationURL,
TokenURL: req.TokenURL,
Extensions: req.Extensions,
})
}
sort.Sort(security)
return
}
// gatherExtraSchemas produces a sorted list of extra schemas.
//
// ExtraSchemas are inlined types rendered in the same model file.
func gatherExtraSchemas(extraMap map[string]GenSchema) (extras GenSchemaList) {
var extraKeys []string
for k := range extraMap {
extraKeys = append(extraKeys, k)
}
sort.Strings(extraKeys)
for _, k := range extraKeys {
// figure out if top level validations are needed
p := extraMap[k]
p.HasValidations = shallowValidationLookup(p)
extras = append(extras, p)
}
return
}
func sharedValidationsFromSimple(v spec.CommonValidations, isRequired bool) (sh sharedValidations) {
sh = sharedValidations{
Required: isRequired,
Maximum: v.Maximum,
ExclusiveMaximum: v.ExclusiveMaximum,
Minimum: v.Minimum,
ExclusiveMinimum: v.ExclusiveMinimum,
MaxLength: v.MaxLength,
MinLength: v.MinLength,
Pattern: v.Pattern,
MaxItems: v.MaxItems,
MinItems: v.MinItems,
UniqueItems: v.UniqueItems,
MultipleOf: v.MultipleOf,
Enum: v.Enum,
}
return
}
func sharedValidationsFromSchema(v spec.Schema, isRequired bool) (sh sharedValidations) {
sh = sharedValidations{
Required: isRequired,
Maximum: v.Maximum,
ExclusiveMaximum: v.ExclusiveMaximum,
Minimum: v.Minimum,
ExclusiveMinimum: v.ExclusiveMinimum,
MaxLength: v.MaxLength,
MinLength: v.MinLength,
Pattern: v.Pattern,
MaxItems: v.MaxItems,
MinItems: v.MinItems,
UniqueItems: v.UniqueItems,
MultipleOf: v.MultipleOf,
Enum: v.Enum,
}
return
}