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>
		
			
				
	
	
		
			443 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			443 lines
		
	
	
		
			20 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 "mockbuilddependenciesstorage.h"
 | 
						|
#include "mockclangpathwatcher.h"
 | 
						|
#include "mockfilepathcaching.h"
 | 
						|
#include "mockgeneratedfiles.h"
 | 
						|
#include "mockpchmanagerclient.h"
 | 
						|
#include "mockpchtaskgenerator.h"
 | 
						|
#include "mockprojectpartsmanager.h"
 | 
						|
 | 
						|
#include <filepathcaching.h>
 | 
						|
#include <pchmanagerserver.h>
 | 
						|
#include <precompiledheadersupdatedmessage.h>
 | 
						|
#include <progressmessage.h>
 | 
						|
#include <refactoringdatabaseinitializer.h>
 | 
						|
#include <removegeneratedfilesmessage.h>
 | 
						|
#include <removeprojectpartsmessage.h>
 | 
						|
#include <updategeneratedfilesmessage.h>
 | 
						|
#include <updateprojectpartsmessage.h>
 | 
						|
 | 
						|
namespace {
 | 
						|
using ClangBackEnd::FilePathId;
 | 
						|
using ClangBackEnd::ProjectPartContainer;
 | 
						|
using ClangBackEnd::ProjectPartContainers;
 | 
						|
using ClangBackEnd::V2::FileContainer;
 | 
						|
using ClangBackEnd::V2::FileContainers;
 | 
						|
using Utils::PathString;
 | 
						|
using Utils::SmallString;
 | 
						|
using UpToDataProjectParts = ClangBackEnd::ProjectPartsManagerInterface::UpToDataProjectParts;
 | 
						|
 | 
						|
class PchManagerServer : public ::testing::Test
 | 
						|
{
 | 
						|
    void SetUp() override
 | 
						|
    {
 | 
						|
        server.setClient(&mockPchManagerClient);
 | 
						|
 | 
						|
        ON_CALL(mockProjectPartsManager, update(projectParts))
 | 
						|
            .WillByDefault(Return(UpToDataProjectParts{{}, projectParts, projectParts4}));
 | 
						|
        ON_CALL(mockGeneratedFiles, isValid()).WillByDefault(Return(true));
 | 
						|
   }
 | 
						|
 | 
						|
    ClangBackEnd::FilePathId id(Utils::SmallStringView path) const
 | 
						|
    {
 | 
						|
        return filePathCache.filePathId(ClangBackEnd::FilePathView(path));
 | 
						|
    }
 | 
						|
 | 
						|
protected:
 | 
						|
    NiceMock<MockPchTaskGenerator> mockPchTaskGenerator;
 | 
						|
    NiceMock<MockClangPathWatcher> mockClangPathWatcher;
 | 
						|
    NiceMock<MockProjectPartsManager> mockProjectPartsManager;
 | 
						|
    NiceMock<MockGeneratedFiles> mockGeneratedFiles;
 | 
						|
    NiceMock<MockBuildDependenciesStorage> mockBuildDependenciesStorage;
 | 
						|
    Sqlite::Database database{":memory:", Sqlite::JournalMode::Memory};
 | 
						|
    ClangBackEnd::RefactoringDatabaseInitializer<Sqlite::Database> initializer{database};
 | 
						|
    ClangBackEnd::FilePathCaching filePathCache{database};
 | 
						|
    NiceMock<MockFilePathCaching> mockFilePathCache;
 | 
						|
    ClangBackEnd::PchManagerServer server{mockClangPathWatcher,
 | 
						|
                                          mockPchTaskGenerator,
 | 
						|
                                          mockProjectPartsManager,
 | 
						|
                                          mockGeneratedFiles,
 | 
						|
                                          mockBuildDependenciesStorage,
 | 
						|
                                          mockFilePathCache};
 | 
						|
    NiceMock<MockPchManagerClient> mockPchManagerClient;
 | 
						|
    ClangBackEnd::ProjectPartId projectPartId1{1};
 | 
						|
    ClangBackEnd::ProjectPartId projectPartId2{2};
 | 
						|
    ClangBackEnd::ProjectPartId projectPartId3{3};
 | 
						|
    ClangBackEnd::ProjectPartId projectPartId4{4};
 | 
						|
    ClangBackEnd::ProjectPartId projectPartId5{5};
 | 
						|
    ClangBackEnd::ProjectPartId projectPartId6{6};
 | 
						|
    PathString main1Path = TESTDATA_DIR "/BuildDependencyCollector_main3.cpp";
 | 
						|
    PathString main2Path = TESTDATA_DIR "/BuildDependencyCollector_main2.cpp";
 | 
						|
    PathString header1Path = TESTDATA_DIR "/BuildDependencyCollector_header1.h";
 | 
						|
    PathString header2Path = TESTDATA_DIR "/BuildDependencyCollector_header2.h";
 | 
						|
    ClangBackEnd::IdPaths idPath{{projectPartId1, ClangBackEnd::SourceType::Source}, {1, 2}};
 | 
						|
    ProjectPartContainer projectPart1{
 | 
						|
        projectPartId1,
 | 
						|
        {"-I", TESTDATA_DIR, "-Wno-pragma-once-outside-header"},
 | 
						|
        {{"DEFINE", "1", 1}},
 | 
						|
        {{TESTDATA_DIR "/symbolscollector/include", 1, ClangBackEnd::IncludeSearchPathType::BuiltIn}},
 | 
						|
        {{"/project/includes", 1, ClangBackEnd::IncludeSearchPathType::User}},
 | 
						|
        {id(header1Path)},
 | 
						|
        {id(main1Path)},
 | 
						|
        Utils::Language::C,
 | 
						|
        Utils::LanguageVersion::C11,
 | 
						|
        Utils::LanguageExtension::All};
 | 
						|
    ProjectPartContainer projectPart2{
 | 
						|
        projectPartId2,
 | 
						|
        {"-x", "c++-header", "-Wno-pragma-once-outside-header"},
 | 
						|
        {{"DEFINE", "1", 1}},
 | 
						|
        {{"/includes", 1, ClangBackEnd::IncludeSearchPathType::BuiltIn}},
 | 
						|
        {{TESTDATA_DIR "/builddependencycollector", 1, ClangBackEnd::IncludeSearchPathType::User}},
 | 
						|
        {id(header2Path)},
 | 
						|
        {id(main2Path)},
 | 
						|
        Utils::Language::C,
 | 
						|
        Utils::LanguageVersion::C11,
 | 
						|
        Utils::LanguageExtension::All};
 | 
						|
    ProjectPartContainer projectPart3{
 | 
						|
        projectPartId3,
 | 
						|
        {"-I", TESTDATA_DIR, "-Wno-pragma-once-outside-header"},
 | 
						|
        {{"DEFINE", "1", 1}},
 | 
						|
        {{"/includes", 1, ClangBackEnd::IncludeSearchPathType::BuiltIn}},
 | 
						|
        {{"/project/includes", 1, ClangBackEnd::IncludeSearchPathType::User}},
 | 
						|
        {id(header1Path)},
 | 
						|
        {id(main1Path)},
 | 
						|
        Utils::Language::C,
 | 
						|
        Utils::LanguageVersion::C11,
 | 
						|
        Utils::LanguageExtension::All};
 | 
						|
    ProjectPartContainer projectPart4{
 | 
						|
        projectPartId4,
 | 
						|
        {"-x", "c++-header", "-Wno-pragma-once-outside-header"},
 | 
						|
        {{"DEFINE", "1", 1}},
 | 
						|
        {{"/includes", 1, ClangBackEnd::IncludeSearchPathType::BuiltIn}},
 | 
						|
        {{"/project/includes", 1, ClangBackEnd::IncludeSearchPathType::User}},
 | 
						|
        {id(header2Path)},
 | 
						|
        {id(main2Path)},
 | 
						|
        Utils::Language::C,
 | 
						|
        Utils::LanguageVersion::C11,
 | 
						|
        Utils::LanguageExtension::All};
 | 
						|
    ProjectPartContainer projectPart5{
 | 
						|
        projectPartId5,
 | 
						|
        {"-I", TESTDATA_DIR, "-Wno-pragma-once-outside-header"},
 | 
						|
        {{"DEFINE", "1", 1}},
 | 
						|
        {{"/includes", 1, ClangBackEnd::IncludeSearchPathType::BuiltIn}},
 | 
						|
        {{"/project/includes", 1, ClangBackEnd::IncludeSearchPathType::User}},
 | 
						|
        {id(header1Path)},
 | 
						|
        {id(main1Path)},
 | 
						|
        Utils::Language::C,
 | 
						|
        Utils::LanguageVersion::C11,
 | 
						|
        Utils::LanguageExtension::All};
 | 
						|
    ProjectPartContainer projectPart6{
 | 
						|
        projectPartId6,
 | 
						|
        {"-x", "c++-header", "-Wno-pragma-once-outside-header"},
 | 
						|
        {{"DEFINE", "1", 1}},
 | 
						|
        {{"/includes", 1, ClangBackEnd::IncludeSearchPathType::BuiltIn}},
 | 
						|
        {{"/project/includes", 1, ClangBackEnd::IncludeSearchPathType::User}},
 | 
						|
        {id(header2Path)},
 | 
						|
        {id(main2Path)},
 | 
						|
        Utils::Language::C,
 | 
						|
        Utils::LanguageVersion::C11,
 | 
						|
        Utils::LanguageExtension::All};
 | 
						|
    std::vector<ProjectPartContainer> projectParts{projectPart1, projectPart2};
 | 
						|
    std::vector<ProjectPartContainer> projectParts1{projectPart1};
 | 
						|
    std::vector<ProjectPartContainer> projectParts2{projectPart2};
 | 
						|
    std::vector<ProjectPartContainer> projectParts3{projectPart3};
 | 
						|
    std::vector<ProjectPartContainer> projectParts4{projectPart3, projectPart4};
 | 
						|
    FileContainer generatedFile{{"/path/to/", "file"}, id("/path/to/file"), "content", {}};
 | 
						|
    ClangBackEnd::UpdateProjectPartsMessage updateProjectPartsMessage{
 | 
						|
        Utils::clone(projectParts), {"toolChainArgument"}};
 | 
						|
    ClangBackEnd::RemoveProjectPartsMessage removeProjectPartsMessage{
 | 
						|
        {projectPart1.projectPartId, projectPart2.projectPartId}};
 | 
						|
};
 | 
						|
 | 
						|
TEST_F(PchManagerServer, FilterProjectPartsAndSendThemToQueue)
 | 
						|
{
 | 
						|
    InSequence s;
 | 
						|
 | 
						|
    EXPECT_CALL(mockProjectPartsManager, update(updateProjectPartsMessage.projectsParts))
 | 
						|
        .WillOnce(Return(UpToDataProjectParts{{}, projectParts2, projectParts4}));
 | 
						|
    EXPECT_CALL(
 | 
						|
        mockPchTaskGenerator, addProjectParts(Eq(projectParts2), ElementsAre("toolChainArgument")));
 | 
						|
    EXPECT_CALL(mockPchTaskGenerator,
 | 
						|
                addNonSystemProjectParts(Eq(projectParts4), ElementsAre("toolChainArgument")));
 | 
						|
 | 
						|
    server.updateProjectParts(updateProjectPartsMessage.clone());
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(PchManagerServer, UpdateGeneratedFilesCallsUpdate)
 | 
						|
{
 | 
						|
    ClangBackEnd::UpdateGeneratedFilesMessage updateGeneratedFilesMessage{{generatedFile}};
 | 
						|
 | 
						|
    EXPECT_CALL(mockGeneratedFiles, update(updateGeneratedFilesMessage.generatedFiles));
 | 
						|
 | 
						|
    server.updateGeneratedFiles(updateGeneratedFilesMessage.clone());
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(PchManagerServer, RemoveGeneratedFilesCallsRemove)
 | 
						|
{
 | 
						|
    ClangBackEnd::RemoveGeneratedFilesMessage removeGeneratedFilesMessage{{generatedFile.filePath}};
 | 
						|
 | 
						|
    EXPECT_CALL(mockGeneratedFiles, remove(Utils::clone(removeGeneratedFilesMessage.generatedFiles)));
 | 
						|
 | 
						|
    server.removeGeneratedFiles(removeGeneratedFilesMessage.clone());
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(PchManagerServer, RemoveIncludesFromFileWatcher)
 | 
						|
{
 | 
						|
    EXPECT_CALL(mockClangPathWatcher, removeIds(removeProjectPartsMessage.projectsPartIds));
 | 
						|
 | 
						|
    server.removeProjectParts(removeProjectPartsMessage.clone());
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(PchManagerServer, RemoveProjectPartsFromProjectParts)
 | 
						|
{
 | 
						|
    EXPECT_CALL(mockProjectPartsManager, remove(removeProjectPartsMessage.projectsPartIds));
 | 
						|
 | 
						|
    server.removeProjectParts(removeProjectPartsMessage.clone());
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(PchManagerServer, SetPathWatcherNotifier)
 | 
						|
{
 | 
						|
    EXPECT_CALL(mockClangPathWatcher, setNotifier(_));
 | 
						|
 | 
						|
    ClangBackEnd::PchManagerServer server{mockClangPathWatcher,
 | 
						|
                                          mockPchTaskGenerator,
 | 
						|
                                          mockProjectPartsManager,
 | 
						|
                                          mockGeneratedFiles,
 | 
						|
                                          mockBuildDependenciesStorage,
 | 
						|
                                          filePathCache};
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(PchManagerServer, UpdateProjectPartQueueByPathIds)
 | 
						|
{
 | 
						|
    InSequence s;
 | 
						|
    server.updateProjectParts(ClangBackEnd::UpdateProjectPartsMessage{
 | 
						|
        {projectPart1, projectPart2, projectPart3, projectPart4, projectPart5, projectPart6},
 | 
						|
        {"toolChainArgument"}});
 | 
						|
 | 
						|
    EXPECT_CALL(mockProjectPartsManager,
 | 
						|
                projects(ElementsAre(projectPart1.projectPartId, projectPart2.projectPartId)))
 | 
						|
        .WillOnce(
 | 
						|
            Return(std::vector<ClangBackEnd::ProjectPartContainer>{{projectPart1, projectPart2}}));
 | 
						|
    EXPECT_CALL(mockPchTaskGenerator,
 | 
						|
                addProjectParts(ElementsAre(projectPart1, projectPart2),
 | 
						|
                                ElementsAre("toolChainArgument")));
 | 
						|
    EXPECT_CALL(mockProjectPartsManager,
 | 
						|
                projects(ElementsAre(projectPart4.projectPartId, projectPart5.projectPartId)))
 | 
						|
        .WillOnce(
 | 
						|
            Return(std::vector<ClangBackEnd::ProjectPartContainer>{{projectPart4, projectPart5}}));
 | 
						|
    EXPECT_CALL(mockPchTaskGenerator,
 | 
						|
                addNonSystemProjectParts(ElementsAre(projectPart4, projectPart5),
 | 
						|
                                         ElementsAre("toolChainArgument")));
 | 
						|
 | 
						|
    server.pathsWithIdsChanged({{{projectPartId1, ClangBackEnd::SourceType::TopSystemInclude}, {}},
 | 
						|
                                {{projectPartId1, ClangBackEnd::SourceType::ProjectInclude}, {}},
 | 
						|
                                {{projectPartId1, ClangBackEnd::SourceType::UserInclude}, {}},
 | 
						|
                                {{projectPartId2, ClangBackEnd::SourceType::SystemInclude}, {}},
 | 
						|
                                {{projectPartId2, ClangBackEnd::SourceType::ProjectInclude}, {}},
 | 
						|
                                {{projectPartId3, ClangBackEnd::SourceType::UserInclude}, {}},
 | 
						|
                                {{projectPartId4, ClangBackEnd::SourceType::TopProjectInclude}, {}},
 | 
						|
                                {{projectPartId4, ClangBackEnd::SourceType::Source}, {}},
 | 
						|
                                {{projectPartId4, ClangBackEnd::SourceType::UserInclude}, {}},
 | 
						|
                                {{projectPartId5, ClangBackEnd::SourceType::ProjectInclude}, {}},
 | 
						|
                                {{projectPartId6, ClangBackEnd::SourceType::Source}, {}}});
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(PchManagerServer, DontUpdateProjectPartQueueByPathIdsIfItUserFile)
 | 
						|
{
 | 
						|
    server.updateProjectParts(ClangBackEnd::UpdateProjectPartsMessage{{projectPart1, projectPart2},
 | 
						|
                                                                      {"toolChainArgument"}});
 | 
						|
 | 
						|
    EXPECT_CALL(mockProjectPartsManager, projects(_)).Times(0);
 | 
						|
    EXPECT_CALL(mockPchTaskGenerator, addProjectParts(_, _)).Times(0);
 | 
						|
    EXPECT_CALL(mockPchTaskGenerator, addNonSystemProjectParts(_, _)).Times(0);
 | 
						|
 | 
						|
    server.pathsWithIdsChanged({{{projectPartId1, ClangBackEnd::SourceType::UserInclude}, {}},
 | 
						|
                                {{projectPartId2, ClangBackEnd::SourceType::Source}, {}}});
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(PchManagerServer, ShortcutPrecompiledHeaderGenerationForUserIncludesAndSources)
 | 
						|
{
 | 
						|
    EXPECT_CALL(mockPchManagerClient,
 | 
						|
                precompiledHeadersUpdated(
 | 
						|
                    Field(&ClangBackEnd::PrecompiledHeadersUpdatedMessage::projectPartIds,
 | 
						|
                          UnorderedElementsAre(projectPartId3, projectPartId6))));
 | 
						|
 | 
						|
    server.pathsWithIdsChanged({{{projectPartId1, ClangBackEnd::SourceType::TopProjectInclude}, {}},
 | 
						|
                                {{projectPartId2, ClangBackEnd::SourceType::TopSystemInclude}, {}},
 | 
						|
                                {{projectPartId3, ClangBackEnd::SourceType::UserInclude}, {}},
 | 
						|
                                {{projectPartId4, ClangBackEnd::SourceType::ProjectInclude}, {}},
 | 
						|
                                {{projectPartId5, ClangBackEnd::SourceType::SystemInclude}, {}},
 | 
						|
                                {{projectPartId6, ClangBackEnd::SourceType::Source}, {}}});
 | 
						|
}
 | 
						|
TEST_F(PchManagerServer, ResetTimeStampForChangedFilesInDatabase)
 | 
						|
{
 | 
						|
    EXPECT_CALL(mockBuildDependenciesStorage,
 | 
						|
                insertOrUpdateIndexingTimeStamps(ElementsAre(FilePathId{1}, FilePathId{3}, FilePathId{5}),
 | 
						|
                                                 TypedEq<ClangBackEnd::TimeStamp>(-1)));
 | 
						|
 | 
						|
    server.pathsChanged({1, 3, 5});
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(PchManagerServer, SetPchCreationProgress)
 | 
						|
{
 | 
						|
    EXPECT_CALL(mockPchManagerClient,
 | 
						|
                progress(AllOf(Field(&ClangBackEnd::ProgressMessage::progressType,
 | 
						|
                                     ClangBackEnd::ProgressType::PrecompiledHeader),
 | 
						|
                               Field(&ClangBackEnd::ProgressMessage::progress, 20),
 | 
						|
                               Field(&ClangBackEnd::ProgressMessage::total, 30))));
 | 
						|
 | 
						|
    server.setPchCreationProgress(20, 30);
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(PchManagerServer, SetDependencyCreationProgress)
 | 
						|
{
 | 
						|
    EXPECT_CALL(mockPchManagerClient,
 | 
						|
                progress(AllOf(Field(&ClangBackEnd::ProgressMessage::progressType,
 | 
						|
                                     ClangBackEnd::ProgressType::DependencyCreation),
 | 
						|
                               Field(&ClangBackEnd::ProgressMessage::progress, 20),
 | 
						|
                               Field(&ClangBackEnd::ProgressMessage::total, 30))));
 | 
						|
 | 
						|
    server.setDependencyCreationProgress(20, 30);
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(PchManagerServer, RemoveToolChainsArguments)
 | 
						|
{
 | 
						|
    server.updateProjectParts(
 | 
						|
        ClangBackEnd::UpdateProjectPartsMessage{{projectPart1}, {"toolChainArgument"}});
 | 
						|
 | 
						|
    EXPECT_CALL(mockPchTaskGenerator, addProjectParts(_, _)).Times(0);
 | 
						|
    EXPECT_CALL(mockPchTaskGenerator, addNonSystemProjectParts(_, _)).Times(0);
 | 
						|
    server.removeProjectParts(removeProjectPartsMessage.clone());
 | 
						|
 | 
						|
    server.pathsWithIdsChanged({{{projectPart1.projectPartId, ClangBackEnd::SourceType::Source}, {}}});
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(PchManagerServer, DontGeneratePchIfGeneratedFilesAreNotValid)
 | 
						|
{
 | 
						|
    InSequence s;
 | 
						|
 | 
						|
    EXPECT_CALL(mockProjectPartsManager, update(ElementsAre(projectPart1)))
 | 
						|
        .WillOnce(Return(UpToDataProjectParts{{}, {projectPart1}, {projectPart2}}));
 | 
						|
    EXPECT_CALL(mockGeneratedFiles, isValid()).WillOnce(Return(false));
 | 
						|
    EXPECT_CALL(mockPchTaskGenerator, addProjectParts(_, _)).Times(0);
 | 
						|
    EXPECT_CALL(mockPchTaskGenerator, addNonSystemProjectParts(_, _)).Times(0);
 | 
						|
    EXPECT_CALL(mockProjectPartsManager,
 | 
						|
                updateDeferred(ElementsAre(projectPart1), ElementsAre(projectPart2)));
 | 
						|
 | 
						|
    server.updateProjectParts(
 | 
						|
        ClangBackEnd::UpdateProjectPartsMessage{{projectPart1}, {"toolChainArgument"}});
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(PchManagerServer, GeneratePchIfGeneratedFilesAreValid)
 | 
						|
{
 | 
						|
    InSequence s;
 | 
						|
 | 
						|
    EXPECT_CALL(mockProjectPartsManager, update(ElementsAre(projectPart1)))
 | 
						|
        .WillOnce(Return(UpToDataProjectParts{{}, {projectPart1}, {projectPart2}}));
 | 
						|
    EXPECT_CALL(mockGeneratedFiles, isValid()).WillOnce(Return(true));
 | 
						|
    EXPECT_CALL(mockPchTaskGenerator, addProjectParts(_, _));
 | 
						|
    EXPECT_CALL(mockPchTaskGenerator, addNonSystemProjectParts(_, _));
 | 
						|
    EXPECT_CALL(mockProjectPartsManager, updateDeferred(_, _)).Times(0);
 | 
						|
 | 
						|
    server.updateProjectParts(
 | 
						|
        ClangBackEnd::UpdateProjectPartsMessage{{projectPart1}, {"toolChainArgument"}});
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(PchManagerServer, AfterUpdatingGeneratedFilesAreValidSoGeneratePchs)
 | 
						|
{
 | 
						|
    InSequence s;
 | 
						|
    ClangBackEnd::UpdateGeneratedFilesMessage updateGeneratedFilesMessage{{generatedFile}};
 | 
						|
    ON_CALL(mockGeneratedFiles, isValid()).WillByDefault(Return(false));
 | 
						|
    server.updateProjectParts(ClangBackEnd::UpdateProjectPartsMessage{{projectPart1, projectPart2},
 | 
						|
                                                                      {"toolChainArgument"}});
 | 
						|
 | 
						|
    EXPECT_CALL(mockGeneratedFiles, update(updateGeneratedFilesMessage.generatedFiles));
 | 
						|
    EXPECT_CALL(mockGeneratedFiles, isValid()).WillOnce(Return(true));
 | 
						|
    EXPECT_CALL(mockProjectPartsManager, deferredSystemUpdates())
 | 
						|
        .WillOnce(Return(ClangBackEnd::ProjectPartContainers{projectPart1}));
 | 
						|
    EXPECT_CALL(mockPchTaskGenerator,
 | 
						|
                addProjectParts(ElementsAre(projectPart1), ElementsAre("toolChainArgument")));
 | 
						|
    EXPECT_CALL(mockProjectPartsManager, deferredProjectUpdates())
 | 
						|
        .WillOnce(Return(ClangBackEnd::ProjectPartContainers{projectPart2}));
 | 
						|
    EXPECT_CALL(mockPchTaskGenerator,
 | 
						|
                addNonSystemProjectParts(ElementsAre(projectPart2), ElementsAre("toolChainArgument")));
 | 
						|
 | 
						|
    server.updateGeneratedFiles(updateGeneratedFilesMessage.clone());
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(PchManagerServer, AfterUpdatingGeneratedFilesAreStillInvalidSoNoPchsGeneration)
 | 
						|
{
 | 
						|
    InSequence s;
 | 
						|
    ClangBackEnd::UpdateGeneratedFilesMessage updateGeneratedFilesMessage{{generatedFile}};
 | 
						|
    ON_CALL(mockGeneratedFiles, isValid()).WillByDefault(Return(false));
 | 
						|
    server.updateProjectParts(ClangBackEnd::UpdateProjectPartsMessage{{projectPart1, projectPart2},
 | 
						|
                                                                      {"toolChainArgument"}});
 | 
						|
 | 
						|
    EXPECT_CALL(mockGeneratedFiles, update(updateGeneratedFilesMessage.generatedFiles));
 | 
						|
    EXPECT_CALL(mockGeneratedFiles, isValid()).WillOnce(Return(false));
 | 
						|
    EXPECT_CALL(mockProjectPartsManager, deferredSystemUpdates()).Times(0);
 | 
						|
    EXPECT_CALL(mockPchTaskGenerator, addProjectParts(_, _)).Times(0);
 | 
						|
    EXPECT_CALL(mockPchTaskGenerator, addNonSystemProjectParts(_, _)).Times(0);
 | 
						|
 | 
						|
    server.updateGeneratedFiles(updateGeneratedFilesMessage.clone());
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(PchManagerServer, SentUpToDateProjectPartIdsToClient)
 | 
						|
{
 | 
						|
    InSequence s;
 | 
						|
 | 
						|
    EXPECT_CALL(mockProjectPartsManager, update(updateProjectPartsMessage.projectsParts))
 | 
						|
        .WillOnce(Return(UpToDataProjectParts{projectParts1, projectParts2, projectParts3}));
 | 
						|
    EXPECT_CALL(mockPchTaskGenerator,
 | 
						|
                addProjectParts(Eq(projectParts2), ElementsAre("toolChainArgument")));
 | 
						|
    EXPECT_CALL(mockPchManagerClient,
 | 
						|
                precompiledHeadersUpdated(
 | 
						|
                    Field(&ClangBackEnd::PrecompiledHeadersUpdatedMessage::projectPartIds,
 | 
						|
                          ElementsAre(Eq(projectPart1.projectPartId)))));
 | 
						|
 | 
						|
    server.updateProjectParts(updateProjectPartsMessage.clone());
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(PchManagerServer, AddingIncludesToFileCacheForProjectUpdates)
 | 
						|
{
 | 
						|
    InSequence s;
 | 
						|
 | 
						|
    EXPECT_CALL(mockFilePathCache, populateIfEmpty());
 | 
						|
    EXPECT_CALL(mockFilePathCache,
 | 
						|
                addFilePaths(AllOf(
 | 
						|
                    Contains(Eq(TESTDATA_DIR "/symbolscollector/include/unmodified_header.h")),
 | 
						|
                    Contains(Eq(TESTDATA_DIR "/symbolscollector/include/unmodified_header2.h")),
 | 
						|
                    Contains(Eq(TESTDATA_DIR "/builddependencycollector/project/header2.h")),
 | 
						|
                    Contains(Eq(TESTDATA_DIR "/builddependencycollector/external/external3.h")))));
 | 
						|
    EXPECT_CALL(mockProjectPartsManager, update(_));
 | 
						|
 | 
						|
    server.updateProjectParts(updateProjectPartsMessage.clone());
 | 
						|
}
 | 
						|
} // namespace
 |