From b427d1985e76f1e3362056107670aba2a223346d Mon Sep 17 00:00:00 2001 From: Zach Mullen Date: Fri, 28 Aug 2009 11:08:39 -0400 Subject: Added ctest -N test. Fixed ctest working directory bug. MemCheck fix coming soon... --- Source/CTest/cmCTestMemCheckHandler.cxx | 2 +- Source/CTest/cmCTestMultiProcessHandler.cxx | 95 +++++++++++------------- Source/CTest/cmCTestMultiProcessHandler.h | 6 +- Source/CTest/cmCTestRunTest.cxx | 108 ++++++++++++++-------------- Source/CTest/cmCTestRunTest.h | 15 ++-- Source/CTest/cmCTestTestHandler.cxx | 40 +++-------- Source/CTest/cmCTestTestHandler.h | 1 - Tests/CMakeLists.txt | 3 + 8 files changed, 117 insertions(+), 153 deletions(-) diff --git a/Source/CTest/cmCTestMemCheckHandler.cxx b/Source/CTest/cmCTestMemCheckHandler.cxx index eabc6cf..a0d74a3 100644 --- a/Source/CTest/cmCTestMemCheckHandler.cxx +++ b/Source/CTest/cmCTestMemCheckHandler.cxx @@ -867,7 +867,7 @@ bool cmCTestMemCheckHandler::ProcessMemCheckBoundsCheckerOutput( } return true; } - +//TODO ZACH move this logic into cmCTestRunTest void cmCTestMemCheckHandler::ProcessOneTest(cmCTestTestProperties *props, std::vector &passed, diff --git a/Source/CTest/cmCTestMultiProcessHandler.cxx b/Source/CTest/cmCTestMultiProcessHandler.cxx index 1c81825..88d1696 100644 --- a/Source/CTest/cmCTestMultiProcessHandler.cxx +++ b/Source/CTest/cmCTestMultiProcessHandler.cxx @@ -24,6 +24,7 @@ cmCTestMultiProcessHandler::cmCTestMultiProcessHandler() { this->ParallelLevel = 1; + this->Completed = 0; } // Set the tests void @@ -38,6 +39,7 @@ cmCTestMultiProcessHandler::SetTests(TestMap& tests, this->TestFinishMap[i->first] = false; } this->Tests = tests; + this->Total = this->Tests.size(); this->Properties = properties; } @@ -81,7 +83,7 @@ void cmCTestMultiProcessHandler::StartTestProcess(int test) } else { - testRun->EndTest(); + testRun->EndTest(this->Completed, this->Total); } } @@ -130,9 +132,9 @@ bool cmCTestMultiProcessHandler::StartTest(int test) void cmCTestMultiProcessHandler::StartNextTests() { - cmCTestLog(this->CTest, HANDLER_VERBOSE_OUTPUT, std::endl - << "Number of running tests : " << this->RunningTests.size() - << "\n"); + //cmCTestLog(this->CTest, HANDLER_VERBOSE_OUTPUT, std::endl + // << "Number of running tests : " << this->RunningTests.size() + // << "\n"); size_t numToStart = this->ParallelLevel - this->RunningTests.size(); if(numToStart == 0) { @@ -183,16 +185,17 @@ bool cmCTestMultiProcessHandler::CheckOutput() for( std::vector::iterator i = finished.begin(); i != finished.end(); ++i) { + this->Completed++; cmCTestRunTest* p = *i; int test = p->GetIndex(); - - if(p->EndTest()) + + if(p->EndTest(this->Completed, this->Total)) { - this->Passed->push_back(p->GetTestProperties()->Name); + this->Passed->push_back(p->GetTestProperties()->Name); } else { - this->Failed->push_back(p->GetTestProperties()->Name); + this->Failed->push_back(p->GetTestProperties()->Name); } for(TestMap::iterator j = this->Tests.begin(); j!= this->Tests.end(); ++j) @@ -226,19 +229,38 @@ void cmCTestMultiProcessHandler::MarkFinished() cmSystemTools::RemoveFile(fname.c_str()); } -void cmCTestMultiProcessHandler::PrintTests() +//--------------------------------------------------------------------- +//For ShowOnly mode +void cmCTestMultiProcessHandler::PrintTestList() { -#undef cout - for( TestMap::iterator i = this->Tests.begin(); - i!= this->Tests.end(); ++i) + int count = 0; + for (PropertiesMap::iterator it = this->Properties.begin(); + it != this->Properties.end(); it ++ ) { - std::cout << "Test " << i->first << " ("; - for(TestSet::iterator j = i->second.begin(); - j != i->second.end(); ++j) + count++; + cmCTestTestHandler::cmCTestTestProperties& p = *it->second; + + cmCTestRunTest testRun; + testRun.SetCTest(this->CTest); + testRun.SetTestHandler(this->TestHandler); + testRun.SetIndex(p.Index); + testRun.SetTestProperties(&p); + testRun.ComputeArguments(); //logs the command in verbose mode + + cmCTestLog(this->CTest, HANDLER_OUTPUT, std::setw(3) + << count << "/"); + cmCTestLog(this->CTest, HANDLER_OUTPUT, std::setw(3) + << this->Total << " "); + if (this->TestHandler->MemCheck) + { + cmCTestLog(this->CTest, HANDLER_OUTPUT, "Memory Check"); + } + else { - std::cout << *j << " "; + cmCTestLog(this->CTest, HANDLER_OUTPUT, "Testing"); } - std::cout << ")\n"; + cmCTestLog(this->CTest, HANDLER_OUTPUT, " "); + cmCTestLog(this->CTest, HANDLER_OUTPUT, p.Name.c_str() << std::endl); } } @@ -282,42 +304,5 @@ void cmCTestMultiProcessHandler::RemoveTest(int index) this->Properties.erase(index); this->TestRunningMap[index] = false; this->TestFinishMap[index] = true; + this->Completed++; } - -#if 0 -int main() -{ - cmCTestMultiProcessHandler h; - h.SetParallelLevel(4); - std::map > tests; - std::set depends; - for(int i =1; i < 92; i++) - { - tests[i] = depends; - } - depends.clear(); - depends.insert(45); subprject - tests[46] = depends; subproject-stage2 - depends.clear(); - depends.insert(55); simpleinstall simpleinstall-s2 - tests[56] = depends; - depends.clear(); - depends.insert(70); wrapping - tests[71] = depends; qtwrapping - depends.clear(); - depends.insert(71); qtwrapping - tests[72] = depends; testdriver1 - depends.clear(); - depends.insert(72) testdriver1 - tests[73] = depends; testdriver2 - depends.clear(); - depends.insert(73); testdriver2 - tests[74] = depends; testdriver3 - depends.clear(); - depends.insert(79); linkorder1 - tests[80] = depends; linkorder2 - h.SetTests(tests); - h.PrintTests(); - h.RunTests(); -} -#endif diff --git a/Source/CTest/cmCTestMultiProcessHandler.h b/Source/CTest/cmCTestMultiProcessHandler.h index 3a47a36..cb3aecb 100644 --- a/Source/CTest/cmCTestMultiProcessHandler.h +++ b/Source/CTest/cmCTestMultiProcessHandler.h @@ -40,7 +40,7 @@ public: // Set the max number of tests that can be run at the same time. void SetParallelLevel(size_t); void RunTests(); - void PrintTests(); + void PrintTestList(); //void SetCTestCommand(const char* c) { this->CTestCommand = c;} //void SetTestCacheFile(const char* c) { this->CTestCacheFile = c;} void SetPassFailVectors(std::vector* passed, @@ -80,6 +80,10 @@ protected: void CheckResume(); // map from test number to set of depend tests TestMap Tests; + //Total number of tests we'll be running + int Total; + //Number of tests that are complete + int Completed; //list of test properties (indices concurrent to the test map) PropertiesMap Properties; std::map TestRunningMap; diff --git a/Source/CTest/cmCTestRunTest.cxx b/Source/CTest/cmCTestRunTest.cxx index 7b42a3d..b9bcb08 100644 --- a/Source/CTest/cmCTestRunTest.cxx +++ b/Source/CTest/cmCTestRunTest.cxx @@ -55,14 +55,14 @@ void cmCTestRunTest::CheckOutput() } //--------------------------------------------------------- -bool cmCTestRunTest::EndTest() +bool cmCTestRunTest::EndTest(int completed, int total) { //restore the old environment if (this->ModifyEnv) { cmSystemTools::RestoreEnv(this->OrigEnv); } - this->WriteLogOutputTop(); + this->WriteLogOutputTop(completed, total); std::string reason; bool passed = true; int res = this->TestProcess->GetProcessStatus(); @@ -259,6 +259,7 @@ void cmCTestRunTest::SetTestHandler(cmCTestTestHandler * handler) // Starts the execution of a test. Returns once it has started bool cmCTestRunTest::StartTest() { + this->ComputeArguments(); std::vector& args = this->TestProperties->Args; this->TestResult.Properties = this->TestProperties; this->TestResult.ExecutionTime = 0; @@ -268,12 +269,6 @@ bool cmCTestRunTest::StartTest() this->TestResult.Name = this->TestProperties->Name; this->TestResult.Path = this->TestProperties->Directory.c_str(); - // find the test executable - this->ActualCommand - = this->TestHandler->FindTheExecutable(args[1].c_str()); - this->TestCommand - = cmSystemTools::ConvertToOutputPath(this->ActualCommand.c_str()); - // continue if we did not find the executable if (this->TestCommand == "") { @@ -289,22 +284,42 @@ bool cmCTestRunTest::StartTest() return false; } } + this->StartTime = this->CTest->CurrentTime(); + + return this->CreateProcess(this->TestProperties->Timeout, + &this->TestProperties->Environment); +} + +void cmCTestRunTest::ComputeArguments() +{ + std::vector& args = this->TestProperties->Args; + // find the test executable + this->ActualCommand + = this->TestHandler->FindTheExecutable(args[1].c_str()); + this->TestCommand + = cmSystemTools::ConvertToOutputPath(this->ActualCommand.c_str()); + // add the arguments + std::vector::const_iterator j = + this->TestProperties->Args.begin(); + ++j; // skip test name + ++j; // skip command as it is in actualCommand + + //TODO ZACH the problem is here for memcheck. We need to call +//memcheckhandler.GenerateTestCommand BEFORE we run the process. +// this->TestHandler->GenerateTestCommand(this->Arguments); + for(;j != this->TestProperties->Args.end(); ++j) + { + this->TestCommand += " "; + this->TestCommand += cmSystemTools::EscapeSpaces(j->c_str()); + this->Arguments.push_back(*j); + } + this->TestResult.FullCommandLine = this->TestCommand; - /** - * Run an executable command and put the stdout in output. - */ cmCTestLog(this->CTest, HANDLER_VERBOSE_OUTPUT, std::endl << this->Index << ": " << (this->TestHandler->MemCheck?"MemCheck":"Test") << " command: " << this->TestCommand << std::endl); - - this->StartTime = this->CTest->CurrentTime(); - - return this->CreateProcess(this->ActualCommand, - this->TestProperties->Args, - this->TestProperties->Timeout, - &this->TestProperties->Environment); } //---------------------------------------------------------------------- @@ -330,24 +345,14 @@ void cmCTestRunTest::DartProcessing() } //---------------------------------------------------------------------- -bool cmCTestRunTest::CreateProcess(std::string command, - std::vector args, - double testTimeOut, +bool cmCTestRunTest::CreateProcess(double testTimeOut, std::vector* environment) { - std::vector commandArgs; - std::vector::iterator i = args.begin(); - - ++i; //skip test name - ++i; //skip executable name - for(; i != args.end(); ++i) - { - commandArgs.push_back(*i); - } this->TestProcess = new cmProcess; this->TestProcess->SetId(this->Index); - this->TestProcess->SetCommand(command.c_str()); - this->TestProcess->SetCommandArguments(commandArgs); + this->TestProcess->SetWorkingDirectory(this->TestProperties->Directory.c_str()); + this->TestProcess->SetCommand(this->ActualCommand.c_str()); + this->TestProcess->SetCommandArguments(this->Arguments); std::vector origEnv; this->ModifyEnv = (environment && environment->size()>0); @@ -380,15 +385,16 @@ bool cmCTestRunTest::CreateProcess(std::string command, return this->TestProcess->StartProcess(); } -void cmCTestRunTest::WriteLogOutputTop() +void cmCTestRunTest::WriteLogOutputTop(int completed, int total) { /* Not sure whether we want to prepend the test index anymore cmCTestLog(this->CTest, HANDLER_OUTPUT, std::setw(3) << this->Index << ": ");*/ cmCTestLog(this->CTest, HANDLER_OUTPUT, std::setw(3) - << this->TestProperties->Index << "/"); + << completed << "/"); cmCTestLog(this->CTest, HANDLER_OUTPUT, std::setw(3) - << this->TestHandler->TotalNumberOfTests << " "); + << total << " "); + if ( this->TestHandler->MemCheck ) { cmCTestLog(this->CTest, HANDLER_OUTPUT, "Memory Check"); @@ -397,39 +403,29 @@ void cmCTestRunTest::WriteLogOutputTop() { cmCTestLog(this->CTest, HANDLER_OUTPUT, "Testing"); } + + std::stringstream indexStr; + indexStr << " (" << this->Index << ")"; + cmCTestLog(this->CTest, HANDLER_OUTPUT, std::setw(6) + << indexStr.str().c_str()); cmCTestLog(this->CTest, HANDLER_OUTPUT, " "); const int maxTestNameWidth = this->CTest->GetMaxTestNameWidth(); std::string outname = this->TestProperties->Name + " "; outname.resize(maxTestNameWidth, '.'); - // add the arguments - std::vector::const_iterator j = - this->TestProperties->Args.begin(); - ++j; // skip test name - ++j; // skip command as it is in actualCommand - std::vector arguments; - this->TestHandler->GenerateTestCommand(arguments); - arguments.push_back(this->ActualCommand.c_str()); - for(;j != this->TestProperties->Args.end(); ++j) - { - this->TestCommand += " "; - this->TestCommand += cmSystemTools::EscapeSpaces(j->c_str()); - arguments.push_back(j->c_str()); - } - arguments.push_back(0); - this->TestResult.FullCommandLine = this->TestCommand; - *this->TestHandler->LogFile << this->TestProperties->Index << "/" << this->TestHandler->TotalNumberOfTests << " Testing: " << this->TestProperties->Name << std::endl; *this->TestHandler->LogFile << this->TestProperties->Index << "/" << this->TestHandler->TotalNumberOfTests << " Test: " << this->TestProperties->Name.c_str() << std::endl; - *this->TestHandler->LogFile << "Command: "; - std::vector::size_type ll; - for ( ll = 0; ll < arguments.size()-1; ll ++ ) + *this->TestHandler->LogFile << "Command: \"" << this->ActualCommand << "\""; + + for (std::vector::iterator i = this->Arguments.begin(); + i != this->Arguments.end(); ++i) { - *this->TestHandler->LogFile << "\"" << arguments[ll] << "\" "; + *this->TestHandler->LogFile + << " \"" << i->c_str() << "\""; } *this->TestHandler->LogFile << std::endl << "Directory: " << this->TestProperties->Directory << std::endl diff --git a/Source/CTest/cmCTestRunTest.h b/Source/CTest/cmCTestRunTest.h index 2609808..521759a 100644 --- a/Source/CTest/cmCTestRunTest.h +++ b/Source/CTest/cmCTestRunTest.h @@ -55,15 +55,15 @@ public: void CheckOutput(); //launch the test process, return whether it started correctly bool StartTest(); - //capture the test results and send them back to the test handler - bool EndTest(); -protected: + //capture and report the test results + bool EndTest(int completed, int total); + //Called by ctest -N to log the command string + void ComputeArguments(); +private: void DartProcessing(); - bool CreateProcess(std::string executable, - std::vector args, - double testTimeOut, + bool CreateProcess(double testTimeOut, std::vector* environment); - void WriteLogOutputTop(); + void WriteLogOutputTop(int completed, int total); cmCTestTestHandler::cmCTestTestProperties * TestProperties; //Pointer back to the "parent"; the handler that invoked this test run @@ -86,6 +86,7 @@ protected: std::string StartTime; std::string TestCommand; std::string ActualCommand; + std::vector Arguments; }; #endif diff --git a/Source/CTest/cmCTestTestHandler.cxx b/Source/CTest/cmCTestTestHandler.cxx index b51b7c5..f2e917a 100644 --- a/Source/CTest/cmCTestTestHandler.cxx +++ b/Source/CTest/cmCTestTestHandler.cxx @@ -1304,41 +1304,11 @@ bool cmCTestTestHandler::GetValue(const char* tag, } //--------------------------------------------------------------------- -void cmCTestTestHandler::PrintTestList() -{ - int total = this->TotalNumberOfTests; - for (ListOfTests::iterator it = this->TestList.begin(); - it != this->TestList.end(); it ++ ) - { - cmCTestTestProperties& p = *it; - cmCTestLog(this->CTest, HANDLER_OUTPUT, std::setw(3) - << p.Index << "/"); - cmCTestLog(this->CTest, HANDLER_OUTPUT, std::setw(3) - << total << " "); - if (this->MemCheck) - { - cmCTestLog(this->CTest, HANDLER_OUTPUT, "Memory Check"); - } - else - { - cmCTestLog(this->CTest, HANDLER_OUTPUT, "Testing"); - } - cmCTestLog(this->CTest, HANDLER_OUTPUT, " "); - cmCTestLog(this->CTest, HANDLER_OUTPUT, p.Name.c_str() << std::endl); - } -} - -//--------------------------------------------------------------------- void cmCTestTestHandler::ProcessDirectory(std::vector &passed, std::vector &failed) { this->ComputeTestList(); - if(this->CTest->GetShowOnly()) - { - this->PrintTestList(); - return; - } cmCTestMultiProcessHandler parallel; parallel.SetCTest(this->CTest); parallel.SetParallelLevel(this->CTest->GetParallelLevel()); @@ -1380,8 +1350,14 @@ void cmCTestTestHandler::ProcessDirectory(std::vector &passed, parallel.SetPassFailVectors(&passed, &failed); this->TestResults.clear(); parallel.SetTestResults(&this->TestResults); - parallel.RunTests(); - + if(this->CTest->GetShowOnly()) + { + parallel.PrintTestList(); + } + else + { + parallel.RunTests(); + } *this->LogFile << "End testing: " << this->CTest->CurrentTime() << std::endl; } diff --git a/Source/CTest/cmCTestTestHandler.h b/Source/CTest/cmCTestTestHandler.h index 9dbb52c..55fb853 100644 --- a/Source/CTest/cmCTestTestHandler.h +++ b/Source/CTest/cmCTestTestHandler.h @@ -141,7 +141,6 @@ protected: void WriteTestResultHeader(std::ostream& os, cmCTestTestResult* result); void WriteTestResultFooter(std::ostream& os, cmCTestTestResult* result); - void PrintTestList(); //! Clean test output to specified length bool CleanTestOutput(std::string& output, size_t length); diff --git a/Tests/CMakeLists.txt b/Tests/CMakeLists.txt index c7e6715..6084e79 100644 --- a/Tests/CMakeLists.txt +++ b/Tests/CMakeLists.txt @@ -1052,6 +1052,9 @@ ${CMake_BINARY_DIR}/bin/cmake -DVERSION=CVS -P ${CMake_SOURCE_DIR}/Utilities/Rel -S "${CMake_BINARY_DIR}/Tests/CTestTest3/test.cmake" -V --output-log "${CMake_BINARY_DIR}/Tests/CTestTest3/testOutput.log" ) + ADD_TEST(CTestTestShowOnly ${CMAKE_CTEST_COMMAND} + -N + ) # these tests take a long time, make sure they have it # if timeouts have not already been set GET_TEST_PROPERTY(CTestTest TIMEOUT PREVIOUS_TIMEOUT) -- cgit v0.12