mirror of
https://github.com/Kitware/CMake.git
synced 2025-10-14 02:08:27 +08:00

Since commit 6a6f1d1edd
(CTest: exit nonzero after
message(SEND_ERROR|FATAL_ERROR), 2020-04-03, v3.19.0-rc1~260^2), `ctest`
no longer runs tests if there are errors before the full set of tests is
defined. Such errors were previously treated more like warnings.
The change exposed some cases where we were issuing an error message but
proceeding to run tests anyway. The above commit downgraded one such
case (missing `DartConfiguration.tcl`) to a warning explicitly in order
to restore its former warning-like semantics.
Downgrade the Update step's diagnostic about modified or conflicting
files to a warning for the same reason.
Fixes: #21783
354 lines
11 KiB
C++
354 lines
11 KiB
C++
/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
|
|
file Copyright.txt or https://cmake.org/licensing for details. */
|
|
#include "cmCTestUpdateHandler.h"
|
|
|
|
#include <chrono>
|
|
#include <sstream>
|
|
|
|
#include <cm/memory>
|
|
|
|
#include "cmCLocaleEnvironmentScope.h"
|
|
#include "cmCTest.h"
|
|
#include "cmCTestBZR.h"
|
|
#include "cmCTestCVS.h"
|
|
#include "cmCTestGIT.h"
|
|
#include "cmCTestHG.h"
|
|
#include "cmCTestP4.h"
|
|
#include "cmCTestSVN.h"
|
|
#include "cmCTestVC.h"
|
|
#include "cmGeneratedFileStream.h"
|
|
#include "cmStringAlgorithms.h"
|
|
#include "cmSystemTools.h"
|
|
#include "cmVersion.h"
|
|
#include "cmXMLWriter.h"
|
|
|
|
static const char* cmCTestUpdateHandlerUpdateStrings[] = {
|
|
"Unknown", "CVS", "SVN", "BZR", "GIT", "HG", "P4"
|
|
};
|
|
|
|
static const char* cmCTestUpdateHandlerUpdateToString(int type)
|
|
{
|
|
if (type < cmCTestUpdateHandler::e_UNKNOWN ||
|
|
type >= cmCTestUpdateHandler::e_LAST) {
|
|
return cmCTestUpdateHandlerUpdateStrings[cmCTestUpdateHandler::e_UNKNOWN];
|
|
}
|
|
return cmCTestUpdateHandlerUpdateStrings[type];
|
|
}
|
|
|
|
cmCTestUpdateHandler::cmCTestUpdateHandler() = default;
|
|
|
|
void cmCTestUpdateHandler::Initialize()
|
|
{
|
|
this->Superclass::Initialize();
|
|
this->UpdateCommand.clear();
|
|
this->UpdateType = e_CVS;
|
|
}
|
|
|
|
int cmCTestUpdateHandler::DetermineType(const char* cmd, const char* type)
|
|
{
|
|
cmCTestOptionalLog(this->CTest, DEBUG,
|
|
"Determine update type from command: "
|
|
<< cmd << " and type: " << type << std::endl,
|
|
this->Quiet);
|
|
if (type && *type) {
|
|
cmCTestOptionalLog(this->CTest, DEBUG,
|
|
"Type specified: " << type << std::endl, this->Quiet);
|
|
std::string stype = cmSystemTools::LowerCase(type);
|
|
if (stype.find("cvs") != std::string::npos) {
|
|
return cmCTestUpdateHandler::e_CVS;
|
|
}
|
|
if (stype.find("svn") != std::string::npos) {
|
|
return cmCTestUpdateHandler::e_SVN;
|
|
}
|
|
if (stype.find("bzr") != std::string::npos) {
|
|
return cmCTestUpdateHandler::e_BZR;
|
|
}
|
|
if (stype.find("git") != std::string::npos) {
|
|
return cmCTestUpdateHandler::e_GIT;
|
|
}
|
|
if (stype.find("hg") != std::string::npos) {
|
|
return cmCTestUpdateHandler::e_HG;
|
|
}
|
|
if (stype.find("p4") != std::string::npos) {
|
|
return cmCTestUpdateHandler::e_P4;
|
|
}
|
|
} else {
|
|
cmCTestOptionalLog(
|
|
this->CTest, DEBUG,
|
|
"Type not specified, check command: " << cmd << std::endl, this->Quiet);
|
|
std::string stype = cmSystemTools::LowerCase(cmd);
|
|
if (stype.find("cvs") != std::string::npos) {
|
|
return cmCTestUpdateHandler::e_CVS;
|
|
}
|
|
if (stype.find("svn") != std::string::npos) {
|
|
return cmCTestUpdateHandler::e_SVN;
|
|
}
|
|
if (stype.find("bzr") != std::string::npos) {
|
|
return cmCTestUpdateHandler::e_BZR;
|
|
}
|
|
if (stype.find("git") != std::string::npos) {
|
|
return cmCTestUpdateHandler::e_GIT;
|
|
}
|
|
if (stype.find("hg") != std::string::npos) {
|
|
return cmCTestUpdateHandler::e_HG;
|
|
}
|
|
if (stype.find("p4") != std::string::npos) {
|
|
return cmCTestUpdateHandler::e_P4;
|
|
}
|
|
}
|
|
return cmCTestUpdateHandler::e_UNKNOWN;
|
|
}
|
|
|
|
// clearly it would be nice if this were broken up into a few smaller
|
|
// functions and commented...
|
|
int cmCTestUpdateHandler::ProcessHandler()
|
|
{
|
|
// Make sure VCS tool messages are in English so we can parse them.
|
|
cmCLocaleEnvironmentScope fixLocale;
|
|
static_cast<void>(fixLocale);
|
|
|
|
// Get source dir
|
|
const char* sourceDirectory = this->GetOption("SourceDirectory");
|
|
if (!sourceDirectory) {
|
|
cmCTestLog(this->CTest, ERROR_MESSAGE,
|
|
"Cannot find SourceDirectory key in the DartConfiguration.tcl"
|
|
<< std::endl);
|
|
return -1;
|
|
}
|
|
|
|
cmGeneratedFileStream ofs;
|
|
if (!this->CTest->GetShowOnly()) {
|
|
this->StartLogFile("Update", ofs);
|
|
}
|
|
|
|
cmCTestOptionalLog(this->CTest, HANDLER_OUTPUT,
|
|
" Updating the repository: " << sourceDirectory
|
|
<< std::endl,
|
|
this->Quiet);
|
|
|
|
if (!this->SelectVCS()) {
|
|
return -1;
|
|
}
|
|
|
|
cmCTestOptionalLog(this->CTest, HANDLER_OUTPUT,
|
|
" Use "
|
|
<< cmCTestUpdateHandlerUpdateToString(this->UpdateType)
|
|
<< " repository type" << std::endl;
|
|
, this->Quiet);
|
|
|
|
// Create an object to interact with the VCS tool.
|
|
std::unique_ptr<cmCTestVC> vc;
|
|
switch (this->UpdateType) {
|
|
case e_CVS:
|
|
vc = cm::make_unique<cmCTestCVS>(this->CTest, ofs);
|
|
break;
|
|
case e_SVN:
|
|
vc = cm::make_unique<cmCTestSVN>(this->CTest, ofs);
|
|
break;
|
|
case e_BZR:
|
|
vc = cm::make_unique<cmCTestBZR>(this->CTest, ofs);
|
|
break;
|
|
case e_GIT:
|
|
vc = cm::make_unique<cmCTestGIT>(this->CTest, ofs);
|
|
break;
|
|
case e_HG:
|
|
vc = cm::make_unique<cmCTestHG>(this->CTest, ofs);
|
|
break;
|
|
case e_P4:
|
|
vc = cm::make_unique<cmCTestP4>(this->CTest, ofs);
|
|
break;
|
|
default:
|
|
vc = cm::make_unique<cmCTestVC>(this->CTest, ofs);
|
|
break;
|
|
}
|
|
vc->SetCommandLineTool(this->UpdateCommand);
|
|
vc->SetSourceDirectory(sourceDirectory);
|
|
|
|
// Cleanup the working tree.
|
|
vc->Cleanup();
|
|
|
|
//
|
|
// Now update repository and remember what files were updated
|
|
//
|
|
cmGeneratedFileStream os;
|
|
if (!this->StartResultingXML(cmCTest::PartUpdate, "Update", os)) {
|
|
cmCTestLog(this->CTest, ERROR_MESSAGE,
|
|
"Cannot open log file" << std::endl);
|
|
return -1;
|
|
}
|
|
std::string start_time = this->CTest->CurrentTime();
|
|
auto start_time_time = std::chrono::system_clock::now();
|
|
auto elapsed_time_start = std::chrono::steady_clock::now();
|
|
|
|
bool updated = vc->Update();
|
|
std::string buildname =
|
|
cmCTest::SafeBuildIdField(this->CTest->GetCTestConfiguration("BuildName"));
|
|
|
|
cmXMLWriter xml(os);
|
|
xml.StartDocument();
|
|
xml.StartElement("Update");
|
|
xml.Attribute("mode", "Client");
|
|
xml.Attribute("Generator",
|
|
std::string("ctest-") + cmVersion::GetCMakeVersion());
|
|
xml.Element("Site", this->CTest->GetCTestConfiguration("Site"));
|
|
xml.Element("BuildName", buildname);
|
|
xml.Element("BuildStamp",
|
|
this->CTest->GetCurrentTag() + "-" +
|
|
this->CTest->GetTestModelString());
|
|
xml.Element("StartDateTime", start_time);
|
|
xml.Element("StartTime", start_time_time);
|
|
xml.Element("UpdateCommand", vc->GetUpdateCommandLine());
|
|
xml.Element("UpdateType",
|
|
cmCTestUpdateHandlerUpdateToString(this->UpdateType));
|
|
std::string changeId = this->CTest->GetCTestConfiguration("ChangeId");
|
|
if (!changeId.empty()) {
|
|
xml.Element("ChangeId", changeId);
|
|
}
|
|
|
|
bool loadedMods = vc->WriteXML(xml);
|
|
|
|
int localModifications = 0;
|
|
int numUpdated = vc->GetPathCount(cmCTestVC::PathUpdated);
|
|
if (numUpdated) {
|
|
cmCTestOptionalLog(this->CTest, HANDLER_OUTPUT,
|
|
" Found " << numUpdated << " updated files\n",
|
|
this->Quiet);
|
|
}
|
|
if (int numModified = vc->GetPathCount(cmCTestVC::PathModified)) {
|
|
cmCTestOptionalLog(
|
|
this->CTest, HANDLER_OUTPUT,
|
|
" Found " << numModified << " locally modified files\n", this->Quiet);
|
|
localModifications += numModified;
|
|
}
|
|
if (int numConflicting = vc->GetPathCount(cmCTestVC::PathConflicting)) {
|
|
cmCTestOptionalLog(this->CTest, HANDLER_OUTPUT,
|
|
" Found " << numConflicting << " conflicting files\n",
|
|
this->Quiet);
|
|
localModifications += numConflicting;
|
|
}
|
|
|
|
cmCTestOptionalLog(this->CTest, DEBUG, "End" << std::endl, this->Quiet);
|
|
std::string end_time = this->CTest->CurrentTime();
|
|
xml.Element("EndDateTime", end_time);
|
|
xml.Element("EndTime", std::chrono::system_clock::now());
|
|
xml.Element("ElapsedMinutes",
|
|
std::chrono::duration_cast<std::chrono::minutes>(
|
|
std::chrono::steady_clock::now() - elapsed_time_start)
|
|
.count());
|
|
|
|
xml.StartElement("UpdateReturnStatus");
|
|
if (localModifications) {
|
|
xml.Content("Update error: "
|
|
"There are modified or conflicting files in the repository");
|
|
cmCTestLog(this->CTest, WARNING,
|
|
" There are modified or conflicting files in the repository"
|
|
<< std::endl);
|
|
}
|
|
if (!updated) {
|
|
xml.Content("Update command failed:\n");
|
|
xml.Content(vc->GetUpdateCommandLine());
|
|
cmCTestLog(this->CTest, HANDLER_OUTPUT,
|
|
" Update command failed: " << vc->GetUpdateCommandLine()
|
|
<< "\n");
|
|
}
|
|
xml.EndElement(); // UpdateReturnStatus
|
|
xml.EndElement(); // Update
|
|
xml.EndDocument();
|
|
return updated && loadedMods ? numUpdated : -1;
|
|
}
|
|
|
|
int cmCTestUpdateHandler::DetectVCS(const char* dir)
|
|
{
|
|
std::string sourceDirectory = dir;
|
|
cmCTestOptionalLog(this->CTest, DEBUG,
|
|
"Check directory: " << sourceDirectory << std::endl,
|
|
this->Quiet);
|
|
sourceDirectory += "/.svn";
|
|
if (cmSystemTools::FileExists(sourceDirectory)) {
|
|
return cmCTestUpdateHandler::e_SVN;
|
|
}
|
|
sourceDirectory = cmStrCat(dir, "/CVS");
|
|
if (cmSystemTools::FileExists(sourceDirectory)) {
|
|
return cmCTestUpdateHandler::e_CVS;
|
|
}
|
|
sourceDirectory = cmStrCat(dir, "/.bzr");
|
|
if (cmSystemTools::FileExists(sourceDirectory)) {
|
|
return cmCTestUpdateHandler::e_BZR;
|
|
}
|
|
sourceDirectory = cmStrCat(dir, "/.git");
|
|
if (cmSystemTools::FileExists(sourceDirectory)) {
|
|
return cmCTestUpdateHandler::e_GIT;
|
|
}
|
|
sourceDirectory = cmStrCat(dir, "/.hg");
|
|
if (cmSystemTools::FileExists(sourceDirectory)) {
|
|
return cmCTestUpdateHandler::e_HG;
|
|
}
|
|
sourceDirectory = cmStrCat(dir, "/.p4");
|
|
if (cmSystemTools::FileExists(sourceDirectory)) {
|
|
return cmCTestUpdateHandler::e_P4;
|
|
}
|
|
sourceDirectory = cmStrCat(dir, "/.p4config");
|
|
if (cmSystemTools::FileExists(sourceDirectory)) {
|
|
return cmCTestUpdateHandler::e_P4;
|
|
}
|
|
return cmCTestUpdateHandler::e_UNKNOWN;
|
|
}
|
|
|
|
bool cmCTestUpdateHandler::SelectVCS()
|
|
{
|
|
// Get update command
|
|
this->UpdateCommand = this->CTest->GetCTestConfiguration("UpdateCommand");
|
|
|
|
// Detect the VCS managing the source tree.
|
|
this->UpdateType = this->DetectVCS(this->GetOption("SourceDirectory"));
|
|
if (this->UpdateType == e_UNKNOWN) {
|
|
// The source tree does not have a recognized VCS. Check the
|
|
// configuration value or command name.
|
|
this->UpdateType = this->DetermineType(
|
|
this->UpdateCommand.c_str(),
|
|
this->CTest->GetCTestConfiguration("UpdateType").c_str());
|
|
}
|
|
|
|
// If no update command was specified, lookup one for this VCS tool.
|
|
if (this->UpdateCommand.empty()) {
|
|
const char* key = nullptr;
|
|
switch (this->UpdateType) {
|
|
case e_CVS:
|
|
key = "CVSCommand";
|
|
break;
|
|
case e_SVN:
|
|
key = "SVNCommand";
|
|
break;
|
|
case e_BZR:
|
|
key = "BZRCommand";
|
|
break;
|
|
case e_GIT:
|
|
key = "GITCommand";
|
|
break;
|
|
case e_HG:
|
|
key = "HGCommand";
|
|
break;
|
|
case e_P4:
|
|
key = "P4Command";
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
if (key) {
|
|
this->UpdateCommand = this->CTest->GetCTestConfiguration(key);
|
|
}
|
|
if (this->UpdateCommand.empty()) {
|
|
std::ostringstream e;
|
|
e << "Cannot find UpdateCommand ";
|
|
if (key) {
|
|
e << "or " << key;
|
|
}
|
|
e << " configuration key.";
|
|
cmCTestLog(this->CTest, ERROR_MESSAGE, e.str() << std::endl);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|