| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416 |
- /*=========================================================================
- 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.
- =========================================================================*/
- #include "cmCTestGIT.h"
- #include "cmCTest.h"
- #include "cmSystemTools.h"
- #include "cmXMLSafe.h"
- #include <cmsys/RegularExpression.hxx>
- #include <cmsys/ios/sstream>
- #include <cmsys/Process.h>
- #include <ctype.h>
- //----------------------------------------------------------------------------
- cmCTestGIT::cmCTestGIT(cmCTest* ct, std::ostream& log):
- cmCTestGlobalVC(ct, log)
- {
- this->PriorRev = this->Unknown;
- }
- //----------------------------------------------------------------------------
- cmCTestGIT::~cmCTestGIT()
- {
- }
- //----------------------------------------------------------------------------
- class cmCTestGIT::OneLineParser: public cmCTestVC::LineParser
- {
- public:
- OneLineParser(cmCTestGIT* git, const char* prefix,
- std::string& l): Line1(l)
- {
- this->SetLog(&git->Log, prefix);
- }
- private:
- std::string& Line1;
- virtual bool ProcessLine()
- {
- // Only the first line is of interest.
- this->Line1 = this->Line;
- return false;
- }
- };
- //----------------------------------------------------------------------------
- std::string cmCTestGIT::GetWorkingRevision()
- {
- // Run plumbing "git rev-list" to get work tree revision.
- const char* git = this->CommandLineTool.c_str();
- const char* git_rev_list[] = {git, "rev-list", "-n", "1", "HEAD", 0};
- std::string rev;
- OneLineParser out(this, "rl-out> ", rev);
- OutputLogger err(this->Log, "rl-err> ");
- this->RunChild(git_rev_list, &out, &err);
- return rev;
- }
- //----------------------------------------------------------------------------
- void cmCTestGIT::NoteOldRevision()
- {
- this->OldRevision = this->GetWorkingRevision();
- cmCTestLog(this->CTest, HANDLER_OUTPUT, " Old revision of repository is: "
- << this->OldRevision << "\n");
- this->PriorRev.Rev = this->OldRevision;
- }
- //----------------------------------------------------------------------------
- void cmCTestGIT::NoteNewRevision()
- {
- this->NewRevision = this->GetWorkingRevision();
- cmCTestLog(this->CTest, HANDLER_OUTPUT, " New revision of repository is: "
- << this->NewRevision << "\n");
- }
- //----------------------------------------------------------------------------
- bool cmCTestGIT::UpdateImpl()
- {
- // Use "git pull" to update the working tree.
- std::vector<char const*> git_pull;
- git_pull.push_back(this->CommandLineTool.c_str());
- git_pull.push_back("pull");
- // TODO: if(this->CTest->GetTestModel() == cmCTest::NIGHTLY)
- // Add user-specified update options.
- std::string opts = this->CTest->GetCTestConfiguration("UpdateOptions");
- if(opts.empty())
- {
- opts = this->CTest->GetCTestConfiguration("GITUpdateOptions");
- }
- std::vector<cmStdString> args = cmSystemTools::ParseArguments(opts.c_str());
- for(std::vector<cmStdString>::const_iterator ai = args.begin();
- ai != args.end(); ++ai)
- {
- git_pull.push_back(ai->c_str());
- }
- // Sentinel argument.
- git_pull.push_back(0);
- OutputLogger out(this->Log, "pull-out> ");
- OutputLogger err(this->Log, "pull-err> ");
- return this->RunUpdateCommand(&git_pull[0], &out, &err);
- }
- //----------------------------------------------------------------------------
- /* Diff format:
- :src-mode dst-mode src-sha1 dst-sha1 status\0
- src-path\0
- [dst-path\0]
- The format is repeated for every file changed. The [dst-path\0]
- line appears only for lines with status 'C' or 'R'. See 'git help
- diff-tree' for details.
- */
- class cmCTestGIT::DiffParser: public cmCTestVC::LineParser
- {
- public:
- DiffParser(cmCTestGIT* git, const char* prefix):
- LineParser('\0', false), GIT(git), DiffField(DiffFieldNone)
- {
- this->SetLog(&git->Log, prefix);
- }
- typedef cmCTestGIT::Change Change;
- std::vector<Change> Changes;
- protected:
- cmCTestGIT* GIT;
- enum DiffFieldType { DiffFieldNone, DiffFieldChange,
- DiffFieldSrc, DiffFieldDst };
- DiffFieldType DiffField;
- Change CurChange;
- void DiffReset()
- {
- this->DiffField = DiffFieldNone;
- this->Changes.clear();
- }
- virtual bool ProcessLine()
- {
- if(this->Line[0] == ':')
- {
- this->DiffField = DiffFieldChange;
- this->CurChange = Change();
- }
- if(this->DiffField == DiffFieldChange)
- {
- // :src-mode dst-mode src-sha1 dst-sha1 status
- if(this->Line[0] != ':')
- {
- this->DiffField = DiffFieldNone;
- return true;
- }
- const char* src_mode_first = this->Line.c_str()+1;
- const char* src_mode_last = this->ConsumeField(src_mode_first);
- const char* dst_mode_first = this->ConsumeSpace(src_mode_last);
- const char* dst_mode_last = this->ConsumeField(dst_mode_first);
- const char* src_sha1_first = this->ConsumeSpace(dst_mode_last);
- const char* src_sha1_last = this->ConsumeField(src_sha1_first);
- const char* dst_sha1_first = this->ConsumeSpace(src_sha1_last);
- const char* dst_sha1_last = this->ConsumeField(dst_sha1_first);
- const char* status_first = this->ConsumeSpace(dst_sha1_last);
- const char* status_last = this->ConsumeField(status_first);
- if(status_first != status_last)
- {
- this->CurChange.Action = *status_first;
- this->DiffField = DiffFieldSrc;
- }
- else
- {
- this->DiffField = DiffFieldNone;
- }
- }
- else if(this->DiffField == DiffFieldSrc)
- {
- // src-path
- if(this->CurChange.Action == 'C')
- {
- // Convert copy to addition of destination.
- this->CurChange.Action = 'A';
- this->DiffField = DiffFieldDst;
- }
- else if(this->CurChange.Action == 'R')
- {
- // Convert rename to deletion of source and addition of destination.
- this->CurChange.Action = 'D';
- this->CurChange.Path = this->Line;
- this->Changes.push_back(this->CurChange);
- this->CurChange = Change('A');
- this->DiffField = DiffFieldDst;
- }
- else
- {
- this->CurChange.Path = this->Line;
- this->Changes.push_back(this->CurChange);
- this->DiffField = this->DiffFieldNone;
- }
- }
- else if(this->DiffField == DiffFieldDst)
- {
- // dst-path
- this->CurChange.Path = this->Line;
- this->Changes.push_back(this->CurChange);
- this->DiffField = this->DiffFieldNone;
- }
- return true;
- }
- const char* ConsumeSpace(const char* c)
- {
- while(*c && isspace(*c)) { ++c; }
- return c;
- }
- const char* ConsumeField(const char* c)
- {
- while(*c && !isspace(*c)) { ++c; }
- return c;
- }
- };
- //----------------------------------------------------------------------------
- /* Commit format:
- commit ...\n
- tree ...\n
- parent ...\n
- author ...\n
- committer ...\n
- \n
- Log message indented by (4) spaces\n
- (even blank lines have the spaces)\n
- \n
- [Diff format]
- The header may have more fields. See 'git help diff-tree'.
- */
- class cmCTestGIT::CommitParser: public DiffParser
- {
- public:
- CommitParser(cmCTestGIT* git, const char* prefix):
- DiffParser(git, prefix), Section(SectionHeader)
- {
- this->Separator = SectionSep[this->Section];
- }
- private:
- typedef cmCTestGIT::Revision Revision;
- enum SectionType { SectionHeader, SectionBody, SectionDiff, SectionCount };
- static char const SectionSep[SectionCount];
- SectionType Section;
- Revision Rev;
- struct Person
- {
- std::string Name;
- std::string EMail;
- unsigned long Time;
- long TimeZone;
- Person(): Name(), EMail(), Time(0), TimeZone(0) {}
- };
- void ParsePerson(const char* str, Person& person)
- {
- // Person Name <[email protected]> 1234567890 +0000
- const char* c = str;
- while(*c && isspace(*c)) { ++c; }
- const char* name_first = c;
- while(*c && *c != '<') { ++c; }
- const char* name_last = c;
- while(name_last != name_first && isspace(*(name_last-1))) { --name_last; }
- person.Name.assign(name_first, name_last-name_first);
- const char* email_first = *c? ++c : c;
- while(*c && *c != '>') { ++c; }
- const char* email_last = *c? c++ : c;
- person.EMail.assign(email_first, email_last-email_first);
- person.Time = strtoul(c, (char**)&c, 10);
- person.TimeZone = strtol(c, (char**)&c, 10);
- }
- virtual bool ProcessLine()
- {
- if(this->Line.empty())
- {
- this->NextSection();
- }
- else
- {
- switch(this->Section)
- {
- case SectionHeader: this->DoHeaderLine(); break;
- case SectionBody: this->DoBodyLine(); break;
- case SectionDiff: this->DiffParser::ProcessLine(); break;
- case SectionCount: break; // never happens
- }
- }
- return true;
- }
- void NextSection()
- {
- this->Section = SectionType((this->Section+1) % SectionCount);
- this->Separator = SectionSep[this->Section];
- if(this->Section == SectionHeader)
- {
- this->GIT->DoRevision(this->Rev, this->Changes);
- this->Rev = Revision();
- this->DiffReset();
- }
- }
- void DoHeaderLine()
- {
- // Look for header fields that we need.
- if(strncmp(this->Line.c_str(), "commit ", 7) == 0)
- {
- this->Rev.Rev = this->Line.c_str()+7;
- }
- else if(strncmp(this->Line.c_str(), "author ", 7) == 0)
- {
- Person author;
- this->ParsePerson(this->Line.c_str()+7, author);
- this->Rev.Author = author.Name;
- char buf[1024];
- if(author.TimeZone >= 0)
- {
- sprintf(buf, "%lu +%04ld", author.Time, author.TimeZone);
- }
- else
- {
- sprintf(buf, "%lu -%04ld", author.Time, -author.TimeZone);
- }
- this->Rev.Date = buf;
- }
- }
- void DoBodyLine()
- {
- // Commit log lines are indented by 4 spaces.
- if(this->Line.size() >= 4)
- {
- this->Rev.Log += this->Line.substr(4);
- }
- this->Rev.Log += "\n";
- }
- };
- char const cmCTestGIT::CommitParser::SectionSep[SectionCount] =
- {'\n', '\n', '\0'};
- //----------------------------------------------------------------------------
- void cmCTestGIT::LoadRevisions()
- {
- // Use 'git rev-list ... | git diff-tree ...' to get revisions.
- std::string range = this->OldRevision + ".." + this->NewRevision;
- const char* git = this->CommandLineTool.c_str();
- const char* git_rev_list[] =
- {git, "rev-list", "--reverse", range.c_str(), "--", 0};
- const char* git_diff_tree[] =
- {git, "diff-tree", "--stdin", "--always", "-z", "-r", "--pretty=raw",
- "--encoding=utf-8", 0};
- this->Log << this->ComputeCommandLine(git_rev_list) << " | "
- << this->ComputeCommandLine(git_diff_tree) << "\n";
- cmsysProcess* cp = cmsysProcess_New();
- cmsysProcess_AddCommand(cp, git_rev_list);
- cmsysProcess_AddCommand(cp, git_diff_tree);
- cmsysProcess_SetWorkingDirectory(cp, this->SourceDirectory.c_str());
- CommitParser out(this, "dt-out> ");
- OutputLogger err(this->Log, "dt-err> ");
- this->RunProcess(cp, &out, &err);
- // Send one extra zero-byte to terminate the last record.
- out.Process("", 1);
- cmsysProcess_Delete(cp);
- }
- //----------------------------------------------------------------------------
- void cmCTestGIT::LoadModifications()
- {
- // Use 'git diff-index' to get modified files.
- const char* git = this->CommandLineTool.c_str();
- const char* git_diff_index[] = {git, "diff-index", "-z", "HEAD", 0};
- DiffParser out(this, "di-out> ");
- OutputLogger err(this->Log, "di-err> ");
- this->RunChild(git_diff_index, &out, &err);
- for(std::vector<Change>::const_iterator ci = out.Changes.begin();
- ci != out.Changes.end(); ++ci)
- {
- this->DoModification(PathModified, ci->Path);
- }
- }
|