diff --git a/README.md b/README.md index a6f9fdf..cb4d454 100644 --- a/README.md +++ b/README.md @@ -58,7 +58,7 @@ void MyAnalysis::BeginInputData( const SInputData& ) throw( SError ) { void MyAnalysis::ExecuteEvent( const SInputData&, Double_t ) throw( SError ) { - bool isGood = true; + // might be better in a separate function (e.g. bool MyAnalysis::isGoodEvent(runNumber, lumiSection)) if (m_isData) { bool isGood = m_grl.HasRunLumiBlock( runNumber, lumiSection ); if( !isGood ) { @@ -101,4 +101,5 @@ In the `JobConfiguration` part, add before the `Library` of your analysis code: ## Acknowledgements -The JSON parser is taken from https://github.com/open-source-parsers/jsoncpp \ No newline at end of file +The JSON parser is taken from https://github.com/open-source-parsers/jsoncpp +Most of the other code is taken from Max Baak and Attila Krasznahorkay, who developed this code for ATLAS analyses. \ No newline at end of file diff --git a/app/anyoption.cxx b/app/anyoption.cxx new file mode 100644 index 0000000..a84f491 --- /dev/null +++ b/app/anyoption.cxx @@ -0,0 +1,1176 @@ +/* + * AnyOption 1.3 + * + * kishan at hackorama dot com www.hackorama.com JULY 2001 + * + * + Acts as a common facade class for reading + * commandline options as well as options from + * an optionfile with delimited type value pairs + * + * + Handles the POSIX style single character options ( -w ) + * as well as the newer GNU long options ( --width ) + * + * + The option file assumes the traditional format of + * first character based comment lines and type value + * pairs with a delimiter , and flags which are not pairs + * + * # this is a coment + * # next line is an option value pair + * width : 100 + * # next line is a flag + * noimages + * + * + Supports printing out Help and Usage + * + * + Why not just use getopt() ? + * + * getopt() Its a POSIX standard not part of ANSI-C. + * So it may not be available on platforms like Windows. + * + * + Why it is so long ? + * + * The actual code which does command line parsing + * and option file parsing are done in few methods. + * Most of the extra code are for providing a flexible + * common public interface to both a resourcefile and + * and command line supporting POSIX style and + * GNU long option as well as mixing of both. + * + * + Please see "anyoption.h" for public method descriptions + * + */ + +/* Updated Auguest 2004 + * Fix from Michael D Peters (mpeters at sandia.gov) + * to remove static local variables, allowing multiple instantiations + * of the reader (for using multiple configuration files). There is + * an error in the destructor when using multiple instances, so you + * cannot delete your objects (it will crash), but not calling the + * destructor only introduces a small memory leak, so I + * have not bothered tracking it down. + * + * Also updated to use modern C++ style headers, rather than + * depricated iostream.h (it was causing my compiler problems) +*/ + +/* + * Updated September 2006 + * Fix from Boyan Asenov for a bug in mixing up option indexes + * leading to exception when mixing different options types + */ + +#include "string.h" + +#include "anyoption.h" + +AnyOption::AnyOption() +{ + init(); +} + +AnyOption::AnyOption(int maxopt) +{ + init( maxopt , maxopt ); +} + +AnyOption::AnyOption(int maxopt, int maxcharopt) +{ + init( maxopt , maxcharopt ); +} + +AnyOption::~AnyOption() +{ + if( mem_allocated ) + cleanup(); +} + +void +AnyOption::init() +{ + init( DEFAULT_MAXOPTS , DEFAULT_MAXOPTS ); +} + +void +AnyOption::init(int maxopt, int maxcharopt ) +{ + + max_options = maxopt; + max_char_options = maxcharopt; + max_usage_lines = DEFAULT_MAXUSAGE; + usage_lines = 0 ; + argc = 0; + argv = NULL; + posix_style = true; + verbose = false; + filename = NULL; + appname = NULL; + option_counter = 0; + optchar_counter = 0; + new_argv = NULL; + new_argc = 0 ; + max_legal_args = 0 ; + command_set = false; + file_set = false; + values = NULL; + g_value_counter = 0; + mem_allocated = false; + command_set = false; + file_set = false; + opt_prefix_char = '-'; + file_delimiter_char = ':'; + file_comment_char = '#'; + equalsign = '='; + comment = '#' ; + delimiter = ':' ; + endofline = '\n'; + whitespace = ' ' ; + nullterminate = '\0'; + set = false; + once = true; + hasoptions = false; + autousage = false; + + strcpy( long_opt_prefix , "--" ); + + if( alloc() == false ){ + cout << endl << "OPTIONS ERROR : Failed allocating memory" ; + cout << endl ; + cout << "Exiting." << endl ; + exit (0); + } +} + +bool +AnyOption::alloc() +{ + int i = 0 ; + int size = 0 ; + + if( mem_allocated ) + return true; + + size = (max_options+1) * sizeof(const char*); + options = (const char**)malloc( size ); + optiontype = (int*) malloc( (max_options+1)*sizeof(int) ); + optionindex = (int*) malloc( (max_options+1)*sizeof(int) ); + if( options == NULL || optiontype == NULL || optionindex == NULL ) + return false; + else + mem_allocated = true; + for( i = 0 ; i < max_options ; i++ ){ + options[i] = NULL; + optiontype[i] = 0 ; + optionindex[i] = -1 ; + } + optionchars = (char*) malloc( (max_char_options+1)*sizeof(char) ); + optchartype = (int*) malloc( (max_char_options+1)*sizeof(int) ); + optcharindex = (int*) malloc( (max_char_options+1)*sizeof(int) ); + if( optionchars == NULL || + optchartype == NULL || + optcharindex == NULL ) + { + mem_allocated = false; + return false; + } + for( i = 0 ; i < max_char_options ; i++ ){ + optionchars[i] = '0'; + optchartype[i] = 0 ; + optcharindex[i] = -1 ; + } + + size = (max_usage_lines+1) * sizeof(const char*) ; + usage = (const char**) malloc( size ); + + if( usage == NULL ){ + mem_allocated = false; + return false; + } + for( i = 0 ; i < max_usage_lines ; i++ ) + usage[i] = NULL; + + return true; +} + +bool +AnyOption::doubleOptStorage() +{ + options = (const char**)realloc( options, + ((2*max_options)+1) * sizeof( const char*) ); + optiontype = (int*) realloc( optiontype , + ((2 * max_options)+1)* sizeof(int) ); + optionindex = (int*) realloc( optionindex, + ((2 * max_options)+1) * sizeof(int) ); + if( options == NULL || optiontype == NULL || optionindex == NULL ) + return false; + /* init new storage */ + for( int i = max_options ; i < 2*max_options ; i++ ){ + options[i] = NULL; + optiontype[i] = 0 ; + optionindex[i] = -1 ; + } + max_options = 2 * max_options ; + return true; +} + +bool +AnyOption::doubleCharStorage() +{ + optionchars = (char*) realloc( optionchars, + ((2*max_char_options)+1)*sizeof(char) ); + optchartype = (int*) realloc( optchartype, + ((2*max_char_options)+1)*sizeof(int) ); + optcharindex = (int*) realloc( optcharindex, + ((2*max_char_options)+1)*sizeof(int) ); + if( optionchars == NULL || + optchartype == NULL || + optcharindex == NULL ) + return false; + /* init new storage */ + for( int i = max_char_options ; i < 2*max_char_options ; i++ ){ + optionchars[i] = '0'; + optchartype[i] = 0 ; + optcharindex[i] = -1 ; + } + max_char_options = 2 * max_char_options; + return true; +} + +bool +AnyOption::doubleUsageStorage() +{ + usage = (const char**)realloc( usage, + ((2*max_usage_lines)+1) * sizeof( const char*) ); + if ( usage == NULL ) + return false; + for( int i = max_usage_lines ; i < 2*max_usage_lines ; i++ ) + usage[i] = NULL; + max_usage_lines = 2 * max_usage_lines ; + return true; + +} + + +void +AnyOption::cleanup() +{ + free (options); + free (optiontype); + free (optionindex); + free (optionchars); + free (optchartype); + free (optcharindex); + free (usage); + if( values != NULL ) + free (values); + if( new_argv != NULL ) + free (new_argv); +} + +void +AnyOption::setCommandPrefixChar( char _prefix ) +{ + opt_prefix_char = _prefix; +} + +void +AnyOption::setCommandLongPrefix( char *_prefix ) +{ + if( strlen( _prefix ) > MAX_LONG_PREFIX_LENGTH ){ + *( _prefix + MAX_LONG_PREFIX_LENGTH ) = '\0'; + } + + strcpy (long_opt_prefix, _prefix); +} + +void +AnyOption::setFileCommentChar( char _comment ) +{ + file_delimiter_char = _comment; +} + + +void +AnyOption::setFileDelimiterChar( char _delimiter ) +{ + file_comment_char = _delimiter ; +} + +bool +AnyOption::CommandSet() +{ + return( command_set ); +} + +bool +AnyOption::FileSet() +{ + return( file_set ); +} + +void +AnyOption::noPOSIX() +{ + posix_style = false; +} + +bool +AnyOption::POSIX() +{ + return posix_style; +} + + +void +AnyOption::setVerbose() +{ + verbose = true ; +} + +void +AnyOption::printVerbose() +{ + if( verbose ) + cout << endl ; +} +void +AnyOption::printVerbose( const char *msg ) +{ + if( verbose ) + cout << msg ; +} + +void +AnyOption::printVerbose( char *msg ) +{ + if( verbose ) + cout << msg ; +} + +void +AnyOption::printVerbose( char ch ) +{ + if( verbose ) + cout << ch ; +} + +bool +AnyOption::hasOptions() +{ + return hasoptions; +} + +void +AnyOption::autoUsagePrint(bool _autousage) +{ + autousage = _autousage; +} + +void +AnyOption::useCommandArgs( int _argc, char **_argv ) +{ + argc = _argc; + argv = _argv; + command_set = true; + appname = argv[0]; + if(argc > 1) hasoptions = true; +} + +void +AnyOption::useFiileName( const char *_filename ) +{ + filename = _filename; + file_set = true; +} + +/* + * set methods for options + */ + +void +AnyOption::setCommandOption( const char *opt ) +{ + addOption( opt , COMMAND_OPT ); + g_value_counter++; +} + +void +AnyOption::setCommandOption( char opt ) +{ + addOption( opt , COMMAND_OPT ); + g_value_counter++; +} + +void +AnyOption::setCommandOption( const char *opt , char optchar ) +{ + addOption( opt , COMMAND_OPT ); + addOption( optchar , COMMAND_OPT ); + g_value_counter++; +} + +void +AnyOption::setCommandFlag( const char *opt ) +{ + addOption( opt , COMMAND_FLAG ); + g_value_counter++; +} + +void +AnyOption::setCommandFlag( char opt ) +{ + addOption( opt , COMMAND_FLAG ); + g_value_counter++; +} + +void +AnyOption::setCommandFlag( const char *opt , char optchar ) +{ + addOption( opt , COMMAND_FLAG ); + addOption( optchar , COMMAND_FLAG ); + g_value_counter++; +} + +void +AnyOption::setFileOption( const char *opt ) +{ + addOption( opt , FILE_OPT ); + g_value_counter++; +} + +void +AnyOption::setFileOption( char opt ) +{ + addOption( opt , FILE_OPT ); + g_value_counter++; +} + +void +AnyOption::setFileOption( const char *opt , char optchar ) +{ + addOption( opt , FILE_OPT ); + addOption( optchar, FILE_OPT ); + g_value_counter++; +} + +void +AnyOption::setFileFlag( const char *opt ) +{ + addOption( opt , FILE_FLAG ); + g_value_counter++; +} + +void +AnyOption::setFileFlag( char opt ) +{ + addOption( opt , FILE_FLAG ); + g_value_counter++; +} + +void +AnyOption::setFileFlag( const char *opt , char optchar ) +{ + addOption( opt , FILE_FLAG ); + addOption( optchar , FILE_FLAG ); + g_value_counter++; +} + +void +AnyOption::setOption( const char *opt ) +{ + addOption( opt , COMMON_OPT ); + g_value_counter++; +} + +void +AnyOption::setOption( char opt ) +{ + addOption( opt , COMMON_OPT ); + g_value_counter++; +} + +void +AnyOption::setOption( const char *opt , char optchar ) +{ + addOption( opt , COMMON_OPT ); + addOption( optchar , COMMON_OPT ); + g_value_counter++; +} + +void +AnyOption::setFlag( const char *opt ) +{ + addOption( opt , COMMON_FLAG ); + g_value_counter++; +} + +void +AnyOption::setFlag( const char opt ) +{ + addOption( opt , COMMON_FLAG ); + g_value_counter++; +} + +void +AnyOption::setFlag( const char *opt , char optchar ) +{ + addOption( opt , COMMON_FLAG ); + addOption( optchar , COMMON_FLAG ); + g_value_counter++; +} + +void +AnyOption::addOption( const char *opt, int type ) +{ + if( option_counter >= max_options ){ + if( doubleOptStorage() == false ){ + addOptionError( opt ); + return; + } + } + options[ option_counter ] = opt ; + optiontype[ option_counter ] = type ; + optionindex[ option_counter ] = g_value_counter; + option_counter++; +} + +void +AnyOption::addOption( char opt, int type ) +{ + if( !POSIX() ){ + printVerbose("Ignoring the option character \""); + printVerbose( opt ); + printVerbose( "\" ( POSIX options are turned off )" ); + printVerbose(); + return; + } + + + if( optchar_counter >= max_char_options ){ + if( doubleCharStorage() == false ){ + addOptionError( opt ); + return; + } + } + optionchars[ optchar_counter ] = opt ; + optchartype[ optchar_counter ] = type ; + optcharindex[ optchar_counter ] = g_value_counter; + optchar_counter++; +} + +void +AnyOption::addOptionError( const char *opt ) +{ + cout << endl ; + cout << "OPTIONS ERROR : Failed allocating extra memory " << endl ; + cout << "While adding the option : \""<< opt << "\"" << endl; + cout << "Exiting." << endl ; + cout << endl ; + exit(0); +} + +void +AnyOption::addOptionError( char opt ) +{ + cout << endl ; + cout << "OPTIONS ERROR : Failed allocating extra memory " << endl ; + cout << "While adding the option: \""<< opt << "\"" << endl; + cout << "Exiting." << endl ; + cout << endl ; + exit(0); +} + +void +AnyOption::processOptions() +{ + if( ! valueStoreOK() ) + return; +} + +void +AnyOption::processCommandArgs(int max_args) +{ + max_legal_args = max_args; + processCommandArgs(); +} + +void +AnyOption::processCommandArgs( int _argc, char **_argv, int max_args ) +{ + max_legal_args = max_args; + processCommandArgs( _argc, _argv ); +} + +void +AnyOption::processCommandArgs( int _argc, char **_argv ) +{ + useCommandArgs( _argc, _argv ); + processCommandArgs(); +} + +void +AnyOption::processCommandArgs() +{ + if( ! ( valueStoreOK() && CommandSet() ) ) + return; + + if( max_legal_args == 0 ) + max_legal_args = argc; + new_argv = (int*) malloc( (max_legal_args+1) * sizeof(int) ); + for( int i = 1 ; i < argc ; i++ ){/* ignore first argv */ + if( argv[i][0] == long_opt_prefix[0] && + argv[i][1] == long_opt_prefix[1] ) { /* long GNU option */ + int match_at = parseGNU( argv[i]+2 ); /* skip -- */ + if( match_at >= 0 && i < argc-1 ) /* found match */ + setValue( options[match_at] , argv[++i] ); + }else if( argv[i][0] == opt_prefix_char ) { /* POSIX char */ + if( POSIX() ){ + char ch = parsePOSIX( argv[i]+1 );/* skip - */ + if( ch != '0' && i < argc-1 ) /* matching char */ + setValue( ch , argv[++i] ); + } else { /* treat it as GNU option with a - */ + int match_at = parseGNU( argv[i]+1 ); /* skip - */ + if( match_at >= 0 && i < argc-1 ) /* found match */ + setValue( options[match_at] , argv[++i] ); + } + }else { /* not option but an argument keep index */ + if( new_argc < max_legal_args ){ + new_argv[ new_argc ] = i ; + new_argc++; + }else{ /* ignore extra arguments */ + printVerbose( "Ignoring extra argument: " ); + printVerbose( argv[i] ); + printVerbose( ); + printAutoUsage(); + } + printVerbose( "Unknown command argument option : " ); + printVerbose( argv[i] ); + printVerbose( ); + printAutoUsage(); + } + } +} + +char +AnyOption::parsePOSIX( char* arg ) +{ + + for( unsigned int i = 0 ; i < strlen(arg) ; i++ ){ + char ch = arg[i] ; + if( matchChar(ch) ) { /* keep matching flags till an option */ + /*if last char argv[++i] is the value */ + if( i == strlen(arg)-1 ){ + return ch; + }else{/* else the rest of arg is the value */ + i++; /* skip any '=' and ' ' */ + while( arg[i] == whitespace + || arg[i] == equalsign ) + i++; + setValue( ch , arg+i ); + return '0'; + } + } + } + printVerbose( "Unknown command argument option : " ); + printVerbose( arg ); + printVerbose( ); + printAutoUsage(); + return '0'; +} + +int +AnyOption::parseGNU( char *arg ) +{ + int split_at = 0; + /* if has a '=' sign get value */ + for( unsigned int i = 0 ; i < strlen(arg) ; i++ ){ + if(arg[i] == equalsign ){ + split_at = i ; /* store index */ + i = strlen(arg); /* get out of loop */ + } + } + if( split_at > 0 ){ /* it is an option value pair */ + char* tmp = (char*) malloc( (split_at+1)*sizeof(char) ); + for( int i = 0 ; i < split_at ; i++ ) + tmp[i] = arg[i]; + tmp[split_at] = '\0'; + + if ( matchOpt( tmp ) >= 0 ){ + setValue( options[matchOpt(tmp)] , arg+split_at+1 ); + free (tmp); + }else{ + printVerbose( "Unknown command argument option : " ); + printVerbose( arg ); + printVerbose( ); + printAutoUsage(); + free (tmp); + return -1; + } + }else{ /* regular options with no '=' sign */ + return matchOpt(arg); + } + return -1; +} + + +int +AnyOption::matchOpt( char *opt ) +{ + for( int i = 0 ; i < option_counter ; i++ ){ + if( strcmp( options[i], opt ) == 0 ){ + if( optiontype[i] == COMMON_OPT || + optiontype[i] == COMMAND_OPT ) + { /* found option return index */ + return i; + }else if( optiontype[i] == COMMON_FLAG || + optiontype[i] == COMMAND_FLAG ) + { /* found flag, set it */ + setFlagOn( opt ); + return -1; + } + } + } + printVerbose( "Unknown command argument option : " ); + printVerbose( opt ) ; + printVerbose( ); + printAutoUsage(); + return -1; +} +bool +AnyOption::matchChar( char c ) +{ + for( int i = 0 ; i < optchar_counter ; i++ ){ + if( optionchars[i] == c ) { /* found match */ + if(optchartype[i] == COMMON_OPT || + optchartype[i] == COMMAND_OPT ) + { /* an option store and stop scanning */ + return true; + }else if( optchartype[i] == COMMON_FLAG || + optchartype[i] == COMMAND_FLAG ) { /* a flag store and keep scanning */ + setFlagOn( c ); + return false; + } + } + } + printVerbose( "Unknown command argument option : " ); + printVerbose( c ) ; + printVerbose( ); + printAutoUsage(); + return false; +} + +bool +AnyOption::valueStoreOK( ) +{ + int size= 0; + if( !set ){ + if( g_value_counter > 0 ){ + size = g_value_counter * sizeof(char*); + values = (char**)malloc( size ); + for( int i = 0 ; i < g_value_counter ; i++) + values[i] = NULL; + set = true; + } + } + return set; +} + +/* + * public get methods + */ +char* +AnyOption::getValue( const char *option ) +{ + if( !valueStoreOK() ) + return NULL; + + for( int i = 0 ; i < option_counter ; i++ ){ + if( strcmp( options[i], option ) == 0 ) + return values[ optionindex[i] ]; + } + return NULL; +} + +bool +AnyOption::getFlag( const char *option ) +{ + if( !valueStoreOK() ) + return false; + for( int i = 0 ; i < option_counter ; i++ ){ + if( strcmp( options[i], option ) == 0 ) + return findFlag( values[ optionindex[i] ] ); + } + return false; +} + +char* +AnyOption::getValue( char option ) +{ + if( !valueStoreOK() ) + return NULL; + for( int i = 0 ; i < optchar_counter ; i++ ){ + if( optionchars[i] == option ) + return values[ optcharindex[i] ]; + } + return NULL; +} + +bool +AnyOption::getFlag( char option ) +{ + if( !valueStoreOK() ) + return false; + for( int i = 0 ; i < optchar_counter ; i++ ){ + if( optionchars[i] == option ) + return findFlag( values[ optcharindex[i] ] ) ; + } + return false; +} + +bool +AnyOption::findFlag( char* val ) +{ + if( val == NULL ) + return false; + + if( strcmp( TRUE_FLAG , val ) == 0 ) + return true; + + return false; +} + +/* + * private set methods + */ +bool +AnyOption::setValue( const char *option , char *value ) +{ + if( !valueStoreOK() ) + return false; + for( int i = 0 ; i < option_counter ; i++ ){ + if( strcmp( options[i], option ) == 0 ){ + values[ optionindex[i] ] = (char*) malloc((strlen(value)+1)*sizeof(char)); + strcpy( values[ optionindex[i] ], value ); + return true; + } + } + return false; +} + +bool +AnyOption::setFlagOn( const char *option ) +{ + if( !valueStoreOK() ) + return false; + for( int i = 0 ; i < option_counter ; i++ ){ + if( strcmp( options[i], option ) == 0 ){ + values[ optionindex[i] ] = (char*) malloc((strlen(TRUE_FLAG)+1)*sizeof(char)); + strcpy( values[ optionindex[i] ] , TRUE_FLAG ); + return true; + } + } + return false; +} + +bool +AnyOption::setValue( char option , char *value ) +{ + if( !valueStoreOK() ) + return false; + for( int i = 0 ; i < optchar_counter ; i++ ){ + if( optionchars[i] == option ){ + values[ optcharindex[i] ] = (char*) malloc((strlen(value)+1)*sizeof(char)); + strcpy( values[ optcharindex[i] ], value ); + return true; + } + } + return false; +} + +bool +AnyOption::setFlagOn( char option ) +{ + if( !valueStoreOK() ) + return false; + for( int i = 0 ; i < optchar_counter ; i++ ){ + if( optionchars[i] == option ){ + values[ optcharindex[i] ] = (char*) malloc((strlen(TRUE_FLAG)+1)*sizeof(char)); + strcpy( values[ optcharindex[i] ] , TRUE_FLAG ); + return true; + } + } + return false; +} + + +int +AnyOption::getArgc( ) +{ + return new_argc; +} + +char* +AnyOption::getArgv( int index ) +{ + if( index < new_argc ){ + return ( argv[ new_argv[ index ] ] ); + } + return NULL; +} + +/* dotfile sub routines */ + +bool +AnyOption::processFile() +{ + if( ! (valueStoreOK() && FileSet()) ) + return false; + return ( consumeFile(readFile()) ); +} + +bool +AnyOption::processFile( const char *filename ) +{ + useFiileName(filename ); + return ( processFile() ); +} + +char* +AnyOption::readFile() +{ + return ( readFile(filename) ); +} + +/* + * read the file contents to a character buffer + */ + +char* +AnyOption::readFile( const char* fname ) +{ + int length; + char *buffer; + ifstream is; + is.open ( fname , ifstream::in ); + if( ! is.good() ){ + is.close(); + return NULL; + } + is.seekg (0, ios::end); + length = is.tellg(); + is.seekg (0, ios::beg); + buffer = (char*) malloc(length*sizeof(char)); + is.read (buffer,length); + is.close(); + return buffer; +} + +/* + * scans a char* buffer for lines that does not + * start with the specified comment character. + */ +bool +AnyOption::consumeFile( char *buffer ) +{ + + if( buffer == NULL ) + return false; + + char *cursor = buffer;/* preserve the ptr */ + char *pline = NULL ; + int linelength = 0; + bool newline = true; + for( unsigned int i = 0 ; i < strlen( buffer ) ; i++ ){ + if( *cursor == endofline ) { /* end of line */ + if( pline != NULL ) /* valid line */ + processLine( pline, linelength ); + pline = NULL; + newline = true; + }else if( newline ){ /* start of line */ + newline = false; + if( (*cursor != comment ) ){ /* not a comment */ + pline = cursor ; + linelength = 0 ; + } + } + cursor++; /* keep moving */ + linelength++; + } + free (buffer); + return true; +} + + +/* + * find a valid type value pair separated by a delimiter + * character and pass it to valuePairs() + * any line which is not valid will be considered a value + * and will get passed on to justValue() + * + * assuming delimiter is ':' the behaviour will be, + * + * width:10 - valid pair valuePairs( width, 10 ); + * width : 10 - valid pair valuepairs( width, 10 ); + * + * :::: - not valid + * width - not valid + * :10 - not valid + * width: - not valid + * :: - not valid + * : - not valid + * + */ + +void +AnyOption::processLine( char *theline, int length ) +{ + bool found = false; + char *pline = (char*) malloc( (length+1)*sizeof(char) ); + for( int i = 0 ; i < length ; i ++ ) + pline[i]= *(theline++); + pline[length] = nullterminate; + char *cursor = pline ; /* preserve the ptr */ + if( *cursor == delimiter || *(cursor+length-1) == delimiter ){ + justValue( pline );/* line with start/end delimiter */ + }else{ + for( int i = 1 ; i < length-1 && !found ; i++){/* delimiter */ + if( *cursor == delimiter ){ + *(cursor-1) = nullterminate; /* two strings */ + found = true; + valuePairs( pline , cursor+1 ); + } + cursor++; + } + cursor++; + if( !found ) /* not a pair */ + justValue( pline ); + } + free (pline); +} + +/* + * removes trailing and preceeding whitespaces from a string + */ +char* +AnyOption::chomp( char *str ) +{ + while( *str == whitespace ) + str++; + char *end = str+strlen(str)-1; + while( *end == whitespace ) + end--; + *(end+1) = nullterminate; + return str; +} + +void +AnyOption::valuePairs( char *type, char *value ) +{ + if ( strlen(chomp(type)) == 1 ){ /* this is a char option */ + for( int i = 0 ; i < optchar_counter ; i++ ){ + if( optionchars[i] == type[0] ){ /* match */ + if( optchartype[i] == COMMON_OPT || + optchartype[i] == FILE_OPT ) + { + setValue( type[0] , chomp(value) ); + return; + } + } + } + } + /* if no char options matched */ + for( int i = 0 ; i < option_counter ; i++ ){ + if( strcmp( options[i], type ) == 0 ){ /* match */ + if( optiontype[i] == COMMON_OPT || + optiontype[i] == FILE_OPT ) + { + setValue( type , chomp(value) ); + return; + } + } + } + printVerbose( "Unknown option in resourcefile : " ); + printVerbose( type ); + printVerbose( ); +} + +void +AnyOption::justValue( char *type ) +{ + + if ( strlen(chomp(type)) == 1 ){ /* this is a char option */ + for( int i = 0 ; i < optchar_counter ; i++ ){ + if( optionchars[i] == type[0] ){ /* match */ + if( optchartype[i] == COMMON_FLAG || + optchartype[i] == FILE_FLAG ) + { + setFlagOn( type[0] ); + return; + } + } + } + } + /* if no char options matched */ + for( int i = 0 ; i < option_counter ; i++ ){ + if( strcmp( options[i], type ) == 0 ){ /* match */ + if( optiontype[i] == COMMON_FLAG || + optiontype[i] == FILE_FLAG ) + { + setFlagOn( type ); + return; + } + } + } + printVerbose( "Unknown option in resourcefile : " ); + printVerbose( type ); + printVerbose( ); +} + +/* + * usage and help + */ + + +void +AnyOption::printAutoUsage() +{ + if( autousage ) printUsage(); +} + +void +AnyOption::printUsage() +{ + + if( once ) { + once = false ; + cout << endl ; + for( int i = 0 ; i < usage_lines ; i++ ) + cout << usage[i] << endl ; + cout << endl ; + } +} + + +void +AnyOption::addUsage( const char *line ) +{ + if( usage_lines >= max_usage_lines ){ + if( doubleUsageStorage() == false ){ + addUsageError( line ); + exit(1); + } + } + usage[ usage_lines ] = line ; + usage_lines++; +} + +void +AnyOption::addUsageError( const char *line ) +{ + cout << endl ; + cout << "OPTIONS ERROR : Failed allocating extra memory " << endl ; + cout << "While adding the usage/help : \""<< line << "\"" << endl; + cout << "Exiting." << endl ; + cout << endl ; + exit(0); + +} diff --git a/app/anyoption.h b/app/anyoption.h new file mode 100644 index 0000000..3f7a5de --- /dev/null +++ b/app/anyoption.h @@ -0,0 +1,270 @@ +#ifndef _ANYOPTION_H +#define _ANYOPTION_H + +#include +#include +#include +#include + +#define COMMON_OPT 1 +#define COMMAND_OPT 2 +#define FILE_OPT 3 +#define COMMON_FLAG 4 +#define COMMAND_FLAG 5 +#define FILE_FLAG 6 + +#define COMMAND_OPTION_TYPE 1 +#define COMMAND_FLAG_TYPE 2 +#define FILE_OPTION_TYPE 3 +#define FILE_FLAG_TYPE 4 +#define UNKNOWN_TYPE 5 + +#define DEFAULT_MAXOPTS 10 +#define MAX_LONG_PREFIX_LENGTH 2 + +#define DEFAULT_MAXUSAGE 3 +#define DEFAULT_MAXHELP 10 + +#define TRUE_FLAG "true" + +using namespace std; + +class AnyOption +{ + +public: /* the public interface */ + AnyOption(); + AnyOption(int maxoptions ); + AnyOption(int maxoptions , int maxcharoptions); + ~AnyOption(); + + /* + * following set methods specifies the + * special characters and delimiters + * if not set traditional defaults will be used + */ + + void setCommandPrefixChar( char _prefix ); /* '-' in "-w" */ + void setCommandLongPrefix( char *_prefix ); /* '--' in "--width" */ + void setFileCommentChar( char _comment ); /* '#' in shellscripts */ + void setFileDelimiterChar( char _delimiter );/* ':' in "width : 100" */ + + /* + * provide the input for the options + * like argv[] for commndline and the + * option file name to use; + */ + + void useCommandArgs( int _argc, char **_argv ); + void useFiileName( const char *_filename ); + + /* + * turn off the POSIX style options + * this means anything starting with a '-' or "--" + * will be considered a valid option + * which alo means you cannot add a bunch of + * POIX options chars together like "-lr" for "-l -r" + * + */ + + void noPOSIX(); + + /* + * prints warning verbose if you set anything wrong + */ + void setVerbose(); + + + /* + * there are two types of options + * + * Option - has an associated value ( -w 100 ) + * Flag - no value, just a boolean flag ( -nogui ) + * + * the options can be either a string ( GNU style ) + * or a character ( traditional POSIX style ) + * or both ( --width, -w ) + * + * the options can be common to the commandline and + * the optionfile, or can belong only to either of + * commandline and optionfile + * + * following set methods, handle all the aboove + * cases of options. + */ + + /* options comman to command line and option file */ + void setOption( const char *opt_string ); + void setOption( char opt_char ); + void setOption( const char *opt_string , char opt_char ); + void setFlag( const char *opt_string ); + void setFlag( char opt_char ); + void setFlag( const char *opt_string , char opt_char ); + + /* options read from commandline only */ + void setCommandOption( const char *opt_string ); + void setCommandOption( char opt_char ); + void setCommandOption( const char *opt_string , char opt_char ); + void setCommandFlag( const char *opt_string ); + void setCommandFlag( char opt_char ); + void setCommandFlag( const char *opt_string , char opt_char ); + + /* options read from an option file only */ + void setFileOption( const char *opt_string ); + void setFileOption( char opt_char ); + void setFileOption( const char *opt_string , char opt_char ); + void setFileFlag( const char *opt_string ); + void setFileFlag( char opt_char ); + void setFileFlag( const char *opt_string , char opt_char ); + + /* + * process the options, registerd using + * useCommandArgs() and useFileName(); + */ + void processOptions(); + void processCommandArgs(); + void processCommandArgs( int max_args ); + bool processFile(); + + /* + * process the specified options + */ + void processCommandArgs( int _argc, char **_argv ); + void processCommandArgs( int _argc, char **_argv, int max_args ); + bool processFile( const char *_filename ); + + /* + * get the value of the options + * will return NULL if no value is set + */ + char *getValue( const char *_option ); + bool getFlag( const char *_option ); + char *getValue( char _optchar ); + bool getFlag( char _optchar ); + + /* + * Print Usage + */ + void printUsage(); + void printAutoUsage(); + void addUsage( const char *line ); + void printHelp(); + /* print auto usage printing for unknown options or flag */ + void autoUsagePrint(bool flag); + + /* + * get the argument count and arguments sans the options + */ + int getArgc(); + char* getArgv( int index ); + bool hasOptions(); + +private: /* the hidden data structure */ + int argc; /* commandline arg count */ + char **argv; /* commndline args */ + const char* filename; /* the option file */ + char* appname; /* the application name from argv[0] */ + + int *new_argv; /* arguments sans options (index to argv) */ + int new_argc; /* argument count sans the options */ + int max_legal_args; /* ignore extra arguments */ + + + /* option strings storage + indexing */ + int max_options; /* maximum number of options */ + const char **options; /* storage */ + int *optiontype; /* type - common, command, file */ + int *optionindex; /* index into value storage */ + int option_counter; /* counter for added options */ + + /* option chars storage + indexing */ + int max_char_options; /* maximum number options */ + char *optionchars; /* storage */ + int *optchartype; /* type - common, command, file */ + int *optcharindex; /* index into value storage */ + int optchar_counter; /* counter for added options */ + + /* values */ + char **values; /* common value storage */ + int g_value_counter; /* globally updated value index LAME! */ + + /* help and usage */ + const char **usage; /* usage */ + int max_usage_lines; /* max usage lines reseverd */ + int usage_lines; /* number of usage lines */ + + bool command_set; /* if argc/argv were provided */ + bool file_set; /* if a filename was provided */ + bool mem_allocated; /* if memory allocated in init() */ + bool posix_style; /* enables to turn off POSIX style options */ + bool verbose; /* silent|verbose */ + bool print_usage; /* usage verbose */ + bool print_help; /* help verbose */ + + char opt_prefix_char; /* '-' in "-w" */ + char long_opt_prefix[MAX_LONG_PREFIX_LENGTH]; /* '--' in "--width" */ + char file_delimiter_char; /* ':' in width : 100 */ + char file_comment_char; /* '#' in "#this is a comment" */ + char equalsign; + char comment; + char delimiter; + char endofline; + char whitespace; + char nullterminate; + + bool set; //was static member + bool once; //was static member + + bool hasoptions; + bool autousage; + +private: /* the hidden utils */ + void init(); + void init(int maxopt, int maxcharopt ); + bool alloc(); + void cleanup(); + bool valueStoreOK(); + + /* grow storage arrays as required */ + bool doubleOptStorage(); + bool doubleCharStorage(); + bool doubleUsageStorage(); + + bool setValue( const char *option , char *value ); + bool setFlagOn( const char *option ); + bool setValue( char optchar , char *value); + bool setFlagOn( char optchar ); + + void addOption( const char* option , int type ); + void addOption( char optchar , int type ); + void addOptionError( const char *opt); + void addOptionError( char opt); + bool findFlag( char* value ); + void addUsageError( const char *line ); + bool CommandSet(); + bool FileSet(); + bool POSIX(); + + char parsePOSIX( char* arg ); + int parseGNU( char *arg ); + bool matchChar( char c ); + int matchOpt( char *opt ); + + /* dot file methods */ + char *readFile(); + char *readFile( const char* fname ); + bool consumeFile( char *buffer ); + void processLine( char *theline, int length ); + char *chomp( char *str ); + void valuePairs( char *type, char *value ); + void justValue( char *value ); + + void printVerbose( const char *msg ); + void printVerbose( char *msg ); + void printVerbose( char ch ); + void printVerbose( ); + + +}; + +#endif /* ! _ANYOPTION_H */ diff --git a/app/fileLumiGRLCreator.cxx b/app/fileLumiGRLCreator.cxx new file mode 100644 index 0000000..514ee35 --- /dev/null +++ b/app/fileLumiGRLCreator.cxx @@ -0,0 +1,246 @@ +// $Id: fileLumiGRLCreator.cxx 183599 2011-07-11 11:40:30Z krasznaa $ + +// ROOT include(s): +#include "TString.h" +#include "TObjString.h" +#include "TFile.h" + +// SFrame include(s): +#include "core/include/SLogger.h" + +// The option parser code: +#include "anyoption.h" + +// Local include(s): +#include "../include/TGoodRunsList.h" +#include "../include/TGoodRunsListReader.h" +#include "../include/TGoodRunsListWriter.h" + +/// A global logger object +static SLogger m_logger( "fileLumiGRLCreator" ); + +/// The luminosity directory name in the D3PDs +static const char* LUMI_DIR_NAME = "Lumi"; + +namespace { + + /// Helper function for printing vectors + /** + * The tool needs to print the contents of a few vectors. This function + * takes care about doing that conveniently. + * + * @param out The SLogger object to write the contents to + * @param vec The vector that should be printed + * @returns The SLogger object that was given to the functions + */ + template< typename T > + SLogger& operator<< ( SLogger& out, const std::vector< T >& vec ) { + out << "["; + typename std::vector< T >::const_iterator itr = vec.begin(); + typename std::vector< T >::const_iterator end = vec.end(); + for( ; itr != end; ++itr ) { + out << *itr; + if( ++itr != end ) { + out << ", "; + } + --itr; + } + out << "]"; + return out; + } + +} // private namespace + +/// Read the XML strings from each file and add it to the GRL reader +static void ReadLumiInfo( Root::TGoodRunsListReader& grlReader, + const std::vector< TString >& fileNames ); + +int main( int argc, char* argv[] ) { + + // Create the command line parser: + AnyOption opt; + + // Set the usage/help messages: + opt.addUsage( "Application for creating a lumiblock list in the format of" ); + opt.addUsage( "a GRL file from a list of D3PD files plus an actual GRL file." ); + opt.addUsage( "" ); + opt.addUsage( "Usage:" ); + opt.addUsage( " fileLumiGRLCreator [options] file1 file2 ..." ); + opt.addUsage( "" ); + opt.addUsage( "Options:" ); + opt.addUsage( " -h --help Prints this help" ); + opt.addUsage( " -g --grl GRL to use" ); + opt.addUsage( " -o --output Output file (XML or ROOT)" ); + + // Set the options: + opt.setFlag( "help", 'h' ); + opt.setOption( "dataset", 'd' ); + opt.setOption( "grl", 'g' ); + opt.setOption( "output", 'o' ); + opt.setOption( "server", 's' ); + + // Process the command line arguments: + opt.processCommandArgs( argc, argv ); + if( ! opt.hasOptions() ) { + opt.printUsage(); + return 1; + } + + // Check if the user just asked for the help: + if( opt.getFlag( "help" ) || opt.getFlag( 'h' ) ) { + opt.printUsage(); + return 0; + } + + // Check if a GRL was defined: + if( ! opt.getValue( 'g' ) ) { + m_logger << WARNING << "You should define a GRL as well. Creating info just based" + << SLogger::endmsg; + m_logger << WARNING << "on the D3PD files for now..." << SLogger::endmsg; + } + TString grlName( opt.getValue( 'g' ) ? opt.getValue( 'g' ) : "" ); + + // Check if an output file was specified: + if( ! opt.getValue( 'o' ) ) { + m_logger << INFO << "No output file specified. Using 'grl_output.xml'" + << SLogger::endmsg; + } + TString outputName( opt.getValue( 'o' ) ? opt.getValue( 'o' ) : "grl_output.xml" ); + if( ( ! outputName.EndsWith( ".xml" ) ) && + ( ! outputName.EndsWith( ".root" ) ) ) { + REPORT_ERROR( "Output file name's format not recognised. Should end either in '.root' or '.xml'" ); + return 1; + } + + // Collect the file names: + std::vector< TString > fileNames; + for( int i = 0 ; i < opt.getArgc() ; ++i ) { + fileNames.push_back( opt.getArgv( i ) ); + } + if( ! fileNames.size() ) { + REPORT_ERROR( "You should specify at least one D3PD file name!" ); + return 1; + } + + // Tell the user what we're about to do: + m_logger << INFO << "Creating lumiblock list using:" << SLogger::endmsg; + m_logger << INFO << " File names = " << fileNames << SLogger::endmsg; + m_logger << INFO << " GRL file = " << ( grlName == "" ? "NOT DEFINED" : grlName ) + << SLogger::endmsg; + m_logger << INFO << " Output file = " << outputName << SLogger::endmsg; + + // Read in the merged lumiblock info from the dataset: + Root::TGoodRunsListReader lbReader; + ReadLumiInfo( lbReader, fileNames ); + if( ! lbReader.Interpret() ) { + REPORT_ERROR( "Couldn't interpret the collected lumiblock information" ); + return 1; + } + Root::TGoodRunsList d3pdGRL = lbReader.GetMergedGoodRunsList(); + + // Get its overlap with a GRL if one was specified: + if( grlName != "" ) { + // Read in the GRL file: + Root::TGoodRunsListReader grlReader( grlName ); + if( ! grlReader.Interpret() ) { + REPORT_ERROR( "Couldn't interpret the GRL: " << grlName ); + return 1; + } + // Take its overlap with the D3PD lumiblock list: + Root::TGoodRunsList xmlGRL = grlReader.GetMergedGoodRunsList(); + d3pdGRL = d3pdGRL.GetOverlapWith( xmlGRL ); + } + + // Print the info about the collected lumiblocks: + d3pdGRL.Summary(); + + // Set some additional things on the GRL: + d3pdGRL.SetVersion( "20" ); + d3pdGRL.AddMetaData( "Description", "Lumiblock list created from D3PD files" + " plus GRL '" + grlName + "'" ); + + // Turn the GRL into an XML string: + Root::TGoodRunsListWriter writer; + writer.SetGoodRunsList( d3pdGRL ); + + if( outputName.EndsWith( ".xml" ) ) { + m_logger << INFO << "Writing out: " << outputName << SLogger::endmsg; + // Write out the XML file: + writer.SetFilename( outputName ); + if( ! writer.WriteXMLFile() ) { + REPORT_ERROR( "Failed to write out: " << outputName ); + return 1; + } + } else if( outputName.EndsWith( ".root" ) ) { + // Write out this merged/overlapped GRL: + TFile* ofile = TFile::Open( outputName, "RECREATE" ); + + // Create the Lumi directory in it: + TDirectory* lumidir = ofile->mkdir( LUMI_DIR_NAME, LUMI_DIR_NAME ); + lumidir->cd(); + + // The object that will be written out: + TObjString ostring( writer.GetXMLString() ); + + // Write out this object string: + lumidir->WriteObjectAny( &ostring, "TObjString", "physics", "overwrite" ); + + // Close and delete the output file: + ofile->Write(); + ofile->Close(); + delete ofile; + } + + return 0; +} + +void ReadLumiInfo( Root::TGoodRunsListReader& grlReader, + const std::vector< TString >& fileNames ) { + + std::vector< TString >::const_iterator file_itr = fileNames.begin(); + std::vector< TString >::const_iterator file_end = fileNames.end(); + for( ; file_itr != file_end; ++file_itr ) { + + // Open the file: + TFile* file = TFile::Open( *file_itr, "READ" ); + + // Access the Lumi directory in the file: + TDirectoryFile* dir = + dynamic_cast< TDirectoryFile* >( file->GetDirectory( LUMI_DIR_NAME ) ); + if( ! dir ) { + REPORT_ERROR( "Lumi information not found in file: " << *file_itr ); + file->Close(); + delete file; + continue; + } + + // Get a list of all objects in this file: + TList* keyList = dir->GetListOfKeys(); + + // Loop over all these objects: + for( Int_t j = 0; j < keyList->GetEntries(); ++j ) { + + TObjString* ostring = + dynamic_cast< TObjString* >( dir->Get( keyList->At( j )->GetName() ) ); + if( ! ostring ) continue; + + // Check that it has the right format: + if( ostring->GetString().BeginsWith( "GetString().Contains( "DOCTYPE LumiRangeCollection" ) ) { + + // If it is, let's add it to the reader: + grlReader.AddXMLString( ostring->GetString() ); + } else { + m_logger << WARNING << "Lumi string not in the expected format: " + << ostring->GetString() << SLogger::endmsg; + } + + } + + // Close and delete the file: + file->Close(); + delete file; + } + + return; +} diff --git a/app/pq2LumiGRLCreator.cxx b/app/pq2LumiGRLCreator.cxx new file mode 100644 index 0000000..dbfd110 --- /dev/null +++ b/app/pq2LumiGRLCreator.cxx @@ -0,0 +1,291 @@ +// $Id: pq2LumiGRLCreator.cxx 183599 2011-07-11 11:40:30Z krasznaa $ + +// ROOT include(s): +#include "TString.h" +#include "TProof.h" +#include "TFileCollection.h" +#include "TFileInfo.h" +#include "THashList.h" +#include "TFile.h" +#include "TDirectoryFile.h" + +// SFrame include(s): +#include "core/include/SLogger.h" +#include "core/include/SError.h" +#include "core/include/SProofManager.h" + +// The option parser code: +#include "anyoption.h" + +// Local include(s): +#include "../include/TGoodRunsList.h" +#include "../include/TGoodRunsListReader.h" +#include "../include/TGoodRunsListWriter.h" + +/// A global logger object +static SLogger m_logger( "pq2LumiGRLCreator" ); + +/// The luminosity directory name in the D3PDs +static const char* LUMI_DIR_NAME = "Lumi"; + +/// Read the XML strings from each file and add it to the GRL reader +static void ReadLumiInfo( Root::TGoodRunsListReader& grlReader, const TString& server, + const TString& dataset ) throw( SError ); + +/** + * @short Application for creating lumiblock lists off of PQ2 datasets + * + * This application can be used to collect information about PROOF (PQ2) + * datasets. The PQ2 dataset is expected to contain only D3PD files that + * have luminosity metadata in them. + * + * The application opens each file in a given dataset, collects the + * lumiblock information from each of them, possibly ANDs it with + * a GRL, and writes out the results into a ROOT or XML file. + * + * The resulting file can then be used to calculate the exact luminosity + * for processing the PQ2 dataset using the GRL file that was used + * with this application. + * + * @author Attila Krasznahorkay + * + * $Revision: 183599 $ + * $Date: 2011-07-11 13:40:30 +0200 (Mon, 11 Jul 2011) $ + */ +int main( int argc, char* argv[] ) { + + // Create the command line parser: + AnyOption opt; + + // Set the usage/help messages: + opt.addUsage( "Application for creating a lumiblock list in the format of" ); + opt.addUsage( "a GRL file from a PQ2 dataset plus an actual GRL file." ); + opt.addUsage( "" ); + opt.addUsage( "Usage:" ); + opt.addUsage( "" ); + opt.addUsage( " -h --help Prints this help" ); + opt.addUsage( " -d --dataset PQ2 dataset to use" ); + opt.addUsage( " -g --grl GRL to use" ); + opt.addUsage( " -o --output Output file (XML or ROOT)" ); + opt.addUsage( " -s --server PROOF server name" ); + + // Set the options: + opt.setFlag( "help", 'h' ); + opt.setOption( "dataset", 'd' ); + opt.setOption( "grl", 'g' ); + opt.setOption( "output", 'o' ); + opt.setOption( "server", 's' ); + + // Process the command line arguments: + opt.processCommandArgs( argc, argv ); + if( ! opt.hasOptions() ) { + opt.printUsage(); + return 1; + } + + // Check if the user just asked for the help: + if( opt.getFlag( "help" ) || opt.getFlag( 'h' ) ) { + opt.printUsage(); + return 0; + } + + // Check if a dataset was defined: + if( ! opt.getValue( 'd' ) ) { + REPORT_FATAL( "You have to define a PQ2 dataset name" ); + opt.printUsage(); + return 1; + } + TString dsetName( opt.getValue( 'd' ) ); + + // Check if a GRL was defined: + if( ! opt.getValue( 'g' ) ) { + m_logger << WARNING << "You should define a GRL as well. Creating info just based" + << SLogger::endmsg; + m_logger << WARNING << "on the D3PD files for now..." << SLogger::endmsg; + } + TString grlName( opt.getValue( 'g' ) ? opt.getValue( 'g' ) : "" ); + + // Check if an output file was specified: + if( ! opt.getValue( 'o' ) ) { + m_logger << INFO << "No output file specified. Using 'grl_output.xml'" + << SLogger::endmsg; + } + TString outputName( opt.getValue( 'o' ) ? opt.getValue( 'o' ) : "grl_output.xml" ); + + // Remove the extension from the file name: + if( ( ! outputName.EndsWith( ".xml" ) ) && + ( ! outputName.EndsWith( ".root" ) ) ) { + REPORT_ERROR( "Output file name's format not recognised. Should end either in '.root' or '.xml'" ); + return 1; + } + + // Check if a PROOF server was specified: + if( ! opt.getValue( 's' ) ) { + REPORT_FATAL( "You have to specify a PROOF server name!" ); + return 1; + } + TString serverName( opt.getValue( 's' ) ); + + // Tell the user what we're about to do: + m_logger << INFO << "Creating lumiblock list using:" << SLogger::endmsg; + m_logger << INFO << " PQ2 dataset = " << dsetName << SLogger::endmsg; + m_logger << INFO << " GRL file = " << ( grlName == "" ? "NOT DEFINED" : grlName ) + << SLogger::endmsg; + m_logger << INFO << " Output file = " << outputName << SLogger::endmsg; + + // Read in the merged lumiblock info from the dataset: + Root::TGoodRunsListReader lbReader; + ReadLumiInfo( lbReader, serverName, dsetName ); + if( ! lbReader.Interpret() ) { + REPORT_ERROR( "Couldn't interpret the collected lumiblock information" ); + return 1; + } + Root::TGoodRunsList d3pdGRL = lbReader.GetMergedGoodRunsList(); + + // Get its overlap with a GRL if one was specified: + if( grlName != "" ) { + // Read in the GRL file: + Root::TGoodRunsListReader grlReader( grlName ); + if( ! grlReader.Interpret() ) { + REPORT_ERROR( "Couldn't interpret the GRL: " << grlName ); + return 1; + } + // Take its overlap with the D3PD lumiblock list: + Root::TGoodRunsList xmlGRL = grlReader.GetMergedGoodRunsList(); + d3pdGRL = d3pdGRL.GetOverlapWith( xmlGRL ); + } + + // Print the info about the collected lumiblocks: + d3pdGRL.Summary(); + + // Set some additional things on the GRL: + d3pdGRL.SetVersion( "20" ); + d3pdGRL.AddMetaData( "Description", "Lumiblock list created from dataset '" + + dsetName + "' plus GRL '" + grlName + "'" ); + + // Turn the GRL into an XML string: + Root::TGoodRunsListWriter writer; + writer.SetGoodRunsList( d3pdGRL ); + + if( outputName.EndsWith( ".xml" ) ) { + m_logger << INFO << "Writing out: " << outputName << SLogger::endmsg; + // Write out the XML file: + writer.SetFilename( outputName ); + if( ! writer.WriteXMLFile() ) { + REPORT_ERROR( "Failed to write out: " << outputName ); + return 1; + } + } else if( outputName.EndsWith( ".root" ) ) { + // Write out this merged/overlapped GRL: + TFile* ofile = TFile::Open( outputName, "RECREATE" ); + + // Create the Lumi directory in it: + TDirectory* lumidir = ofile->mkdir( LUMI_DIR_NAME, LUMI_DIR_NAME ); + lumidir->cd(); + + // The object that will be written out: + TObjString ostring( writer.GetXMLString() ); + + // Write out this object string: + lumidir->WriteObjectAny( &ostring, "TObjString", "physics", "overwrite" ); + + // Close and delete the output file: + ofile->Write(); + ofile->Close(); + delete ofile; + } + + // Close the PROOF connection(s): + SProofManager::Instance()->Cleanup(); + + return 0; +} + +/** + * This function is used to extract the XML strings out of all the files that are in + * a PQ2 dataset. + * + * @param grlReader The GRL reader to which the XML strings should be defined + * @param server The name of the PROOF server + * @param dataset The name of the dataset on the PROOF server + */ +void ReadLumiInfo( Root::TGoodRunsListReader& grlReader, const TString& server, + const TString& dataset ) throw( SError ) { + + // Connect to the PROOF server: + TProof* pserver = SProofManager::Instance()->Open( server, "masteronly" ); + if( ! pserver ) { + REPORT_FATAL( "Couldn't connect to PROOF server: " << server ); + throw SError( "Couldn't connect to PROOF server", + SError::StopExecution ); + } + + // Check if the dataset even exists: + if( ! pserver->ExistsDataSet( dataset ) ) { + REPORT_ERROR( "The dataset (" << dataset << ") doesn't exist" ); + return; + } + + // Access the dataset: + TFileCollection* dset = pserver->GetDataSet( dataset ); + // Get the file list out of it: + THashList* flist = dset->GetList(); + + // Give some info: + m_logger << INFO << "Extracting lumiblock information out of " + << flist->GetEntries() << " files" << SLogger::endmsg; + + // List the file names: + for( Int_t i = 0; i < flist->GetEntries(); ++i ) { + TFileInfo* finfo = dynamic_cast< TFileInfo* >( flist->At( i ) ); + if( ! finfo ) { + REPORT_FATAL( "The dataset doesn't have the expected format" ); + throw SError( "The dataset doesn't have the expected format", + SError::StopExecution ); + } + TUrl* url = finfo->GetCurrentUrl(); + m_logger << DEBUG << "Processing file: " << url->GetUrl() << SLogger::endmsg; + + // Open the file: + TFile* file = TFile::Open( url->GetUrl(), "READ" ); + + // Access the Lumi directory in the file: + TDirectoryFile* dir = + dynamic_cast< TDirectoryFile* >( file->GetDirectory( LUMI_DIR_NAME ) ); + if( ! dir ) { + REPORT_ERROR( "Lumi information not found in file: " << url->GetUrl() ); + file->Close(); + delete file; + continue; + } + + // Get a list of all objects in this file: + TList* keyList = dir->GetListOfKeys(); + + // Loop over all these objects: + for( Int_t j = 0; j < keyList->GetEntries(); ++j ) { + + TObjString* ostring = + dynamic_cast< TObjString* >( dir->Get( keyList->At( j )->GetName() ) ); + if( ! ostring ) continue; + + // Check that it has the right format: + if( ostring->GetString().BeginsWith( "GetString().Contains( "DOCTYPE LumiRangeCollection" ) ) { + + // If it is, let's add it to the reader: + grlReader.AddXMLString( ostring->GetString() ); + } else { + m_logger << WARNING << "Lumi string not in the expected format: " + << ostring->GetString() << SLogger::endmsg; + } + + } + + // Close and delete the file: + file->Close(); + delete file; + } + + return; +}