Skip to content

Commit

Permalink
Add flag to enable mock reset methods (#181)
Browse files Browse the repository at this point in the history
The optional with-resets flag adds methods to reset method calls. Calls
can be reset individually or all at once with these.
  • Loading branch information
cbaker authored Mar 8, 2023
1 parent a464ccc commit d0d706b
Show file tree
Hide file tree
Showing 9 changed files with 299 additions and 28 deletions.
3 changes: 3 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,8 @@ moq [flags] source-dir interface [interface2 [interface3 [...]]]
-skip-ensure
suppress mock implementation check, avoid import cycle if mocks
generated outside of the tested package
-with-resets
generate functions to facilitate resetting calls made to a mock
Specifying an alias for the mock is also supported with the format 'interface:alias'
Expand Down Expand Up @@ -120,6 +122,7 @@ The mocked structure implements the interface, where each method calls the assoc
* Name arguments in the interface for a better experience
* Use closured variables inside your test function to capture details about the calls to the methods
* Use `.MethodCalls()` to track the calls
* Use `.ResetCalls()` to reset calls within an invidual mock's context
* Use `go:generate` to invoke the `moq` command
* If Moq fails with a `go/format` error, it indicates the generated code was not valid.
You can run the same command with `-fmt noop` to print the generated source code without attempting to format it.
Expand Down
21 changes: 20 additions & 1 deletion internal/template/template.go
Original file line number Diff line number Diff line change
Expand Up @@ -182,8 +182,27 @@ func (mock *{{$mock.MockName}}
mock.lock{{.Name}}.RUnlock()
return calls
}
{{- if $.WithResets}}
// Reset{{.Name}}Calls reset all the calls that were made to {{.Name}}.
func (mock *{{$mock.MockName}}) Reset{{.Name}}Calls() {
mock.lock{{.Name}}.Lock()
mock.calls.{{.Name}} = nil
mock.lock{{.Name}}.Unlock()
}
{{end}}
{{end -}}
{{- if $.WithResets}}
// ResetCalls reset all the calls that were made to all mocked methods.
func (mock *{{$mock.MockName}}) ResetCalls() {
{{- range .Methods}}
mock.lock{{.Name}}.Lock()
mock.calls.{{.Name}} = nil
mock.lock{{.Name}}.Unlock()
{{end -}}
}
{{end -}}
{{end -}}
{{end -}}`
`

// This list comes from the golint codebase. Golint will complain about any of
// these being mixed-case, like "Id" instead of "ID".
Expand Down
1 change: 1 addition & 0 deletions internal/template/template_data.go
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ type Data struct {
Mocks []MockData
StubImpl bool
SkipEnsure bool
WithResets bool
}

// MocksSomeMethod returns true of any one of the Mocks has at least 1
Expand Down
8 changes: 6 additions & 2 deletions main.go
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ type userFlags struct {
formatter string
stubImpl bool
skipEnsure bool
withResets bool
remove bool
args []string
}
Expand All @@ -37,6 +38,8 @@ func main() {
flag.BoolVar(&flags.skipEnsure, "skip-ensure", false,
"suppress mock implementation check, avoid import cycle if mocks generated outside of the tested package")
flag.BoolVar(&flags.remove, "rm", false, "first remove output file, if it exists")
flag.BoolVar(&flags.withResets, "with-resets", false,
"generate functions to facilitate resetting calls made to a mock")

flag.Usage = func() {
fmt.Println(`moq [flags] source-dir interface [interface2 [interface3 [...]]]`)
Expand Down Expand Up @@ -86,6 +89,7 @@ func run(flags userFlags) error {
Formatter: flags.formatter,
StubImpl: flags.stubImpl,
SkipEnsure: flags.skipEnsure,
WithResets: flags.withResets,
})
if err != nil {
return err
Expand All @@ -100,10 +104,10 @@ func run(flags userFlags) error {
}

// create the file
err = os.MkdirAll(filepath.Dir(flags.outFile), 0750)
err = os.MkdirAll(filepath.Dir(flags.outFile), 0o750)
if err != nil {
return err
}

return ioutil.WriteFile(flags.outFile, buf.Bytes(), 0600)
return ioutil.WriteFile(flags.outFile, buf.Bytes(), 0o600)
}
2 changes: 2 additions & 0 deletions pkg/moq/moq.go
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@ type Config struct {
Formatter string
StubImpl bool
SkipEnsure bool
WithResets bool
}

// New makes a new Mocker for the specified package directory.
Expand Down Expand Up @@ -81,6 +82,7 @@ func (m *Mocker) Mock(w io.Writer, namePairs ...string) error {
Mocks: mocks,
StubImpl: m.cfg.StubImpl,
SkipEnsure: m.cfg.SkipEnsure,
WithResets: m.cfg.WithResets,
}

if data.MocksSomeMethod() {
Expand Down
34 changes: 20 additions & 14 deletions pkg/moq/moq_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@ func TestMoq(t *testing.T) {
}
s := buf.String()
// assertions of things that should be mentioned
var strs = []string{
strs := []string{
"package example",
"type PersonStoreMock struct",
"CreateFunc func(ctx context.Context, person *Person, confirm bool) error",
Expand Down Expand Up @@ -62,7 +62,7 @@ func TestMoqWithStaticCheck(t *testing.T) {
}
s := buf.String()
// assertions of things that should be mentioned
var strs = []string{
strs := []string{
"package example",
"var _ PersonStore = &PersonStoreMock{}",
"type PersonStoreMock struct",
Expand Down Expand Up @@ -96,7 +96,7 @@ func TestMoqWithAlias(t *testing.T) {
}
s := buf.String()
// assertions of things that should be mentioned
var strs = []string{
strs := []string{
"package example",
"type AnotherPersonStoreMock struct",
"CreateFunc func(ctx context.Context, person *Person, confirm bool) error",
Expand Down Expand Up @@ -129,7 +129,7 @@ func TestMoqExplicitPackage(t *testing.T) {
}
s := buf.String()
// assertions of things that should be mentioned
var strs = []string{
strs := []string{
"package different",
"type PersonStoreMock struct",
"CreateFunc func(ctx context.Context, person *example.Person, confirm bool) error",
Expand All @@ -156,7 +156,7 @@ func TestMoqExplicitPackageWithStaticCheck(t *testing.T) {
}
s := buf.String()
// assertions of things that should be mentioned
var strs = []string{
strs := []string{
"package different",
"var _ example.PersonStore = &PersonStoreMock{}",
"type PersonStoreMock struct",
Expand Down Expand Up @@ -184,7 +184,7 @@ func TestMoqSkipEnsure(t *testing.T) {
}
s := buf.String()
// assertions of things that should be mentioned
var strs = []string{
strs := []string{
"package different",
"type PersonStoreMock struct",
"CreateFunc func(ctx context.Context, person *example.Person, confirm bool) error",
Expand Down Expand Up @@ -233,7 +233,7 @@ func TestVariadicArguments(t *testing.T) {
}
s := buf.String()
// assertions of things that should be mentioned
var strs = []string{
strs := []string{
"package variadic",
"type GreeterMock struct",
"GreetFunc func(ctx context.Context, names ...string) string",
Expand Down Expand Up @@ -261,7 +261,7 @@ func TestNothingToReturn(t *testing.T) {
t.Errorf("should not have return for items that have no return arguments")
}
// assertions of things that should be mentioned
var strs = []string{
strs := []string{
"mock.ClearCacheFunc(id)",
}
for _, str := range strs {
Expand All @@ -282,7 +282,7 @@ func TestImports(t *testing.T) {
t.Errorf("m.Mock: %s", err)
}
s := buf.String()
var strs = []string{
strs := []string{
` "sync"`,
` "github.com/matryer/moq/pkg/moq/testpackages/imports/one"`,
}
Expand Down Expand Up @@ -395,6 +395,12 @@ func TestMockGolden(t *testing.T) {
interfaces: []string{"Transient"},
goldenFile: filepath.Join("testpackages/transientimport", "transient_moq.golden.go"),
},
{
name: "WithResets",
cfg: Config{SrcDir: "testpackages/withresets", WithResets: true},
interfaces: []string{"ResetStore"},
goldenFile: filepath.Join("testpackages/withresets", "withresets_moq.golden.go"),
},
}
for _, tc := range cases {
t.Run(tc.name, func(t *testing.T) {
Expand Down Expand Up @@ -449,10 +455,10 @@ 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 := os.MkdirAll(filepath.Dir(goldenFile), 0750); err != nil {
if err := os.MkdirAll(filepath.Dir(goldenFile), 0o750); err != nil {
return fmt.Errorf("create dir: %s", err)
}
if err := ioutil.WriteFile(goldenFile, actual, 0600); err != nil {
if err := ioutil.WriteFile(goldenFile, actual, 0o600); err != nil {
return fmt.Errorf("write: %s", err)
}

Expand Down Expand Up @@ -495,7 +501,7 @@ func TestVendoredPackages(t *testing.T) {
}
s := buf.String()
// assertions of things that should be mentioned
var strs = []string{
strs := []string{
`"github.com/sudo-suhas/moq-test-pkgs/somerepo"`,
}
for _, str := range strs {
Expand All @@ -520,7 +526,7 @@ func TestVendoredInterface(t *testing.T) {
}
s := buf.String()
// assertions of things that should be mentioned
var strs = []string{
strs := []string{
`"github.com/sudo-suhas/moq-test-pkgs/somerepo"`,
}
for _, str := range strs {
Expand All @@ -546,7 +552,7 @@ func TestVendoredBuildConstraints(t *testing.T) {
}
s := buf.String()
// assertions of things that should be mentioned
var strs = []string{
strs := []string{
`"github.com/sudo-suhas/moq-test-pkgs/buildconstraints"`,
}
for _, str := range strs {
Expand Down
23 changes: 12 additions & 11 deletions pkg/moq/testpackages/dotimport/service_moq_test.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

18 changes: 18 additions & 0 deletions pkg/moq/testpackages/withresets/withresets.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
package withresets

import "context"

// Reset is a reset.
type Reset struct {
ID string
Name string
Company string
Website string
}

// ResetStore stores resets.
type ResetStore interface {
Get(ctx context.Context, id string) (*Reset, error)
Create(ctx context.Context, person *Reset, confirm bool) error
ClearCache(id string)
}
Loading

0 comments on commit d0d706b

Please sign in to comment.