2004-09-07 17:17:15 +04:00
/*=========================================================================
Program : CMake - Cross - Platform Makefile Generator
Module : $ RCSfile $
Language : C + +
Date : $ Date $
Version : $ Revision $
Copyright ( c ) 2002 Kitware , Inc . , Insight Consortium . 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 "cmCTestUpdateHandler.h"
# include "cmCTest.h"
# include "cmake.h"
# include "cmMakefile.h"
# include "cmLocalGenerator.h"
# include "cmGlobalGenerator.h"
2004-10-22 23:44:54 +04:00
# include "cmVersion.h"
2005-01-27 18:15:01 +03:00
# include "cmGeneratedFileStream.h"
2005-02-16 21:28:47 +03:00
# include "cmXMLParser.h"
2004-09-07 17:17:15 +04:00
//#include <cmsys/RegularExpression.hxx>
# include <cmsys/Process.h>
// used for sleep
# ifdef _WIN32
# include "windows.h"
# endif
# include <stdlib.h>
# include <math.h>
# include <float.h>
2005-06-17 21:04:56 +04:00
//----------------------------------------------------------------------
static const char * cmCTestUpdateHandlerUpdateStrings [ ] =
{
" Unknown " ,
" CVS " ,
" SVN "
} ;
static const char * cmCTestUpdateHandlerUpdateToString ( int type )
{
if ( type < cmCTestUpdateHandler : : e_UNKNOWN | | type > = cmCTestUpdateHandler : : e_LAST )
{
return cmCTestUpdateHandlerUpdateStrings [ cmCTestUpdateHandler : : e_UNKNOWN ] ;
}
return cmCTestUpdateHandlerUpdateStrings [ type ] ;
}
2005-02-16 21:28:47 +03:00
//----------------------------------------------------------------------
//**********************************************************************
class cmCTestUpdateHandlerSVNXMLParser : public cmXMLParser
{
public :
struct t_CommitLog
{
int m_Revision ;
std : : string m_Author ;
std : : string m_Date ;
std : : string m_Message ;
} ;
cmCTestUpdateHandlerSVNXMLParser ( cmCTestUpdateHandler * up )
2005-06-01 01:32:40 +04:00
: cmXMLParser ( ) , m_UpdateHandler ( up ) , m_MinRevision ( - 1 ) , m_MaxRevision ( - 1 )
2005-02-16 21:28:47 +03:00
{
}
int Parse ( const char * str )
{
m_MinRevision = - 1 ;
m_MaxRevision = - 1 ;
int res = this - > cmXMLParser : : Parse ( str ) ;
if ( m_MinRevision = = - 1 | | m_MaxRevision = = - 1 )
{
return 0 ;
}
return res ;
}
typedef std : : vector < t_CommitLog > t_VectorOfCommits ;
t_VectorOfCommits * GetCommits ( ) { return & m_Commits ; }
int GetMinRevision ( ) { return m_MinRevision ; }
int GetMaxRevision ( ) { return m_MaxRevision ; }
protected :
void StartElement ( const char * name , const char * * atts )
{
if ( strcmp ( name , " logentry " ) = = 0 )
{
m_CommitLog = t_CommitLog ( ) ;
const char * rev = this - > FindAttribute ( atts , " revision " ) ;
if ( rev )
{
m_CommitLog . m_Revision = atoi ( rev ) ;
if ( m_MinRevision < 0 | | m_MinRevision > m_CommitLog . m_Revision )
{
m_MinRevision = m_CommitLog . m_Revision ;
}
if ( m_MaxRevision < 0 | | m_MaxRevision < m_CommitLog . m_Revision )
{
m_MaxRevision = m_CommitLog . m_Revision ;
}
}
}
m_CharacterData . erase ( m_CharacterData . begin ( ) , m_CharacterData . end ( ) ) ;
}
void EndElement ( const char * name )
{
if ( strcmp ( name , " logentry " ) = = 0 )
{
2005-06-01 01:32:40 +04:00
cmCTestLog ( m_UpdateHandler - > GetCTestInstance ( ) , HANDLER_VERBOSE_OUTPUT , " \t Revision: " < < m_CommitLog . m_Revision < < std : : endl
< < " \t Author: " < < m_CommitLog . m_Author . c_str ( ) < < std : : endl
< < " \t Date: " < < m_CommitLog . m_Date . c_str ( ) < < std : : endl
< < " \t Message: " < < m_CommitLog . m_Message . c_str ( ) < < std : : endl ) ;
2005-02-16 21:28:47 +03:00
m_Commits . push_back ( m_CommitLog ) ;
}
else if ( strcmp ( name , " author " ) = = 0 )
{
m_CommitLog . m_Author . assign ( & ( * ( m_CharacterData . begin ( ) ) ) , m_CharacterData . size ( ) ) ;
}
else if ( strcmp ( name , " date " ) = = 0 )
{
m_CommitLog . m_Date . assign ( & ( * ( m_CharacterData . begin ( ) ) ) , m_CharacterData . size ( ) ) ;
}
else if ( strcmp ( name , " msg " ) = = 0 )
{
m_CommitLog . m_Message . assign ( & ( * ( m_CharacterData . begin ( ) ) ) , m_CharacterData . size ( ) ) ;
}
m_CharacterData . erase ( m_CharacterData . begin ( ) , m_CharacterData . end ( ) ) ;
}
void CharacterDataHandler ( const char * data , int length )
{
m_CharacterData . insert ( m_CharacterData . end ( ) , data , data + length ) ;
}
const char * FindAttribute ( const char * * atts , const char * attribute )
{
if ( ! atts | | ! attribute )
{
return 0 ;
}
const char * * atr = atts ;
while ( * atr & & * * atr & & * * ( atr + 1 ) )
{
if ( strcmp ( * atr , attribute ) = = 0 )
{
return * ( atr + 1 ) ;
}
atr + = 2 ;
}
return 0 ;
}
private :
std : : vector < char > m_CharacterData ;
cmCTestUpdateHandler * m_UpdateHandler ;
t_CommitLog m_CommitLog ;
t_VectorOfCommits m_Commits ;
int m_MinRevision ;
int m_MaxRevision ;
} ;
//**********************************************************************
//----------------------------------------------------------------------
2004-09-07 17:17:15 +04:00
//----------------------------------------------------------------------
cmCTestUpdateHandler : : cmCTestUpdateHandler ( )
{
2005-06-17 21:04:56 +04:00
}
//----------------------------------------------------------------------
void cmCTestUpdateHandler : : Initialize ( )
{
2005-06-23 21:04:18 +04:00
this - > Superclass : : Initialize ( ) ;
2004-09-07 17:17:15 +04:00
}
2005-02-16 21:28:47 +03:00
//----------------------------------------------------------------------
int cmCTestUpdateHandler : : DetermineType ( const char * cmd , const char * type )
{
2005-06-17 21:04:56 +04:00
cmCTestLog ( m_CTest , DEBUG , " Determine update type from command: " < < cmd < < " and type: " < < type < < std : : endl ) ;
2005-02-16 21:28:47 +03:00
if ( type & & * type )
{
2005-06-17 21:04:56 +04:00
cmCTestLog ( m_CTest , DEBUG , " Type specified: " < < type < < std : : endl ) ;
2005-02-16 21:28:47 +03:00
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 ;
}
}
else
{
2005-06-17 21:04:56 +04:00
cmCTestLog ( m_CTest , DEBUG , " Type not specified, check command: " < < cmd < < std : : endl ) ;
2005-02-16 21:28:47 +03:00
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 ;
}
}
2005-06-17 21:04:56 +04:00
std : : string sourceDirectory = this - > GetOption ( " SourceDirectory " ) ;
cmCTestLog ( m_CTest , DEBUG , " Check directory: " < < sourceDirectory . c_str ( ) < < std : : endl ) ;
sourceDirectory + = " /.svn " ;
if ( cmSystemTools : : FileExists ( sourceDirectory . c_str ( ) ) )
{
return cmCTestUpdateHandler : : e_SVN ;
}
sourceDirectory = this - > GetOption ( " SourceDirectory " ) ;
sourceDirectory + = " /CVS " ;
if ( cmSystemTools : : FileExists ( sourceDirectory . c_str ( ) ) )
{
return cmCTestUpdateHandler : : e_CVS ;
}
return cmCTestUpdateHandler : : e_UNKNOWN ;
2005-02-16 21:28:47 +03:00
}
2004-09-07 17:17:15 +04:00
//----------------------------------------------------------------------
//clearly it would be nice if this were broken up into a few smaller
//functions and commented...
2005-01-27 23:54:47 +03:00
int cmCTestUpdateHandler : : ProcessHandler ( )
2004-09-07 17:17:15 +04:00
{
int count = 0 ;
2005-02-16 21:28:47 +03:00
int updateType = e_CVS ;
2004-09-07 17:17:15 +04:00
std : : string : : size_type cc , kk ;
2005-04-24 20:32:56 +04:00
bool updateProducedError = false ;
2005-06-16 21:18:21 +04:00
std : : string goutput ;
std : : string errors ;
2005-02-16 21:28:47 +03:00
2005-06-16 21:18:21 +04:00
std : : string checkoutErrorMessages ;
int retVal = 0 ;
2005-02-16 21:28:47 +03:00
// Get source dir
2005-02-17 23:23:00 +03:00
const char * sourceDirectory = this - > GetOption ( " SourceDirectory " ) ;
if ( ! sourceDirectory )
2004-09-07 17:17:15 +04:00
{
2005-06-01 02:40:43 +04:00
cmCTestLog ( m_CTest , ERROR_MESSAGE , " Cannot find SourceDirectory key in the DartConfiguration.tcl " < < std : : endl ) ;
2004-09-07 17:17:15 +04:00
return - 1 ;
}
2005-02-16 21:28:47 +03:00
2005-06-16 21:18:21 +04:00
cmGeneratedFileStream ofs ;
if ( ! m_CTest - > GetShowOnly ( ) )
{
2005-06-23 21:04:18 +04:00
this - > StartLogFile ( " Update " , ofs ) ;
2005-06-16 21:18:21 +04:00
}
cmCTestLog ( m_CTest , HANDLER_OUTPUT , " Updating the repository " < < std : : endl ) ;
const char * initialCheckoutCommand = this - > GetOption ( " InitialCheckout " ) ;
if ( initialCheckoutCommand )
{
cmCTestLog ( m_CTest , HANDLER_OUTPUT , " First perform the initil checkout: " < < initialCheckoutCommand < < std : : endl ) ;
cmStdString parent = cmSystemTools : : GetParentDirectory ( sourceDirectory ) ;
if ( parent . empty ( ) )
{
cmCTestLog ( m_CTest , ERROR_MESSAGE ,
" Something went wrong when trying to determine the parent directory of " < < sourceDirectory < < std : : endl ) ;
return - 1 ;
}
cmCTestLog ( m_CTest , HANDLER_OUTPUT , " Perform checkout in directory: " < < parent . c_str ( ) < < std : : endl ) ;
if ( ! cmSystemTools : : MakeDirectory ( parent . c_str ( ) ) )
{
cmCTestLog ( m_CTest , ERROR_MESSAGE ,
" Cannot create parent directory: " < < parent . c_str ( ) < < " of the source directory: " < < sourceDirectory < < std : : endl ) ;
return - 1 ;
}
ofs < < " * Run initial checkout " < < std : : endl ;
ofs < < " Command: " < < initialCheckoutCommand < < std : : endl ;
cmCTestLog ( m_CTest , DEBUG , " Before: " < < initialCheckoutCommand < < std : : endl ) ;
bool retic = m_CTest - > RunCommand ( initialCheckoutCommand , & goutput , & errors , & retVal , parent . c_str ( ) , 0 /* Timeout */ ) ;
cmCTestLog ( m_CTest , DEBUG , " After: " < < initialCheckoutCommand < < std : : endl ) ;
ofs < < " Output: " < < goutput . c_str ( ) < < std : : endl ;
ofs < < " Errors: " < < errors . c_str ( ) < < std : : endl ;
if ( ! retic | | retVal )
{
cmOStringStream ostr ;
ostr < < " Problem running initial checkout Output [ " < < goutput < < " ] Errors [ " < < errors < < " ] " ;
cmCTestLog ( m_CTest , HANDLER_OUTPUT , ostr . str ( ) . c_str ( ) < < std : : endl ) ;
checkoutErrorMessages + = ostr . str ( ) ;
updateProducedError = true ;
}
m_CTest - > InitializeFromCommand ( m_Command ) ;
}
cmCTestLog ( m_CTest , HANDLER_OUTPUT , " Updating the repository: " < < sourceDirectory < < std : : endl ) ;
2005-02-16 21:28:47 +03:00
// Get update command
2005-05-03 17:40:16 +04:00
std : : string updateCommand = m_CTest - > GetCTestConfiguration ( " UpdateCommand " ) ;
2005-02-16 21:28:47 +03:00
if ( updateCommand . empty ( ) )
2004-09-07 17:17:15 +04:00
{
2005-05-03 17:40:16 +04:00
updateCommand = m_CTest - > GetCTestConfiguration ( " CVSCommand " ) ;
2005-02-16 21:28:47 +03:00
if ( updateCommand . empty ( ) )
{
2005-05-03 17:40:16 +04:00
updateCommand = m_CTest - > GetCTestConfiguration ( " SVNCommand " ) ;
2005-02-16 21:28:47 +03:00
if ( updateCommand . empty ( ) )
{
2005-06-01 02:40:43 +04:00
cmCTestLog ( m_CTest , ERROR_MESSAGE , " Cannot find CVSCommand, SVNCommand, or UpdateCommand key in the DartConfiguration.tcl " < < std : : endl ) ;
2005-02-16 21:28:47 +03:00
return - 1 ;
}
else
{
updateType = e_SVN ;
}
}
else
{
updateType = e_CVS ;
}
}
else
{
2005-05-03 17:40:16 +04:00
updateType = this - > DetermineType ( updateCommand . c_str ( ) , m_CTest - > GetCTestConfiguration ( " UpdateType " ) . c_str ( ) ) ;
2004-09-07 17:17:15 +04:00
}
2005-06-17 21:04:56 +04:00
cmCTestLog ( m_CTest , HANDLER_OUTPUT , " Use " < < cmCTestUpdateHandlerUpdateToString ( updateType ) < < " repository type " < < std : : endl ; ) ;
2005-02-16 21:28:47 +03:00
// And update options
2005-05-03 17:40:16 +04:00
std : : string updateOptions = m_CTest - > GetCTestConfiguration ( " UpdateOptions " ) ;
2005-02-16 21:28:47 +03:00
if ( updateOptions . empty ( ) )
2004-09-07 17:17:15 +04:00
{
2005-02-16 21:28:47 +03:00
switch ( updateType )
{
case cmCTestUpdateHandler : : e_CVS :
2005-05-03 17:40:16 +04:00
updateOptions = m_CTest - > GetCTestConfiguration ( " CVSUpdateOptions " ) ;
2005-07-11 19:59:51 +04:00
if ( updateOptions . empty ( ) )
{
updateOptions = " -dP " ;
}
2005-02-16 21:28:47 +03:00
break ;
case cmCTestUpdateHandler : : e_SVN :
2005-05-03 17:40:16 +04:00
updateOptions = m_CTest - > GetCTestConfiguration ( " SVNUpdateOptions " ) ;
2005-02-16 21:28:47 +03:00
break ;
}
2004-09-07 17:17:15 +04:00
}
2005-02-16 21:28:47 +03:00
// Get update time
2004-09-07 17:17:15 +04:00
std : : string extra_update_opts ;
if ( m_CTest - > GetTestModel ( ) = = cmCTest : : NIGHTLY )
{
2005-06-01 01:32:40 +04:00
struct tm * t = m_CTest - > GetNightlyTime ( m_CTest - > GetCTestConfiguration ( " NightlyStartTime " ) ,
m_CTest - > GetTomorrowTag ( ) ) ;
2004-09-07 17:17:15 +04:00
char current_time [ 1024 ] ;
2005-02-16 21:28:47 +03:00
sprintf ( current_time , " %04d-%02d-%02d %02d:%02d:%02d " ,
2004-09-07 17:17:15 +04:00
t - > tm_year + 1900 ,
t - > tm_mon + 1 ,
t - > tm_mday ,
t - > tm_hour ,
t - > tm_min ,
t - > tm_sec ) ;
std : : string today_update_date = current_time ;
2005-02-16 21:28:47 +03:00
// TODO: SVN
switch ( updateType )
{
case cmCTestUpdateHandler : : e_CVS :
extra_update_opts + = " -D \" " + today_update_date + " UTC \" " ;
break ;
case cmCTestUpdateHandler : : e_SVN :
extra_update_opts + = " -r \" { " + today_update_date + " +0000} \" " ;
break ;
}
}
2005-05-04 19:16:53 +04:00
updateCommand = " \" " + updateCommand + " \" " ;
2005-02-16 21:28:47 +03:00
// First, check what the current state of repository is
std : : string command = " " ;
switch ( updateType )
{
case cmCTestUpdateHandler : : e_CVS :
// TODO: CVS - for now just leave empty
break ;
case cmCTestUpdateHandler : : e_SVN :
command = updateCommand + " info " ;
break ;
}
// CVS variables
// SVN variables
int svn_current_revision = 0 ;
int svn_latest_revision = 0 ;
int svn_use_status = 0 ;
bool res = true ;
2005-04-24 21:57:11 +04:00
//
// Get initial repository information if that is possible. With subversion, this will check the current revision.
//
2005-02-16 21:28:47 +03:00
if ( ! command . empty ( ) )
{
2005-06-01 01:32:40 +04:00
cmCTestLog ( m_CTest , HANDLER_VERBOSE_OUTPUT , " * Get repository information: " < < command . c_str ( ) < < std : : endl ) ;
2005-02-16 21:28:47 +03:00
if ( ! m_CTest - > GetShowOnly ( ) )
{
2005-06-14 19:42:53 +04:00
ofs < < " * Get repository information " < < std : : endl ;
ofs < < " Command: " < < command . c_str ( ) < < std : : endl ;
res = m_CTest - > RunCommand ( command . c_str ( ) , & goutput , & errors ,
& retVal , sourceDirectory , 0 /*m_TimeOut*/ ) ;
ofs < < " Output: " < < goutput . c_str ( ) < < std : : endl ;
ofs < < " Errors: " < < errors . c_str ( ) < < std : : endl ;
2005-02-16 21:28:47 +03:00
if ( ofs )
{
ofs < < " --- Update information --- " < < std : : endl ;
ofs < < goutput < < std : : endl ;
}
switch ( updateType )
{
case cmCTestUpdateHandler : : e_CVS :
// TODO: CVS - for now just leave empty
break ;
case cmCTestUpdateHandler : : e_SVN :
{
cmsys : : RegularExpression current_revision_regex ( " Revision: ([0-9]+) " ) ;
if ( current_revision_regex . find ( goutput . c_str ( ) ) )
{
std : : string currentRevisionString = current_revision_regex . match ( 1 ) ;
svn_current_revision = atoi ( currentRevisionString . c_str ( ) ) ;
2005-06-01 01:32:40 +04:00
cmCTestLog ( m_CTest , HANDLER_OUTPUT , " Old revision of repository is: " < < svn_current_revision < < std : : endl ) ;
2005-02-16 21:28:47 +03:00
}
}
break ;
}
}
else
{
2005-06-01 01:32:40 +04:00
cmCTestLog ( m_CTest , HANDLER_VERBOSE_OUTPUT , " Update with command: " < < command < < std : : endl ) ;
2005-02-16 21:28:47 +03:00
}
2004-09-07 17:17:15 +04:00
}
2005-02-16 21:28:47 +03:00
2005-04-24 21:57:11 +04:00
//
// Now update repository and remember what files were updated
//
2005-01-27 18:15:01 +03:00
cmGeneratedFileStream os ;
2005-06-23 21:04:18 +04:00
if ( ! this - > StartResultingXML ( " Update " , os ) )
2004-09-07 17:17:15 +04:00
{
2005-06-01 02:40:43 +04:00
cmCTestLog ( m_CTest , ERROR_MESSAGE , " Cannot open log file " < < std : : endl ) ;
2004-09-07 17:17:15 +04:00
}
std : : string start_time = m_CTest - > CurrentTime ( ) ;
double elapsed_time_start = cmSystemTools : : GetTime ( ) ;
2005-06-01 01:32:40 +04:00
cmCTestLog ( m_CTest , HANDLER_VERBOSE_OUTPUT , " * Update repository: " < < command . c_str ( ) < < std : : endl ) ;
2004-09-07 17:17:15 +04:00
if ( ! m_CTest - > GetShowOnly ( ) )
{
2005-04-24 21:57:11 +04:00
command = " " ;
switch ( updateType )
{
case cmCTestUpdateHandler : : e_CVS :
command = updateCommand + " -z3 update " + updateOptions +
" " + extra_update_opts ;
2005-06-14 19:42:53 +04:00
ofs < < " * Update repository: " < < std : : endl ;
ofs < < " Command: " < < command . c_str ( ) < < std : : endl ;
res = m_CTest - > RunCommand ( command . c_str ( ) , & goutput , & errors ,
& retVal , sourceDirectory , 0 /*m_TimeOut*/ ) ;
ofs < < " Output: " < < goutput . c_str ( ) < < std : : endl ;
ofs < < " Errors: " < < errors . c_str ( ) < < std : : endl ;
2005-04-24 21:57:11 +04:00
break ;
case cmCTestUpdateHandler : : e_SVN :
{
std : : string partialOutput ;
command = updateCommand + " update " + updateOptions +
" " + extra_update_opts ;
2005-06-14 19:42:53 +04:00
ofs < < " * Update repository: " < < std : : endl ;
ofs < < " Command: " < < command . c_str ( ) < < std : : endl ;
bool res1 = m_CTest - > RunCommand ( command . c_str ( ) , & partialOutput , & errors ,
& retVal , sourceDirectory , 0 /*m_TimeOut*/ ) ;
ofs < < " Output: " < < partialOutput . c_str ( ) < < std : : endl ;
ofs < < " Errors: " < < errors . c_str ( ) < < std : : endl ;
2005-06-17 21:04:56 +04:00
goutput = partialOutput ;
2005-04-24 21:57:11 +04:00
command = updateCommand + " status " ;
2005-06-14 19:42:53 +04:00
ofs < < " * Status repository: " < < std : : endl ;
ofs < < " Command: " < < command . c_str ( ) < < std : : endl ;
res = m_CTest - > RunCommand ( command . c_str ( ) , & partialOutput , & errors ,
& retVal , sourceDirectory , 0 /*m_TimeOut*/ ) ;
ofs < < " Output: " < < partialOutput . c_str ( ) < < std : : endl ;
ofs < < " Errors: " < < errors . c_str ( ) < < std : : endl ;
2005-04-24 21:57:11 +04:00
goutput + = partialOutput ;
2005-04-26 19:09:29 +04:00
res = res & & res1 ;
2005-06-17 21:04:56 +04:00
ofs < < " Total output of update: " < < goutput . c_str ( ) < < std : : endl ;
2005-04-24 21:57:11 +04:00
}
}
2005-02-16 21:28:47 +03:00
if ( ofs )
2004-09-07 17:17:15 +04:00
{
2005-02-16 21:28:47 +03:00
ofs < < " --- Update repository --- " < < std : : endl ;
2004-09-07 17:17:15 +04:00
ofs < < goutput < < std : : endl ; ;
}
}
2005-04-22 02:23:28 +04:00
if ( ! res | | retVal )
{
2005-04-24 20:32:56 +04:00
updateProducedError = true ;
2005-06-16 21:18:21 +04:00
checkoutErrorMessages + = " " + goutput ;
2005-04-22 02:23:28 +04:00
}
2004-09-07 17:17:15 +04:00
os < < " <?xml version= \" 1.0 \" encoding= \" UTF-8 \" ?> \n "
2005-02-16 21:28:47 +03:00
< < " <Update mode= \" Client \" Generator= \" ctest- "
< < cmVersion : : GetCMakeVersion ( ) < < " \" > \n "
2005-05-03 17:40:16 +04:00
< < " \t <Site> " < < m_CTest - > GetCTestConfiguration ( " Site " ) < < " </Site> \n "
< < " \t <BuildName> " < < m_CTest - > GetCTestConfiguration ( " BuildName " )
2005-02-16 21:28:47 +03:00
< < " </BuildName> \n "
< < " \t <BuildStamp> " < < m_CTest - > GetCurrentTag ( ) < < " - "
< < m_CTest - > GetTestModelString ( ) < < " </BuildStamp> " < < std : : endl ;
2004-09-07 17:17:15 +04:00
os < < " \t <StartDateTime> " < < start_time < < " </StartDateTime> \n "
2005-02-16 21:28:47 +03:00
< < " \t <UpdateCommand> " < < m_CTest - > MakeXMLSafe ( command )
2005-06-17 21:04:56 +04:00
< < " </UpdateCommand> \n "
< < " \t <UpdateType> " < < m_CTest - > MakeXMLSafe ( cmCTestUpdateHandlerUpdateToString ( updateType ) )
< < " </UpdateType> \n " ;
2005-02-16 21:28:47 +03:00
// Even though it failed, we may have some useful information. Try to continue...
std : : vector < cmStdString > lines ;
cmSystemTools : : Split ( goutput . c_str ( ) , lines ) ;
// CVS style regular expressions
cmsys : : RegularExpression cvs_date_author_regex ( " ^date: +([^;]+); +author: +([^;]+); +state: +[^;]+; " ) ;
cmsys : : RegularExpression cvs_revision_regex ( " ^revision +([^ ]*) *$ " ) ;
cmsys : : RegularExpression cvs_end_of_file_regex ( " ^=============================================================================$ " ) ;
cmsys : : RegularExpression cvs_end_of_comment_regex ( " ^----------------------------$ " ) ;
// Subversion style regular expressions
cmsys : : RegularExpression svn_status_line_regex ( " ^ *([0-9]+) *([0-9]+) *([^ ]+) *([^ ][^ \t \r \n ]*)[ \t \r \n ]*$ " ) ;
2005-02-16 21:45:08 +03:00
cmsys : : RegularExpression svn_latest_revision_regex ( " (Updated to|At) revision ([0-9]+) \\ . " ) ;
2005-02-16 21:28:47 +03:00
cmsys : : RegularExpression file_update_line ( " ([A-Z]) *(.*) " ) ;
std : : string current_path = " <no-path> " ;
bool first_file = true ;
cmCTestUpdateHandler : : AuthorsToUpdatesMap authors_files_map ;
int num_updated = 0 ;
int num_modified = 0 ;
int num_conflicting = 0 ;
// In subversion, get the latest revision
if ( updateType = = cmCTestUpdateHandler : : e_SVN )
{
2004-09-07 17:17:15 +04:00
for ( cc = 0 ; cc < lines . size ( ) ; cc + + )
{
const char * line = lines [ cc ] . c_str ( ) ;
2005-02-16 21:28:47 +03:00
if ( svn_latest_revision_regex . find ( line ) )
2004-09-07 17:17:15 +04:00
{
2005-02-16 21:45:08 +03:00
svn_latest_revision = atoi ( svn_latest_revision_regex . match ( 2 ) . c_str ( ) ) ;
2005-02-16 21:28:47 +03:00
}
}
2005-07-11 19:16:14 +04:00
if ( svn_latest_revision < = 0 )
{
cmCTestLog ( m_CTest , ERROR_MESSAGE , " Problem determining the current revision of the repository from output: " < < std : : endl < < goutput . c_str ( ) < < std : : endl ) ;
}
else
{
cmCTestLog ( m_CTest , HANDLER_OUTPUT , " Current revision of repository is: " < < svn_latest_revision < < std : : endl ) ;
}
2005-02-16 21:28:47 +03:00
}
2005-06-01 01:32:40 +04:00
cmCTestLog ( m_CTest , HANDLER_OUTPUT , " Gathering version information (each . represents one updated file): " < < std : : endl ) ;
2005-02-16 22:24:43 +03:00
int file_count = 0 ;
2005-02-16 21:28:47 +03:00
for ( cc = 0 ; cc < lines . size ( ) ; cc + + )
{
const char * line = lines [ cc ] . c_str ( ) ;
if ( file_update_line . find ( line ) )
{
2005-06-01 01:32:40 +04:00
if ( file_count = = 0 )
2005-02-16 21:28:47 +03:00
{
2005-06-01 01:32:40 +04:00
cmCTestLog ( m_CTest , HANDLER_OUTPUT , " " < < std : : flush ) ;
2005-02-16 21:28:47 +03:00
}
2005-06-01 01:32:40 +04:00
cmCTestLog ( m_CTest , HANDLER_OUTPUT , " . " < < std : : flush ) ;
2005-02-16 21:28:47 +03:00
std : : string upChar = file_update_line . match ( 1 ) ;
std : : string upFile = file_update_line . match ( 2 ) ;
char mod = upChar [ 0 ] ;
2005-04-24 21:57:11 +04:00
bool modifiedOrConflict = false ;
2005-02-16 22:24:43 +03:00
if ( mod ! = ' M ' & & mod ! = ' C ' & & mod ! = ' G ' )
2005-02-16 21:28:47 +03:00
{
count + + ;
2005-04-24 21:57:11 +04:00
modifiedOrConflict = true ;
2005-02-16 21:28:47 +03:00
}
const char * file = upFile . c_str ( ) ;
2005-06-01 01:32:40 +04:00
cmCTestLog ( m_CTest , DEBUG , " Line " < < cc < < " : " < < mod < < " - " < < file < < std : : endl ) ;
2005-02-16 21:28:47 +03:00
2005-04-24 21:57:11 +04:00
std : : string output ;
if ( modifiedOrConflict )
2005-02-16 21:28:47 +03:00
{
2005-04-24 21:57:11 +04:00
std : : string logcommand ;
switch ( updateType )
2004-09-07 17:17:15 +04:00
{
2005-04-24 21:57:11 +04:00
case cmCTestUpdateHandler : : e_CVS :
logcommand = updateCommand + " -z3 log -N \" " + file + " \" " ;
break ;
case cmCTestUpdateHandler : : e_SVN :
if ( svn_latest_revision > 0 & & svn_latest_revision > svn_current_revision )
{
cmOStringStream logCommandStream ;
logCommandStream < < updateCommand < < " log -r " < < svn_current_revision < < " : " < < svn_latest_revision
< < " --xml \" " < < file < < " \" " ;
logcommand = logCommandStream . str ( ) ;
}
else
{
logcommand = updateCommand + " status --verbose \" " + file + " \" " ;
svn_use_status = 1 ;
}
break ;
2004-09-07 17:17:15 +04:00
}
2005-06-01 01:32:40 +04:00
cmCTestLog ( m_CTest , DEBUG , " Do log: " < < logcommand < < std : : endl ) ;
cmCTestLog ( m_CTest , HANDLER_VERBOSE_OUTPUT , " * Get file update information: " < < logcommand . c_str ( ) < < std : : endl ) ;
2005-06-14 19:42:53 +04:00
ofs < < " * Get log information for file: " < < file < < std : : endl ;
ofs < < " Command: " < < logcommand . c_str ( ) < < std : : endl ;
res = m_CTest - > RunCommand ( logcommand . c_str ( ) , & output , & errors ,
& retVal , sourceDirectory , 0 /*m_TimeOut*/ ) ;
ofs < < " Output: " < < output . c_str ( ) < < std : : endl ;
ofs < < " Errors: " < < errors . c_str ( ) < < std : : endl ;
2005-04-24 21:57:11 +04:00
if ( ofs )
{
ofs < < output < < std : : endl ;
2004-09-07 17:17:15 +04:00
}
2005-02-16 21:28:47 +03:00
}
if ( res & & retVal = = 0 )
{
2005-06-01 01:32:40 +04:00
cmCTestLog ( m_CTest , DEBUG , output < < std : : endl ) ;
2005-02-16 21:28:47 +03:00
std : : string : : size_type sline = 0 ;
std : : string srevision1 = " Unknown " ;
std : : string sdate1 = " Unknown " ;
std : : string sauthor1 = " Unknown " ;
std : : string semail1 = " Unknown " ;
std : : string comment1 = " " ;
std : : string srevision2 = " Unknown " ;
std : : string sdate2 = " Unknown " ;
std : : string sauthor2 = " Unknown " ;
std : : string comment2 = " " ;
std : : string semail2 = " Unknown " ;
if ( updateType = = cmCTestUpdateHandler : : e_CVS )
2004-09-07 17:17:15 +04:00
{
bool have_first = false ;
bool have_second = false ;
2005-02-16 21:28:47 +03:00
std : : vector < cmStdString > ulines ;
cmSystemTools : : Split ( output . c_str ( ) , ulines ) ;
2004-09-07 17:17:15 +04:00
for ( kk = 0 ; kk < ulines . size ( ) ; kk + + )
{
const char * clp = ulines [ kk ] . c_str ( ) ;
2005-02-16 21:28:47 +03:00
if ( ! have_second & & ! sline & & cvs_revision_regex . find ( clp ) )
2004-09-07 17:17:15 +04:00
{
if ( ! have_first )
{
2005-02-16 21:28:47 +03:00
srevision1 = cvs_revision_regex . match ( 1 ) ;
2004-09-07 17:17:15 +04:00
}
else
{
2005-02-16 21:28:47 +03:00
srevision2 = cvs_revision_regex . match ( 1 ) ;
2004-09-07 17:17:15 +04:00
}
}
2005-02-16 21:28:47 +03:00
else if ( ! have_second & & ! sline & & cvs_date_author_regex . find ( clp ) )
2004-09-07 17:17:15 +04:00
{
sline = kk + 1 ;
if ( ! have_first )
{
2005-02-16 21:28:47 +03:00
sdate1 = cvs_date_author_regex . match ( 1 ) ;
sauthor1 = cvs_date_author_regex . match ( 2 ) ;
2004-09-07 17:17:15 +04:00
}
else
{
2005-02-16 21:28:47 +03:00
sdate2 = cvs_date_author_regex . match ( 1 ) ;
sauthor2 = cvs_date_author_regex . match ( 2 ) ;
2004-09-07 17:17:15 +04:00
}
}
2005-02-16 21:28:47 +03:00
else if ( sline & & cvs_end_of_comment_regex . find ( clp ) | | cvs_end_of_file_regex . find ( clp ) )
2004-09-07 17:17:15 +04:00
{
if ( ! have_first )
{
have_first = true ;
}
else if ( ! have_second )
{
have_second = true ;
}
sline = 0 ;
}
else if ( sline )
{
if ( ! have_first )
{
comment1 + = clp ;
comment1 + = " \n " ;
}
else
{
comment2 + = clp ;
comment2 + = " \n " ;
}
}
}
2005-02-16 21:28:47 +03:00
}
else if ( updateType = = cmCTestUpdateHandler : : e_SVN )
{
if ( svn_use_status )
2004-09-07 17:17:15 +04:00
{
2005-02-16 21:28:47 +03:00
cmOStringStream str ;
str < < svn_current_revision ;
srevision1 = str . str ( ) ;
if ( ! svn_status_line_regex . find ( output ) )
{
2005-06-01 02:40:43 +04:00
cmCTestLog ( m_CTest , ERROR_MESSAGE , " Bad output from SVN status command: " < < output < < std : : endl ) ;
2005-02-16 21:28:47 +03:00
}
else if ( svn_status_line_regex . match ( 4 ) ! = file )
2004-09-07 17:17:15 +04:00
{
2005-06-01 02:40:43 +04:00
cmCTestLog ( m_CTest , ERROR_MESSAGE , " Bad output from SVN status command. The file name returned: \" " < < svn_status_line_regex . match ( 4 ) < < " \" was different than the file specified: \" " < < file < < " \" " < < std : : endl ) ;
2004-09-07 17:17:15 +04:00
}
else
{
2005-02-16 21:28:47 +03:00
srevision1 = svn_status_line_regex . match ( 2 ) ;
int latest_revision = atoi ( svn_status_line_regex . match ( 2 ) . c_str ( ) ) ;
if ( svn_current_revision < latest_revision )
{
srevision2 = str . str ( ) ;
}
sauthor1 = svn_status_line_regex . match ( 3 ) ;
2004-09-07 17:17:15 +04:00
}
}
else
{
2005-02-16 21:28:47 +03:00
cmCTestUpdateHandlerSVNXMLParser parser ( this ) ;
if ( parser . Parse ( output . c_str ( ) ) )
{
int minrev = parser . GetMinRevision ( ) ;
int maxrev = parser . GetMaxRevision ( ) ;
cmCTestUpdateHandlerSVNXMLParser : : t_VectorOfCommits : : iterator it ;
for ( it = parser . GetCommits ( ) - > begin ( ) ;
it ! = parser . GetCommits ( ) - > end ( ) ;
+ + it )
{
if ( it - > m_Revision = = maxrev )
{
cmOStringStream mRevStream ;
mRevStream < < maxrev ;
srevision1 = mRevStream . str ( ) ;
sauthor1 = it - > m_Author ;
comment1 = it - > m_Message ;
sdate1 = it - > m_Date ;
}
else if ( it - > m_Revision = = minrev )
{
cmOStringStream mRevStream ;
mRevStream < < minrev ;
srevision2 = mRevStream . str ( ) ;
sauthor2 = it - > m_Author ;
comment2 = it - > m_Message ;
sdate2 = it - > m_Date ;
}
}
}
2004-09-07 17:17:15 +04:00
}
2005-02-16 21:28:47 +03:00
}
if ( mod = = ' M ' )
{
comment1 = " Locally modified file \n " ;
2005-04-24 21:57:11 +04:00
sauthor1 = " Local User " ;
2005-02-16 21:28:47 +03:00
}
if ( mod = = ' C ' )
{
comment1 = " Conflict while updating \n " ;
2005-04-24 21:57:11 +04:00
sauthor1 = " Local User " ;
2005-02-16 21:28:47 +03:00
}
std : : string path = cmSystemTools : : GetFilenamePath ( file ) ;
std : : string fname = cmSystemTools : : GetFilenameName ( file ) ;
if ( path ! = current_path )
{
if ( ! first_file )
2004-09-07 17:17:15 +04:00
{
2005-02-16 21:28:47 +03:00
os < < " \t </Directory> " < < std : : endl ;
2004-09-07 17:17:15 +04:00
}
else
{
2005-02-16 21:28:47 +03:00
first_file = false ;
2004-09-07 17:17:15 +04:00
}
2005-02-16 21:28:47 +03:00
os < < " \t <Directory> \n "
< < " \t \t <Name> " < < path < < " </Name> " < < std : : endl ;
}
if ( mod = = ' C ' )
{
num_conflicting + + ;
os < < " \t <Conflicting> " < < std : : endl ;
}
2005-02-16 22:24:43 +03:00
else if ( mod = = ' G ' )
{
num_conflicting + + ;
os < < " \t <Conflicting> " < < std : : endl ;
}
2005-02-16 21:28:47 +03:00
else if ( mod = = ' M ' )
{
num_modified + + ;
os < < " \t <Modified> " < < std : : endl ;
2004-09-07 17:17:15 +04:00
}
2005-02-16 21:28:47 +03:00
else
{
num_updated + + ;
os < < " \t <Updated> " < < std : : endl ;
}
if ( srevision2 = = " Unknown " )
{
srevision2 = srevision1 ;
}
2005-06-01 01:32:40 +04:00
cmCTestLog ( m_CTest , HANDLER_VERBOSE_OUTPUT , " File: " < < path . c_str ( ) < < " / " < < fname . c_str ( ) < < " was updated by "
< < sauthor1 . c_str ( ) < < " to revision: " < < srevision1 . c_str ( )
< < " from revision: " < < srevision2 . c_str ( ) < < std : : endl ) ;
2005-02-16 21:28:47 +03:00
os < < " \t \t <File Directory= \" " < < cmCTest : : MakeXMLSafe ( path ) < < " \" > " < < cmCTest : : MakeXMLSafe ( fname )
< < " </File> \n "
< < " \t \t <Directory> " < < cmCTest : : MakeXMLSafe ( path ) < < " </Directory> \n "
< < " \t \t <FullName> " < < cmCTest : : MakeXMLSafe ( file ) < < " </FullName> \n "
< < " \t \t <CheckinDate> " < < cmCTest : : MakeXMLSafe ( sdate1 ) < < " </CheckinDate> \n "
< < " \t \t <Author> " < < cmCTest : : MakeXMLSafe ( sauthor1 ) < < " </Author> \n "
< < " \t \t <Email> " < < cmCTest : : MakeXMLSafe ( semail1 ) < < " </Email> \n "
< < " \t \t <Log> " < < cmCTest : : MakeXMLSafe ( comment1 ) < < " </Log> \n "
< < " \t \t <Revision> " < < srevision1 < < " </Revision> \n "
< < " \t \t <PriorRevision> " < < srevision2 < < " </PriorRevision> "
< < std : : endl ;
if ( srevision2 ! = srevision1 )
{
os
< < " \t \t <Revisions> \n "
< < " \t \t \t <Revision> " < < srevision1 < < " </Revision> \n "
< < " \t \t \t <PreviousRevision> " < < srevision2 < < " </PreviousRevision> \n "
< < " \t \t \t <Author> " < < cmCTest : : MakeXMLSafe ( sauthor1 ) < < " </Author> \n "
< < " \t \t \t <Date> " < < cmCTest : : MakeXMLSafe ( sdate1 ) < < " </Date> \n "
< < " \t \t \t <Comment> " < < cmCTest : : MakeXMLSafe ( comment1 ) < < " </Comment> \n "
< < " \t \t \t <Email> " < < cmCTest : : MakeXMLSafe ( semail1 ) < < " </Email> \n "
< < " \t \t </Revisions> \n "
< < " \t \t <Revisions> \n "
< < " \t \t \t <Revision> " < < srevision2 < < " </Revision> \n "
< < " \t \t \t <PreviousRevision> " < < srevision2 < < " </PreviousRevision> \n "
< < " \t \t \t <Author> " < < cmCTest : : MakeXMLSafe ( sauthor2 ) < < " </Author> \n "
< < " \t \t \t <Date> " < < cmCTest : : MakeXMLSafe ( sdate2 ) < < " </Date> \n "
< < " \t \t \t <Comment> " < < cmCTest : : MakeXMLSafe ( comment2 ) < < " </Comment> \n "
< < " \t \t \t <Email> " < < cmCTest : : MakeXMLSafe ( semail2 ) < < " </Email> \n "
< < " \t \t </Revisions> " < < std : : endl ;
}
if ( mod = = ' C ' )
{
os < < " \t </Conflicting> " < < std : : endl ;
}
2005-02-16 22:24:43 +03:00
else if ( mod = = ' G ' )
{
os < < " \t </Conflicting> " < < std : : endl ;
}
2005-02-16 21:28:47 +03:00
else if ( mod = = ' M ' )
{
os < < " \t </Modified> " < < std : : endl ;
}
else
{
os < < " \t </Updated> " < < std : : endl ;
}
cmCTestUpdateHandler : : UpdateFiles * u = & authors_files_map [ sauthor1 ] ;
cmCTestUpdateHandler : : StringPair p ;
p . first = path ;
p . second = fname ;
u - > push_back ( p ) ;
current_path = path ;
2004-09-07 17:17:15 +04:00
}
2005-02-16 22:24:43 +03:00
file_count + + ;
2004-09-07 17:17:15 +04:00
}
2005-02-16 21:28:47 +03:00
}
2005-02-16 22:24:43 +03:00
if ( file_count )
{
2005-06-01 01:32:40 +04:00
cmCTestLog ( m_CTest , HANDLER_OUTPUT , std : : endl ) ;
2005-02-16 22:24:43 +03:00
}
2005-02-16 21:28:47 +03:00
if ( num_updated )
{
2005-06-01 01:32:40 +04:00
cmCTestLog ( m_CTest , HANDLER_OUTPUT , " Found " < < num_updated < < " updated files " < < std : : endl ) ;
2005-02-16 21:28:47 +03:00
}
if ( num_modified )
{
2005-06-01 01:32:40 +04:00
cmCTestLog ( m_CTest , HANDLER_OUTPUT , " Found " < < num_modified < < " locally modified files "
< < std : : endl ) ;
2005-02-16 21:28:47 +03:00
}
if ( num_conflicting )
{
2005-06-01 01:32:40 +04:00
cmCTestLog ( m_CTest , HANDLER_OUTPUT , " Found " < < num_conflicting < < " conflicting files "
< < std : : endl ) ;
2005-02-16 21:28:47 +03:00
}
2005-02-16 22:24:43 +03:00
if ( num_modified = = 0 & & num_conflicting = = 0 & & num_updated = = 0 )
{
2005-06-01 01:32:40 +04:00
cmCTestLog ( m_CTest , HANDLER_OUTPUT , " Project is up-to-date " < < std : : endl ) ;
2005-02-16 22:24:43 +03:00
}
2005-02-16 21:28:47 +03:00
if ( ! first_file )
{
os < < " \t </Directory> " < < std : : endl ;
}
2004-09-07 17:17:15 +04:00
2005-02-16 21:28:47 +03:00
cmCTestUpdateHandler : : AuthorsToUpdatesMap : : iterator it ;
for ( it = authors_files_map . begin ( ) ;
it ! = authors_files_map . end ( ) ;
it + + )
{
os < < " \t <Author> \n "
< < " \t \t <Name> " < < it - > first < < " </Name> " < < std : : endl ;
cmCTestUpdateHandler : : UpdateFiles * u = & ( it - > second ) ;
for ( cc = 0 ; cc < u - > size ( ) ; cc + + )
2004-09-07 17:17:15 +04:00
{
2005-02-16 21:28:47 +03:00
os < < " \t \t <File Directory= \" " < < ( * u ) [ cc ] . first < < " \" > "
< < ( * u ) [ cc ] . second < < " </File> " < < std : : endl ;
2004-09-07 17:17:15 +04:00
}
2005-02-16 21:28:47 +03:00
os < < " \t </Author> " < < std : : endl ;
2004-09-07 17:17:15 +04:00
}
2005-06-01 01:32:40 +04:00
cmCTestLog ( m_CTest , DEBUG , " End " < < std : : endl ) ;
2004-09-07 17:17:15 +04:00
std : : string end_time = m_CTest - > CurrentTime ( ) ;
os < < " \t <EndDateTime> " < < end_time < < " </EndDateTime> \n "
2005-02-16 21:28:47 +03:00
< < " <ElapsedMinutes> " < <
2004-09-07 17:17:15 +04:00
static_cast < int > ( ( cmSystemTools : : GetTime ( ) - elapsed_time_start ) / 6 ) / 10.0
2005-04-22 02:23:28 +04:00
< < " </ElapsedMinutes> \n "
< < " \t <UpdateReturnStatus> " ;
if ( num_modified > 0 | | num_conflicting > 0 )
{
os < < " Update error: There are modified or conflicting files in the repository " ;
2005-06-01 02:40:43 +04:00
cmCTestLog ( m_CTest , ERROR_MESSAGE , " There are modified or conflicting files in the repository " < < std : : endl ) ;
2005-04-22 02:23:28 +04:00
}
2005-04-24 20:32:56 +04:00
if ( updateProducedError )
2005-04-22 02:23:28 +04:00
{
os < < " Update error: " ;
2005-06-16 21:18:21 +04:00
os < < m_CTest - > MakeXMLSafe ( checkoutErrorMessages ) ;
2005-06-01 02:40:43 +04:00
cmCTestLog ( m_CTest , ERROR_MESSAGE , " Update with command: " < < command < < " failed " < < std : : endl ) ;
2005-04-22 02:23:28 +04:00
}
os < < " </UpdateReturnStatus> " < < std : : endl ;
os < < " </Update> " < < std : : endl ;
2005-02-16 21:28:47 +03:00
2004-09-07 17:17:15 +04:00
if ( ofs )
{
ofs . close ( ) ;
}
if ( ! res | | retVal )
{
2005-06-01 02:40:43 +04:00
cmCTestLog ( m_CTest , ERROR_MESSAGE , " Error(s) when updating the project " < < std : : endl ) ;
cmCTestLog ( m_CTest , ERROR_MESSAGE , " Output: " < < goutput < < std : : endl ) ;
2004-09-07 17:17:15 +04:00
return - 1 ;
}
return count ;
}