forked from qt-creator/qt-creator
TestResult: Devirtualize the class - part 2 of 5
Step 2 - implement findTestItemHook. Change-Id: Iabc843740343e549f79b02f74f94f5b1b4713af3 Reviewed-by: Christian Stenger <christian.stenger@qt.io>
This commit is contained in:
@@ -11,6 +11,8 @@
|
||||
|
||||
#include <QRegularExpression>
|
||||
|
||||
using namespace Utils;
|
||||
|
||||
namespace Autotest {
|
||||
namespace Internal {
|
||||
|
||||
@@ -35,15 +37,65 @@ static ResultHooks::OutputStringHook outputStringHook(const QString &testCaseNam
|
||||
};
|
||||
}
|
||||
|
||||
static bool matches(const FilePath &fileName, const FilePath &projectFile, const QString &testCaseName,
|
||||
const QString &testSuiteName, const BoostTestTreeItem *item)
|
||||
{
|
||||
// due to lacking information on the result side and a not fully appropriate tree we
|
||||
// might end up here with a differing set of tests, but it's the best we can do
|
||||
if (!item)
|
||||
return false;
|
||||
if (testCaseName.isEmpty()) // a top level module node
|
||||
return item->proFile() == projectFile;
|
||||
if (item->proFile() != projectFile)
|
||||
return false;
|
||||
if (!fileName.isEmpty() && fileName != item->filePath())
|
||||
return false;
|
||||
|
||||
QString fullName = "::" + testCaseName;
|
||||
fullName.prepend(testSuiteName.isEmpty() ? QString(BoostTest::Constants::BOOST_MASTER_SUITE)
|
||||
: testSuiteName);
|
||||
|
||||
BoostTestTreeItem::TestStates states = item->state();
|
||||
if (states & BoostTestTreeItem::Templated) {
|
||||
const QRegularExpression regex(
|
||||
QRegularExpression::wildcardToRegularExpression(item->fullName() + "<*>"));
|
||||
return regex.match(fullName).hasMatch();
|
||||
} else if (states & BoostTestTreeItem::Parameterized) {
|
||||
const QRegularExpression regex(
|
||||
QRegularExpression::anchoredPattern(item->fullName() + "_\\d+"));
|
||||
return regex.isValid() && regex.match(fullName).hasMatch();
|
||||
}
|
||||
return item->fullName() == fullName;
|
||||
}
|
||||
|
||||
static ResultHooks::FindTestItemHook findTestItemHook(const FilePath &projectFile,
|
||||
const QString &testCaseName,
|
||||
const QString &testSuiteName)
|
||||
{
|
||||
return [=](const TestResult &result) -> ITestTreeItem * {
|
||||
const Id id = Id(Constants::FRAMEWORK_PREFIX).withSuffix(BoostTest::Constants::FRAMEWORK_NAME);
|
||||
ITestFramework *framework = TestFrameworkManager::frameworkForId(id);
|
||||
QTC_ASSERT(framework, return nullptr);
|
||||
const TestTreeItem *rootNode = framework->rootNode();
|
||||
if (!rootNode)
|
||||
return nullptr;
|
||||
|
||||
return rootNode->findAnyChild([&](const TreeItem *item) {
|
||||
const auto testTreeItem = static_cast<const BoostTestTreeItem *>(item);
|
||||
return testTreeItem && matches(result.fileName(), projectFile, testCaseName,
|
||||
testSuiteName, testTreeItem);
|
||||
});
|
||||
};
|
||||
}
|
||||
|
||||
BoostTestResult::BoostTestResult(const QString &id, const QString &name,
|
||||
const Utils::FilePath &projectFile,
|
||||
const QString &testCaseName, const QString &testSuiteName)
|
||||
: TestResult(id, name, {outputStringHook(testCaseName)})
|
||||
const FilePath &projectFile, const QString &testCaseName,
|
||||
const QString &testSuiteName)
|
||||
: TestResult(id, name, {outputStringHook(testCaseName),
|
||||
findTestItemHook(projectFile, testCaseName, testSuiteName)})
|
||||
, m_projectFile(projectFile)
|
||||
, m_testCaseName(testCaseName)
|
||||
, m_testSuiteName(testSuiteName)
|
||||
{
|
||||
}
|
||||
, m_testSuiteName(testSuiteName) {}
|
||||
|
||||
bool BoostTestResult::isDirectParentOf(const TestResult *other, bool *needsIntermediate) const
|
||||
{
|
||||
@@ -75,50 +127,6 @@ bool BoostTestResult::isDirectParentOf(const TestResult *other, bool *needsInter
|
||||
return false;
|
||||
}
|
||||
|
||||
const ITestTreeItem *BoostTestResult::findTestTreeItem() const
|
||||
{
|
||||
auto id = Utils::Id(Constants::FRAMEWORK_PREFIX).withSuffix(BoostTest::Constants::FRAMEWORK_NAME);
|
||||
ITestFramework *framework = TestFrameworkManager::frameworkForId(id);
|
||||
QTC_ASSERT(framework, return nullptr);
|
||||
const TestTreeItem *rootNode = framework->rootNode();
|
||||
if (!rootNode)
|
||||
return nullptr;
|
||||
|
||||
return rootNode->findAnyChild([this](const Utils::TreeItem *item) {
|
||||
return matches(static_cast<const BoostTestTreeItem *>(item));
|
||||
});
|
||||
}
|
||||
|
||||
bool BoostTestResult::matches(const BoostTestTreeItem *item) const
|
||||
{
|
||||
// due to lacking information on the result side and a not fully appropriate tree we
|
||||
// might end up here with a differing set of tests, but it's the best we can do
|
||||
if (!item)
|
||||
return false;
|
||||
if (m_testCaseName.isEmpty()) // a top level module node
|
||||
return item->proFile() == m_projectFile;
|
||||
if (item->proFile() != m_projectFile)
|
||||
return false;
|
||||
if (!fileName().isEmpty() && fileName() != item->filePath())
|
||||
return false;
|
||||
|
||||
QString fullName = "::" + m_testCaseName;
|
||||
fullName.prepend(m_testSuiteName.isEmpty() ? QString(BoostTest::Constants::BOOST_MASTER_SUITE)
|
||||
: m_testSuiteName);
|
||||
|
||||
BoostTestTreeItem::TestStates states = item->state();
|
||||
if (states & BoostTestTreeItem::Templated) {
|
||||
const QRegularExpression regex(
|
||||
QRegularExpression::wildcardToRegularExpression(item->fullName() + "<*>"));
|
||||
return regex.match(fullName).hasMatch();
|
||||
} else if (states & BoostTestTreeItem::Parameterized) {
|
||||
const QRegularExpression regex(
|
||||
QRegularExpression::anchoredPattern(item->fullName() + "_\\d+"));
|
||||
return regex.isValid() && regex.match(fullName).hasMatch();
|
||||
}
|
||||
return item->fullName() == fullName;
|
||||
}
|
||||
|
||||
} // namespace Internal
|
||||
} // namespace Autotest
|
||||
|
||||
|
@@ -17,9 +17,7 @@ public:
|
||||
const QString &testCaseName = {}, const QString &testSuiteName = {});
|
||||
|
||||
bool isDirectParentOf(const TestResult *other, bool *needsIntermediate) const override;
|
||||
const ITestTreeItem * findTestTreeItem() const override;
|
||||
private:
|
||||
bool matches(const BoostTestTreeItem *item) const;
|
||||
|
||||
Utils::FilePath m_projectFile;
|
||||
QString m_testCaseName;
|
||||
|
@@ -9,15 +9,36 @@
|
||||
#include <utils/id.h>
|
||||
#include <utils/qtcassert.h>
|
||||
|
||||
using namespace Utils;
|
||||
|
||||
namespace Autotest {
|
||||
namespace Internal {
|
||||
|
||||
CatchResult::CatchResult(const QString &id, const QString &name, int depth)
|
||||
: TestResult(id, name)
|
||||
, m_sectionDepth(depth)
|
||||
static ResultHooks::FindTestItemHook findTestItemHook()
|
||||
{
|
||||
return [=](const TestResult &result) -> ITestTreeItem * {
|
||||
const Id id = Id(Constants::FRAMEWORK_PREFIX).withSuffix("Catch");
|
||||
ITestFramework *framework = TestFrameworkManager::frameworkForId(id);
|
||||
QTC_ASSERT(framework, return nullptr);
|
||||
const TestTreeItem *rootNode = framework->rootNode();
|
||||
if (!rootNode)
|
||||
return nullptr;
|
||||
|
||||
return rootNode->findAnyChild([&](const TreeItem *item) {
|
||||
const auto treeItem = static_cast<const CatchTreeItem *>(item);
|
||||
if (!treeItem || treeItem->filePath() != result.fileName())
|
||||
return false;
|
||||
const bool parameterized = treeItem->states() & CatchTreeItem::Parameterized;
|
||||
return parameterized ? result.name().startsWith(treeItem->name() + " - ")
|
||||
: result.name() == treeItem->name();
|
||||
});
|
||||
};
|
||||
}
|
||||
|
||||
CatchResult::CatchResult(const QString &id, const QString &name, int depth)
|
||||
: TestResult(id, name, {{}, findTestItemHook()})
|
||||
, m_sectionDepth(depth) {}
|
||||
|
||||
bool CatchResult::isDirectParentOf(const TestResult *other, bool *needsIntermediate) const
|
||||
{
|
||||
if (!TestResult::isDirectParentOf(other, needsIntermediate))
|
||||
@@ -43,26 +64,5 @@ bool CatchResult::isDirectParentOf(const TestResult *other, bool *needsIntermedi
|
||||
return name() == catchOther->name();
|
||||
}
|
||||
|
||||
const ITestTreeItem *CatchResult::findTestTreeItem() const
|
||||
{
|
||||
auto id = Utils::Id(Constants::FRAMEWORK_PREFIX).withSuffix("Catch");
|
||||
ITestFramework *framework = TestFrameworkManager::frameworkForId(id);
|
||||
QTC_ASSERT(framework, return nullptr);
|
||||
const TestTreeItem *rootNode = framework->rootNode();
|
||||
if (!rootNode)
|
||||
return nullptr;
|
||||
|
||||
const QString tcName = name();
|
||||
const Utils::FilePath tcFilePath = fileName();
|
||||
return rootNode->findAnyChild([&tcName, &tcFilePath](const Utils::TreeItem *item) {
|
||||
const auto treeItem = static_cast<const CatchTreeItem *>(item);
|
||||
if (!treeItem || treeItem->filePath() != tcFilePath)
|
||||
return false;
|
||||
const bool parameterized = treeItem->states() & CatchTreeItem::Parameterized;
|
||||
return parameterized ? tcName.startsWith(treeItem->name() + " - ")
|
||||
: tcName == treeItem->name();
|
||||
});
|
||||
}
|
||||
|
||||
} // namespace Internal
|
||||
} // namespace Autotest
|
||||
|
@@ -14,7 +14,6 @@ public:
|
||||
CatchResult(const QString &id, const QString &name, int depth);
|
||||
|
||||
bool isDirectParentOf(const TestResult *other, bool *needsIntermediate) const override;
|
||||
const ITestTreeItem *findTestTreeItem() const override;
|
||||
|
||||
private:
|
||||
int sectionDepth() const { return m_sectionDepth; }
|
||||
|
@@ -3,26 +3,45 @@
|
||||
|
||||
#include "ctestoutputreader.h"
|
||||
|
||||
#include "ctesttreeitem.h"
|
||||
#include "../autotesttr.h"
|
||||
#include "../testframeworkmanager.h"
|
||||
#include "../testresult.h"
|
||||
#include "../testtreeitem.h"
|
||||
|
||||
#include <cmakeprojectmanager/cmakeprojectconstants.h>
|
||||
|
||||
#include <utils/qtcassert.h>
|
||||
#include <utils/treemodel.h>
|
||||
|
||||
#include <QRegularExpression>
|
||||
|
||||
using namespace Utils;
|
||||
|
||||
namespace Autotest {
|
||||
namespace Internal {
|
||||
|
||||
static ResultHooks::FindTestItemHook findTestItemHook(const QString &testCaseName)
|
||||
{
|
||||
return [=](const TestResult &result) -> ITestTreeItem * {
|
||||
Q_UNUSED(result)
|
||||
ITestTool *testTool = TestFrameworkManager::testToolForBuildSystemId(
|
||||
CMakeProjectManager::Constants::CMAKE_PROJECT_ID);
|
||||
QTC_ASSERT(testTool, return nullptr);
|
||||
const ITestTreeItem *rootNode = testTool->rootNode();
|
||||
if (!rootNode)
|
||||
return nullptr;
|
||||
|
||||
return rootNode->findFirstLevelChild([&](const ITestTreeItem *item) {
|
||||
return item && item->name() == testCaseName;
|
||||
});
|
||||
};
|
||||
}
|
||||
|
||||
class CTestResult : public TestResult
|
||||
{
|
||||
public:
|
||||
CTestResult(const QString &id, const QString &project, const QString &testCase)
|
||||
: TestResult(id, project)
|
||||
, m_testCase(testCase)
|
||||
CTestResult(const QString &id, const QString &project, const QString &testCaseName)
|
||||
: TestResult(id, project, {{}, findTestItemHook(testCaseName)})
|
||||
{}
|
||||
|
||||
bool isDirectParentOf(const TestResult *other, bool *needsIntermediate) const override
|
||||
@@ -31,23 +50,6 @@ public:
|
||||
return false;
|
||||
return result() == ResultType::TestStart;
|
||||
}
|
||||
|
||||
const ITestTreeItem *findTestTreeItem() const override
|
||||
{
|
||||
ITestTool *testTool = TestFrameworkManager::testToolForBuildSystemId(
|
||||
CMakeProjectManager::Constants::CMAKE_PROJECT_ID);
|
||||
QTC_ASSERT(testTool, return nullptr);
|
||||
const ITestTreeItem *rootNode = testTool->rootNode();
|
||||
if (!rootNode)
|
||||
return nullptr;
|
||||
|
||||
return rootNode->findFirstLevelChild([this](const ITestTreeItem *item) {
|
||||
return item && item->name() == m_testCase;
|
||||
});
|
||||
}
|
||||
|
||||
private:
|
||||
QString m_testCase;
|
||||
};
|
||||
|
||||
CTestOutputReader::CTestOutputReader(const QFutureInterface<TestResultPtr> &futureInterface,
|
||||
|
@@ -11,6 +11,8 @@
|
||||
|
||||
#include <QRegularExpression>
|
||||
|
||||
using namespace Utils;
|
||||
|
||||
namespace Autotest {
|
||||
namespace Internal {
|
||||
|
||||
@@ -35,15 +37,74 @@ static ResultHooks::OutputStringHook outputStringHook(const QString &testCaseNam
|
||||
};
|
||||
}
|
||||
|
||||
GTestResult::GTestResult(const QString &id, const QString &name, const Utils::FilePath &projectFile,
|
||||
const QString &testCaseName, int iteration)
|
||||
: TestResult(id, name, {outputStringHook(testCaseName)})
|
||||
, m_projectFile(projectFile)
|
||||
, m_testCaseName(testCaseName)
|
||||
, m_iteration(iteration)
|
||||
static QString normalizeName(const QString &name)
|
||||
{
|
||||
static QRegularExpression parameterIndex("/\\d+");
|
||||
|
||||
QString nameWithoutParameterIndices = name;
|
||||
nameWithoutParameterIndices.remove(parameterIndex);
|
||||
|
||||
return nameWithoutParameterIndices.split('/').last();
|
||||
}
|
||||
|
||||
static QString normalizeTestName(const QString &testname)
|
||||
{
|
||||
return normalizeName(testname.split(',').first());
|
||||
}
|
||||
|
||||
static bool matchesTestCase(const QString &testCaseName, const TestTreeItem *treeItem)
|
||||
{
|
||||
if (treeItem->type() != TestTreeItem::TestCase)
|
||||
return false;
|
||||
|
||||
const QString testItemTestCase = treeItem->parentItem()->name() + '.' + treeItem->name();
|
||||
return testItemTestCase == normalizeName(testCaseName);
|
||||
}
|
||||
|
||||
static bool matchesTestSuite(const QString &name, const TestTreeItem *treeItem)
|
||||
{
|
||||
if (treeItem->type() != TestTreeItem::TestSuite)
|
||||
return false;
|
||||
|
||||
return treeItem->name() == normalizeTestName(name);
|
||||
}
|
||||
|
||||
static bool matches(const QString &name, const FilePath &projectFile,
|
||||
const QString &testCaseName, const TestTreeItem *treeItem)
|
||||
{
|
||||
if (treeItem->proFile() != projectFile)
|
||||
return false;
|
||||
|
||||
if (!testCaseName.isEmpty())
|
||||
return matchesTestCase(testCaseName, treeItem);
|
||||
return matchesTestSuite(name, treeItem);
|
||||
}
|
||||
|
||||
static ResultHooks::FindTestItemHook findTestItemHook(const FilePath &projectFile,
|
||||
const QString &testCaseName)
|
||||
{
|
||||
return [=](const TestResult &result) -> ITestTreeItem * {
|
||||
const Id id = Id(Constants::FRAMEWORK_PREFIX).withSuffix(GTest::Constants::FRAMEWORK_NAME);
|
||||
ITestFramework *framework = TestFrameworkManager::frameworkForId(id);
|
||||
QTC_ASSERT(framework, return nullptr);
|
||||
const TestTreeItem *rootNode = framework->rootNode();
|
||||
if (!rootNode)
|
||||
return nullptr;
|
||||
|
||||
return rootNode->findAnyChild([&](const TreeItem *item) {
|
||||
const auto testTreeItem = static_cast<const TestTreeItem *>(item);
|
||||
return testTreeItem && matches(result.name(), projectFile, testCaseName, testTreeItem);
|
||||
});
|
||||
};
|
||||
}
|
||||
|
||||
GTestResult::GTestResult(const QString &id, const QString &name, const FilePath &projectFile,
|
||||
const QString &testCaseName, int iteration)
|
||||
: TestResult(id, name, {outputStringHook(testCaseName),
|
||||
findTestItemHook(projectFile, testCaseName)})
|
||||
, m_testCaseName(testCaseName)
|
||||
, m_iteration(iteration) {}
|
||||
|
||||
bool GTestResult::isDirectParentOf(const TestResult *other, bool *needsIntermediate) const
|
||||
{
|
||||
if (!TestResult::isDirectParentOf(other, needsIntermediate))
|
||||
@@ -60,65 +121,5 @@ bool GTestResult::isDirectParentOf(const TestResult *other, bool *needsIntermedi
|
||||
return isTestSuite() && gtOther->isTestCase();
|
||||
}
|
||||
|
||||
static QString normalizeName(const QString &name)
|
||||
{
|
||||
static QRegularExpression parameterIndex("/\\d+");
|
||||
|
||||
QString nameWithoutParameterIndices = name;
|
||||
nameWithoutParameterIndices.remove(parameterIndex);
|
||||
|
||||
return nameWithoutParameterIndices.split('/').last();
|
||||
}
|
||||
|
||||
static QString normalizeTestName(const QString &testname)
|
||||
{
|
||||
QString nameWithoutTypeParam = testname.split(',').first();
|
||||
|
||||
return normalizeName(nameWithoutTypeParam);
|
||||
}
|
||||
|
||||
const ITestTreeItem *GTestResult::findTestTreeItem() const
|
||||
{
|
||||
auto id = Utils::Id(Constants::FRAMEWORK_PREFIX).withSuffix(GTest::Constants::FRAMEWORK_NAME);
|
||||
ITestFramework *framework = TestFrameworkManager::frameworkForId(id);
|
||||
QTC_ASSERT(framework, return nullptr);
|
||||
const TestTreeItem *rootNode = framework->rootNode();
|
||||
if (!rootNode)
|
||||
return nullptr;
|
||||
|
||||
return rootNode->findAnyChild([this](const Utils::TreeItem *item) {
|
||||
const auto treeItem = static_cast<const TestTreeItem *>(item);
|
||||
return treeItem && matches(treeItem);
|
||||
});
|
||||
}
|
||||
|
||||
bool GTestResult::matches(const TestTreeItem *treeItem) const
|
||||
{
|
||||
if (treeItem->proFile() != m_projectFile)
|
||||
return false;
|
||||
|
||||
if (isTestSuite())
|
||||
return matchesTestSuite(treeItem);
|
||||
|
||||
return matchesTestCase(treeItem);
|
||||
}
|
||||
|
||||
bool GTestResult::matchesTestCase(const TestTreeItem *treeItem) const
|
||||
{
|
||||
if (treeItem->type() != TestTreeItem::TestCase)
|
||||
return false;
|
||||
|
||||
const QString testItemTestCase = treeItem->parentItem()->name() + '.' + treeItem->name();
|
||||
return testItemTestCase == normalizeName(m_testCaseName);
|
||||
}
|
||||
|
||||
bool GTestResult::matchesTestSuite(const TestTreeItem *treeItem) const
|
||||
{
|
||||
if (treeItem->type() != TestTreeItem::TestSuite)
|
||||
return false;
|
||||
|
||||
return treeItem->name() == normalizeTestName(name());
|
||||
}
|
||||
|
||||
} // namespace Internal
|
||||
} // namespace Autotest
|
||||
|
@@ -18,17 +18,11 @@ public:
|
||||
const QString &testCaseName = {}, int iteration = 1);
|
||||
|
||||
bool isDirectParentOf(const TestResult *other, bool *needsIntermediate) const override;
|
||||
virtual const ITestTreeItem *findTestTreeItem() const override;
|
||||
|
||||
private:
|
||||
bool isTestSuite() const { return m_testCaseName.isEmpty(); }
|
||||
bool isTestCase() const { return !m_testCaseName.isEmpty(); }
|
||||
|
||||
bool matches(const TestTreeItem *item) const;
|
||||
bool matchesTestCase(const TestTreeItem *treeItem) const;
|
||||
bool matchesTestSuite(const TestTreeItem *treeItem) const;
|
||||
|
||||
Utils::FilePath m_projectFile;
|
||||
QString m_testCaseName;
|
||||
int m_iteration = 1;
|
||||
};
|
||||
|
@@ -9,12 +9,14 @@
|
||||
#include <utils/id.h>
|
||||
#include <utils/qtcassert.h>
|
||||
|
||||
using namespace Utils;
|
||||
|
||||
namespace Autotest {
|
||||
namespace Internal {
|
||||
|
||||
static ResultHooks::OutputStringHook outputStringHook(const QString &function, const QString &dataTag)
|
||||
{
|
||||
return [function, dataTag](const TestResult &result, bool selected) {
|
||||
return [=](const TestResult &result, bool selected) {
|
||||
const QString &desc = result.description();
|
||||
const QString &className = result.name();
|
||||
QString output;
|
||||
@@ -56,10 +58,97 @@ static ResultHooks::OutputStringHook outputStringHook(const QString &function, c
|
||||
};
|
||||
}
|
||||
|
||||
QtTestResult::QtTestResult(const QString &id, const QString &name,
|
||||
const Utils::FilePath &projectFile, TestType type,
|
||||
const QString &functionName, const QString &dataTag)
|
||||
: TestResult(id, name, {outputStringHook(functionName, dataTag)})
|
||||
static bool matchesTestCase(const QString &name, const TestTreeItem *item)
|
||||
{
|
||||
// FIXME this will never work for Quick Tests
|
||||
return item->name() == name;
|
||||
}
|
||||
|
||||
static bool matchesTestFunction(const QString &name, TestType testType, const QString &functionName,
|
||||
const QString &dataTag, const TestTreeItem *item)
|
||||
{
|
||||
TestTreeItem *parentItem = item->parentItem();
|
||||
const TestTreeItem::Type type = item->type();
|
||||
if (testType == TestType::QuickTest) { // Quick tests have slightly different layout // BAD/WRONG!
|
||||
const QStringList tmp = functionName.split("::");
|
||||
return tmp.size() == 2 && item->name() == tmp.last() && parentItem->name() == tmp.first();
|
||||
}
|
||||
if (type == TestTreeItem::TestDataTag) {
|
||||
TestTreeItem *grandParentItem = parentItem->parentItem();
|
||||
return parentItem->name() == functionName && grandParentItem->name() == name
|
||||
&& item->name() == dataTag;
|
||||
}
|
||||
return item->name() == functionName && parentItem->name() == name;
|
||||
}
|
||||
|
||||
static bool matches(const QString &name, const FilePath &projectFile, TestType testType,
|
||||
const QString &functionName, const QString &dataTag, const TestTreeItem *item)
|
||||
{
|
||||
const auto isTestCase = [&] { return functionName.isEmpty() && dataTag.isEmpty(); };
|
||||
const auto isTestFunction = [&] { return !functionName.isEmpty() && dataTag.isEmpty(); };
|
||||
const auto isDataTag = [&] { return !functionName.isEmpty() && !dataTag.isEmpty(); };
|
||||
|
||||
QTC_ASSERT(item, return false);
|
||||
TestTreeItem *parentItem = item->parentItem();
|
||||
QTC_ASSERT(parentItem, return false);
|
||||
|
||||
TestTreeItem::Type type = item->type();
|
||||
switch (type) {
|
||||
case TestTreeItem::TestCase:
|
||||
if (!isTestCase())
|
||||
return false;
|
||||
if (item->proFile() != projectFile)
|
||||
return false;
|
||||
return matchesTestCase(name, item);
|
||||
case TestTreeItem::TestFunction:
|
||||
case TestTreeItem::TestSpecialFunction:
|
||||
// QuickTest data tags have no dedicated TestTreeItem, so treat these results as function
|
||||
if (!isTestFunction() && !(testType == TestType::QuickTest && isDataTag()))
|
||||
return false;
|
||||
if (parentItem->proFile() != projectFile)
|
||||
return false;
|
||||
return matchesTestFunction(name, testType, functionName, dataTag, item);
|
||||
case TestTreeItem::TestDataTag: {
|
||||
if (!isDataTag())
|
||||
return false;
|
||||
TestTreeItem *grandParentItem = parentItem->parentItem();
|
||||
QTC_ASSERT(grandParentItem, return false);
|
||||
if (grandParentItem->proFile() != projectFile)
|
||||
return false;
|
||||
return matchesTestFunction(name, testType, functionName, dataTag, item);
|
||||
}
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
static ResultHooks::FindTestItemHook findTestItemHook(const FilePath &projectFile, TestType type,
|
||||
const QString &functionName,
|
||||
const QString &dataTag)
|
||||
{
|
||||
return [=](const TestResult &result) -> ITestTreeItem * {
|
||||
const Id id = type == TestType::QtTest
|
||||
? Id(Constants::FRAMEWORK_PREFIX).withSuffix(QtTest::Constants::FRAMEWORK_NAME)
|
||||
: Id(Constants::FRAMEWORK_PREFIX).withSuffix(QuickTest::Constants::FRAMEWORK_NAME);
|
||||
ITestFramework *framework = TestFrameworkManager::frameworkForId(id);
|
||||
QTC_ASSERT(framework, return nullptr);
|
||||
const TestTreeItem *rootNode = framework->rootNode();
|
||||
QTC_ASSERT(rootNode, return nullptr);
|
||||
|
||||
return rootNode->findAnyChild([&](const TreeItem *item) {
|
||||
const TestTreeItem *testTreeItem = static_cast<const TestTreeItem *>(item);
|
||||
return testTreeItem && matches(result.name(), projectFile, type, functionName, dataTag,
|
||||
testTreeItem);
|
||||
});
|
||||
};
|
||||
}
|
||||
|
||||
QtTestResult::QtTestResult(const QString &id, const QString &name, const FilePath &projectFile,
|
||||
TestType type, const QString &functionName, const QString &dataTag)
|
||||
: TestResult(id, name, {outputStringHook(functionName, dataTag),
|
||||
findTestItemHook(projectFile, type, functionName, dataTag)})
|
||||
, m_projectFile(projectFile)
|
||||
, m_type(type)
|
||||
, m_function(functionName)
|
||||
@@ -116,85 +205,5 @@ TestResult *QtTestResult::createIntermediateResultFor(const TestResult *other) c
|
||||
return intermediate;
|
||||
}
|
||||
|
||||
const ITestTreeItem *QtTestResult::findTestTreeItem() const
|
||||
{
|
||||
Utils::Id id;
|
||||
if (m_type == TestType::QtTest)
|
||||
id = Utils::Id(Constants::FRAMEWORK_PREFIX).withSuffix(QtTest::Constants::FRAMEWORK_NAME);
|
||||
else
|
||||
id = Utils::Id(Constants::FRAMEWORK_PREFIX).withSuffix(QuickTest::Constants::FRAMEWORK_NAME);
|
||||
ITestFramework *framework = TestFrameworkManager::frameworkForId(id);
|
||||
QTC_ASSERT(framework, return nullptr);
|
||||
const TestTreeItem *rootNode = framework->rootNode();
|
||||
QTC_ASSERT(rootNode, return nullptr);
|
||||
|
||||
return rootNode->findAnyChild([this](const Utils::TreeItem *item) {
|
||||
const TestTreeItem *treeItem = static_cast<const TestTreeItem *>(item);
|
||||
return treeItem && matches(treeItem);
|
||||
});
|
||||
}
|
||||
|
||||
bool QtTestResult::matches(const TestTreeItem *item) const
|
||||
{
|
||||
QTC_ASSERT(item, return false);
|
||||
TestTreeItem *parentItem = item->parentItem();
|
||||
QTC_ASSERT(parentItem, return false);
|
||||
|
||||
TestTreeItem::Type type = item->type();
|
||||
switch (type) {
|
||||
case TestTreeItem::TestCase:
|
||||
if (!isTestCase())
|
||||
return false;
|
||||
if (item->proFile() != m_projectFile)
|
||||
return false;
|
||||
return matchesTestCase(item);
|
||||
case TestTreeItem::TestFunction:
|
||||
case TestTreeItem::TestSpecialFunction:
|
||||
// QuickTest data tags have no dedicated TestTreeItem, so treat these results as function
|
||||
if (!isTestFunction() && !(m_type == TestType::QuickTest && isDataTag()))
|
||||
return false;
|
||||
if (parentItem->proFile() != m_projectFile)
|
||||
return false;
|
||||
return matchesTestFunction(item);
|
||||
case TestTreeItem::TestDataTag: {
|
||||
if (!isDataTag())
|
||||
return false;
|
||||
TestTreeItem *grandParentItem = parentItem->parentItem();
|
||||
QTC_ASSERT(grandParentItem, return false);
|
||||
if (grandParentItem->proFile() != m_projectFile)
|
||||
return false;
|
||||
return matchesTestFunction(item);
|
||||
}
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
bool QtTestResult::matchesTestCase(const TestTreeItem *item) const
|
||||
{
|
||||
// FIXME this will never work for Quick Tests
|
||||
if (item->name() == name())
|
||||
return true;
|
||||
return false;
|
||||
}
|
||||
|
||||
bool QtTestResult::matchesTestFunction(const TestTreeItem *item) const
|
||||
{
|
||||
TestTreeItem *parentItem = item->parentItem();
|
||||
TestTreeItem::Type type = item->type();
|
||||
if (m_type == TestType::QuickTest) { // Quick tests have slightly different layout // BAD/WRONG!
|
||||
const QStringList tmp = m_function.split("::");
|
||||
return tmp.size() == 2 && item->name() == tmp.last() && parentItem->name() == tmp.first();
|
||||
}
|
||||
if (type == TestTreeItem::TestDataTag) {
|
||||
TestTreeItem *grandParentItem = parentItem->parentItem();
|
||||
return parentItem->name() == m_function && grandParentItem->name() == name()
|
||||
&& item->name() == m_dataTag;
|
||||
}
|
||||
return item->name() == m_function && parentItem->name() == name();
|
||||
}
|
||||
|
||||
} // namespace Internal
|
||||
} // namespace Autotest
|
||||
|
@@ -21,16 +21,12 @@ public:
|
||||
bool isDirectParentOf(const TestResult *other, bool *needsIntermediate) const override;
|
||||
bool isIntermediateFor(const TestResult *other) const override;
|
||||
TestResult *createIntermediateResultFor(const TestResult *other) const override;
|
||||
const ITestTreeItem *findTestTreeItem() const override;
|
||||
|
||||
private:
|
||||
bool isTestCase() const { return m_function.isEmpty() && m_dataTag.isEmpty(); }
|
||||
bool isTestFunction() const { return !m_function.isEmpty() && m_dataTag.isEmpty(); }
|
||||
bool isDataTag() const { return !m_function.isEmpty() && !m_dataTag.isEmpty(); }
|
||||
|
||||
bool matches(const TestTreeItem *item) const;
|
||||
bool matchesTestCase(const TestTreeItem *item) const;
|
||||
bool matchesTestFunction(const TestTreeItem *item) const;
|
||||
|
||||
Utils::FilePath m_projectFile;
|
||||
TestType m_type;
|
||||
QString m_function;
|
||||
|
@@ -27,6 +27,8 @@ const QString TestResult::outputString(bool selected) const
|
||||
|
||||
const ITestTreeItem *TestResult::findTestTreeItem() const
|
||||
{
|
||||
if (m_hooks.findTestItem)
|
||||
return m_hooks.findTestItem(*this);
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
|
@@ -63,7 +63,9 @@ class TestResult;
|
||||
struct ResultHooks
|
||||
{
|
||||
using OutputStringHook = std::function<QString(const TestResult &, bool)>;
|
||||
using FindTestItemHook = std::function<ITestTreeItem *(const TestResult &)>;
|
||||
OutputStringHook outputString;
|
||||
FindTestItemHook findTestItem;
|
||||
};
|
||||
|
||||
class TestResult
|
||||
@@ -74,7 +76,7 @@ public:
|
||||
virtual ~TestResult() {}
|
||||
|
||||
const QString outputString(bool selected) const;
|
||||
virtual const ITestTreeItem *findTestTreeItem() const;
|
||||
const ITestTreeItem *findTestTreeItem() const;
|
||||
|
||||
QString id() const { return m_id; }
|
||||
QString name() const { return m_name; }
|
||||
|
Reference in New Issue
Block a user