forked from qt-creator/qt-creator
383 lines
16 KiB
C++
383 lines
16 KiB
C++
![]() |
/****************************************************************************
|
|||
|
**
|
|||
|
** Copyright (C) 2015 The Qt Company Ltd.
|
|||
|
** Contact: http://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 Digia. For licensing terms and
|
|||
|
** conditions see http://www.qt.io/licensing. For further information
|
|||
|
** use the contact form at http://www.qt.io/contact-us.
|
|||
|
**
|
|||
|
** GNU Lesser General Public License Usage
|
|||
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|||
|
** General Public License version 2.1 or version 3 as published by the Free
|
|||
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
|||
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
|||
|
** following information to ensure the GNU Lesser General Public License
|
|||
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
|||
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|||
|
**
|
|||
|
** In addition, as a special exception, Digia gives you certain additional
|
|||
|
** rights. These rights are described in the Digia Qt LGPL Exception
|
|||
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|||
|
**
|
|||
|
****************************************************************************/
|
|||
|
|
|||
|
#include "gtest/gtest.h"
|
|||
|
#include "gmock/gmock-matchers.h"
|
|||
|
#include "gmock/gmock.h"
|
|||
|
|
|||
|
#include "gtest-qt-printing.h"
|
|||
|
|
|||
|
#include <ipcclientproxy.h>
|
|||
|
#include <ipcserverproxy.h>
|
|||
|
#include <clangipcserver.h>
|
|||
|
#include <translationunitdoesnotexistexception.h>
|
|||
|
#include <translationunitparseerrorexception.h>
|
|||
|
|
|||
|
#include <cmbcodecompletedcommand.h>
|
|||
|
#include <cmbcompletecodecommand.h>
|
|||
|
#include <cmbechocommand.h>
|
|||
|
#include <cmbregistertranslationunitsforcodecompletioncommand.h>
|
|||
|
#include <cmbunregistertranslationunitsforcodecompletioncommand.h>
|
|||
|
#include <cmbregisterprojectsforcodecompletioncommand.h>
|
|||
|
#include <cmbunregisterprojectsforcodecompletioncommand.h>
|
|||
|
#include <translationunitdoesnotexistcommand.h>
|
|||
|
#include <projectpartsdonotexistcommand.h>
|
|||
|
|
|||
|
#include <QBuffer>
|
|||
|
#include <QFile>
|
|||
|
#include <projectpartsdonotexistcommand.h>
|
|||
|
|
|||
|
#include "mockipclient.h"
|
|||
|
|
|||
|
using testing::Property;
|
|||
|
using testing::Contains;
|
|||
|
using testing::Not;
|
|||
|
using testing::Eq;
|
|||
|
|
|||
|
namespace {
|
|||
|
|
|||
|
using CodeModelBackEnd::RegisterTranslationUnitForCodeCompletionCommand;
|
|||
|
using CodeModelBackEnd::UnregisterTranslationUnitsForCodeCompletionCommand;
|
|||
|
using CodeModelBackEnd::RegisterProjectPartsForCodeCompletionCommand;
|
|||
|
using CodeModelBackEnd::UnregisterProjectPartsForCodeCompletionCommand;
|
|||
|
using CodeModelBackEnd::CompleteCodeCommand;
|
|||
|
using CodeModelBackEnd::CodeCompletedCommand;
|
|||
|
using CodeModelBackEnd::CodeCompletion;
|
|||
|
using CodeModelBackEnd::FileContainer;
|
|||
|
using CodeModelBackEnd::ProjectPartContainer;
|
|||
|
using CodeModelBackEnd::TranslationUnitDoesNotExistCommand;
|
|||
|
using CodeModelBackEnd::ProjectPartsDoNotExistCommand;
|
|||
|
|
|||
|
class ClangIpcServer : public ::testing::Test
|
|||
|
{
|
|||
|
protected:
|
|||
|
void SetUp() override;
|
|||
|
|
|||
|
void registerFiles();
|
|||
|
void registerProjectPart();
|
|||
|
void changeProjectPartArguments();
|
|||
|
void changeProjectPartArgumentsToWrongValues();
|
|||
|
static const Utf8String unsavedContent(const QString &unsavedFilePath);
|
|||
|
|
|||
|
protected:
|
|||
|
MockIpcClient mockIpcClient;
|
|||
|
CodeModelBackEnd::ClangIpcServer clangServer;
|
|||
|
const Utf8String projectPartId = Utf8StringLiteral("pathToProjectPart.pro");
|
|||
|
const Utf8String functionTestFilePath = Utf8StringLiteral(TESTDATA_DIR"/complete_extractor_function.cpp");
|
|||
|
const Utf8String variableTestFilePath = Utf8StringLiteral(TESTDATA_DIR"/complete_extractor_variable.cpp");
|
|||
|
const QString unsavedTestFilePath = QStringLiteral(TESTDATA_DIR) + QStringLiteral("/complete_extractor_function_unsaved.cpp");
|
|||
|
const QString updatedUnsavedTestFilePath = QStringLiteral(TESTDATA_DIR) + QStringLiteral("/complete_extractor_function_unsaved_2.cpp");
|
|||
|
const Utf8String parseErrorTestFilePath = Utf8StringLiteral(TESTDATA_DIR"/complete_translationunit_parse_error.cpp");
|
|||
|
};
|
|||
|
|
|||
|
|
|||
|
void ClangIpcServer::SetUp()
|
|||
|
{
|
|||
|
clangServer.addClient(&mockIpcClient);
|
|||
|
registerProjectPart();
|
|||
|
registerFiles();
|
|||
|
}
|
|||
|
|
|||
|
void ClangIpcServer::registerFiles()
|
|||
|
{
|
|||
|
RegisterTranslationUnitForCodeCompletionCommand command({FileContainer(functionTestFilePath, projectPartId, unsavedContent(unsavedTestFilePath), true),
|
|||
|
FileContainer(variableTestFilePath, projectPartId)});
|
|||
|
|
|||
|
clangServer.registerTranslationUnitsForCodeCompletion(command);
|
|||
|
}
|
|||
|
|
|||
|
void ClangIpcServer::registerProjectPart()
|
|||
|
{
|
|||
|
RegisterProjectPartsForCodeCompletionCommand command({ProjectPartContainer(projectPartId)});
|
|||
|
|
|||
|
clangServer.registerProjectPartsForCodeCompletion(command);
|
|||
|
}
|
|||
|
|
|||
|
void ClangIpcServer::changeProjectPartArguments()
|
|||
|
{
|
|||
|
RegisterProjectPartsForCodeCompletionCommand command({ProjectPartContainer(projectPartId, {Utf8StringLiteral("-DArgumentDefinition")})});
|
|||
|
|
|||
|
clangServer.registerProjectPartsForCodeCompletion(command);
|
|||
|
}
|
|||
|
|
|||
|
void ClangIpcServer::changeProjectPartArgumentsToWrongValues()
|
|||
|
{
|
|||
|
RegisterProjectPartsForCodeCompletionCommand command({ProjectPartContainer(projectPartId, {Utf8StringLiteral("-blah")})});
|
|||
|
|
|||
|
clangServer.registerProjectPartsForCodeCompletion(command);
|
|||
|
}
|
|||
|
|
|||
|
const Utf8String ClangIpcServer::unsavedContent(const QString &unsavedFilePath)
|
|||
|
{
|
|||
|
QFile unsavedFileContentFile(unsavedFilePath);
|
|||
|
bool isOpen = unsavedFileContentFile.open(QIODevice::ReadOnly | QIODevice::Text);
|
|||
|
if (!isOpen)
|
|||
|
ADD_FAILURE() << "File with the unsaved content cannot be opened!";
|
|||
|
|
|||
|
return Utf8String::fromByteArray(unsavedFileContentFile.readAll());
|
|||
|
}
|
|||
|
|
|||
|
TEST_F(ClangIpcServer, GetCodeCompletion)
|
|||
|
{
|
|||
|
CompleteCodeCommand completeCodeCommand(functionTestFilePath,
|
|||
|
20,
|
|||
|
1,
|
|||
|
projectPartId);
|
|||
|
CodeCompletion codeCompletion(Utf8StringLiteral("Function"),
|
|||
|
Utf8String(),
|
|||
|
Utf8String(),
|
|||
|
34,
|
|||
|
CodeCompletion::FunctionCompletionKind);
|
|||
|
|
|||
|
EXPECT_CALL(mockIpcClient, codeCompleted(Property(&CodeCompletedCommand::codeCompletions, Contains(codeCompletion))))
|
|||
|
.Times(1);
|
|||
|
|
|||
|
clangServer.completeCode(completeCodeCommand);
|
|||
|
}
|
|||
|
|
|||
|
TEST_F(ClangIpcServer, GetCodeCompletionDependingOnArgumets)
|
|||
|
{
|
|||
|
CompleteCodeCommand completeCodeCommand(variableTestFilePath,
|
|||
|
35,
|
|||
|
1,
|
|||
|
projectPartId);
|
|||
|
CodeCompletion codeCompletion(Utf8StringLiteral("ArgumentDefinitionVariable"),
|
|||
|
Utf8String(),
|
|||
|
Utf8String(),
|
|||
|
34,
|
|||
|
CodeCompletion::VariableCompletionKind);
|
|||
|
|
|||
|
EXPECT_CALL(mockIpcClient, codeCompleted(Property(&CodeCompletedCommand::codeCompletions, Contains(codeCompletion))))
|
|||
|
.Times(1);
|
|||
|
|
|||
|
changeProjectPartArguments();
|
|||
|
clangServer.completeCode(completeCodeCommand);
|
|||
|
}
|
|||
|
|
|||
|
TEST_F(ClangIpcServer, GetTranslationUnitDoesNotExistForCodeCompletionOnNonExistingTranslationUnit)
|
|||
|
{
|
|||
|
CompleteCodeCommand completeCodeCommand(Utf8StringLiteral("dontexists.cpp"),
|
|||
|
34,
|
|||
|
1,
|
|||
|
projectPartId);
|
|||
|
TranslationUnitDoesNotExistCommand translationUnitDoesNotExistCommand(Utf8StringLiteral("dontexists.cpp"), projectPartId);
|
|||
|
|
|||
|
EXPECT_CALL(mockIpcClient, translationUnitDoesNotExist(translationUnitDoesNotExistCommand))
|
|||
|
.Times(1);
|
|||
|
|
|||
|
clangServer.completeCode(completeCodeCommand);
|
|||
|
}
|
|||
|
|
|||
|
TEST_F(ClangIpcServer, GetTranslationUnitDoesNotExistForCompletingUnregisteredFile)
|
|||
|
{
|
|||
|
CompleteCodeCommand completeCodeCommand(parseErrorTestFilePath,
|
|||
|
20,
|
|||
|
1,
|
|||
|
projectPartId);
|
|||
|
TranslationUnitDoesNotExistCommand translationUnitDoesNotExistCommand(parseErrorTestFilePath, projectPartId);
|
|||
|
|
|||
|
EXPECT_CALL(mockIpcClient, translationUnitDoesNotExist(translationUnitDoesNotExistCommand))
|
|||
|
.Times(1);
|
|||
|
|
|||
|
clangServer.completeCode(completeCodeCommand);
|
|||
|
}
|
|||
|
|
|||
|
TEST_F(ClangIpcServer, GetCodeCompletionForUnsavedFile)
|
|||
|
{
|
|||
|
CompleteCodeCommand completeCodeCommand(functionTestFilePath,
|
|||
|
20,
|
|||
|
1,
|
|||
|
projectPartId);
|
|||
|
CodeCompletion codeCompletion(Utf8StringLiteral("Method2"),
|
|||
|
Utf8String(),
|
|||
|
Utf8String(),
|
|||
|
34,
|
|||
|
CodeCompletion::FunctionCompletionKind);
|
|||
|
|
|||
|
EXPECT_CALL(mockIpcClient, codeCompleted(Property(&CodeCompletedCommand::codeCompletions, Contains(codeCompletion))))
|
|||
|
.Times(1);
|
|||
|
|
|||
|
clangServer.completeCode(completeCodeCommand);
|
|||
|
}
|
|||
|
|
|||
|
TEST_F(ClangIpcServer, GetNoCodeCompletionAfterRemovingUnsavedFile)
|
|||
|
{
|
|||
|
clangServer.registerTranslationUnitsForCodeCompletion(RegisterTranslationUnitForCodeCompletionCommand({FileContainer(functionTestFilePath, projectPartId)}));
|
|||
|
CompleteCodeCommand completeCodeCommand(functionTestFilePath,
|
|||
|
20,
|
|||
|
1,
|
|||
|
projectPartId);
|
|||
|
CodeCompletion codeCompletion(Utf8StringLiteral("Method2"),
|
|||
|
Utf8String(),
|
|||
|
Utf8String(),
|
|||
|
34,
|
|||
|
CodeCompletion::FunctionCompletionKind);
|
|||
|
|
|||
|
EXPECT_CALL(mockIpcClient, codeCompleted(Property(&CodeCompletedCommand::codeCompletions, Not(Contains(codeCompletion)))))
|
|||
|
.Times(1);
|
|||
|
|
|||
|
clangServer.completeCode(completeCodeCommand);
|
|||
|
}
|
|||
|
|
|||
|
TEST_F(ClangIpcServer, GetNewCodeCompletionAfterUpdatingUnsavedFile)
|
|||
|
{
|
|||
|
clangServer.registerTranslationUnitsForCodeCompletion(RegisterTranslationUnitForCodeCompletionCommand({FileContainer(functionTestFilePath,
|
|||
|
projectPartId,
|
|||
|
unsavedContent(updatedUnsavedTestFilePath),
|
|||
|
true)}));
|
|||
|
CompleteCodeCommand completeCodeCommand(functionTestFilePath,
|
|||
|
20,
|
|||
|
1,
|
|||
|
projectPartId);
|
|||
|
CodeCompletion codeCompletion(Utf8StringLiteral("Method3"),
|
|||
|
Utf8String(),
|
|||
|
Utf8String(),
|
|||
|
34,
|
|||
|
CodeCompletion::FunctionCompletionKind);
|
|||
|
|
|||
|
EXPECT_CALL(mockIpcClient, codeCompleted(Property(&CodeCompletedCommand::codeCompletions, Contains(codeCompletion))))
|
|||
|
.Times(1);
|
|||
|
|
|||
|
clangServer.completeCode(completeCodeCommand);
|
|||
|
}
|
|||
|
|
|||
|
TEST_F(ClangIpcServer, GetTranslationUnitDoesNotExistForUnregisterTranslationUnitWithWrongFilePath)
|
|||
|
{
|
|||
|
FileContainer fileContainer(Utf8StringLiteral("foo.cpp"), projectPartId);
|
|||
|
UnregisterTranslationUnitsForCodeCompletionCommand command({fileContainer});
|
|||
|
TranslationUnitDoesNotExistCommand translationUnitDoesNotExistCommand(fileContainer);
|
|||
|
|
|||
|
EXPECT_CALL(mockIpcClient, translationUnitDoesNotExist(translationUnitDoesNotExistCommand))
|
|||
|
.Times(1);
|
|||
|
|
|||
|
clangServer.unregisterTranslationUnitsForCodeCompletion(command);
|
|||
|
}
|
|||
|
|
|||
|
TEST_F(ClangIpcServer, UnregisterTranslationUnitAndTestFailingCompletion)
|
|||
|
{
|
|||
|
FileContainer fileContainer(functionTestFilePath, projectPartId);
|
|||
|
UnregisterTranslationUnitsForCodeCompletionCommand command({fileContainer});
|
|||
|
clangServer.unregisterTranslationUnitsForCodeCompletion(command);
|
|||
|
CompleteCodeCommand completeCodeCommand(functionTestFilePath,
|
|||
|
20,
|
|||
|
1,
|
|||
|
projectPartId);
|
|||
|
TranslationUnitDoesNotExistCommand translationUnitDoesNotExistCommand(fileContainer);
|
|||
|
|
|||
|
EXPECT_CALL(mockIpcClient, translationUnitDoesNotExist(translationUnitDoesNotExistCommand))
|
|||
|
.Times(1);
|
|||
|
|
|||
|
clangServer.completeCode(completeCodeCommand);
|
|||
|
}
|
|||
|
|
|||
|
TEST_F(ClangIpcServer, GetProjectPartDoesNotExistUnregisterProjectPartInexistingProjectPart)
|
|||
|
{
|
|||
|
Utf8StringVector inexistingProjectPartFilePath = {Utf8StringLiteral("projectpartsdoesnotexist.pro"), Utf8StringLiteral("project2doesnotexists.pro")};
|
|||
|
UnregisterProjectPartsForCodeCompletionCommand unregisterProjectPartsForCodeCompletionCommand(inexistingProjectPartFilePath);
|
|||
|
ProjectPartsDoNotExistCommand projectPartsDoNotExistCommand(inexistingProjectPartFilePath);
|
|||
|
|
|||
|
EXPECT_CALL(mockIpcClient, projectPartsDoNotExist(projectPartsDoNotExistCommand))
|
|||
|
.Times(1);
|
|||
|
|
|||
|
clangServer.unregisterProjectPartsForCodeCompletion(unregisterProjectPartsForCodeCompletionCommand);
|
|||
|
}
|
|||
|
|
|||
|
TEST_F(ClangIpcServer, GetProjectPartDoesNotExistRegisterTranslationUnitWithInexistingProjectPart)
|
|||
|
{
|
|||
|
Utf8String inexistingProjectPartFilePath = Utf8StringLiteral("projectpartsdoesnotexist.pro");
|
|||
|
RegisterTranslationUnitForCodeCompletionCommand registerFileForCodeCompletionCommand({FileContainer(variableTestFilePath, inexistingProjectPartFilePath)});
|
|||
|
ProjectPartsDoNotExistCommand projectPartsDoNotExistCommand({inexistingProjectPartFilePath});
|
|||
|
|
|||
|
EXPECT_CALL(mockIpcClient, projectPartsDoNotExist(projectPartsDoNotExistCommand))
|
|||
|
.Times(1);
|
|||
|
|
|||
|
clangServer.registerTranslationUnitsForCodeCompletion(registerFileForCodeCompletionCommand);
|
|||
|
}
|
|||
|
|
|||
|
TEST_F(ClangIpcServer, GetProjectPartDoesNotExistUnregisterTranslationUnitWithInexistingProjectPart)
|
|||
|
{
|
|||
|
Utf8String inexistingProjectPartFilePath = Utf8StringLiteral("projectpartsdoesnotexist.pro");
|
|||
|
UnregisterTranslationUnitsForCodeCompletionCommand unregisterFileForCodeCompletionCommand({FileContainer(variableTestFilePath, inexistingProjectPartFilePath)});
|
|||
|
ProjectPartsDoNotExistCommand projectPartsDoNotExistCommand({inexistingProjectPartFilePath});
|
|||
|
|
|||
|
EXPECT_CALL(mockIpcClient, projectPartsDoNotExist(projectPartsDoNotExistCommand))
|
|||
|
.Times(1);
|
|||
|
|
|||
|
clangServer.unregisterTranslationUnitsForCodeCompletion(unregisterFileForCodeCompletionCommand);
|
|||
|
}
|
|||
|
|
|||
|
TEST_F(ClangIpcServer, GetProjectPartDoesNotExistForCompletingProjectPartFile)
|
|||
|
{
|
|||
|
Utf8String inexistingProjectPartFilePath = Utf8StringLiteral("projectpartsdoesnotexist.pro");
|
|||
|
CompleteCodeCommand completeCodeCommand(variableTestFilePath,
|
|||
|
20,
|
|||
|
1,
|
|||
|
inexistingProjectPartFilePath);
|
|||
|
ProjectPartsDoNotExistCommand projectPartsDoNotExistCommand({inexistingProjectPartFilePath});
|
|||
|
|
|||
|
EXPECT_CALL(mockIpcClient, projectPartsDoNotExist(projectPartsDoNotExistCommand))
|
|||
|
.Times(1);
|
|||
|
|
|||
|
clangServer.completeCode(completeCodeCommand);
|
|||
|
}
|
|||
|
|
|||
|
TEST_F(ClangIpcServer, GetProjectPartDoesNotExistForCompletingUnregisteredFile)
|
|||
|
{
|
|||
|
CompleteCodeCommand completeCodeCommand(parseErrorTestFilePath,
|
|||
|
20,
|
|||
|
1,
|
|||
|
projectPartId);
|
|||
|
TranslationUnitDoesNotExistCommand translationUnitDoesNotExistCommand(parseErrorTestFilePath, projectPartId);
|
|||
|
|
|||
|
EXPECT_CALL(mockIpcClient, translationUnitDoesNotExist(translationUnitDoesNotExistCommand))
|
|||
|
.Times(1);
|
|||
|
|
|||
|
clangServer.completeCode(completeCodeCommand);
|
|||
|
}
|
|||
|
|
|||
|
TEST_F(ClangIpcServer, TicketNumberIsForwarded)
|
|||
|
{
|
|||
|
CompleteCodeCommand completeCodeCommand(functionTestFilePath,
|
|||
|
20,
|
|||
|
1,
|
|||
|
projectPartId);
|
|||
|
CodeCompletion codeCompletion(Utf8StringLiteral("Function"),
|
|||
|
Utf8String(),
|
|||
|
Utf8String(),
|
|||
|
34,
|
|||
|
CodeCompletion::FunctionCompletionKind);
|
|||
|
|
|||
|
EXPECT_CALL(mockIpcClient, codeCompleted(Property(&CodeCompletedCommand::ticketNumber, Eq(completeCodeCommand.ticketNumber()))))
|
|||
|
.Times(1);
|
|||
|
|
|||
|
clangServer.completeCode(completeCodeCommand);
|
|||
|
}
|
|||
|
}
|