blob: bcd5353ac87418916956648ca47492cfaf604f83 [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/graph/graph_impl_operations.h"
#include <algorithm>
#include "base/bind.h"
#include "base/stl_util.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 GraphImplOperationsTest : 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_;
};
} // namespace
TEST_F(GraphImplOperationsTest, GetAssociatedPageNodes) {
auto page_nodes =
GraphImplOperations::GetAssociatedPageNodes(process1_.get());
EXPECT_EQ(2u, page_nodes.size());
EXPECT_THAT(page_nodes,
testing::UnorderedElementsAre(page1_.get(), page2_.get()));
}
TEST_F(GraphImplOperationsTest, GetAssociatedProcessNodes) {
auto process_nodes =
GraphImplOperations::GetAssociatedProcessNodes(page1_.get());
EXPECT_EQ(2u, process_nodes.size());
EXPECT_THAT(process_nodes,
testing::UnorderedElementsAre(process1_.get(), process2_.get()));
}
TEST_F(GraphImplOperationsTest, GetFrameNodes) {
// Add a grandchild frame.
auto grandchild = CreateFrameNodeAutoId(process1_.get(), page1_.get(),
childframe1a_.get(), 6);
auto frame_nodes = GraphImplOperations::GetFrameNodes(page1_.get());
EXPECT_THAT(frame_nodes, testing::UnorderedElementsAre(
mainframe1_.get(), childframe1a_.get(),
childframe1b_.get(), 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(mainframe1_.get(), frame_nodes[0]);
EXPECT_EQ(grandchild.get(), frame_nodes[3]);
}
TEST_F(GraphImplOperationsTest, VisitFrameTree) {
auto frame_nodes = GraphImplOperations::GetFrameNodes(page1_.get());
std::vector<FrameNodeImpl*> visited;
GraphImplOperations::VisitFrameTreePreOrder(
page1_.get(), [&visited](FrameNodeImpl* frame_node) -> bool {
visited.push_back(frame_node);
return true;
});
EXPECT_THAT(visited,
testing::UnorderedElementsAre(
mainframe1_.get(), childframe1a_.get(), childframe1b_.get()));
// In pre-order the main frame is first.
EXPECT_EQ(mainframe1_.get(), visited[0]);
visited.clear();
GraphImplOperations::VisitFrameTreePostOrder(
page1_.get(), [&visited](FrameNodeImpl* frame_node) -> bool {
visited.push_back(frame_node);
return true;
});
EXPECT_THAT(visited,
testing::UnorderedElementsAre(
mainframe1_.get(), childframe1a_.get(), childframe1b_.get()));
// In post-order the main frame is last.
EXPECT_EQ(mainframe1_.get(), visited[2]);
}
TEST_F(GraphImplOperationsTest, HasFrame) {
EXPECT_TRUE(GraphImplOperations::HasFrame(page1_.get(), childframe1a_.get()));
EXPECT_FALSE(
GraphImplOperations::HasFrame(page1_.get(), childframe2a_.get()));
}
} // namespace performance_manager