183 lines
5.8 KiB
Go
183 lines
5.8 KiB
Go
|
// Package args has common command-line flags for generation programs.
|
||
|
package args
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"flag"
|
||
|
"fmt"
|
||
|
"io/ioutil"
|
||
|
"os"
|
||
|
"path"
|
||
|
"path/filepath"
|
||
|
"strconv"
|
||
|
"strings"
|
||
|
"time"
|
||
|
|
||
|
"go-common/app/tool/gengo/generator"
|
||
|
"go-common/app/tool/gengo/namer"
|
||
|
"go-common/app/tool/gengo/parser"
|
||
|
"go-common/app/tool/gengo/types"
|
||
|
)
|
||
|
|
||
|
// Default returns a defaulted GeneratorArgs. You may change the defaults
|
||
|
// before calling AddFlags.
|
||
|
func Default() *GeneratorArgs {
|
||
|
return &GeneratorArgs{
|
||
|
OutputBase: DefaultSourceTree(),
|
||
|
GoHeaderFilePath: filepath.Join(DefaultSourceTree(), "go-common/app/tool/gengo/boilerplate/boilerplate.go.txt"),
|
||
|
GeneratedBuildTag: "ignore_autogenerated",
|
||
|
GeneratedByCommentTemplate: "// Code generated by GENERATOR_NAME. DO NOT EDIT.",
|
||
|
defaultCommandLineFlags: true,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// GeneratorArgs has arguments that are passed to generators.
|
||
|
type GeneratorArgs struct {
|
||
|
// Which directories to parse.
|
||
|
InputDirs StringSliceVar
|
||
|
|
||
|
// Source tree to write results to.
|
||
|
OutputBase string
|
||
|
|
||
|
// Package path within the source tree.
|
||
|
OutputPackagePath string
|
||
|
|
||
|
// Output file name.
|
||
|
OutputFileBaseName string
|
||
|
|
||
|
// Where to get copyright header text.
|
||
|
GoHeaderFilePath string
|
||
|
|
||
|
// If GeneratedByCommentTemplate is set, generate a "Code generated by" comment
|
||
|
// below the bloilerplate, of the format defined by this string.
|
||
|
// Any instances of "GENERATOR_NAME" will be replaced with the name of the code generator.
|
||
|
GeneratedByCommentTemplate string
|
||
|
|
||
|
// If true, only verify, don't write anything.
|
||
|
VerifyOnly bool
|
||
|
|
||
|
// GeneratedBuildTag is the tag used to identify code generated by execution
|
||
|
// of this type. Each generator should use a different tag, and different
|
||
|
// groups of generators (external API that depends on Kube generations) should
|
||
|
// keep tags distinct as well.
|
||
|
GeneratedBuildTag string
|
||
|
|
||
|
// Any custom arguments go here
|
||
|
CustomArgs interface{}
|
||
|
|
||
|
// Whether to use default command line flags
|
||
|
defaultCommandLineFlags bool
|
||
|
}
|
||
|
|
||
|
// WithoutDefaultFlagParsing disables implicit addition of command line flags and parsing.
|
||
|
func (g *GeneratorArgs) WithoutDefaultFlagParsing() *GeneratorArgs {
|
||
|
g.defaultCommandLineFlags = false
|
||
|
return g
|
||
|
}
|
||
|
|
||
|
// AddFlags is
|
||
|
func (g *GeneratorArgs) AddFlags(fs *flag.FlagSet) {
|
||
|
fs.Var(&g.InputDirs, "input-dirs", "Comma-separated list of import paths to get input types from.")
|
||
|
fs.StringVar(&g.OutputBase, "output-base", g.OutputBase, "Output base; defaults to $GOPATH/src/ or ./ if $GOPATH is not set.")
|
||
|
fs.StringVar(&g.OutputPackagePath, "output-package", g.OutputPackagePath, "Base package path.")
|
||
|
fs.StringVar(&g.OutputFileBaseName, "output-file-base", g.OutputFileBaseName, "Base name (without .go suffix) for output files.")
|
||
|
fs.StringVar(&g.GoHeaderFilePath, "go-header-file", g.GoHeaderFilePath, "File containing boilerplate header text. The string YEAR will be replaced with the current 4-digit year.")
|
||
|
fs.BoolVar(&g.VerifyOnly, "verify-only", g.VerifyOnly, "If true, only verify existing output, do not write anything.")
|
||
|
fs.StringVar(&g.GeneratedBuildTag, "build-tag", g.GeneratedBuildTag, "A Go build tag to use to identify files generated by this command. Should be unique.")
|
||
|
}
|
||
|
|
||
|
// LoadGoBoilerplate loads the boilerplate file passed to --go-header-file.
|
||
|
func (g *GeneratorArgs) LoadGoBoilerplate() ([]byte, error) {
|
||
|
b, err := ioutil.ReadFile(g.GoHeaderFilePath)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
b = bytes.Replace(b, []byte("YEAR"), []byte(strconv.Itoa(time.Now().Year())), -1)
|
||
|
|
||
|
if g.GeneratedByCommentTemplate != "" {
|
||
|
if len(b) != 0 {
|
||
|
b = append(b, byte('\n'))
|
||
|
}
|
||
|
generatorName := path.Base(os.Args[0])
|
||
|
generatedByComment := strings.Replace(g.GeneratedByCommentTemplate, "GENERATOR_NAME", generatorName, -1)
|
||
|
s := fmt.Sprintf("%s\n\n", generatedByComment)
|
||
|
b = append(b, []byte(s)...)
|
||
|
}
|
||
|
return b, nil
|
||
|
}
|
||
|
|
||
|
// NewBuilder makes a new parser.Builder and populates it with the input
|
||
|
// directories.
|
||
|
func (g *GeneratorArgs) NewBuilder() (*parser.Builder, error) {
|
||
|
b := parser.New()
|
||
|
// Ignore all auto-generated files.
|
||
|
b.AddBuildTags(g.GeneratedBuildTag)
|
||
|
|
||
|
for _, d := range g.InputDirs {
|
||
|
var err error
|
||
|
if strings.HasSuffix(d, "/...") {
|
||
|
err = b.AddDirRecursive(strings.TrimSuffix(d, "/..."))
|
||
|
} else {
|
||
|
err = b.AddDir(d)
|
||
|
}
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("unable to add directory %q: %v", d, err)
|
||
|
}
|
||
|
}
|
||
|
return b, nil
|
||
|
}
|
||
|
|
||
|
// InputIncludes returns true if the given package is a (sub) package of one of
|
||
|
// the InputDirs.
|
||
|
func (g *GeneratorArgs) InputIncludes(p *types.Package) bool {
|
||
|
for _, dir := range g.InputDirs {
|
||
|
d := dir
|
||
|
if strings.HasSuffix(d, "...") {
|
||
|
d = strings.TrimSuffix(d, "...")
|
||
|
}
|
||
|
if strings.HasPrefix(p.Path, d) {
|
||
|
return true
|
||
|
}
|
||
|
}
|
||
|
return false
|
||
|
}
|
||
|
|
||
|
// DefaultSourceTree returns the /src directory of the first entry in $GOPATH.
|
||
|
// If $GOPATH is empty, it returns "./". Useful as a default output location.
|
||
|
func DefaultSourceTree() string {
|
||
|
paths := strings.Split(os.Getenv("GOPATH"), string(filepath.ListSeparator))
|
||
|
if len(paths) > 0 && len(paths[0]) > 0 {
|
||
|
return filepath.Join(paths[0], "src")
|
||
|
}
|
||
|
return "./"
|
||
|
}
|
||
|
|
||
|
// Execute implements main().
|
||
|
// If you don't need any non-default behavior, use as:
|
||
|
// args.Default().Execute(...)
|
||
|
func (g *GeneratorArgs) Execute(nameSystems namer.NameSystems, defaultSystem string, pkgs func(*generator.Context, *GeneratorArgs) generator.Packages) error {
|
||
|
if g.defaultCommandLineFlags {
|
||
|
g.AddFlags(flag.CommandLine)
|
||
|
// flag.CommandLine.AddGoFlagSet(goflag.CommandLine)
|
||
|
flag.Parse()
|
||
|
}
|
||
|
|
||
|
b, err := g.NewBuilder()
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("Failed making a parser: %v", err)
|
||
|
}
|
||
|
|
||
|
c, err := generator.NewContext(b, nameSystems, defaultSystem)
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("Failed making a context: %v", err)
|
||
|
}
|
||
|
|
||
|
c.Verify = g.VerifyOnly
|
||
|
packages := pkgs(c, g)
|
||
|
if err := c.ExecutePackages(g.OutputBase, packages); err != nil {
|
||
|
return fmt.Errorf("Failed executing generator: %v", err)
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|