diff --git a/platform/platform.cpp b/platform/platform.cpp index 5eeb043ff3..f17f278740 100644 --- a/platform/platform.cpp +++ b/platform/platform.cpp @@ -3,6 +3,7 @@ #include "coding/base64.hpp" #include "coding/file_name_utils.hpp" +#include "coding/internal/file_data.hpp" #include "coding/sha2.hpp" #include "coding/writer.hpp" @@ -15,6 +16,14 @@ #include +namespace +{ +bool IsSpecialDirName(string const & dirName) +{ + return dirName == "." || dirName == ".."; +} +} // namespace + // static Platform::EError Platform::ErrnoToError() { @@ -33,6 +42,38 @@ Platform::EError Platform::ErrnoToError() } } +// static +bool Platform::RmDirRecursively(string const & dirName) +{ + if (dirName.empty() || IsSpecialDirName(dirName)) + return false; + + bool res = true; + + TFilesWithType fwts; + GetFilesByType(dirName, FILE_TYPE_REGULAR | FILE_TYPE_DIRECTORY, fwts); + for (auto const & fwt : fwts) + { + string const & name = fwt.first; + EFileType const type = fwt.second; + if (type == FILE_TYPE_REGULAR) + { + if (!my::DeleteFileX(my::JoinFoldersToPath(dirName, name))) + res = false; + } + else if (type == FILE_TYPE_DIRECTORY && !IsSpecialDirName(name)) + { + if (!RmDirRecursively(my::JoinFoldersToPath(dirName, name))) + res = false; + } + } + + if (RmDir(dirName) != ERR_OK) + res = false; + + return res; +} + string Platform::ReadPathForFile(string const & file, string searchScope) const { if (searchScope.empty()) diff --git a/platform/platform.hpp b/platform/platform.hpp index d7e071dc38..127c7c019e 100644 --- a/platform/platform.hpp +++ b/platform/platform.hpp @@ -106,6 +106,11 @@ public: /// Removes empty directory from the filesystem. static EError RmDir(string const & dirName); + /// Removes directory from the filesystem. + /// @note Directory can be non empty. + /// @note If function fails, directory can be partially removed. + static bool RmDirRecursively(string const & dirName); + /// @TODO create join method for string concatenation /// @return path for directory with temporary files with slash at the end diff --git a/platform/platform_tests/platform_test.cpp b/platform/platform_tests/platform_test.cpp index b17bf60b99..52a881d5a7 100644 --- a/platform/platform_tests/platform_test.cpp +++ b/platform/platform_tests/platform_test.cpp @@ -212,3 +212,39 @@ UNIT_TEST(GetWritableStorageStatus) TEST_EQUAL(Platform::STORAGE_OK, GetPlatform().GetWritableStorageStatus(100000), ()); TEST_EQUAL(Platform::NOT_ENOUGH_SPACE, GetPlatform().GetWritableStorageStatus(uint64_t(-1)), ()); } + +UNIT_TEST(RmDirRecursively) +{ + Platform & platform = GetPlatform(); + + string const testDir1 = my::JoinFoldersToPath(platform.WritableDir(), "test_dir1"); + TEST_EQUAL(platform.MkDir(testDir1), Platform::ERR_OK, ()); + MY_SCOPE_GUARD(removeTestDir1, bind(&Platform::RmDir, testDir1)); + + string const testDir2 = my::JoinFoldersToPath(testDir1, "test_dir2"); + TEST_EQUAL(platform.MkDir(testDir2), Platform::ERR_OK, ()); + MY_SCOPE_GUARD(removeTestDir2, bind(&Platform::RmDir, testDir2)); + + string const filePath = my::JoinFoldersToPath(testDir2, "test_file"); + { + FileWriter testFile(filePath); + testFile.Write("HOHOHO", 6); + } + MY_SCOPE_GUARD(removeTestFile, bind(&my::DeleteFileX, filePath)); + + TEST(Platform::IsFileExistsByFullPath(filePath), ()); + TEST(Platform::IsFileExistsByFullPath(testDir1), ()); + TEST(Platform::IsFileExistsByFullPath(testDir2), ()); + + TEST_EQUAL(Platform::ERR_DIRECTORY_NOT_EMPTY, Platform::RmDir(testDir1), ()); + + TEST(Platform::IsFileExistsByFullPath(filePath), ()); + TEST(Platform::IsFileExistsByFullPath(testDir1), ()); + TEST(Platform::IsFileExistsByFullPath(testDir2), ()); + + TEST(Platform::RmDirRecursively(testDir1), ()); + + TEST(!Platform::IsFileExistsByFullPath(filePath), ()); + TEST(!Platform::IsFileExistsByFullPath(testDir1), ()); + TEST(!Platform::IsFileExistsByFullPath(testDir2), ()); +}