2005-10-12 01:46:19 +02:00
|
|
|
#include "TestRunner.hpp"
|
|
|
|
#include "textui/TextTestResult.h"
|
2008-08-10 00:19:21 +02:00
|
|
|
#include "textui/TableTestResult.hpp"
|
2010-01-11 18:02:11 +01:00
|
|
|
#include "textui/XmlTestResult.hpp"
|
2005-10-12 01:46:19 +02:00
|
|
|
#include <iostream>
|
2010-01-12 10:42:08 +01:00
|
|
|
#include <fstream>
|
2005-10-12 01:46:19 +02:00
|
|
|
|
|
|
|
//////////////////////////////////////////
|
|
|
|
/*
|
|
|
|
* A command line based tool to run tests.
|
|
|
|
* TestRunner expects as its only argument the name of a TestCase class.
|
|
|
|
* TestRunner prints out a trace as the tests are executed followed by a
|
|
|
|
* summary at the end.
|
|
|
|
*
|
|
|
|
* You can add to the tests that the TestRunner knows about by
|
|
|
|
* making additional calls to "addTest (...)" in main.
|
|
|
|
*
|
|
|
|
* Here is the synopsis:
|
|
|
|
*
|
|
|
|
* TestRunner [-wait] ExampleTestCase
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
using namespace std;
|
|
|
|
|
2007-08-29 04:03:12 +02:00
|
|
|
|
2005-10-12 01:46:19 +02:00
|
|
|
typedef pair<string, Test *> mapping;
|
|
|
|
typedef vector<pair<string, Test *> > mappings;
|
|
|
|
|
2008-08-10 00:19:21 +02:00
|
|
|
template<typename result_type>
|
2010-01-12 13:07:32 +01:00
|
|
|
bool run(const string& name, Test *test, bool verbose, const string& logprefix)
|
2007-08-29 04:03:12 +02:00
|
|
|
{
|
|
|
|
if(verbose)
|
|
|
|
cout << "Running " << name << endl;
|
2008-08-10 00:19:21 +02:00
|
|
|
result_type result(name, verbose);
|
2007-08-29 04:03:12 +02:00
|
|
|
test->run (&result);
|
|
|
|
cout << result;
|
2010-01-12 13:07:32 +01:00
|
|
|
if(!logprefix.empty())
|
2010-01-12 10:42:08 +01:00
|
|
|
{
|
2010-01-12 13:07:32 +01:00
|
|
|
std::string filename = logprefix + name + ".log";
|
2010-01-12 10:42:08 +01:00
|
|
|
std::ofstream of(filename.c_str());
|
|
|
|
of << result;
|
|
|
|
of.close();
|
|
|
|
} // if ...
|
2007-12-08 21:46:40 +01:00
|
|
|
return result.wasSuccessful();
|
2007-08-29 04:03:12 +02:00
|
|
|
} // run
|
|
|
|
|
2010-01-12 13:07:32 +01:00
|
|
|
bool textrun(const string& name, Test *test, bool verbose, const string& logprefix)
|
2008-08-10 00:19:21 +02:00
|
|
|
{
|
2010-01-12 13:07:32 +01:00
|
|
|
return run<TextTestResult>(name, test, verbose, logprefix);
|
2008-08-10 00:19:21 +02:00
|
|
|
} // textrun
|
|
|
|
|
2010-01-12 13:07:32 +01:00
|
|
|
bool tablerun(const string& name, Test *test, bool verbose, const string& logprefix)
|
2008-08-10 00:19:21 +02:00
|
|
|
{
|
2010-01-12 13:07:32 +01:00
|
|
|
return run<TableTestResult>(name, test, verbose, logprefix);
|
2008-08-10 00:19:21 +02:00
|
|
|
} // tablerun
|
|
|
|
|
2010-01-12 13:07:32 +01:00
|
|
|
bool xmlrun(const string& name, Test *test, bool verbose, const string& logprefix)
|
2010-01-11 18:02:11 +01:00
|
|
|
{
|
2010-01-12 13:07:32 +01:00
|
|
|
return run<XmlTestResult>(name, test, verbose, logprefix);
|
2010-01-11 18:02:11 +01:00
|
|
|
} // xmlrun
|
|
|
|
|
2008-08-10 00:19:21 +02:00
|
|
|
|
|
|
|
|
2007-08-29 04:03:12 +02:00
|
|
|
void printBanner ()
|
2005-10-12 01:46:19 +02:00
|
|
|
{
|
2008-08-10 00:19:21 +02:00
|
|
|
cout << "Usage: driver [-v] [-table] [ -wait ] testName, where name is the name of a test case class" << endl;
|
2005-10-12 01:46:19 +02:00
|
|
|
} // printBanner
|
|
|
|
|
2010-01-12 13:07:32 +01:00
|
|
|
typedef bool (*runFn)(const string& name, Test *test, bool verbose, const string& logprefix);
|
2008-08-10 00:19:21 +02:00
|
|
|
|
2007-12-08 21:46:40 +01:00
|
|
|
bool TestRunner::run(int ac, const char **av)
|
2005-10-12 01:46:19 +02:00
|
|
|
{
|
2007-12-08 21:46:40 +01:00
|
|
|
bool ok = true;
|
2005-10-12 01:46:19 +02:00
|
|
|
string testCase;
|
|
|
|
int numberOfTests = 0;
|
2006-10-12 00:51:04 +02:00
|
|
|
int opt = 0;
|
2008-08-10 00:19:21 +02:00
|
|
|
runFn runner = textrun;
|
2005-10-12 01:46:19 +02:00
|
|
|
|
2010-01-12 16:56:44 +01:00
|
|
|
std::string executable = av[0];
|
|
|
|
size_t slash = executable.find_last_of("/\\");
|
|
|
|
if(slash != -1)
|
|
|
|
executable.erase(0, slash+1);
|
|
|
|
executable += "-";
|
|
|
|
|
2005-10-12 01:46:19 +02:00
|
|
|
for(int i = 1; i < ac; i++)
|
|
|
|
{
|
|
|
|
if(string(av[i]) == "-wait")
|
|
|
|
{
|
|
|
|
m_wait = true;
|
2006-10-12 00:51:04 +02:00
|
|
|
++opt;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2007-08-29 04:03:12 +02:00
|
|
|
if(string(av[i]) == "-v")
|
2006-10-12 00:51:04 +02:00
|
|
|
{
|
2007-08-29 04:03:12 +02:00
|
|
|
verbose_ = true;
|
2006-10-12 00:51:04 +02:00
|
|
|
++opt;
|
2005-10-12 01:46:19 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2008-08-10 00:19:21 +02:00
|
|
|
if(string(av[i]) == "-table")
|
|
|
|
{
|
|
|
|
runner = tablerun;
|
|
|
|
++opt;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2010-01-11 18:02:11 +01:00
|
|
|
if(string(av[i]) == "-xml")
|
|
|
|
{
|
|
|
|
runner = xmlrun;
|
|
|
|
++opt;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2010-01-12 10:42:08 +01:00
|
|
|
if(string(av[i]) == "-log")
|
|
|
|
{
|
2010-01-12 13:07:32 +01:00
|
|
|
logprefix_ = av[++i];
|
2010-01-12 16:56:44 +01:00
|
|
|
logprefix_ += executable;
|
|
|
|
cout << "logprefix=" << logprefix_ << std::endl;
|
2010-01-12 10:42:08 +01:00
|
|
|
opt += 2;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2008-08-10 00:19:21 +02:00
|
|
|
|
2005-10-12 01:46:19 +02:00
|
|
|
testCase = av[i];
|
|
|
|
|
|
|
|
if(testCase == "")
|
|
|
|
{
|
|
|
|
printBanner ();
|
2007-12-08 21:46:40 +01:00
|
|
|
return ok;
|
2005-10-12 01:46:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Test *testToRun = NULL;
|
|
|
|
|
|
|
|
for(mappings::iterator it = m_mappings.begin();
|
|
|
|
it != m_mappings.end();
|
|
|
|
++it)
|
|
|
|
{
|
|
|
|
if((*it).first == testCase)
|
|
|
|
{
|
|
|
|
testToRun = (*it).second;
|
2010-01-12 16:56:44 +01:00
|
|
|
ok &= runner(executable + (*it).first, testToRun, verbose_, logprefix_);
|
2005-10-12 01:46:19 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
numberOfTests++;
|
|
|
|
|
|
|
|
if(!testToRun)
|
|
|
|
{
|
|
|
|
cout << "Test " << testCase << " not found." << endl;
|
2007-12-08 21:46:40 +01:00
|
|
|
return false;
|
2005-10-12 01:46:19 +02:00
|
|
|
}
|
|
|
|
} // for ...
|
|
|
|
|
2006-10-12 00:51:04 +02:00
|
|
|
if((ac-opt) == 1)
|
2005-10-12 01:46:19 +02:00
|
|
|
{
|
|
|
|
// run everything
|
|
|
|
for(mappings::iterator it = m_mappings.begin(); it != m_mappings.end(); ++it)
|
|
|
|
{
|
2010-01-12 16:56:44 +01:00
|
|
|
ok &= runner(executable + (*it).first, (*it).second, verbose_, logprefix_);
|
2005-10-12 01:46:19 +02:00
|
|
|
}
|
2007-12-08 21:46:40 +01:00
|
|
|
return ok;
|
2005-10-12 01:46:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if(numberOfTests == 0)
|
|
|
|
{
|
|
|
|
printBanner ();
|
2007-12-08 21:46:40 +01:00
|
|
|
return false;
|
2005-10-12 01:46:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if(m_wait)
|
|
|
|
{
|
|
|
|
cout << "<RETURN> to continue" << endl;
|
|
|
|
cin.get ();
|
|
|
|
}
|
2007-12-08 21:46:40 +01:00
|
|
|
|
|
|
|
return ok;
|
2005-10-12 01:46:19 +02:00
|
|
|
} // run
|
|
|
|
|
|
|
|
TestRunner::~TestRunner ()
|
|
|
|
{
|
|
|
|
for(mappings::iterator it = m_mappings.begin ();
|
|
|
|
it != m_mappings.end ();
|
|
|
|
++it)
|
|
|
|
delete it->second;
|
|
|
|
} // ~TestRunner
|
|
|
|
|