From cb7d651a5a623c9730d731671ec69285d63f65ef Mon Sep 17 00:00:00 2001 From: thediveo Date: Fri, 6 Dec 2024 20:33:52 +0100 Subject: [PATCH] resolves onsi/gomega#696: make HaveField great on pointer receivers given only a non-addressable value Signed-off-by: thediveo --- matchers/have_field.go | 7 ++++++- matchers/have_field_test.go | 10 +++++----- 2 files changed, 11 insertions(+), 6 deletions(-) diff --git a/matchers/have_field.go b/matchers/have_field.go index 8dd3f871a..293457e85 100644 --- a/matchers/have_field.go +++ b/matchers/have_field.go @@ -40,7 +40,12 @@ func extractField(actual interface{}, field string, matchername string) (any, er extractedValue = actualValue.Addr().MethodByName(strings.TrimSuffix(fields[0], "()")) } if extractedValue == (reflect.Value{}) { - return nil, missingFieldError(fmt.Sprintf("%s could not find method named '%s' in struct of type %T.", matchername, fields[0], actual)) + ptr := reflect.New(actualValue.Type()) + ptr.Elem().Set(actualValue) + extractedValue = ptr.MethodByName(strings.TrimSuffix(fields[0], "()")) + if extractedValue == (reflect.Value{}) { + return nil, missingFieldError(fmt.Sprintf("%s could not find method named '%s' in struct of type %T.", matchername, fields[0], actual)) + } } t := extractedValue.Type() if t.NumIn() != 0 || t.NumOut() != 1 { diff --git a/matchers/have_field_test.go b/matchers/have_field_test.go index c979944ad..4bc1edffe 100644 --- a/matchers/have_field_test.go +++ b/matchers/have_field_test.go @@ -162,7 +162,7 @@ var _ = Describe("HaveField", func() { }) Describe("receiver lookup", func() { - DescribeTable("(pointer) receiver lookup works", + DescribeTable("(pointer) receiver lookup on book pointer works", func(field string, expected interface{}) { Ω(&book).Should(HaveField(field, expected)) }, @@ -170,15 +170,15 @@ var _ = Describe("HaveField", func() { Entry("pointer receiver", "PointerReceiverTitle()", "Les Miserables"), ) + It("correctly looks up a pointer receiver on a book value", func() { + Ω(book).Should(HaveField("PointerReceiverTitle()", "Les Miserables")) + }) + It("correctly fails", func() { matcher := HaveField("ReceiverTitle()", "Les Miserables") answer := struct{}{} Ω(matcher.Match(answer)).Error().Should(MatchError( "HaveField could not find method named 'ReceiverTitle()' in struct of type struct {}.")) - - matcher = HaveField("PointerReceiverTitle()", "Les Miserables") - Ω(matcher.Match(book)).Error().Should(MatchError( - "HaveField could not find method named 'PointerReceiverTitle()' in struct of type matchers_test.Book.")) }) })