-
Notifications
You must be signed in to change notification settings - Fork 608
Improve default failed equality match output to more clearly show diffs #616
Comments
I'm also running into this issue a lot. It would be great if the library exposed a way to set a diff function with a signature like When a test fails and the got/want output is difficult to read, I like to use the following trick to get a pretty diff. It's not an elegant solution, especially when wrapping several values, but it works for debugging purposes. example.gopackage example
//go:generate mockgen -destination mocks/$GOFILE -package ${GOPACKAGE}_mocks -source $GOFILE
type Foo struct {
Bar *string
}
type Fooer interface {
DoSomething(Foo)
} example_test.gopackage example_test
import (
"example"
example_mocks "example/mocks"
"log"
"testing"
"github.com/golang/mock/gomock"
"github.com/kr/pretty"
)
func String(s string) *string { return &s }
func TestFoo_Standard(t *testing.T) {
ctrl := gomock.NewController(t)
defer ctrl.Finish()
mockFooer := example_mocks.NewMockFooer(ctrl)
mockFooer.EXPECT().
DoSomething(
example.Foo{Bar: String("expected")},
)
mockFooer.DoSomething(example.Foo{
Bar: String("actual"),
})
}
func TestFoo_Improved(t *testing.T) {
ctrl := gomock.NewController(t)
defer ctrl.Finish()
mockFooer := example_mocks.NewMockFooer(ctrl)
mockFooer.EXPECT().
DoSomething(
&diffWrapper{
v: example.Foo{Bar: String("expected")},
},
)
mockFooer.DoSomething(example.Foo{
Bar: String("actual"),
})
}
type diffWrapper struct {
v interface{}
}
// Implement the gomock.Matcher interface
func (m *diffWrapper) Matches(v interface{}) bool {
ok := gomock.Eq(m.v).Matches(v) // wrap the gomock.Eq matcher
if !ok { // print the diff
log.Printf("This is the diff: %s", pretty.Diff(m.v, v)) // any diff func here
}
return ok // return the result of the match
}
// Implement the gomock.Matcher interface
func (m *diffWrapper) String() string {
return gomock.Eq(m.v).String() // again wrap the gomock.Eq matcher
} When running the
|
While not perfect (and quite hackish to be honest), it's possible to have a diff in the output. type eqMatcher struct {
want interface{}
}
func EqMatcher(want interface{}) eqMatcher {
return eqMatcher{
want: want,
}
}
func (eq eqMatcher) Matches(got interface{}) bool {
return gomock.Eq(eq.want).Matches(got)
}
func (eq eqMatcher) Got(got interface{}) string {
return fmt.Sprintf("%v (%T)\nDiff (-got +want):\n%s", got, got, strings.TrimSpace(cmp.Diff(got, eq.want)))
}
func (eq eqMatcher) String() string {
return fmt.Sprintf("%v (%T)\n", eq.want, eq.want)
} And then just use as you would with any other matcher: m.EXPECT().DoSomething(EqMatcher(...)) Which outputs:
But I agree, some optional, native form of diffing (perhaps even using go-cmp?) would be better. |
Any chance we can rework the Controller to accept a diff function? Something along the lines of: ctrl := gomock.NewController(t, gomock.WithDiffFunc(cmp.Diff)) The "got" and "want" formatters could be added in the same way. |
I think @palsivertsen's solution by requiring developers to wrap ctrl := gomock.NewController(t, gomock.WithDiffOpts(cmp.AllowUnexported(&Something{}), protocmp.Transform())) |
Looks like DoAndReturn has better diff. |
Requested feature
Sometimes I'll generate a mock and
EXPECT
some calls. In cases where the test fails because the mock was called with a different input, it can be very hard to tell what the difference is. This is especially apparent in cases where some fields are pointers.I've tried to set up an example in Go Playground to show what I mean: https://go.dev/play/p/pVxqpZpqiwz
Basically, I have a struct that looks like
and an interface I generate a mock for that looks like
In my
TestWithMock
case, I expect a call with a certainObo
and provide a call with a slightly different one, causing a test failure. The only difference between the two provided is that the first one has ana
value set to&"apple"
and the second set to&"banana"
.Here's the output I see when the test fails:
In reality, the diff is quite small, just a single field with a different value. However, the output from the failure provides basically no information on this mismatch. Every field appears different because the only thing that it displays is the pointer, not the value. This makes the issue much more difficult to debug!
Compare this to the output of
TestWithAssert
, a similar case that directly compares the twoObo
objects usingassert.Equal
from thegithub.heygears.com/stretchr/testify/assert
package.With this output, the nice diff immediately makes it clear what the difference was between the two structs, vastly simplifying my debugging experience!
I believe this is related to #265, which was closed. If I understand correctly, the proposed solution would be to use
WantFormatter
andGotFormatter
to generate a nicer output myself. Personally, having never used these features before, it feels like a lot of effort for a simple and common problem! Would I need to use these in every test case I have? I think it would be much more helpful to improve the default output when using an equality matcher.(Optional) Proposed solution
Some potential ideas for how I'd like to see the output improved
This would make it easier to manually inspect and see what the difference is
assert
function in my example above. This is even better since it makes it really easy to tell the difference.The text was updated successfully, but these errors were encountered: