blob: 3678cb4ccbbc558bd48389509d4e90e213c23b1b [file] [log] [blame]
// Copyright 2022 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "content/browser/renderer_host/indexed_db_client_state_checker_factory.h"
#include "base/functional/bind.h"
#include "base/functional/callback_forward.h"
#include "base/test/scoped_feature_list.h"
#include "content/browser/renderer_host/render_frame_host_impl.h"
#include "content/public/browser/global_routing_id.h"
#include "content/public/browser/render_frame_host.h"
#include "content/public/common/content_features.h"
#include "content/public/test/navigation_simulator.h"
#include "content/public/test/prerender_test_util.h"
#include "content/public/test/test_renderer_host.h"
#include "content/public/test/web_contents_tester.h"
#include "mojo/public/cpp/bindings/pending_receiver.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace content {
class IndexedDBClientStateCheckerFactoryTest
: public RenderViewHostTestHarness {
public:
IndexedDBClientStateCheckerFactoryTest() {
// Set up the features so that we can test the behaviour of the
// `DocumentIndexedDBClientStateChecker` with document in different states
// such as prerendering or back/forward cache.
scoped_feature_list_.InitWithFeatures(
{blink::features::kPrerender2, features::kBackForwardCache},
{blink::features::kPrerender2MemoryControls});
}
IndexedDBClientStateCheckerFactoryTest(
const IndexedDBClientStateCheckerFactoryTest&) = delete;
IndexedDBClientStateCheckerFactoryTest& operator=(
const IndexedDBClientStateCheckerFactoryTest&) = delete;
~IndexedDBClientStateCheckerFactoryTest() override = default;
// A helper function used for testing the `RequireClientToBeActive` method
// with callback.
void TestRequireClientToBeActive(
storage::mojom::IndexedDBClientStateChecker* checker,
storage::mojom::DisallowClientActivationReason reason,
bool expected_was_active) {
mojo::Remote<storage::mojom::IndexedDBClientKeepActive> test_remote_;
base::RunLoop run_loop;
base::OnceClosure quit_closure = run_loop.QuitClosure();
checker->RequireClientToBeActiveAndKeepActive(
reason, test_remote_.BindNewPipeAndPassReceiver(),
base::BindOnce(
[](base::OnceClosure closure, bool expected_was_active,
bool was_active) {
EXPECT_EQ(was_active, expected_was_active);
std::move(closure).Run();
},
std::move(quit_closure), expected_was_active));
run_loop.Run();
}
private:
base::test::ScopedFeatureList scoped_feature_list_;
};
TEST_F(IndexedDBClientStateCheckerFactoryTest,
RequireClientToBeActive_DocumentBFCache) {
RenderFrameHostImpl* rfh = static_cast<RenderFrameHostImpl*>(main_rfh());
ASSERT_TRUE(rfh);
storage::mojom::IndexedDBClientStateChecker* checker =
IndexedDBClientStateCheckerFactory::
GetOrCreateIndexedDBClientStateCheckerForTesting(rfh->GetGlobalId());
// The document is initially in active state.
EXPECT_EQ(rfh->GetLifecycleState(), RenderFrameHost::LifecycleState::kActive);
// For the active document, the client state checker should claim that it's
// active.
TestRequireClientToBeActive(
checker,
storage::mojom::DisallowClientActivationReason::kClientEventIsTriggered,
/*expected_was_active=*/true);
// There is no side effect to the active document.
EXPECT_EQ(rfh->GetLifecycleState(),
RenderFrameHostImpl::LifecycleState::kActive);
// Set the lifecycle state to `kInBackForwardCache`.
rfh->SetLifecycleState(
RenderFrameHostImpl::LifecycleStateImpl::kInBackForwardCache);
// Now the client state check should report that the document is not active.
TestRequireClientToBeActive(
checker,
storage::mojom::DisallowClientActivationReason::kClientEventIsTriggered,
/*expected_was_active=*/false);
// The page will be evicted from back/forward cache as the side effect.
EXPECT_TRUE(rfh->is_evicted_from_back_forward_cache());
}
TEST_F(IndexedDBClientStateCheckerFactoryTest,
RequireClientToBeActive_DocumentPrerendering) {
// Set up a `RenderFrameHost` that's in `kPrerendering`.
test::ScopedPrerenderWebContentsDelegate web_contents_delegate(
*web_contents());
GURL url = GURL("https://example.com");
NavigationSimulator::NavigateAndCommitFromBrowser(web_contents(), url);
RenderFrameHost* rfh = WebContentsTester::For(web_contents())
->AddPrerenderAndCommitNavigation(GURL(url));
EXPECT_EQ(rfh->GetLifecycleState(),
RenderFrameHost::LifecycleState::kPrerendering);
storage::mojom::IndexedDBClientStateChecker* checker =
IndexedDBClientStateCheckerFactory::
GetOrCreateIndexedDBClientStateCheckerForTesting(rfh->GetGlobalId());
// For prerendering case, the client state checker should claim it as
// active, since IndexedDB is supported in prerendering page.
TestRequireClientToBeActive(
checker,
storage::mojom::DisallowClientActivationReason::kClientEventIsTriggered,
/*expected_was_active=*/true);
EXPECT_EQ(rfh->GetLifecycleState(),
RenderFrameHost::LifecycleState::kPrerendering);
}
} // namespace content