From 198dcbad602d1540aec7051aaefd99279e1df1bc Mon Sep 17 00:00:00 2001 From: Nisheeth Barthwal Date: Mon, 9 Nov 2020 15:26:28 +0100 Subject: [PATCH 1/3] allow testing for functional options --- mock/mock.go | 101 ++++++++++++++++++++++++++++++++++++++++++++++ mock/mock_test.go | 40 ++++++++++++++++++ 2 files changed, 141 insertions(+) diff --git a/mock/mock.go b/mock/mock.go index c6df4485a..69a40a82c 100644 --- a/mock/mock.go +++ b/mock/mock.go @@ -3,6 +3,7 @@ package mock import ( "errors" "fmt" + "path" "reflect" "regexp" "runtime" @@ -13,6 +14,7 @@ import ( "github.com/davecgh/go-spew/spew" "github.com/pmezard/go-difflib/difflib" "github.com/stretchr/objx" + "github.com/stretchr/testify/assert" ) @@ -324,6 +326,10 @@ func callString(method string, arguments Arguments, includeArgumentValues bool) if includeArgumentValues { var argVals []string for argIndex, arg := range arguments { + if _, ok := arg.(*FunctionalOptionsArgument); ok { + argVals = append(argVals, fmt.Sprintf("%d: %s", argIndex, arg)) + continue + } argVals = append(argVals, fmt.Sprintf("%d: %#v", argIndex, arg)) } argValsString = fmt.Sprintf("\n\t\t%s", strings.Join(argVals, "\n\t\t")) @@ -659,6 +665,22 @@ func IsType(t interface{}) *IsTypeArgument { return &IsTypeArgument{t: t} } +type FunctionalOptionsArgument struct { + name string + value interface{} +} + +func (f *FunctionalOptionsArgument) String() string { + return strings.Replace(fmt.Sprintf("%#v", f.value), "[]interface {}", f.name, 1) +} + +func FunctionalOptions(name string, value ...interface{}) *FunctionalOptionsArgument { + return &FunctionalOptionsArgument{ + name: name, + value: value, + } +} + // argumentMatcher performs custom argument matching, returning whether or // not the argument is matched by the expectation fixture function. type argumentMatcher struct { @@ -798,6 +820,23 @@ func (args Arguments) Diff(objects []interface{}) (string, int) { differences++ output = fmt.Sprintf("%s\t%d: FAIL: type %s != type %s - %s\n", output, i, reflect.TypeOf(t).Name(), reflect.TypeOf(actual).Name(), actualFmt) } + } else if reflect.TypeOf(expected) == reflect.TypeOf((*FunctionalOptionsArgument)(nil)) { + name := expected.(*FunctionalOptionsArgument).name + t := expected.(*FunctionalOptionsArgument).value + tName := reflect.TypeOf(t).Name() + if name != reflect.TypeOf(actual).String() { + differences++ + output = fmt.Sprintf("%s\t%d: FAIL: type %s != type %s - %s\n", output, i, tName, reflect.TypeOf(actual).Name(), actualFmt) + } else { + if ef, af := assertOpts(t, actual); ef == "" && af == "" { + // match + output = fmt.Sprintf("%s\t%d: PASS: %s == %s\n", output, i, tName, tName) + } else { + // not match + differences++ + output = fmt.Sprintf("%s\t%d: FAIL: %s != %s\n", output, i, af, ef) + } + } } else { // normal checking @@ -979,3 +1018,65 @@ var spewConfig = spew.ConfigState{ type tHelper interface { Helper() } + +func assertOpts(expected, actual interface{}) (expectedFmt, actualFmt string) { + expectedOpts := reflect.ValueOf(expected) + actualOpts := reflect.ValueOf(actual) + var expectedNames []string + for i := 0; i < expectedOpts.Len(); i++ { + expectedNames = append(expectedNames, funcName(expectedOpts.Index(i).Interface())) + } + var actualNames []string + for i := 0; i < actualOpts.Len(); i++ { + actualNames = append(actualNames, funcName(actualOpts.Index(i).Interface())) + } + if !assert.ObjectsAreEqual(expectedNames, actualNames) { + expectedFmt = fmt.Sprintf("%v", expectedNames) + actualFmt = fmt.Sprintf("%v", actualNames) + return + } + + for i := 0; i < expectedOpts.Len(); i++ { + expectedOpt := expectedOpts.Index(i).Interface() + actualOpt := actualOpts.Index(i).Interface() + + expectedFunc := expectedNames[i] + actualFunc := actualNames[i] + if expectedFunc != actualFunc { + expectedFmt = expectedFunc + actualFmt = actualFunc + return + } + + ot := reflect.TypeOf(expectedOpt) + var expectedValues []reflect.Value + var actualValues []reflect.Value + if ot.NumIn() == 0 { + return + } + + for i := 0; i < ot.NumIn(); i++ { + vt := ot.In(i).Elem() + expectedValues = append(expectedValues, reflect.New(vt)) + actualValues = append(actualValues, reflect.New(vt)) + } + + reflect.ValueOf(expectedOpt).Call(expectedValues) + reflect.ValueOf(actualOpt).Call(actualValues) + + for i := 0; i < ot.NumIn(); i++ { + if !assert.ObjectsAreEqual(expectedValues[i].Interface(), actualValues[i].Interface()) { + expectedFmt = fmt.Sprintf("%s %+v", expectedNames[i], expectedValues[i].Interface()) + actualFmt = fmt.Sprintf("%s %+v", expectedNames[i], actualValues[i].Interface()) + return + } + } + } + + return "", "" +} + +func funcName(opt interface{}) string { + n := runtime.FuncForPC(reflect.ValueOf(opt).Pointer()).Name() + return strings.TrimSuffix(path.Base(n), path.Ext(n)) +} diff --git a/mock/mock_test.go b/mock/mock_test.go index cb4954145..0d782c556 100644 --- a/mock/mock_test.go +++ b/mock/mock_test.go @@ -32,6 +32,29 @@ func (i *TestExampleImplementation) TheExampleMethod(a, b, c int) (int, error) { return args.Int(0), errors.New("Whoops") } +type options struct { + num int + str string +} + +type OptionFn func(*options) + +func OpNum(n int) OptionFn { + return func(o *options) { + o.num = n + } +} + +func OpStr(s string) OptionFn { + return func(o *options) { + o.str = s + } +} +func (i *TestExampleImplementation) TheExampleMethodFunctionalOptions(x string, opts ...OptionFn) error { + args := i.Called(x, opts) + return args.Error(0) +} + //go:noinline func (i *TestExampleImplementation) TheExampleMethod2(yesorno bool) { i.Called(yesorno) @@ -1044,6 +1067,23 @@ func Test_Mock_AssertExpectationsCustomType(t *testing.T) { } +func Test_Mock_AssertExpectationsFunctionalOptionsType(t *testing.T) { + + var mockedService = new(TestExampleImplementation) + + mockedService.On("TheExampleMethodFunctionalOptions", "test", FunctionalOptions("[]mock.OptionFn", OpNum(1), OpStr("foo"))).Return(nil).Once() + + tt := new(testing.T) + assert.False(t, mockedService.AssertExpectations(tt)) + + // make the call now + mockedService.TheExampleMethodFunctionalOptions("test", OpNum(1), OpStr("foo")) + + // now assert expectations + assert.True(t, mockedService.AssertExpectations(tt)) + +} + func Test_Mock_AssertExpectations_With_Repeatability(t *testing.T) { var mockedService = new(TestExampleImplementation) From 3b93fae0ea161f8d0ca47dd7aa3d8a949799d261 Mon Sep 17 00:00:00 2001 From: Nisheeth Barthwal Date: Mon, 9 Nov 2020 15:31:17 +0100 Subject: [PATCH 2/3] add linting docs --- mock/mock.go | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/mock/mock.go b/mock/mock.go index 69a40a82c..df67dfbe1 100644 --- a/mock/mock.go +++ b/mock/mock.go @@ -665,15 +665,23 @@ func IsType(t interface{}) *IsTypeArgument { return &IsTypeArgument{t: t} } +// FunctionalOptionsArgument is a struct that contains the type and value of an functional option argument +// for use when type checking. type FunctionalOptionsArgument struct { name string value interface{} } +// String returns the string representation of FunctionalOptionsArgument func (f *FunctionalOptionsArgument) String() string { return strings.Replace(fmt.Sprintf("%#v", f.value), "[]interface {}", f.name, 1) } +// FunctionalOptions returns an FunctionalOptionsArgument object containing the functional option type +// and the values to check of +// +// For example: +// Assert(t, FunctionalOptions("[]foo.FunctionalOption", foo.Opt1(), foo.Opt2())) func FunctionalOptions(name string, value ...interface{}) *FunctionalOptionsArgument { return &FunctionalOptionsArgument{ name: name, From a706293304275448e7eeca780f9617e95018f2d9 Mon Sep 17 00:00:00 2001 From: Nisheeth Barthwal Date: Wed, 18 Jan 2023 00:27:10 +0100 Subject: [PATCH 3/3] use reflect magic to obtain FunctionalOption name Co-authored-by: dillonstreator --- mock/mock.go | 22 ++++++++++++++++------ mock/mock_test.go | 19 ++++++++++++++++++- 2 files changed, 34 insertions(+), 7 deletions(-) diff --git a/mock/mock.go b/mock/mock.go index df67dfbe1..cc7ba00bc 100644 --- a/mock/mock.go +++ b/mock/mock.go @@ -668,13 +668,18 @@ func IsType(t interface{}) *IsTypeArgument { // FunctionalOptionsArgument is a struct that contains the type and value of an functional option argument // for use when type checking. type FunctionalOptionsArgument struct { - name string value interface{} } // String returns the string representation of FunctionalOptionsArgument func (f *FunctionalOptionsArgument) String() string { - return strings.Replace(fmt.Sprintf("%#v", f.value), "[]interface {}", f.name, 1) + var name string + tValue := reflect.ValueOf(f.value) + if tValue.Len() > 0 { + name = "[]" + reflect.TypeOf(tValue.Index(0).Interface()).String() + } + + return strings.Replace(fmt.Sprintf("%#v", f.value), "[]interface {}", name, 1) } // FunctionalOptions returns an FunctionalOptionsArgument object containing the functional option type @@ -682,9 +687,8 @@ func (f *FunctionalOptionsArgument) String() string { // // For example: // Assert(t, FunctionalOptions("[]foo.FunctionalOption", foo.Opt1(), foo.Opt2())) -func FunctionalOptions(name string, value ...interface{}) *FunctionalOptionsArgument { +func FunctionalOptions(value ...interface{}) *FunctionalOptionsArgument { return &FunctionalOptionsArgument{ - name: name, value: value, } } @@ -829,10 +833,16 @@ func (args Arguments) Diff(objects []interface{}) (string, int) { output = fmt.Sprintf("%s\t%d: FAIL: type %s != type %s - %s\n", output, i, reflect.TypeOf(t).Name(), reflect.TypeOf(actual).Name(), actualFmt) } } else if reflect.TypeOf(expected) == reflect.TypeOf((*FunctionalOptionsArgument)(nil)) { - name := expected.(*FunctionalOptionsArgument).name t := expected.(*FunctionalOptionsArgument).value + + var name string + tValue := reflect.ValueOf(t) + if tValue.Len() > 0 { + name = "[]" + reflect.TypeOf(tValue.Index(0).Interface()).String() + } + tName := reflect.TypeOf(t).Name() - if name != reflect.TypeOf(actual).String() { + if name != reflect.TypeOf(actual).String() && tValue.Len() != 0 { differences++ output = fmt.Sprintf("%s\t%d: FAIL: type %s != type %s - %s\n", output, i, tName, reflect.TypeOf(actual).Name(), actualFmt) } else { diff --git a/mock/mock_test.go b/mock/mock_test.go index 0d782c556..4797e30c0 100644 --- a/mock/mock_test.go +++ b/mock/mock_test.go @@ -1071,7 +1071,7 @@ func Test_Mock_AssertExpectationsFunctionalOptionsType(t *testing.T) { var mockedService = new(TestExampleImplementation) - mockedService.On("TheExampleMethodFunctionalOptions", "test", FunctionalOptions("[]mock.OptionFn", OpNum(1), OpStr("foo"))).Return(nil).Once() + mockedService.On("TheExampleMethodFunctionalOptions", "test", FunctionalOptions(OpNum(1), OpStr("foo"))).Return(nil).Once() tt := new(testing.T) assert.False(t, mockedService.AssertExpectations(tt)) @@ -1084,6 +1084,23 @@ func Test_Mock_AssertExpectationsFunctionalOptionsType(t *testing.T) { } +func Test_Mock_AssertExpectationsFunctionalOptionsType_Empty(t *testing.T) { + + var mockedService = new(TestExampleImplementation) + + mockedService.On("TheExampleMethodFunctionalOptions", "test", FunctionalOptions()).Return(nil).Once() + + tt := new(testing.T) + assert.False(t, mockedService.AssertExpectations(tt)) + + // make the call now + mockedService.TheExampleMethodFunctionalOptions("test") + + // now assert expectations + assert.True(t, mockedService.AssertExpectations(tt)) + +} + func Test_Mock_AssertExpectations_With_Repeatability(t *testing.T) { var mockedService = new(TestExampleImplementation)