Skip to content

Commit d442089

Browse files
Abseil Teammattcalabrese-google
Abseil Team
authored andcommitted
Googletest export
Detect when C++ parametric tests (TEST_P) are not instantiated. When an un-instantiated TEST_P is found, a new test will be inserted that will emit a warning message. This can be made to error with minor code edits. In the future, that is intended to be the default. PiperOrigin-RevId: 284901666
1 parent 88ba008 commit d442089

File tree

5 files changed

+96
-4
lines changed

5 files changed

+96
-4
lines changed

googletest/include/gtest/internal/gtest-param-util.h

+14-1
Original file line numberDiff line numberDiff line change
@@ -42,12 +42,14 @@
4242
#include <memory>
4343
#include <set>
4444
#include <tuple>
45+
#include <type_traits>
4546
#include <utility>
4647
#include <vector>
4748

4849
#include "gtest/internal/gtest-internal.h"
4950
#include "gtest/internal/gtest-port.h"
5051
#include "gtest/gtest-printers.h"
52+
#include "gtest/gtest-test-part.h"
5153

5254
namespace testing {
5355
// Input to a parameterized test name generator, describing a test parameter.
@@ -472,6 +474,8 @@ class ParameterizedTestSuiteInfoBase {
472474
GTEST_DISALLOW_COPY_AND_ASSIGN_(ParameterizedTestSuiteInfoBase);
473475
};
474476

477+
void InsertSyntheticTestCase(const std::string &name, CodeLocation location);
478+
475479
// INTERNAL IMPLEMENTATION - DO NOT USE IN USER CODE.
476480
//
477481
// ParameterizedTestSuiteInfo accumulates tests obtained from TEST_P
@@ -522,11 +526,13 @@ class ParameterizedTestSuiteInfo : public ParameterizedTestSuiteInfoBase {
522526
return 0; // Return value used only to run this method in namespace scope.
523527
}
524528
// UnitTest class invokes this method to register tests in this test suite
525-
// test suites right before running tests in RUN_ALL_TESTS macro.
529+
// right before running tests in RUN_ALL_TESTS macro.
526530
// This method should not be called more than once on any single
527531
// instance of a ParameterizedTestSuiteInfoBase derived class.
528532
// UnitTest has a guard to prevent from calling this method more than once.
529533
void RegisterTests() override {
534+
bool generated_instantiations = false;
535+
530536
for (typename TestInfoContainer::iterator test_it = tests_.begin();
531537
test_it != tests_.end(); ++test_it) {
532538
std::shared_ptr<TestInfo> test_info = *test_it;
@@ -549,6 +555,8 @@ class ParameterizedTestSuiteInfo : public ParameterizedTestSuiteInfoBase {
549555
for (typename ParamGenerator<ParamType>::iterator param_it =
550556
generator.begin();
551557
param_it != generator.end(); ++param_it, ++i) {
558+
generated_instantiations = true;
559+
552560
Message test_name_stream;
553561

554562
std::string param_name = name_func(
@@ -577,6 +585,11 @@ class ParameterizedTestSuiteInfo : public ParameterizedTestSuiteInfoBase {
577585
} // for param_it
578586
} // for gen_it
579587
} // for test_it
588+
589+
if (!generated_instantiations) {
590+
// There are no generaotrs, or they all generate nothing ...
591+
InsertSyntheticTestCase(GetTestSuiteName(), code_location_);
592+
}
580593
} // RegisterTests
581594

582595
private:

googletest/src/gtest.cc

+60
Original file line numberDiff line numberDiff line change
@@ -407,6 +407,66 @@ void AssertHelper::operator=(const Message& message) const {
407407
); // NOLINT
408408
}
409409

410+
namespace {
411+
412+
// When TEST_P is found without a matching INSTANTIATE_TEST_SUITE_P
413+
// to creates test cases for it, a syntetic test case is
414+
// inserted to report ether an error or a log message.
415+
//
416+
// This configuration bit will likely be removed at some point.
417+
constexpr bool kErrorOnUninstantiatedParameterizedTest = false;
418+
419+
// A test that fails at a given file/line location with a given message.
420+
class FailureTest : public Test {
421+
public:
422+
explicit FailureTest(const CodeLocation& loc, std::string error_message,
423+
bool as_error)
424+
: loc_(loc),
425+
error_message_(std::move(error_message)),
426+
as_error_(as_error) {}
427+
428+
void TestBody() override {
429+
if (as_error_) {
430+
AssertHelper(TestPartResult::kNonFatalFailure, loc_.file.c_str(),
431+
loc_.line, "") = Message() << error_message_;
432+
} else {
433+
std::cout << error_message_ << std::endl;
434+
}
435+
}
436+
437+
private:
438+
const CodeLocation loc_;
439+
const std::string error_message_;
440+
const bool as_error_;
441+
};
442+
443+
444+
} // namespace
445+
446+
// If this parameterized test suite has no instantiations (and that
447+
// has not been marked as okay), emit a test case reporting that.
448+
void InsertSyntheticTestCase(const std::string &name, CodeLocation location) {
449+
std::string message =
450+
"Paramaterized test suite " + name +
451+
" is defined via TEST_P, but never instantiated. None of the test cases "
452+
"will run. Either no INSTANTIATE_TEST_SUITE_P is provided or the only "
453+
"ones provided expand to nothing."
454+
"\n\n"
455+
"Ideally, TEST_P definitions should only ever be included as part of "
456+
"binaries that intend to use them. (As opposed to, for example, being "
457+
"placed in a library that may be linked in to get other utilities.)";
458+
459+
std::string full_name = "UninstantiatedParamaterizedTestSuite<" + name + ">";
460+
RegisterTest( //
461+
"GoogleTestVerification", full_name.c_str(),
462+
nullptr, // No type parameter.
463+
nullptr, // No value parameter.
464+
location.file.c_str(), location.line, [message, location] {
465+
return new FailureTest(location, message,
466+
kErrorOnUninstantiatedParameterizedTest);
467+
});
468+
}
469+
410470
// A copy of all command line arguments. Set by InitGoogleTest().
411471
static ::std::vector<std::string> g_argvs;
412472

googletest/test/googletest-output-test-golden-lin.txt

+9-3
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@ Expected equality of these values:
1212
3
1313
Stack trace: (omitted)
1414

15-
[==========] Running 84 tests from 39 test suites.
15+
[==========] Running 85 tests from 40 test suites.
1616
[----------] Global test environment set-up.
1717
FooEnvironment::SetUp() called.
1818
BarEnvironment::SetUp() called.
@@ -979,6 +979,12 @@ Expected failure
979979
Stack trace: (omitted)
980980

981981
[ FAILED ] PrintingStrings/ParamTest.Failure/a, where GetParam() = "a"
982+
[----------] 1 test from GoogleTestVerification
983+
[ RUN ] GoogleTestVerification.UninstantiatedParamaterizedTestSuite<DetectNotInstantiatedTest>
984+
Paramaterized test suite DetectNotInstantiatedTest is defined via TEST_P, but never instantiated. None of the test cases will run. Either no INSTANTIATE_TEST_SUITE_P is provided or the only ones provided expand to nothing.
985+
986+
Ideally, TEST_P definitions should only ever be included as part of binaries that intend to use them. (As opposed to, for example, being placed in a library that may be linked in to get other utilities.)
987+
[ OK ] GoogleTestVerification.UninstantiatedParamaterizedTestSuite<DetectNotInstantiatedTest>
982988
[----------] Global test environment tear-down
983989
BarEnvironment::TearDown() called.
984990
googletest-output-test_.cc:#: Failure
@@ -992,8 +998,8 @@ Failed
992998
Expected fatal failure.
993999
Stack trace: (omitted)
9941000

995-
[==========] 84 tests from 39 test suites ran.
996-
[ PASSED ] 30 tests.
1001+
[==========] 85 tests from 40 test suites ran.
1002+
[ PASSED ] 31 tests.
9971003
[ FAILED ] 54 tests, listed below:
9981004
[ FAILED ] NonfatalFailureTest.EscapesStringOperands
9991005
[ FAILED ] NonfatalFailureTest.DiffForLongStrings

googletest/test/googletest-output-test_.cc

+7
Original file line numberDiff line numberDiff line change
@@ -782,6 +782,13 @@ INSTANTIATE_TEST_SUITE_P(PrintingStrings,
782782
testing::Values(std::string("a")),
783783
ParamNameFunc);
784784

785+
// fails under kErrorOnUninstantiatedParameterizedTest=true
786+
class DetectNotInstantiatedTest : public testing::TestWithParam<int> {};
787+
TEST_P(DetectNotInstantiatedTest, Used) { }
788+
789+
// This would make the test failure from the above go away.
790+
// INSTANTIATE_TEST_SUITE_P(Fix, DetectNotInstantiatedTest, testing::Values(1));
791+
785792
// This #ifdef block tests the output of typed tests.
786793
#if GTEST_HAS_TYPED_TEST
787794

googletest/test/googletest-param-test-test.cc

+6
Original file line numberDiff line numberDiff line change
@@ -1068,6 +1068,12 @@ TEST_P(MyEnumTest, ChecksParamMoreThanZero) { EXPECT_GE(10, GetParam()); }
10681068
INSTANTIATE_TEST_SUITE_P(MyEnumTests, MyEnumTest,
10691069
::testing::Values(ENUM1, ENUM2, 0));
10701070

1071+
namespace works_here {
1072+
// Never used not instantiated, this should work.
1073+
class NotUsedTest : public testing::TestWithParam<int> {};
1074+
1075+
} // namespace works_here
1076+
10711077
int main(int argc, char **argv) {
10721078
// Used in TestGenerationTest test suite.
10731079
AddGlobalTestEnvironment(TestGenerationTest::Environment::Instance());

0 commit comments

Comments
 (0)