Files
netris-cdc-file-transfer/common/dir_iter_test.cc
Christian Schneider 4326e972ac Releasing the former Stadia file transfer tools
The tools allow efficient and fast synchronization of large directory
trees from a Windows workstation to a Linux target machine.

cdc_rsync* support efficient copy of files by using content-defined
chunking (CDC) to identify chunks within files that can be reused.

asset_stream_manager + cdc_fuse_fs support efficient streaming of a
local directory to a remote virtual file system based on FUSE. It also
employs CDC to identify and reuse unchanged data chunks.
2022-11-03 10:39:10 +01:00

268 lines
8.3 KiB
C++

// Copyright 2022 Google LLC
//
// 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 "common/dir_iter.h"
#include <algorithm>
#include <iterator>
#include "common/path.h"
#include "common/platform.h"
#include "common/status_test_macros.h"
#include "common/test_main.h"
#include "gtest/gtest.h"
namespace cdc_ft {
namespace {
class DirectoryIteratorTest : public ::testing::Test {
public:
using StringSet = std::set<std::string>;
DirectoryIteratorTest() {}
void SetUp() override {
expected_files_.insert("a/aa/aaa1.txt");
expected_files_.insert("a/aa/aaa2.txt");
expected_files_.insert("a/aa1.txt");
expected_files_.insert("a/aa2.txt");
expected_files_.insert("a/ab/aab1.txt");
expected_files_.insert("a/ab/aab2.txt");
expected_files_.insert("b/ba/bba1.txt");
expected_files_.insert("b/ba/bba2.txt");
expected_files_.insert("b/bb/bbb1.txt");
expected_files_.insert("b/bb/bbb2.txt");
expected_files_.insert("c/c1.txt");
expected_files_.insert("c/c2.txt");
expected_files_.insert("d/d1.txt");
expected_files_.insert("d/d2.txt");
expected_files_.insert("root.txt");
expected_dirs_.insert("a");
expected_dirs_.insert("a/aa");
expected_dirs_.insert("a/ab");
expected_dirs_.insert("b");
expected_dirs_.insert("b/ba");
expected_dirs_.insert("b/bb");
expected_dirs_.insert("c");
expected_dirs_.insert("d");
}
static StringSet Union(const StringSet& a, const StringSet& b) {
StringSet out(a);
out.insert(b.begin(), b.end());
return out;
}
static StringSet Filter(
const StringSet& items,
std::function<bool(const std::string& item)> filter_fn) {
StringSet out;
std::copy_if(items.begin(), items.end(), std::inserter(out, out.begin()),
[&](const std::string& f) { return !filter_fn(f); });
return out;
}
protected:
std::set<std::string> expected_files_;
std::set<std::string> expected_dirs_;
std::string test_data_dir_ = GetTestDataDir("dir_iter");
};
TEST_F(DirectoryIteratorTest, EmptyDirIterator) {
DirectoryIterator dit;
DirectoryEntry dent;
EXPECT_OK(dit.Status());
EXPECT_FALSE(dit.Valid());
EXPECT_EQ(dit.Path(), std::string());
EXPECT_FALSE(dit.NextEntry(&dent));
}
TEST_F(DirectoryIteratorTest, ValidDirIterator) {
DirectoryIterator dit(test_data_dir_);
DirectoryEntry dent;
EXPECT_OK(dit.Status());
EXPECT_TRUE(dit.Valid());
EXPECT_EQ(dit.Path(), test_data_dir_);
EXPECT_TRUE(dit.NextEntry(&dent));
}
TEST_F(DirectoryIteratorTest, FindFiles) {
DirectoryIterator dit(test_data_dir_, DirectorySearchFlags::kFiles, false);
DirectoryEntry dent;
std::set<std::string> found;
while (dit.NextEntry(&dent)) {
found.insert(path::ToUnix(dent.RelPathName()));
}
EXPECT_OK(dit.Status());
std::set<std::string> expected = Filter(
expected_files_,
[](const std::string& f) { return f.find('/') != std::string::npos; });
EXPECT_EQ(found, expected);
}
TEST_F(DirectoryIteratorTest, FindDirectories) {
DirectoryIterator dit(test_data_dir_, DirectorySearchFlags::kDirectories,
false);
DirectoryEntry dent;
std::set<std::string> found;
while (dit.NextEntry(&dent)) {
found.insert(path::ToUnix(dent.RelPathName()));
}
EXPECT_OK(dit.Status());
std::set<std::string> expected = Filter(
expected_dirs_,
[](const std::string& f) { return f.find('/') != std::string::npos; });
EXPECT_EQ(found, expected);
}
TEST_F(DirectoryIteratorTest, FindFilesAndDirectories) {
DirectoryIterator dit(test_data_dir_,
DirectorySearchFlags::kFilesAndDirectories, false);
DirectoryEntry dent;
std::set<std::string> found;
while (dit.NextEntry(&dent)) {
found.insert(path::ToUnix(dent.RelPathName()));
}
EXPECT_OK(dit.Status());
std::set<std::string> expected = Filter(
Union(expected_dirs_, expected_files_),
[](const std::string& f) { return f.find('/') != std::string::npos; });
EXPECT_EQ(found, expected);
}
TEST_F(DirectoryIteratorTest, FindFilesRecursive) {
DirectoryIterator dit(test_data_dir_, DirectorySearchFlags::kFiles);
DirectoryEntry dent;
std::set<std::string> found;
while (dit.NextEntry(&dent)) {
found.insert(path::ToUnix(dent.RelPathName()));
}
EXPECT_OK(dit.Status());
EXPECT_EQ(found, expected_files_);
}
TEST_F(DirectoryIteratorTest, FindDirectoriesRecursive) {
DirectoryIterator dit(test_data_dir_, DirectorySearchFlags::kDirectories);
DirectoryEntry dent;
std::set<std::string> found;
while (dit.NextEntry(&dent)) {
found.insert(path::ToUnix(dent.RelPathName()));
}
EXPECT_OK(dit.Status());
EXPECT_EQ(found, expected_dirs_);
}
TEST_F(DirectoryIteratorTest, FindFilesAndDirectoriesRecursive) {
DirectoryIterator dit(test_data_dir_);
DirectoryEntry dent;
std::set<std::string> found;
while (dit.NextEntry(&dent)) {
found.insert(path::ToUnix(dent.RelPathName()));
}
EXPECT_OK(dit.Status());
EXPECT_EQ(found, Union(expected_dirs_, expected_files_));
}
TEST_F(DirectoryIteratorTest, DirNotFound) {
DirectoryIterator dit("does/not/exist");
DirectoryEntry dent;
std::set<std::string> found;
while (dit.NextEntry(&dent)) {
found.insert(path::ToUnix(dent.RelPathName()));
}
EXPECT_TRUE(found.empty());
EXPECT_TRUE(absl::IsNotFound(dit.Status()));
}
TEST_F(DirectoryIteratorTest, HandleError) {
// Trying to open a file instead of a directory.
DirectoryIterator dit(path::Join(test_data_dir_, "root.txt"));
DirectoryEntry dent;
std::set<std::string> found;
while (dit.NextEntry(&dent)) {
found.insert(path::ToUnix(dent.RelPathName()));
}
EXPECT_TRUE(found.empty());
EXPECT_NOT_OK(dit.Status());
}
TEST_F(DirectoryIteratorTest, EmptyDirEntry) {
DirectoryEntry dent;
EXPECT_FALSE(dent.Valid());
EXPECT_FALSE(dent.IsDir());
EXPECT_FALSE(dent.IsRegularFile());
EXPECT_FALSE(dent.IsSymlink());
EXPECT_EQ(dent.Name(), std::string());
EXPECT_EQ(dent.RelPath(), std::string());
EXPECT_EQ(dent.RelPathName(), std::string());
}
TEST_F(DirectoryIteratorTest, DirEntryForFile) {
DirectoryIterator dit(test_data_dir_, DirectorySearchFlags::kFiles, false);
DirectoryEntry dent;
EXPECT_TRUE(dit.NextEntry(&dent));
EXPECT_OK(dit.Status());
EXPECT_TRUE(dent.Valid());
EXPECT_FALSE(dent.IsDir());
EXPECT_TRUE(dent.IsRegularFile());
#ifdef PLATFORM_WINDOWS
// On Windows, Bazel does not copy nor link data dependencies.
EXPECT_FALSE(dent.IsSymlink());
#else
// On Linux, Bazel creates symlinks for files in data dependencies.
EXPECT_TRUE(dent.IsSymlink());
#endif
EXPECT_EQ(dent.Name(), "root.txt");
EXPECT_EQ(dent.RelPath(), std::string());
EXPECT_EQ(dent.RelPathName(), "root.txt");
}
TEST_F(DirectoryIteratorTest, DirEntryForDir) {
std::set<std::string> expected_any = Filter(
expected_dirs_,
[](const std::string& f) { return f.find('/') != std::string::npos; });
DirectoryIterator dit(test_data_dir_, DirectorySearchFlags::kDirectories,
false);
DirectoryEntry dent;
EXPECT_TRUE(dit.NextEntry(&dent));
EXPECT_OK(dit.Status());
EXPECT_TRUE(dent.Valid());
EXPECT_TRUE(dent.IsDir());
EXPECT_FALSE(dent.IsRegularFile());
EXPECT_FALSE(dent.IsSymlink());
EXPECT_TRUE(expected_any.find(dent.Name()) != expected_any.end());
EXPECT_EQ(dent.RelPath(), std::string());
EXPECT_TRUE(expected_any.find(dent.RelPathName()) != expected_any.end());
}
TEST_F(DirectoryIteratorTest, DirEntryClear) {
DirectoryIterator dit(test_data_dir_, DirectorySearchFlags::kFiles, false);
DirectoryEntry dent;
EXPECT_TRUE(dit.NextEntry(&dent));
EXPECT_OK(dit.Status());
dent.Clear();
EXPECT_FALSE(dent.Valid());
EXPECT_FALSE(dent.IsDir());
EXPECT_FALSE(dent.IsRegularFile());
EXPECT_FALSE(dent.IsSymlink());
EXPECT_EQ(dent.Name(), std::string());
EXPECT_EQ(dent.RelPath(), std::string());
EXPECT_EQ(dent.RelPathName(), std::string());
}
} // namespace
} // namespace cdc_ft