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 "quicktestvisitors.h"
|
|
|
|
|
|
2018-07-09 10:43:31 +02:00
|
|
|
#include <cplusplus/Overview.h>
|
2016-05-11 13:02:42 +02:00
|
|
|
#include <qmljs/parser/qmljsast_p.h>
|
2017-02-27 09:30:16 +01:00
|
|
|
#include <qmljs/qmljsbind.h>
|
|
|
|
|
#include <qmljs/qmljslink.h>
|
|
|
|
|
#include <qmljs/qmljsutils.h>
|
|
|
|
|
#include <utils/algorithm.h>
|
2016-05-11 13:02:42 +02:00
|
|
|
|
|
|
|
|
namespace Autotest {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
2017-02-22 15:09:35 +01:00
|
|
|
static QStringList specialFunctions({"initTestCase", "cleanupTestCase", "init", "cleanup"});
|
2016-05-11 13:02:42 +02:00
|
|
|
|
2017-02-27 09:30:16 +01:00
|
|
|
TestQmlVisitor::TestQmlVisitor(QmlJS::Document::Ptr doc, const QmlJS::Snapshot &snapshot)
|
2016-05-11 13:02:42 +02:00
|
|
|
: m_currentDoc(doc)
|
2017-02-27 09:30:16 +01:00
|
|
|
, m_snapshot(snapshot)
|
2016-05-11 13:02:42 +02:00
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2017-02-27 09:30:16 +01:00
|
|
|
static bool documentImportsQtTest(const QmlJS::Document *doc)
|
|
|
|
|
{
|
|
|
|
|
if (const QmlJS::Bind *bind = doc->bind()) {
|
|
|
|
|
return Utils::anyOf(bind->imports(), [] (const QmlJS::ImportInfo &info) {
|
|
|
|
|
return info.isValid() && info.name() == "QtTest";
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static bool isDerivedFromTestCase(QmlJS::AST::UiQualifiedId *id, const QmlJS::Document::Ptr &doc,
|
|
|
|
|
const QmlJS::Snapshot &snapshot)
|
|
|
|
|
{
|
|
|
|
|
if (!id)
|
|
|
|
|
return false;
|
|
|
|
|
QmlJS::Link link(snapshot, QmlJS::ViewerContext(), QmlJS::LibraryInfo());
|
|
|
|
|
const QmlJS::ContextPtr context = link();
|
|
|
|
|
|
|
|
|
|
const QmlJS::ObjectValue *value = context->lookupType(doc.data(), id);
|
|
|
|
|
if (!value)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
QmlJS::PrototypeIterator protoIterator(value, context);
|
|
|
|
|
const QList<const QmlJS::ObjectValue *> prototypes = protoIterator.all();
|
|
|
|
|
for (const QmlJS::ObjectValue *val : prototypes) {
|
|
|
|
|
if (auto prototype = val->prototype()) {
|
|
|
|
|
if (auto qmlPrototypeRef = prototype->asQmlPrototypeReference()) {
|
|
|
|
|
if (auto qmlTypeName = qmlPrototypeRef->qmlTypeName()) {
|
|
|
|
|
if (qmlTypeName->name == "TestCase") {
|
|
|
|
|
if (auto astObjVal = val->asAstObjectValue())
|
|
|
|
|
return documentImportsQtTest(astObjVal->document());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2016-05-11 13:02:42 +02:00
|
|
|
bool TestQmlVisitor::visit(QmlJS::AST::UiObjectDefinition *ast)
|
|
|
|
|
{
|
|
|
|
|
const QStringRef name = ast->qualifiedTypeNameId->name;
|
2018-06-20 10:12:04 +02:00
|
|
|
m_objectStack.push(name.toString());
|
2017-02-27 09:30:16 +01:00
|
|
|
if (name != "TestCase") {
|
2018-06-20 10:12:04 +02:00
|
|
|
m_insideTestCase = false;
|
2017-02-27 09:30:16 +01:00
|
|
|
if (!isDerivedFromTestCase(ast->qualifiedTypeNameId, m_currentDoc, m_snapshot))
|
|
|
|
|
return true;
|
|
|
|
|
} else if (!documentImportsQtTest(m_currentDoc.data())) {
|
2016-05-11 13:02:42 +02:00
|
|
|
return true; // find nested TestCase items as well
|
2017-02-27 09:30:16 +01:00
|
|
|
}
|
2016-05-11 13:02:42 +02:00
|
|
|
|
2017-02-27 09:30:16 +01:00
|
|
|
m_typeIsTestCase = true;
|
2018-06-20 10:12:04 +02:00
|
|
|
m_insideTestCase = true;
|
2016-05-11 13:02:42 +02:00
|
|
|
m_currentTestCaseName.clear();
|
|
|
|
|
const auto sourceLocation = ast->firstSourceLocation();
|
|
|
|
|
m_testCaseLocation.m_name = m_currentDoc->fileName();
|
|
|
|
|
m_testCaseLocation.m_line = sourceLocation.startLine;
|
|
|
|
|
m_testCaseLocation.m_column = sourceLocation.startColumn - 1;
|
|
|
|
|
m_testCaseLocation.m_type = TestTreeItem::TestCase;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2018-06-20 10:12:04 +02:00
|
|
|
void TestQmlVisitor::endVisit(QmlJS::AST::UiObjectDefinition *)
|
|
|
|
|
{
|
|
|
|
|
m_insideTestCase = m_objectStack.pop() == "TestCase";
|
|
|
|
|
}
|
|
|
|
|
|
2016-05-11 13:02:42 +02:00
|
|
|
bool TestQmlVisitor::visit(QmlJS::AST::ExpressionStatement *ast)
|
|
|
|
|
{
|
|
|
|
|
const QmlJS::AST::ExpressionNode *expr = ast->expression;
|
|
|
|
|
return expr->kind == QmlJS::AST::Node::Kind_StringLiteral;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool TestQmlVisitor::visit(QmlJS::AST::UiScriptBinding *ast)
|
|
|
|
|
{
|
2018-06-20 10:12:04 +02:00
|
|
|
if (m_insideTestCase)
|
|
|
|
|
m_expectTestCaseName = ast->qualifiedId->name == "name";
|
|
|
|
|
return m_expectTestCaseName;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void TestQmlVisitor::endVisit(QmlJS::AST::UiScriptBinding *)
|
|
|
|
|
{
|
|
|
|
|
if (m_expectTestCaseName)
|
|
|
|
|
m_expectTestCaseName = false;
|
2016-05-11 13:02:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool TestQmlVisitor::visit(QmlJS::AST::FunctionDeclaration *ast)
|
|
|
|
|
{
|
|
|
|
|
const QStringRef name = ast->name;
|
2016-09-29 12:15:43 +02:00
|
|
|
if (name.startsWith("test_")
|
|
|
|
|
|| name.startsWith("benchmark_")
|
|
|
|
|
|| name.endsWith("_data")
|
2016-05-11 13:02:42 +02:00
|
|
|
|| specialFunctions.contains(name.toString())) {
|
|
|
|
|
const auto sourceLocation = ast->firstSourceLocation();
|
|
|
|
|
TestCodeLocationAndType locationAndType;
|
|
|
|
|
locationAndType.m_name = m_currentDoc->fileName();
|
|
|
|
|
locationAndType.m_line = sourceLocation.startLine;
|
|
|
|
|
locationAndType.m_column = sourceLocation.startColumn - 1;
|
|
|
|
|
if (specialFunctions.contains(name.toString()))
|
|
|
|
|
locationAndType.m_type = TestTreeItem::TestSpecialFunction;
|
2016-09-29 12:15:43 +02:00
|
|
|
else if (name.endsWith("_data"))
|
2016-05-11 13:02:42 +02:00
|
|
|
locationAndType.m_type = TestTreeItem::TestDataFunction;
|
|
|
|
|
else
|
|
|
|
|
locationAndType.m_type = TestTreeItem::TestFunctionOrSet;
|
|
|
|
|
|
|
|
|
|
m_testFunctions.insert(name.toString(), locationAndType);
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool TestQmlVisitor::visit(QmlJS::AST::StringLiteral *ast)
|
|
|
|
|
{
|
2018-06-20 10:12:04 +02:00
|
|
|
if (m_expectTestCaseName && m_currentTestCaseName.isEmpty()) {
|
2017-02-27 09:30:16 +01:00
|
|
|
m_currentTestCaseName = ast->value.toString();
|
2018-06-20 10:12:04 +02:00
|
|
|
m_expectTestCaseName = false;
|
|
|
|
|
}
|
2016-05-11 13:02:42 +02:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-09 10:43:31 +02:00
|
|
|
/************************************** QuickTestAstVisitor *************************************/
|
|
|
|
|
|
|
|
|
|
QuickTestAstVisitor::QuickTestAstVisitor(CPlusPlus::Document::Ptr doc,
|
|
|
|
|
const CPlusPlus::Snapshot &snapshot)
|
|
|
|
|
: ASTVisitor(doc->translationUnit())
|
|
|
|
|
, m_currentDoc(doc)
|
|
|
|
|
, m_snapshot(snapshot)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool QuickTestAstVisitor::visit(CPlusPlus::CallAST *ast)
|
|
|
|
|
{
|
|
|
|
|
if (m_currentDoc.isNull())
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
if (const auto expressionAST = ast->base_expression) {
|
|
|
|
|
if (const auto idExpressionAST = expressionAST->asIdExpression()) {
|
|
|
|
|
if (const auto simpleNameAST = idExpressionAST->name->asSimpleName()) {
|
|
|
|
|
const CPlusPlus::Overview o;
|
|
|
|
|
const QString prettyName = o.prettyName(simpleNameAST->name);
|
|
|
|
|
if (prettyName == "quick_test_main" || prettyName == "quick_test_main_with_setup") {
|
|
|
|
|
if (auto expressionListAST = ast->expression_list) {
|
|
|
|
|
// third argument is the one we need, so skip current and next
|
|
|
|
|
expressionListAST = expressionListAST->next; // argv
|
|
|
|
|
expressionListAST = expressionListAST ? expressionListAST->next : nullptr; // testcase literal
|
|
|
|
|
|
|
|
|
|
if (expressionListAST && expressionListAST->value) {
|
|
|
|
|
const auto *stringLitAST = expressionListAST->value->asStringLiteral();
|
|
|
|
|
const auto *string
|
|
|
|
|
= translationUnit()->stringLiteral(stringLitAST->literal_token);
|
|
|
|
|
if (string) {
|
|
|
|
|
m_testBaseName = QString::fromUtf8(string->chars(),
|
|
|
|
|
int(string->size()));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2016-05-11 13:02:42 +02:00
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace Autotest
|