| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530 | /* 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        textAffected 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);}
 |