Skip to content

Commit a462188

Browse files
Abseil Teamderekmauro
Abseil Team
authored andcommitted
Googletest export
Add ::testing::FieldsAre matcher for objects that support get<> and structured bindings. PiperOrigin-RevId: 337165285
1 parent f3dbe3e commit a462188

File tree

4 files changed

+364
-3
lines changed

4 files changed

+364
-3
lines changed

googlemock/docs/cheat_sheet.md

+1
Original file line numberDiff line numberDiff line change
@@ -403,6 +403,7 @@ messages, you can use:
403403
| `Field(&class::field, m)` | `argument.field` (or `argument->field` when `argument` is a plain pointer) matches matcher `m`, where `argument` is an object of type _class_. |
404404
| `Key(e)` | `argument.first` matches `e`, which can be either a value or a matcher. E.g. `Contains(Key(Le(5)))` can verify that a `map` contains a key `<= 5`. |
405405
| `Pair(m1, m2)` | `argument` is an `std::pair` whose `first` field matches `m1` and `second` field matches `m2`. |
406+
| `FieldsAre(m...)` | `argument` is a compatible object where each field matches piecewise with `m...`. A compatible object is any that supports the `std::tuple_size<Obj>`+`get<I>(obj)` protocol. In C++17 and up this also supports types compatible with structured bindings, like aggregates. |
406407
| `Property(&class::property, m)` | `argument.property()` (or `argument->property()` when `argument` is a plain pointer) matches matcher `m`, where `argument` is an object of type _class_. |
407408
<!-- mdformat on -->
408409

googlemock/include/gmock/gmock-matchers.h

+213
Original file line numberDiff line numberDiff line change
@@ -2879,6 +2879,203 @@ class PairMatcher {
28792879
const SecondMatcher second_matcher_;
28802880
};
28812881

