/*========================================================================= Program: CMake - Cross-Platform Makefile Generator Module: $RCSfile$ Language: C++ Date: $Date$ Version: $Revision$ Copyright (c) 2002 Kitware, Inc., Insight Consortium. All rights reserved. See Copyright.txt or http://www.cmake.org/HTML/Copyright.html for details. This software is distributed WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the above copyright notices for more information. =========================================================================*/ #include "cmCTestTestHandler.h" #include "cmCTestMultiProcessHandler.h" #include "cmCTest.h" #include "cmCTestRunTest.h" #include "cmake.h" #include "cmGeneratedFileStream.h" #include #include #include #include "cmMakefile.h" #include "cmGlobalGenerator.h" #include "cmLocalGenerator.h" #include "cmCommand.h" #include "cmSystemTools.h" #include "cmXMLSafe.h" #include #include #include #include // auto_ptr //---------------------------------------------------------------------- class cmCTestSubdirCommand : public cmCommand { public: /** * This is a virtual constructor for the command. */ virtual cmCommand* Clone() { cmCTestSubdirCommand* c = new cmCTestSubdirCommand; c->TestHandler = this->TestHandler; return c; } /** * This is called when the command is first encountered in * the CMakeLists.txt file. */ virtual bool InitialPass(std::vector const& args, cmExecutionStatus &); /** * The name of the command as specified in CMakeList.txt. */ virtual const char* GetName() { return "subdirs";} // Unused methods virtual const char* GetTerseDocumentation() { return ""; } virtual const char* GetFullDocumentation() { return ""; } cmTypeMacro(cmCTestSubdirCommand, cmCommand); cmCTestTestHandler* TestHandler; }; //---------------------------------------------------------------------- bool cmCTestSubdirCommand ::InitialPass(std::vector const& args, cmExecutionStatus &) { if(args.size() < 1 ) { this->SetError("called with incorrect number of arguments"); return false; } std::vector::const_iterator it; std::string cwd = cmSystemTools::GetCurrentWorkingDirectory(); for ( it = args.begin(); it != args.end(); ++ it ) { cmSystemTools::ChangeDirectory(cwd.c_str()); std::string fname = cwd; fname += "/"; fname += *it; if ( !cmSystemTools::FileExists(fname.c_str()) ) { // No subdirectory? So what... continue; } cmSystemTools::ChangeDirectory(fname.c_str()); const char* testFilename; if( cmSystemTools::FileExists("CTestTestfile.cmake") ) { // does the CTestTestfile.cmake exist ? testFilename = "CTestTestfile.cmake"; } else if( cmSystemTools::FileExists("DartTestfile.txt") ) { // does the DartTestfile.txt exist ? testFilename = "DartTestfile.txt"; } else { // No CTestTestfile? Who cares... cmSystemTools::ChangeDirectory(cwd.c_str()); continue; } fname += "/"; fname += testFilename; bool readit = this->Makefile->ReadListFile(this->Makefile->GetCurrentListFile(), fname.c_str()); cmSystemTools::ChangeDirectory(cwd.c_str()); if(!readit) { std::string m = "Could not find include file: "; m += fname; this->SetError(m.c_str()); return false; } } return true; } //---------------------------------------------------------------------- class cmCTestAddSubdirectoryCommand : public cmCommand { public: /** * This is a virtual constructor for the command. */ virtual cmCommand* Clone() { cmCTestAddSubdirectoryCommand* c = new cmCTestAddSubdirectoryCommand; c->TestHandler = this->TestHandler; return c; } /** * This is called when the command is first encountered in * the CMakeLists.txt file. */ virtual bool InitialPass(std::vector const& args, cmExecutionStatus &); /** * The name of the command as specified in CMakeList.txt. */ virtual const char* GetName() { return "add_subdirectory";} // Unused methods virtual const char* GetTerseDocumentation() { return ""; } virtual const char* GetFullDocumentation() { return ""; } cmTypeMacro(cmCTestAddSubdirectoryCommand, cmCommand); cmCTestTestHandler* TestHandler; }; //---------------------------------------------------------------------- bool cmCTestAddSubdirectoryCommand ::InitialPass(std::vector const& args, cmExecutionStatus &) { if(args.size() < 1 ) { this->SetError("called with incorrect number of arguments"); return false; } std::string cwd = cmSystemTools::GetCurrentWorkingDirectory(); cmSystemTools::ChangeDirectory(cwd.c_str()); std::string fname = cwd; fname += "/"; fname += args[1]; if ( !cmSystemTools::FileExists(fname.c_str()) ) { // No subdirectory? So what... return true; } cmSystemTools::ChangeDirectory(fname.c_str()); const char* testFilename; if( cmSystemTools::FileExists("CTestTestfile.cmake") ) { // does the CTestTestfile.cmake exist ? testFilename = "CTestTestfile.cmake"; } else if( cmSystemTools::FileExists("DartTestfile.txt") ) { // does the DartTestfile.txt exist ? testFilename = "DartTestfile.txt"; } else { // No CTestTestfile? Who cares... cmSystemTools::ChangeDirectory(cwd.c_str()); return true; } fname += "/"; fname += testFilename; bool readit = this->Makefile->ReadListFile(this->Makefile->GetCurrentListFile(), fname.c_str()); cmSystemTools::ChangeDirectory(cwd.c_str()); if(!readit) { std::string m = "Could not find include file: "; m += fname; this->SetError(m.c_str()); return false; } return true; } //---------------------------------------------------------------------- class cmCTestAddTestCommand : public cmCommand { public: /** * This is a virtual constructor for the command. */ virtual cmCommand* Clone() { cmCTestAddTestCommand* c = new cmCTestAddTestCommand; c->TestHandler = this->TestHandler; return c; } /** * This is called when the command is first encountered in * the CMakeLists.txt file. */ virtual bool InitialPass(std::vector const&, cmExecutionStatus &); /** * The name of the command as specified in CMakeList.txt. */ virtual const char* GetName() { return "ADD_TEST";} // Unused methods virtual const char* GetTerseDocumentation() { return ""; } virtual const char* GetFullDocumentation() { return ""; } cmTypeMacro(cmCTestAddTestCommand, cmCommand); cmCTestTestHandler* TestHandler; }; //---------------------------------------------------------------------- bool cmCTestAddTestCommand ::InitialPass(std::vector const& args, cmExecutionStatus &) { if ( args.size() < 2 ) { this->SetError("called with incorrect number of arguments"); return false; } return this->TestHandler->AddTest(args); } //---------------------------------------------------------------------- class cmCTestSetTestsPropertiesCommand : public cmCommand { public: /** * This is a virtual constructor for the command. */ virtual cmCommand* Clone() { cmCTestSetTestsPropertiesCommand* c = new cmCTestSetTestsPropertiesCommand; c->TestHandler = this->TestHandler; return c; } /** * This is called when the command is first encountered in * the CMakeLists.txt file. */ virtual bool InitialPass(std::vector const&, cmExecutionStatus &); /** * The name of the command as specified in CMakeList.txt. */ virtual const char* GetName() { return "SET_TESTS_PROPERTIES";} // Unused methods virtual const char* GetTerseDocumentation() { return ""; } virtual const char* GetFullDocumentation() { return ""; } cmTypeMacro(cmCTestSetTestsPropertiesCommand, cmCommand); cmCTestTestHandler* TestHandler; }; //---------------------------------------------------------------------- bool cmCTestSetTestsPropertiesCommand ::InitialPass(std::vector const& args, cmExecutionStatus &) { return this->TestHandler->SetTestsProperties(args); } //---------------------------------------------------------------------- // get the next number in a string with numbers separated by , // pos is the start of the search and pos2 is the end of the search // pos becomes pos2 after a call to GetNextNumber. // -1 is returned at the end of the list. inline int GetNextNumber(std::string const& in, int& val, std::string::size_type& pos, std::string::size_type& pos2) { pos2 = in.find(',', pos); if(pos2 != in.npos) { if(pos2-pos == 0) { val = -1; } else { val = atoi(in.substr(pos, pos2-pos).c_str()); } pos = pos2+1; return 1; } else { if(in.size()-pos == 0) { val = -1; } else { val = atoi(in.substr(pos, in.size()-pos).c_str()); } return 0; } } //---------------------------------------------------------------------- // get the next number in a string with numbers separated by , // pos is the start of the search and pos2 is the end of the search // pos becomes pos2 after a call to GetNextNumber. // -1 is returned at the end of the list. inline int GetNextRealNumber(std::string const& in, double& val, std::string::size_type& pos, std::string::size_type& pos2) { pos2 = in.find(',', pos); if(pos2 != in.npos) { if(pos2-pos == 0) { val = -1; } else { val = atof(in.substr(pos, pos2-pos).c_str()); } pos = pos2+1; return 1; } else { if(in.size()-pos == 0) { val = -1; } else { val = atof(in.substr(pos, in.size()-pos).c_str()); } return 0; } } //---------------------------------------------------------------------- cmCTestTestHandler::cmCTestTestHandler() { this->UseUnion = false; this->UseIncludeLabelRegExpFlag = false; this->UseExcludeLabelRegExpFlag = false; this->UseIncludeRegExpFlag = false; this->UseExcludeRegExpFlag = false; this->UseExcludeRegExpFirst = false; this->CustomMaximumPassedTestOutputSize = 1 * 1024; this->CustomMaximumFailedTestOutputSize = 300 * 1024; this->MemCheck = false; this->LogFile = 0; // regex to detect ... this->DartStuff.compile( "()"); // regex to detect each individual ... this->DartStuff1.compile( "()"); } //---------------------------------------------------------------------- void cmCTestTestHandler::Initialize() { this->Superclass::Initialize(); this->ElapsedTestingTime = -1; this->TestResults.clear(); this->CustomTestsIgnore.clear(); this->StartTest = ""; this->EndTest = ""; this->CustomPreTest.clear(); this->CustomPostTest.clear(); this->CustomMaximumPassedTestOutputSize = 1 * 1024; this->CustomMaximumFailedTestOutputSize = 300 * 1024; this->TestsToRun.clear(); this->UseIncludeRegExpFlag = false; this->UseExcludeRegExpFlag = false; this->UseExcludeRegExpFirst = false; this->IncludeLabelRegularExpression = ""; this->ExcludeLabelRegularExpression = ""; this->IncludeRegExp = ""; this->ExcludeRegExp = ""; TestsToRunString = ""; this->UseUnion = false; this->TestList.clear(); } //---------------------------------------------------------------------- void cmCTestTestHandler::PopulateCustomVectors(cmMakefile *mf) { this->CTest->PopulateCustomVector(mf, "CTEST_CUSTOM_PRE_TEST", this->CustomPreTest); this->CTest->PopulateCustomVector(mf, "CTEST_CUSTOM_POST_TEST", this->CustomPostTest); this->CTest->PopulateCustomVector(mf, "CTEST_CUSTOM_TESTS_IGNORE", this->CustomTestsIgnore); this->CTest->PopulateCustomInteger(mf, "CTEST_CUSTOM_MAXIMUM_PASSED_TEST_OUTPUT_SIZE", this->CustomMaximumPassedTestOutputSize); this->CTest->PopulateCustomInteger(mf, "CTEST_CUSTOM_MAXIMUM_FAILED_TEST_OUTPUT_SIZE", this->CustomMaximumFailedTestOutputSize); } //---------------------------------------------------------------------- int cmCTestTestHandler::PreProcessHandler() { if ( !this->ExecuteCommands(this->CustomPreTest) ) { cmCTestLog(this->CTest, ERROR_MESSAGE, "Problem executing pre-test command(s)." << std::endl); return 0; } return 1; } //---------------------------------------------------------------------- int cmCTestTestHandler::PostProcessHandler() { if ( !this->ExecuteCommands(this->CustomPostTest) ) { cmCTestLog(this->CTest, ERROR_MESSAGE, "Problem executing post-test command(s)." << std::endl); return 0; } return 1; } //---------------------------------------------------------------------- //clearly it would be nice if this were broken up into a few smaller //functions and commented... int cmCTestTestHandler::ProcessHandler() { // Update internal data structure from generic one this->SetTestsToRunInformation(this->GetOption("TestsToRunInformation")); this->SetUseUnion(cmSystemTools::IsOn(this->GetOption("UseUnion"))); const char* val; val = this->GetOption("LabelRegularExpression"); if ( val ) { this->UseIncludeLabelRegExpFlag = true; this->IncludeLabelRegExp = val; } val = this->GetOption("ExcludeLabelRegularExpression"); if ( val ) { this->UseExcludeLabelRegExpFlag = true; this->ExcludeLabelRegularExpression = val; } val = this->GetOption("IncludeRegularExpression"); if ( val ) { this->UseIncludeRegExp(); this->SetIncludeRegExp(val); } val = this->GetOption("ExcludeRegularExpression"); if ( val ) { this->UseExcludeRegExp(); this->SetExcludeRegExp(val); } this->TestResults.clear(); // do not output startup if this is a sub-process for parallel tests if(!this->CTest->GetParallelSubprocess()) { cmCTestLog(this->CTest, HANDLER_OUTPUT, (this->MemCheck ? "Memory check" : "Test") << " project " << cmSystemTools::GetCurrentWorkingDirectory() << std::endl); } if ( ! this->PreProcessHandler() ) { return -1; } cmGeneratedFileStream mLogFile; this->StartLogFile((this->MemCheck ? "DynamicAnalysis" : "Test"), mLogFile); this->LogFile = &mLogFile; std::vector passed; std::vector failed; int total; this->ProcessDirectory(passed, failed); total = int(passed.size()) + int(failed.size()); if (total == 0) { if ( !this->CTest->GetShowOnly() ) { cmCTestLog(this->CTest, ERROR_MESSAGE, "No tests were found!!!" << std::endl); } } else { if (this->HandlerVerbose && passed.size() && (this->UseIncludeRegExpFlag || this->UseExcludeRegExpFlag)) { cmCTestLog(this->CTest, HANDLER_VERBOSE_OUTPUT, std::endl << "The following tests passed:" << std::endl); for(std::vector::iterator j = passed.begin(); j != passed.end(); ++j) { cmCTestLog(this->CTest, HANDLER_VERBOSE_OUTPUT, "\t" << *j << std::endl); } } float percent = float(passed.size()) * 100.0f / total; if ( failed.size() > 0 && percent > 99) { percent = 99; } if(!this->CTest->GetParallelSubprocess()) { cmCTestLog(this->CTest, HANDLER_OUTPUT, std::endl << static_cast(percent + .5) << "% tests passed, " << failed.size() << " tests failed out of " << total << std::endl); double totalTestTime = 0; for(cmCTestTestHandler::TestResultsVector::size_type cc = 0; cc < this->TestResults.size(); cc ++ ) { cmCTestTestResult *result = &this->TestResults[cc]; totalTestTime += result->ExecutionTime; } char buf[1024]; sprintf(buf, "%6.2f sec", totalTestTime); cmCTestLog(this->CTest, HANDLER_OUTPUT, "\nTotal Test time = " << buf << "\n" ); } if (failed.size()) { cmGeneratedFileStream ofs; if(!this->CTest->GetParallelSubprocess()) { cmCTestLog(this->CTest, ERROR_MESSAGE, std::endl << "The following tests FAILED:" << std::endl); this->StartLogFile("TestsFailed", ofs); std::vector::iterator ftit; for(ftit = this->TestResults.begin(); ftit != this->TestResults.end(); ++ftit) { if ( ftit->Status != cmCTestTestHandler::COMPLETED ) { ofs << ftit->TestCount << ":" << ftit->Name << std::endl; cmCTestLog(this->CTest, HANDLER_OUTPUT, "\t" << std::setw(3) << ftit->TestCount << " - " << ftit->Name.c_str() << " (" << this->GetTestStatus(ftit->Status) << ")" << std::endl); } } } } } if ( this->CTest->GetProduceXML() ) { cmGeneratedFileStream xmlfile; if( !this->StartResultingXML( (this->MemCheck ? cmCTest::PartMemCheck : cmCTest::PartTest), (this->MemCheck ? "DynamicAnalysis" : "Test"), xmlfile) ) { cmCTestLog(this->CTest, ERROR_MESSAGE, "Cannot create " << (this->MemCheck ? "memory check" : "testing") << " XML file" << std::endl); this->LogFile = 0; return 1; } this->GenerateDartOutput(xmlfile); } if ( ! this->PostProcessHandler() ) { this->LogFile = 0; return -1; } if ( !failed.empty() ) { this->LogFile = 0; return -1; } this->LogFile = 0; return 0; } //---------------------------------------------------------------------- void cmCTestTestHandler::PrintLabelSummary() { cmCTestTestHandler::ListOfTests::iterator it = this->TestList.begin(); cmCTestTestHandler::TestResultsVector::iterator ri = this->TestResults.begin(); std::map labelTimes; std::set labels; // initialize maps for(; it != this->TestList.end(); ++it) { cmCTestTestProperties& p = *it; if(p.Labels.size() != 0) { for(std::vector::iterator l = p.Labels.begin(); l != p.Labels.end(); ++l) { labels.insert(*l); labelTimes[*l] = 0; } } } it = this->TestList.begin(); ri = this->TestResults.begin(); // fill maps for(; it != this->TestList.end(); ++it, ++ri) { cmCTestTestProperties& p = *it; cmCTestTestResult &result = *ri; if(p.Labels.size() != 0) { for(std::vector::iterator l = p.Labels.begin(); l != p.Labels.end(); ++l) { labelTimes[*l] += result.ExecutionTime; } } } // now print times for(std::set::const_iterator i = labels.begin(); i != labels.end(); ++i) { cmCTestLog(this->CTest, HANDLER_OUTPUT, "\nTime in " << *i << " = " << labelTimes[*i] << " sec" ); if ( this->LogFile ) { *this->LogFile << "\nTime in " << *i << " = " << labelTimes[*i] << " sec" << std::endl; } } } //---------------------------------------------------------------------- void cmCTestTestHandler::ProcessOneTest(cmCTestTestProperties *it, std::vector &passed, std::vector &failed, int cnt, int tmsize) { const std::string& testname = it->Name; std::vector& args = it->Args; cmCTestTestResult cres; cres.Properties = &*it; cres.ExecutionTime = 0; cres.ReturnValue = -1; cres.Status = cmCTestTestHandler::NOT_RUN; cres.TestCount = cnt; cres.Name = testname; cres.Path = it->Directory.c_str(); cmCTestLog(this->CTest, HANDLER_OUTPUT, std::setw(3) << cnt << "/"); cmCTestLog(this->CTest, HANDLER_OUTPUT, std::setw(3) << tmsize << " "); if ( this->MemCheck ) { cmCTestLog(this->CTest, HANDLER_OUTPUT, "Memory Check"); } else { cmCTestLog(this->CTest, HANDLER_OUTPUT, "Testing"); } cmCTestLog(this->CTest, HANDLER_OUTPUT, " "); const int maxTestNameWidth = this->CTest->GetMaxTestNameWidth(); std::string outname = testname + " "; outname.resize(maxTestNameWidth, '.'); *this->LogFile << cnt << "/" << tmsize << " Testing: " << testname << std::endl; if ( this->CTest->GetShowOnly() ) { cmCTestLog(this->CTest, HANDLER_OUTPUT, outname.c_str() << std::endl); } else { cmCTestLog(this->CTest, HANDLER_OUTPUT, outname.c_str()); } cmCTestLog(this->CTest, DEBUG, "Testing " << args[0].c_str() << " ... "); // find the test executable std::string actualCommand = this->FindTheExecutable(args[1].c_str()); std::string testCommand = cmSystemTools::ConvertToOutputPath(actualCommand.c_str()); // continue if we did not find the executable if (testCommand == "") { *this->LogFile << "Unable to find executable: " << args[1].c_str() << std::endl; cmCTestLog(this->CTest, ERROR_MESSAGE, "Unable to find executable: " << args[1].c_str() << std::endl); cres.Output = "Unable to find executable: " + args[1]; if ( !this->CTest->GetShowOnly() ) { cres.FullCommandLine = actualCommand; this->TestResults.push_back( cres ); failed.push_back(testname); return; } } // add the arguments std::vector::const_iterator j = args.begin(); ++j; // skip test name ++j; // skip command as it is in actualCommand std::vector arguments; this->GenerateTestCommand(arguments); arguments.push_back(actualCommand.c_str()); for(;j != args.end(); ++j) { testCommand += " "; testCommand += cmSystemTools::EscapeSpaces(j->c_str()); arguments.push_back(j->c_str()); } arguments.push_back(0); /** * Run an executable command and put the stdout in output. */ std::string output; int retVal = 0; cmCTestLog(this->CTest, HANDLER_VERBOSE_OUTPUT, std::endl << (this->MemCheck?"MemCheck":"Test") << " command: " << testCommand << std::endl); *this->LogFile << cnt << "/" << tmsize << " Test: " << testname.c_str() << std::endl; *this->LogFile << "Command: "; std::vector::size_type ll; for ( ll = 0; ll < arguments.size()-1; ll ++ ) { *this->LogFile << "\"" << arguments[ll] << "\" "; } *this->LogFile << std::endl << "Directory: " << it->Directory << std::endl << "\"" << testname.c_str() << "\" start time: " << this->CTest->CurrentTime() << std::endl << "Output:" << std::endl << "----------------------------------------------------------" << std::endl; int res = 0; double clock_start, clock_finish; clock_start = cmSystemTools::GetTime(); if ( !this->CTest->GetShowOnly() ) { res = this->CTest->RunTest(arguments, &output, &retVal, this->LogFile, it->Timeout, &it->Environment); } clock_finish = cmSystemTools::GetTime(); cres.ExecutionTime = (double)(clock_finish - clock_start); cres.FullCommandLine = testCommand; std::string reason; if ( !this->CTest->GetShowOnly() ) { std::vector >::iterator passIt; bool forceFail = false; if ( it->RequiredRegularExpressions.size() > 0 ) { bool found = false; for ( passIt = it->RequiredRegularExpressions.begin(); passIt != it->RequiredRegularExpressions.end(); ++ passIt ) { if ( passIt->first.find(output.c_str()) ) { found = true; reason = "Required regular expression found."; } } if ( !found ) { reason = "Required regular expression not found."; forceFail = true; } reason += "Regex=["; for ( passIt = it->RequiredRegularExpressions.begin(); passIt != it->RequiredRegularExpressions.end(); ++ passIt ) { reason += passIt->second; reason += "\n"; } reason += "]"; } if ( it->ErrorRegularExpressions.size() > 0 ) { for ( passIt = it->ErrorRegularExpressions.begin(); passIt != it->ErrorRegularExpressions.end(); ++ passIt ) { if ( passIt->first.find(output.c_str()) ) { reason = "Error regular expression found in output."; reason += " Regex=["; reason += passIt->second; reason += "]"; forceFail = true; } } } if (res == cmsysProcess_State_Exited) { bool success = !forceFail && (retVal == 0 || it->RequiredRegularExpressions.size()); if((success && !it->WillFail) || (!success && it->WillFail)) { cres.Status = cmCTestTestHandler::COMPLETED; cmCTestLog(this->CTest, HANDLER_OUTPUT, " Passed " ); } else { cres.Status = cmCTestTestHandler::FAILED; cmCTestLog(this->CTest, HANDLER_OUTPUT, "***Failed " << reason ); } } else if ( res == cmsysProcess_State_Expired ) { cmCTestLog(this->CTest, HANDLER_OUTPUT, "***Timeout"); cres.Status = cmCTestTestHandler::TIMEOUT; } else if ( res == cmsysProcess_State_Exception ) { cmCTestLog(this->CTest, HANDLER_OUTPUT, "***Exception: "); switch ( retVal ) { case cmsysProcess_Exception_Fault: cmCTestLog(this->CTest, HANDLER_OUTPUT, "SegFault"); cres.Status = cmCTestTestHandler::SEGFAULT; break; case cmsysProcess_Exception_Illegal: cmCTestLog(this->CTest, HANDLER_OUTPUT, "Illegal"); cres.Status = cmCTestTestHandler::ILLEGAL; break; case cmsysProcess_Exception_Interrupt: cmCTestLog(this->CTest, HANDLER_OUTPUT, "Interrupt"); cres.Status = cmCTestTestHandler::INTERRUPT; break; case cmsysProcess_Exception_Numerical: cmCTestLog(this->CTest, HANDLER_OUTPUT, "Numerical"); cres.Status = cmCTestTestHandler::NUMERICAL; break; default: cmCTestLog(this->CTest, HANDLER_OUTPUT, "Other"); cres.Status = cmCTestTestHandler::OTHER_FAULT; } } else // if ( res == cmsysProcess_State_Error ) { cmCTestLog(this->CTest, HANDLER_OUTPUT, "***Bad command " << res ); cres.Status = cmCTestTestHandler::BAD_COMMAND; } if(cres.Status == cmCTestTestHandler::COMPLETED) { passed.push_back(testname); } else { failed.push_back(testname); } char buf[1024]; sprintf(buf, "%6.2f sec", cres.ExecutionTime); cmCTestLog(this->CTest, HANDLER_OUTPUT, buf << "\n" ); if (!output.empty() && output.find("DartStuff.find(output.c_str())) { std::string dartString = this->DartStuff.match(1); // keep searching and replacing until none are left while (this->DartStuff1.find(output.c_str())) { // replace the exact match for the string cmSystemTools::ReplaceString(output, this->DartStuff1.match(1).c_str(), ""); } cres.RegressionImages = this->GenerateRegressionImages(dartString); } } } // if this is doing MemCheck then all the output needs to be put into // Output since that it what is parsed to by cmCTestMemCheckHandler if(!this->MemCheck) { if ( cres.Status == cmCTestTestHandler::COMPLETED ) { this->CleanTestOutput(output, static_cast (this->CustomMaximumPassedTestOutputSize)); } else { this->CleanTestOutput(output, static_cast (this->CustomMaximumFailedTestOutputSize)); } } cres.Reason = reason; if ( this->LogFile ) { bool pass = true; const char* reasonType = "Test Pass Reason"; if(cres.Status != cmCTestTestHandler::COMPLETED && cres.Status != cmCTestTestHandler::NOT_RUN) { reasonType = "Test Fail Reason"; pass = false; } double ttime = clock_finish - clock_start; int hours = static_cast(ttime / (60 * 60)); int minutes = static_cast(ttime / 60) % 60; int seconds = static_cast(ttime) % 60; char buffer[100]; sprintf(buffer, "%02d:%02d:%02d", hours, minutes, seconds); *this->LogFile << "----------------------------------------------------------" << std::endl; if(cres.Reason.size()) { *this->LogFile << reasonType << ":\n" << cres.Reason << "\n"; } else { if(pass) { *this->LogFile << "Test Passed.\n"; } else { *this->LogFile << "Test Failed.\n"; } } *this->LogFile << "\"" << testname.c_str() << "\" end time: " << this->CTest->CurrentTime() << std::endl << "\"" << testname.c_str() << "\" time elapsed: " << buffer << std::endl << "----------------------------------------------------------" << std::endl << std::endl; } cres.Output = output; cres.ReturnValue = retVal; cres.CompletionStatus = "Completed"; this->TestResults.push_back( cres ); } //---------------------------------------------------------------------- void cmCTestTestHandler::CheckLabelFilterInclude(cmCTestTestProperties& it) { // if not using Labels to filter then return if (!this->UseIncludeLabelRegExpFlag ) { return; } // if there are no labels and we are filtering by labels // then exclude the test as it does not have the label if(it.Labels.size() == 0 ) { it.IsInBasedOnREOptions = false; return; } // check to see if the label regular expression matches bool found = false; // assume it does not match // loop over all labels and look for match for(std::vector::iterator l = it.Labels.begin(); l != it.Labels.end(); ++l) { if(this->IncludeLabelRegularExpression.find(*l)) { found = true; } } // if no match was found, exclude the test if(!found) { it.IsInBasedOnREOptions = false; } } //---------------------------------------------------------------------- void cmCTestTestHandler::CheckLabelFilterExclude(cmCTestTestProperties& it) { // if not using Labels to filter then return if (!this->UseExcludeLabelRegExpFlag ) { return; } // if there are no labels and we are excluding by labels // then do nothing as a no label can not be a match if(it.Labels.size() == 0 ) { return; } // check to see if the label regular expression matches bool found = false; // assume it does not match // loop over all labels and look for match for(std::vector::iterator l = it.Labels.begin(); l != it.Labels.end(); ++l) { if(this->ExcludeLabelRegularExpression.find(*l)) { found = true; } } // if match was found, exclude the test if(found) { it.IsInBasedOnREOptions = false; } } //---------------------------------------------------------------------- void cmCTestTestHandler::CheckLabelFilter(cmCTestTestProperties& it) { this->CheckLabelFilterInclude(it); this->CheckLabelFilterExclude(it); } //---------------------------------------------------------------------- void cmCTestTestHandler::ComputeTestList() { this->TestList.clear(); // clear list of test if(this->CTest->GetParallelSubprocess()) { this->LoadTestList(); return; } else { this->GetListOfTests(); } cmCTestTestHandler::ListOfTests::size_type tmsize = this->TestList.size(); // how many tests are in based on RegExp? int inREcnt = 0; cmCTestTestHandler::ListOfTests::iterator it; for ( it = this->TestList.begin(); it != this->TestList.end(); it ++ ) { this->CheckLabelFilter(*it); if (it->IsInBasedOnREOptions) { inREcnt ++; } } // expand the test list based on the union flag if (this->UseUnion) { this->ExpandTestsToRunInformation((int)tmsize); } else { this->ExpandTestsToRunInformation(inREcnt); } // Now create a final list of tests to run int cnt = 0; inREcnt = 0; std::string last_directory = ""; ListOfTests finalList; for ( it = this->TestList.begin(); it != this->TestList.end(); it ++ ) { cnt ++; if (it->IsInBasedOnREOptions) { inREcnt++; } if (this->UseUnion) { // if it is not in the list and not in the regexp then skip if ((this->TestsToRun.size() && std::find(this->TestsToRun.begin(), this->TestsToRun.end(), cnt) == this->TestsToRun.end()) && !it->IsInBasedOnREOptions) { continue; } } else { // is this test in the list of tests to run? If not then skip it if ((this->TestsToRun.size() && std::find(this->TestsToRun.begin(), this->TestsToRun.end(), inREcnt) == this->TestsToRun.end()) || !it->IsInBasedOnREOptions) { continue; } } it->Index = cnt; // save the index into the test list for this test finalList.push_back(*it); } // Save the total number of tests before exclusions this->TotalNumberOfTests = this->TestList.size(); // Set the TestList to the final list of all test this->TestList = finalList; std::string::size_type max = this->CTest->GetMaxTestNameWidth(); for (it = this->TestList.begin(); it != this->TestList.end(); it ++ ) { cmCTestTestProperties& p = *it; if(max < p.Name.size()) { max = p.Name.size(); } } if(static_cast(this->CTest->GetMaxTestNameWidth()) != max) { this->CTest->SetMaxTestNameWidth(static_cast(max)); } } bool cmCTestTestHandler::GetValue(const char* tag, int& value, std::ifstream& fin) { std::string line; bool ret = true; cmSystemTools::GetLineFromStream(fin, line); if(line == tag) { fin >> value; ret = cmSystemTools::GetLineFromStream(fin, line); // read blank line } else { cmCTestLog(this->CTest, ERROR_MESSAGE, "parse error: missing tag: " << tag << " found [" << line << "]" << std::endl); ret = false; } return ret; } bool cmCTestTestHandler::GetValue(const char* tag, double& value, std::ifstream& fin) { std::string line; cmSystemTools::GetLineFromStream(fin, line); bool ret = true; if(line == tag) { fin >> value; ret = cmSystemTools::GetLineFromStream(fin, line); // read blank line } else { cmCTestLog(this->CTest, ERROR_MESSAGE, "parse error: missing tag: " << tag << " found [" << line << "]" << std::endl); ret = false; } return ret; } bool cmCTestTestHandler::GetValue(const char* tag, bool& value, std::ifstream& fin) { std::string line; cmSystemTools::GetLineFromStream(fin, line); bool ret = true; if(line == tag) { #ifdef __HAIKU__ int tmp = 0; fin >> tmp; value = false; if(tmp) { value = true; } #else fin >> value; #endif ret = cmSystemTools::GetLineFromStream(fin, line); // read blank line } else { cmCTestLog(this->CTest, ERROR_MESSAGE, "parse error: missing tag: " << tag << " found [" << line << "]" << std::endl); ret = false; } return ret; } bool cmCTestTestHandler::GetValue(const char* tag, size_t& value, std::ifstream& fin) { std::string line; cmSystemTools::GetLineFromStream(fin, line); bool ret = true; if(line == tag) { fin >> value; ret = cmSystemTools::GetLineFromStream(fin, line); // read blank line } else { cmCTestLog(this->CTest, ERROR_MESSAGE, "parse error: missing tag: " << tag << " found [" << line.c_str() << "]" << std::endl); ret = false; } return ret; } bool cmCTestTestHandler::GetValue(const char* tag, std::string& value, std::ifstream& fin) { std::string line; cmSystemTools::GetLineFromStream(fin, line); bool ret = true; if(line == tag) { ret = cmSystemTools::GetLineFromStream(fin, value); } else { cmCTestLog(this->CTest, ERROR_MESSAGE, "parse error: missing tag: " << tag << " found [" << line << "]" << std::endl); ret = false; } return ret; } // This should load only one test and is used in -j N mode. // it is used by the sub-process ctest runs which should have // only one -I N test to run. void cmCTestTestHandler::LoadTestList() { this->TestList.clear(); std::string fname = this->CTest->GetBinaryDir() + "/Testing/Temporary/PCache.txt"; std::ifstream fin(fname.c_str()); std::string line; if(!fin) { cmCTestLog(this->CTest, ERROR_MESSAGE, "Could not load PCache.txt file: " << fname.c_str() << std::endl); return; } bool ok = true; int numTestsToRun = 0; ok = ok && this->GetValue("TotalNumberOfTests:", this->TotalNumberOfTests, fin); ok = ok && this->GetValue("NumberOfTestsToRun:", numTestsToRun, fin); this->ExpandTestsToRunInformation(this->TotalNumberOfTests); if(this->TestsToRun.size() != 1) { cmCTestLog(this->CTest, ERROR_MESSAGE, "Error when in parallel mode only one test should be run: " << this->TestsToRun.size() << std::endl); } int testIndexToRun = this->TestsToRun[0]; this->CTest->SetParallelSubprocessId(testIndexToRun); if(!ok) { return; } for(int i =0; i < numTestsToRun; i++) { cmCTestTestProperties p; int numArgs = 0; ok = this->GetValue("Name:", p.Name, fin); ok = ok && this->GetValue("Directory:", p.Directory, fin); ok = ok && this->GetValue("Args:", numArgs, fin); for(int j =0; j < numArgs; ++j) { cmSystemTools::GetLineFromStream(fin, line); p.Args.push_back(line); } int numDep = 0; ok = ok && this->GetValue("Depends:", numDep, fin); for(int j =0; j < numDep; ++j) { cmSystemTools::GetLineFromStream(fin, line); p.Depends.push_back(line); } int numErrRegex = 0; ok = ok && this->GetValue("ErrorRegularExpressions:", numErrRegex, fin); for(int j =0; j < numErrRegex; j++) { cmSystemTools::GetLineFromStream(fin, line); std::pair rpair; rpair.first.compile(line.c_str()); rpair.second = line; p.ErrorRegularExpressions.push_back(rpair); } int numReqRegex = 0; ok = ok && this->GetValue("RequiredRegularExpressions:", numReqRegex, fin); for(int j =0; j < numReqRegex; j++) { cmSystemTools::GetLineFromStream(fin, line); std::pair rpair; rpair.first.compile(line.c_str()); rpair.second = line; p.RequiredRegularExpressions.push_back(rpair); } int numMeasure = 0; ok = ok && this->GetValue("Measurements:", numMeasure, fin); for(int j =0; j < numMeasure; j++) { cmStdString m; cmStdString v; cmSystemTools::GetLineFromStream(fin, line); m = line; cmSystemTools::GetLineFromStream(fin, line); v = line; p.Measurements[m] = v; } int isinre; ok = ok && this->GetValue("IsInBasedOnREOptions:", isinre, fin); ok = ok && this->GetValue("WillFail:", p.WillFail, fin); ok = ok && this->GetValue("TimeOut:", p.Timeout, fin); ok = ok && this->GetValue("Index:", p.Index, fin); int numEnv = 0; ok = ok && this->GetValue("Environment:", numEnv, fin); for(int j =0; j < numEnv; j++) { cmSystemTools::GetLineFromStream(fin, line); p.Environment.push_back(line); } int numLabels = 0; ok = ok && this->GetValue("Labels:", numLabels, fin); for(int j =0; j < numLabels; j++) { cmSystemTools::GetLineFromStream(fin, line); p.Labels.push_back(line); } if(!ok) { cmCTestLog(this->CTest, ERROR_MESSAGE, "Internal Error reading cached test information." << std::endl); return; } if(p.Index == testIndexToRun) { // add the one test and stop reading this->TestList.push_back(p); return; } } } std::string cmCTestTestHandler::SaveTestList() { std::string fname = this->CTest->GetBinaryDir() + "/Testing/Temporary/PCache.txt"; cmGeneratedFileStream fout(fname.c_str()); if(!fout) { cmCTestLog(this->CTest, ERROR_MESSAGE, std::endl << "Could not open PCache.txt for write:" << fname.c_str() << std::endl); } fout << "TotalNumberOfTests:\n"; fout << this->TotalNumberOfTests << "\n"; fout << "NumberOfTestsToRun:\n"; fout << this->TestList.size() << "\n"; for (ListOfTests::iterator it = this->TestList.begin(); it != this->TestList.end(); it ++ ) { cmCTestTestProperties& p = *it; fout << "Name:\n" << p.Name.c_str() << "\nDirectory:\n" << p.Directory.c_str() << "\nArgs:\n" << p.Args.size() << "\n"; for(std::vector::iterator i = p.Args.begin(); i != p.Args.end(); ++i) { fout << i->c_str() << "\n"; } fout << "Depends:\n" << p.Depends.size() << "\n"; for(std::vector::iterator i = p.Depends.begin(); i != p.Depends.end(); ++i) { fout << i->c_str() << "\n"; } std::vector >::iterator regxi; fout << "ErrorRegularExpressions:\n" << p.ErrorRegularExpressions.size() << "\n"; for(regxi = p.ErrorRegularExpressions.begin(); regxi != p.ErrorRegularExpressions.end(); regxi++) { fout << regxi->second << "\n"; } fout << "RequiredRegularExpressions:\n" << p.RequiredRegularExpressions.size() << "\n"; for(regxi = p.RequiredRegularExpressions.begin(); regxi != p.RequiredRegularExpressions.end(); regxi++) { fout << regxi->second << "\n"; } fout << "Measurements:\n" << p.Measurements.size() << "\n"; for(std::map::const_iterator m = p.Measurements.begin(); m != p.Measurements.end(); ++m) { fout << m->first << "\n"; fout << m->second << "\n"; } fout << "IsInBasedOnREOptions:\n" << p.IsInBasedOnREOptions << "\nWillFail:\n" << p.WillFail << "\nTimeOut:\n" << p.Timeout << "\nIndex:\n" << p.Index << "\n"; fout << "Environment:\n" << p.Environment.size() << "\n"; for(std::vector::const_iterator e = p.Environment.begin(); e != p.Environment.end(); ++e) { fout << *e << "\n"; } fout << "Labels:\n" << p.Labels.size() << "\n"; for(std::vector::const_iterator e = p.Labels.begin(); e != p.Labels.end(); ++e) { fout << *e << "\n"; } } fout.close(); return fname; } void cmCTestTestHandler::ProcessParallel(std::vector &passed, std::vector &failed) { this->ComputeTestList(); cmCTestMultiProcessHandler parallel; parallel.SetCTest(this->CTest); parallel.SetParallelLevel(this->CTest->GetParallelLevel()); cmCTestMultiProcessHandler::TestSet depends; cmCTestMultiProcessHandler::TestMap tests; std::map testnames; for (ListOfTests::iterator it = this->TestList.begin(); it != this->TestList.end(); it ++ ) { cmCTestTestProperties& p = *it; testnames[p.Index] = p.Name; if(p.Depends.size()) { for(std::vector::iterator i = p.Depends.begin(); i != p.Depends.end(); ++i) { for(ListOfTests::iterator it2 = this->TestList.begin(); it2 != this->TestList.end(); it2 ++ ) { if(it2->Name == *i) { depends.insert(it2->Index); break; // break out of test loop as name can only match 1 } } } } tests[it->Index] = depends; } parallel.SetCTestCommand(this->CTest->GetCTestExecutable()); parallel.SetTests(tests, testnames); std::string fname = this->SaveTestList(); parallel.SetTestCacheFile(fname.c_str()); parallel.SetPassFailVectors(&passed, &failed); this->TestResults.clear(); parallel.SetTestResults(&this->TestResults); parallel.RunTests(); cmSystemTools::RemoveFile(fname.c_str()); } //---------------------------------------------------------------------- void cmCTestTestHandler::ProcessDirectory(std::vector &passed, std::vector &failed) { if(this->CTest->GetParallelLevel() > 0) { this->ProcessParallel(passed, failed); return; } // save the current working directory std::string current_dir = cmSystemTools::GetCurrentWorkingDirectory(); // compute the list of tests to run this->ComputeTestList(); this->StartTest = this->CTest->CurrentTime(); this->StartTestTime = static_cast(cmSystemTools::GetTime()); double elapsed_time_start = cmSystemTools::GetTime(); *this->LogFile << "Start testing: " << this->StartTest << std::endl << "----------------------------------------------------------" << std::endl; std::string last_directory = ""; // run each test for (ListOfTests::iterator it = this->TestList.begin(); it != this->TestList.end(); it ++ ) { if (!(last_directory == it->Directory)) { cmCTestLog(this->CTest, HANDLER_VERBOSE_OUTPUT, "Changing directory into " << it->Directory.c_str() << "\n"); *this->LogFile << "Changing directory into: " << it->Directory.c_str() << std::endl; last_directory = it->Directory; cmSystemTools::ChangeDirectory(it->Directory.c_str()); } // process this one test cmCTestRunTest testRun; testRun.SetTestProperties(&(*it)); testRun.SetTestHandler(this); bool testPassed = testRun.Execute(); //run the test if(testPassed && !this->CTest->GetShowOnly()) { passed.push_back(it->Name); } else if(!testPassed) { failed.push_back(it->Name); } } this->EndTest = this->CTest->CurrentTime(); this->EndTestTime = static_cast(cmSystemTools::GetTime()); this->ElapsedTestingTime = cmSystemTools::GetTime() - elapsed_time_start; if ( this->LogFile ) { *this->LogFile << "End testing: " << this->EndTest << std::endl; } cmSystemTools::ChangeDirectory(current_dir.c_str()); } //---------------------------------------------------------------------- void cmCTestTestHandler::GenerateTestCommand(std::vector&) { } //---------------------------------------------------------------------- void cmCTestTestHandler::GenerateDartOutput(std::ostream& os) { if ( !this->CTest->GetProduceXML() ) { return; } this->CTest->StartXML(os, this->AppendXML); os << "\n" << "\t" << this->StartTest << "\n" << "\t" << this->StartTestTime << "\n" << "\t\n"; cmCTestTestHandler::TestResultsVector::size_type cc; for ( cc = 0; cc < this->TestResults.size(); cc ++ ) { cmCTestTestResult *result = &this->TestResults[cc]; std::string testPath = result->Path + "/" + result->Name; os << "\t\t" << cmXMLSafe( this->CTest->GetShortPathToFile(testPath.c_str())) << "" << std::endl; } os << "\t\n"; for ( cc = 0; cc < this->TestResults.size(); cc ++ ) { cmCTestTestResult *result = &this->TestResults[cc]; this->WriteTestResultHeader(os, result); os << "\t\t" << std::endl; if ( result->Status != cmCTestTestHandler::NOT_RUN ) { if ( result->Status != cmCTestTestHandler::COMPLETED || result->ReturnValue ) { os << "\t\t\t" << cmXMLSafe(this->GetTestStatus(result->Status)) << "" "\n" << "\t\t\t" << result->ReturnValue << "" << std::endl; } os << result->RegressionImages; os << "\t\t\t" << result->ExecutionTime << "\n"; if(result->Reason.size()) { const char* reasonType = "Pass Reason"; if(result->Status != cmCTestTestHandler::COMPLETED && result->Status != cmCTestTestHandler::NOT_RUN) { reasonType = "Fail Reason"; } os << "\t\t\t" << cmXMLSafe(result->Reason) << "\n"; } os << "\t\t\t" << cmXMLSafe(result->CompletionStatus) << "\n"; } os << "\t\t\t" << cmXMLSafe(result->FullCommandLine) << "\n"; std::map::iterator measureIt; for ( measureIt = result->Properties->Measurements.begin(); measureIt != result->Properties->Measurements.end(); ++ measureIt ) { os << "\t\t\tfirst.c_str() << "\">" << cmXMLSafe(measureIt->second) << "\n"; } os << "\t\t\t\n" << "\t\t\t\t"; os << cmXMLSafe(result->Output); os << "\n" << "\t\t\t\n" << "\t\t\n"; this->WriteTestResultFooter(os, result); } os << "\t" << this->EndTest << "\n" << "\t" << this->EndTestTime << "\n" << "" << static_cast(this->ElapsedTestingTime/6)/10.0 << "" << "" << std::endl; this->CTest->EndXML(os); } //---------------------------------------------------------------------------- void cmCTestTestHandler::WriteTestResultHeader(std::ostream& os, cmCTestTestResult* result) { os << "\tStatus == cmCTestTestHandler::COMPLETED ) { os << "passed"; } else if ( result->Status == cmCTestTestHandler::NOT_RUN ) { os << "notrun"; } else { os << "failed"; } std::string testPath = result->Path + "/" + result->Name; os << "\">\n" << "\t\t" << cmXMLSafe(result->Name) << "\n" << "\t\t" << cmXMLSafe( this->CTest->GetShortPathToFile(result->Path.c_str())) << "\n" << "\t\t" << cmXMLSafe( this->CTest->GetShortPathToFile(testPath.c_str())) << "\n" << "\t\t" << cmXMLSafe(result->FullCommandLine) << "\n"; } //---------------------------------------------------------------------------- void cmCTestTestHandler::WriteTestResultFooter(std::ostream& os, cmCTestTestResult* result) { if(!result->Properties->Labels.empty()) { os << "\t\t\n"; std::vector const& labels = result->Properties->Labels; for(std::vector::const_iterator li = labels.begin(); li != labels.end(); ++li) { os << "\t\t\t\n"; } os << "\t\t\n"; } os << "\t" << std::endl; } //---------------------------------------------------------------------- int cmCTestTestHandler::ExecuteCommands(std::vector& vec) { std::vector::iterator it; for ( it = vec.begin(); it != vec.end(); ++it ) { int retVal = 0; cmCTestLog(this->CTest, HANDLER_VERBOSE_OUTPUT, "Run command: " << *it << std::endl); if ( !cmSystemTools::RunSingleCommand(it->c_str(), 0, &retVal, 0, true /*this->Verbose*/) || retVal != 0 ) { cmCTestLog(this->CTest, ERROR_MESSAGE, "Problem running command: " << *it << std::endl); return 0; } } return 1; } //---------------------------------------------------------------------- // Find the appropriate executable to run for a test std::string cmCTestTestHandler::FindTheExecutable(const char *exe) { std::string resConfig; std::vector extraPaths; std::vector failedPaths; return cmCTestTestHandler::FindExecutable(this->CTest, exe, resConfig, extraPaths, failedPaths); } // add additional configurations to the search path void cmCTestTestHandler ::AddConfigurations(cmCTest *ctest, std::vector &attempted, std::vector &attemptedConfigs, std::string filepath, std::string &filename) { std::string tempPath; if (filepath.size() && filepath[filepath.size()-1] != '/') { filepath += "/"; } tempPath = filepath + filename; attempted.push_back(tempPath); attemptedConfigs.push_back(""); if(ctest->GetConfigType().size()) { tempPath = filepath; tempPath += ctest->GetConfigType(); tempPath += "/"; tempPath += filename; attempted.push_back(tempPath); attemptedConfigs.push_back(ctest->GetConfigType()); // If the file is an OSX bundle then the configtyp // will be at the start of the path tempPath = ctest->GetConfigType(); tempPath += "/"; tempPath += filepath; tempPath += filename; attempted.push_back(tempPath); attemptedConfigs.push_back(ctest->GetConfigType()); } else { // no config specified to try some options tempPath = filepath; tempPath += "Release/"; tempPath += filename; attempted.push_back(tempPath); attemptedConfigs.push_back("Release"); tempPath = filepath; tempPath += "Debug/"; tempPath += filename; attempted.push_back(tempPath); attemptedConfigs.push_back("Debug"); tempPath = filepath; tempPath += "MinSizeRel/"; tempPath += filename; attempted.push_back(tempPath); attemptedConfigs.push_back("MinSizeRel"); tempPath = filepath; tempPath += "RelWithDebInfo/"; tempPath += filename; attempted.push_back(tempPath); attemptedConfigs.push_back("RelWithDebInfo"); tempPath = filepath; tempPath += "Deployment/"; tempPath += filename; attempted.push_back(tempPath); attemptedConfigs.push_back("Deployment"); tempPath = filepath; tempPath += "Development/"; tempPath += filename; attempted.push_back(tempPath); attemptedConfigs.push_back("Deployment"); } } //---------------------------------------------------------------------- // Find the appropriate executable to run for a test std::string cmCTestTestHandler ::FindExecutable(cmCTest *ctest, const char *testCommand, std::string &resultingConfig, std::vector &extraPaths, std::vector &failed) { // now run the compiled test if we can find it std::vector attempted; std::vector attemptedConfigs; std::string tempPath; std::string filepath = cmSystemTools::GetFilenamePath(testCommand); std::string filename = cmSystemTools::GetFilenameName(testCommand); cmCTestTestHandler::AddConfigurations(ctest, attempted, attemptedConfigs, filepath,filename); // even if a fullpath was specified also try it relative to the current // directory if (filepath.size() && filepath[0] == '/') { std::string localfilepath = filepath.substr(1,filepath.size()-1); cmCTestTestHandler::AddConfigurations(ctest, attempted, attemptedConfigs, localfilepath,filename); } // if extraPaths are provided and we were not passed a full path, try them, // try any extra paths if (filepath.size() == 0) { for (unsigned int i = 0; i < extraPaths.size(); ++i) { std::string filepathExtra = cmSystemTools::GetFilenamePath(extraPaths[i]); std::string filenameExtra = cmSystemTools::GetFilenameName(extraPaths[i]); cmCTestTestHandler::AddConfigurations(ctest,attempted, attemptedConfigs, filepathExtra, filenameExtra); } } // store the final location in fullPath std::string fullPath; // now look in the paths we specified above for(unsigned int ai=0; ai < attempted.size() && fullPath.size() == 0; ++ai) { // first check without exe extension if(cmSystemTools::FileExists(attempted[ai].c_str()) && !cmSystemTools::FileIsDirectory(attempted[ai].c_str())) { fullPath = cmSystemTools::CollapseFullPath(attempted[ai].c_str()); resultingConfig = attemptedConfigs[ai]; } // then try with the exe extension else { failed.push_back(attempted[ai].c_str()); tempPath = attempted[ai]; tempPath += cmSystemTools::GetExecutableExtension(); if(cmSystemTools::FileExists(tempPath.c_str()) && !cmSystemTools::FileIsDirectory(tempPath.c_str())) { fullPath = cmSystemTools::CollapseFullPath(tempPath.c_str()); resultingConfig = attemptedConfigs[ai]; } else { failed.push_back(tempPath.c_str()); } } } // if everything else failed, check the users path, but only if a full path // wasn't specified if (fullPath.size() == 0 && filepath.size() == 0) { std::string path = cmSystemTools::FindProgram(filename.c_str()); if (path != "") { resultingConfig = ""; return path; } } if(fullPath.size() == 0) { cmCTestLog(ctest, HANDLER_OUTPUT, "Could not find executable " << testCommand << "\n" << "Looked in the following places:\n"); for(std::vector::iterator i = failed.begin(); i != failed.end(); ++i) { cmCTestLog(ctest, HANDLER_OUTPUT, i->c_str() << "\n"); } } return fullPath; } //---------------------------------------------------------------------- void cmCTestTestHandler::GetListOfTests() { if ( !this->IncludeLabelRegExp.empty() ) { this->IncludeLabelRegularExpression. compile(this->IncludeLabelRegExp.c_str()); } if ( !this->IncludeLabelRegExp.empty() ) { this->ExcludeLabelRegularExpression. compile(this->ExcludeLabelRegExp.c_str()); } if ( !this->IncludeRegExp.empty() ) { this->IncludeTestsRegularExpression.compile(this->IncludeRegExp.c_str()); } if ( !this->ExcludeRegExp.empty() ) { this->ExcludeTestsRegularExpression.compile(this->ExcludeRegExp.c_str()); } cmCTestLog(this->CTest, HANDLER_VERBOSE_OUTPUT, "Constructing a list of tests" << std::endl); cmake cm; cmGlobalGenerator gg; gg.SetCMakeInstance(&cm); std::auto_ptr lg(gg.CreateLocalGenerator()); cmMakefile *mf = lg->GetMakefile(); mf->AddDefinition("CTEST_CONFIGURATION_TYPE", this->CTest->GetConfigType().c_str()); // Add handler for ADD_TEST cmCTestAddTestCommand* newCom1 = new cmCTestAddTestCommand; newCom1->TestHandler = this; cm.AddCommand(newCom1); // Add handler for SUBDIRS cmCTestSubdirCommand* newCom2 = new cmCTestSubdirCommand; newCom2->TestHandler = this; cm.AddCommand(newCom2); // Add handler for ADD_SUBDIRECTORY cmCTestAddSubdirectoryCommand* newCom3 = new cmCTestAddSubdirectoryCommand; newCom3->TestHandler = this; cm.AddCommand(newCom3); // Add handler for SET_SOURCE_FILES_PROPERTIES cmCTestSetTestsPropertiesCommand* newCom4 = new cmCTestSetTestsPropertiesCommand; newCom4->TestHandler = this; cm.AddCommand(newCom4); const char* testFilename; if( cmSystemTools::FileExists("CTestTestfile.cmake") ) { // does the CTestTestfile.cmake exist ? testFilename = "CTestTestfile.cmake"; } else if( cmSystemTools::FileExists("DartTestfile.txt") ) { // does the DartTestfile.txt exist ? testFilename = "DartTestfile.txt"; } else { return; } if ( !mf->ReadListFile(0, testFilename) ) { return; } if ( cmSystemTools::GetErrorOccuredFlag() ) { return; } cmCTestLog(this->CTest, HANDLER_VERBOSE_OUTPUT, "Done constructing a list of tests" << std::endl); } //---------------------------------------------------------------------- void cmCTestTestHandler::UseIncludeRegExp() { this->UseIncludeRegExpFlag = true; } //---------------------------------------------------------------------- void cmCTestTestHandler::UseExcludeRegExp() { this->UseExcludeRegExpFlag = true; this->UseExcludeRegExpFirst = this->UseIncludeRegExpFlag ? false : true; } //---------------------------------------------------------------------- const char* cmCTestTestHandler::GetTestStatus(int status) { static const char statuses[][100] = { "Not Run", "Timeout", "SEGFAULT", "ILLEGAL", "INTERRUPT", "NUMERICAL", "OTHER_FAULT", "Failed", "BAD_COMMAND", "Completed" }; if ( status < cmCTestTestHandler::NOT_RUN || status > cmCTestTestHandler::COMPLETED ) { return "No Status"; } return statuses[status]; } //---------------------------------------------------------------------- void cmCTestTestHandler::ExpandTestsToRunInformation(size_t numTests) { if (this->TestsToRunString.empty()) { return; } int start; int end = -1; double stride = -1; std::string::size_type pos = 0; std::string::size_type pos2; // read start if(GetNextNumber(this->TestsToRunString, start, pos, pos2)) { // read end if(GetNextNumber(this->TestsToRunString, end, pos, pos2)) { // read stride if(GetNextRealNumber(this->TestsToRunString, stride, pos, pos2)) { int val =0; // now read specific numbers while(GetNextNumber(this->TestsToRunString, val, pos, pos2)) { this->TestsToRun.push_back(val); } this->TestsToRun.push_back(val); } } } // if start is not specified then we assume we start at 1 if(start == -1) { start = 1; } // if end isnot specified then we assume we end with the last test if(end == -1) { end = static_cast(numTests); } // if the stride wasn't specified then it defaults to 1 if(stride == -1) { stride = 1; } // if we have a range then add it if(end != -1 && start != -1 && stride > 0) { int i = 0; while (i*stride + start <= end) { this->TestsToRun.push_back(static_cast(i*stride+start)); ++i; } } // sort the array std::sort(this->TestsToRun.begin(), this->TestsToRun.end(), std::less()); // remove duplicates std::vector::iterator new_end = std::unique(this->TestsToRun.begin(), this->TestsToRun.end()); this->TestsToRun.erase(new_end, this->TestsToRun.end()); } //---------------------------------------------------------------------- // Just for convenience #define SPACE_REGEX "[ \t\r\n]" //---------------------------------------------------------------------- std::string cmCTestTestHandler::GenerateRegressionImages( const std::string& xml) { cmsys::RegularExpression twoattributes( "([^<]*)"); cmsys::RegularExpression threeattributes( "([^<]*)"); cmsys::RegularExpression fourattributes( "([^<]*)"); cmsys::RegularExpression cdatastart( "" SPACE_REGEX "*" SPACE_REGEX "*"); cmsys::RegularExpression measurementfile( "([^<]*)"); cmOStringStream ostr; bool done = false; std::string cxml = xml; while ( ! done ) { if ( twoattributes.find(cxml) ) { ostr << "\t\t\t" << twoattributes.match(5) << "" << std::endl; cxml.erase(twoattributes.start(), twoattributes.end() - twoattributes.start()); } else if ( threeattributes.find(cxml) ) { ostr << "\t\t\t" << threeattributes.match(7) << "" << std::endl; cxml.erase(threeattributes.start(), threeattributes.end() - threeattributes.start()); } else if ( fourattributes.find(cxml) ) { ostr << "\t\t\t" << fourattributes.match(9) << "" << std::endl; cxml.erase(fourattributes.start(), fourattributes.end() - fourattributes.start()); } else if ( cdatastart.find(cxml) && cdataend.find(cxml) ) { ostr << "\t\t\t" << std::endl; cxml.erase(cdatastart.start(), cdataend.end() - cdatastart.start()); } else if ( measurementfile.find(cxml) ) { const std::string& filename = cmCTest::CleanString(measurementfile.match(5)); if ( cmSystemTools::FileExists(filename.c_str()) ) { long len = cmSystemTools::FileLength(filename.c_str()); if ( len == 0 ) { std::string k1 = measurementfile.match(1); std::string v1 = measurementfile.match(2); std::string k2 = measurementfile.match(3); std::string v2 = measurementfile.match(4); if ( cmSystemTools::LowerCase(k1) == "type" ) { v1 = "text/string"; } if ( cmSystemTools::LowerCase(k2) == "type" ) { v2 = "text/string"; } ostr << "\t\t\tImage " << filename.c_str() << " is empty"; } else { std::ifstream ifs(filename.c_str(), std::ios::in #ifdef _WIN32 | std::ios::binary #endif ); unsigned char *file_buffer = new unsigned char [ len + 1 ]; ifs.read(reinterpret_cast(file_buffer), len); unsigned char *encoded_buffer = new unsigned char [ static_cast(len * 1.5 + 5) ]; unsigned long rlen = cmsysBase64_Encode(file_buffer, len, encoded_buffer, 1); unsigned long cc; ostr << "\t\t\t" << std::endl << "\t\t\t\t"; for ( cc = 0; cc < rlen; cc ++ ) { ostr << encoded_buffer[cc]; if ( cc % 60 == 0 && cc ) { ostr << std::endl; } } ostr << "" << std::endl << "\t\t\t" << std::endl; delete [] file_buffer; delete [] encoded_buffer; } } else { int idx = 4; if ( measurementfile.match(1) == "name" ) { idx = 2; } ostr << "\t\t\tFile " << filename.c_str() << " not found" << std::endl; cmCTestLog(this->CTest, HANDLER_OUTPUT, "File \"" << filename.c_str() << "\" not found." << std::endl); } cxml.erase(measurementfile.start(), measurementfile.end() - measurementfile.start()); } else { done = true; } } return ostr.str(); } //---------------------------------------------------------------------- void cmCTestTestHandler::SetIncludeRegExp(const char *arg) { this->IncludeRegExp = arg; } //---------------------------------------------------------------------- void cmCTestTestHandler::SetExcludeRegExp(const char *arg) { this->ExcludeRegExp = arg; } //---------------------------------------------------------------------- void cmCTestTestHandler::SetTestsToRunInformation(const char* in) { if ( !in ) { return; } this->TestsToRunString = in; // if the argument is a file, then read it and use the contents as the // string if(cmSystemTools::FileExists(in)) { std::ifstream fin(in); unsigned long filelen = cmSystemTools::FileLength(in); char* buff = new char[filelen+1]; fin.getline(buff, filelen); buff[fin.gcount()] = 0; this->TestsToRunString = buff; } } //---------------------------------------------------------------------- bool cmCTestTestHandler::CleanTestOutput(std::string& output, size_t remove_threshold) { if ( remove_threshold == 0 ) { return true; } if ( output.find("CTEST_FULL_OUTPUT") != output.npos ) { return true; } cmOStringStream ostr; std::string::size_type cc; std::string::size_type skipsize = 0; int inTag = 0; int skipped = 0; for ( cc = 0; cc < output.size(); cc ++ ) { int ch = output[cc]; if ( ch < 0 || ch > 255 ) { break; } if ( ch == '<' ) { inTag = 1; } if ( !inTag ) { int notskip = 0; // Skip if ( skipsize < remove_threshold ) { ostr << static_cast(ch); notskip = 1; } skipsize ++; if ( notskip && skipsize >= remove_threshold ) { skipped = 1; } } else { ostr << static_cast(ch); } if ( ch == '>' ) { inTag = 0; } } if ( skipped ) { ostr << "..." << std::endl << "The rest of the test output was removed " "since it exceeds the threshold of " << remove_threshold << " characters." << std::endl; } output = ostr.str(); return true; } //---------------------------------------------------------------------- bool cmCTestTestHandler::SetTestsProperties( const std::vector& args) { std::vector::const_iterator it; std::vector tests; bool found = false; for ( it = args.begin(); it != args.end(); ++ it ) { if ( *it == "PROPERTIES" ) { found = true; break; } tests.push_back(*it); } if ( !found ) { return false; } ++ it; // skip PROPERTIES for ( ; it != args.end(); ++ it ) { std::string key = *it; ++ it; if ( it == args.end() ) { break; } std::string val = *it; std::vector::const_iterator tit; for ( tit = tests.begin(); tit != tests.end(); ++ tit ) { cmCTestTestHandler::ListOfTests::iterator rtit; for ( rtit = this->TestList.begin(); rtit != this->TestList.end(); ++ rtit ) { if ( *tit == rtit->Name ) { if ( key == "WILL_FAIL" ) { rtit->WillFail = cmSystemTools::IsOn(val.c_str()); } if ( key == "TIMEOUT" ) { rtit->Timeout = atof(val.c_str()); } if ( key == "FAIL_REGULAR_EXPRESSION" ) { std::vector lval; cmSystemTools::ExpandListArgument(val.c_str(), lval); std::vector::iterator crit; for ( crit = lval.begin(); crit != lval.end(); ++ crit ) { rtit->ErrorRegularExpressions.push_back( std::pair( cmsys::RegularExpression(crit->c_str()), std::string(crit->c_str()))); } } if ( key == "DEPENDS" ) { std::vector lval; cmSystemTools::ExpandListArgument(val.c_str(), lval); std::vector::iterator crit; for ( crit = lval.begin(); crit != lval.end(); ++ crit ) { rtit->Depends.push_back(*crit); } } if ( key == "ENVIRONMENT" ) { std::vector lval; cmSystemTools::ExpandListArgument(val.c_str(), lval); std::vector::iterator crit; for ( crit = lval.begin(); crit != lval.end(); ++ crit ) { rtit->Environment.push_back(*crit); } } if ( key == "LABELS" ) { std::vector lval; cmSystemTools::ExpandListArgument(val.c_str(), lval); std::vector::iterator crit; for ( crit = lval.begin(); crit != lval.end(); ++ crit ) { rtit->Labels.push_back(*crit); } } if ( key == "MEASUREMENT" ) { size_t pos = val.find_first_of("="); if ( pos != val.npos ) { std::string mKey = val.substr(0, pos); const char* mVal = val.c_str() + pos + 1; rtit->Measurements[mKey] = mVal; } else { rtit->Measurements[val] = "1"; } } if ( key == "PASS_REGULAR_EXPRESSION" ) { std::vector lval; cmSystemTools::ExpandListArgument(val.c_str(), lval); std::vector::iterator crit; for ( crit = lval.begin(); crit != lval.end(); ++ crit ) { rtit->RequiredRegularExpressions.push_back( std::pair( cmsys::RegularExpression(crit->c_str()), std::string(crit->c_str()))); } } } } } } return true; } //---------------------------------------------------------------------- bool cmCTestTestHandler::AddTest(const std::vector& args) { const std::string& testname = args[0]; cmCTestLog(this->CTest, DEBUG, "Add test: " << args[0] << std::endl); if (this->UseExcludeRegExpFlag && this->UseExcludeRegExpFirst && this->ExcludeTestsRegularExpression.find(testname.c_str())) { return true; } if ( this->MemCheck ) { std::vector::iterator it; bool found = false; for ( it = this->CustomTestsIgnore.begin(); it != this->CustomTestsIgnore.end(); ++ it ) { if ( *it == testname ) { found = true; break; } } if ( found ) { cmCTestLog(this->CTest, HANDLER_VERBOSE_OUTPUT, "Ignore memcheck: " << *it << std::endl); return true; } } else { std::vector::iterator it; bool found = false; for ( it = this->CustomTestsIgnore.begin(); it != this->CustomTestsIgnore.end(); ++ it ) { if ( *it == testname ) { found = true; break; } } if ( found ) { cmCTestLog(this->CTest, HANDLER_VERBOSE_OUTPUT, "Ignore test: " << *it << std::endl); return true; } } cmCTestTestProperties test; test.Name = testname; test.Args = args; test.Directory = cmSystemTools::GetCurrentWorkingDirectory(); cmCTestLog(this->CTest, DEBUG, "Set test directory: " << test.Directory << std::endl); test.IsInBasedOnREOptions = true; test.WillFail = false; test.Timeout = 0; if (this->UseIncludeRegExpFlag && !this->IncludeTestsRegularExpression.find(testname.c_str())) { test.IsInBasedOnREOptions = false; } else if (this->UseExcludeRegExpFlag && !this->UseExcludeRegExpFirst && this->ExcludeTestsRegularExpression.find(testname.c_str())) { test.IsInBasedOnREOptions = false; } this->TestList.push_back(test); return true; }