| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230 | /* *  Created by Phil on 02/11/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) */#include "catch_commandline.h"#include "catch_string_manip.h"#include "catch_interfaces_registry_hub.h"#include "catch_interfaces_reporter.h"#include <fstream>#include <ctime>namespace Catch {    clara::Parser makeCommandLineParser( ConfigData& config ) {        using namespace clara;        auto const setWarning = [&]( std::string const& warning ) {                auto warningSet = [&]() {                    if( warning == "NoAssertions" )                        return WarnAbout::NoAssertions;                    if ( warning == "NoTests" )                        return WarnAbout::NoTests;                    return WarnAbout::Nothing;                }();                if (warningSet == WarnAbout::Nothing)                    return ParserResult::runtimeError( "Unrecognised warning: '" + warning + "'" );                config.warnings = static_cast<WarnAbout::What>( config.warnings | warningSet );                return ParserResult::ok( ParseResultType::Matched );            };        auto const loadTestNamesFromFile = [&]( std::string const& filename ) {                std::ifstream f( filename.c_str() );                if( !f.is_open() )                    return ParserResult::runtimeError( "Unable to load input file: '" + filename + "'" );                std::string line;                while( std::getline( f, line ) ) {                    line = trim(line);                    if( !line.empty() && !startsWith( line, '#' ) ) {                        if( !startsWith( line, '"' ) )                            line = '"' + line + '"';                        config.testsOrTags.push_back( line );                        config.testsOrTags.emplace_back( "," );                    }                }                //Remove comma in the end                if(!config.testsOrTags.empty())                    config.testsOrTags.erase( config.testsOrTags.end()-1 );                return ParserResult::ok( ParseResultType::Matched );            };        auto const setTestOrder = [&]( std::string const& order ) {                if( startsWith( "declared", order ) )                    config.runOrder = RunTests::InDeclarationOrder;                else if( startsWith( "lexical", order ) )                    config.runOrder = RunTests::InLexicographicalOrder;                else if( startsWith( "random", order ) )                    config.runOrder = RunTests::InRandomOrder;                else                    return clara::ParserResult::runtimeError( "Unrecognised ordering: '" + order + "'" );                return ParserResult::ok( ParseResultType::Matched );            };        auto const setRngSeed = [&]( std::string const& seed ) {                if( seed != "time" )                    return clara::detail::convertInto( seed, config.rngSeed );                config.rngSeed = static_cast<unsigned int>( std::time(nullptr) );                return ParserResult::ok( ParseResultType::Matched );            };        auto const setColourUsage = [&]( std::string const& useColour ) {                    auto mode = toLower( useColour );                    if( mode == "yes" )                        config.useColour = UseColour::Yes;                    else if( mode == "no" )                        config.useColour = UseColour::No;                    else if( mode == "auto" )                        config.useColour = UseColour::Auto;                    else                        return ParserResult::runtimeError( "colour mode must be one of: auto, yes or no. '" + useColour + "' not recognised" );                return ParserResult::ok( ParseResultType::Matched );            };        auto const setWaitForKeypress = [&]( std::string const& keypress ) {                auto keypressLc = toLower( keypress );                if (keypressLc == "never")                    config.waitForKeypress = WaitForKeypress::Never;                else if( keypressLc == "start" )                    config.waitForKeypress = WaitForKeypress::BeforeStart;                else if( keypressLc == "exit" )                    config.waitForKeypress = WaitForKeypress::BeforeExit;                else if( keypressLc == "both" )                    config.waitForKeypress = WaitForKeypress::BeforeStartAndExit;                else                    return ParserResult::runtimeError( "keypress argument must be one of: never, start, exit or both. '" + keypress + "' not recognised" );            return ParserResult::ok( ParseResultType::Matched );            };        auto const setVerbosity = [&]( std::string const& verbosity ) {            auto lcVerbosity = toLower( verbosity );            if( lcVerbosity == "quiet" )                config.verbosity = Verbosity::Quiet;            else if( lcVerbosity == "normal" )                config.verbosity = Verbosity::Normal;            else if( lcVerbosity == "high" )                config.verbosity = Verbosity::High;            else                return ParserResult::runtimeError( "Unrecognised verbosity, '" + verbosity + "'" );            return ParserResult::ok( ParseResultType::Matched );        };        auto const setReporter = [&]( std::string const& reporter ) {            IReporterRegistry::FactoryMap const& factories = getRegistryHub().getReporterRegistry().getFactories();            auto lcReporter = toLower( reporter );            auto result = factories.find( lcReporter );            if( factories.end() != result )                config.reporterName = lcReporter;            else                return ParserResult::runtimeError( "Unrecognized reporter, '" + reporter + "'. Check available with --list-reporters" );            return ParserResult::ok( ParseResultType::Matched );        };        auto cli            = ExeName( config.processName )            | Help( config.showHelp )            | Opt( config.listTests )                ["-l"]["--list-tests"]                ( "list all/matching test cases" )            | Opt( config.listTags )                ["-t"]["--list-tags"]                ( "list all/matching tags" )            | Opt( config.showSuccessfulTests )                ["-s"]["--success"]                ( "include successful tests in output" )            | Opt( config.shouldDebugBreak )                ["-b"]["--break"]                ( "break into debugger on failure" )            | Opt( config.noThrow )                ["-e"]["--nothrow"]                ( "skip exception tests" )            | Opt( config.showInvisibles )                ["-i"]["--invisibles"]                ( "show invisibles (tabs, newlines)" )            | Opt( config.outputFilename, "filename" )                ["-o"]["--out"]                ( "output filename" )            | Opt( setReporter, "name" )                ["-r"]["--reporter"]                ( "reporter to use (defaults to console)" )            | Opt( config.name, "name" )                ["-n"]["--name"]                ( "suite name" )            | Opt( [&]( bool ){ config.abortAfter = 1; } )                ["-a"]["--abort"]                ( "abort at first failure" )            | Opt( [&]( int x ){ config.abortAfter = x; }, "no. failures" )                ["-x"]["--abortx"]                ( "abort after x failures" )            | Opt( setWarning, "warning name" )                ["-w"]["--warn"]                ( "enable warnings" )            | Opt( [&]( bool flag ) { config.showDurations = flag ? ShowDurations::Always : ShowDurations::Never; }, "yes|no" )                ["-d"]["--durations"]                ( "show test durations" )            | Opt( config.minDuration, "seconds" )                ["-D"]["--min-duration"]                ( "show test durations for tests taking at least the given number of seconds" )            | Opt( loadTestNamesFromFile, "filename" )                ["-f"]["--input-file"]                ( "load test names to run from a file" )            | Opt( config.filenamesAsTags )                ["-#"]["--filenames-as-tags"]                ( "adds a tag for the filename" )            | Opt( config.sectionsToRun, "section name" )                ["-c"]["--section"]                ( "specify section to run" )            | Opt( setVerbosity, "quiet|normal|high" )                ["-v"]["--verbosity"]                ( "set output verbosity" )            | Opt( config.listTestNamesOnly )                ["--list-test-names-only"]                ( "list all/matching test cases names only" )            | Opt( config.listReporters )                ["--list-reporters"]                ( "list all reporters" )            | Opt( setTestOrder, "decl|lex|rand" )                ["--order"]                ( "test case order (defaults to decl)" )            | Opt( setRngSeed, "'time'|number" )                ["--rng-seed"]                ( "set a specific seed for random numbers" )            | Opt( setColourUsage, "yes|no" )                ["--use-colour"]                ( "should output be colourised" )            | Opt( config.libIdentify )                ["--libidentify"]                ( "report name and version according to libidentify standard" )            | Opt( setWaitForKeypress, "never|start|exit|both" )                ["--wait-for-keypress"]                ( "waits for a keypress before exiting" )            | Opt( config.benchmarkSamples, "samples" )                ["--benchmark-samples"]                ( "number of samples to collect (default: 100)" )            | Opt( config.benchmarkResamples, "resamples" )                ["--benchmark-resamples"]                ( "number of resamples for the bootstrap (default: 100000)" )            | Opt( config.benchmarkConfidenceInterval, "confidence interval" )                ["--benchmark-confidence-interval"]                ( "confidence interval for the bootstrap (between 0 and 1, default: 0.95)" )            | Opt( config.benchmarkNoAnalysis )                ["--benchmark-no-analysis"]                ( "perform only measurements; do not perform any analysis" )            | Opt( config.benchmarkWarmupTime, "benchmarkWarmupTime" )                ["--benchmark-warmup-time"]                ( "amount of time in milliseconds spent on warming up each test (default: 100)" )            | Arg( config.testsOrTags, "test name|pattern|tags" )                ( "which test or tests to use" );        return cli;    }} // end namespace Catch
 |