2016-05-11 13:02:42 +02:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
|
|
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
|
|
|
|
**
|
|
|
|
|
** This file is part of Qt Creator.
|
|
|
|
|
**
|
|
|
|
|
** Commercial License Usage
|
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
|
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
|
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
|
|
|
|
**
|
|
|
|
|
** GNU General Public License Usage
|
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
|
|
|
** General Public License version 3 as published by the Free Software
|
|
|
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
|
|
|
** included in the packaging of this file. Please review the following
|
|
|
|
|
** information to ensure the GNU General Public License requirements will
|
|
|
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
|
|
|
|
**
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include "quicktesttreeitem.h"
|
|
|
|
|
#include "quicktestconfiguration.h"
|
|
|
|
|
#include "quicktestparser.h"
|
|
|
|
|
|
2017-06-09 09:30:21 +02:00
|
|
|
#include <cpptools/cppmodelmanager.h>
|
2016-05-11 13:02:42 +02:00
|
|
|
#include <projectexplorer/session.h>
|
|
|
|
|
#include <utils/qtcassert.h>
|
|
|
|
|
|
|
|
|
|
namespace Autotest {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
|
|
QVariant QuickTestTreeItem::data(int column, int role) const
|
|
|
|
|
{
|
|
|
|
|
switch (role) {
|
|
|
|
|
case Qt::DisplayRole:
|
|
|
|
|
if (type() == TestCase && name().isEmpty())
|
|
|
|
|
return QCoreApplication::translate("QuickTestTreeItem", "<unnamed>");
|
|
|
|
|
break;
|
|
|
|
|
case Qt::ToolTipRole:
|
|
|
|
|
if (type() == TestCase && name().isEmpty())
|
|
|
|
|
return QCoreApplication::translate("QuickTestTreeItem",
|
|
|
|
|
"<p>Give all test cases a name to ensure correct "
|
|
|
|
|
"behavior when running test cases and to be able to "
|
|
|
|
|
"select them.</p>");
|
|
|
|
|
break;
|
|
|
|
|
case Qt::CheckStateRole:
|
|
|
|
|
switch (type()) {
|
|
|
|
|
case Root:
|
|
|
|
|
case TestDataFunction:
|
|
|
|
|
case TestSpecialFunction:
|
|
|
|
|
case TestDataTag:
|
|
|
|
|
return QVariant();
|
|
|
|
|
case TestCase:
|
|
|
|
|
return name().isEmpty() ? QVariant() : checked();
|
|
|
|
|
case TestFunctionOrSet:
|
|
|
|
|
return (parentItem() && !parentItem()->name().isEmpty()) ? checked() : QVariant();
|
|
|
|
|
default:
|
|
|
|
|
return checked();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case ItalicRole:
|
|
|
|
|
switch (type()) {
|
|
|
|
|
case TestDataFunction:
|
|
|
|
|
case TestSpecialFunction:
|
|
|
|
|
return true;
|
|
|
|
|
case TestCase:
|
|
|
|
|
return name().isEmpty();
|
|
|
|
|
case TestFunctionOrSet:
|
|
|
|
|
return parentItem() ? parentItem()->name().isEmpty() : false;
|
|
|
|
|
default:
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return TestTreeItem::data(column, role);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Qt::ItemFlags QuickTestTreeItem::flags(int column) const
|
|
|
|
|
{
|
|
|
|
|
// handle unnamed quick tests and their functions
|
|
|
|
|
switch (type()) {
|
|
|
|
|
case TestCase:
|
|
|
|
|
if (name().isEmpty())
|
|
|
|
|
return Qt::ItemIsEnabled | Qt::ItemIsSelectable;
|
|
|
|
|
break;
|
|
|
|
|
case TestFunctionOrSet:
|
|
|
|
|
if (parentItem()->name().isEmpty())
|
|
|
|
|
return Qt::ItemIsEnabled | Qt::ItemIsSelectable;
|
|
|
|
|
break;
|
|
|
|
|
default: {} // avoid warning regarding unhandled enum values
|
|
|
|
|
}
|
|
|
|
|
return TestTreeItem::flags(column);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool QuickTestTreeItem::canProvideTestConfiguration() const
|
|
|
|
|
{
|
|
|
|
|
switch (type()) {
|
|
|
|
|
case TestCase:
|
|
|
|
|
return !name().isEmpty();
|
|
|
|
|
case TestFunctionOrSet:
|
|
|
|
|
return !parentItem()->name().isEmpty();
|
|
|
|
|
default:
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TestConfiguration *QuickTestTreeItem::testConfiguration() const
|
|
|
|
|
{
|
|
|
|
|
ProjectExplorer::Project *project = ProjectExplorer::SessionManager::startupProject();
|
2017-02-13 10:05:06 +01:00
|
|
|
QTC_ASSERT(project, return nullptr);
|
2016-05-11 13:02:42 +02:00
|
|
|
|
2017-02-13 10:05:06 +01:00
|
|
|
QuickTestConfiguration *config = nullptr;
|
2016-05-11 13:02:42 +02:00
|
|
|
switch (type()) {
|
|
|
|
|
case TestCase: {
|
|
|
|
|
QStringList testFunctions;
|
|
|
|
|
for (int row = 0, count = childCount(); row < count; ++row)
|
2016-09-29 12:15:43 +02:00
|
|
|
testFunctions << name() + "::" + childItem(row)->name();
|
2016-05-11 13:02:42 +02:00
|
|
|
config = new QuickTestConfiguration;
|
|
|
|
|
config->setTestCases(testFunctions);
|
2017-02-21 14:53:58 +01:00
|
|
|
config->setProjectFile(proFile());
|
2016-05-11 13:02:42 +02:00
|
|
|
config->setProject(project);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case TestFunctionOrSet: {
|
|
|
|
|
TestTreeItem *parent = parentItem();
|
2016-09-29 12:15:43 +02:00
|
|
|
QStringList testFunction(parent->name() + "::" + name());
|
2016-05-11 13:02:42 +02:00
|
|
|
config = new QuickTestConfiguration;
|
|
|
|
|
config->setTestCases(testFunction);
|
2017-02-21 14:53:58 +01:00
|
|
|
config->setProjectFile(parent->proFile());
|
2016-05-11 13:02:42 +02:00
|
|
|
config->setProject(project);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
default:
|
2017-02-13 10:05:06 +01:00
|
|
|
return nullptr;
|
2016-05-11 13:02:42 +02:00
|
|
|
}
|
2017-06-09 09:30:21 +02:00
|
|
|
if (config)
|
|
|
|
|
config->setInternalTargets(internalTargets());
|
2016-05-11 13:02:42 +02:00
|
|
|
return config;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QList<TestConfiguration *> QuickTestTreeItem::getAllTestConfigurations() const
|
|
|
|
|
{
|
|
|
|
|
QList<TestConfiguration *> result;
|
|
|
|
|
|
|
|
|
|
ProjectExplorer::Project *project = ProjectExplorer::SessionManager::startupProject();
|
|
|
|
|
if (!project || type() != Root)
|
|
|
|
|
return result;
|
|
|
|
|
|
|
|
|
|
QHash<QString, int> foundProFiles;
|
2017-06-09 09:30:21 +02:00
|
|
|
QHash<QString, QSet<QString> > proFilesWithTargets;
|
2016-05-11 13:02:42 +02:00
|
|
|
for (int row = 0, count = childCount(); row < count; ++row) {
|
|
|
|
|
const TestTreeItem *child = childItem(row);
|
|
|
|
|
// unnamed Quick Tests must be handled separately
|
|
|
|
|
if (child->name().isEmpty()) {
|
|
|
|
|
for (int childRow = 0, ccount = child->childCount(); childRow < ccount; ++ childRow) {
|
|
|
|
|
const TestTreeItem *grandChild = child->childItem(childRow);
|
|
|
|
|
const QString &proFile = grandChild->proFile();
|
|
|
|
|
foundProFiles.insert(proFile, foundProFiles[proFile] + 1);
|
2017-06-09 09:30:21 +02:00
|
|
|
proFilesWithTargets.insert(proFile, grandChild->internalTargets());
|
2016-05-11 13:02:42 +02:00
|
|
|
}
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
// named Quick Test
|
|
|
|
|
const QString &proFile = child->proFile();
|
|
|
|
|
foundProFiles.insert(proFile, foundProFiles[proFile] + child->childCount());
|
2017-06-09 09:30:21 +02:00
|
|
|
proFilesWithTargets.insert(proFile, child->internalTargets());
|
2016-05-11 13:02:42 +02:00
|
|
|
}
|
|
|
|
|
// create TestConfiguration for each project file
|
|
|
|
|
QHash<QString, int>::ConstIterator it = foundProFiles.begin();
|
|
|
|
|
QHash<QString, int>::ConstIterator end = foundProFiles.end();
|
|
|
|
|
for ( ; it != end; ++it) {
|
|
|
|
|
QuickTestConfiguration *tc = new QuickTestConfiguration;
|
|
|
|
|
tc->setTestCaseCount(it.value());
|
2017-02-21 14:53:58 +01:00
|
|
|
tc->setProjectFile(it.key());
|
2016-05-11 13:02:42 +02:00
|
|
|
tc->setProject(project);
|
2017-06-09 09:30:21 +02:00
|
|
|
tc->setInternalTargets(proFilesWithTargets[it.key()]);
|
2016-05-11 13:02:42 +02:00
|
|
|
result << tc;
|
|
|
|
|
}
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QList<TestConfiguration *> QuickTestTreeItem::getSelectedTestConfigurations() const
|
|
|
|
|
{
|
|
|
|
|
QList<TestConfiguration *> result;
|
|
|
|
|
ProjectExplorer::Project *project = ProjectExplorer::SessionManager::startupProject();
|
|
|
|
|
if (!project || type() != Root)
|
|
|
|
|
return result;
|
|
|
|
|
|
2017-02-13 10:05:06 +01:00
|
|
|
QuickTestConfiguration *tc = nullptr;
|
2016-05-11 13:02:42 +02:00
|
|
|
QHash<QString, QuickTestConfiguration *> foundProFiles;
|
|
|
|
|
// unnamed Quick Tests must be handled first
|
|
|
|
|
if (TestTreeItem *unnamed = unnamedQuickTests()) {
|
|
|
|
|
for (int childRow = 0, ccount = unnamed->childCount(); childRow < ccount; ++ childRow) {
|
|
|
|
|
const TestTreeItem *grandChild = unnamed->childItem(childRow);
|
|
|
|
|
const QString &proFile = grandChild->proFile();
|
|
|
|
|
if (foundProFiles.contains(proFile)) {
|
|
|
|
|
QTC_ASSERT(tc,
|
|
|
|
|
qWarning() << "Illegal state (unnamed Quick Test listed as named)";
|
|
|
|
|
return QList<TestConfiguration *>());
|
|
|
|
|
foundProFiles[proFile]->setTestCaseCount(tc->testCaseCount() + 1);
|
|
|
|
|
} else {
|
|
|
|
|
tc = new QuickTestConfiguration;
|
|
|
|
|
tc->setTestCaseCount(1);
|
|
|
|
|
tc->setUnnamedOnly(true);
|
2017-02-21 14:53:58 +01:00
|
|
|
tc->setProjectFile(proFile);
|
2016-05-11 13:02:42 +02:00
|
|
|
tc->setProject(project);
|
2017-06-09 09:30:21 +02:00
|
|
|
tc->setInternalTargets(grandChild->internalTargets());
|
2016-05-11 13:02:42 +02:00
|
|
|
foundProFiles.insert(proFile, tc);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (int row = 0, count = childCount(); row < count; ++row) {
|
|
|
|
|
const TestTreeItem *child = childItem(row);
|
|
|
|
|
// unnamed Quick Tests have been handled separately already
|
|
|
|
|
if (child->name().isEmpty())
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
// named Quick Tests
|
|
|
|
|
switch (child->checked()) {
|
|
|
|
|
case Qt::Unchecked:
|
|
|
|
|
continue;
|
|
|
|
|
case Qt::Checked:
|
|
|
|
|
case Qt::PartiallyChecked:
|
|
|
|
|
default:
|
|
|
|
|
QStringList testFunctions;
|
|
|
|
|
int grandChildCount = child->childCount();
|
|
|
|
|
for (int grandChildRow = 0; grandChildRow < grandChildCount; ++grandChildRow) {
|
|
|
|
|
const TestTreeItem *grandChild = child->childItem(grandChildRow);
|
2016-10-10 12:42:48 +02:00
|
|
|
if (grandChild->checked() != Qt::Checked || grandChild->type() != TestFunctionOrSet)
|
2016-05-11 13:02:42 +02:00
|
|
|
continue;
|
2016-09-29 12:15:43 +02:00
|
|
|
testFunctions << child->name() + "::" + grandChild->name();
|
2016-05-11 13:02:42 +02:00
|
|
|
}
|
|
|
|
|
if (foundProFiles.contains(child->proFile())) {
|
|
|
|
|
tc = foundProFiles[child->proFile()];
|
|
|
|
|
QStringList oldFunctions(tc->testCases());
|
|
|
|
|
// if oldFunctions.size() is 0 this test configuration is used for at least one
|
|
|
|
|
// unnamed test case
|
|
|
|
|
if (oldFunctions.size() == 0) {
|
|
|
|
|
tc->setTestCaseCount(tc->testCaseCount() + testFunctions.size());
|
|
|
|
|
tc->setUnnamedOnly(false);
|
|
|
|
|
} else {
|
|
|
|
|
oldFunctions << testFunctions;
|
|
|
|
|
tc->setTestCases(oldFunctions);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
tc = new QuickTestConfiguration;
|
|
|
|
|
tc->setTestCases(testFunctions);
|
2017-02-21 14:53:58 +01:00
|
|
|
tc->setProjectFile(child->proFile());
|
2016-05-11 13:02:42 +02:00
|
|
|
tc->setProject(project);
|
2017-06-09 09:30:21 +02:00
|
|
|
tc->setInternalTargets(child->internalTargets());
|
2016-05-11 13:02:42 +02:00
|
|
|
foundProFiles.insert(child->proFile(), tc);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
QHash<QString, QuickTestConfiguration *>::ConstIterator it = foundProFiles.begin();
|
|
|
|
|
QHash<QString, QuickTestConfiguration *>::ConstIterator end = foundProFiles.end();
|
|
|
|
|
for ( ; it != end; ++it) {
|
|
|
|
|
QuickTestConfiguration *config = it.value();
|
|
|
|
|
if (!config->unnamedOnly())
|
|
|
|
|
result << config;
|
|
|
|
|
else
|
|
|
|
|
delete config;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TestTreeItem *QuickTestTreeItem::find(const TestParseResult *result)
|
|
|
|
|
{
|
2017-02-13 10:05:06 +01:00
|
|
|
QTC_ASSERT(result, return nullptr);
|
2016-05-11 13:02:42 +02:00
|
|
|
|
|
|
|
|
switch (type()) {
|
|
|
|
|
case Root:
|
|
|
|
|
return result->name.isEmpty() ? unnamedQuickTests() : findChildByFile(result->fileName);
|
|
|
|
|
case TestCase:
|
|
|
|
|
return name().isEmpty() ? findChildByNameAndFile(result->name, result->fileName)
|
|
|
|
|
: findChildByName(result->name);
|
|
|
|
|
default:
|
2017-02-13 10:05:06 +01:00
|
|
|
return nullptr;
|
2016-05-11 13:02:42 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool QuickTestTreeItem::modify(const TestParseResult *result)
|
|
|
|
|
{
|
|
|
|
|
QTC_ASSERT(result, return false);
|
|
|
|
|
|
|
|
|
|
switch (type()) {
|
|
|
|
|
case TestCase:
|
2017-05-10 13:42:44 +02:00
|
|
|
return result->name.isEmpty() ? false : modifyTestCaseContent(result);
|
2016-05-11 13:02:42 +02:00
|
|
|
case TestFunctionOrSet:
|
|
|
|
|
case TestDataFunction:
|
|
|
|
|
case TestSpecialFunction:
|
2017-05-10 13:42:44 +02:00
|
|
|
return name().isEmpty() ? modifyLineAndColumn(result)
|
2016-05-11 13:02:42 +02:00
|
|
|
: modifyTestFunctionContent(result);
|
|
|
|
|
default:
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool QuickTestTreeItem::lessThan(const TestTreeItem *other, TestTreeItem::SortMode mode) const
|
|
|
|
|
{
|
|
|
|
|
// handle special item (<unnamed>)
|
|
|
|
|
if (name().isEmpty())
|
|
|
|
|
return false;
|
|
|
|
|
if (other->name().isEmpty())
|
|
|
|
|
return true;
|
|
|
|
|
return TestTreeItem::lessThan(other, mode);
|
|
|
|
|
}
|
|
|
|
|
|
2017-06-09 09:30:21 +02:00
|
|
|
QSet<QString> QuickTestTreeItem::internalTargets() const
|
|
|
|
|
{
|
|
|
|
|
QSet<QString> result;
|
|
|
|
|
const auto cppMM = CppTools::CppModelManager::instance();
|
|
|
|
|
const auto projectInfo = cppMM->projectInfo(ProjectExplorer::SessionManager::startupProject());
|
|
|
|
|
for (const CppTools::ProjectPart::Ptr projectPart : projectInfo.projectParts()) {
|
|
|
|
|
if (projectPart->projectFile == proFile()) {
|
|
|
|
|
result.insert(projectPart->buildSystemTarget);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2016-05-11 13:02:42 +02:00
|
|
|
TestTreeItem *QuickTestTreeItem::unnamedQuickTests() const
|
|
|
|
|
{
|
|
|
|
|
if (type() != Root)
|
2017-02-13 10:05:06 +01:00
|
|
|
return nullptr;
|
2016-05-11 13:02:42 +02:00
|
|
|
|
|
|
|
|
for (int row = 0, count = childCount(); row < count; ++row) {
|
|
|
|
|
TestTreeItem *child = childItem(row);
|
|
|
|
|
if (child->name().isEmpty())
|
|
|
|
|
return child;
|
|
|
|
|
}
|
2017-02-13 10:05:06 +01:00
|
|
|
return nullptr;
|
2016-05-11 13:02:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace Autotest
|