cppcheck/test/testsuite.cpp

228 lines
5.7 KiB
C++
Raw Normal View History

2008-12-18 22:28:57 +01:00
/*
* Cppcheck - A tool for static C/C++ code analysis
* Copyright (C) 2007-2010 Daniel Marjamäki and Cppcheck team.
2008-12-18 22:28:57 +01:00
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
2008-12-18 22:28:57 +01:00
*/
#include "testsuite.h"
#include <iostream>
#include <list>
std::ostringstream errout;
std::ostringstream output;
2008-12-18 22:28:57 +01:00
/**
* TestRegistry
**/
class TestRegistry
{
private:
std::list<TestFixture *> _tests;
public:
static TestRegistry &theInstance()
{
static TestRegistry testreg;
return testreg;
}
void addTest(TestFixture *t)
2008-12-18 22:28:57 +01:00
{
_tests.push_back(t);
2008-12-18 22:28:57 +01:00
}
const std::list<TestFixture *> &tests() const
{
return _tests;
}
};
/**
* TestFixture
**/
std::ostringstream TestFixture::errmsg;
unsigned int TestFixture::countTests;
size_t TestFixture::fails_counter = 0;
size_t TestFixture::todos_counter = 0;
2008-12-18 22:28:57 +01:00
TestFixture::TestFixture(const std::string &_name) : classname(_name)
{
TestRegistry::theInstance().addTest(this);
}
bool TestFixture::runTest(const char testname[])
{
if (testToRun.empty() || testToRun == testname)
2008-12-18 22:28:57 +01:00
{
2009-01-01 23:22:28 +01:00
++countTests;
2008-12-18 22:28:57 +01:00
std::cout << classname << "::" << testname << "\n";
return true;
2008-12-18 22:28:57 +01:00
}
return false;
2008-12-18 22:28:57 +01:00
}
static std::string writestr(const std::string &str)
2008-12-18 22:28:57 +01:00
{
std::ostringstream ostr;
ostr << "\"";
for (unsigned int i = 0; i < str.length(); ++i)
2008-12-18 22:28:57 +01:00
{
char ch = str[i];
if (ch == '\n')
2008-12-18 22:28:57 +01:00
ostr << "\\n";
else if (ch == '\t')
2008-12-18 22:28:57 +01:00
ostr << "\\t";
else if (ch == '\"')
2008-12-18 22:28:57 +01:00
ostr << "\\\"";
else
ostr << std::string(1, ch);
}
ostr << "\"";
return ostr.str();
}
void TestFixture::assert(const char *filename, int linenr, bool condition)
{
if (!condition)
{
++fails_counter;
errmsg << "Assertion failed in " << filename << " at line " << linenr << std::endl;
}
}
void TestFixture::assertEquals(const char *filename, int linenr, const std::string &expected, const std::string &actual, const std::string &msg)
2008-12-18 22:28:57 +01:00
{
if (expected != actual)
2008-12-18 22:28:57 +01:00
{
++fails_counter;
2008-12-18 22:28:57 +01:00
errmsg << "Assertion failed in " << filename << " at line " << linenr << std::endl
2010-04-15 20:08:51 +02:00
<< "Expected:" << std::endl
<< writestr(expected) << std::endl
<< "Actual:" << std::endl
<< writestr(actual) << std::endl;
2010-05-14 18:40:37 +02:00
if (!msg.empty())
{
errmsg << msg << std::endl;
}
2008-12-18 22:28:57 +01:00
}
}
void TestFixture::assertEquals(const char *filename, int linenr, double expected, double actual, const std::string &msg)
2008-12-18 22:28:57 +01:00
{
std::ostringstream ostr1;
ostr1 << expected;
std::ostringstream ostr2;
ostr2 << actual;
assertEquals(filename, linenr, ostr1.str(), ostr2.str(), msg);
2008-12-18 22:28:57 +01:00
}
void TestFixture::todoAssertEquals(const char *filename, int linenr, const std::string &expected, const std::string &actual)
{
if (expected == actual)
assertEquals(filename, linenr, "TODO assertion", "The assertion succeeded");
else
++todos_counter;
}
void TestFixture::todoAssertEquals(const char *filename, int linenr, unsigned int expected, unsigned int actual)
{
std::ostringstream ostr1;
ostr1 << expected;
std::ostringstream ostr2;
ostr2 << actual;
todoAssertEquals(filename, linenr, ostr1.str(), ostr2.str());
}
void TestFixture::assertThrowFail(const char *filename, int linenr)
{
++fails_counter;
errmsg << "Assertion failed in " << filename << " at line " << linenr << std::endl
2010-04-15 20:08:51 +02:00
<< "The expected exception was not thrown" << std::endl;
}
2008-12-18 22:28:57 +01:00
void TestFixture::printTests()
{
const std::list<TestFixture *> &tests = TestRegistry::theInstance().tests();
for (std::list<TestFixture *>::const_iterator it = tests.begin(); it != tests.end(); ++it)
2008-12-18 22:28:57 +01:00
{
std::cout << (*it)->classname << std::endl;
}
}
void TestFixture::run(const std::string &str)
{
testToRun = str;
run();
}
2008-12-18 22:28:57 +01:00
size_t TestFixture::runTests(const char cmd[])
2008-12-18 22:28:57 +01:00
{
std::string classname(cmd ? cmd : "");
std::string testname("");
if (classname.find("::") != std::string::npos)
2008-12-18 22:28:57 +01:00
{
testname = classname.substr(classname.find("::") + 2);
classname.erase(classname.find("::"));
2008-12-18 22:28:57 +01:00
}
countTests = 0;
errmsg.str("");
const std::list<TestFixture *> &tests = TestRegistry::theInstance().tests();
for (std::list<TestFixture *>::const_iterator it = tests.begin(); it != tests.end(); ++it)
2008-12-18 22:28:57 +01:00
{
if (classname.empty() || (*it)->classname == classname)
2008-12-18 22:28:57 +01:00
{
(*it)->run(testname);
2008-12-18 22:28:57 +01:00
}
}
std::cout << "\n\nTesting Complete\nNumber of tests: " << countTests << "\n";
std::cout << "Number of todos: " << todos_counter << "\n";
// calling flush here, to do all output before the error messages (in case the output is buffered)
std::cout.flush();
2008-12-18 22:28:57 +01:00
std::cerr << "Tests failed: " << fails_counter << "\n";
2008-12-18 22:28:57 +01:00
std::cerr << errmsg.str();
std::cerr.flush();
return fails_counter;
2008-12-18 22:28:57 +01:00
}
void TestFixture::reportOut(const std::string & outmsg)
2008-12-18 22:28:57 +01:00
{
output << outmsg << std::endl;
2008-12-18 22:28:57 +01:00
}
2009-02-01 19:00:47 +01:00
void TestFixture::reportErr(const ErrorLogger::ErrorMessage &msg)
2009-02-01 19:00:47 +01:00
{
errout << msg.toString() << std::endl;
2009-02-01 19:00:47 +01:00
}