From 8a26410e641f99ceaaa4ae2e4b0d03bfafe07c18 Mon Sep 17 00:00:00 2001 From: stoorx Date: Mon, 24 Feb 2025 16:02:28 +0300 Subject: [PATCH] Add `CombineTo()` generator function `CombineTo()` allows to construct the required type directly from combined arguments. As it would be a composition of `ConvertGenerator()` and `Combine()`, but without `std::tuple` in between. --- googletest/include/gtest/gtest-param-test.h | 41 ++++++++++++ .../include/gtest/internal/gtest-param-util.h | 3 +- googletest/test/googletest-param-test-test.cc | 64 +++++++++++++++++++ 3 files changed, 107 insertions(+), 1 deletion(-) diff --git a/googletest/include/gtest/gtest-param-test.h b/googletest/include/gtest/gtest-param-test.h index f5caed68..9bf66b49 100644 --- a/googletest/include/gtest/gtest-param-test.h +++ b/googletest/include/gtest/gtest-param-test.h @@ -360,6 +360,47 @@ internal::ParamGenerator> Values(Ts... vs) { // inline internal::ParamGenerator Bool() { return Values(false, true); } +// CombineTo() allows the user to combine two or more sequences to produce +// values of a Cartesian product of those sequences' elements converted to +// the required type. +// +// Synopsis: +// CombineTo(gen1, gen2, ..., genN) +// - returns a generator producing sequences with elements coming from +// the Cartesian product of elements from the sequences generated by +// gen1, gen2, ..., genN. The sequence elements will have a type of +// Myclass where elements from sequences produced by gen1, gen2, ..., genN +// was provided to the MyClass constructor parameters. +// +// Example: +// +// This will instantiate tests in test suite AnimalTest each one with +// the parameter values Animal("cat", BLACK), Animal("cat", WHITE), +// Animal("dog", BLACK), and Animal("dog", WHITE): +// enum Color { BLACK, GRAY, WHITE }; +// +// struct Animal { +// std::string name; +// Color color; +// }; +// +// class AnimalTest +// : public testing::TestWithParam {...}; +// +// TEST_P(AnimalTest, AnimalLooksNice) {...} +// +// INSTANTIATE_TEST_SUITE_P(AnimalVariations, AnimalTest, +// CombineTo(Values("cat", "dog"), +// Values(BLACK, WHITE))); +// +template +internal::ParamGenerator CombineTo( + internal::ParamGenerator&&... generators) { + return internal::ParamGenerator( + new internal::CartesianProductGenerator( + std::forward(generators)...)); +} + // Combine() allows the user to combine two or more sequences to produce // values of a Cartesian product of those sequences' elements. // diff --git a/googletest/include/gtest/internal/gtest-param-util.h b/googletest/include/gtest/internal/gtest-param-util.h index b2bdf16d..8c7c34fb 100644 --- a/googletest/include/gtest/internal/gtest-param-util.h +++ b/googletest/include/gtest/internal/gtest-param-util.h @@ -902,7 +902,8 @@ class CartesianProductGenerator : public ParamGeneratorInterface { void ComputeCurrentValue() { if (!AtEnd()) - current_value_ = std::make_shared(*std::get(current_)...); + current_value_ = + std::make_shared(ParamType{*std::get(current_)...}); } bool AtEnd() const { bool at_end = false; diff --git a/googletest/test/googletest-param-test-test.cc b/googletest/test/googletest-param-test-test.cc index bc130601..6c354196 100644 --- a/googletest/test/googletest-param-test-test.cc +++ b/googletest/test/googletest-param-test-test.cc @@ -588,6 +588,70 @@ TEST(ConvertTest, NonDefaultConstructAssign) { EXPECT_TRUE(it == gen.end()); } +TEST(CombineToTest, DefaultConstructible) { + struct DefaultConstructible { + int x; + std::string s; + + bool operator==(const DefaultConstructible& other) const { + return x == other.x && s == other.s; + } + }; + + static_assert(std::is_default_constructible_v); + ParamGenerator gen = + testing::CombineTo(Values(0, 1), Values("A", "B")); + + DefaultConstructible expected_values[] = { + {0, "A"}, {0, "B"}, {1, "A"}, {1, "B"}}; + VerifyGenerator(gen, expected_values); +} + +TEST(CombineToTest, NonDefaultConstructible) { + class NonDefaultConstructible { + public: + NonDefaultConstructible(const int xArg, std::string sArg) + : x(xArg), s(std::move(sArg)) {} + + bool operator==(const NonDefaultConstructible& other) const { + return x == other.x && s == other.s; + } + + private: + int x; + std::string s; + }; + + static_assert(not std::is_default_constructible_v); + ParamGenerator gen = + testing::CombineTo(Values(0, 1), + Values("A", "B")); + + NonDefaultConstructible expected_values[] = { + {0, "A"}, {0, "B"}, {1, "A"}, {1, "B"}}; + VerifyGenerator(gen, expected_values); +} + +TEST(CombineToTest, CopyConstructible) { + struct CopyConstructible { + CopyConstructible(const CopyConstructible& other) = default; + + bool operator==(const CopyConstructible& other) const { + return x == other.x && s == other.s; + } + + int x; + std::string s; + }; + + static_assert(std::is_copy_constructible_v); + ParamGenerator gen = testing::CombineTo( + Values(CopyConstructible{0, "A"}, CopyConstructible{1, "B"})); + CopyConstructible expected_values[] = {CopyConstructible{0, "A"}, + CopyConstructible{1, "B"}}; + VerifyGenerator(gen, expected_values); +} + TEST(ConvertTest, WithConverterLambdaAndDeducedType) { const ParamGenerator> gen = ConvertGenerator(Values("0", std::string("1")), [](const std::string& s) {