309 строки
8,4 КиБ
Go
309 строки
8,4 КиБ
Go
package godog
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"go/build"
|
|
"go/parser"
|
|
"go/token"
|
|
"io/ioutil"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"strings"
|
|
"text/template"
|
|
"time"
|
|
"unicode"
|
|
)
|
|
|
|
var compiler = filepath.Join(build.ToolDir, "compile")
|
|
var linker = filepath.Join(build.ToolDir, "link")
|
|
var gopaths = filepath.SplitList(build.Default.GOPATH)
|
|
var goarch = build.Default.GOARCH
|
|
var goos = build.Default.GOOS
|
|
|
|
var godogImportPath = "github.com/DATA-DOG/godog"
|
|
var runnerTemplate = template.Must(template.New("testmain").Parse(`package main
|
|
|
|
import (
|
|
"github.com/DATA-DOG/godog"
|
|
{{if .Contexts}}_test "{{.ImportPath}}"{{end}}
|
|
"os"
|
|
)
|
|
|
|
func main() {
|
|
status := godog.Run(func (suite *godog.Suite) {
|
|
{{range .Contexts}}
|
|
_test.{{ . }}(suite)
|
|
{{end}}
|
|
})
|
|
os.Exit(status)
|
|
}`))
|
|
|
|
// Build creates a test package like go test command.
|
|
// If there are no go files in tested directory, then
|
|
// it simply builds a godog executable to scan features.
|
|
//
|
|
// If there are go test files, it first builds a test
|
|
// package with standard go test command.
|
|
//
|
|
// Finally it generates godog suite executable which
|
|
// registers exported godog contexts from the test files
|
|
// of tested package.
|
|
//
|
|
// Returns the path to generated executable
|
|
func Build() (string, error) {
|
|
abs, err := filepath.Abs(".")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
bin := filepath.Join(abs, "godog.test")
|
|
// suffix with .exe for windows
|
|
if goos == "windows" {
|
|
bin += ".exe"
|
|
}
|
|
|
|
// we allow package to be nil, if godog is run only when
|
|
// there is a feature file in empty directory
|
|
pkg, _ := build.ImportDir(abs, 0)
|
|
src, anyContexts, err := buildTestMain(pkg)
|
|
if err != nil {
|
|
return bin, err
|
|
}
|
|
|
|
workdir := fmt.Sprintf(filepath.Join("%s", "godog-%d"), os.TempDir(), time.Now().UnixNano())
|
|
testdir := workdir
|
|
|
|
// if none of test files exist, or there are no contexts found
|
|
// we will skip test package compilation, since it is useless
|
|
if anyContexts {
|
|
// first of all compile test package dependencies
|
|
// that will save was many compilations for dependencies
|
|
// go does it better
|
|
out, err := exec.Command("go", "test", "-i").CombinedOutput()
|
|
if err != nil {
|
|
return bin, fmt.Errorf("failed to compile package %s:\n%s", pkg.Name, string(out))
|
|
}
|
|
|
|
// let go do the dirty work and compile test
|
|
// package with it's dependencies. Older go
|
|
// versions does not accept existing file output
|
|
// so we create a temporary executable which will
|
|
// removed.
|
|
temp := fmt.Sprintf(filepath.Join("%s", "temp-%d.test"), os.TempDir(), time.Now().UnixNano())
|
|
|
|
// builds and compile the tested package.
|
|
// generated test executable will be removed
|
|
// since we do not need it for godog suite.
|
|
// we also print back the temp WORK directory
|
|
// go has built. We will reuse it for our suite workdir.
|
|
out, err = exec.Command("go", "test", "-c", "-work", "-o", temp).CombinedOutput()
|
|
if err != nil {
|
|
return bin, fmt.Errorf("failed to compile tested package %s:\n%s", pkg.Name, string(out))
|
|
}
|
|
defer os.Remove(temp)
|
|
|
|
// extract go-build temporary directory as our workdir
|
|
workdir = strings.TrimSpace(string(out))
|
|
if !strings.HasPrefix(workdir, "WORK=") {
|
|
return bin, fmt.Errorf("expected WORK dir path, but got: %s", workdir)
|
|
}
|
|
workdir = strings.Replace(workdir, "WORK=", "", 1)
|
|
testdir = filepath.Join(workdir, pkg.ImportPath, "_test")
|
|
} else {
|
|
// still need to create temporary workdir
|
|
if err = os.MkdirAll(testdir, 0755); err != nil {
|
|
return bin, err
|
|
}
|
|
}
|
|
defer os.RemoveAll(workdir)
|
|
|
|
// replace _testmain.go file with our own
|
|
testmain := filepath.Join(testdir, "_testmain.go")
|
|
err = ioutil.WriteFile(testmain, src, 0644)
|
|
if err != nil {
|
|
return bin, err
|
|
}
|
|
|
|
// godog library may not be imported in tested package
|
|
// but we need it for our testmain package.
|
|
// So we look it up in available source paths
|
|
// including vendor directory, supported since 1.5.
|
|
try := maybeVendorPaths(abs)
|
|
for _, d := range build.Default.SrcDirs() {
|
|
try = append(try, filepath.Join(d, godogImportPath))
|
|
}
|
|
godogPkg, err := locatePackage(try)
|
|
if err != nil {
|
|
return bin, err
|
|
}
|
|
|
|
// make sure godog package archive is installed, gherkin
|
|
// will be installed as dependency of godog
|
|
cmd := exec.Command("go", "install", godogPkg.ImportPath)
|
|
cmd.Env = os.Environ()
|
|
out, err := cmd.CombinedOutput()
|
|
if err != nil {
|
|
return bin, fmt.Errorf("failed to install godog package:\n%s", string(out))
|
|
}
|
|
|
|
// collect all possible package dirs, will be
|
|
// used for includes and linker
|
|
pkgDirs := []string{workdir, testdir}
|
|
for _, gopath := range gopaths {
|
|
pkgDirs = append(pkgDirs, filepath.Join(gopath, "pkg", goos+"_"+goarch))
|
|
}
|
|
pkgDirs = uniqStringList(pkgDirs)
|
|
|
|
// compile godog testmain package archive
|
|
// we do not depend on CGO so a lot of checks are not necessary
|
|
testMainPkgOut := filepath.Join(testdir, "main.a")
|
|
args := []string{
|
|
"-o", testMainPkgOut,
|
|
// "-trimpath", workdir,
|
|
"-p", "main",
|
|
"-complete",
|
|
}
|
|
// if godog library is in vendor directory
|
|
// link it with import map
|
|
if i := strings.LastIndex(godogPkg.ImportPath, "vendor/"); i != -1 {
|
|
args = append(args, "-importmap", godogImportPath+"="+godogPkg.ImportPath)
|
|
}
|
|
for _, inc := range pkgDirs {
|
|
args = append(args, "-I", inc)
|
|
}
|
|
args = append(args, "-pack", testmain)
|
|
cmd = exec.Command(compiler, args...)
|
|
cmd.Env = os.Environ()
|
|
out, err = cmd.CombinedOutput()
|
|
if err != nil {
|
|
return bin, fmt.Errorf("failed to compile testmain package:\n%s", string(out))
|
|
}
|
|
|
|
// link test suite executable
|
|
args = []string{
|
|
"-o", bin,
|
|
"-extld", build.Default.Compiler,
|
|
"-buildmode=exe",
|
|
}
|
|
for _, link := range pkgDirs {
|
|
args = append(args, "-L", link)
|
|
}
|
|
args = append(args, testMainPkgOut)
|
|
cmd = exec.Command(linker, args...)
|
|
cmd.Env = os.Environ()
|
|
|
|
out, err = cmd.CombinedOutput()
|
|
if err != nil {
|
|
return bin, fmt.Errorf("failed to link test executable:\n%s", string(out))
|
|
}
|
|
|
|
return bin, nil
|
|
}
|
|
|
|
func locatePackage(try []string) (*build.Package, error) {
|
|
for _, path := range try {
|
|
abs, err := filepath.Abs(path)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
pkg, err := build.ImportDir(abs, 0)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
return pkg, nil
|
|
}
|
|
return nil, fmt.Errorf("failed to find godog package in any of:\n%s", strings.Join(try, "\n"))
|
|
}
|
|
|
|
func uniqStringList(strs []string) (unique []string) {
|
|
uniq := make(map[string]void, len(strs))
|
|
for _, s := range strs {
|
|
if _, ok := uniq[s]; !ok {
|
|
uniq[s] = void{}
|
|
unique = append(unique, s)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
// buildTestMain if given package is valid
|
|
// it scans test files for contexts
|
|
// and produces a testmain source code.
|
|
func buildTestMain(pkg *build.Package) ([]byte, bool, error) {
|
|
var contexts []string
|
|
var importPath string
|
|
name := "main"
|
|
if nil != pkg {
|
|
ctxs, err := processPackageTestFiles(
|
|
pkg.TestGoFiles,
|
|
pkg.XTestGoFiles,
|
|
)
|
|
if err != nil {
|
|
return nil, false, err
|
|
}
|
|
contexts = ctxs
|
|
importPath = pkg.ImportPath
|
|
name = pkg.Name
|
|
}
|
|
|
|
data := struct {
|
|
Name string
|
|
Contexts []string
|
|
ImportPath string
|
|
}{name, contexts, importPath}
|
|
|
|
var buf bytes.Buffer
|
|
if err := runnerTemplate.Execute(&buf, data); err != nil {
|
|
return nil, len(contexts) > 0, err
|
|
}
|
|
return buf.Bytes(), len(contexts) > 0, nil
|
|
}
|
|
|
|
// maybeVendorPaths determines possible vendor paths
|
|
// which goes levels down from given directory
|
|
// until it reaches GOPATH source dir
|
|
func maybeVendorPaths(dir string) []string {
|
|
paths := []string{filepath.Join(dir, "vendor", godogImportPath)}
|
|
for _, gopath := range gopaths {
|
|
if !strings.HasPrefix(dir, gopath) {
|
|
continue
|
|
}
|
|
|
|
for p := filepath.Dir(dir); p != filepath.Join(gopath, "src"); p = filepath.Dir(p) {
|
|
paths = append(paths, filepath.Join(p, "vendor", godogImportPath))
|
|
}
|
|
}
|
|
return paths
|
|
}
|
|
|
|
// processPackageTestFiles runs through ast of each test
|
|
// file pack and looks for godog suite contexts to register
|
|
// on run
|
|
func processPackageTestFiles(packs ...[]string) ([]string, error) {
|
|
var ctxs []string
|
|
fset := token.NewFileSet()
|
|
for _, pack := range packs {
|
|
for _, testFile := range pack {
|
|
node, err := parser.ParseFile(fset, testFile, nil, 0)
|
|
if err != nil {
|
|
return ctxs, err
|
|
}
|
|
|
|
ctxs = append(ctxs, astContexts(node)...)
|
|
}
|
|
}
|
|
var failed []string
|
|
for _, ctx := range ctxs {
|
|
runes := []rune(ctx)
|
|
if unicode.IsLower(runes[0]) {
|
|
expected := append([]rune{unicode.ToUpper(runes[0])}, runes[1:]...)
|
|
failed = append(failed, fmt.Sprintf("%s - should be: %s", ctx, string(expected)))
|
|
}
|
|
}
|
|
if len(failed) > 0 {
|
|
return ctxs, fmt.Errorf("godog contexts must be exported:\n\t%s", strings.Join(failed, "\n\t"))
|
|
}
|
|
return ctxs, nil
|
|
}
|