2882+
template <typename T, size_t... I>
2883+
auto UnpackStructImpl(const T& t, IndexSequence<I...>, int)
2884+
-> decltype(std::tie(get<I>(t)...)) {
2885+
static_assert(std::tuple_size<T>::value == sizeof...(I),
2886+
"Number of arguments doesn't match the number of fields.");
2887+
return std::tie(get<I>(t)...);
2888+
}
2889+
2890+
#if defined(__cpp_structured_bindings) && __cpp_structured_bindings >= 201606
2891+
template <typename T>
2892+
auto UnpackStructImpl(const T& t, MakeIndexSequence<1>, char) {
2893+
const auto& [a] = t;
2894+
return std::tie(a);
2895+
}
2896+
template <typename T>
2897+
auto UnpackStructImpl(const T& t, MakeIndexSequence<2>, char) {
2898+
const auto& [a, b] = t;
2899+
return std::tie(a, b);
2900+
}
2901+
template <typename T>
2902+
auto UnpackStructImpl(const T& t, MakeIndexSequence<3>, char) {
2903+
const auto& [a, b, c] = t;
2904+
return std::tie(a, b, c);
2905+
}
2906+
template <typename T>
2907+
auto UnpackStructImpl(const T& t, MakeIndexSequence<4>, char) {
2908+
const auto& [a, b, c, d] = t;
2909+
return std::tie(a, b, c, d);
2910+
}
2911+
template <typename T>
2912+
auto UnpackStructImpl(const T& t, MakeIndexSequence<5>, char) {
2913+
const auto& [a, b, c, d, e] = t;
2914+
return std::tie(a, b, c, d, e);
2915+
}
2916+
template <typename T>
2917+
auto UnpackStructImpl(const T& t, MakeIndexSequence<6>, char) {
2918+
const auto& [a, b, c, d, e, f] = t;
2919+
return std::tie(a, b, c, d, e, f);
2920+
}
2921+
template <typename T>
2922+
auto UnpackStructImpl(const T& t, MakeIndexSequence<7>, char) {
2923+
const auto& [a, b, c, d, e, f, g] = t;
2924+
return std::tie(a, b, c, d, e, f, g);
2925+
}
2926+
template <typename T>
2927+
auto UnpackStructImpl(const T& t, MakeIndexSequence<8>, char) {
2928+
const auto& [a, b, c, d, e, f, g, h] = t;
2929+
return std::tie(a, b, c, d, e, f, g, h);
2930+
}
2931+
template <typename T>
2932+
auto UnpackStructImpl(const T& t, MakeIndexSequence<9>, char) {
2933+
const auto& [a, b, c, d, e, f, g, h, i] = t;
2934+
return std::tie(a, b, c, d, e, f, g, h, i);
2935+
}
2936+
template <typename T>
2937+
auto UnpackStructImpl(const T& t, MakeIndexSequence<10>, char) {
2938+
const auto& [a, b, c, d, e, f, g, h, i, j] = t;
2939+
return std::tie(a, b, c, d, e, f, g, h, i, j);
2940+
}
2941+
template <typename T>
2942+
auto UnpackStructImpl(const T& t, MakeIndexSequence<11>, char) {
2943+
const auto& [a, b, c, d, e, f, g, h, i, j, k] = t;
2944+
return std::tie(a, b, c, d, e, f, g, h, i, j, k);
2945+
}
2946+
template <typename T>
2947+
auto UnpackStructImpl(const T& t, MakeIndexSequence<12>, char) {
2948+
const auto& [a, b, c, d, e, f, g, h, i, j, k, l] = t;
2949+
return std::tie(a, b, c, d, e, f, g, h, i, j, k, l);
2950+
}
2951+
template <typename T>
2952+
auto UnpackStructImpl(const T& t, MakeIndexSequence<13>, char) {
2953+
const auto& [a, b, c, d, e, f, g, h, i, j, k, l, m] = t;
2954+
return std::tie(a, b, c, d, e, f, g, h, i, j, k, l, m);
2955+
}
2956+
template <typename T>
2957+
auto UnpackStructImpl(const T& t, MakeIndexSequence<14>, char) {
2958+
const auto& [a, b, c, d, e, f, g, h, i, j, k, l, m, n] = t;
2959+
return std::tie(a, b, c, d, e, f, g, h, i, j, k, l, m, n);
2960+
}
2961+
template <typename T>
2962+
auto UnpackStructImpl(const T& t, MakeIndexSequence<15>, char) {
2963+
const auto& [a, b, c, d, e, f, g, h, i, j, k, l, m, n, o] = t;
2964+
return std::tie(a, b, c, d, e, f, g, h, i, j, k, l, m, n, o);
2965+
}
2966+
template <typename T>
2967+
auto UnpackStructImpl(const T& t, MakeIndexSequence<16>, char) {
2968+
const auto& [a, b, c, d, e, f, g, h, i, j, k, l, m, n, o, p] = t;
2969+
return std::tie(a, b, c, d, e, f, g, h, i, j, k, l, m, n, o, p);
2970+
}
2971+
#endif // defined(__cpp_structured_bindings)
2972+
2973+
template <size_t I, typename T>
2974+
auto UnpackStruct(const T& t)
2975+
-> decltype((UnpackStructImpl)(t, MakeIndexSequence<I>{}, 0)) {
2976+
return (UnpackStructImpl)(t, MakeIndexSequence<I>{}, 0);
2977+
}
2978+
2979+
// Helper function to do comma folding in C++11.
2980+
// The array ensures left-to-right order of evaluation.
2981+
// Usage: VariadicExpand({expr...});
2982+
template <typename T, size_t N>
2983+
void VariadicExpand(const T (&a)[N]) {}
2984+
2985+
template <typename Struct, typename StructSize>
2986+
class FieldsAreMatcherImpl;
2987+
2988+
template <typename Struct, size_t... I>
2989+
class FieldsAreMatcherImpl<Struct, IndexSequence<I...>>
2990+
: public MatcherInterface<Struct> {
2991+
using UnpackedType =
2992+
decltype(UnpackStruct<sizeof...(I)>(std::declval<const Struct&>()));
2993+
using MatchersType = std::tuple<
2994+
Matcher<const typename std::tuple_element<I, UnpackedType>::type&>...>;
2995+
2996+
public:
2997+
template <typename Inner>
2998+
explicit FieldsAreMatcherImpl(const Inner& matchers)
2999+
: matchers_(testing::SafeMatcherCast<
3000+
const typename std::tuple_element<I, UnpackedType>::type&>(
3001+
std::get<I>(matchers))...) {}
3002+
3003+
void DescribeTo(::std::ostream* os) const override {
3004+
const char* separator = "";
3005+
VariadicExpand(
3006+
{(*os << separator << "has field #" << I << " that ",
3007+
std::get<I>(matchers_).DescribeTo(os), separator = ", and ")...});
3008+
}
3009+
3010+
void DescribeNegationTo(::std::ostream* os) const override {
3011+
const char* separator = "";
3012+
VariadicExpand({(*os << separator << "has field #" << I << " that ",
3013+
std::get<I>(matchers_).DescribeNegationTo(os),
3014+
separator = ", or ")...});
3015+
}
3016+
3017+
bool MatchAndExplain(Struct t, MatchResultListener* listener) const override {
3018+
return MatchInternal((UnpackStruct<sizeof...(I)>)(t), listener);
3019+
}
3020+
3021+
private:
3022+
bool MatchInternal(UnpackedType tuple, MatchResultListener* listener) const {
3023+
if (!listener->IsInterested()) {
3024+
// If the listener is not interested, we don't need to construct the
3025+
// explanation.
3026+
bool good = true;
3027+
VariadicExpand({good = good && std::get<I>(matchers_).Matches(
3028+
std::get<I>(tuple))...});
3029+
return good;
3030+
}
3031+
3032+
int failed_pos = -1;
3033+
3034+
std::vector<StringMatchResultListener> inner_listener(sizeof...(I));
3035+
3036+
VariadicExpand(
3037+
{failed_pos == -1 && !std::get<I>(matchers_).MatchAndExplain(
3038+
std::get<I>(tuple), &inner_listener[I])
3039+
? failed_pos = I
3040+
: 0 ...});
3041+
if (failed_pos != ~size_t{}) {
3042+
*listener << "whose field #" << failed_pos << " does not match";
3043+
PrintIfNotEmpty(inner_listener[failed_pos].str(), listener->stream());
3044+
return false;
3045+
}
3046+
3047+
*listener << "whose all elements match";
3048+
const char* separator = ", where";
3049+
for (size_t index = 0; index < sizeof...(I); ++index) {
3050+
const std::string str = inner_listener[index].str();
3051+
if (!str.empty()) {
3052+
*listener << separator << " field #" << index << " is a value " << str;
3053+
separator = ", and";
3054+
}
3055+
}
3056+
3057+
return true;
3058+
}
3059+
3060+
MatchersType matchers_;
3061+
};
3062+
3063+
template <typename... Inner>
3064+
class FieldsAreMatcher {
3065+
public:
3066+
explicit FieldsAreMatcher(Inner... inner) : matchers_(std::move(inner)...) {}
3067+
3068+
template <typename Struct>
3069+
operator Matcher<Struct>() const { // NOLINT
3070+
return Matcher<Struct>(
3071+
new FieldsAreMatcherImpl<const Struct&, IndexSequenceFor<Inner...>>(
3072+
matchers_));
3073+
}
3074+
3075+
private:
3076+
std::tuple<Inner...> matchers_;
3077+
};
3078+
28823079
// Implements ElementsAre() and ElementsAreArray().
28833080
template <typename Container>
28843081
class ElementsAreMatcherImpl : public MatcherInterface<Container> {
@@ -4514,6 +4711,19 @@ Pair(FirstMatcher first_matcher, SecondMatcher second_matcher) {
45144711
first_matcher, second_matcher);
45154712
}
45164713

