/**************************************************************************** ** ** Copyright (C) 2017 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 "mockmutex.h" #include "mocksqlitedatabase.h" #include "mocksqlitereadstatement.h" #include "mocksqlitewritestatement.h" #include #include namespace { using StatementFactory = ClangBackEnd::FilePathStorageSqliteStatementFactory>; using Storage = ClangBackEnd::FilePathStorage; using ClangBackEnd::Sources::Directory; using ClangBackEnd::Sources::Source; class FilePathStorage : public testing::Test { protected: void SetUp(); protected: NiceMock mockDatabase; StatementFactory factory{mockDatabase}; MockSqliteReadStatement &selectDirectoryIdFromDirectoriesByDirectoryPath = factory.selectDirectoryIdFromDirectoriesByDirectoryPath; MockSqliteReadStatement &selectSourceIdFromSourcesByDirectoryIdAndSourceName = factory.selectSourceIdFromSourcesByDirectoryIdAndSourceName; MockSqliteReadStatement &selectDirectoryPathFromDirectoriesByDirectoryId = factory.selectDirectoryPathFromDirectoriesByDirectoryId; MockSqliteReadStatement &selectSourceNameFromSourcesBySourceId = factory.selectSourceNameFromSourcesBySourceId; MockSqliteReadStatement &selectAllDirectories = factory.selectAllDirectories; MockSqliteWriteStatement &insertIntoDirectories = factory.insertIntoDirectories; MockSqliteWriteStatement &insertIntoSources = factory.insertIntoSources; MockSqliteReadStatement &selectAllSources = factory.selectAllSources; Storage storage{factory}; }; TEST_F(FilePathStorage, ReadDirectoryIdForNotContainedPath) { auto directoryId = storage.readDirectoryId("/some/not/known/path"); ASSERT_FALSE(directoryId); } TEST_F(FilePathStorage, ReadSourceIdForNotContainedPathAndDirectoryId) { auto sourceId = storage.readSourceId(23, "/some/not/known/path"); ASSERT_FALSE(sourceId); } TEST_F(FilePathStorage, ReadDirectoryIdForEmptyPath) { auto directoryId = storage.readDirectoryId(""); ASSERT_THAT(directoryId.value(), 0); } TEST_F(FilePathStorage, ReadSourceIdForEmptyNameAndZeroDirectoryId) { auto sourceId = storage.readSourceId(0, ""); ASSERT_THAT(sourceId.value(), 0); } TEST_F(FilePathStorage, ReadDirectoryIdForPath) { auto directoryId = storage.readDirectoryId("/path/to"); ASSERT_THAT(directoryId.value(), 5); } TEST_F(FilePathStorage, ReadSourceIdForPathAndDirectoryId) { auto sourceId = storage.readSourceId(5, "file.h"); ASSERT_THAT(sourceId.value(), 42); } TEST_F(FilePathStorage, FetchDirectoryIdForEmptyPath) { auto directoryId = storage.fetchDirectoryId(""); ASSERT_THAT(directoryId, 0); } TEST_F(FilePathStorage, FetchSourceIdForEmptyNameAndZeroDirectoryId) { auto sourceId = storage.fetchSourceId(0, ""); ASSERT_THAT(sourceId, 0); } TEST_F(FilePathStorage, FetchDirectoryIdForPath) { auto directoryId = storage.fetchDirectoryId("/path/to"); ASSERT_THAT(directoryId, 5); } TEST_F(FilePathStorage, FetchSourceIdForPathAndDirectoryId) { auto sourceId = storage.fetchSourceId(5, "file.h"); ASSERT_THAT(sourceId, 42); } TEST_F(FilePathStorage, CallWriteForWriteDirectory) { EXPECT_CALL(insertIntoDirectories, write(TypedEq("/some/not/known/path"))); storage.writeDirectoryId("/some/not/known/path"); } TEST_F(FilePathStorage, CallWriteForWriteSource) { EXPECT_CALL(insertIntoSources, write(5, TypedEq("unknownfile.h"))); storage.writeSourceId(5, "unknownfile.h"); } TEST_F(FilePathStorage, GetTheDirectoryIdBackAfterWritingANewEntryInDirectories) { auto directoryId = storage.writeDirectoryId("/some/not/known/path"); ASSERT_THAT(directoryId, 12); } TEST_F(FilePathStorage, GetTheSourceIdBackAfterWritingANewEntryInSources) { auto sourceId = storage.writeSourceId(5, "unknownfile.h"); ASSERT_THAT(sourceId, 12); } TEST_F(FilePathStorage, GetTheDirectoryIdBackAfterFetchingANewEntryFromDirectories) { auto directoryId = storage.fetchDirectoryId("/some/not/known/path"); ASSERT_THAT(directoryId, 12); } TEST_F(FilePathStorage, GetTheSourceIdBackAfterFetchingANewEntryFromSources) { auto sourceId = storage.fetchSourceId(5, "unknownfile.h"); ASSERT_THAT(sourceId, 12); } TEST_F(FilePathStorage, CallSelectForFetchingDirectoryIdForKnownPath) { InSequence s; EXPECT_CALL(mockDatabase, deferredBegin()); EXPECT_CALL(selectDirectoryIdFromDirectoriesByDirectoryPath, valueReturnInt32(Eq("/path/to"))); EXPECT_CALL(mockDatabase, commit()); storage.fetchDirectoryId("/path/to"); } TEST_F(FilePathStorage, CallSelectForFetchingSourceIdForKnownPath) { InSequence s; EXPECT_CALL(mockDatabase, deferredBegin()); EXPECT_CALL(selectSourceIdFromSourcesByDirectoryIdAndSourceName, valueReturnInt32(5, Eq("file.h"))); EXPECT_CALL(mockDatabase, commit()); storage.fetchSourceId(5, "file.h"); } TEST_F(FilePathStorage, CallNotWriteForFetchingDirectoryIdForKnownPath) { EXPECT_CALL(insertIntoDirectories, write(An())).Times(0); storage.fetchDirectoryId("/path/to"); } TEST_F(FilePathStorage, CallNotWriteForFetchingSoureIdForKnownEntry) { EXPECT_CALL(insertIntoSources, write(An(), An())).Times(0); storage.fetchSourceId(5, "file.h"); } TEST_F(FilePathStorage, CallSelectAndWriteForFetchingDirectoryIdForUnknownPath) { InSequence s; EXPECT_CALL(mockDatabase, deferredBegin()); EXPECT_CALL(selectDirectoryIdFromDirectoriesByDirectoryPath, valueReturnInt32(Eq("/some/not/known/path"))); EXPECT_CALL(insertIntoDirectories, write(TypedEq("/some/not/known/path"))); EXPECT_CALL(mockDatabase, commit()); storage.fetchDirectoryId("/some/not/known/path"); } TEST_F(FilePathStorage, CallSelectAndWriteForFetchingSourceIdForUnknownEntry) { InSequence s; EXPECT_CALL(mockDatabase, deferredBegin()); EXPECT_CALL(selectSourceIdFromSourcesByDirectoryIdAndSourceName, valueReturnInt32(5, Eq("unknownfile.h"))); EXPECT_CALL(insertIntoSources, write(5, TypedEq("unknownfile.h"))); EXPECT_CALL(mockDatabase, commit()); storage.fetchSourceId(5, "unknownfile.h"); } TEST_F(FilePathStorage, RestartFetchDirectoryIDIfTheDatabaseIsBusyInBeginBecauseTheTableAlreadyChanged) { InSequence s; EXPECT_CALL(mockDatabase, deferredBegin()).WillOnce(Throw(Sqlite::StatementIsBusy("busy"))); EXPECT_CALL(mockDatabase, rollback()).Times(0); EXPECT_CALL(mockDatabase, deferredBegin()); EXPECT_CALL(selectDirectoryIdFromDirectoriesByDirectoryPath, valueReturnInt32(Eq("/other/unknow/path"))); EXPECT_CALL(insertIntoDirectories, write(TypedEq("/other/unknow/path"))); EXPECT_CALL(mockDatabase, commit()); storage.fetchDirectoryId("/other/unknow/path"); } TEST_F(FilePathStorage, CallSelectAndWriteForFetchingDirectoryIdTwoTimesIfTheDatabaseIsBusyInWriteBecauseTheTableAlreadyChanged) { InSequence s; EXPECT_CALL(mockDatabase, deferredBegin()); EXPECT_CALL(selectDirectoryIdFromDirectoriesByDirectoryPath, valueReturnInt32(Eq("/other/unknow/path"))); EXPECT_CALL(insertIntoDirectories, write(TypedEq("/other/unknow/path"))) .WillOnce(Throw(Sqlite::StatementIsBusy("busy"))); EXPECT_CALL(mockDatabase, rollback()); EXPECT_CALL(mockDatabase, deferredBegin()); EXPECT_CALL(selectDirectoryIdFromDirectoriesByDirectoryPath, valueReturnInt32(Eq("/other/unknow/path"))); EXPECT_CALL(insertIntoDirectories, write(TypedEq("/other/unknow/path"))); EXPECT_CALL(mockDatabase, commit()); storage.fetchDirectoryId("/other/unknow/path"); } TEST_F(FilePathStorage, CallSelectAndWriteForFetchingDirectoryIdTwoTimesIfTheIndexIsConstraintBecauseTheEntryExistsAlready) { InSequence s; EXPECT_CALL(mockDatabase,deferredBegin()); EXPECT_CALL(selectDirectoryIdFromDirectoriesByDirectoryPath, valueReturnInt32(Eq("/other/unknow/path"))); EXPECT_CALL(insertIntoDirectories, write(TypedEq("/other/unknow/path"))) .WillOnce(Throw(Sqlite::ConstraintPreventsModification("busy"))); EXPECT_CALL(mockDatabase, rollback()); EXPECT_CALL(mockDatabase,deferredBegin()); EXPECT_CALL(selectDirectoryIdFromDirectoriesByDirectoryPath, valueReturnInt32(Eq("/other/unknow/path"))); EXPECT_CALL(mockDatabase, commit()); storage.fetchDirectoryId("/other/unknow/path"); } TEST_F(FilePathStorage, RestartFetchSourceIdIfTheDatabaseIsBusyInBeginBecauseTheTableAlreadyChanged) { InSequence s; EXPECT_CALL(mockDatabase, deferredBegin()).WillOnce(Throw(Sqlite::StatementIsBusy("busy"))); EXPECT_CALL(mockDatabase, rollback()).Times(0); EXPECT_CALL(mockDatabase, deferredBegin()); EXPECT_CALL(selectSourceIdFromSourcesByDirectoryIdAndSourceName, valueReturnInt32(5, Eq("otherunknownfile.h"))); EXPECT_CALL(insertIntoSources, write(5, TypedEq("otherunknownfile.h"))); EXPECT_CALL(mockDatabase, commit()); storage.fetchSourceId(5, "otherunknownfile.h"); } TEST_F(FilePathStorage, CallSelectAndWriteForFetchingSourceTwoTimesIfTheDatabaseIsBusyInWriteBecauseTheTableAlreadyChanged) { InSequence s; EXPECT_CALL(mockDatabase, deferredBegin()); EXPECT_CALL(selectSourceIdFromSourcesByDirectoryIdAndSourceName, valueReturnInt32(5, Eq("otherunknownfile.h"))); EXPECT_CALL(insertIntoSources, write(5, TypedEq("otherunknownfile.h"))) .WillOnce(Throw(Sqlite::StatementIsBusy("busy")));; EXPECT_CALL(mockDatabase, rollback()); EXPECT_CALL(mockDatabase, deferredBegin()); EXPECT_CALL(selectSourceIdFromSourcesByDirectoryIdAndSourceName, valueReturnInt32(5, Eq("otherunknownfile.h"))); EXPECT_CALL(insertIntoSources, write(5, TypedEq("otherunknownfile.h"))); EXPECT_CALL(mockDatabase, commit()); storage.fetchSourceId(5, "otherunknownfile.h"); } TEST_F(FilePathStorage, CallSelectAndWriteForFetchingSourceTwoTimesIfTheIndexIsConstraintBecauseTheEntryExistsAlready) { InSequence s; EXPECT_CALL(mockDatabase,deferredBegin()); EXPECT_CALL(selectSourceIdFromSourcesByDirectoryIdAndSourceName, valueReturnInt32(5, Eq("otherunknownfile.h"))); EXPECT_CALL(insertIntoSources, write(5, TypedEq("otherunknownfile.h"))) .WillOnce(Throw(Sqlite::ConstraintPreventsModification("busy")));; EXPECT_CALL(mockDatabase, rollback()); EXPECT_CALL(mockDatabase,deferredBegin()); EXPECT_CALL(selectSourceIdFromSourcesByDirectoryIdAndSourceName, valueReturnInt32(5, Eq("otherunknownfile.h"))); EXPECT_CALL(mockDatabase, commit()); storage.fetchSourceId(5, "otherunknownfile.h"); } TEST_F(FilePathStorage, SelectAllDirectories) { auto directories = storage.fetchAllDirectories(); ASSERT_THAT(directories, ElementsAre(Directory{1, "/path/to"}, Directory{2, "/other/path"})); } TEST_F(FilePathStorage, SelectAllSources) { auto sources = storage.fetchAllSources(); ASSERT_THAT(sources, ElementsAre(Source{1, "file.h"}, Source{4, "file.cpp"})); } TEST_F(FilePathStorage, CallSelectAllDirectories) { EXPECT_CALL(mockDatabase, deferredBegin()); EXPECT_CALL(selectAllDirectories, valuesReturnStdVectorDirectory(256)); EXPECT_CALL(mockDatabase, commit()); storage.fetchAllDirectories(); } TEST_F(FilePathStorage, CallSelectAllSources) { EXPECT_CALL(mockDatabase, deferredBegin()); EXPECT_CALL(selectAllSources, valuesReturnStdVectorSource(8192)); EXPECT_CALL(mockDatabase, commit()); storage.fetchAllSources(); } TEST_F(FilePathStorage, CallValueForFetchDirectoryPathForId) { EXPECT_CALL(mockDatabase, deferredBegin()); EXPECT_CALL(selectDirectoryPathFromDirectoriesByDirectoryId, valueReturnPathString(5)); EXPECT_CALL(mockDatabase, commit()); storage.fetchDirectoryPath(5); } TEST_F(FilePathStorage, FetchDirectoryPathForId) { auto path = storage.fetchDirectoryPath(5); ASSERT_THAT(path, Eq("/path/to")); } TEST_F(FilePathStorage, ThrowAsFetchingDirectoryPathForNonExistingId) { ASSERT_THROW(storage.fetchDirectoryPath(12), ClangBackEnd::DirectoryPathIdDoesNotExists); } TEST_F(FilePathStorage, CallValueForFetchSoureNameForId) { EXPECT_CALL(mockDatabase, deferredBegin()); EXPECT_CALL(selectSourceNameFromSourcesBySourceId, valueReturnSmallString(42)); EXPECT_CALL(mockDatabase, commit()); storage.fetchSourceName(42); } TEST_F(FilePathStorage, FetchSoureNameForId) { auto path = storage.fetchSourceName(42); ASSERT_THAT(path, Eq("file.cpp")); } TEST_F(FilePathStorage, ThrowAsFetchingSourceNameForNonExistingId) { ASSERT_THROW(storage.fetchSourceName(12), ClangBackEnd::SourceNameIdDoesNotExists); } TEST_F(FilePathStorage, RestartFetchSourceNameIfTheDatabaseIsBusyInBegin) { InSequence s; EXPECT_CALL(mockDatabase, lock()); EXPECT_CALL(mockDatabase, deferredBegin()).WillOnce(Throw(Sqlite::StatementIsBusy("busy"))); EXPECT_CALL(mockDatabase, rollback()).Times(0); EXPECT_CALL(mockDatabase, unlock()); EXPECT_CALL(mockDatabase, lock()); EXPECT_CALL(mockDatabase, deferredBegin()); EXPECT_CALL(selectSourceNameFromSourcesBySourceId, valueReturnSmallString(42)); EXPECT_CALL(mockDatabase, commit()); EXPECT_CALL(mockDatabase, unlock()); storage.fetchSourceName(42); } TEST_F(FilePathStorage, RestartFetchDirectoryPathIfTheDatabaseIsBusyInBegin) { InSequence s; EXPECT_CALL(mockDatabase, lock()); EXPECT_CALL(mockDatabase, deferredBegin()).WillOnce(Throw(Sqlite::StatementIsBusy("busy"))); EXPECT_CALL(mockDatabase, rollback()).Times(0); EXPECT_CALL(mockDatabase, unlock()); EXPECT_CALL(mockDatabase, lock()); EXPECT_CALL(mockDatabase, deferredBegin()); EXPECT_CALL(selectDirectoryPathFromDirectoriesByDirectoryId, valueReturnPathString(5)); EXPECT_CALL(mockDatabase, commit()); EXPECT_CALL(mockDatabase, unlock()); storage.fetchDirectoryPath(5); } TEST_F(FilePathStorage, RestartFetchAllDirectoriesIfBeginIsBusy) { InSequence s; EXPECT_CALL(mockDatabase, lock()); EXPECT_CALL(mockDatabase, deferredBegin()).WillOnce(Throw(Sqlite::StatementIsBusy("busy"))); EXPECT_CALL(mockDatabase, rollback()).Times(0); EXPECT_CALL(mockDatabase, unlock()); EXPECT_CALL(mockDatabase, lock()); EXPECT_CALL(mockDatabase, deferredBegin()); EXPECT_CALL(selectAllDirectories, valuesReturnStdVectorDirectory(256)); EXPECT_CALL(mockDatabase, commit()); EXPECT_CALL(mockDatabase, unlock()); storage.fetchAllDirectories(); } TEST_F(FilePathStorage, RestartFetchAllSourcesIfBeginIsBusy) { InSequence s; EXPECT_CALL(mockDatabase, lock()); EXPECT_CALL(mockDatabase, deferredBegin()).WillOnce(Throw(Sqlite::StatementIsBusy("busy"))); EXPECT_CALL(mockDatabase, rollback()).Times(0); EXPECT_CALL(mockDatabase, unlock()); EXPECT_CALL(mockDatabase, lock()); EXPECT_CALL(mockDatabase, deferredBegin()); EXPECT_CALL(selectAllSources, valuesReturnStdVectorSource(8192)); EXPECT_CALL(mockDatabase, commit()); EXPECT_CALL(mockDatabase, unlock()); storage.fetchAllSources(); } void FilePathStorage::SetUp() { ON_CALL(selectDirectoryIdFromDirectoriesByDirectoryPath, valueReturnInt32(_)) .WillByDefault(Return(Utils::optional())); ON_CALL(selectDirectoryIdFromDirectoriesByDirectoryPath, valueReturnInt32(Utils::SmallStringView(""))) .WillByDefault(Return(Utils::optional(0))); ON_CALL(selectDirectoryIdFromDirectoriesByDirectoryPath, valueReturnInt32(Utils::SmallStringView("/path/to"))) .WillByDefault(Return(Utils::optional(5))); ON_CALL(mockDatabase, lastInsertedRowId()) .WillByDefault(Return(12)); ON_CALL(selectAllDirectories, valuesReturnStdVectorDirectory(_)) .WillByDefault(Return(std::vector{{1, "/path/to"}, {2, "/other/path"}})); ON_CALL(selectSourceIdFromSourcesByDirectoryIdAndSourceName, valueReturnInt32(_, _)) .WillByDefault(Return(Utils::optional())); ON_CALL(selectSourceIdFromSourcesByDirectoryIdAndSourceName, valueReturnInt32(0, Utils::SmallStringView(""))) .WillByDefault(Return(Utils::optional(0))); ON_CALL(selectSourceIdFromSourcesByDirectoryIdAndSourceName, valueReturnInt32(5, Utils::SmallStringView("file.h"))) .WillByDefault(Return(Utils::optional(42))); ON_CALL(selectAllSources, valuesReturnStdVectorSource(_)) .WillByDefault(Return(std::vector{{1, "file.h"}, {4, "file.cpp"}})); ON_CALL(selectDirectoryPathFromDirectoriesByDirectoryId, valueReturnPathString(5)) .WillByDefault(Return(Utils::optional("/path/to"))); ON_CALL(selectSourceNameFromSourcesBySourceId, valueReturnSmallString(42)) .WillByDefault(Return(Utils::optional("file.cpp"))); EXPECT_CALL(selectDirectoryIdFromDirectoriesByDirectoryPath, valueReturnInt32(_)) .Times(AnyNumber()); EXPECT_CALL(selectSourceIdFromSourcesByDirectoryIdAndSourceName, valueReturnInt32(_, _)) .Times(AnyNumber()); EXPECT_CALL(insertIntoDirectories, write(An())) .Times(AnyNumber()); EXPECT_CALL(insertIntoSources, write(An(), A())) .Times(AnyNumber()); EXPECT_CALL(selectAllDirectories, valuesReturnStdVectorDirectory(_)) .Times(AnyNumber()); EXPECT_CALL(selectAllSources, valuesReturnStdVectorSource(_)) .Times(AnyNumber()); EXPECT_CALL(selectDirectoryPathFromDirectoriesByDirectoryId, valueReturnPathString(_)) .Times(AnyNumber()); EXPECT_CALL(selectSourceNameFromSourcesBySourceId, valueReturnSmallString(_)) .Times(AnyNumber()); } }