Skip to content

Commit 48ec640

Browse files
Abseil Teamderekmauro
Abseil Team
authored andcommitted
Googletest export
Fix DoAll to work with move-only sink arguments. This changes types of the first n - 1 actions so that they only get a readonly view of the arguments. The last action will accept move only objects. PiperOrigin-RevId: 324600664
1 parent e6e2d3b commit 48ec640

File tree

3 files changed

+21
-7
lines changed

3 files changed

+21
-7
lines changed

googlemock/docs/cheat_sheet.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -616,7 +616,7 @@ composite action - trying to do so will result in a run-time error.
616616
<!-- mdformat off(no multiline tables) -->
617617
| | |
618618
| :----------------------------- | :------------------------------------------ |
619-
| `DoAll(a1, a2, ..., an)` | Do all actions `a1` to `an` and return the result of `an` in each invocation. The first `n - 1` sub-actions must return void. |
619+
| `DoAll(a1, a2, ..., an)` | Do all actions `a1` to `an` and return the result of `an` in each invocation. The first `n - 1` sub-actions must return void and will receive a readonly view of the arguments. |
620620
| `IgnoreResult(a)` | Perform action `a` and ignore its result. `a` must not return void. |
621621
| `WithArg<N>(a)` | Pass the `N`-th (0-based) argument of the mock function to action `a` and perform it. |
622622
| `WithArgs<N1, N2, ..., Nk>(a)` | Pass the selected (0-based) arguments of the mock function to action `a` and perform it. |

googlemock/include/gmock/gmock-actions.h

+11-6
Original file line numberDiff line numberDiff line change
@@ -1032,8 +1032,13 @@ struct WithArgsAction {
10321032
template <typename... Actions>
10331033
struct DoAllAction {
10341034
private:
1035+
template <typename T>
1036+
using NonFinalType =
1037+
typename std::conditional<std::is_scalar<T>::value, T, const T&>::type;
1038+
10351039
template <typename... Args, size_t... I>
1036-
std::vector<Action<void(Args...)>> Convert(IndexSequence<I...>) const {
1040+
std::vector<Action<void(NonFinalType<Args>...)>> Convert(
1041+
IndexSequence<I...>) const {
10371042
return {std::get<I>(actions)...};
10381043
}
10391044

@@ -1043,14 +1048,13 @@ struct DoAllAction {
10431048
template <typename R, typename... Args>
10441049
operator Action<R(Args...)>() const { // NOLINT
10451050
struct Op {
1046-
std::vector<Action<void(Args...)>> converted;
1051+
std::vector<Action<void(NonFinalType<Args>...)>> converted;
10471052
Action<R(Args...)> last;
10481053
R operator()(Args... args) const {
1049-
auto tuple_args = std::forward_as_tuple(std::forward<Args>(args)...);
10501054
for (auto& a : converted) {
1051-
a.Perform(tuple_args);
1055+
a.Perform(std::forward_as_tuple(std::forward<Args>(args)...));
10521056
}
1053-
return last.Perform(tuple_args);
1057+
return last.Perform(std::forward_as_tuple(std::forward<Args>(args)...));
10541058
}
10551059
};
10561060
return Op{Convert<Args...>(MakeIndexSequence<sizeof...(Actions) - 1>()),
@@ -1093,7 +1097,8 @@ struct DoAllAction {
10931097
typedef internal::IgnoredValue Unused;
10941098

10951099
// Creates an action that does actions a1, a2, ..., sequentially in
1096-
// each invocation.
1100+
// each invocation. All but the last action will have a readonly view of the
1101+
// arguments.
10971102
template <typename... Action>
10981103
internal::DoAllAction<typename std::decay<Action>::type...> DoAll(
10991104
Action&&... action) {

googlemock/test/gmock-generated-actions_test.cc

+9
Original file line numberDiff line numberDiff line change
@@ -422,6 +422,15 @@ TEST(DoAllTest, TenActions) {
422422
EXPECT_EQ('g', g);
423423
}
424424

425+
TEST(DoAllTest, MoveOnlyArgs) {
426+
bool ran_first = false;
427+
Action<int(std::unique_ptr<int>)> a =
428+
DoAll(InvokeWithoutArgs([&] { ran_first = true; }),
429+
[](std::unique_ptr<int> p) { return *p; });
430+
EXPECT_EQ(7, a.Perform(std::make_tuple(std::unique_ptr<int>(new int(7)))));
431+
EXPECT_TRUE(ran_first);
432+
}
433+
425434
// The ACTION*() macros trigger warning C4100 (unreferenced formal
426435
// parameter) in MSVC with -W4. Unfortunately they cannot be fixed in
427436
// the macro definition, as the warnings are generated when the macro

0 commit comments

Comments
 (0)