4714+
namespace no_adl {
4715+
// FieldsAre(matchers...) matches piecewise the fields of compatible structs.
4716+
// These include those that support `get<I>(obj)`, and when structured bindings
4717+
// are enabled any class that supports them.
4718+
// In particular, `std::tuple`, `std::pair`, `std::array` and aggregate types.
4719+
template <typename... M>
4720+
internal::FieldsAreMatcher<typename std::decay<M>::type...> FieldsAre(
4721+
M&&... matchers) {
4722+
return internal::FieldsAreMatcher<typename std::decay<M>::type...>(
4723+
std::forward<M>(matchers)...);
4724+
}
4725+
} // namespace no_adl
4726+
45174727
// Returns a predicate that is satisfied by anything that matches the
45184728
// given matcher.
45194729
template <typename M>
@@ -5053,6 +5263,9 @@ PolymorphicMatcher<internal::ExceptionMatcherImpl<Err>> ThrowsMessage(
50535263
#define GMOCK_INTERNAL_MATCHER_ARG_USAGE(i, data_unused, arg_unused) \
50545264
, gmock_p##i
50555265

5266+
// To prevent ADL on certain functions we put them on a separate namespace.
5267+
using namespace no_adl; // NOLINT
5268+
50565269
} // namespace testing
50575270

