diff options
author | Andy Cedilnik <andy.cedilnik@kitware.com> | 2005-06-03 20:10:55 (GMT) |
---|---|---|
committer | Andy Cedilnik <andy.cedilnik@kitware.com> | 2005-06-03 20:10:55 (GMT) |
commit | 1a165dc60de12499eb3610db7efcc5fcbf48b780 (patch) | |
tree | 2fb1a463dbdcb9d5c7435c36ed0dbf77bb61d6e8 /Source | |
parent | 6fe633e1f75d296caf8890ca75c5899492de9f13 (diff) | |
download | CMake-1a165dc60de12499eb3610db7efcc5fcbf48b780.zip CMake-1a165dc60de12499eb3610db7efcc5fcbf48b780.tar.gz CMake-1a165dc60de12499eb3610db7efcc5fcbf48b780.tar.bz2 |
ENH: Move the build-and-test code to a handler
Diffstat (limited to 'Source')
-rw-r--r-- | Source/CMakeLists.txt | 1 | ||||
-rw-r--r-- | Source/CTest/cmCTestBuildAndTestHandler.cxx | 451 | ||||
-rw-r--r-- | Source/CTest/cmCTestBuildAndTestHandler.h | 78 | ||||
-rw-r--r-- | Source/cmCTest.cxx | 465 | ||||
-rw-r--r-- | Source/cmCTest.h | 21 |
5 files changed, 603 insertions, 413 deletions
diff --git a/Source/CMakeLists.txt b/Source/CMakeLists.txt index 845c74e..b3f31e5 100644 --- a/Source/CMakeLists.txt +++ b/Source/CMakeLists.txt @@ -156,6 +156,7 @@ INCLUDE_DIRECTORIES( SET(CMTEST_SRCS cmCTest.cxx CTest/cmCTestBuildCommand.cxx CTest/cmCTestBuildHandler.cxx + CTest/cmCTestBuildAndTestHandler.cxx CTest/cmCTestConfigureCommand.cxx CTest/cmCTestConfigureHandler.cxx CTest/cmCTestCoverageHandler.cxx diff --git a/Source/CTest/cmCTestBuildAndTestHandler.cxx b/Source/CTest/cmCTestBuildAndTestHandler.cxx new file mode 100644 index 0000000..aba5f4a --- /dev/null +++ b/Source/CTest/cmCTestBuildAndTestHandler.cxx @@ -0,0 +1,451 @@ +/*========================================================================= + + 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 "cmCTestBuildAndTestHandler.h" + +#include "cmSystemTools.h" +#include "cmCTest.h" +#include "cmake.h" +#include "cmGlobalGenerator.h" +#include <cmsys/Process.h> + +//---------------------------------------------------------------------- +cmCTestBuildAndTestHandler::cmCTestBuildAndTestHandler() +{ + m_BuildTwoConfig = false; + m_BuildNoClean = false; + m_BuildNoCMake = false; +} + +//---------------------------------------------------------------------- +const char* cmCTestBuildAndTestHandler::GetOutput() +{ + return m_Output.c_str(); +} +//---------------------------------------------------------------------- +int cmCTestBuildAndTestHandler::ProcessHandler() +{ + m_Output = ""; + std::string output; + cmSystemTools::ResetErrorOccuredFlag(); + cmListFileCache::ClearCache(); + int retv = this->RunCMakeAndTest(&m_Output); + cmSystemTools::ResetErrorOccuredFlag(); + cmListFileCache::ClearCache(); + return retv; +} + +//---------------------------------------------------------------------- +int cmCTestBuildAndTestHandler::RunCMake(std::string* outstring, cmOStringStream &out, + std::string &cmakeOutString, std::string &cwd, + cmake *cm) +{ + unsigned int k; + std::vector<std::string> args; + args.push_back(m_CTest->GetCMakeExecutable()); + args.push_back(m_SourceDir); + if(m_BuildGenerator.size()) + { + std::string generator = "-G"; + generator += m_BuildGenerator; + args.push_back(generator); + } + if ( m_CTest->GetConfigType().size() > 0 ) + { + std::string btype = "-DBUILD_TYPE:STRING=" + m_CTest->GetConfigType(); + args.push_back(btype); + } + + for(k=0; k < m_BuildOptions.size(); ++k) + { + args.push_back(m_BuildOptions[k]); + } + if (cm->Run(args) != 0) + { + out << "Error: cmake execution failed\n"; + out << cmakeOutString << "\n"; + // return to the original directory + cmSystemTools::ChangeDirectory(cwd.c_str()); + if(outstring) + { + *outstring = out.str(); + } + else + { + cmCTestLog(m_CTest, ERROR_MESSAGE, out.str() << std::endl); + } + return 1; + } + // do another config? + if(m_BuildTwoConfig) + { + if (cm->Run(args) != 0) + { + out << "Error: cmake execution failed\n"; + out << cmakeOutString << "\n"; + // return to the original directory + cmSystemTools::ChangeDirectory(cwd.c_str()); + if(outstring) + { + *outstring = out.str(); + } + else + { + cmCTestLog(m_CTest, ERROR_MESSAGE, out.str() << std::endl); + } + return 1; + } + } + return 0; +} + +//---------------------------------------------------------------------- +void CMakeMessageCallback(const char* m, const char*, bool&, void* s) +{ + std::string* out = (std::string*)s; + *out += m; + *out += "\n"; +} + +//---------------------------------------------------------------------- +void CMakeStdoutCallback(const char* m, int len, void* s) +{ + std::string* out = (std::string*)s; + out->append(m, len); +} + +//---------------------------------------------------------------------- +int cmCTestBuildAndTestHandler::RunCMakeAndTest(std::string* outstring) +{ + unsigned int k; + std::string cmakeOutString; + cmSystemTools::SetErrorCallback(CMakeMessageCallback, &cmakeOutString); + cmSystemTools::SetStdoutCallback(CMakeStdoutCallback, &cmakeOutString); + cmOStringStream out; + // What is this? double timeout = m_CTest->GetTimeOut(); + int retVal = 0; + + // if the generator and make program are not specified then it is an error + if (!m_BuildGenerator.size() || !m_BuildMakeProgram.size()) + { + if(outstring) + { + *outstring = + "--build-and-test requires that both the generator and makeprogram " + "be provided using the --build-generator and --build-makeprogram " + "command line options. "; + } + return 1; + } + + // make sure the binary dir is there + std::string cwd = cmSystemTools::GetCurrentWorkingDirectory(); + out << "Internal cmake changing into directory: " << m_BinaryDir << "\n"; + if (!cmSystemTools::FileIsDirectory(m_BinaryDir.c_str())) + { + cmSystemTools::MakeDirectory(m_BinaryDir.c_str()); + } + cmSystemTools::ChangeDirectory(m_BinaryDir.c_str()); + + // should we cmake? + cmake cm; + cm.SetGlobalGenerator(cm.CreateGlobalGenerator(m_BuildGenerator.c_str())); + + if(!m_BuildNoCMake) + { + // do the cmake step + if (this->RunCMake(outstring,out,cmakeOutString,cwd,&cm)) + { + return 1; + } + } + + // do the build + std::string output; + retVal = cm.GetGlobalGenerator()->Build( + m_SourceDir.c_str(), m_BinaryDir.c_str(), + m_BuildProject.c_str(), m_BuildTarget.c_str(), + &output, m_BuildMakeProgram.c_str(), + m_CTest->GetConfigType().c_str(),!m_BuildNoClean); + + out << output; + if(outstring) + { + *outstring = out.str(); + } + + // if the build failed then return + if (retVal) + { + return 1; + } + + // if not test was specified then we are done + if (!m_TestCommand.size()) + { + return 0; + } + + // now run the compiled test if we can find it + std::vector<std::string> attempted; + std::vector<std::string> failed; + std::string tempPath; + std::string filepath = + cmSystemTools::GetFilenamePath(m_TestCommand); + std::string filename = + cmSystemTools::GetFilenameName(m_TestCommand); + // if full path specified then search that first + if (filepath.size()) + { + tempPath = filepath; + tempPath += "/"; + tempPath += filename; + attempted.push_back(tempPath); + if(m_CTest->GetConfigType().size()) + { + tempPath = filepath; + tempPath += "/"; + tempPath += m_CTest->GetConfigType(); + tempPath += "/"; + tempPath += filename; + attempted.push_back(tempPath); + } + } + // otherwise search local dirs + else + { + attempted.push_back(filename); + if(m_CTest->GetConfigType().size()) + { + tempPath = m_CTest->GetConfigType(); + tempPath += "/"; + tempPath += filename; + attempted.push_back(tempPath); + } + } + // if m_ExecutableDirectory is set try that as well + if (m_ExecutableDirectory.size()) + { + tempPath = m_ExecutableDirectory; + tempPath += "/"; + tempPath += m_TestCommand; + attempted.push_back(tempPath); + if(m_CTest->GetConfigType().size()) + { + tempPath = m_ExecutableDirectory; + tempPath += "/"; + tempPath += m_CTest->GetConfigType(); + tempPath += "/"; + tempPath += filename; + attempted.push_back(tempPath); + } + } + + // 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()); + } + // 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()); + } + else + { + failed.push_back(tempPath.c_str()); + } + } + } + + if(!cmSystemTools::FileExists(fullPath.c_str())) + { + out << "Could not find path to executable, perhaps it was not built: " << + m_TestCommand << "\n"; + out << "tried to find it in these places:\n"; + out << fullPath.c_str() << "\n"; + for(unsigned int i=0; i < failed.size(); ++i) + { + out << failed[i] << "\n"; + } + if(outstring) + { + *outstring = out.str(); + } + else + { + cmCTestLog(m_CTest, ERROR_MESSAGE, out.str()); + } + // return to the original directory + cmSystemTools::ChangeDirectory(cwd.c_str()); + return 1; + } + + std::vector<const char*> testCommand; + testCommand.push_back(fullPath.c_str()); + for(k=0; k < m_TestCommandArgs.size(); ++k) + { + testCommand.push_back(m_TestCommandArgs[k].c_str()); + } + testCommand.push_back(0); + std::string outs; + int retval = 0; + // run the test from the m_BuildRunDir if set + if(m_BuildRunDir.size()) + { + out << "Run test in directory: " << m_BuildRunDir << "\n"; + cmSystemTools::ChangeDirectory(m_BuildRunDir.c_str()); + } + out << "Running test executable: " << fullPath << " "; + for(k=0; k < m_TestCommandArgs.size(); ++k) + { + out << m_TestCommandArgs[k] << " "; + } + out << "\n"; + // What is this? m_TimeOut = timeout; + int runTestRes = m_CTest->RunTest(testCommand, &outs, &retval, 0); + if(runTestRes != cmsysProcess_State_Exited || retval != 0) + { + out << "Failed to run test command: " << testCommand[0] << "\n"; + retval = 1; + } + + out << outs << "\n"; + if(outstring) + { + *outstring = out.str(); + } + else + { + cmCTestLog(m_CTest, OUTPUT, out.str() << std::endl); + } + return retval; +} + +//---------------------------------------------------------------------- +int cmCTestBuildAndTestHandler::ProcessCommandLineArguments( + const std::string& currentArg, size_t& idx, + const std::vector<std::string>& allArgs) +{ + // --build-and-test options + if(currentArg.find("--build-and-test",0) == 0 && idx < allArgs.size() - 1) + { + if(idx+2 < allArgs.size()) + { + idx++; + m_SourceDir = allArgs[idx]; + idx++; + m_BinaryDir = allArgs[idx]; + // dir must exist before CollapseFullPath is called + cmSystemTools::MakeDirectory(m_BinaryDir.c_str()); + m_BinaryDir = cmSystemTools::CollapseFullPath(m_BinaryDir.c_str()); + m_SourceDir = cmSystemTools::CollapseFullPath(m_SourceDir.c_str()); + } + else + { + cmCTestLog(m_CTest, ERROR_MESSAGE, "--build-and-test must have source and binary dir" << std::endl); + return 0; + } + } + if(currentArg.find("--build-target",0) == 0 && idx < allArgs.size() - 1) + { + idx++; + m_BuildTarget = allArgs[idx]; + } + if(currentArg.find("--build-nocmake",0) == 0) + { + m_BuildNoCMake = true; + } + if(currentArg.find("--build-run-dir",0) == 0 && idx < allArgs.size() - 1) + { + idx++; + m_BuildRunDir = allArgs[idx]; + } + if(currentArg.find("--build-two-config",0) == 0) + { + m_BuildTwoConfig = true; + } + if(currentArg.find("--build-exe-dir",0) == 0 && idx < allArgs.size() - 1) + { + idx++; + m_ExecutableDirectory = allArgs[idx]; + } + if(currentArg.find("--build-generator",0) == 0 && idx < allArgs.size() - 1) + { + idx++; + m_BuildGenerator = allArgs[idx]; + } + if(currentArg.find("--build-project",0) == 0 && idx < allArgs.size() - 1) + { + idx++; + m_BuildProject = allArgs[idx]; + } + if(currentArg.find("--build-makeprogram",0) == 0 && idx < allArgs.size() - 1) + { + idx++; + m_BuildMakeProgram = allArgs[idx]; + } + if(currentArg.find("--build-noclean",0) == 0) + { + m_BuildNoClean = true; + } + if(currentArg.find("--build-options",0) == 0 && idx < allArgs.size() - 1) + { + ++idx; + bool done = false; + while(idx < allArgs.size() && !done) + { + m_BuildOptions.push_back(allArgs[idx]); + if(idx+1 < allArgs.size() + && (allArgs[idx+1] == "--build-target" || allArgs[idx+1] == "--test-command")) + { + done = true; + } + else + { + ++idx; + } + } + } + if(currentArg.find("--test-command",0) == 0 && idx < allArgs.size() - 1) + { + ++idx; + m_TestCommand = allArgs[idx]; + while(idx+1 < allArgs.size()) + { + ++idx; + m_TestCommandArgs.push_back(allArgs[idx]); + } + } + return 1; +} + + diff --git a/Source/CTest/cmCTestBuildAndTestHandler.h b/Source/CTest/cmCTestBuildAndTestHandler.h new file mode 100644 index 0000000..788b28f --- /dev/null +++ b/Source/CTest/cmCTestBuildAndTestHandler.h @@ -0,0 +1,78 @@ +/*========================================================================= + + Program: CMake - Cross-Platform Makefile Generator + Module: $RCSfile$ + Language: C++ + Date: $Date$ + Version: $Revision$ + + Copyright (c) 2002 Kitware, Inc. 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. + +=========================================================================*/ + +#ifndef cmCTestBuildAndTestHandler_h +#define cmCTestBuildAndTestHandler_h + + +#include "cmCTestGenericHandler.h" +#include "cmListFileCache.h" + +class cmake; + +/** \class cmCTestBuildAndTestHandler + * \brief A class that handles ctest -S invocations + * + */ +class cmCTestBuildAndTestHandler : public cmCTestGenericHandler +{ +public: + + /* + * The main entry point for this class + */ + int ProcessHandler(); + + //! Set all the build and test arguments + virtual int ProcessCommandLineArguments( + const std::string& currentArg, size_t& idx, + const std::vector<std::string>& allArgs); + + /* + * Get the output variable + */ + const char* GetOutput(); + + cmCTestBuildAndTestHandler(); + +protected: + ///! Run CMake and build a test and then run it as a single test. + int RunCMakeAndTest(std::string* output); + int RunCMake(std::string* outstring, cmOStringStream &out, + std::string &cmakeOutString, + std::string &cwd, cmake *cm); + + cmStdString m_Output; + + std::string m_BuildGenerator; + std::vector<std::string> m_BuildOptions; + bool m_BuildTwoConfig; + std::string m_BuildMakeProgram; + std::string m_SourceDir; + std::string m_BinaryDir; + std::string m_BuildProject; + std::string m_TestCommand; + bool m_BuildNoClean; + std::string m_BuildRunDir; + std::string m_ExecutableDirectory; + std::vector<std::string> m_TestCommandArgs; + std::string m_BuildTarget; + bool m_BuildNoCMake; +}; + +#endif + diff --git a/Source/cmCTest.cxx b/Source/cmCTest.cxx index 99ed55c..9f6d77a 100644 --- a/Source/cmCTest.cxx +++ b/Source/cmCTest.cxx @@ -27,6 +27,7 @@ #include "cmGeneratedFileStream.h" #include "cmCTestBuildHandler.h" +#include "cmCTestBuildAndTestHandler.h" #include "cmCTestConfigureHandler.h" #include "cmCTestCoverageHandler.h" #include "cmCTestMemCheckHandler.h" @@ -49,6 +50,7 @@ #define DEBUGOUT std::cout << __LINE__ << " "; std::cout #define DEBUGERR std::cerr << __LINE__ << " "; std::cerr +//---------------------------------------------------------------------- struct tm* cmCTest::GetNightlyTime(std::string str, bool tomorrowtag) { @@ -97,6 +99,7 @@ struct tm* cmCTest::GetNightlyTime(std::string str, return lctime; } +//---------------------------------------------------------------------- std::string cmCTest::CleanString(const std::string& str) { std::string::size_type spos = str.find_first_not_of(" \n\t\r\f\v"); @@ -112,6 +115,7 @@ std::string cmCTest::CleanString(const std::string& str) return str.substr(spos, epos); } +//---------------------------------------------------------------------- std::string cmCTest::CurrentTime() { time_t currenttime = time(0); @@ -131,6 +135,7 @@ std::string cmCTest::CurrentTime() } +//---------------------------------------------------------------------- std::string cmCTest::MakeXMLSafe(const std::string& str) { cmOStringStream ost; @@ -174,6 +179,7 @@ std::string cmCTest::MakeXMLSafe(const std::string& str) return ost.str(); } +//---------------------------------------------------------------------- std::string cmCTest::MakeURLSafe(const std::string& str) { cmOStringStream ost; @@ -200,13 +206,11 @@ std::string cmCTest::MakeURLSafe(const std::string& str) return ost.str(); } +//---------------------------------------------------------------------- cmCTest::cmCTest() { m_ForceNewCTestProcess = false; m_TomorrowTag = false; - m_BuildNoCMake = false; - m_BuildNoClean = false; - m_BuildTwoConfig = false; m_Verbose = false; m_Debug = false; m_Quiet = false; @@ -231,6 +235,7 @@ cmCTest::cmCTest() m_ShortDateFormat = true; m_TestingHandlers["build"] = new cmCTestBuildHandler; + m_TestingHandlers["buildtest"] = new cmCTestBuildAndTestHandler; m_TestingHandlers["coverage"] = new cmCTestCoverageHandler; m_TestingHandlers["script"] = new cmCTestScriptHandler; m_TestingHandlers["test"] = new cmCTestTestHandler; @@ -246,6 +251,7 @@ cmCTest::cmCTest() } } +//---------------------------------------------------------------------- cmCTest::~cmCTest() { cmCTest::t_TestingHandlers::iterator it; @@ -257,6 +263,7 @@ cmCTest::~cmCTest() this->SetOutputLogFileName(0); } +//---------------------------------------------------------------------- int cmCTest::Initialize(const char* binary_dir, bool new_tag) { if(!m_InteractiveDebugMode) @@ -357,6 +364,7 @@ int cmCTest::Initialize(const char* binary_dir, bool new_tag) return 1; } +//---------------------------------------------------------------------- bool cmCTest::UpdateCTestConfiguration() { if ( m_SuppressUpdatingCTestConfiguration ) @@ -425,6 +433,7 @@ bool cmCTest::UpdateCTestConfiguration() return true; } +//---------------------------------------------------------------------- void cmCTest::BlockTestErrorDiagnostics() { cmSystemTools::PutEnv("DART_TEST_FROM_DART=1"); @@ -434,12 +443,14 @@ void cmCTest::BlockTestErrorDiagnostics() #endif } +//---------------------------------------------------------------------- void cmCTest::SetTestModel(int mode) { m_InteractiveDebugMode = false; m_TestModel = mode; } +//---------------------------------------------------------------------- bool cmCTest::SetTest(const char* ttype, bool report) { if ( cmSystemTools::LowerCase(ttype) == "all" ) @@ -493,12 +504,12 @@ bool cmCTest::SetTest(const char* ttype, bool report) return true; } +//---------------------------------------------------------------------- void cmCTest::Finalize() { } - - +//---------------------------------------------------------------------- bool cmCTest::OpenOutputFile(const std::string& path, const std::string& name, cmGeneratedFileStream& stream, bool compress) @@ -544,6 +555,7 @@ bool cmCTest::OpenOutputFile(const std::string& path, return true; } +//---------------------------------------------------------------------- bool cmCTest::AddIfExists(tm_VectorOfStrings& files, const char* file) { if ( this->CTestFileExists(file) ) @@ -566,6 +578,7 @@ bool cmCTest::AddIfExists(tm_VectorOfStrings& files, const char* file) return true; } +//---------------------------------------------------------------------- bool cmCTest::CTestFileExists(const std::string& filename) { std::string testingDir = m_BinaryDir + "/Testing/" + m_CurrentTag + "/" + @@ -573,6 +586,7 @@ bool cmCTest::CTestFileExists(const std::string& filename) return cmSystemTools::FileExists(testingDir.c_str()); } +//---------------------------------------------------------------------- cmCTestGenericHandler* cmCTest::GetHandler(const char* handler) { cmCTest::t_TestingHandlers::iterator it = m_TestingHandlers.find(handler); @@ -583,6 +597,7 @@ cmCTestGenericHandler* cmCTest::GetHandler(const char* handler) return it->second; } +//---------------------------------------------------------------------- int cmCTest::ExecuteHandler(const char* shandler) { cmCTestGenericHandler* handler = this->GetHandler(shandler); @@ -593,6 +608,7 @@ int cmCTest::ExecuteHandler(const char* shandler) return handler->ProcessHandler(); } +//---------------------------------------------------------------------- int cmCTest::ProcessTests() { int res = 0; @@ -705,6 +721,7 @@ int cmCTest::ProcessTests() return res; } +//---------------------------------------------------------------------- std::string cmCTest::GetTestModelString() { switch ( m_TestModel ) @@ -717,6 +734,7 @@ std::string cmCTest::GetTestModelString() return "Experimental"; } +//---------------------------------------------------------------------- int cmCTest::GetTestModelFromString(const char* str) { if ( !str ) @@ -735,6 +753,7 @@ int cmCTest::GetTestModelFromString(const char* str) return cmCTest::EXPERIMENTAL; } +//---------------------------------------------------------------------- int cmCTest::RunMakeCommand(const char* command, std::string* output, int* retVal, const char* dir, int timeout, std::ofstream& ofs) { @@ -835,6 +854,7 @@ int cmCTest::RunMakeCommand(const char* command, std::string* output, return result; } +//---------------------------------------------------------------------- int cmCTest::RunTest(std::vector<const char*> argv, std::string* output, int *retVal, std::ostream* log) @@ -934,6 +954,7 @@ int cmCTest::RunTest(std::vector<const char*> argv, return result; } +//---------------------------------------------------------------------- void cmCTest::StartXML(std::ostream& ostr) { ostr << "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n" @@ -945,11 +966,13 @@ void cmCTest::StartXML(std::ostream& ostr) << "\">" << std::endl; } +//---------------------------------------------------------------------- void cmCTest::EndXML(std::ostream& ostr) { ostr << "</Site>" << std::endl; } +//---------------------------------------------------------------------- int cmCTest::GenerateCTestNotesOutput(std::ostream& os, const cmCTest::tm_VectorOfStrings& files) { cmCTest::tm_VectorOfStrings::const_iterator it; @@ -992,6 +1015,7 @@ int cmCTest::GenerateCTestNotesOutput(std::ostream& os, const cmCTest::tm_Vector return 1; } +//---------------------------------------------------------------------- int cmCTest::GenerateNotesFile(const std::vector<cmStdString> &files) { cmGeneratedFileStream ofs; @@ -1005,6 +1029,7 @@ int cmCTest::GenerateNotesFile(const std::vector<cmStdString> &files) return 0; } +//---------------------------------------------------------------------- int cmCTest::GenerateNotesFile(const char* cfiles) { if ( !cfiles ) @@ -1025,13 +1050,14 @@ int cmCTest::GenerateNotesFile(const char* cfiles) return this->GenerateNotesFile(files); } +//---------------------------------------------------------------------- int cmCTest::Run(std::vector<std::string>const& args, std::string* output) { this->FindRunningCMake(args[0].c_str()); const char* ctestExec = "ctest"; bool cmakeAndTest = false; bool performSomeTest = true; - for(unsigned int i=1; i < args.size(); ++i) + for(size_t i=1; i < args.size(); ++i) { std::string arg = args[i]; if(arg.find("--ctest-config",0) == 0 && i < args.size() - 1) @@ -1361,7 +1387,7 @@ int cmCTest::Run(std::vector<std::string>const& args, std::string* output) i++; this->GetHandler("test")->SetOption("IncludeRegularExpression", args[i].c_str()); } - + if(arg.find("-E",0) == 0 && i < args.size() - 1) { i++; @@ -1375,107 +1401,45 @@ int cmCTest::Run(std::vector<std::string>const& args, std::string* output) i++; this->SetNotesFiles(args[i].c_str()); } - // --build-and-test options if(arg.find("--build-and-test",0) == 0 && i < args.size() - 1) { cmakeAndTest = true; - if(i+2 < args.size()) - { - i++; - m_SourceDir = args[i]; - i++; - m_BinaryDir = args[i]; - // dir must exist before CollapseFullPath is called - cmSystemTools::MakeDirectory(m_BinaryDir.c_str()); - m_BinaryDir = cmSystemTools::CollapseFullPath(m_BinaryDir.c_str()); - m_SourceDir = cmSystemTools::CollapseFullPath(m_SourceDir.c_str()); - } - else - { - cmCTestLog(this, ERROR_MESSAGE, "--build-and-test must have source and binary dir" << std::endl); - } - } - if(arg.find("--build-target",0) == 0 && i < args.size() - 1) - { - i++; - m_BuildTarget = args[i]; - } - if(arg.find("--build-nocmake",0) == 0) - { - m_BuildNoCMake = true; - } - if(arg.find("--build-run-dir",0) == 0 && i < args.size() - 1) - { - i++; - m_BuildRunDir = args[i]; - } - if(arg.find("--build-two-config",0) == 0) - { - m_BuildTwoConfig = true; - } - if(arg.find("--build-exe-dir",0) == 0 && i < args.size() - 1) - { - i++; - m_ExecutableDirectory = args[i]; - } - if(arg.find("--build-generator",0) == 0 && i < args.size() - 1) - { - i++; - m_BuildGenerator = args[i]; - } - if(arg.find("--build-project",0) == 0 && i < args.size() - 1) - { - i++; - m_BuildProject = args[i]; - } - if(arg.find("--build-makeprogram",0) == 0 && i < args.size() - 1) - { - i++; - m_BuildMakeProgram = args[i]; - } - if(arg.find("--build-noclean",0) == 0) - { - m_BuildNoClean = true; - } - if(arg.find("--build-options",0) == 0 && i < args.size() - 1) - { - ++i; - bool done = false; - while(i < args.size() && !done) - { - m_BuildOptions.push_back(args[i]); - if(i+1 < args.size() - && (args[i+1] == "--build-target" || args[i+1] == "--test-command")) - { - done = true; - } - else - { - ++i; - } - } } - if(arg.find("--test-command",0) == 0 && i < args.size() - 1) + cmCTest::t_TestingHandlers::iterator it; + for ( it = m_TestingHandlers.begin(); it != m_TestingHandlers.end(); ++ it ) { - ++i; - m_TestCommand = args[i]; - while(i+1 < args.size()) + if ( !it->second->ProcessCommandLineArguments(arg, i, args) ) { - ++i; - m_TestCommandArgs.push_back(args[i]); + cmCTestLog(this, ERROR_MESSAGE, "Problem parsing command line arguments within a handler"); + return 0; } } } + // default to the build type of ctest itself + if(m_ConfigType.size() == 0) + { +#ifdef CMAKE_INTDIR + m_ConfigType = CMAKE_INTDIR; +#endif + } + + if(cmakeAndTest) { m_Verbose = true; + cmCTestBuildAndTestHandler* handler = + static_cast<cmCTestBuildAndTestHandler*>(this->GetHandler("buildtest")); + int retv = handler->ProcessHandler(); + /* cmSystemTools::ResetErrorOccuredFlag(); cmListFileCache::ClearCache(); int retv = this->RunCMakeAndTest(output); cmSystemTools::ResetErrorOccuredFlag(); cmListFileCache::ClearCache(); + */ + *output = handler->GetOutput(); #ifdef CMAKE_BUILD_WITH_CMAKE cmDynamicLoader::FlushCache(); #endif @@ -1520,6 +1484,7 @@ int cmCTest::Run(std::vector<std::string>const& args, std::string* output) return 1; } +//---------------------------------------------------------------------- void cmCTest::FindRunningCMake(const char* arg0) { // Find our own executable. @@ -1587,311 +1552,7 @@ void cmCTest::FindRunningCMake(const char* arg0) } } -void CMakeMessageCallback(const char* m, const char*, bool&, void* s) -{ - std::string* out = (std::string*)s; - *out += m; - *out += "\n"; -} - -void CMakeStdoutCallback(const char* m, int len, void* s) -{ - std::string* out = (std::string*)s; - out->append(m, len); -} - -int cmCTest::RunCMake(std::string* outstring, cmOStringStream &out, - std::string &cmakeOutString, std::string &cwd, - cmake *cm) -{ - unsigned int k; - std::vector<std::string> args; - args.push_back(m_CMakeSelf); - args.push_back(m_SourceDir); - if(m_BuildGenerator.size()) - { - std::string generator = "-G"; - generator += m_BuildGenerator; - args.push_back(generator); - } - if ( m_ConfigType.size() > 0 ) - { - std::string btype = "-DBUILD_TYPE:STRING=" + m_ConfigType; - args.push_back(btype); - } - - for(k=0; k < m_BuildOptions.size(); ++k) - { - args.push_back(m_BuildOptions[k]); - } - if (cm->Run(args) != 0) - { - out << "Error: cmake execution failed\n"; - out << cmakeOutString << "\n"; - // return to the original directory - cmSystemTools::ChangeDirectory(cwd.c_str()); - if(outstring) - { - *outstring = out.str(); - } - else - { - cmCTestLog(this, ERROR_MESSAGE, out.str() << std::endl); - } - return 1; - } - // do another config? - if(m_BuildTwoConfig) - { - if (cm->Run(args) != 0) - { - out << "Error: cmake execution failed\n"; - out << cmakeOutString << "\n"; - // return to the original directory - cmSystemTools::ChangeDirectory(cwd.c_str()); - if(outstring) - { - *outstring = out.str(); - } - else - { - cmCTestLog(this, ERROR_MESSAGE, out.str() << std::endl); - } - return 1; - } - } - return 0; -} - -int cmCTest::RunCMakeAndTest(std::string* outstring) -{ - unsigned int k; - std::string cmakeOutString; - cmSystemTools::SetErrorCallback(CMakeMessageCallback, &cmakeOutString); - cmSystemTools::SetStdoutCallback(CMakeStdoutCallback, &cmakeOutString); - cmOStringStream out; - double timeout = m_TimeOut; - int retVal = 0; - - // if the generator and make program are not specified then it is an error - if (!m_BuildGenerator.size() || !m_BuildMakeProgram.size()) - { - if(outstring) - { - *outstring = - "--build-and-test requires that both the generator and makeprogram " - "be provided using the --build-generator and --build-makeprogram " - "command line options. "; - } - return 1; - } - - // default to the build type of ctest itself - if(m_ConfigType.size() == 0) - { -#ifdef CMAKE_INTDIR - m_ConfigType = CMAKE_INTDIR; -#endif - } - - // make sure the binary dir is there - std::string cwd = cmSystemTools::GetCurrentWorkingDirectory(); - out << "Internal cmake changing into directory: " << m_BinaryDir << "\n"; - if (!cmSystemTools::FileIsDirectory(m_BinaryDir.c_str())) - { - cmSystemTools::MakeDirectory(m_BinaryDir.c_str()); - } - cmSystemTools::ChangeDirectory(m_BinaryDir.c_str()); - - // should we cmake? - cmake cm; - cm.SetGlobalGenerator(cm.CreateGlobalGenerator(m_BuildGenerator.c_str())); - - if(!m_BuildNoCMake) - { - // do the cmake step - if (this->RunCMake(outstring,out,cmakeOutString,cwd,&cm)) - { - return 1; - } - } - - // do the build - std::string output; - retVal = cm.GetGlobalGenerator()->Build( - m_SourceDir.c_str(), m_BinaryDir.c_str(), - m_BuildProject.c_str(), m_BuildTarget.c_str(), - &output, m_BuildMakeProgram.c_str(), - m_ConfigType.c_str(),!m_BuildNoClean); - - out << output; - if(outstring) - { - *outstring = out.str(); - } - - // if the build failed then return - if (retVal) - { - return 1; - } - - // if not test was specified then we are done - if (!m_TestCommand.size()) - { - return 0; - } - - // now run the compiled test if we can find it - std::vector<std::string> attempted; - std::vector<std::string> failed; - std::string tempPath; - std::string filepath = - cmSystemTools::GetFilenamePath(m_TestCommand); - std::string filename = - cmSystemTools::GetFilenameName(m_TestCommand); - // if full path specified then search that first - if (filepath.size()) - { - tempPath = filepath; - tempPath += "/"; - tempPath += filename; - attempted.push_back(tempPath); - if(m_ConfigType.size()) - { - tempPath = filepath; - tempPath += "/"; - tempPath += m_ConfigType; - tempPath += "/"; - tempPath += filename; - attempted.push_back(tempPath); - } - } - // otherwise search local dirs - else - { - attempted.push_back(filename); - if(m_ConfigType.size()) - { - tempPath = m_ConfigType; - tempPath += "/"; - tempPath += filename; - attempted.push_back(tempPath); - } - } - // if m_ExecutableDirectory is set try that as well - if (m_ExecutableDirectory.size()) - { - tempPath = m_ExecutableDirectory; - tempPath += "/"; - tempPath += m_TestCommand; - attempted.push_back(tempPath); - if(m_ConfigType.size()) - { - tempPath = m_ExecutableDirectory; - tempPath += "/"; - tempPath += m_ConfigType; - tempPath += "/"; - tempPath += filename; - attempted.push_back(tempPath); - } - } - - // 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()); - } - // 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()); - } - else - { - failed.push_back(tempPath.c_str()); - } - } - } - - if(!cmSystemTools::FileExists(fullPath.c_str())) - { - out << "Could not find path to executable, perhaps it was not built: " << - m_TestCommand << "\n"; - out << "tried to find it in these places:\n"; - out << fullPath.c_str() << "\n"; - for(unsigned int i=0; i < failed.size(); ++i) - { - out << failed[i] << "\n"; - } - if(outstring) - { - *outstring = out.str(); - } - else - { - cmCTestLog(this, ERROR_MESSAGE, out.str()); - } - // return to the original directory - cmSystemTools::ChangeDirectory(cwd.c_str()); - return 1; - } - - std::vector<const char*> testCommand; - testCommand.push_back(fullPath.c_str()); - for(k=0; k < m_TestCommandArgs.size(); ++k) - { - testCommand.push_back(m_TestCommandArgs[k].c_str()); - } - testCommand.push_back(0); - std::string outs; - int retval = 0; - // run the test from the m_BuildRunDir if set - if(m_BuildRunDir.size()) - { - out << "Run test in directory: " << m_BuildRunDir << "\n"; - cmSystemTools::ChangeDirectory(m_BuildRunDir.c_str()); - } - out << "Running test executable: " << fullPath << " "; - for(k=0; k < m_TestCommandArgs.size(); ++k) - { - out << m_TestCommandArgs[k] << " "; - } - out << "\n"; - m_TimeOut = timeout; - int runTestRes = this->RunTest(testCommand, &outs, &retval, 0); - if(runTestRes != cmsysProcess_State_Exited || retval != 0) - { - out << "Failed to run test command: " << testCommand[0] << "\n"; - retval = 1; - } - - out << outs << "\n"; - if(outstring) - { - *outstring = out.str(); - } - else - { - cmCTestLog(this, OUTPUT, out.str() << std::endl); - } - return retval; -} - +//---------------------------------------------------------------------- void cmCTest::SetNotesFiles(const char* notes) { if ( !notes ) @@ -1901,6 +1562,7 @@ void cmCTest::SetNotesFiles(const char* notes) m_NotesFiles = notes; } +//---------------------------------------------------------------------- int cmCTest::ReadCustomConfigurationFileTree(const char* dir) { tm_VectorOfStrings dirs; @@ -1938,6 +1600,7 @@ int cmCTest::ReadCustomConfigurationFileTree(const char* dir) return 1; } +//---------------------------------------------------------------------- void cmCTest::PopulateCustomVector(cmMakefile* mf, const char* def, tm_VectorOfStrings& vec) { if ( !def) @@ -1959,6 +1622,7 @@ void cmCTest::PopulateCustomVector(cmMakefile* mf, const char* def, tm_VectorOfS } } +//---------------------------------------------------------------------- void cmCTest::PopulateCustomInteger(cmMakefile* mf, const char* def, int& val) { if ( !def) @@ -1973,6 +1637,7 @@ void cmCTest::PopulateCustomInteger(cmMakefile* mf, const char* def, int& val) val = atoi(dval); } +//---------------------------------------------------------------------- std::string cmCTest::GetShortPathToFile(const char* cfname) { const std::string& sourceDir = this->GetCTestConfiguration("SourceDirectory"); @@ -2026,11 +1691,13 @@ std::string cmCTest::GetShortPathToFile(const char* cfname) return path; } +//---------------------------------------------------------------------- std::string cmCTest::GetCTestConfiguration(const char *name) { return m_CTestConfiguration[name]; } +//---------------------------------------------------------------------- void cmCTest::SetCTestConfiguration(const char *name, const char* value) { if ( !name ) @@ -2046,36 +1713,43 @@ void cmCTest::SetCTestConfiguration(const char *name, const char* value) } +//---------------------------------------------------------------------- std::string cmCTest::GetCurrentTag() { return m_CurrentTag; } +//---------------------------------------------------------------------- std::string cmCTest::GetBinaryDir() { return m_BinaryDir; } +//---------------------------------------------------------------------- std::string cmCTest::GetConfigType() { return m_ConfigType; } +//---------------------------------------------------------------------- bool cmCTest::GetShowOnly() { return m_ShowOnly; } +//---------------------------------------------------------------------- void cmCTest::SetProduceXML(bool v) { m_ProduceXML = v; } +//---------------------------------------------------------------------- bool cmCTest::GetProduceXML() { return m_ProduceXML; } +//---------------------------------------------------------------------- bool cmCTest::SetCTestConfigurationFromCMakeVariable(cmMakefile* mf, const char* dconfig, const char* cmake_var) { const char* ctvar; @@ -2088,6 +1762,7 @@ bool cmCTest::SetCTestConfigurationFromCMakeVariable(cmMakefile* mf, const char* return true; } +//---------------------------------------------------------------------- void cmCTest::SetOutputLogFileName(const char* name) { if ( m_OutputLogFile) @@ -2101,6 +1776,7 @@ void cmCTest::SetOutputLogFileName(const char* name) } } +//---------------------------------------------------------------------- static const char* cmCTestStringLogType[] = { "DEBUG", @@ -2112,6 +1788,7 @@ static const char* cmCTestStringLogType[] = 0 }; +//---------------------------------------------------------------------- #ifdef cerr # undef cerr #endif diff --git a/Source/cmCTest.h b/Source/cmCTest.h index 00ce135..bb5cf4c 100644 --- a/Source/cmCTest.h +++ b/Source/cmCTest.h @@ -88,6 +88,7 @@ public: ///! what is the configuraiton type, e.g. Debug, Release etc. std::string GetConfigType(); + double GetTimeOut() { return m_TimeOut; } /** * Check if CTest file exists @@ -289,22 +290,10 @@ private: int m_CompatibilityMode; // information for the --build-and-test options - std::string m_ExecutableDirectory; std::string m_CMakeSelf; std::string m_CTestSelf; - std::string m_SourceDir; std::string m_BinaryDir; - std::string m_BuildRunDir; - std::string m_BuildGenerator; - std::string m_BuildMakeProgram; - std::string m_BuildProject; - std::string m_BuildTarget; - std::vector<std::string> m_BuildOptions; - std::string m_TestCommand; - std::vector<std::string> m_TestCommandArgs; - bool m_BuildTwoConfig; - bool m_BuildNoClean; - bool m_BuildNoCMake; + std::string m_NotesFiles; @@ -325,12 +314,6 @@ private: //! Create not from files. int GenerateCTestNotesOutput(std::ostream& os, const tm_VectorOfStrings& files); - ///! Run CMake and build a test and then run it as a single test. - int RunCMakeAndTest(std::string* output); - int RunCMake(std::string* outstring, cmOStringStream &out, - std::string &cmakeOutString, - std::string &cwd, cmake *cm); - ///! Find the running cmake void FindRunningCMake(const char* arg0); |