blob: 6496f8084373a96d26c468054be6baed46349240 [file] [log] [blame]
#include <gtest/gtest.h>
#include "node-inl.h"
#include <algorithm>
#include <limits>
#include <memory>
#include <mutex>
using mediaprovider::fuse::dirhandle;
using mediaprovider::fuse::handle;
using mediaprovider::fuse::node;
using mediaprovider::fuse::NodeTracker;
// Listed as a friend class to struct node so it can observe implementation
// details if required. The only implementation detail that is worth writing
// tests around at the moment is the reference count.
class NodeTest : public ::testing::Test {
public:
NodeTest() : tracker_(NodeTracker(&lock_)) {}
uint32_t GetRefCount(node* node) { return node->refcount_; }
std::recursive_mutex lock_;
NodeTracker tracker_;
// Forward destruction here, as NodeTest is a friend class.
static void destroy(node* node) { delete node; }
static void acquire(node* node) { node->Acquire(); }
typedef std::unique_ptr<node, decltype(&NodeTest::destroy)> unique_node_ptr;
unique_node_ptr CreateNode(node* parent, const std::string& path, const int transforms = 0) {
return unique_node_ptr(
node::Create(parent, path, "", true, true, transforms, &lock_, &tracker_),
&NodeTest::destroy);
}
static class node* ForChild(class node* node, const std::string& name,
const std::function<bool(class node*)>& callback) {
return node->ForChild(name, callback);
}
// Expose NodeCompare for testing.
node::NodeCompare cmp;
};
TEST_F(NodeTest, TestCreate) {
unique_node_ptr node = CreateNode(nullptr, "/path");
ASSERT_EQ("/path", node->GetName());
ASSERT_EQ(1, GetRefCount(node.get()));
ASSERT_FALSE(node->HasCachedHandle());
}
TEST_F(NodeTest, TestCreate_withParent) {
unique_node_ptr parent = CreateNode(nullptr, "/path");
ASSERT_EQ(1, GetRefCount(parent.get()));
// Adding a child to a parent node increments its refcount.
unique_node_ptr child = CreateNode(parent.get(), "subdir");
ASSERT_EQ(2, GetRefCount(parent.get()));
// Make sure the node has been added to the parents list of children.
ASSERT_EQ(child.get(), parent->LookupChildByName("subdir", false /* acquire */));
ASSERT_EQ(1, GetRefCount(child.get()));
}
TEST_F(NodeTest, TestRelease) {
node* node = node::Create(nullptr, "/path", "", false, true, 0, &lock_, &tracker_);
acquire(node);
acquire(node);
ASSERT_EQ(3, GetRefCount(node));
ASSERT_FALSE(node->Release(1));
ASSERT_EQ(2, GetRefCount(node));
// A Release that makes refcount go negative should be a no-op.
ASSERT_FALSE(node->Release(10000));
ASSERT_EQ(2, GetRefCount(node));
// Finally, let the refcount go to zero.
ASSERT_TRUE(node->Release(2));
}
TEST_F(NodeTest, TestRenameName) {
unique_node_ptr parent = CreateNode(nullptr, "/path");
unique_node_ptr child = CreateNode(parent.get(), "subdir");
ASSERT_EQ(2, GetRefCount(parent.get()));
ASSERT_EQ(child.get(), parent->LookupChildByName("subdir", false /* acquire */));
child->Rename("subdir_new", parent.get());
ASSERT_EQ(2, GetRefCount(parent.get()));
ASSERT_EQ(nullptr, parent->LookupChildByName("subdir", false /* acquire */));
ASSERT_EQ(child.get(), parent->LookupChildByName("subdir_new", false /* acquire */));
ASSERT_EQ("/path/subdir_new", child->BuildPath());
ASSERT_EQ(1, GetRefCount(child.get()));
}
TEST_F(NodeTest, TestRenameParent) {
unique_node_ptr parent1 = CreateNode(nullptr, "/path1");
unique_node_ptr parent2 = CreateNode(nullptr, "/path2");
unique_node_ptr child = CreateNode(parent1.get(), "subdir");
ASSERT_EQ(2, GetRefCount(parent1.get()));
ASSERT_EQ(child.get(), parent1->LookupChildByName("subdir", false /* acquire */));
child->Rename("subdir", parent2.get());
ASSERT_EQ(1, GetRefCount(parent1.get()));
ASSERT_EQ(nullptr, parent1->LookupChildByName("subdir", false /* acquire */));
ASSERT_EQ(2, GetRefCount(parent2.get()));
ASSERT_EQ(child.get(), parent2->LookupChildByName("subdir", false /* acquire */));
ASSERT_EQ("/path2/subdir", child->BuildPath());
ASSERT_EQ(1, GetRefCount(child.get()));
}
TEST_F(NodeTest, TestRenameNameAndParent) {
unique_node_ptr parent1 = CreateNode(nullptr, "/path1");
unique_node_ptr parent2 = CreateNode(nullptr, "/path2");
unique_node_ptr child = CreateNode(parent1.get(), "subdir");
ASSERT_EQ(2, GetRefCount(parent1.get()));
ASSERT_EQ(child.get(), parent1->LookupChildByName("subdir", false /* acquire */));
child->Rename("subdir_new", parent2.get());
ASSERT_EQ(1, GetRefCount(parent1.get()));
ASSERT_EQ(nullptr, parent1->LookupChildByName("subdir", false /* acquire */));
ASSERT_EQ(nullptr, parent1->LookupChildByName("subdir_new", false /* acquire */));
ASSERT_EQ(2, GetRefCount(parent2.get()));
ASSERT_EQ(child.get(), parent2->LookupChildByName("subdir_new", false /* acquire */));
ASSERT_EQ("/path2/subdir_new", child->BuildPath());
ASSERT_EQ(1, GetRefCount(child.get()));
}
TEST_F(NodeTest, TestRenameNameForChild) {
unique_node_ptr parent = CreateNode(nullptr, "/path");
unique_node_ptr child0 = CreateNode(parent.get(), "subdir", 0 /* transforms */);
unique_node_ptr child1 = CreateNode(parent.get(), "subdir", 1 /* transforms */);
ASSERT_EQ(3, GetRefCount(parent.get()));
ASSERT_EQ(child0.get(),
parent->LookupChildByName("subdir", false /* acquire */, 0 /* transforms */));
ASSERT_EQ(child1.get(),
parent->LookupChildByName("subdir", false /* acquire */, 1 /* transforms */));
parent->RenameChild("subdir", "subdir_new", parent.get());
ASSERT_EQ(3, GetRefCount(parent.get()));
ASSERT_EQ(nullptr,
parent->LookupChildByName("subdir", false /* acquire */, 0 /* transforms */));
ASSERT_EQ(nullptr,
parent->LookupChildByName("subdir", false /* acquire */, 1 /* transforms */));
ASSERT_EQ(child0.get(),
parent->LookupChildByName("subdir_new", false /* acquire */, 0 /* transforms */));
ASSERT_EQ(child1.get(),
parent->LookupChildByName("subdir_new", false /* acquire */, 1 /* transforms */));
ASSERT_EQ("/path/subdir_new", child0->BuildPath());
ASSERT_EQ("/path/subdir_new", child1->BuildPath());
ASSERT_EQ(1, GetRefCount(child0.get()));
ASSERT_EQ(1, GetRefCount(child1.get()));
}
TEST_F(NodeTest, TestRenameParentForChild) {
unique_node_ptr parent1 = CreateNode(nullptr, "/path1");
unique_node_ptr parent2 = CreateNode(nullptr, "/path2");
unique_node_ptr child0 = CreateNode(parent1.get(), "subdir", 0 /* transforms */);
unique_node_ptr child1 = CreateNode(parent1.get(), "subdir", 1 /* transforms */);
ASSERT_EQ(3, GetRefCount(parent1.get()));
ASSERT_EQ(child0.get(),
parent1->LookupChildByName("subdir", false /* acquire */, 0 /* transforms */));
ASSERT_EQ(child1.get(),
parent1->LookupChildByName("subdir", false /* acquire */, 1 /* transforms */));
parent1->RenameChild("subdir", "subdir", parent2.get());
ASSERT_EQ(1, GetRefCount(parent1.get()));
ASSERT_EQ(nullptr,
parent1->LookupChildByName("subdir", false /* acquire */, 0 /* transforms */));
ASSERT_EQ(nullptr,
parent1->LookupChildByName("subdir", false /* acquire */, 1 /* transforms */));
ASSERT_EQ(3, GetRefCount(parent2.get()));
ASSERT_EQ(child0.get(),
parent2->LookupChildByName("subdir", false /* acquire */, 0 /* transforms */));
ASSERT_EQ(child1.get(),
parent2->LookupChildByName("subdir", false /* acquire */, 1 /* transforms */));
ASSERT_EQ("/path2/subdir", child0->BuildPath());
ASSERT_EQ("/path2/subdir", child1->BuildPath());
ASSERT_EQ(1, GetRefCount(child0.get()));
ASSERT_EQ(1, GetRefCount(child1.get()));
}
TEST_F(NodeTest, TestRenameNameAndParentForChild) {
unique_node_ptr parent1 = CreateNode(nullptr, "/path1");
unique_node_ptr parent2 = CreateNode(nullptr, "/path2");
unique_node_ptr child0 = CreateNode(parent1.get(), "subdir", 0 /* transforms */);
unique_node_ptr child1 = CreateNode(parent1.get(), "subdir", 1 /* transforms */);
ASSERT_EQ(3, GetRefCount(parent1.get()));
ASSERT_EQ(child0.get(),
parent1->LookupChildByName("subdir", false /* acquire */, 0 /* transforms */));
ASSERT_EQ(child1.get(),
parent1->LookupChildByName("subdir", false /* acquire */, 1 /* transforms */));
parent1->RenameChild("subdir", "subdir_new", parent2.get());
ASSERT_EQ(1, GetRefCount(parent1.get()));
ASSERT_EQ(nullptr,
parent1->LookupChildByName("subdir", false /* acquire */, 0 /* transforms */));
ASSERT_EQ(nullptr,
parent1->LookupChildByName("subdir_new", false /* acquire */, 0 /* transforms */));
ASSERT_EQ(nullptr,
parent1->LookupChildByName("subdir", false /* acquire */, 1 /* transforms */));
ASSERT_EQ(nullptr,
parent1->LookupChildByName("subdir_new", false /* acquire */, 1 /* transforms */));
ASSERT_EQ(3, GetRefCount(parent2.get()));
ASSERT_EQ(nullptr,
parent1->LookupChildByName("subdir_new", false /* acquire */, 0 /* transforms */));
ASSERT_EQ(nullptr,
parent1->LookupChildByName("subdir_new", false /* acquire */, 1 /* transforms */));
ASSERT_EQ("/path2/subdir_new", child0->BuildPath());
ASSERT_EQ("/path2/subdir_new", child1->BuildPath());
ASSERT_EQ(1, GetRefCount(child0.get()));
ASSERT_EQ(1, GetRefCount(child1.get()));
}
TEST_F(NodeTest, TestBuildPath) {
unique_node_ptr parent = CreateNode(nullptr, "/path");
ASSERT_EQ("/path", parent->BuildPath());
unique_node_ptr child = CreateNode(parent.get(), "subdir");
ASSERT_EQ("/path/subdir", child->BuildPath());
unique_node_ptr child2 = CreateNode(parent.get(), "subdir2");
ASSERT_EQ("/path/subdir2", child2->BuildPath());
unique_node_ptr subchild = CreateNode(child2.get(), "subsubdir");
ASSERT_EQ("/path/subdir2/subsubdir", subchild->BuildPath());
}
TEST_F(NodeTest, TestSetDeleted) {
unique_node_ptr parent = CreateNode(nullptr, "/path");
unique_node_ptr child = CreateNode(parent.get(), "subdir");
ASSERT_EQ(child.get(), parent->LookupChildByName("subdir", false /* acquire */));
child->SetDeleted();
ASSERT_EQ(nullptr, parent->LookupChildByName("subdir", false /* acquire */));
}
TEST_F(NodeTest, TestSetDeletedForChild) {
unique_node_ptr parent = CreateNode(nullptr, "/path");
unique_node_ptr child0 = CreateNode(parent.get(), "subdir", 0 /* transforms */);
unique_node_ptr child1 = CreateNode(parent.get(), "subdir", 1 /* transforms */);
ASSERT_EQ(child0.get(),
parent->LookupChildByName("subdir", false /* acquire */, 0 /* transforms */));
ASSERT_EQ(child1.get(),
parent->LookupChildByName("subdir", false /* acquire */, 1 /* transforms */));
parent->SetDeletedForChild("subdir");
ASSERT_EQ(nullptr,
parent->LookupChildByName("subdir", false /* acquire */, 0 /* transforms */));
ASSERT_EQ(nullptr,
parent->LookupChildByName("subdir", false /* acquire */, 1 /* transforms */));
}
TEST_F(NodeTest, DeleteTree) {
unique_node_ptr parent = CreateNode(nullptr, "/path");
// This is the tree that we intend to delete.
node* child = node::Create(parent.get(), "subdir", "", false, true, 0, &lock_, &tracker_);
node::Create(child, "s1", "", false, true, 0, &lock_, &tracker_);
node* subchild2 = node::Create(child, "s2", "", false, true, 0, &lock_, &tracker_);
node::Create(subchild2, "sc2", "", false, true, 0, &lock_, &tracker_);
ASSERT_EQ(child, parent->LookupChildByName("subdir", false /* acquire */));
node::DeleteTree(child);
ASSERT_EQ(nullptr, parent->LookupChildByName("subdir", false /* acquire */));
}
TEST_F(NodeTest, LookupChildByName_empty) {
unique_node_ptr parent = CreateNode(nullptr, "/path");
unique_node_ptr child = CreateNode(parent.get(), "subdir");
ASSERT_EQ(child.get(), parent->LookupChildByName("subdir", false /* acquire */));
ASSERT_EQ(nullptr, parent->LookupChildByName("", false /* acquire */));
}
TEST_F(NodeTest, LookupChildByName_transforms) {
unique_node_ptr parent = CreateNode(nullptr, "/path");
unique_node_ptr child0 = CreateNode(parent.get(), "subdir", 0 /* transforms */);
unique_node_ptr child1 = CreateNode(parent.get(), "subdir", 1 /* transforms */);
ASSERT_EQ(child0.get(), parent->LookupChildByName("subdir", false /* acquire */));
ASSERT_EQ(child0.get(),
parent->LookupChildByName("subdir", false /* acquire */, 0 /* transforms */));
ASSERT_EQ(child1.get(),
parent->LookupChildByName("subdir", false /* acquire */, 1 /* transforms */));
ASSERT_EQ(nullptr,
parent->LookupChildByName("subdir", false /* acquire */, 2 /* transforms */));
}
TEST_F(NodeTest, LookupChildByName_refcounts) {
unique_node_ptr parent = CreateNode(nullptr, "/path");
unique_node_ptr child = CreateNode(parent.get(), "subdir");
ASSERT_EQ(child.get(), parent->LookupChildByName("subdir", false /* acquire */));
ASSERT_EQ(1, GetRefCount(child.get()));
ASSERT_EQ(child.get(), parent->LookupChildByName("subdir", true /* acquire */));
ASSERT_EQ(2, GetRefCount(child.get()));
}
TEST_F(NodeTest, LookupAbsolutePath) {
unique_node_ptr parent = CreateNode(nullptr, "/path");
unique_node_ptr child = CreateNode(parent.get(), "subdir");
unique_node_ptr child2 = CreateNode(parent.get(), "subdir2");
unique_node_ptr subchild = CreateNode(child2.get(), "subsubdir");
ASSERT_EQ(parent.get(), node::LookupAbsolutePath(parent.get(), "/path"));
ASSERT_EQ(parent.get(), node::LookupAbsolutePath(parent.get(), "/path/"));
ASSERT_EQ(nullptr, node::LookupAbsolutePath(parent.get(), "/path2"));
ASSERT_EQ(child.get(), node::LookupAbsolutePath(parent.get(), "/path/subdir"));
ASSERT_EQ(child.get(), node::LookupAbsolutePath(parent.get(), "/path/subdir/"));
// TODO(narayan): Are the two cases below intentional behaviour ?
ASSERT_EQ(child.get(), node::LookupAbsolutePath(parent.get(), "/path//subdir"));
ASSERT_EQ(child.get(), node::LookupAbsolutePath(parent.get(), "/path///subdir"));
ASSERT_EQ(child2.get(), node::LookupAbsolutePath(parent.get(), "/path/subdir2"));
ASSERT_EQ(child2.get(), node::LookupAbsolutePath(parent.get(), "/path/subdir2/"));
ASSERT_EQ(nullptr, node::LookupAbsolutePath(parent.get(), "/path/subdir3/"));
ASSERT_EQ(subchild.get(), node::LookupAbsolutePath(parent.get(), "/path/subdir2/subsubdir"));
ASSERT_EQ(nullptr, node::LookupAbsolutePath(parent.get(), "/path/subdir/subsubdir"));
}
TEST_F(NodeTest, AddDestroyHandle) {
unique_node_ptr node = CreateNode(nullptr, "/path");
handle* h = new handle(-1, new mediaprovider::fuse::RedactionInfo, true /* cached */,
false /* passthrough */, 0 /* uid */);
node->AddHandle(h);
ASSERT_TRUE(node->HasCachedHandle());
node->DestroyHandle(h);
ASSERT_FALSE(node->HasCachedHandle());
// Should all crash the process as the handle is no longer associated with
// the node in question.
EXPECT_DEATH(node->DestroyHandle(h), "");
EXPECT_DEATH(node->DestroyHandle(nullptr), "");
std::unique_ptr<handle> h2(new handle(-1, new mediaprovider::fuse::RedactionInfo,
true /* cached */, false /* passthrough */, 0 /* uid */));
EXPECT_DEATH(node->DestroyHandle(h2.get()), "");
}
TEST_F(NodeTest, CaseInsensitive) {
unique_node_ptr parent = CreateNode(nullptr, "/path");
unique_node_ptr mixed_child = CreateNode(parent.get(), "cHiLd");
node* upper_child = parent->LookupChildByName("CHILD", false /* acquire */);
node* lower_child = parent->LookupChildByName("child", false /* acquire */);
ASSERT_EQ(mixed_child.get(), lower_child);
ASSERT_EQ(mixed_child.get(), upper_child);
}
TEST_F(NodeTest, RenameSameNameSameParent) {
unique_node_ptr parent = CreateNode(nullptr, "/path1");
unique_node_ptr child = CreateNode(parent.get(), "subdir");
ASSERT_EQ(child.get(), parent->LookupChildByName("SuBdIr", false /* acquire */));
ASSERT_EQ(2, GetRefCount(parent.get()));
child->Rename("subdir", parent.get());
ASSERT_EQ(child.get(), parent->LookupChildByName("SuBdIr", false /* acquire */));
ASSERT_EQ(2, GetRefCount(parent.get()));
}
TEST_F(NodeTest, RenameRoot) {
unique_node_ptr root = CreateNode(nullptr, "/root");
ASSERT_EQ(1, GetRefCount(root.get()));
root->Rename("/i-am-root!", nullptr);
ASSERT_EQ("/i-am-root!", root->GetName());
ASSERT_EQ(1, GetRefCount(root.get()));
}
TEST_F(NodeTest, NodeCompareDefinesLinearOrder) {
unique_node_ptr node_a = CreateNode(nullptr, "a");
unique_node_ptr node_b = CreateNode(nullptr, "B");
unique_node_ptr node_c = CreateNode(nullptr, "c");
ASSERT_FALSE(cmp.operator()(node_a.get(), node_a.get()));
ASSERT_FALSE(cmp.operator()(node_b.get(), node_b.get()));
ASSERT_FALSE(cmp.operator()(node_c.get(), node_c.get()));
auto check_fn = [&](const node* lhs_node, const node* rhs_node) {
ASSERT_TRUE(cmp.operator()(lhs_node, rhs_node));
ASSERT_FALSE(cmp.operator()(rhs_node, lhs_node));
};
check_fn(node_a.get(), node_b.get());
check_fn(node_b.get(), node_c.get());
check_fn(node_a.get(), node_c.get());
// ("A", 0) < node_a < ("a", max_uintptr_t) < node_b
ASSERT_TRUE(cmp.operator()(std::make_pair("A", 0), node_a.get()));
ASSERT_TRUE(cmp.operator()(node_a.get(),
std::make_pair("A", std::numeric_limits<uintptr_t>::max())));
ASSERT_TRUE(cmp.operator()(std::make_pair("A", std::numeric_limits<uintptr_t>::max()),
node_b.get()));
}
TEST_F(NodeTest, LookupChildByName_ChildrenWithSameName) {
unique_node_ptr parent = CreateNode(nullptr, "/path");
unique_node_ptr foo1 = CreateNode(parent.get(), "FoO");
unique_node_ptr foo2 = CreateNode(parent.get(), "fOo");
unique_node_ptr bar1 = CreateNode(parent.get(), "BAR");
unique_node_ptr bar2 = CreateNode(parent.get(), "bar");
unique_node_ptr baz1 = CreateNode(parent.get(), "baZ");
unique_node_ptr baz2 = CreateNode(parent.get(), "Baz");
auto test_fn = [&](const std::string& name, node* first, node* second) {
auto node1 = parent->LookupChildByName(name, false /* acquire */);
ASSERT_EQ(std::min(first, second), node1);
node1->SetDeleted();
auto node2 = parent->LookupChildByName(name, false /* acquire */);
ASSERT_EQ(std::max(first, second), node2);
node2->SetDeleted();
ASSERT_EQ(nullptr, parent->LookupChildByName(name, false /* acquire */));
};
test_fn("foo", foo1.get(), foo2.get());
test_fn("bAr", bar1.get(), bar2.get());
test_fn("BaZ", baz1.get(), baz2.get());
}
TEST_F(NodeTest, ForChild) {
unique_node_ptr parent = CreateNode(nullptr, "/path");
unique_node_ptr foo1 = CreateNode(parent.get(), "FoO");
unique_node_ptr foo2 = CreateNode(parent.get(), "fOo");
unique_node_ptr foo3 = CreateNode(parent.get(), "foo");
foo3->SetDeleted();
std::vector<node*> match_all;
auto test_fn_match_all = [&](node* child) {
match_all.push_back(child);
return false;
};
std::vector<node*> match_first;
auto test_fn_match_first = [&](node* child) {
match_first.push_back(child);
return true;
};
std::vector<node*> match_none;
auto test_fn_match_none = [&](node* child) {
match_none.push_back(child);
return false;
};
node* node_all = ForChild(parent.get(), "foo", test_fn_match_all);
ASSERT_EQ(nullptr, node_all);
ASSERT_EQ(2, match_all.size());
ASSERT_EQ(std::min(foo1.get(), foo2.get()), match_all[0]);
ASSERT_EQ(std::max(foo1.get(), foo2.get()), match_all[1]);
node* node_first = ForChild(parent.get(), "foo", test_fn_match_first);
ASSERT_EQ(std::min(foo1.get(), foo2.get()), node_first);
ASSERT_EQ(1, match_first.size());
ASSERT_EQ(std::min(foo1.get(), foo2.get()), match_first[0]);
node* node_none = ForChild(parent.get(), "bar", test_fn_match_none);
ASSERT_EQ(nullptr, node_none);
ASSERT_TRUE(match_none.empty());
}