diff options
Diffstat (limited to 'test')
-rw-r--r-- | test/CMakeLists.txt | 1 | ||||
-rw-r--r-- | test/test_directory.cpp | 147 |
2 files changed, 148 insertions, 0 deletions
diff --git a/test/CMakeLists.txt b/test/CMakeLists.txt index 73968fe..9061d55 100644 --- a/test/CMakeLists.txt +++ b/test/CMakeLists.txt @@ -54,6 +54,7 @@ endfunction() fud_add_test(test_result SOURCES test_result.cpp) fud_add_test(test_string SOURCES test_string.cpp) fud_add_test(test_sqlite SOURCES test_sqlite.cpp) +fud_add_test(test_directory SOURCES test_directory.cpp) # fud_add_test(test_deserialize_number SOURCES test_deserialize_number.cpp) # fud_add_test(test_ext_algorithm SOURCES test_algorithm.cpp) # fud_add_test(test_ext_array SOURCES diff --git a/test/test_directory.cpp b/test/test_directory.cpp new file mode 100644 index 0000000..9cec80d --- /dev/null +++ b/test/test_directory.cpp @@ -0,0 +1,147 @@ +/* + * libfud + * Copyright 2024 Dominick Allen + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include "fud_array.hpp" +#include "fud_c_file.hpp" +#include "fud_directory.hpp" +#include "fud_memory.hpp" +#include "fud_string.hpp" +#include "test_common.hpp" + +#include "gtest/gtest.h" +#include <algorithm> +#include <cerrno> +#include <fcntl.h> +#include <ftw.h> +#include <ranges> + +namespace fud { + +int unlink_cb(const char* fpath, const struct stat* sb, int typeflag, struct FTW* ftwbuf) +{ + int retValue = remove(fpath); + + EXPECT_EQ(retValue, 0); + if (retValue) { + perror(fpath); + } + + return retValue; +} + +FudStatus removeRecursive(const String& path) +{ + if (!path.utf8Valid()) { + return FudStatus::Utf8Invalid; + } + if (path.length() < 5) { + return FudStatus::InvalidInput; + } + const String prefix{"/tmp/"}; + auto diffResult = compareMem(path.data(), path.length(), prefix.data(), prefix.length()); + if (diffResult.isError()) { + return FudStatus::InvalidInput; + } + auto diff = diffResult.getOkay(); + if (diff != 0) { + return FudStatus::InvalidInput; + } + constexpr int maxOpenFd = 64; + auto status = nftw(path.c_str(), unlink_cb, maxOpenFd, FTW_DEPTH | FTW_PHYS); + if (status == 0) { + return FudStatus::Success; + } + + if (errno == ENOENT) { + return FudStatus::Success; + } + + return FudStatus::Failure; +} + +TEST(FudDirectory, Basic) +{ + const String testDirName{"/tmp/fud_directory_test"}; + ASSERT_TRUE(testDirName.utf8Valid()); + constexpr mode_t pathMode = 0777; + const Array<String, 2> files{ + String{"file1"}, + String{"file2"}, + }; + ASSERT_TRUE(files[0].utf8Valid()); + ASSERT_TRUE(files[1].utf8Valid()); + + ASSERT_EQ(removeRecursive(testDirName), FudStatus::Success); + + auto mkdirResult = mkdir(testDirName.c_str(), pathMode); + EXPECT_EQ(mkdirResult, 0); + if (mkdirResult != 0) { + ASSERT_EQ(removeRecursive(testDirName), FudStatus::Success); + return; + } + + const String testDirNamePrefix = testDirName.catenate("/"); + ASSERT_TRUE(testDirNamePrefix.utf8Valid()); + for (const auto& fnameBase : files) { + const auto fname = testDirNamePrefix.catenate(fnameBase); + ASSERT_TRUE(fname.utf8Valid()); + CBinaryFile file{fname, CFileMode::ReadWriteTruncate}; + ASSERT_EQ(file.open(), FileStatus::Success); + Array<utf8, 5> data{"test"}; + WriteResult expected{data.size(), FileStatus::Success}; + auto writeResult = file.write(data); + ASSERT_EQ(writeResult.bytesWritten, expected.bytesWritten); + ASSERT_EQ(writeResult.status, expected.status); + } + + Directory directory{testDirName}; + ASSERT_EQ(directory.status(), FudStatus::Success); + ASSERT_EQ(directory.errorCode(), 0); + + const Array<DirectoryEntry, 4> expectedFiles{ + DirectoryEntry{String{"."}, 0, 2, 0, DirectoryEntryType::Directory}, + DirectoryEntry{String{".."}, 0, 1, 0, DirectoryEntryType::Directory}, + DirectoryEntry{files[0], files[0].size(), 1, 0, DirectoryEntryType::RegularFile}, + DirectoryEntry{files[1], files[1].size(), 1, 0, DirectoryEntryType::RegularFile}, + }; + ASSERT_TRUE(expectedFiles[0].name.compare(expectedFiles[0].name)); + + for (auto idx = 0; idx < expectedFiles.size(); ++idx) { + auto dirEntryResult = directory.getNextEntry(); + EXPECT_TRUE(dirEntryResult.isOkay()); + const auto dirEntryOpt = dirEntryResult.getOkay(); + if (dirEntryOpt == std::nullopt) { + break; + } + const auto dirEntry = *dirEntryOpt; + const auto expected = std::find_if( + expectedFiles.begin(), + expectedFiles.end(), + [&dirEntry](const DirectoryEntry& entry) { return entry.name.compare(dirEntry.name) && entry.entryType == dirEntry.entryType; }); + EXPECT_NE(expected, nullptr); + EXPECT_NE(expected, expectedFiles.end()); + printf("%s %u\n", dirEntry.name.c_str(), static_cast<uint8_t>(dirEntry.entryType)); + } + + auto finalDirEntryResult = directory.getNextEntry(); + EXPECT_TRUE(finalDirEntryResult.isOkay()); + EXPECT_EQ(finalDirEntryResult.getOkay(), std::nullopt); + + // ASSERT_EQ(removeRecursive(testDirName), FudStatus::Success); +} + +} // namespace fud |