@@ -2879,6 +2879,203 @@ class PairMatcher {
2879
2879
const SecondMatcher second_matcher_;
2880
2880
};
2881
2881
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
+
2882
3079
// Implements ElementsAre() and ElementsAreArray().
2883
3080
template <typename Container>
2884
3081
class ElementsAreMatcherImpl : public MatcherInterface <Container> {
@@ -4514,6 +4711,19 @@ Pair(FirstMatcher first_matcher, SecondMatcher second_matcher) {
4514
4711
first_matcher, second_matcher);
4515
4712
}
4516
4713
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
+
4517
4727
// Returns a predicate that is satisfied by anything that matches the
4518
4728
// given matcher.
4519
4729
template <typename M>
@@ -5053,6 +5263,9 @@ PolymorphicMatcher<internal::ExceptionMatcherImpl<Err>> ThrowsMessage(
5053
5263
#define GMOCK_INTERNAL_MATCHER_ARG_USAGE (i, data_unused, arg_unused ) \
5054
5264
, gmock_p##i
5055
5265
5266
+ // To prevent ADL on certain functions we put them on a separate namespace.
5267
+ using namespace no_adl ; // NOLINT
5268
+
5056
5269
} // namespace testing
5057
5270
5058
5271
GTEST_DISABLE_MSC_WARNINGS_POP_ () // 4251 5046
0 commit comments