/**************************************************************************** ** ** 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 "mockprogressmanager.h" #include "mockrefactoringserver.h" #include "mocksearch.h" #include "mocksearchhandle.h" #include #include #include #include #include #include namespace { using testing::_; using testing::AllOf; using testing::Field; using testing::NiceMock; using testing::NotNull; using testing::Return; using testing::ReturnNew; using testing::DefaultValue; using testing::ByMove; using CppTools::CompilerOptionsBuilder; using ClangBackEnd::V2::FileContainer; std::vector createCommandLines( const std::vector &projectParts) { using Filter = ClangRefactoring::ClangQueryProjectsFindFilter; std::vector commandLines; for (const CppTools::ProjectPart::Ptr &projectPart : projectParts) { for (const CppTools::ProjectFile &projectFile : projectPart->files) { Utils::SmallStringVector commandLine = Filter::compilerArguments(projectPart.data(), projectFile.kind); commandLine.emplace_back(projectFile.path); commandLines.push_back(commandLine); } } return commandLines; } class ClangQueryProjectFindFilter : public ::testing::Test { protected: void SetUp() { projectsParts = createProjectParts(); commandLines = createCommandLines(projectsParts); findFilter.setProjectParts(projectsParts); findFilter.setUnsavedContent({unsavedContent.clone()}); ON_CALL(mockSearch, startNewSearch(QStringLiteral("Clang Query"), findDeclQueryText)) .WillByDefault(Return(ByMove(createSearchHandle()))); } std::unique_ptr createSearchHandle() { auto handle = std::make_unique>(); handle->setRefactoringServer(&mockRefactoringServer); return handle; } std::vector createProjectParts() { auto projectPart1 = CppTools::ProjectPart::Ptr(new CppTools::ProjectPart); projectPart1->project = &project; projectPart1->files.append({"/path/to/file1.h", CppTools::ProjectFile::CXXHeader}); projectPart1->files.append({"/path/to/file1.cpp", CppTools::ProjectFile::CXXSource}); auto projectPart2 = CppTools::ProjectPart::Ptr(new CppTools::ProjectPart); projectPart2->project = &project; projectPart2->files.append({"/path/to/file2.cpp", CppTools::ProjectFile::CXXSource}); projectPart2->files.append({"/path/to/unsaved.cpp", CppTools::ProjectFile::CXXSource}); projectPart2->files.append({"/path/to/cheader.h", CppTools::ProjectFile::CHeader}); return {projectPart1, projectPart2}; } protected: NiceMock mockRefactoringServer; NiceMock mockSearch; NiceMock mockProgressManager; ClangRefactoring::RefactoringClient refactoringClient{mockProgressManager}; ClangRefactoring::ClangQueryProjectsFindFilter findFilter{mockRefactoringServer, mockSearch, refactoringClient}; QString findDeclQueryText{"functionDecl()"}; QString curentDocumentFilePath{"/path/to/file.cpp"}; QString unsavedDocumentContent{"void f();"}; std::vector commandLines; std::vector projectsParts; ClangBackEnd::V2::FileContainer unsavedContent{{"/path/to", "unsaved.cpp"}, "void f();", {}}; ProjectExplorer::Project project; }; TEST_F(ClangQueryProjectFindFilter, SupportedFindFlags) { auto findFlags = findFilter.supportedFindFlags(); ASSERT_FALSE(findFlags); } TEST_F(ClangQueryProjectFindFilter, IsNotUsableForUnusableServer) { auto isUsable = findFilter.isAvailable(); ASSERT_FALSE(isUsable); } TEST_F(ClangQueryProjectFindFilter, IsUsableForUsableServer) { mockRefactoringServer.setAvailable(true); auto isUsable = findFilter.isAvailable(); ASSERT_TRUE(isUsable); } TEST_F(ClangQueryProjectFindFilter, ServerIsUsableForUsableFindFilter) { findFilter.setAvailable(true); auto isUsable = mockRefactoringServer.isAvailable(); ASSERT_TRUE(isUsable); } TEST_F(ClangQueryProjectFindFilter, SearchHandleSetIsSetAfterFindAll) { findFilter.find(findDeclQueryText); auto searchHandle = refactoringClient.searchHandle(); ASSERT_THAT(searchHandle, NotNull()); } TEST_F(ClangQueryProjectFindFilter, FindAllIsCallingStartNewSearch) { EXPECT_CALL(mockSearch, startNewSearch(QStringLiteral("Clang Query"), findDeclQueryText)); findFilter.find(findDeclQueryText); } TEST_F(ClangQueryProjectFindFilter, FindAllIsSettingExprectedResultCountInTheRefactoringClient) { findFilter.find(findDeclQueryText); ASSERT_THAT(refactoringClient.expectedResultCount(), 3); } TEST_F(ClangQueryProjectFindFilter, FindAllIsCallingRequestSourceRangesAndDiagnosticsForQueryMessage) { ClangBackEnd::RequestSourceRangesForQueryMessage message(findDeclQueryText, {{{"/path/to", "file1.h"}, "", commandLines[0].clone()}, {{"/path/to", "file1.cpp"}, "", commandLines[1].clone()}, {{"/path/to", "file2.cpp"}, "", commandLines[2].clone()}}, {unsavedContent.clone()}); EXPECT_CALL(mockRefactoringServer, requestSourceRangesForQueryMessage(message)); findFilter.find(findDeclQueryText); } TEST_F(ClangQueryProjectFindFilter, CancelSearch) { auto searchHandle = findFilter.find(findDeclQueryText); EXPECT_CALL(mockRefactoringServer, cancel()); searchHandle->cancel(); } TEST_F(ClangQueryProjectFindFilter, CallingRequestSourceRangesAndDiagnostics) { using Message = ClangBackEnd::RequestSourceRangesAndDiagnosticsForQueryMessage; Utils::SmallString queryText = "functionDecl()"; Utils::SmallString exampleContent = "void foo();"; EXPECT_CALL(mockRefactoringServer, requestSourceRangesAndDiagnosticsForQueryMessage( AllOf( Field(&Message::source, Field(&FileContainer::unsavedFileContent, exampleContent)), Field(&Message::query, queryText)))); findFilter.requestSourceRangesAndDiagnostics(QString(queryText), QString(exampleContent)); } }