|  | @@ -0,0 +1,416 @@
 | 
	
		
			
				|  |  | +/*=========================================================================
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  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);
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +}
 |