forked from qt-creator/qt-creator
If we prefetch data from the database to the caches we reduce the database transaction calls which are quite expensive. Change-Id: I617a0d886807402e0a94291a913a77f989970b55 Reviewed-by: Tim Jenssen <tim.jenssen@qt.io>
362 lines
15 KiB
C++
362 lines
15 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 "googletest.h"
|
|
|
|
#include "filesystem-utilities.h"
|
|
#include "mockfilepathcaching.h"
|
|
#include "mockrefactoringclient.h"
|
|
#include "mocksymbolindexing.h"
|
|
#include "sourcerangecontainer-matcher.h"
|
|
|
|
#include <clangrefactoringmessages.h>
|
|
#include <filepathcaching.h>
|
|
#include <refactoringdatabaseinitializer.h>
|
|
#include <refactoringserver.h>
|
|
#include <sqlitedatabase.h>
|
|
|
|
#include <utils/temporarydirectory.h>
|
|
|
|
#include <QTemporaryFile>
|
|
|
|
namespace {
|
|
|
|
using testing::AllOf;
|
|
using testing::Contains;
|
|
using testing::Field;
|
|
using testing::IsEmpty;
|
|
using testing::NiceMock;
|
|
using testing::Not;
|
|
using testing::Pair;
|
|
using testing::PrintToString;
|
|
using testing::Property;
|
|
using testing::_;
|
|
|
|
using ClangBackEnd::FilePath;
|
|
using ClangBackEnd::IncludeSearchPaths;
|
|
using ClangBackEnd::IncludeSearchPathType;
|
|
using ClangBackEnd::RequestSourceRangesAndDiagnosticsForQueryMessage;
|
|
using ClangBackEnd::RequestSourceRangesForQueryMessage;
|
|
using ClangBackEnd::SourceLocationsContainer;
|
|
using ClangBackEnd::SourceRangesAndDiagnosticsForQueryMessage;
|
|
using ClangBackEnd::SourceRangesContainer;
|
|
using ClangBackEnd::SourceRangesForQueryMessage;
|
|
using ClangBackEnd::V2::FileContainer;
|
|
using ClangBackEnd::V2::FileContainers;
|
|
using ClangBackEnd::ProjectPartContainer;
|
|
using ClangBackEnd::ProjectPartContainers;
|
|
|
|
MATCHER_P2(IsSourceLocation, line, column,
|
|
std::string(negation ? "isn't " : "is ")
|
|
+ "(" + PrintToString(line)
|
|
+ ", " + PrintToString(column)
|
|
+ ")"
|
|
)
|
|
{
|
|
return arg.line == uint(line)
|
|
&& arg.column == uint(column);
|
|
}
|
|
|
|
class RefactoringServer : public ::testing::Test
|
|
{
|
|
protected:
|
|
void SetUp() override;
|
|
void TearDown() override;
|
|
|
|
ClangBackEnd::FilePathId filePathId(Utils::SmallStringView string)
|
|
{
|
|
return filePathCache.filePathId(ClangBackEnd::FilePathView{string});
|
|
}
|
|
|
|
protected:
|
|
NiceMock<MockRefactoringClient> mockRefactoringClient;
|
|
NiceMock<MockSymbolIndexing> mockSymbolIndexing;
|
|
Sqlite::Database database{":memory:", Sqlite::JournalMode::Memory};
|
|
ClangBackEnd::RefactoringDatabaseInitializer<Sqlite::Database> databaseInitializer{database};
|
|
ClangBackEnd::FilePathCaching filePathCache{database};
|
|
ClangBackEnd::GeneratedFiles generatedFiles;
|
|
ClangBackEnd::RefactoringServer refactoringServer{mockSymbolIndexing, filePathCache, generatedFiles};
|
|
Utils::SmallString sourceContent{"void f()\n {}"};
|
|
ClangBackEnd::FilePath filePath{TESTDATA_DIR, "query_simplefunction.cpp"};
|
|
FileContainer source{filePath.clone(),
|
|
filePathCache.filePathId(filePath),
|
|
sourceContent.clone(),
|
|
{"cc"}};
|
|
QTemporaryFile temporaryFile{Utils::TemporaryDirectory::masterDirectoryPath()
|
|
+ "/clangQuery-XXXXXX.cpp"};
|
|
int processingSlotCount = 2;
|
|
};
|
|
|
|
using RefactoringServerSlowTest = RefactoringServer;
|
|
using RefactoringServerVerySlowTest = RefactoringServer;
|
|
|
|
TEST_F(RefactoringServerSlowTest, RequestSingleSourceRangesForQueryMessage)
|
|
{
|
|
RequestSourceRangesForQueryMessage message{"functionDecl()",
|
|
{source.clone()},
|
|
{}};
|
|
|
|
EXPECT_CALL(mockRefactoringClient,
|
|
sourceRangesForQueryMessage(
|
|
Field(&SourceRangesForQueryMessage::sourceRanges,
|
|
Field(&SourceRangesContainer::sourceRangeWithTextContainers,
|
|
Contains(IsSourceRangeWithText(1, 1, 2, 4, sourceContent))))));
|
|
|
|
refactoringServer.requestSourceRangesForQueryMessage(std::move(message));
|
|
}
|
|
|
|
TEST_F(RefactoringServerSlowTest, RequestSingleSourceRangesAndDiagnosticsWithUnsavedContentForQueryMessage)
|
|
{
|
|
Utils::SmallString unsavedContent{"void f();"};
|
|
FileContainer source{{TESTDATA_DIR, "query_simplefunction.cpp"},
|
|
filePathCache.filePathId(FilePath{TESTDATA_DIR, "query_simplefunction.cpp"}),
|
|
"#include \"query_simplefunction.h\"",
|
|
{"cc"}};
|
|
FileContainer unsaved{{TESTDATA_DIR, "query_simplefunction.h"},
|
|
filePathCache.filePathId(FilePath{TESTDATA_DIR, "query_simplefunction.h"}),
|
|
unsavedContent.clone(),
|
|
{}};
|
|
RequestSourceRangesForQueryMessage message{"functionDecl()", {source.clone()}, {unsaved.clone()}};
|
|
|
|
EXPECT_CALL(mockRefactoringClient,
|
|
sourceRangesForQueryMessage(
|
|
Field(&SourceRangesForQueryMessage::sourceRanges,
|
|
Field(&SourceRangesContainer::sourceRangeWithTextContainers,
|
|
Contains(IsSourceRangeWithText(1, 1, 1, 9, unsavedContent))))));
|
|
|
|
refactoringServer.requestSourceRangesForQueryMessage(std::move(message));
|
|
}
|
|
|
|
TEST_F(RefactoringServerSlowTest, RequestTwoSourceRangesForQueryMessage)
|
|
{
|
|
RequestSourceRangesForQueryMessage message{"functionDecl()",
|
|
{source.clone(), source.clone()},
|
|
{}};
|
|
|
|
EXPECT_CALL(mockRefactoringClient,
|
|
sourceRangesForQueryMessage(
|
|
Field(&SourceRangesForQueryMessage::sourceRanges,
|
|
Field(&SourceRangesContainer::sourceRangeWithTextContainers,
|
|
Contains(IsSourceRangeWithText(1, 1, 2, 4, sourceContent))))));
|
|
EXPECT_CALL(mockRefactoringClient,
|
|
sourceRangesForQueryMessage(
|
|
Field(&SourceRangesForQueryMessage::sourceRanges,
|
|
Field(&SourceRangesContainer::sourceRangeWithTextContainers,
|
|
Not(Contains(IsSourceRangeWithText(1, 1, 2, 4, sourceContent)))))));
|
|
|
|
refactoringServer.requestSourceRangesForQueryMessage(std::move(message));
|
|
}
|
|
|
|
TEST_F(RefactoringServerVerySlowTest, RequestManySourceRangesForQueryMessage)
|
|
{
|
|
std::vector<FileContainer> sources;
|
|
std::fill_n(std::back_inserter(sources),
|
|
processingSlotCount + 3,
|
|
source.clone());
|
|
RequestSourceRangesForQueryMessage message{"functionDecl()",
|
|
std::move(sources),
|
|
{}};
|
|
|
|
EXPECT_CALL(mockRefactoringClient,
|
|
sourceRangesForQueryMessage(
|
|
Field(&SourceRangesForQueryMessage::sourceRanges,
|
|
Field(&SourceRangesContainer::sourceRangeWithTextContainers,
|
|
Contains(IsSourceRangeWithText(1, 1, 2, 4, sourceContent))))));
|
|
EXPECT_CALL(mockRefactoringClient,
|
|
sourceRangesForQueryMessage(
|
|
Field(&SourceRangesForQueryMessage::sourceRanges,
|
|
Field(&SourceRangesContainer::sourceRangeWithTextContainers,
|
|
Not(Contains(IsSourceRangeWithText(1, 1, 2, 4, sourceContent)))))))
|
|
.Times(processingSlotCount + 2);
|
|
|
|
refactoringServer.requestSourceRangesForQueryMessage(std::move(message));
|
|
}
|
|
|
|
TEST_F(RefactoringServer, CancelJobs)
|
|
{
|
|
std::vector<FileContainer> sources;
|
|
std::fill_n(std::back_inserter(sources),
|
|
std::thread::hardware_concurrency() + 3,
|
|
source.clone());
|
|
RequestSourceRangesForQueryMessage message{"functionDecl()",
|
|
std::move(sources),
|
|
{}};
|
|
refactoringServer.requestSourceRangesForQueryMessage(std::move(message));
|
|
|
|
refactoringServer.cancel();
|
|
|
|
ASSERT_TRUE(refactoringServer.isCancelingJobs());
|
|
}
|
|
|
|
TEST_F(RefactoringServer, PollTimerIsActiveAfterStart)
|
|
{
|
|
RequestSourceRangesForQueryMessage message{"functionDecl()",
|
|
{source},
|
|
{}};
|
|
|
|
refactoringServer.requestSourceRangesForQueryMessage(std::move(message));
|
|
|
|
ASSERT_TRUE(refactoringServer.pollTimerIsActive());
|
|
}
|
|
|
|
TEST_F(RefactoringServer, PollTimerIsNotActiveAfterFinishing)
|
|
{
|
|
RequestSourceRangesForQueryMessage message{"functionDecl()",
|
|
{source},
|
|
{}};
|
|
refactoringServer.requestSourceRangesForQueryMessage(std::move(message));
|
|
|
|
refactoringServer.waitThatSourceRangesForQueryMessagesAreFinished();
|
|
|
|
ASSERT_FALSE(refactoringServer.pollTimerIsActive());
|
|
}
|
|
|
|
TEST_F(RefactoringServer, PollTimerNotIsActiveAfterCanceling)
|
|
{
|
|
RequestSourceRangesForQueryMessage message{"functionDecl()",
|
|
{source},
|
|
{}};
|
|
refactoringServer.requestSourceRangesForQueryMessage(std::move(message));
|
|
|
|
refactoringServer.cancel();
|
|
|
|
ASSERT_FALSE(refactoringServer.pollTimerIsActive());
|
|
}
|
|
|
|
TEST_F(RefactoringServerSlowTest, ForValidRequestSourceRangesAndDiagnosticsGetSourceRange)
|
|
{
|
|
RequestSourceRangesAndDiagnosticsForQueryMessage message("functionDecl()",
|
|
{FilePath(temporaryFile.fileName()),
|
|
filePathCache.filePathId(FilePath(
|
|
temporaryFile.fileName())),
|
|
"void f() {}",
|
|
{"cc"}});
|
|
|
|
EXPECT_CALL(mockRefactoringClient,
|
|
sourceRangesAndDiagnosticsForQueryMessage(
|
|
AllOf(
|
|
Field(&SourceRangesAndDiagnosticsForQueryMessage::sourceRanges,
|
|
Field(&SourceRangesContainer::sourceRangeWithTextContainers,
|
|
Contains(IsSourceRangeWithText(1, 1, 1, 12, "void f() {}")))),
|
|
Field(&SourceRangesAndDiagnosticsForQueryMessage::diagnostics,
|
|
IsEmpty()))));
|
|
|
|
refactoringServer.requestSourceRangesAndDiagnosticsForQueryMessage(std::move(message));
|
|
}
|
|
|
|
TEST_F(RefactoringServerSlowTest, ForInvalidRequestSourceRangesAndDiagnosticsGetDiagnostics)
|
|
{
|
|
RequestSourceRangesAndDiagnosticsForQueryMessage message("func()",
|
|
{FilePath(temporaryFile.fileName()),
|
|
filePathCache.filePathId(FilePath(
|
|
temporaryFile.fileName())),
|
|
"void f() {}",
|
|
{"cc"}});
|
|
|
|
EXPECT_CALL(mockRefactoringClient,
|
|
sourceRangesAndDiagnosticsForQueryMessage(
|
|
AllOf(
|
|
Field(&SourceRangesAndDiagnosticsForQueryMessage::sourceRanges,
|
|
Field(&SourceRangesContainer::sourceRangeWithTextContainers,
|
|
IsEmpty())),
|
|
Field(&SourceRangesAndDiagnosticsForQueryMessage::diagnostics,
|
|
Not(IsEmpty())))));
|
|
|
|
refactoringServer.requestSourceRangesAndDiagnosticsForQueryMessage(std::move(message));
|
|
}
|
|
|
|
TEST_F(RefactoringServer, UpdateGeneratedFilesSetMemberWhichIsUsedForSymbolIndexing)
|
|
{
|
|
FileContainers unsaved{{{TESTDATA_DIR, "query_simplefunction.h"},
|
|
filePathCache.filePathId(FilePath{TESTDATA_DIR, "query_simplefunction.h"}),
|
|
"void f();",
|
|
{}}};
|
|
|
|
refactoringServer.updateGeneratedFiles(Utils::clone(unsaved));
|
|
|
|
ASSERT_THAT(generatedFiles.fileContainers(), ElementsAre(unsaved.front()));
|
|
}
|
|
|
|
TEST_F(RefactoringServer, RemoveGeneratedFilesSetMemberWhichIsUsedForSymbolIndexing)
|
|
{
|
|
FileContainers unsaved{{{TESTDATA_DIR, "query_simplefunction.h"},
|
|
filePathCache.filePathId(FilePath{TESTDATA_DIR, "query_simplefunction.h"}),
|
|
"void f();",
|
|
{}}};
|
|
refactoringServer.updateGeneratedFiles(Utils::clone(unsaved));
|
|
|
|
refactoringServer.removeGeneratedFiles({{{TESTDATA_DIR, "query_simplefunction.h"}}});
|
|
|
|
ASSERT_THAT(generatedFiles.fileContainers(), IsEmpty());
|
|
}
|
|
|
|
TEST_F(RefactoringServer, UpdateProjectPartsCalls)
|
|
{
|
|
NiceMock<MockFilePathCaching> mockFilePathCaching;
|
|
ClangBackEnd::RefactoringServer refactoringServer{mockSymbolIndexing,
|
|
mockFilePathCaching,
|
|
generatedFiles};
|
|
ProjectPartContainers projectParts{
|
|
{{1,
|
|
{"-I", TESTDATA_DIR},
|
|
{{"DEFINE", "1", 1}},
|
|
IncludeSearchPaths{{"/includes", 1, IncludeSearchPathType::BuiltIn},
|
|
{"/other/includes", 2, IncludeSearchPathType::System}},
|
|
IncludeSearchPaths{{"/project/includes", 1, IncludeSearchPathType::User},
|
|
{"/other/project/includes", 2, IncludeSearchPathType::User}},
|
|
{filePathId("header1.h")},
|
|
{filePathId("main.cpp")},
|
|
Utils::Language::C,
|
|
Utils::LanguageVersion::C11,
|
|
Utils::LanguageExtension::All}}};
|
|
|
|
EXPECT_CALL(mockFilePathCaching, populateIfEmpty());
|
|
EXPECT_CALL(mockSymbolIndexing, updateProjectParts(projectParts));
|
|
|
|
refactoringServer.updateProjectParts({Utils::clone(projectParts), {}});
|
|
}
|
|
|
|
TEST_F(RefactoringServer, SetProgress)
|
|
{
|
|
EXPECT_CALL(mockRefactoringClient, progress(AllOf(Field(&ClangBackEnd::ProgressMessage::progress, 20),
|
|
Field(&ClangBackEnd::ProgressMessage::total, 30))));
|
|
|
|
refactoringServer.setProgress(20, 30);
|
|
}
|
|
|
|
void RefactoringServer::SetUp()
|
|
{
|
|
temporaryFile.open();
|
|
refactoringServer.setClient(&mockRefactoringClient);
|
|
}
|
|
|
|
void RefactoringServer::TearDown()
|
|
{
|
|
refactoringServer.setGathererProcessingSlotCount(uint(processingSlotCount));
|
|
refactoringServer.waitThatSourceRangesForQueryMessagesAreFinished();
|
|
}
|
|
|
|
}
|