vmime/tests/testRunner.cpp

306 lines
7.0 KiB
C++
Raw Normal View History

//
// VMime library (http://www.vmime.org)
2018-09-05 21:54:48 +00:00
// Copyright (C) 2002 Vincent Richard <vincent@vmime.org>
//
// 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.
//
2005-09-17 10:10:29 +00:00
// You should have received a copy of the GNU General Public License along
// with this program; if not, write to the Free Software Foundation, Inc.,
// 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
//
// Linking this library statically or dynamically with other modules is making
// a combined work based on this library. Thus, the terms and conditions of
// the GNU General Public License cover the whole combination.
//
2005-08-27 10:35:39 +00:00
#include <sys/time.h>
#include <time.h>
#include <iostream>
#include <vector>
#include <algorithm>
2008-07-11 21:27:24 +00:00
#include <memory>
2005-08-27 10:35:39 +00:00
#include <cppunit/XmlOutputter.h>
#include <cppunit/extensions/TestFactoryRegistry.h>
#include <cppunit/ui/text/TestRunner.h>
2005-08-27 10:35:39 +00:00
#include <cppunit/TestListener.h>
#include <cppunit/TestResult.h>
#include <cppunit/TestResultCollector.h>
#include <cppunit/TestFailure.h>
#include <cppunit/SourceLine.h>
#include <cppunit/Exception.h>
#include <cppunit/tools/XmlDocument.h>
#include <cppunit/tools/XmlElement.h>
#include "vmime/vmime.hpp"
#include "vmime/platforms/posix/posixHandler.hpp"
2018-09-05 21:54:48 +00:00
class Clock {
2005-08-27 10:35:39 +00:00
public:
2018-09-05 21:54:48 +00:00
void reset() {
2005-08-27 10:35:39 +00:00
struct timezone tz;
gettimeofday(&m_start, &tz);
}
2018-09-05 21:54:48 +00:00
double getDuration() const {
2005-08-27 10:35:39 +00:00
struct timeval tv;
struct timezone tz;
gettimeofday(&tv, &tz);
return static_cast <double>(tv.tv_sec - m_start.tv_sec)
+ static_cast <double>(tv.tv_usec - m_start.tv_usec) / 1000000.0;
}
private:
struct timeval m_start;
};
2018-09-05 21:54:48 +00:00
class XmlTestListener : public CppUnit::TestListener {
2005-08-27 10:35:39 +00:00
public:
XmlTestListener()
2018-09-05 21:54:48 +00:00
: m_doc("utf-8"),
m_testElt(NULL) {
2005-08-27 10:35:39 +00:00
m_doc.setRootElement(new CppUnit::XmlElement("TestRun"));
}
2018-09-05 21:54:48 +00:00
void startTest(CppUnit::Test* test) {
2005-08-27 10:35:39 +00:00
m_testElt = new CppUnit::XmlElement("Test");
m_suiteElt.back()->addElement(m_testElt);
m_testElt->addElement(new CppUnit::XmlElement("Name", test->getName()));
m_chrono.reset();
}
2018-09-05 21:54:48 +00:00
void addFailure(const CppUnit::TestFailure& failure) {
2005-08-27 10:35:39 +00:00
CppUnit::XmlElement* failElt = new CppUnit::XmlElement("Failure");
m_testElt->addElement(failElt);
2018-09-05 21:54:48 +00:00
failElt->addElement(
new CppUnit::XmlElement("FailureType", failure.isError() ? "Error" : "Assertion")
);
if (failure.sourceLine().isValid()) {
2005-08-27 10:35:39 +00:00
CppUnit::XmlElement* locElt = new CppUnit::XmlElement("Location");
failElt->addElement(locElt);
locElt->addElement(new CppUnit::XmlElement("File", failure.sourceLine().fileName()));
locElt->addElement(new CppUnit::XmlElement("Line", failure.sourceLine().lineNumber()));
}
CppUnit::XmlElement* exElt = new CppUnit::XmlElement("Exception");
failElt->addElement(exElt);
exElt->addElement(new CppUnit::XmlElement("Message", failure.thrownException()->what()));
}
2018-09-05 21:54:48 +00:00
void endTest(CppUnit::Test* /* test */) {
2005-08-27 10:35:39 +00:00
std::ostringstream ossTime;
ossTime << (m_chrono.getDuration() * 1000.0);
m_testElt->addElement(new CppUnit::XmlElement("Time", ossTime.str()));
m_testElt = NULL;
}
2018-09-05 21:54:48 +00:00
void startSuite(CppUnit::Test* suite) {
if (suite->getName() == "All Tests") {
2005-08-27 10:35:39 +00:00
return;
2018-09-05 21:54:48 +00:00
}
2005-08-27 10:35:39 +00:00
CppUnit::XmlElement* suiteElt = new CppUnit::XmlElement("Suite");
2018-09-05 21:54:48 +00:00
if (m_suiteElt.size() == 0) {
2005-08-27 10:35:39 +00:00
m_doc.rootElement().addElement(suiteElt);
2018-09-05 21:54:48 +00:00
} else {
2005-08-27 10:35:39 +00:00
m_suiteElt.back()->addElement(suiteElt);
2018-09-05 21:54:48 +00:00
}
2005-08-27 10:35:39 +00:00
m_suiteElt.push_back(suiteElt);
suiteElt->addElement(new CppUnit::XmlElement("Name", suite->getName()));
}
2018-09-05 21:54:48 +00:00
void endSuite(CppUnit::Test* /* suite */) {
if (m_suiteElt.size()) {
2005-08-27 10:35:39 +00:00
m_suiteElt.pop_back();
2018-09-05 21:54:48 +00:00
}
2005-08-27 10:35:39 +00:00
}
2018-09-05 21:54:48 +00:00
void startTestRun(CppUnit::Test* /* test */, CppUnit::TestResult* /* eventManager */) {
2005-08-27 10:35:39 +00:00
}
2018-09-05 21:54:48 +00:00
void endTestRun(CppUnit::Test* /* test */, CppUnit::TestResult* /* eventManager */) {
2005-08-27 10:35:39 +00:00
}
2018-09-05 21:54:48 +00:00
void output(std::ostream& os) {
2005-08-27 10:35:39 +00:00
os << m_doc.toString();
}
private:
Clock m_chrono;
CppUnit::XmlDocument m_doc;
std::vector <CppUnit::XmlElement*> m_suiteElt;
CppUnit::XmlElement* m_testElt;
};
// see testUtils.hpp
2018-09-05 21:54:48 +00:00
std::vector <std::string>& getTestModules() {
static std::vector <std::string> allModules;
return allModules;
}
2018-09-05 21:54:48 +00:00
void registerTestModule(const char* name_) {
std::vector <std::string>& testModules = getTestModules();
std::string name(name_);
2018-09-05 21:54:48 +00:00
if (std::find(testModules.begin(), testModules.end(), name) == testModules.end()) {
testModules.push_back(name);
2018-09-05 21:54:48 +00:00
}
}
2018-09-05 21:54:48 +00:00
const std::string getNormalizedPath(const std::string& path) {
2013-03-08 07:19:50 +00:00
std::string res = path;
2018-09-05 21:54:48 +00:00
for (std::size_t i = 0, n = res.length() ; i < n ; ++i) {
if (res[i] == '\\') {
2013-03-08 07:19:50 +00:00
res[i] = '/';
2018-09-05 21:54:48 +00:00
}
2013-03-08 07:19:50 +00:00
}
return res;
}
2018-09-05 21:54:48 +00:00
const std::string getFileNameFromPath(const std::string& path) {
const std::size_t pos = path.find_last_of('/');
2013-03-08 07:19:50 +00:00
2018-09-05 21:54:48 +00:00
if (pos == std::string::npos) {
2013-03-08 07:19:50 +00:00
return "";
2018-09-05 21:54:48 +00:00
}
2013-03-08 07:19:50 +00:00
return path.substr(pos + 1);
}
static char g_moduleNameBuffer[2048];
2018-09-05 21:54:48 +00:00
const char* getTestModuleNameFromSourceFile(const char *path_) {
2013-03-08 07:19:50 +00:00
static const std::string testRunnerPath(getNormalizedPath(__FILE__));
static const std::string testRunnerFileName(getFileNameFromPath(testRunnerPath));
const std::string path = getNormalizedPath(path_);
// "/path/to/testRunner.cpp" --> "/path/to/"
const std::string basePath
(testRunnerPath.begin(), testRunnerPath.end() - testRunnerFileName.length());
// "/path/to/module/testFile.cpp" --> "module/testFile.cpp"
const std::string testFileName(getFileNameFromPath(path));
const std::string testPath(path.begin() + basePath.length(), path.end());
// "module/testFile.cpp" --> "module"
const std::string moduleName(testPath.substr(0, testPath.length() - testFileName.length() - 1));
std::copy(moduleName.begin(), moduleName.end(), g_moduleNameBuffer);
g_moduleNameBuffer[moduleName.length()] = 0;
2013-03-08 07:19:50 +00:00
return g_moduleNameBuffer;
2013-03-08 07:19:50 +00:00
}
2018-09-05 21:54:48 +00:00
int main(int argc, char* argv[]) {
2005-08-27 10:35:39 +00:00
// Parse arguments
bool xmlOutput = false;
2018-09-05 21:54:48 +00:00
for (int c = 1 ; c < argc ; ++c) {
2005-08-27 10:35:39 +00:00
const std::string arg = argv[c];
2018-09-05 21:54:48 +00:00
if (arg == "--xml") {
2005-08-27 10:35:39 +00:00
xmlOutput = true;
2018-09-05 21:54:48 +00:00
}
2005-08-27 10:35:39 +00:00
}
// Run the tests
2018-09-05 21:54:48 +00:00
if (xmlOutput) {
2006-05-05 20:50:26 +00:00
// Get the test suites from the registry and add them to the list of tests to run
2005-08-27 10:35:39 +00:00
CppUnit::TestRunner runner;
2018-09-05 21:54:48 +00:00
for (unsigned int i = 0 ; i < getTestModules().size() ; ++i) {
runner.addTest(
CppUnit::TestFactoryRegistry::getRegistry(getTestModules()[i]).makeTest()
);
}
2005-08-27 10:35:39 +00:00
2016-03-13 19:15:22 +00:00
XmlTestListener xmlListener;
2005-08-27 10:35:39 +00:00
CppUnit::TestResult controller;
2016-03-13 19:15:22 +00:00
controller.addListener(&xmlListener);
2005-08-27 10:35:39 +00:00
CppUnit::TestResultCollector result;
controller.addListener(&result);
runner.run(controller);
2016-03-13 19:15:22 +00:00
xmlListener.output(std::cout);
2005-08-27 10:35:39 +00:00
2006-05-05 20:50:26 +00:00
// Return error code 1 if a test failed
2005-08-27 10:35:39 +00:00
return result.wasSuccessful() ? 0 : 1;
2018-09-05 21:54:48 +00:00
} else {
2005-08-27 10:35:39 +00:00
// Get the top level suite from the registry
CppUnit::TextUi::TestRunner runner;
runner.addTest(CppUnit::TestFactoryRegistry::getRegistry().makeTest());
2005-08-27 10:35:39 +00:00
return runner.run() ? 0 : 1;
}
}