/* * Created by Phil on 31/10/2010. * Copyright 2010 Two Blue Cubes Ltd. All rights reserved. * * Distributed under the Boost Software License, Version 1.0. (See accompanying * file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) */ #ifndef TWOBLUECUBES_CATCH_RUNNER_HPP_INCLUDED #define TWOBLUECUBES_CATCH_RUNNER_HPP_INCLUDED #include "internal/catch_context_impl.hpp" #include "internal/catch_commandline.hpp" #include "internal/catch_list.hpp" #include "reporters/catch_reporter_basic.hpp" #include "reporters/catch_reporter_xml.hpp" #include "reporters/catch_reporter_junit.hpp" #include #include #include namespace Catch { INTERNAL_CATCH_REGISTER_REPORTER( "basic", BasicReporter ) INTERNAL_CATCH_REGISTER_REPORTER( "xml", XmlReporter ) INTERNAL_CATCH_REGISTER_REPORTER( "junit", JunitReporter ) inline int Main( Config& config ) { // Handle list request if( config.listWhat() != List::None ) return List( config ); // Open output file, if specified std::ofstream ofs; if( !config.getFilename().empty() ) { ofs.open( config.getFilename().c_str() ); if( ofs.fail() ) { std::cerr << "Unable to open file: '" << config.getFilename() << "'" << std::endl; return (std::numeric_limits::max)(); } config.setStreamBuf( ofs.rdbuf() ); } int result = 0; // Scope here for the Runner so it can use the context before it is cleaned-up { Runner runner( config ); // Run test specs specified on the command line - or default to all if( !config.testsSpecified() ) { config.getReporter()->StartGroup( "" ); runner.runAll(); config.getReporter()->EndGroup( "", runner.getTotals() ); } else { // !TBD We should get all the testcases upfront, report any missing, // then just run them std::vector::const_iterator it = config.getTestSpecs().begin(); std::vector::const_iterator itEnd = config.getTestSpecs().end(); for(; it != itEnd; ++it ) { Totals prevTotals = runner.getTotals(); config.getReporter()->StartGroup( *it ); if( runner.runMatching( *it ) == 0 ) { // Use reporter? // std::cerr << "\n[Unable to match any test cases with: " << *it << "]" << std::endl; } config.getReporter()->EndGroup( *it, runner.getTotals() - prevTotals ); } } result = static_cast( runner.getTotals().assertions.failed ); } Catch::Context::cleanUp(); return result; } inline void showUsage( std::ostream& os ) { os << "\t-l, --list [xml]\n" << "\t-t, --test [...]\n" << "\t-r, --reporter \n" << "\t-o, --out |<%stream name>\n" << "\t-s, --success\n" << "\t-b, --break\n" << "\t-n, --name \n" << "\t-a, --abort [#]\n\n" << "\t-nt, --nothrow\n\n" << "For more detail usage please see: https://github.com/philsquared/Catch/wiki/Command-line" << std::endl; } inline void showHelp( std::string exeName ) { std::string::size_type pos = exeName.find_last_of( "/\\" ); if( pos != std::string::npos ) { exeName = exeName.substr( pos+1 ); } std::cout << exeName << " is a CATCH host application. Options are as follows:\n\n"; showUsage( std::cout ); } inline int Main( int argc, char* const argv[], Config& config ) { try { CommandParser parser( argc, argv ); if( Command cmd = parser.find( "-h", "-?", "--help" ) ) { if( cmd.argsCount() != 0 ) cmd.raiseError( "Does not accept arguments" ); showHelp( argv[0] ); Catch::Context::cleanUp(); return 0; } parseIntoConfig( parser, config.data() ); // !TBD: wire up (do this lazily?) if( !config.data().reporter.empty() ) config.setReporter( config.data().reporter ); if( !config.data().stream.empty() ) { if( config.data().stream[0] == '%' ) config.useStream( config.data().stream.substr( 1 ) ); else config.setFilename( config.data().stream ); } } catch( std::exception& ex ) { std::cerr << ex.what() << + "\n\nUsage: ...\n\n"; showUsage( std::cerr ); Catch::Context::cleanUp(); return (std::numeric_limits::max)(); } return Main( config ); } inline int Main( int argc, char* const argv[] ) { Config config; // !TBD: This doesn't always work, for some reason // if( isDebuggerActive() ) // config.useStream( "debug" ); return Main( argc, argv, config ); } } // end namespace Catch #endif // TWOBLUECUBES_CATCH_RUNNER_HPP_INCLUDED