123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527 |
- /* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
- file Copyright.txt or https://cmake.org/licensing for details. */
- #include "cmCTestP4.h"
- #include "cmCTest.h"
- #include "cmCTestVC.h"
- #include "cmProcessTools.h"
- #include "cmSystemTools.h"
- #include "cmsys/RegularExpression.hxx"
- #include <algorithm>
- #include <ostream>
- #include <time.h>
- #include <utility>
- cmCTestP4::cmCTestP4(cmCTest* ct, std::ostream& log)
- : cmCTestGlobalVC(ct, log)
- {
- this->PriorRev = this->Unknown;
- }
- cmCTestP4::~cmCTestP4()
- {
- }
- class cmCTestP4::IdentifyParser : public cmCTestVC::LineParser
- {
- public:
- IdentifyParser(cmCTestP4* p4, const char* prefix, std::string& rev)
- : Rev(rev)
- {
- this->SetLog(&p4->Log, prefix);
- this->RegexIdentify.compile("^Change ([0-9]+) on");
- }
- private:
- std::string& Rev;
- cmsys::RegularExpression RegexIdentify;
- bool ProcessLine() override
- {
- if (this->RegexIdentify.find(this->Line)) {
- this->Rev = this->RegexIdentify.match(1);
- return false;
- }
- return true;
- }
- };
- class cmCTestP4::ChangesParser : public cmCTestVC::LineParser
- {
- public:
- ChangesParser(cmCTestP4* p4, const char* prefix)
- : P4(p4)
- {
- this->SetLog(&P4->Log, prefix);
- this->RegexIdentify.compile("^Change ([0-9]+) on");
- }
- private:
- cmsys::RegularExpression RegexIdentify;
- cmCTestP4* P4;
- bool ProcessLine() override
- {
- if (this->RegexIdentify.find(this->Line)) {
- P4->ChangeLists.push_back(this->RegexIdentify.match(1));
- }
- return true;
- }
- };
- class cmCTestP4::UserParser : public cmCTestVC::LineParser
- {
- public:
- UserParser(cmCTestP4* p4, const char* prefix)
- : P4(p4)
- {
- this->SetLog(&P4->Log, prefix);
- this->RegexUser.compile("^(.+) <(.*)> \\((.*)\\) accessed (.*)$");
- }
- private:
- cmsys::RegularExpression RegexUser;
- cmCTestP4* P4;
- bool ProcessLine() override
- {
- if (this->RegexUser.find(this->Line)) {
- User NewUser;
- NewUser.UserName = this->RegexUser.match(1);
- NewUser.EMail = this->RegexUser.match(2);
- NewUser.Name = this->RegexUser.match(3);
- NewUser.AccessTime = this->RegexUser.match(4);
- P4->Users[this->RegexUser.match(1)] = NewUser;
- return false;
- }
- return true;
- }
- };
- /* Diff format:
- ==== //depot/file#rev - /absolute/path/to/file ====
- (diff data)
- ==== //depot/file2#rev - /absolute/path/to/file2 ====
- (diff data)
- ==== //depot/file3#rev - /absolute/path/to/file3 ====
- ==== //depot/file4#rev - /absolute/path/to/file4 ====
- (diff data)
- */
- class cmCTestP4::DiffParser : public cmCTestVC::LineParser
- {
- public:
- DiffParser(cmCTestP4* p4, const char* prefix)
- : P4(p4)
- , AlreadyNotified(false)
- {
- this->SetLog(&P4->Log, prefix);
- this->RegexDiff.compile("^==== (.*)#[0-9]+ - (.*)");
- }
- private:
- cmCTestP4* P4;
- bool AlreadyNotified;
- std::string CurrentPath;
- cmsys::RegularExpression RegexDiff;
- bool ProcessLine() override
- {
- if (!this->Line.empty() && this->Line[0] == '=' &&
- this->RegexDiff.find(this->Line)) {
- CurrentPath = this->RegexDiff.match(1);
- AlreadyNotified = false;
- } else {
- if (!AlreadyNotified) {
- P4->DoModification(PathModified, CurrentPath);
- AlreadyNotified = true;
- }
- }
- return true;
- }
- };
- cmCTestP4::User cmCTestP4::GetUserData(const std::string& username)
- {
- std::map<std::string, cmCTestP4::User>::const_iterator it =
- Users.find(username);
- if (it == Users.end()) {
- std::vector<char const*> p4_users;
- SetP4Options(p4_users);
- p4_users.push_back("users");
- p4_users.push_back("-m");
- p4_users.push_back("1");
- p4_users.push_back(username.c_str());
- p4_users.push_back(nullptr);
- UserParser out(this, "users-out> ");
- OutputLogger err(this->Log, "users-err> ");
- RunChild(&p4_users[0], &out, &err);
- // The user should now be added to the map. Search again.
- it = Users.find(username);
- if (it == Users.end()) {
- return cmCTestP4::User();
- }
- }
- return it->second;
- }
- /* Commit format:
- Change 1111111 by user@client on 2013/09/26 11:50:36
- text
- text
- Affected files ...
- ... //path/to/file#rev edit
- ... //path/to/file#rev add
- ... //path/to/file#rev delete
- ... //path/to/file#rev integrate
- */
- class cmCTestP4::DescribeParser : public cmCTestVC::LineParser
- {
- public:
- DescribeParser(cmCTestP4* p4, const char* prefix)
- : LineParser('\n', false)
- , P4(p4)
- , Section(SectionHeader)
- {
- this->SetLog(&P4->Log, prefix);
- this->RegexHeader.compile("^Change ([0-9]+) by (.+)@(.+) on (.*)$");
- this->RegexDiff.compile("^\\.\\.\\. (.*)#[0-9]+ ([^ ]+)$");
- }
- private:
- cmsys::RegularExpression RegexHeader;
- cmsys::RegularExpression RegexDiff;
- cmCTestP4* P4;
- typedef cmCTestP4::Revision Revision;
- typedef cmCTestP4::Change Change;
- std::vector<Change> Changes;
- enum SectionType
- {
- SectionHeader,
- SectionBody,
- SectionDiffHeader,
- SectionDiff,
- SectionCount
- };
- SectionType Section;
- Revision Rev;
- bool ProcessLine() override
- {
- if (this->Line.empty()) {
- this->NextSection();
- } else {
- switch (this->Section) {
- case SectionHeader:
- this->DoHeaderLine();
- break;
- case SectionBody:
- this->DoBodyLine();
- break;
- case SectionDiffHeader:
- break; // nothing to do
- case SectionDiff:
- this->DoDiffLine();
- break;
- case SectionCount:
- break; // never happens
- }
- }
- return true;
- }
- void NextSection()
- {
- if (this->Section == SectionDiff) {
- this->P4->DoRevision(this->Rev, this->Changes);
- this->Rev = Revision();
- }
- this->Section = SectionType((this->Section + 1) % SectionCount);
- }
- void DoHeaderLine()
- {
- if (this->RegexHeader.find(this->Line)) {
- this->Rev.Rev = this->RegexHeader.match(1);
- this->Rev.Date = this->RegexHeader.match(4);
- cmCTestP4::User user = P4->GetUserData(this->RegexHeader.match(2));
- this->Rev.Author = user.Name;
- this->Rev.EMail = user.EMail;
- this->Rev.Committer = this->Rev.Author;
- this->Rev.CommitterEMail = this->Rev.EMail;
- this->Rev.CommitDate = this->Rev.Date;
- }
- }
- void DoBodyLine()
- {
- if (this->Line[0] == '\t') {
- this->Rev.Log += this->Line.substr(1);
- }
- this->Rev.Log += "\n";
- }
- void DoDiffLine()
- {
- if (this->RegexDiff.find(this->Line)) {
- Change change;
- std::string Path = this->RegexDiff.match(1);
- if (Path.length() > 2 && Path[0] == '/' && Path[1] == '/') {
- size_t found = Path.find('/', 2);
- if (found != std::string::npos) {
- Path = Path.substr(found + 1);
- }
- }
- change.Path = Path;
- std::string action = this->RegexDiff.match(2);
- if (action == "add") {
- change.Action = 'A';
- } else if (action == "delete") {
- change.Action = 'D';
- } else if (action == "edit" || action == "integrate") {
- change.Action = 'M';
- }
- Changes.push_back(change);
- }
- }
- };
- void cmCTestP4::SetP4Options(std::vector<char const*>& CommandOptions)
- {
- if (P4Options.empty()) {
- const char* p4 = this->CommandLineTool.c_str();
- P4Options.push_back(p4);
- // The CTEST_P4_CLIENT variable sets the P4 client used when issuing
- // Perforce commands, if it's different from the default one.
- std::string client = this->CTest->GetCTestConfiguration("P4Client");
- if (!client.empty()) {
- P4Options.push_back("-c");
- P4Options.push_back(client);
- }
- // Set the message language to be English, in case the P4 admin
- // has localized them
- P4Options.push_back("-L");
- P4Options.push_back("en");
- // The CTEST_P4_OPTIONS variable adds additional Perforce command line
- // options before the main command
- std::string opts = this->CTest->GetCTestConfiguration("P4Options");
- std::vector<std::string> args =
- cmSystemTools::ParseArguments(opts.c_str());
- P4Options.insert(P4Options.end(), args.begin(), args.end());
- }
- CommandOptions.clear();
- for (std::string const& o : P4Options) {
- CommandOptions.push_back(o.c_str());
- }
- }
- std::string cmCTestP4::GetWorkingRevision()
- {
- std::vector<char const*> p4_identify;
- SetP4Options(p4_identify);
- p4_identify.push_back("changes");
- p4_identify.push_back("-m");
- p4_identify.push_back("1");
- p4_identify.push_back("-t");
- std::string source = this->SourceDirectory + "/...#have";
- p4_identify.push_back(source.c_str());
- p4_identify.push_back(nullptr);
- std::string rev;
- IdentifyParser out(this, "p4_changes-out> ", rev);
- OutputLogger err(this->Log, "p4_changes-err> ");
- bool result = RunChild(&p4_identify[0], &out, &err);
- // If there was a problem contacting the server return "<unknown>"
- if (!result) {
- return "<unknown>";
- }
- if (rev.empty()) {
- return "0";
- }
- return rev;
- }
- bool cmCTestP4::NoteOldRevision()
- {
- this->OldRevision = this->GetWorkingRevision();
- cmCTestLog(this->CTest, HANDLER_OUTPUT, " Old revision of repository is: "
- << this->OldRevision << "\n");
- this->PriorRev.Rev = this->OldRevision;
- return true;
- }
- bool cmCTestP4::NoteNewRevision()
- {
- this->NewRevision = this->GetWorkingRevision();
- cmCTestLog(this->CTest, HANDLER_OUTPUT, " New revision of repository is: "
- << this->NewRevision << "\n");
- return true;
- }
- bool cmCTestP4::LoadRevisions()
- {
- std::vector<char const*> p4_changes;
- SetP4Options(p4_changes);
- // Use 'p4 changes ...@old,new' to get a list of changelists
- std::string range = this->SourceDirectory + "/...";
- // If any revision is unknown it means we couldn't contact the server.
- // Do not process updates
- if (this->OldRevision == "<unknown>" || this->NewRevision == "<unknown>") {
- cmCTestLog(this->CTest, HANDLER_OUTPUT, " At least one of the revisions "
- << "is unknown. No repository changes will be reported.\n");
- return false;
- }
- range.append("@")
- .append(this->OldRevision)
- .append(",")
- .append(this->NewRevision);
- p4_changes.push_back("changes");
- p4_changes.push_back(range.c_str());
- p4_changes.push_back(nullptr);
- ChangesParser out(this, "p4_changes-out> ");
- OutputLogger err(this->Log, "p4_changes-err> ");
- ChangeLists.clear();
- this->RunChild(&p4_changes[0], &out, &err);
- if (ChangeLists.empty()) {
- return true;
- }
- // p4 describe -s ...@1111111,2222222
- std::vector<char const*> p4_describe;
- for (std::vector<std::string>::reverse_iterator i = ChangeLists.rbegin();
- i != ChangeLists.rend(); ++i) {
- SetP4Options(p4_describe);
- p4_describe.push_back("describe");
- p4_describe.push_back("-s");
- p4_describe.push_back(i->c_str());
- p4_describe.push_back(nullptr);
- DescribeParser outDescribe(this, "p4_describe-out> ");
- OutputLogger errDescribe(this->Log, "p4_describe-err> ");
- this->RunChild(&p4_describe[0], &outDescribe, &errDescribe);
- }
- return true;
- }
- bool cmCTestP4::LoadModifications()
- {
- std::vector<char const*> p4_diff;
- SetP4Options(p4_diff);
- p4_diff.push_back("diff");
- // Ideally we would use -Od but not all clients support it
- p4_diff.push_back("-dn");
- std::string source = this->SourceDirectory + "/...";
- p4_diff.push_back(source.c_str());
- p4_diff.push_back(nullptr);
- DiffParser out(this, "p4_diff-out> ");
- OutputLogger err(this->Log, "p4_diff-err> ");
- this->RunChild(&p4_diff[0], &out, &err);
- return true;
- }
- bool cmCTestP4::UpdateCustom(const std::string& custom)
- {
- std::vector<std::string> p4_custom_command;
- cmSystemTools::ExpandListArgument(custom, p4_custom_command, true);
- std::vector<char const*> p4_custom;
- p4_custom.reserve(p4_custom_command.size() + 1);
- for (std::string const& i : p4_custom_command) {
- p4_custom.push_back(i.c_str());
- }
- p4_custom.push_back(nullptr);
- OutputLogger custom_out(this->Log, "p4_customsync-out> ");
- OutputLogger custom_err(this->Log, "p4_customsync-err> ");
- return this->RunUpdateCommand(&p4_custom[0], &custom_out, &custom_err);
- }
- bool cmCTestP4::UpdateImpl()
- {
- std::string custom = this->CTest->GetCTestConfiguration("P4UpdateCustom");
- if (!custom.empty()) {
- return this->UpdateCustom(custom);
- }
- // If we couldn't get a revision number before updating, abort.
- if (this->OldRevision == "<unknown>") {
- this->UpdateCommandLine = "Unknown current revision";
- cmCTestLog(this->CTest, ERROR_MESSAGE, " Unknown current revision\n");
- return false;
- }
- std::vector<char const*> p4_sync;
- SetP4Options(p4_sync);
- p4_sync.push_back("sync");
- // Get user-specified update options.
- std::string opts = this->CTest->GetCTestConfiguration("UpdateOptions");
- if (opts.empty()) {
- opts = this->CTest->GetCTestConfiguration("P4UpdateOptions");
- }
- std::vector<std::string> args = cmSystemTools::ParseArguments(opts.c_str());
- for (std::string const& arg : args) {
- p4_sync.push_back(arg.c_str());
- }
- std::string source = this->SourceDirectory + "/...";
- // Specify the start time for nightly testing.
- if (this->CTest->GetTestModel() == cmCTest::NIGHTLY) {
- std::string date = this->GetNightlyTime();
- // CTest reports the date as YYYY-MM-DD, Perforce needs it as YYYY/MM/DD
- std::replace(date.begin(), date.end(), '-', '/');
- // Revision specification: /...@"YYYY/MM/DD HH:MM:SS"
- source.append("@\"").append(date).append("\"");
- }
- p4_sync.push_back(source.c_str());
- p4_sync.push_back(nullptr);
- OutputLogger out(this->Log, "p4_sync-out> ");
- OutputLogger err(this->Log, "p4_sync-err> ");
- return this->RunUpdateCommand(&p4_sync[0], &out, &err);
- }
|