Add fmt flag to specify formatter (#117)
- Support formatting output with gofmt(default) or goimports via flag. - Introduce moq.Config struct to configure moq.Mocker instance. This breaks backward compatibility but facilitates it in the future if and when any features are added. - Use golden file tests for validating formatters. Use github.com/pmezard/go-difflib to print the diff between expected and actual.
This commit is contained in:
parent
7615cbe602
commit
7721994d1b
@ -24,7 +24,9 @@ $ go get github.com/matryer/moq
|
|||||||
### Usage
|
### Usage
|
||||||
|
|
||||||
```
|
```
|
||||||
moq [flags] destination interface [interface2 [interface3 [...]]]
|
moq [flags] source-dir interface [interface2 [interface3 [...]]]
|
||||||
|
-fmt string
|
||||||
|
go pretty-printer: gofmt (default) or goimports
|
||||||
-out string
|
-out string
|
||||||
output file (default stdout)
|
output file (default stdout)
|
||||||
-pkg string
|
-pkg string
|
||||||
@ -33,6 +35,9 @@ Specifying an alias for the mock is also supported with the format 'interface:al
|
|||||||
Ex: moq -pkg different . MyInterface:MyMock
|
Ex: moq -pkg different . MyInterface:MyMock
|
||||||
```
|
```
|
||||||
|
|
||||||
|
**NOTE:** `source-dir` is the directory where the source code (definition) of the target interface is located.
|
||||||
|
It needs to be a path to a directory and not the import statement for a Go package.
|
||||||
|
|
||||||
In a command line:
|
In a command line:
|
||||||
|
|
||||||
```
|
```
|
||||||
|
19
main.go
19
main.go
@ -14,18 +14,20 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type userFlags struct {
|
type userFlags struct {
|
||||||
outFile string
|
outFile string
|
||||||
pkgName string
|
pkgName string
|
||||||
args []string
|
formatter string
|
||||||
|
args []string
|
||||||
}
|
}
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
var flags userFlags
|
var flags userFlags
|
||||||
flag.StringVar(&flags.outFile, "out", "", "output file (default stdout)")
|
flag.StringVar(&flags.outFile, "out", "", "output file (default stdout)")
|
||||||
flag.StringVar(&flags.pkgName, "pkg", "", "package name (default will infer)")
|
flag.StringVar(&flags.pkgName, "pkg", "", "package name (default will infer)")
|
||||||
|
flag.StringVar(&flags.formatter, "fmt", "", "go pretty-printer: gofmt (default) or goimports")
|
||||||
|
|
||||||
flag.Usage = func() {
|
flag.Usage = func() {
|
||||||
fmt.Println(`moq [flags] destination interface [interface2 [interface3 [...]]]`)
|
fmt.Println(`moq [flags] source-dir interface [interface2 [interface3 [...]]]`)
|
||||||
flag.PrintDefaults()
|
flag.PrintDefaults()
|
||||||
fmt.Println(`Specifying an alias for the mock is also supported with the format 'interface:alias'`)
|
fmt.Println(`Specifying an alias for the mock is also supported with the format 'interface:alias'`)
|
||||||
fmt.Println(`Ex: moq -pkg different . MyInterface:MyMock`)
|
fmt.Println(`Ex: moq -pkg different . MyInterface:MyMock`)
|
||||||
@ -52,9 +54,12 @@ func run(flags userFlags) error {
|
|||||||
out = &buf
|
out = &buf
|
||||||
}
|
}
|
||||||
|
|
||||||
destination := flags.args[0]
|
srcDir, args := flags.args[0], flags.args[1:]
|
||||||
args := flags.args[1:]
|
m, err := moq.New(moq.Config{
|
||||||
m, err := moq.New(destination, flags.pkgName)
|
SrcDir: srcDir,
|
||||||
|
PkgName: flags.pkgName,
|
||||||
|
Formatter: flags.formatter,
|
||||||
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
31
pkg/moq/formatter.go
Normal file
31
pkg/moq/formatter.go
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
package moq
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"go/format"
|
||||||
|
|
||||||
|
"golang.org/x/tools/imports"
|
||||||
|
)
|
||||||
|
|
||||||
|
func goimports(src []byte) ([]byte, error) {
|
||||||
|
formatted, err := imports.Process("filename", src, &imports.Options{
|
||||||
|
TabWidth: 8,
|
||||||
|
TabIndent: true,
|
||||||
|
Comments: true,
|
||||||
|
Fragment: true,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("goimports: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return formatted, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func gofmt(src []byte) ([]byte, error) {
|
||||||
|
formatted, err := format.Source(src)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("go/format: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return formatted, nil
|
||||||
|
}
|
44
pkg/moq/golint_initialisms.go
Normal file
44
pkg/moq/golint_initialisms.go
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
package moq
|
||||||
|
|
||||||
|
// This list comes from the golint codebase. Golint will complain about any of
|
||||||
|
// these being mixed-case, like "Id" instead of "ID".
|
||||||
|
var golintInitialisms = []string{
|
||||||
|
"ACL",
|
||||||
|
"API",
|
||||||
|
"ASCII",
|
||||||
|
"CPU",
|
||||||
|
"CSS",
|
||||||
|
"DNS",
|
||||||
|
"EOF",
|
||||||
|
"GUID",
|
||||||
|
"HTML",
|
||||||
|
"HTTP",
|
||||||
|
"HTTPS",
|
||||||
|
"ID",
|
||||||
|
"IP",
|
||||||
|
"JSON",
|
||||||
|
"LHS",
|
||||||
|
"QPS",
|
||||||
|
"RAM",
|
||||||
|
"RHS",
|
||||||
|
"RPC",
|
||||||
|
"SLA",
|
||||||
|
"SMTP",
|
||||||
|
"SQL",
|
||||||
|
"SSH",
|
||||||
|
"TCP",
|
||||||
|
"TLS",
|
||||||
|
"TTL",
|
||||||
|
"UDP",
|
||||||
|
"UI",
|
||||||
|
"UID",
|
||||||
|
"UUID",
|
||||||
|
"URI",
|
||||||
|
"URL",
|
||||||
|
"UTF8",
|
||||||
|
"VM",
|
||||||
|
"XML",
|
||||||
|
"XMPP",
|
||||||
|
"XSRF",
|
||||||
|
"XSS",
|
||||||
|
}
|
@ -5,7 +5,6 @@ import (
|
|||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"go/build"
|
"go/build"
|
||||||
"go/format"
|
|
||||||
"go/types"
|
"go/types"
|
||||||
"io"
|
"io"
|
||||||
"os"
|
"os"
|
||||||
@ -17,66 +16,38 @@ import (
|
|||||||
"golang.org/x/tools/go/packages"
|
"golang.org/x/tools/go/packages"
|
||||||
)
|
)
|
||||||
|
|
||||||
// This list comes from the golint codebase. Golint will complain about any of
|
|
||||||
// these being mixed-case, like "Id" instead of "ID".
|
|
||||||
var golintInitialisms = []string{
|
|
||||||
"ACL",
|
|
||||||
"API",
|
|
||||||
"ASCII",
|
|
||||||
"CPU",
|
|
||||||
"CSS",
|
|
||||||
"DNS",
|
|
||||||
"EOF",
|
|
||||||
"GUID",
|
|
||||||
"HTML",
|
|
||||||
"HTTP",
|
|
||||||
"HTTPS",
|
|
||||||
"ID",
|
|
||||||
"IP",
|
|
||||||
"JSON",
|
|
||||||
"LHS",
|
|
||||||
"QPS",
|
|
||||||
"RAM",
|
|
||||||
"RHS",
|
|
||||||
"RPC",
|
|
||||||
"SLA",
|
|
||||||
"SMTP",
|
|
||||||
"SQL",
|
|
||||||
"SSH",
|
|
||||||
"TCP",
|
|
||||||
"TLS",
|
|
||||||
"TTL",
|
|
||||||
"UDP",
|
|
||||||
"UI",
|
|
||||||
"UID",
|
|
||||||
"UUID",
|
|
||||||
"URI",
|
|
||||||
"URL",
|
|
||||||
"UTF8",
|
|
||||||
"VM",
|
|
||||||
"XML",
|
|
||||||
"XMPP",
|
|
||||||
"XSRF",
|
|
||||||
"XSS",
|
|
||||||
}
|
|
||||||
|
|
||||||
// Mocker can generate mock structs.
|
// Mocker can generate mock structs.
|
||||||
type Mocker struct {
|
type Mocker struct {
|
||||||
srcPkg *packages.Package
|
srcPkg *packages.Package
|
||||||
tmpl *template.Template
|
tmpl *template.Template
|
||||||
pkgName string
|
pkgName string
|
||||||
pkgPath string
|
pkgPath string
|
||||||
|
fmter func(src []byte) ([]byte, error)
|
||||||
|
|
||||||
imports map[string]bool
|
imports map[string]bool
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Config specifies details about how interfaces should be mocked.
|
||||||
|
// SrcDir is the only field which needs be specified.
|
||||||
|
type Config struct {
|
||||||
|
SrcDir string
|
||||||
|
PkgName string
|
||||||
|
Formatter string
|
||||||
|
}
|
||||||
|
|
||||||
// New makes a new Mocker for the specified package directory.
|
// New makes a new Mocker for the specified package directory.
|
||||||
func New(src, packageName string) (*Mocker, error) {
|
func New(conf Config) (*Mocker, error) {
|
||||||
srcPkg, err := pkgInfoFromPath(src, packages.NeedName|packages.NeedTypes|packages.NeedTypesInfo)
|
srcPkg, err := pkgInfoFromPath(conf.SrcDir, packages.NeedName|packages.NeedTypes|packages.NeedTypesInfo)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("couldn't load source package: %s", err)
|
return nil, fmt.Errorf("couldn't load source package: %s", err)
|
||||||
}
|
}
|
||||||
pkgPath, err := findPkgPath(packageName, srcPkg)
|
|
||||||
|
pkgName := conf.PkgName
|
||||||
|
if pkgName == "" {
|
||||||
|
pkgName = srcPkg.Name
|
||||||
|
}
|
||||||
|
|
||||||
|
pkgPath, err := findPkgPath(conf.PkgName, srcPkg)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("couldn't load mock package: %s", err)
|
return nil, fmt.Errorf("couldn't load mock package: %s", err)
|
||||||
}
|
}
|
||||||
@ -85,22 +56,22 @@ func New(src, packageName string) (*Mocker, error) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fmter := gofmt
|
||||||
|
if conf.Formatter == "goimports" {
|
||||||
|
fmter = goimports
|
||||||
|
}
|
||||||
|
|
||||||
return &Mocker{
|
return &Mocker{
|
||||||
tmpl: tmpl,
|
tmpl: tmpl,
|
||||||
srcPkg: srcPkg,
|
srcPkg: srcPkg,
|
||||||
pkgName: preventZeroStr(packageName, srcPkg.Name),
|
pkgName: pkgName,
|
||||||
pkgPath: pkgPath,
|
pkgPath: pkgPath,
|
||||||
|
fmter: fmter,
|
||||||
imports: make(map[string]bool),
|
imports: make(map[string]bool),
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func preventZeroStr(val, defaultVal string) string {
|
|
||||||
if val == "" {
|
|
||||||
return defaultVal
|
|
||||||
}
|
|
||||||
return val
|
|
||||||
}
|
|
||||||
|
|
||||||
func findPkgPath(pkgInputVal string, srcPkg *packages.Package) (string, error) {
|
func findPkgPath(pkgInputVal string, srcPkg *packages.Package) (string, error) {
|
||||||
if pkgInputVal == "" {
|
if pkgInputVal == "" {
|
||||||
return srcPkg.PkgPath, nil
|
return srcPkg.PkgPath, nil
|
||||||
@ -186,9 +157,9 @@ func (m *Mocker) Mock(w io.Writer, names ...string) error {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
formatted, err := format.Source(buf.Bytes())
|
formatted, err := m.fmter(buf.Bytes())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("go/format: %s", err)
|
return err
|
||||||
}
|
}
|
||||||
if _, err := w.Write(formatted); err != nil {
|
if _, err := w.Write(formatted); err != nil {
|
||||||
return err
|
return err
|
||||||
@ -233,12 +204,11 @@ func (m *Mocker) extractArgs(sig *types.Signature, list *types.Tuple, nameFormat
|
|||||||
return params
|
return params
|
||||||
}
|
}
|
||||||
|
|
||||||
func pkgInfoFromPath(src string, mode packages.LoadMode) (*packages.Package, error) {
|
func pkgInfoFromPath(srcDir string, mode packages.LoadMode) (*packages.Package, error) {
|
||||||
conf := packages.Config{
|
pkgs, err := packages.Load(&packages.Config{
|
||||||
Mode: mode,
|
Mode: mode,
|
||||||
Dir: src,
|
Dir: srcDir,
|
||||||
}
|
})
|
||||||
pkgs, err := packages.Load(&conf)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -71,7 +71,7 @@ func TestModulesSamePackage(t *testing.T) {
|
|||||||
t.Fatalf("Test package copy error: %s", err)
|
t.Fatalf("Test package copy error: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
m, err := New(tmpDir, "")
|
m, err := New(Config{SrcDir: tmpDir})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("moq.New: %s", err)
|
t.Fatalf("moq.New: %s", err)
|
||||||
}
|
}
|
||||||
@ -103,7 +103,7 @@ func TestModulesNestedPackage(t *testing.T) {
|
|||||||
t.Fatalf("Test package copy error: %s", err)
|
t.Fatalf("Test package copy error: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
m, err := New(tmpDir, "nested")
|
m, err := New(Config{SrcDir: tmpDir, PkgName: "nested"})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("moq.New: %s", err)
|
t.Fatalf("moq.New: %s", err)
|
||||||
}
|
}
|
||||||
|
@ -2,15 +2,23 @@ package moq
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
|
"flag"
|
||||||
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
|
"io/ioutil"
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
|
"path/filepath"
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/pmezard/go-difflib/difflib"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var update = flag.Bool("update", false, "Update golden files.")
|
||||||
|
|
||||||
func TestMoq(t *testing.T) {
|
func TestMoq(t *testing.T) {
|
||||||
m, err := New("testpackages/example", "")
|
m, err := New(Config{SrcDir: "testpackages/example"})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("moq.New: %s", err)
|
t.Fatalf("moq.New: %s", err)
|
||||||
}
|
}
|
||||||
@ -43,7 +51,7 @@ func TestMoq(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestMoqWithStaticCheck(t *testing.T) {
|
func TestMoqWithStaticCheck(t *testing.T) {
|
||||||
m, err := New("testpackages/example", "")
|
m, err := New(Config{SrcDir: "testpackages/example"})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("moq.New: %s", err)
|
t.Fatalf("moq.New: %s", err)
|
||||||
}
|
}
|
||||||
@ -77,7 +85,7 @@ func TestMoqWithStaticCheck(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestMoqWithAlias(t *testing.T) {
|
func TestMoqWithAlias(t *testing.T) {
|
||||||
m, err := New("testpackages/example", "")
|
m, err := New(Config{SrcDir: "testpackages/example"})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("moq.New: %s", err)
|
t.Fatalf("moq.New: %s", err)
|
||||||
}
|
}
|
||||||
@ -110,7 +118,7 @@ func TestMoqWithAlias(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestMoqExplicitPackage(t *testing.T) {
|
func TestMoqExplicitPackage(t *testing.T) {
|
||||||
m, err := New("testpackages/example", "different")
|
m, err := New(Config{SrcDir: "testpackages/example", PkgName: "different"})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("moq.New: %s", err)
|
t.Fatalf("moq.New: %s", err)
|
||||||
}
|
}
|
||||||
@ -137,7 +145,7 @@ func TestMoqExplicitPackage(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestMoqExplicitPackageWithStaticCheck(t *testing.T) {
|
func TestMoqExplicitPackageWithStaticCheck(t *testing.T) {
|
||||||
m, err := New("testpackages/example", "different")
|
m, err := New(Config{SrcDir: "testpackages/example", PkgName: "different"})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("moq.New: %s", err)
|
t.Fatalf("moq.New: %s", err)
|
||||||
}
|
}
|
||||||
@ -165,7 +173,7 @@ func TestMoqExplicitPackageWithStaticCheck(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestNotCreatingEmptyDirWhenPkgIsGiven(t *testing.T) {
|
func TestNotCreatingEmptyDirWhenPkgIsGiven(t *testing.T) {
|
||||||
m, err := New("testpackages/example", "different")
|
m, err := New(Config{SrcDir: "testpackages/example", PkgName: "different"})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("moq.New: %s", err)
|
t.Fatalf("moq.New: %s", err)
|
||||||
}
|
}
|
||||||
@ -187,7 +195,7 @@ func TestNotCreatingEmptyDirWhenPkgIsGiven(t *testing.T) {
|
|||||||
// expected.
|
// expected.
|
||||||
// see https://github.com/matryer/moq/issues/5
|
// see https://github.com/matryer/moq/issues/5
|
||||||
func TestVariadicArguments(t *testing.T) {
|
func TestVariadicArguments(t *testing.T) {
|
||||||
m, err := New("testpackages/variadic", "")
|
m, err := New(Config{SrcDir: "testpackages/variadic"})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("moq.New: %s", err)
|
t.Fatalf("moq.New: %s", err)
|
||||||
}
|
}
|
||||||
@ -212,7 +220,7 @@ func TestVariadicArguments(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestNothingToReturn(t *testing.T) {
|
func TestNothingToReturn(t *testing.T) {
|
||||||
m, err := New("testpackages/example", "")
|
m, err := New(Config{SrcDir: "testpackages/example"})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("moq.New: %s", err)
|
t.Fatalf("moq.New: %s", err)
|
||||||
}
|
}
|
||||||
@ -237,7 +245,7 @@ func TestNothingToReturn(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestChannelNames(t *testing.T) {
|
func TestChannelNames(t *testing.T) {
|
||||||
m, err := New("testpackages/channels", "")
|
m, err := New(Config{SrcDir: "testpackages/channels"})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("moq.New: %s", err)
|
t.Fatalf("moq.New: %s", err)
|
||||||
}
|
}
|
||||||
@ -258,7 +266,7 @@ func TestChannelNames(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestImports(t *testing.T) {
|
func TestImports(t *testing.T) {
|
||||||
m, err := New("testpackages/imports/two", "")
|
m, err := New(Config{SrcDir: "testpackages/imports/two"})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("moq.New: %s", err)
|
t.Fatalf("moq.New: %s", err)
|
||||||
}
|
}
|
||||||
@ -282,6 +290,69 @@ func TestImports(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestFormatter(t *testing.T) {
|
||||||
|
cases := []struct {
|
||||||
|
name string
|
||||||
|
conf Config
|
||||||
|
}{
|
||||||
|
{name: "gofmt", conf: Config{SrcDir: "testpackages/imports/two"}},
|
||||||
|
{name: "goimports", conf: Config{SrcDir: "testpackages/imports/two", Formatter: "goimports"}},
|
||||||
|
}
|
||||||
|
for _, tc := range cases {
|
||||||
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
|
m, err := New(tc.conf)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("moq.New: %s", err)
|
||||||
|
}
|
||||||
|
var buf bytes.Buffer
|
||||||
|
err = m.Mock(&buf, "DoSomething")
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("m.Mock: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
golden := filepath.Join("testpackages/imports/testdata", tc.name+".golden.go")
|
||||||
|
if err := matchGoldenFile(golden, buf.Bytes()); err != nil {
|
||||||
|
t.Errorf("check golden file: %s", err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func matchGoldenFile(goldenFile string, actual []byte) error {
|
||||||
|
// To update golden files, run the following:
|
||||||
|
// go test -v -run ^<Test-Name>$ github.com/matryer/moq/pkg/moq -update
|
||||||
|
if *update {
|
||||||
|
if err := ioutil.WriteFile(goldenFile, actual, 0644); err != nil {
|
||||||
|
return fmt.Errorf("write: %s: %s", goldenFile, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
expected, err := ioutil.ReadFile(goldenFile)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("read: %s: %s", goldenFile, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Normalise newlines
|
||||||
|
actual, expected = normalize(actual), normalize(expected)
|
||||||
|
if !bytes.Equal(expected, actual) {
|
||||||
|
diff, err := difflib.GetUnifiedDiffString(difflib.UnifiedDiff{
|
||||||
|
A: difflib.SplitLines(string(expected)),
|
||||||
|
B: difflib.SplitLines(string(actual)),
|
||||||
|
FromFile: "Expected",
|
||||||
|
ToFile: "Actual",
|
||||||
|
Context: 1,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("diff: %s", err)
|
||||||
|
}
|
||||||
|
return fmt.Errorf("match: %s:\n%s", goldenFile, diff)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func TestTemplateFuncs(t *testing.T) {
|
func TestTemplateFuncs(t *testing.T) {
|
||||||
fn := templateFuncs["Exported"].(func(string) string)
|
fn := templateFuncs["Exported"].(func(string) string)
|
||||||
if fn("var") != "Var" {
|
if fn("var") != "Var" {
|
||||||
@ -290,7 +361,7 @@ func TestTemplateFuncs(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestVendoredPackages(t *testing.T) {
|
func TestVendoredPackages(t *testing.T) {
|
||||||
m, err := New("testpackages/vendoring/user", "")
|
m, err := New(Config{SrcDir: "testpackages/vendoring/user"})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("moq.New: %s", err)
|
t.Fatalf("moq.New: %s", err)
|
||||||
}
|
}
|
||||||
@ -312,7 +383,10 @@ func TestVendoredPackages(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestVendoredInterface(t *testing.T) {
|
func TestVendoredInterface(t *testing.T) {
|
||||||
m, err := New("testpackages/vendoring/vendor/github.com/matryer/somerepo", "someother")
|
m, err := New(Config{
|
||||||
|
SrcDir: "testpackages/vendoring/vendor/github.com/matryer/somerepo",
|
||||||
|
PkgName: "someother",
|
||||||
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("moq.New: %s", err)
|
t.Fatalf("moq.New: %s", err)
|
||||||
}
|
}
|
||||||
@ -338,7 +412,7 @@ func TestVendoredInterface(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestVendoredBuildConstraints(t *testing.T) {
|
func TestVendoredBuildConstraints(t *testing.T) {
|
||||||
m, err := New("testpackages/buildconstraints/user", "")
|
m, err := New(Config{SrcDir: "testpackages/buildconstraints/user"})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("moq.New: %s", err)
|
t.Fatalf("moq.New: %s", err)
|
||||||
}
|
}
|
||||||
@ -375,7 +449,7 @@ func TestDotImports(t *testing.T) {
|
|||||||
t.Errorf("Chdir back: %s", err)
|
t.Errorf("Chdir back: %s", err)
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
m, err := New(".", "moqtest_test")
|
m, err := New(Config{SrcDir: ".", PkgName: "moqtest_test"})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("moq.New: %s", err)
|
t.Fatalf("moq.New: %s", err)
|
||||||
}
|
}
|
||||||
@ -391,7 +465,7 @@ func TestDotImports(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestEmptyInterface(t *testing.T) {
|
func TestEmptyInterface(t *testing.T) {
|
||||||
m, err := New("testpackages/emptyinterface", "")
|
m, err := New(Config{SrcDir: "testpackages/emptyinterface"})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("moq.New: %s", err)
|
t.Fatalf("moq.New: %s", err)
|
||||||
}
|
}
|
||||||
@ -435,7 +509,7 @@ func TestGoGenerateVendoredPackages(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestImportedPackageWithSameName(t *testing.T) {
|
func TestImportedPackageWithSameName(t *testing.T) {
|
||||||
m, err := New("testpackages/samenameimport", "")
|
m, err := New(Config{SrcDir: "testpackages/samenameimport"})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("moq.New: %s", err)
|
t.Fatalf("moq.New: %s", err)
|
||||||
}
|
}
|
||||||
@ -449,3 +523,14 @@ func TestImportedPackageWithSameName(t *testing.T) {
|
|||||||
t.Error("missing samename.A to address the struct A from the external package samename")
|
t.Error("missing samename.A to address the struct A from the external package samename")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// normalize normalizes \r\n (windows) and \r (mac)
|
||||||
|
// into \n (unix)
|
||||||
|
func normalize(d []byte) []byte {
|
||||||
|
// Source: https://www.programming-books.io/essential/go/normalize-newlines-1d3abcf6f17c4186bb9617fa14074e48
|
||||||
|
// replace CR LF \r\n (windows) with LF \n (unix)
|
||||||
|
d = bytes.Replace(d, []byte{13, 10}, []byte{10}, -1)
|
||||||
|
// replace CF \r (mac) with LF \n (unix)
|
||||||
|
d = bytes.Replace(d, []byte{13}, []byte{10}, -1)
|
||||||
|
return d
|
||||||
|
}
|
||||||
|
120
pkg/moq/testpackages/imports/testdata/gofmt.golden.go
vendored
Normal file
120
pkg/moq/testpackages/imports/testdata/gofmt.golden.go
vendored
Normal file
@ -0,0 +1,120 @@
|
|||||||
|
// Code generated by moq; DO NOT EDIT.
|
||||||
|
// github.com/matryer/moq
|
||||||
|
|
||||||
|
package two
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/matryer/moq/pkg/moq/testpackages/imports/one"
|
||||||
|
"sync"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
lockDoSomethingMockAnother sync.RWMutex
|
||||||
|
lockDoSomethingMockDo sync.RWMutex
|
||||||
|
)
|
||||||
|
|
||||||
|
// Ensure, that DoSomethingMock does implement DoSomething.
|
||||||
|
// If this is not the case, regenerate this file with moq.
|
||||||
|
var _ DoSomething = &DoSomethingMock{}
|
||||||
|
|
||||||
|
// DoSomethingMock is a mock implementation of DoSomething.
|
||||||
|
//
|
||||||
|
// func TestSomethingThatUsesDoSomething(t *testing.T) {
|
||||||
|
//
|
||||||
|
// // make and configure a mocked DoSomething
|
||||||
|
// mockedDoSomething := &DoSomethingMock{
|
||||||
|
// AnotherFunc: func(thing one.Thing) error {
|
||||||
|
// panic("mock out the Another method")
|
||||||
|
// },
|
||||||
|
// DoFunc: func(thing one.Thing) error {
|
||||||
|
// panic("mock out the Do method")
|
||||||
|
// },
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// // use mockedDoSomething in code that requires DoSomething
|
||||||
|
// // and then make assertions.
|
||||||
|
//
|
||||||
|
// }
|
||||||
|
type DoSomethingMock struct {
|
||||||
|
// AnotherFunc mocks the Another method.
|
||||||
|
AnotherFunc func(thing one.Thing) error
|
||||||
|
|
||||||
|
// DoFunc mocks the Do method.
|
||||||
|
DoFunc func(thing one.Thing) error
|
||||||
|
|
||||||
|
// calls tracks calls to the methods.
|
||||||
|
calls struct {
|
||||||
|
// Another holds details about calls to the Another method.
|
||||||
|
Another []struct {
|
||||||
|
// Thing is the thing argument value.
|
||||||
|
Thing one.Thing
|
||||||
|
}
|
||||||
|
// Do holds details about calls to the Do method.
|
||||||
|
Do []struct {
|
||||||
|
// Thing is the thing argument value.
|
||||||
|
Thing one.Thing
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Another calls AnotherFunc.
|
||||||
|
func (mock *DoSomethingMock) Another(thing one.Thing) error {
|
||||||
|
if mock.AnotherFunc == nil {
|
||||||
|
panic("DoSomethingMock.AnotherFunc: method is nil but DoSomething.Another was just called")
|
||||||
|
}
|
||||||
|
callInfo := struct {
|
||||||
|
Thing one.Thing
|
||||||
|
}{
|
||||||
|
Thing: thing,
|
||||||
|
}
|
||||||
|
lockDoSomethingMockAnother.Lock()
|
||||||
|
mock.calls.Another = append(mock.calls.Another, callInfo)
|
||||||
|
lockDoSomethingMockAnother.Unlock()
|
||||||
|
return mock.AnotherFunc(thing)
|
||||||
|
}
|
||||||
|
|
||||||
|
// AnotherCalls gets all the calls that were made to Another.
|
||||||
|
// Check the length with:
|
||||||
|
// len(mockedDoSomething.AnotherCalls())
|
||||||
|
func (mock *DoSomethingMock) AnotherCalls() []struct {
|
||||||
|
Thing one.Thing
|
||||||
|
} {
|
||||||
|
var calls []struct {
|
||||||
|
Thing one.Thing
|
||||||
|
}
|
||||||
|
lockDoSomethingMockAnother.RLock()
|
||||||
|
calls = mock.calls.Another
|
||||||
|
lockDoSomethingMockAnother.RUnlock()
|
||||||
|
return calls
|
||||||
|
}
|
||||||
|
|
||||||
|
// Do calls DoFunc.
|
||||||
|
func (mock *DoSomethingMock) Do(thing one.Thing) error {
|
||||||
|
if mock.DoFunc == nil {
|
||||||
|
panic("DoSomethingMock.DoFunc: method is nil but DoSomething.Do was just called")
|
||||||
|
}
|
||||||
|
callInfo := struct {
|
||||||
|
Thing one.Thing
|
||||||
|
}{
|
||||||
|
Thing: thing,
|
||||||
|
}
|
||||||
|
lockDoSomethingMockDo.Lock()
|
||||||
|
mock.calls.Do = append(mock.calls.Do, callInfo)
|
||||||
|
lockDoSomethingMockDo.Unlock()
|
||||||
|
return mock.DoFunc(thing)
|
||||||
|
}
|
||||||
|
|
||||||
|
// DoCalls gets all the calls that were made to Do.
|
||||||
|
// Check the length with:
|
||||||
|
// len(mockedDoSomething.DoCalls())
|
||||||
|
func (mock *DoSomethingMock) DoCalls() []struct {
|
||||||
|
Thing one.Thing
|
||||||
|
} {
|
||||||
|
var calls []struct {
|
||||||
|
Thing one.Thing
|
||||||
|
}
|
||||||
|
lockDoSomethingMockDo.RLock()
|
||||||
|
calls = mock.calls.Do
|
||||||
|
lockDoSomethingMockDo.RUnlock()
|
||||||
|
return calls
|
||||||
|
}
|
121
pkg/moq/testpackages/imports/testdata/goimports.golden.go
vendored
Normal file
121
pkg/moq/testpackages/imports/testdata/goimports.golden.go
vendored
Normal file
@ -0,0 +1,121 @@
|
|||||||
|
// Code generated by moq; DO NOT EDIT.
|
||||||
|
// github.com/matryer/moq
|
||||||
|
|
||||||
|
package two
|
||||||
|
|
||||||
|
import (
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"github.com/matryer/moq/pkg/moq/testpackages/imports/one"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
lockDoSomethingMockAnother sync.RWMutex
|
||||||
|
lockDoSomethingMockDo sync.RWMutex
|
||||||
|
)
|
||||||
|
|
||||||
|
// Ensure, that DoSomethingMock does implement DoSomething.
|
||||||
|
// If this is not the case, regenerate this file with moq.
|
||||||
|
var _ DoSomething = &DoSomethingMock{}
|
||||||
|
|
||||||
|
// DoSomethingMock is a mock implementation of DoSomething.
|
||||||
|
//
|
||||||
|
// func TestSomethingThatUsesDoSomething(t *testing.T) {
|
||||||
|
//
|
||||||
|
// // make and configure a mocked DoSomething
|
||||||
|
// mockedDoSomething := &DoSomethingMock{
|
||||||
|
// AnotherFunc: func(thing one.Thing) error {
|
||||||
|
// panic("mock out the Another method")
|
||||||
|
// },
|
||||||
|
// DoFunc: func(thing one.Thing) error {
|
||||||
|
// panic("mock out the Do method")
|
||||||
|
// },
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// // use mockedDoSomething in code that requires DoSomething
|
||||||
|
// // and then make assertions.
|
||||||
|
//
|
||||||
|
// }
|
||||||
|
type DoSomethingMock struct {
|
||||||
|
// AnotherFunc mocks the Another method.
|
||||||
|
AnotherFunc func(thing one.Thing) error
|
||||||
|
|
||||||
|
// DoFunc mocks the Do method.
|
||||||
|
DoFunc func(thing one.Thing) error
|
||||||
|
|
||||||
|
// calls tracks calls to the methods.
|
||||||
|
calls struct {
|
||||||
|
// Another holds details about calls to the Another method.
|
||||||
|
Another []struct {
|
||||||
|
// Thing is the thing argument value.
|
||||||
|
Thing one.Thing
|
||||||
|
}
|
||||||
|
// Do holds details about calls to the Do method.
|
||||||
|
Do []struct {
|
||||||
|
// Thing is the thing argument value.
|
||||||
|
Thing one.Thing
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Another calls AnotherFunc.
|
||||||
|
func (mock *DoSomethingMock) Another(thing one.Thing) error {
|
||||||
|
if mock.AnotherFunc == nil {
|
||||||
|
panic("DoSomethingMock.AnotherFunc: method is nil but DoSomething.Another was just called")
|
||||||
|
}
|
||||||
|
callInfo := struct {
|
||||||
|
Thing one.Thing
|
||||||
|
}{
|
||||||
|
Thing: thing,
|
||||||
|
}
|
||||||
|
lockDoSomethingMockAnother.Lock()
|
||||||
|
mock.calls.Another = append(mock.calls.Another, callInfo)
|
||||||
|
lockDoSomethingMockAnother.Unlock()
|
||||||
|
return mock.AnotherFunc(thing)
|
||||||
|
}
|
||||||
|
|
||||||
|
// AnotherCalls gets all the calls that were made to Another.
|
||||||
|
// Check the length with:
|
||||||
|
// len(mockedDoSomething.AnotherCalls())
|
||||||
|
func (mock *DoSomethingMock) AnotherCalls() []struct {
|
||||||
|
Thing one.Thing
|
||||||
|
} {
|
||||||
|
var calls []struct {
|
||||||
|
Thing one.Thing
|
||||||
|
}
|
||||||
|
lockDoSomethingMockAnother.RLock()
|
||||||
|
calls = mock.calls.Another
|
||||||
|
lockDoSomethingMockAnother.RUnlock()
|
||||||
|
return calls
|
||||||
|
}
|
||||||
|
|
||||||
|
// Do calls DoFunc.
|
||||||
|
func (mock *DoSomethingMock) Do(thing one.Thing) error {
|
||||||
|
if mock.DoFunc == nil {
|
||||||
|
panic("DoSomethingMock.DoFunc: method is nil but DoSomething.Do was just called")
|
||||||
|
}
|
||||||
|
callInfo := struct {
|
||||||
|
Thing one.Thing
|
||||||
|
}{
|
||||||
|
Thing: thing,
|
||||||
|
}
|
||||||
|
lockDoSomethingMockDo.Lock()
|
||||||
|
mock.calls.Do = append(mock.calls.Do, callInfo)
|
||||||
|
lockDoSomethingMockDo.Unlock()
|
||||||
|
return mock.DoFunc(thing)
|
||||||
|
}
|
||||||
|
|
||||||
|
// DoCalls gets all the calls that were made to Do.
|
||||||
|
// Check the length with:
|
||||||
|
// len(mockedDoSomething.DoCalls())
|
||||||
|
func (mock *DoSomethingMock) DoCalls() []struct {
|
||||||
|
Thing one.Thing
|
||||||
|
} {
|
||||||
|
var calls []struct {
|
||||||
|
Thing one.Thing
|
||||||
|
}
|
||||||
|
lockDoSomethingMockDo.RLock()
|
||||||
|
calls = mock.calls.Do
|
||||||
|
lockDoSomethingMockDo.RUnlock()
|
||||||
|
return calls
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user