blob: e1e462baef64bb94bbfcee19c1e086e8af64eb23 [file] [log] [blame]
// Copyright 2019 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/performance_manager/public/graph/graph_operations.h"
#include <algorithm>
#include "base/bind.h"
#include "chrome/browser/performance_manager/test_support/graph_test_harness.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace performance_manager {
namespace {
class GraphOperationsTest : public GraphTestHarness {
public:
// Sets up two parallel frame trees that span multiple processes each.
void SetUp() override {
process1_ = CreateNode<ProcessNodeImpl>();
process2_ = CreateNode<ProcessNodeImpl>();
page1_ = CreateNode<PageNodeImpl>();
page2_ = CreateNode<PageNodeImpl>();
mainframe1_ =
CreateFrameNodeAutoId(process1_.get(), page1_.get(), nullptr, 0);
mainframe2_ =
CreateFrameNodeAutoId(process2_.get(), page2_.get(), nullptr, 1);
childframe1a_ = CreateFrameNodeAutoId(process2_.get(), page1_.get(),
mainframe1_.get(), 2);
childframe1b_ = CreateFrameNodeAutoId(process2_.get(), page1_.get(),
mainframe1_.get(), 3);
childframe2a_ = CreateFrameNodeAutoId(process1_.get(), page2_.get(),
mainframe2_.get(), 4);
childframe2b_ = CreateFrameNodeAutoId(process1_.get(), page2_.get(),
mainframe2_.get(), 5);
}
TestNodeWrapper<ProcessNodeImpl> process1_;
TestNodeWrapper<ProcessNodeImpl> process2_;
TestNodeWrapper<PageNodeImpl> page1_;
TestNodeWrapper<PageNodeImpl> page2_;
// Root nodes. |mainframeX_| is in |processX_|.
TestNodeWrapper<FrameNodeImpl> mainframe1_;
TestNodeWrapper<FrameNodeImpl> mainframe2_;
// Children of |mainframe1_|, but in |process2_|.
TestNodeWrapper<FrameNodeImpl> childframe1a_;
TestNodeWrapper<FrameNodeImpl> childframe1b_;
// Children of |mainframe2_|, but in |process1_|.
TestNodeWrapper<FrameNodeImpl> childframe2a_;
TestNodeWrapper<FrameNodeImpl> childframe2b_;
};
const PageNode* ToPublic(PageNodeImpl* page_node) {
return page_node;
}
const FrameNode* ToPublic(FrameNodeImpl* frame_node) {
return frame_node;
}
} // namespace
TEST_F(GraphOperationsTest, GetAssociatedPageNodes) {
auto page_nodes = GraphOperations::GetAssociatedPageNodes(process1_.get());
EXPECT_EQ(2u, page_nodes.size());
EXPECT_THAT(page_nodes, testing::UnorderedElementsAre(
ToPublic(page1_.get()), ToPublic(page2_.get())));
}
TEST_F(GraphOperationsTest, GetAssociatedProcessNodes) {
auto process_nodes = GraphOperations::GetAssociatedProcessNodes(page1_.get());
EXPECT_EQ(2u, process_nodes.size());
EXPECT_THAT(process_nodes,
testing::UnorderedElementsAre(process1_.get(), process2_.get()));
}
TEST_F(GraphOperationsTest, GetFrameNodes) {
// Add a grandchild frame.
auto grandchild = CreateFrameNodeAutoId(process1_.get(), page1_.get(),
childframe1a_.get(), 6);
auto frame_nodes = GraphOperations::GetFrameNodes(page1_.get());
EXPECT_THAT(frame_nodes,
testing::UnorderedElementsAre(
ToPublic(mainframe1_.get()), ToPublic(childframe1a_.get()),
ToPublic(childframe1b_.get()), ToPublic(grandchild.get())));
// In a level order the main-frame is first, and the grandchild is last. The
// two children can come in any order.
EXPECT_EQ(ToPublic(mainframe1_.get()), frame_nodes[0]);
EXPECT_EQ(ToPublic(grandchild.get()), frame_nodes[3]);
}
TEST_F(GraphOperationsTest, VisitFrameTree) {
auto frame_nodes = GraphOperations::GetFrameNodes(page1_.get());
std::vector<const FrameNode*> visited;
GraphOperations::VisitFrameTreePreOrder(
page1_.get(), base::Bind(
[](std::vector<const FrameNode*>* visited,
const FrameNode* frame_node) -> bool {
visited->push_back(frame_node);
return true;
},
base::Unretained(&visited)));
EXPECT_THAT(visited,
testing::UnorderedElementsAre(ToPublic(mainframe1_.get()),
ToPublic(childframe1a_.get()),
ToPublic(childframe1b_.get())));
// In pre-order the main frame is first.
EXPECT_EQ(ToPublic(mainframe1_.get()), visited[0]);
visited.clear();
GraphOperations::VisitFrameTreePostOrder(
page1_.get(), base::Bind(
[](std::vector<const FrameNode*>* visited,
const FrameNode* frame_node) -> bool {
visited->push_back(frame_node);
return true;
},
base::Unretained(&visited)));
EXPECT_THAT(visited,
testing::UnorderedElementsAre(ToPublic(mainframe1_.get()),
ToPublic(childframe1a_.get()),
ToPublic(childframe1b_.get())));
// In post-order the main frame is last.
EXPECT_EQ(mainframe1_.get(), visited[2]);
}
TEST_F(GraphOperationsTest, HasFrame) {
EXPECT_TRUE(GraphOperations::HasFrame(page1_.get(), childframe1a_.get()));
EXPECT_FALSE(GraphOperations::HasFrame(page1_.get(), childframe2a_.get()));
}
} // namespace performance_manager