test runnable source builder
Этот коммит содержится в:
родитель
4157f01786
коммит
5e5234f81f
3 изменённых файлов: 245 добавлений и 26 удалений
44
builder.go
44
builder.go
|
@ -39,8 +39,8 @@ func (b *builder) hasImport(a *ast.ImportSpec) bool {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
func newBuilder(buildPath string) (*builder, error) {
|
func newBuilderSkel() *builder {
|
||||||
b := &builder{
|
return &builder{
|
||||||
files: make(map[string]*ast.File),
|
files: make(map[string]*ast.File),
|
||||||
fset: token.NewFileSet(),
|
fset: token.NewFileSet(),
|
||||||
tpl: template.Must(template.New("main").Parse(`package main
|
tpl: template.Must(template.New("main").Parse(`package main
|
||||||
|
@ -57,15 +57,20 @@ func main() {
|
||||||
})
|
})
|
||||||
}`)),
|
}`)),
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func newBuilder(buildPath string) (*builder, error) {
|
||||||
|
b := newBuilderSkel()
|
||||||
err := filepath.Walk(buildPath, func(path string, file os.FileInfo, err error) error {
|
err := filepath.Walk(buildPath, func(path string, file os.FileInfo, err error) error {
|
||||||
if file.IsDir() && file.Name() != "." {
|
if file.IsDir() && file.Name() != "." {
|
||||||
return filepath.SkipDir
|
return filepath.SkipDir
|
||||||
}
|
}
|
||||||
if err == nil && strings.HasSuffix(path, ".go") {
|
if err == nil && strings.HasSuffix(path, ".go") {
|
||||||
if err := b.parseFile(path); err != nil {
|
f, err := parser.ParseFile(b.fset, path, nil, 0)
|
||||||
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
b.register(f, path)
|
||||||
}
|
}
|
||||||
return err
|
return err
|
||||||
})
|
})
|
||||||
|
@ -73,11 +78,7 @@ func main() {
|
||||||
return b, err
|
return b, err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *builder) parseFile(path string) error {
|
func (b *builder) register(f *ast.File, path string) {
|
||||||
f, err := parser.ParseFile(b.fset, path, nil, 0)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
// mark godog package as internal
|
// mark godog package as internal
|
||||||
if f.Name.Name == "godog" && !b.Internal {
|
if f.Name.Name == "godog" && !b.Internal {
|
||||||
b.Internal = true
|
b.Internal = true
|
||||||
|
@ -86,8 +87,6 @@ func (b *builder) parseFile(path string) error {
|
||||||
b.registerContexts(f)
|
b.registerContexts(f)
|
||||||
b.deleteImports(f)
|
b.deleteImports(f)
|
||||||
b.files[path] = f
|
b.files[path] = f
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *builder) deleteImports(f *ast.File) {
|
func (b *builder) deleteImports(f *ast.File) {
|
||||||
|
@ -149,7 +148,7 @@ func (b *builder) registerContexts(f *ast.File) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *builder) merge() (*ast.File, error) {
|
func (b *builder) merge() ([]byte, error) {
|
||||||
var buf bytes.Buffer
|
var buf bytes.Buffer
|
||||||
if err := b.tpl.Execute(&buf, b); err != nil {
|
if err := b.tpl.Execute(&buf, b); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
@ -168,7 +167,7 @@ func (b *builder) merge() (*ast.File, error) {
|
||||||
|
|
||||||
ret, err := ast.MergePackageFiles(pkg, 0), nil
|
ret, err := ast.MergePackageFiles(pkg, 0), nil
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return ret, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// @TODO: we reread the file, probably something goes wrong with position
|
// @TODO: we reread the file, probably something goes wrong with position
|
||||||
|
@ -190,7 +189,13 @@ func (b *builder) merge() (*ast.File, error) {
|
||||||
ipath, _ := strconv.Unquote(spec.Path.Value)
|
ipath, _ := strconv.Unquote(spec.Path.Value)
|
||||||
addImport(b.fset, ret, name, ipath)
|
addImport(b.fset, ret, name, ipath)
|
||||||
}
|
}
|
||||||
return ret, nil
|
|
||||||
|
buf.Reset()
|
||||||
|
if err := format.Node(&buf, b.fset, ret); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return buf.Bytes(), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Build creates a runnable Godog executable file
|
// Build creates a runnable Godog executable file
|
||||||
|
@ -209,18 +214,7 @@ func Build() ([]byte, error) {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
merged, err := b.merge()
|
return b.merge()
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
var buf bytes.Buffer
|
|
||||||
|
|
||||||
if err := format.Node(&buf, b.fset, merged); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return buf.Bytes(), nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// taken from https://github.com/golang/tools/blob/master/go/ast/astutil/imports.go#L17
|
// taken from https://github.com/golang/tools/blob/master/go/ast/astutil/imports.go#L17
|
||||||
|
|
225
builder_test.go
Обычный файл
225
builder_test.go
Обычный файл
|
@ -0,0 +1,225 @@
|
||||||
|
package godog
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"go/parser"
|
||||||
|
"go/token"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
var builderMainFile = `
|
||||||
|
package main
|
||||||
|
import "fmt"
|
||||||
|
func main() {
|
||||||
|
fmt.Println("hello")
|
||||||
|
}`
|
||||||
|
|
||||||
|
var builderPackAliases = `
|
||||||
|
package main
|
||||||
|
import (
|
||||||
|
a "fmt"
|
||||||
|
b "fmt"
|
||||||
|
)
|
||||||
|
func Tester() {
|
||||||
|
a.Println("a")
|
||||||
|
b.Println("b")
|
||||||
|
}`
|
||||||
|
|
||||||
|
var builderAnonymousImport = `
|
||||||
|
package main
|
||||||
|
import (
|
||||||
|
_ "github.com/go-sql-driver/mysql"
|
||||||
|
)
|
||||||
|
`
|
||||||
|
|
||||||
|
var builderContextSrc = `
|
||||||
|
package main
|
||||||
|
import (
|
||||||
|
"github.com/DATA-DOG/godog"
|
||||||
|
)
|
||||||
|
|
||||||
|
func myContext(s *godog.Suite) {
|
||||||
|
|
||||||
|
}
|
||||||
|
`
|
||||||
|
|
||||||
|
var builderLibrarySrc = `
|
||||||
|
package lib
|
||||||
|
import "fmt"
|
||||||
|
func test() {
|
||||||
|
fmt.Println("hello")
|
||||||
|
}
|
||||||
|
`
|
||||||
|
|
||||||
|
var builderInternalPackageSrc = `
|
||||||
|
package godog
|
||||||
|
import "fmt"
|
||||||
|
func test() {
|
||||||
|
fmt.Println("hello")
|
||||||
|
}
|
||||||
|
`
|
||||||
|
|
||||||
|
func (b *builder) registerMulti(contents []string) error {
|
||||||
|
for i, c := range contents {
|
||||||
|
f, err := parser.ParseFile(token.NewFileSet(), "", []byte(c), 0)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
b.register(f, fmt.Sprintf("path%d", i))
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *builder) cleanSpacing(src string) string {
|
||||||
|
var lines []string
|
||||||
|
for _, ln := range strings.Split(src, "\n") {
|
||||||
|
if ln == "" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
lines = append(lines, strings.TrimSpace(ln))
|
||||||
|
}
|
||||||
|
return strings.Join(lines, "\n")
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestUsualSourceFileMerge(t *testing.T) {
|
||||||
|
b := newBuilderSkel()
|
||||||
|
err := b.registerMulti([]string{
|
||||||
|
builderMainFile, builderPackAliases, builderAnonymousImport,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("unexpected error: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
data, err := b.merge()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("unexpected error: %s", err)
|
||||||
|
}
|
||||||
|
expected := `package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
a "fmt"
|
||||||
|
b "fmt"
|
||||||
|
"github.com/DATA-DOG/godog"
|
||||||
|
_ "github.com/go-sql-driver/mysql"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
godog.Run(func(suite *godog.Suite) {
|
||||||
|
|
||||||
|
})
|
||||||
|
}
|
||||||
|
func Tester() {
|
||||||
|
a.Println("a")
|
||||||
|
b.Println("b")
|
||||||
|
}`
|
||||||
|
|
||||||
|
actual := string(data)
|
||||||
|
if b.cleanSpacing(expected) != b.cleanSpacing(actual) {
|
||||||
|
t.Fatalf("expected output does not match: %s", actual)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestShouldCallContextOnMerged(t *testing.T) {
|
||||||
|
b := newBuilderSkel()
|
||||||
|
err := b.registerMulti([]string{
|
||||||
|
builderMainFile, builderContextSrc,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("unexpected error: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
data, err := b.merge()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("unexpected error: %s", err)
|
||||||
|
}
|
||||||
|
expected := `package main
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"github.com/DATA-DOG/godog"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
godog.Run(func(suite *godog.Suite) {
|
||||||
|
myContext(suite)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func myContext(s *godog.Suite) {
|
||||||
|
}`
|
||||||
|
|
||||||
|
actual := string(data)
|
||||||
|
if b.cleanSpacing(expected) != b.cleanSpacing(actual) {
|
||||||
|
t.Fatalf("expected output does not match: %s", actual)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestBuildLibraryPackage(t *testing.T) {
|
||||||
|
b := newBuilderSkel()
|
||||||
|
err := b.registerMulti([]string{
|
||||||
|
builderLibrarySrc,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("unexpected error: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
data, err := b.merge()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("unexpected error: %s", err)
|
||||||
|
}
|
||||||
|
expected := `package main
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"github.com/DATA-DOG/godog"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
godog.Run(func(suite *godog.Suite) {
|
||||||
|
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func test() {
|
||||||
|
fmt.Println(
|
||||||
|
"hello",
|
||||||
|
)
|
||||||
|
}`
|
||||||
|
|
||||||
|
actual := string(data)
|
||||||
|
if b.cleanSpacing(expected) != b.cleanSpacing(actual) {
|
||||||
|
t.Fatalf("expected output does not match: %s", actual)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestBuildInternalPackage(t *testing.T) {
|
||||||
|
b := newBuilderSkel()
|
||||||
|
err := b.registerMulti([]string{
|
||||||
|
builderInternalPackageSrc,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("unexpected error: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
data, err := b.merge()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("unexpected error: %s", err)
|
||||||
|
}
|
||||||
|
expected := `package main
|
||||||
|
import "fmt"
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
Run(func(suite *Suite) {
|
||||||
|
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func test() {
|
||||||
|
fmt.Println("hello")
|
||||||
|
}`
|
||||||
|
|
||||||
|
actual := string(data)
|
||||||
|
if b.cleanSpacing(expected) != b.cleanSpacing(actual) {
|
||||||
|
t.Fatalf("expected output does not match: %s", actual)
|
||||||
|
}
|
||||||
|
}
|
|
@ -16,7 +16,7 @@ func assertMatchesTagFilter(tags []string, filter string, t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func Test_tag_filter(t *testing.T) {
|
func TestTagFilter(t *testing.T) {
|
||||||
assertMatchesTagFilter([]string{"wip"}, "@wip", t)
|
assertMatchesTagFilter([]string{"wip"}, "@wip", t)
|
||||||
assertMatchesTagFilter([]string{}, "~@wip", t)
|
assertMatchesTagFilter([]string{}, "~@wip", t)
|
||||||
assertMatchesTagFilter([]string{"one", "two"}, "@two,@three", t)
|
assertMatchesTagFilter([]string{"one", "two"}, "@two,@three", t)
|
||||||
|
|
Загрузка…
Создание таблицы
Сослаться в новой задаче