forked from qt-creator/qt-creator
The issue was re-producible with e.g.:
void g()
{
// Type 'foo.' as fast as possible in the next line
}
This led to "foo->" with completion results as if there was no "foo." at
all in that line.
We relied on a correct position for
translationUnit.sourceLocationAtWithoutReparsing(), but the just typed
characters were not yet reparsed. And we do not want to reparse at that
point since takes too long. We already determine the utf8 position for
the dot character, so simply use that instead.
This completes commit 17c1325cc4.
Change-Id: I669888b5c17ee63b2aec7b16c9921f9d79e281f9
Reviewed-by: David Schulz <david.schulz@theqtcompany.com>
444 lines
19 KiB
C++
444 lines
19 KiB
C++
/****************************************************************************
|
|
**
|
|
** 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 <codecompleter.h>
|
|
#include <filecontainer.h>
|
|
#include <projectpart.h>
|
|
#include <projects.h>
|
|
#include <clangtranslationunit.h>
|
|
#include <translationunits.h>
|
|
#include <unsavedfiles.h>
|
|
#include <utf8stringvector.h>
|
|
|
|
#include <QCoreApplication>
|
|
#include <QFile>
|
|
#include <QTemporaryDir>
|
|
|
|
#include <gmock/gmock.h>
|
|
#include <gmock/gmock-matchers.h>
|
|
#include <gtest/gtest.h>
|
|
#include "gtest-qt-printing.h"
|
|
|
|
using ::testing::ElementsAreArray;
|
|
using ::testing::Contains;
|
|
using ::testing::AllOf;
|
|
using ::testing::Not;
|
|
using ::testing::PrintToString;
|
|
|
|
using ClangBackEnd::CodeCompletion;
|
|
using ClangBackEnd::CodeCompleter;
|
|
|
|
namespace {
|
|
|
|
MATCHER_P2(IsCodeCompletion, text, completionKind,
|
|
std::string(negation ? "isn't" : "is") + " code completion with text "
|
|
+ PrintToString(text) + " and kind " + PrintToString(completionKind)
|
|
)
|
|
{
|
|
if (arg.text() != text) {
|
|
*result_listener << "text is " + PrintToString(arg.text()) + " and not " + PrintToString(text);
|
|
return false;
|
|
}
|
|
|
|
if (arg.completionKind() != completionKind) {
|
|
*result_listener << "kind is " + PrintToString(arg.completionKind()) + " and not " + PrintToString(completionKind);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
class CodeCompleter : public ::testing::Test
|
|
{
|
|
protected:
|
|
void SetUp();
|
|
void copyTargetHeaderToTemporaryIncludeDirecory();
|
|
void copyChangedTargetHeaderToTemporaryIncludeDirecory();
|
|
ClangBackEnd::CodeCompleter setupCompleter(const ClangBackEnd::FileContainer &fileContainer);
|
|
static Utf8String readFileContent(const QString &fileName);
|
|
|
|
protected:
|
|
QTemporaryDir includeDirectory;
|
|
Utf8String includePath{QStringLiteral("-I") + includeDirectory.path()};
|
|
QString targetHeaderPath{includeDirectory.path() + QStringLiteral("/complete_target_header.h")};
|
|
ClangBackEnd::ProjectPartContainer projectPart{Utf8StringLiteral("projectPartId"), {includePath}};
|
|
ClangBackEnd::FileContainer mainFileContainer{Utf8StringLiteral(TESTDATA_DIR"/complete_completer_main.cpp"),
|
|
projectPart.projectPartId()};
|
|
ClangBackEnd::ProjectParts projects;
|
|
ClangBackEnd::UnsavedFiles unsavedFiles;
|
|
ClangBackEnd::TranslationUnits translationUnits{projects, unsavedFiles};
|
|
ClangBackEnd::TranslationUnit translationUnit;
|
|
ClangBackEnd::CodeCompleter completer;
|
|
ClangBackEnd::FileContainer unsavedMainFileContainer{mainFileContainer.filePath(),
|
|
projectPart.projectPartId(),
|
|
readFileContent(QStringLiteral("/complete_completer_main_unsaved.cpp")),
|
|
true};
|
|
ClangBackEnd::FileContainer unsavedTargetHeaderFileContainer{targetHeaderPath,
|
|
projectPart.projectPartId(),
|
|
readFileContent(QStringLiteral("/complete_target_header_unsaved.h")),
|
|
true};
|
|
|
|
ClangBackEnd::FileContainer arrowFileContainer{
|
|
Utf8StringLiteral(TESTDATA_DIR"/complete_arrow.cpp"),
|
|
projectPart.projectPartId(),
|
|
readFileContent(QStringLiteral("/complete_arrow.cpp")),
|
|
true
|
|
};
|
|
ClangBackEnd::FileContainer dotArrowCorrectionForPointerFileContainer{
|
|
Utf8StringLiteral(TESTDATA_DIR"/complete_withDotArrowCorrectionForPointer.cpp"),
|
|
projectPart.projectPartId(),
|
|
readFileContent(QStringLiteral("/complete_withDotArrowCorrectionForPointer.cpp")),
|
|
true
|
|
};
|
|
ClangBackEnd::FileContainer dotArrowCorrectionForPointerFileContainerBeforeTyping{
|
|
Utf8StringLiteral(TESTDATA_DIR"/complete_withDotArrowCorrectionForPointer.cpp"),
|
|
projectPart.projectPartId(),
|
|
readFileContent(QStringLiteral("/complete_withDotArrowCorrectionForPointer_beforeTyping.cpp")),
|
|
true
|
|
};
|
|
ClangBackEnd::FileContainer dotArrowCorrectionForPointerFileContainerAfterTyping{
|
|
Utf8StringLiteral(TESTDATA_DIR"/complete_withDotArrowCorrectionForPointer.cpp"),
|
|
projectPart.projectPartId(),
|
|
readFileContent(QStringLiteral("/complete_withDotArrowCorrectionForPointer_afterTyping.cpp")),
|
|
true
|
|
};
|
|
|
|
ClangBackEnd::FileContainer dotArrowCorrectionForPointerFileContainerInitial{
|
|
Utf8StringLiteral(TESTDATA_DIR"/complete_withDotArrowCorrectionForPointer.cpp"),
|
|
projectPart.projectPartId(),
|
|
readFileContent(QStringLiteral("/complete_withDotArrowCorrectionForPointerInitial.cpp")),
|
|
true
|
|
};
|
|
ClangBackEnd::FileContainer dotArrowCorrectionForPointerFileContainerUpdated{
|
|
Utf8StringLiteral(TESTDATA_DIR"/complete_withDotArrowCorrectionForPointer.cpp"),
|
|
projectPart.projectPartId(),
|
|
readFileContent(QStringLiteral("/complete_withDotArrowCorrectionForPointerUpdated.cpp")),
|
|
true
|
|
};
|
|
ClangBackEnd::FileContainer noDotArrowCorrectionForObjectFileContainer{
|
|
Utf8StringLiteral(TESTDATA_DIR"/complete_withNoDotArrowCorrectionForObject.cpp"),
|
|
projectPart.projectPartId(),
|
|
readFileContent(QStringLiteral("/complete_withNoDotArrowCorrectionForObject.cpp")),
|
|
true
|
|
};
|
|
ClangBackEnd::FileContainer noDotArrowCorrectionForFloatFileContainer{
|
|
Utf8StringLiteral(TESTDATA_DIR"/complete_withNoDotArrowCorrectionForFloat.cpp"),
|
|
projectPart.projectPartId(),
|
|
readFileContent(QStringLiteral("/complete_withNoDotArrowCorrectionForFloat.cpp")),
|
|
true
|
|
};
|
|
ClangBackEnd::FileContainer noDotArrowCorrectionForObjectWithArrowOperatortFileContainer{
|
|
Utf8StringLiteral(TESTDATA_DIR"/complete_withNoDotArrowCorrectionForObjectWithArrowOperator.cpp"),
|
|
projectPart.projectPartId(),
|
|
readFileContent(QStringLiteral("/complete_withNoDotArrowCorrectionForObjectWithArrowOperator.cpp")),
|
|
true
|
|
};
|
|
ClangBackEnd::FileContainer noDotArrowCorrectionForDotDotFileContainer{
|
|
Utf8StringLiteral(TESTDATA_DIR"/complete_withNoDotArrowCorrectionForDotDot.cpp"),
|
|
projectPart.projectPartId(),
|
|
readFileContent(QStringLiteral("/complete_withNoDotArrowCorrectionForDotDot.cpp")),
|
|
true
|
|
};
|
|
ClangBackEnd::FileContainer noDotArrowCorrectionForArrowDotFileContainer{
|
|
Utf8StringLiteral(TESTDATA_DIR"/complete_withNoDotArrowCorrectionForArrowDot.cpp"),
|
|
projectPart.projectPartId(),
|
|
readFileContent(QStringLiteral("/complete_withNoDotArrowCorrectionForArrowDot.cpp")),
|
|
true
|
|
};
|
|
ClangBackEnd::FileContainer noDotArrowCorrectionForOnlyDotFileContainer{
|
|
Utf8StringLiteral(TESTDATA_DIR"/complete_withNoDotArrowCorrectionForOnlyDot.cpp"),
|
|
projectPart.projectPartId(),
|
|
readFileContent(QStringLiteral("/complete_withNoDotArrowCorrectionForOnlyDot.cpp")),
|
|
true
|
|
};
|
|
ClangBackEnd::FileContainer noDotArrowCorrectionForColonColonFileContainer{
|
|
Utf8StringLiteral(TESTDATA_DIR"/complete_withNoDotArrowCorrectionForColonColon.cpp"),
|
|
projectPart.projectPartId(),
|
|
readFileContent(QStringLiteral("/complete_withNoDotArrowCorrectionForColonColon.cpp")),
|
|
true
|
|
};
|
|
};
|
|
|
|
Utf8String CodeCompleter::readFileContent(const QString &fileName)
|
|
{
|
|
QFile readFileContentFile(QStringLiteral(TESTDATA_DIR) + fileName);
|
|
bool hasOpened = readFileContentFile.open(QIODevice::ReadOnly | QIODevice::Text);
|
|
|
|
EXPECT_TRUE(hasOpened);
|
|
|
|
return Utf8String::fromByteArray(readFileContentFile.readAll());
|
|
}
|
|
|
|
void CodeCompleter::copyTargetHeaderToTemporaryIncludeDirecory()
|
|
{
|
|
QFile::remove(targetHeaderPath);
|
|
bool hasCopied = QFile::copy(QString::fromUtf8(TESTDATA_DIR "/complete_target_header.h"),
|
|
targetHeaderPath);
|
|
EXPECT_TRUE(hasCopied);
|
|
}
|
|
|
|
void CodeCompleter::copyChangedTargetHeaderToTemporaryIncludeDirecory()
|
|
{
|
|
QFile::remove(targetHeaderPath);
|
|
bool hasCopied = QFile::copy(QString::fromUtf8(TESTDATA_DIR "/complete_target_header_changed.h"),
|
|
targetHeaderPath);
|
|
EXPECT_TRUE(hasCopied);
|
|
}
|
|
|
|
void CodeCompleter::SetUp()
|
|
{
|
|
EXPECT_TRUE(includeDirectory.isValid());
|
|
projects.createOrUpdate({projectPart});
|
|
translationUnits.create({mainFileContainer});
|
|
translationUnit = translationUnits.translationUnit(mainFileContainer);
|
|
completer = ClangBackEnd::CodeCompleter(translationUnit);
|
|
|
|
copyTargetHeaderToTemporaryIncludeDirecory();
|
|
|
|
translationUnit.cxTranslationUnit(); // initialize translation unit so we check changes
|
|
}
|
|
|
|
TEST_F(CodeCompleter, FunctionInUnsavedFile)
|
|
{
|
|
unsavedFiles.createOrUpdate({unsavedMainFileContainer});
|
|
translationUnits.update({unsavedMainFileContainer});
|
|
|
|
ASSERT_THAT(completer.complete(27, 1),
|
|
AllOf(Contains(IsCodeCompletion(Utf8StringLiteral("FunctionWithArguments"),
|
|
CodeCompletion::FunctionCompletionKind)),
|
|
Contains(IsCodeCompletion(Utf8StringLiteral("Function"),
|
|
CodeCompletion::FunctionCompletionKind)),
|
|
Contains(IsCodeCompletion(Utf8StringLiteral("UnsavedFunction"),
|
|
CodeCompletion::FunctionCompletionKind)),
|
|
Contains(IsCodeCompletion(Utf8StringLiteral("f"),
|
|
CodeCompletion::FunctionCompletionKind)),
|
|
Not(Contains(IsCodeCompletion(Utf8StringLiteral("SavedFunction"),
|
|
CodeCompletion::FunctionCompletionKind)))));
|
|
}
|
|
|
|
TEST_F(CodeCompleter, VariableInUnsavedFile)
|
|
{
|
|
unsavedFiles.createOrUpdate({unsavedMainFileContainer});
|
|
translationUnits.update({unsavedMainFileContainer});
|
|
|
|
ASSERT_THAT(completer.complete(27, 1),
|
|
Contains(IsCodeCompletion(Utf8StringLiteral("VariableInUnsavedFile"),
|
|
CodeCompletion::VariableCompletionKind)));
|
|
}
|
|
|
|
TEST_F(CodeCompleter, GlobalVariableInUnsavedFile)
|
|
{
|
|
unsavedFiles.createOrUpdate({unsavedMainFileContainer});
|
|
translationUnits.update({unsavedMainFileContainer});
|
|
|
|
ASSERT_THAT(completer.complete(27, 1),
|
|
Contains(IsCodeCompletion(Utf8StringLiteral("GlobalVariableInUnsavedFile"),
|
|
CodeCompletion::VariableCompletionKind)));
|
|
}
|
|
|
|
TEST_F(CodeCompleter, Macro)
|
|
{
|
|
unsavedFiles.createOrUpdate({unsavedMainFileContainer});
|
|
translationUnits.update({unsavedMainFileContainer});
|
|
|
|
ASSERT_THAT(completer.complete(27, 1),
|
|
Contains(IsCodeCompletion(Utf8StringLiteral("Macro"),
|
|
CodeCompletion::PreProcessorCompletionKind)));
|
|
}
|
|
|
|
TEST_F(CodeCompleter, Keyword)
|
|
{
|
|
ASSERT_THAT(completer.complete(27, 1),
|
|
Contains(IsCodeCompletion(Utf8StringLiteral("switch"),
|
|
CodeCompletion::KeywordCompletionKind)));
|
|
}
|
|
|
|
TEST_F(CodeCompleter, FunctionInIncludedHeader)
|
|
{
|
|
ASSERT_THAT(completer.complete(27, 1),
|
|
Contains(IsCodeCompletion(Utf8StringLiteral("FunctionInIncludedHeader"),
|
|
CodeCompletion::FunctionCompletionKind)));
|
|
}
|
|
|
|
TEST_F(CodeCompleter, FunctionInUnsavedIncludedHeader)
|
|
{
|
|
unsavedFiles.createOrUpdate({unsavedTargetHeaderFileContainer});
|
|
translationUnits.create({unsavedTargetHeaderFileContainer});
|
|
|
|
ASSERT_THAT(completer.complete(27, 1),
|
|
Contains(IsCodeCompletion(Utf8StringLiteral("FunctionInIncludedHeaderUnsaved"),
|
|
CodeCompletion::FunctionCompletionKind)));
|
|
}
|
|
|
|
TEST_F(CodeCompleter, DISABLED_FunctionInChangedIncludedHeader)
|
|
{
|
|
copyChangedTargetHeaderToTemporaryIncludeDirecory();
|
|
|
|
ASSERT_THAT(completer.complete(27, 1),
|
|
Contains(IsCodeCompletion(Utf8StringLiteral("FunctionInIncludedHeaderChanged"),
|
|
CodeCompletion::FunctionCompletionKind)));
|
|
}
|
|
|
|
TEST_F(CodeCompleter, DISABLED_FunctionInChangedIncludedHeaderWithUnsavedContentInMainFile) // it's not that bad because we reparse anyway
|
|
{
|
|
unsavedFiles.createOrUpdate({unsavedMainFileContainer});
|
|
translationUnits.update({unsavedMainFileContainer});
|
|
|
|
copyChangedTargetHeaderToTemporaryIncludeDirecory();
|
|
|
|
ASSERT_THAT(completer.complete(27, 1),
|
|
Contains(IsCodeCompletion(Utf8StringLiteral("FunctionInIncludedHeaderChanged"),
|
|
CodeCompletion::FunctionCompletionKind)));
|
|
}
|
|
|
|
TEST_F(CodeCompleter, ArrowCompletion)
|
|
{
|
|
auto myCompleter = setupCompleter(arrowFileContainer);
|
|
|
|
const ClangBackEnd::CodeCompletions completions = myCompleter.complete(5, 10);
|
|
|
|
ASSERT_THAT(completions,
|
|
Contains(IsCodeCompletion(Utf8StringLiteral("member"),
|
|
CodeCompletion::VariableCompletionKind)));
|
|
ASSERT_THAT(myCompleter.neededCorrection(),
|
|
ClangBackEnd::CompletionCorrection::NoCorrection);
|
|
}
|
|
|
|
TEST_F(CodeCompleter, DotToArrowCompletionForPointer)
|
|
{
|
|
auto myCompleter = setupCompleter(dotArrowCorrectionForPointerFileContainer);
|
|
|
|
const ClangBackEnd::CodeCompletions completions = myCompleter.complete(5, 9);
|
|
|
|
ASSERT_THAT(completions,
|
|
Contains(IsCodeCompletion(Utf8StringLiteral("member"),
|
|
CodeCompletion::VariableCompletionKind)));
|
|
ASSERT_THAT(myCompleter.neededCorrection(),
|
|
ClangBackEnd::CompletionCorrection::DotToArrowCorrection);
|
|
}
|
|
|
|
TEST_F(CodeCompleter, DotToArrowCompletionForPointerInOutdatedTranslationUnit)
|
|
{
|
|
auto fileContainerBeforeTyping = dotArrowCorrectionForPointerFileContainerBeforeTyping;
|
|
auto myCompleter = setupCompleter(fileContainerBeforeTyping);
|
|
auto translationUnit = translationUnits.translationUnit(fileContainerBeforeTyping.filePath(),
|
|
fileContainerBeforeTyping.projectPartId());
|
|
translationUnit.cxTranslationUnit(); // Parse
|
|
unsavedFiles.createOrUpdate({dotArrowCorrectionForPointerFileContainerAfterTyping});
|
|
|
|
const ClangBackEnd::CodeCompletions completions = myCompleter.complete(5, 9);
|
|
|
|
ASSERT_THAT(completions,
|
|
Contains(IsCodeCompletion(Utf8StringLiteral("member"),
|
|
CodeCompletion::VariableCompletionKind)));
|
|
ASSERT_THAT(myCompleter.neededCorrection(),
|
|
ClangBackEnd::CompletionCorrection::DotToArrowCorrection);
|
|
}
|
|
|
|
TEST_F(CodeCompleter, NoDotToArrowCompletionForObject)
|
|
{
|
|
auto myCompleter = setupCompleter(noDotArrowCorrectionForObjectFileContainer);
|
|
|
|
const ClangBackEnd::CodeCompletions completions = myCompleter.complete(5, 9);
|
|
|
|
ASSERT_THAT(completions,
|
|
Contains(IsCodeCompletion(Utf8StringLiteral("member"),
|
|
CodeCompletion::VariableCompletionKind)));
|
|
ASSERT_THAT(myCompleter.neededCorrection(), ClangBackEnd::CompletionCorrection::NoCorrection);
|
|
}
|
|
|
|
TEST_F(CodeCompleter, NoDotToArrowCompletionForFloat)
|
|
{
|
|
auto myCompleter = setupCompleter(noDotArrowCorrectionForFloatFileContainer);
|
|
|
|
const ClangBackEnd::CodeCompletions completions = myCompleter.complete(3, 18);
|
|
|
|
ASSERT_TRUE(completions.isEmpty());
|
|
ASSERT_THAT(myCompleter.neededCorrection(), ClangBackEnd::CompletionCorrection::NoCorrection);
|
|
}
|
|
|
|
TEST_F(CodeCompleter, NoDotArrowCorrectionForObjectWithArrowOperator)
|
|
{
|
|
auto myCompleter = setupCompleter(noDotArrowCorrectionForObjectWithArrowOperatortFileContainer);
|
|
|
|
const ClangBackEnd::CodeCompletions completions = myCompleter.complete(8, 9);
|
|
|
|
ASSERT_THAT(completions,
|
|
Contains(IsCodeCompletion(Utf8StringLiteral("member"),
|
|
CodeCompletion::VariableCompletionKind)));
|
|
ASSERT_THAT(myCompleter.neededCorrection(), ClangBackEnd::CompletionCorrection::NoCorrection);
|
|
}
|
|
|
|
TEST_F(CodeCompleter, NoDotArrowCorrectionForDotDot)
|
|
{
|
|
auto myCompleter = setupCompleter(noDotArrowCorrectionForDotDotFileContainer);
|
|
|
|
const ClangBackEnd::CodeCompletions completions = myCompleter.complete(5, 10);
|
|
|
|
ASSERT_TRUE(completions.isEmpty());
|
|
ASSERT_THAT(myCompleter.neededCorrection(), ClangBackEnd::CompletionCorrection::NoCorrection);
|
|
}
|
|
|
|
TEST_F(CodeCompleter, NoDotArrowCorrectionForArrowDot)
|
|
{
|
|
auto myCompleter = setupCompleter(noDotArrowCorrectionForArrowDotFileContainer);
|
|
|
|
const ClangBackEnd::CodeCompletions completions = myCompleter.complete(5, 11);
|
|
|
|
ASSERT_TRUE(completions.isEmpty());
|
|
ASSERT_THAT(myCompleter.neededCorrection(), ClangBackEnd::CompletionCorrection::NoCorrection);
|
|
}
|
|
|
|
TEST_F(CodeCompleter, NoDotArrowCorrectionForOnlyDot)
|
|
{
|
|
auto myCompleter = setupCompleter(noDotArrowCorrectionForOnlyDotFileContainer);
|
|
|
|
const ClangBackEnd::CodeCompletions completions = myCompleter.complete(5, 6);
|
|
|
|
ASSERT_THAT(completions,
|
|
Contains(IsCodeCompletion(Utf8StringLiteral("Foo"),
|
|
CodeCompletion::ClassCompletionKind)));
|
|
ASSERT_THAT(myCompleter.neededCorrection(), ClangBackEnd::CompletionCorrection::NoCorrection);
|
|
}
|
|
|
|
TEST_F(CodeCompleter, NoDotArrowCorrectionForColonColon)
|
|
{
|
|
auto myCompleter = setupCompleter(noDotArrowCorrectionForColonColonFileContainer);
|
|
const ClangBackEnd::CodeCompletions completions = myCompleter.complete(1, 7);
|
|
|
|
ASSERT_THAT(myCompleter.neededCorrection(), ClangBackEnd::CompletionCorrection::NoCorrection);
|
|
}
|
|
|
|
ClangBackEnd::CodeCompleter CodeCompleter::setupCompleter(
|
|
const ClangBackEnd::FileContainer &fileContainer)
|
|
{
|
|
translationUnits.create({fileContainer});
|
|
unsavedFiles.createOrUpdate({fileContainer});
|
|
|
|
return ClangBackEnd::CodeCompleter(translationUnits.translationUnit(fileContainer));
|
|
}
|
|
|
|
}
|