50585271
GTEST_DISABLE_MSC_WARNINGS_POP_() // 4251 5046

googlemock/test/gmock-matchers_test.cc

+141
Original file line numberDiff line numberDiff line change
@@ -1643,6 +1643,147 @@ TEST(PairTest, InsideContainsUsingMap) {
16431643
EXPECT_THAT(container, Not(Contains(Pair(3, _))));
16441644
}
16451645

1646+
TEST(FieldsAreTest, MatchesCorrectly) {
1647+
std::tuple<int, std::string, double> p(25, "foo", .5);
1648+
1649+
// All fields match.
1650+
EXPECT_THAT(p, FieldsAre(25, "foo", .5));
1651+
EXPECT_THAT(p, FieldsAre(Ge(20), HasSubstr("o"), DoubleEq(.5)));
1652+
1653+
// Some don't match.
1654+
EXPECT_THAT(p, Not(FieldsAre(26, "foo", .5)));
1655+
EXPECT_THAT(p, Not(FieldsAre(25, "fo", .5)));
1656+
EXPECT_THAT(p, Not(FieldsAre(25, "foo", .6)));
1657+
}
1658+
1659+
TEST(FieldsAreTest, CanDescribeSelf) {
1660+
Matcher<const pair<std::string, int>&> m1 = FieldsAre("foo", 42);
1661+
EXPECT_EQ(
1662+
"has field #0 that is equal to \"foo\""
1663+
", and has field #1 that is equal to 42",
1664+
Describe(m1));
1665+
EXPECT_EQ(
1666+
"has field #0 that isn't equal to \"foo\""
1667+
", or has field #1 that isn't equal to 42",
1668+
DescribeNegation(m1));
1669+
}
1670+
1671+
TEST(FieldsAreTest, CanExplainMatchResultTo) {
1672+
// The first one that fails is the one that gives the error.
1673+
Matcher<std::tuple<int, int, int>> m =
1674+
FieldsAre(GreaterThan(0), GreaterThan(0), GreaterThan(0));
1675+
1676+
EXPECT_EQ("whose field #0 does not match, which is 1 less than 0",
1677+
Explain(m, std::make_tuple(-1, -2, -3)));
1678+
EXPECT_EQ("whose field #1 does not match, which is 2 less than 0",
1679+
Explain(m, std::make_tuple(1, -2, -3)));
1680+
EXPECT_EQ("whose field #2 does not match, which is 3 less than 0",
1681+
Explain(m, std::make_tuple(1, 2, -3)));
1682+
1683+
// If they all match, we get a long explanation of success.
1684+
EXPECT_EQ(
1685+
"whose all elements match, "
1686+
"where field #0 is a value which is 1 more than 0"
1687+
", and field #1 is a value which is 2 more than 0"
1688+
", and field #2 is a value which is 3 more than 0",
1689+
Explain(m, std::make_tuple(1, 2, 3)));
1690+
1691+
// Only print those that have an explanation.
1692+
m = FieldsAre(GreaterThan(0), 0, GreaterThan(0));
1693+
EXPECT_EQ(
1694+
"whose all elements match, "
1695+
"where field #0 is a value which is 1 more than 0"
1696+
", and field #2 is a value which is 3 more than 0",
1697+
Explain(m, std::make_tuple(1, 0, 3)));
1698+
1699+
// If only one has an explanation, then print that one.
1700+
m = FieldsAre(0, GreaterThan(0), 0);
1701+
EXPECT_EQ(
1702+
"whose all elements match, "
1703+
"where field #1 is a value which is 1 more than 0",
1704+
Explain(m, std::make_tuple(0, 1, 0)));
1705+
}
1706+
1707+
#if defined(__cpp_structured_bindings) && __cpp_structured_bindings >= 201606
1708+
TEST(FieldsAreTest, StructuredBindings) {
1709+
// testing::FieldsAre can also match aggregates and such with C++17 and up.
1710+
struct MyType {
1711+
int i;
1712+
std::string str;
1713+
};
1714+
EXPECT_THAT((MyType{17, "foo"}), FieldsAre(Eq(17), HasSubstr("oo")));
1715+
1716+
// Test all the supported arities.
1717+
struct MyVarType1 {
1718+
int a;
1719+
};
1720+
EXPECT_THAT(MyVarType1{}, FieldsAre(0));
1721+
struct MyVarType2 {
1722+
int a, b;
1723+
};
1724+
EXPECT_THAT(MyVarType2{}, FieldsAre(0, 0));
1725+
struct MyVarType3 {
1726+
int a, b, c;
1727+
};
1728+
EXPECT_THAT(MyVarType3{}, FieldsAre(0, 0, 0));
1729+
struct MyVarType4 {
1730+
int a, b, c, d;
1731+
};
1732+
EXPECT_THAT(MyVarType4{}, FieldsAre(0, 0, 0, 0));
1733+
struct MyVarType5 {
1734+
int a, b, c, d, e;
1735+
};
1736+
EXPECT_THAT(MyVarType5{}, FieldsAre(0, 0, 0, 0, 0));
1737+
struct MyVarType6 {
1738+
int a, b, c, d, e, f;
1739+
};
1740+
EXPECT_THAT(MyVarType6{}, FieldsAre(0, 0, 0, 0, 0, 0));
1741+
struct MyVarType7 {
1742+
int a, b, c, d, e, f, g;
1743+
};
1744+
EXPECT_THAT(MyVarType7{}, FieldsAre(0, 0, 0, 0, 0, 0, 0));
1745+
struct MyVarType8 {
1746+
int a, b, c, d, e, f, g, h;
1747+
};
1748+
EXPECT_THAT(MyVarType8{}, FieldsAre(0, 0, 0, 0, 0, 0, 0, 0));
1749+
struct MyVarType9 {
1750+
int a, b, c, d, e, f, g, h, i;
1751+
};
1752+
EXPECT_THAT(MyVarType9{}, FieldsAre(0, 0, 0, 0, 0, 0, 0, 0, 0));
1753+
struct MyVarType10 {
1754+
int a, b, c, d, e, f, g, h, i, j;
1755+
};
1756+
EXPECT_THAT(MyVarType10{}, FieldsAre(0, 0, 0, 0, 0, 0, 0, 0, 0, 0));
1757+
struct MyVarType11 {
1758+
int a, b, c, d, e, f, g, h, i, j, k;
1759+
};
1760+
EXPECT_THAT(MyVarType11{}, FieldsAre(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0));
1761+
struct MyVarType12 {
1762+
int a, b, c, d, e, f, g, h, i, j, k, l;
1763+
};
1764+
EXPECT_THAT(MyVarType12{}, FieldsAre(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0));
1765+
struct MyVarType13 {
1766+
int a, b, c, d, e, f, g, h, i, j, k, l, m;
1767+
};
1768+
EXPECT_THAT(MyVarType13{}, FieldsAre(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0));
1769+
struct MyVarType14 {
1770+
int a, b, c, d, e, f, g, h, i, j, k, l, m, n;
1771+
};
1772+
EXPECT_THAT(MyVarType14{},
1773+
FieldsAre(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0));
1774+
struct MyVarType15 {
1775+
int a, b, c, d, e, f, g, h, i, j, k, l, m, n, o;
1776+
};
1777+
EXPECT_THAT(MyVarType15{},
1778+
FieldsAre(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0));
1779+
struct MyVarType16 {
1780+
int a, b, c, d, e, f, g, h, i, j, k, l, m, n, o, p;
1781+
};
1782+
EXPECT_THAT(MyVarType16{},
1783+
FieldsAre(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0));
1784+
}
1785+
#endif
1786+
16461787
TEST(ContainsTest, WorksWithMoveOnly) {
16471788
ContainerHelper helper;
16481789
EXPECT_CALL(helper, Call(Contains(Pointee(2))));

0 commit comments

Comments
 (0)