2015-10-21 05:01:47 -05:00
|
|
|
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4; fill-column: 100 -*- */
|
|
|
|
/*
|
|
|
|
* This file is part of the LibreOffice project.
|
|
|
|
*
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
*/
|
|
|
|
|
2019-03-04 15:02:59 -06:00
|
|
|
#define TST_LOG_REDIRECT
|
2017-12-20 07:06:26 -06:00
|
|
|
#include <test.hpp>
|
2017-06-19 07:55:07 -05:00
|
|
|
|
2017-12-20 07:06:26 -06:00
|
|
|
#include <config.h>
|
2017-03-08 10:38:22 -06:00
|
|
|
|
2016-10-12 03:47:26 -05:00
|
|
|
#include <cstdlib>
|
2015-10-21 05:01:47 -05:00
|
|
|
#include <iostream>
|
2016-08-28 15:32:34 -05:00
|
|
|
|
|
|
|
#include <cppunit/BriefTestProgressListener.h>
|
|
|
|
#include <cppunit/CompilerOutputter.h>
|
2015-10-21 05:01:47 -05:00
|
|
|
#include <cppunit/TestResult.h>
|
2019-05-01 16:12:56 -05:00
|
|
|
#include <cppunit/TestFailure.h>
|
2015-10-21 05:01:47 -05:00
|
|
|
#include <cppunit/TestResultCollector.h>
|
2016-08-28 15:32:34 -05:00
|
|
|
#include <cppunit/TestRunner.h>
|
2016-03-25 18:41:53 -05:00
|
|
|
#include <cppunit/TextTestProgressListener.h>
|
2015-10-21 05:01:47 -05:00
|
|
|
#include <cppunit/extensions/TestFactoryRegistry.h>
|
2016-08-28 15:32:34 -05:00
|
|
|
|
|
|
|
#include <Poco/RegularExpression.h>
|
2017-09-19 15:16:44 -05:00
|
|
|
#include <Poco/DirectoryIterator.h>
|
|
|
|
#include <Poco/FileStream.h>
|
|
|
|
#include <Poco/StreamCopier.h>
|
|
|
|
#include <Poco/StringTokenizer.h>
|
2015-10-21 05:01:47 -05:00
|
|
|
|
2016-09-14 16:43:56 -05:00
|
|
|
#include <Log.hpp>
|
|
|
|
|
2016-04-24 19:59:58 -05:00
|
|
|
class HTTPGetTest;
|
|
|
|
|
2018-10-17 02:34:05 -05:00
|
|
|
bool filterTests(CPPUNIT_NS::TestRunner& runner, CPPUNIT_NS::Test* testRegistry, const std::string& testName)
|
2015-10-21 05:01:47 -05:00
|
|
|
{
|
2016-10-17 10:51:54 -05:00
|
|
|
Poco::RegularExpression re(testName, Poco::RegularExpression::RE_CASELESS);
|
|
|
|
Poco::RegularExpression::Match reMatch;
|
2016-08-28 15:32:34 -05:00
|
|
|
|
2016-10-17 10:51:54 -05:00
|
|
|
bool haveTests = false;
|
|
|
|
for (int i = 0; i < testRegistry->getChildTestCount(); ++i)
|
|
|
|
{
|
|
|
|
CPPUNIT_NS::Test* testSuite = testRegistry->getChildTestAt(i);
|
|
|
|
for (int j = 0; j < testSuite->getChildTestCount(); ++j)
|
2016-07-08 02:04:27 -05:00
|
|
|
{
|
2016-10-17 10:51:54 -05:00
|
|
|
CPPUNIT_NS::Test* testCase = testSuite->getChildTestAt(j);
|
|
|
|
try
|
2016-07-08 02:04:27 -05:00
|
|
|
{
|
2016-10-17 10:51:54 -05:00
|
|
|
if (re.match(testCase->getName(), reMatch))
|
2016-08-28 15:32:34 -05:00
|
|
|
{
|
2016-10-17 10:51:54 -05:00
|
|
|
runner.addTest(testCase);
|
|
|
|
haveTests = true;
|
2016-08-28 15:32:34 -05:00
|
|
|
}
|
2016-07-08 02:04:27 -05:00
|
|
|
}
|
2016-10-17 10:51:54 -05:00
|
|
|
catch (const std::exception& exc)
|
|
|
|
{
|
|
|
|
// Nothing to do; skip.
|
|
|
|
}
|
2016-07-08 02:04:27 -05:00
|
|
|
}
|
|
|
|
}
|
2016-08-28 15:32:34 -05:00
|
|
|
|
2016-10-17 10:51:54 -05:00
|
|
|
return haveTests;
|
2016-08-28 15:32:34 -05:00
|
|
|
}
|
|
|
|
|
2017-04-07 08:37:20 -05:00
|
|
|
int main(int argc, char** argv)
|
2016-08-28 15:32:34 -05:00
|
|
|
{
|
2017-04-19 23:06:28 -05:00
|
|
|
const bool verbose = (argc > 1 && std::string("--verbose") == argv[1]);
|
|
|
|
const char* loglevel = verbose ? "trace" : "error";
|
2017-04-07 08:37:20 -05:00
|
|
|
|
|
|
|
Log::initialize("tst", loglevel, true, false, {});
|
2016-09-14 16:43:56 -05:00
|
|
|
|
2017-08-17 07:04:22 -05:00
|
|
|
return runClientTests(true, verbose)? 0: 1;
|
2017-05-11 21:08:20 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool IsStandalone = false;
|
|
|
|
|
|
|
|
bool isStandalone()
|
|
|
|
{
|
|
|
|
return IsStandalone;
|
|
|
|
}
|
|
|
|
|
2019-03-04 15:02:59 -06:00
|
|
|
static std::mutex errorMutex;
|
2019-05-04 14:16:36 -05:00
|
|
|
static bool IsVerbose = false;
|
2019-03-04 15:02:59 -06:00
|
|
|
static std::stringstream errors;
|
|
|
|
|
|
|
|
void tstLog(const std::ostringstream &stream)
|
|
|
|
{
|
2019-05-04 14:16:36 -05:00
|
|
|
if (IsVerbose)
|
|
|
|
std::cerr << stream.str() << std::endl;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> lock(errorMutex);
|
|
|
|
errors << stream.str();
|
|
|
|
}
|
2019-03-04 15:02:59 -06:00
|
|
|
}
|
|
|
|
|
2017-09-19 13:06:46 -05:00
|
|
|
// returns true on success
|
2017-05-11 21:08:20 -05:00
|
|
|
bool runClientTests(bool standalone, bool verbose)
|
|
|
|
{
|
2019-05-04 14:16:36 -05:00
|
|
|
IsVerbose = verbose;
|
2017-05-11 21:08:20 -05:00
|
|
|
IsStandalone = standalone;
|
|
|
|
|
2016-08-28 15:32:34 -05:00
|
|
|
CPPUNIT_NS::TestResult controller;
|
|
|
|
CPPUNIT_NS::TestResultCollector result;
|
|
|
|
controller.addListener(&result);
|
|
|
|
CPPUNIT_NS::BriefTestProgressListener progress;
|
|
|
|
controller.addListener(&progress);
|
|
|
|
controller.addListener(new CPPUNIT_NS::TextTestProgressListener());
|
|
|
|
|
|
|
|
CPPUNIT_NS::Test* testRegistry = CPPUNIT_NS::TestFactoryRegistry::getRegistry().makeTest();
|
|
|
|
|
|
|
|
CPPUNIT_NS::TestRunner runner;
|
2016-10-17 10:51:54 -05:00
|
|
|
const char* envar = std::getenv("CPPUNIT_TEST_NAME");
|
|
|
|
std::string testName;
|
|
|
|
if (envar)
|
|
|
|
{
|
|
|
|
testName = std::string(envar);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (testName.empty())
|
2016-08-28 15:32:34 -05:00
|
|
|
{
|
2016-10-17 10:51:54 -05:00
|
|
|
// Add all tests.
|
2016-08-28 15:32:34 -05:00
|
|
|
runner.addTest(testRegistry);
|
|
|
|
}
|
2016-10-17 10:51:54 -05:00
|
|
|
else
|
|
|
|
{
|
|
|
|
const bool testsAdded = filterTests(runner, testRegistry, testName);
|
|
|
|
if (!testsAdded)
|
|
|
|
{
|
|
|
|
std::cerr << "Failed to match [" << testName << "] to any names in the external test-suite. "
|
|
|
|
<< "No external tests will be executed" << std::endl;
|
|
|
|
}
|
|
|
|
}
|
2016-07-08 02:04:27 -05:00
|
|
|
|
2017-04-19 23:06:28 -05:00
|
|
|
if (!verbose)
|
|
|
|
{
|
|
|
|
runner.run(controller);
|
2015-10-21 05:01:47 -05:00
|
|
|
|
2017-04-19 23:06:28 -05:00
|
|
|
// output the errors we got during the testing
|
|
|
|
if (!result.wasSuccessful())
|
2019-03-04 15:02:59 -06:00
|
|
|
std::cerr << errors.str() << std::endl;
|
2017-04-19 23:06:28 -05:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
runner.run(controller);
|
|
|
|
}
|
2017-04-07 08:37:20 -05:00
|
|
|
|
2015-10-21 05:01:47 -05:00
|
|
|
CPPUNIT_NS::CompilerOutputter outputter(&result, std::cerr);
|
2016-10-04 03:04:22 -05:00
|
|
|
outputter.setNoWrap();
|
2015-10-21 05:01:47 -05:00
|
|
|
outputter.write();
|
|
|
|
|
2019-05-01 16:12:56 -05:00
|
|
|
const std::deque<CPPUNIT_NS::TestFailure *> &failures = result.failures();
|
|
|
|
if (!envar && failures.size() > 0)
|
|
|
|
{
|
|
|
|
std::cerr << "\nTo reproduce the first test failure use:\n\n";
|
2019-05-03 08:01:57 -05:00
|
|
|
#ifndef UNIT_CLIENT_TESTS
|
2019-05-04 14:16:36 -05:00
|
|
|
std::cerr << "(cd test; CPPUNIT_TEST_NAME=\"" << (*failures.begin())->failedTestName() << "\" ./run_unit.sh --verbose)\n\n";
|
2019-05-03 08:01:57 -05:00
|
|
|
#else
|
|
|
|
std::cerr << "(cd test; CPPUNIT_TEST_NAME=\"" << (*failures.begin())->failedTestName() << "\" make check)\n\n";
|
|
|
|
#endif
|
2019-05-01 16:12:56 -05:00
|
|
|
}
|
|
|
|
|
2017-08-17 07:04:22 -05:00
|
|
|
return result.wasSuccessful();
|
2015-10-21 05:01:47 -05:00
|
|
|
}
|
|
|
|
|
2017-09-19 13:06:46 -05:00
|
|
|
// Versions assuming a single user, on a single machine
|
|
|
|
#ifndef UNIT_CLIENT_TESTS
|
|
|
|
|
2019-05-13 09:39:14 -05:00
|
|
|
std::vector<int> getProcPids(const char* exec_filename)
|
2017-09-19 15:16:44 -05:00
|
|
|
{
|
|
|
|
std::vector<int> pids;
|
|
|
|
|
2019-05-13 09:39:14 -05:00
|
|
|
// Ensure we're in the same group.
|
|
|
|
int grp = getpgrp();
|
|
|
|
|
|
|
|
// Get all lokit processes.
|
2017-09-19 15:16:44 -05:00
|
|
|
for (auto it = Poco::DirectoryIterator(std::string("/proc")); it != Poco::DirectoryIterator(); ++it)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
2018-11-30 01:37:50 -06:00
|
|
|
const Poco::Path& procEntry = it.path();
|
2017-09-19 15:16:44 -05:00
|
|
|
const std::string& fileName = procEntry.getFileName();
|
|
|
|
int pid;
|
|
|
|
std::size_t endPos = 0;
|
|
|
|
try
|
|
|
|
{
|
|
|
|
pid = std::stoi(fileName, &endPos);
|
|
|
|
}
|
|
|
|
catch (const std::invalid_argument&)
|
|
|
|
{
|
|
|
|
pid = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pid > 1 && endPos == fileName.length())
|
|
|
|
{
|
|
|
|
Poco::FileInputStream stat(procEntry.toString() + "/stat");
|
|
|
|
std::string statString;
|
|
|
|
Poco::StreamCopier::copyToString(stat, statString);
|
|
|
|
Poco::StringTokenizer tokens(statString, " ");
|
2019-05-13 09:39:14 -05:00
|
|
|
if (tokens.count() > 6 && tokens[1] == exec_filename)
|
2017-09-19 15:16:44 -05:00
|
|
|
{
|
2019-05-13 09:39:14 -05:00
|
|
|
// We could have several make checks running at once.
|
|
|
|
int kidGrp = std::atoi(tokens[4].c_str());
|
|
|
|
if (kidGrp != grp)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
switch (tokens[2].c_str()[0])
|
2017-09-19 15:16:44 -05:00
|
|
|
{
|
2019-05-13 09:39:14 -05:00
|
|
|
// Dead & zombie markers for old and new kernels.
|
|
|
|
case 'x':
|
|
|
|
case 'X':
|
|
|
|
case 'Z':
|
|
|
|
break;
|
|
|
|
default:
|
2017-09-19 15:16:44 -05:00
|
|
|
pids.push_back(pid);
|
2019-05-13 09:39:14 -05:00
|
|
|
break;
|
|
|
|
}
|
2017-09-19 15:16:44 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (const Poco::Exception&)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return pids;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<int> getKitPids()
|
|
|
|
{
|
|
|
|
std::vector<int> pids;
|
|
|
|
|
|
|
|
pids = getProcPids("(loolkit)");
|
|
|
|
|
|
|
|
return pids;
|
|
|
|
}
|
|
|
|
|
|
|
|
int getLoolKitProcessCount()
|
|
|
|
{
|
2019-05-13 09:39:14 -05:00
|
|
|
return getProcPids("(loolkit)").size();
|
2017-09-19 15:16:44 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<int> getForKitPids()
|
|
|
|
{
|
|
|
|
std::vector<int> pids, pids2;
|
|
|
|
|
|
|
|
pids = getProcPids("(loolforkit)");
|
|
|
|
pids2 = getProcPids("(forkit)");
|
|
|
|
pids.insert(pids.end(), pids2.begin(), pids2.end());
|
|
|
|
|
|
|
|
return pids;
|
|
|
|
}
|
|
|
|
|
2017-09-19 13:06:46 -05:00
|
|
|
#else // UNIT_CLIENT_TESTS
|
|
|
|
|
|
|
|
// Here we are compiled inside UnitClient.cpp and we have
|
|
|
|
// full access to the WSD process internals.
|
|
|
|
|
|
|
|
std::vector<int> getKitPids()
|
|
|
|
{
|
|
|
|
return LOOLWSD::getKitPids();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the PID of the forkit
|
|
|
|
std::vector<int> getForKitPids()
|
|
|
|
{
|
|
|
|
std::vector<int> pids;
|
|
|
|
if (LOOLWSD::ForKitProcId >= 0)
|
|
|
|
pids.push_back(LOOLWSD::ForKitProcId);
|
|
|
|
return pids;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// How many live lookit processes do we have ?
|
|
|
|
int getLoolKitProcessCount()
|
|
|
|
{
|
|
|
|
return getKitPids().size();
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // UNIT_CLIENT_TESTS
|
|
|
|
|
2015-10-21 05:01:47 -05:00
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|