perkeep/vendor/myitcv.io/gogenerate/gogenerate.go

334 lines
9.0 KiB
Go

// Copyright (c) 2016 Paul Jolly <paul@myitcv.org.uk>, all rights reserved.
// Use of this document is governed by a license found in the LICENSE document.
// Package gogenerate exposes some of the unexported internals of the go generate command as a convenience
// for the authors of go generate generators. See https://github.com/myitcv/gogenerate/wiki/Go-Generate-Notes
// for further notes on such generators. It also exposes some convenience functions that might be useful
// to authors of generators
//
package gogenerate // import "myitcv.io/gogenerate"
import (
"bufio"
"bytes"
"crypto/sha1"
"flag"
"fmt"
"go/build"
"io"
"os"
"path/filepath"
"sort"
"strings"
)
// These constants correspond in name and value to the details given in
// go generate --help
const (
GOARCH = "GOARCH"
GOFILE = "GOFILE"
GOLINE = "GOLINE"
GOOS = "GOOS"
GOPACKAGE = "GOPACKAGE"
GOPATH = "GOPATH"
GoGeneratePrefix = "//go:generate"
)
const (
// genStr is the string used in the prefix of generated files
genStr = "gen"
// sep is the separator used between the parts of a file name; the prefix used to identify
// generated files, the name (body) and the suffix used to identify the generator
sep = "_"
// genFilePrefix is the prefix used on all generated files (which strictly speaking
// is limited to Go files as far as this definition is concerned, but in practice need not be)
genFilePrefix = genStr + sep
)
const (
// FlagLog is the name of the common flag shared between go generate generators
// to control logging verbosity
FlagLog = "gglog"
// FlagLicenseFile is the name of the common flag shared between go generate generators
// to provide a license header file
FlagLicenseFile = "licenseFile"
)
type LogLevel string
// The various log levels supported by the flag returned by LogFlag()
const (
LogInfo LogLevel = "info"
LogWarning LogLevel = "warning"
LogError LogLevel = "error"
LogFatal LogLevel = "fatal"
)
// FileIsGenerated determines wheter the Go file located at path is generated or not
// and if it is generated returns the base name of the generator that generated it
func FileIsGenerated(path string) (string, bool) {
fn := filepath.Base(path)
if !strings.HasPrefix(fn, genFilePrefix) || !strings.HasSuffix(fn, ".go") {
return "", false
}
fn = strings.TrimPrefix(fn, genFilePrefix)
fn = strings.TrimSuffix(fn, ".go")
if strings.HasSuffix(fn, "_test") {
fn = strings.TrimSuffix(fn, "_test")
}
// deals with the edge case gen_.go or gen__test.go
if fn == "" {
return "", false
}
parts := strings.Split(fn, sep)
return parts[len(parts)-1], true
}
// FileGeneratedBy returns true if the base name of the supplied path is a Go file that
// would have been generated by the supplied cmd
func FileGeneratedBy(path string, cmd string) bool {
cmd = filepath.Base(cmd)
c, ok := FileIsGenerated(path)
return ok && c == cmd
}
// NameFileFromFile uses the provided filename as a template and returns a generated filename consistent with
// the provided command
func NameFileFromFile(name string, cmd string) (string, bool) {
dir := filepath.Dir(name)
name = filepath.Base(name)
if !strings.HasSuffix(name, ".go") {
return "", false
}
name = strings.TrimSuffix(name, ".go")
cmd = filepath.Base(cmd)
var res string
if strings.HasSuffix(name, "_test") {
name = strings.TrimSuffix(name, "_test")
res = NameTestFile(name, cmd)
} else {
res = NameFile(name, cmd)
}
return filepath.Join(dir, res), true
}
func nameBase(name string, cmd string) string {
res := genStr
if name != "" {
res += sep + name
}
res += sep + cmd
return res
}
// NameFile returns a file name that conforms with the pattern associated with files
// generated by the provided command
func NameFile(name string, cmd string) string {
cmd = filepath.Base(cmd)
return nameBase(name, cmd) + ".go"
}
// NameTestFile returns a file name that conforms with the pattern associated with files
// generated by the provided command
func NameTestFile(name string, cmd string) string {
cmd = filepath.Base(cmd)
return nameBase(name, cmd) + "_test.go"
}
// LogFlag defines a command line string flag named according to the constant
// FlagLog and returns a pointer to the string the flag sets
func LogFlag() *string {
return flag.String(FlagLog, string(LogFatal), "log level; one of info, warning, error, fatal")
}
// LicenseFileFlag defines a command line string flag named according to the constant
// FlagLicenseFile and returns a pointer ot the string that flag set
func LicenseFileFlag() *string {
return flag.String(FlagLicenseFile, "", "file that contains a license header to be inserted at the top of each generated file")
}
// CommentLicenseHeader is a convenience function to be used in conjunction with LicenseFileFlag;
// if a filename is provided it reads the contents of the file and returns a line-commented transformation
// of the contents with a final blank newline
func CommentLicenseHeader(file *string) (string, error) {
if file == nil || *file == "" {
return "", nil
}
fi, err := os.Open(*file)
if err != nil {
return "", fmt.Errorf("could not open file %q: %v", *file, err)
}
res := bytes.NewBuffer(nil)
lastLineEmpty := false
scanner := bufio.NewScanner(fi)
for scanner.Scan() {
line := strings.TrimSpace(scanner.Text())
lastLineEmpty = line == ""
fmt.Fprintln(res, "//", line)
}
if err := scanner.Err(); err != nil {
return "", fmt.Errorf("failed to scan file %v: %v", *file, err)
}
// ensure we have a space before package
if !lastLineEmpty {
fmt.Fprintln(res)
}
return res.String(), nil
}
// DefaultLogLevel is provided simply as a convenience along with LogFlag to ensure a default LogLevel
// in a flag variable. This is necessary because of the interplay between go generate argument parsing
// and the advice given for log levels via gg.
func DefaultLogLevel(f *string, ll LogLevel) {
if f != nil && *f == "" {
*f = string(ll)
}
}
// FilesContainingCmd returns a sorted list of Go file names (defined by go list as
// GoFiles + CgoFiles + TestGoFiles + XTestGoFiles) in the directory dir that
// contain a command matching any of the provided commands after quote and
// variable expansion (as described by go generate -help). When comparing commands,
// the filepath.Base of each is compared. The file names will, by definition, be
// relative to dir
func FilesContainingCmd(dir string, commands ...string) ([]string, error) {
// clean our commands
nonZero := false
for i, c := range commands {
c = strings.TrimSpace(c)
if c != "" {
nonZero = true
}
commands[i] = filepath.Base(c)
}
if !nonZero {
return nil, nil
}
pkg, err := build.ImportDir(dir, build.IgnoreVendor)
if err != nil {
return nil, err
}
// GoFiles+CgoFiles+TestGoFiles+XTestGoFiles per go list
// these are all relative to path
gofiles := make([]string, 0, len(pkg.GoFiles)+len(pkg.CgoFiles)+len(pkg.TestGoFiles)+len(pkg.XTestGoFiles))
gofiles = append(gofiles, pkg.GoFiles...)
gofiles = append(gofiles, pkg.CgoFiles...)
gofiles = append(gofiles, pkg.TestGoFiles...)
gofiles = append(gofiles, pkg.XTestGoFiles...)
matchMap := make(map[string]struct{})
for _, f := range gofiles {
checkMatch := func(line int, args []string) error {
for _, cmd := range commands {
if filepath.Base(args[0]) == cmd {
matchMap[f] = struct{}{}
}
}
return nil
}
fp := filepath.Join(dir, f)
err = DirFunc(pkg.ImportPath, fp, checkMatch)
if err != nil {
return nil, err
}
}
matches := make([]string, 0, len(matchMap))
for k := range matchMap {
matches = append(matches, k)
}
sort.Sort(byBase(matches))
return matches, nil
}
// WriteIfDiff writes the supplied byte slice to the file identified by
// path in the case that either no file exists at that path or the contents
// of the file differ from the supplied bytes. The returned bool indicates
// whether a write was necessary or not; in case of any errors false is returned
// along with the error
func WriteIfDiff(toWrite []byte, path string) (bool, error) {
write := false
// is there an existing file... and is it any different?
if fi, err := os.Open(path); err == nil {
ex := sha1.New()
_, err := io.Copy(ex, fi)
if err != nil {
return false, fmt.Errorf("could not read existing file %v: %v", path, err)
}
n := sha1.New()
_, err = n.Write(toWrite)
if err != nil {
return false, fmt.Errorf("could not compute hash of new output: %v", err)
}
if !bytes.Equal(n.Sum(nil), ex.Sum(nil)) {
write = true
}
} else {
// problem checking; write
write = true
}
if write {
of, err := os.Create(path)
if err != nil {
return false, fmt.Errorf("unable to create output file %v: %v", path, err)
}
_, err = of.Write(toWrite)
of.Close()
if err != nil {
return false, fmt.Errorf("unable to close output file %v: %v", path, err)
}
}
return write, nil
}
type byBase []string
func (f byBase) Len() int { return len(f) }
func (f byBase) Less(i, j int) bool { return filepath.Base(f[i]) < filepath.Base(f[j]) }
func (f byBase) Swap(i, j int) { f[i], f[j] = f[j], f[i] }