blob: 1d9ad4d9a7bbd1c2c8a0f3d2ad765ad3814484d7 [file] [log] [blame]
// Copyright 2018 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 "third_party/blink/renderer/modules/storage/storage_controller.h"
#include "base/run_loop.h"
#include "base/task/post_task.h"
#include "base/test/scoped_feature_list.h"
#include "mojo/public/cpp/bindings/strong_binding.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "third_party/blink/public/common/features.h"
#include "third_party/blink/public/platform/scheduler/test/renderer_scheduler_test_support.h"
#include "third_party/blink/renderer/modules/storage/storage_namespace.h"
#include "third_party/blink/renderer/modules/storage/testing/fake_area_source.h"
#include "third_party/blink/renderer/modules/storage/testing/mock_storage_area.h"
#include "third_party/blink/renderer/platform/cross_thread_functional.h"
#include "third_party/blink/renderer/platform/scheduler/public/post_cross_thread_task.h"
#include "third_party/blink/renderer/platform/uuid.h"
#include "third_party/blink/renderer/platform/wtf/functional.h"
namespace blink {
namespace {
const size_t kTestCacheLimit = 100;
class MockStoragePartitionService
: public mojom::blink::StoragePartitionService {
public:
void OpenLocalStorage(const scoped_refptr<const SecurityOrigin>& origin,
mojom::blink::StorageAreaRequest request) override {}
void OpenSessionStorage(
const String& namespace_id,
mojom::blink::SessionStorageNamespaceRequest request) override {
session_storage_opens++;
}
void GetSessionStorageUsage(int32_t* out) const {
*out = session_storage_opens;
}
int32_t session_storage_opens = 0;
};
} // namespace
TEST(StorageControllerTest, CacheLimit) {
base::test::ScopedFeatureList features;
features.InitAndEnableFeature(features::kOnionSoupDOMStorage);
const auto kOrigin = SecurityOrigin::CreateFromString("http://dom_storage1/");
const auto kOrigin2 =
SecurityOrigin::CreateFromString("http://dom_storage2/");
const auto kOrigin3 =
SecurityOrigin::CreateFromString("http://dom_storage3/");
const String kKey("key");
const String kValue("value");
const KURL kPageUrl("http://dom_storage/page");
Persistent<FakeAreaSource> source_area =
MakeGarbageCollected<FakeAreaSource>(kPageUrl);
mojom::blink::StoragePartitionServicePtr storage_partition_service_ptr;
PostCrossThreadTask(
*base::CreateSequencedTaskRunnerWithTraits({}), FROM_HERE,
CrossThreadBind(
[](mojom::blink::StoragePartitionServiceRequest request) {
mojo::MakeStrongBinding(
std::make_unique<MockStoragePartitionService>(),
std::move(request));
},
WTF::Passed(MakeRequest(&storage_partition_service_ptr))));
StorageController controller(scheduler::GetSingleThreadTaskRunnerForTesting(),
std::move(storage_partition_service_ptr),
kTestCacheLimit);
auto cached_area1 = controller.GetLocalStorageArea(kOrigin.get());
cached_area1->RegisterSource(source_area);
cached_area1->SetItem(kKey, kValue, source_area);
const auto* area1_ptr = cached_area1.get();
size_t expected_total = (kKey.length() + kValue.length()) * 2;
EXPECT_EQ(expected_total, cached_area1->memory_used());
EXPECT_EQ(expected_total, controller.TotalCacheSize());
cached_area1 = nullptr;
auto cached_area2 = controller.GetLocalStorageArea(kOrigin2.get());
cached_area2->RegisterSource(source_area);
cached_area2->SetItem(kKey, kValue, source_area);
// Area for kOrigin should still be alive.
EXPECT_EQ(2 * cached_area2->memory_used(), controller.TotalCacheSize());
EXPECT_EQ(area1_ptr, controller.GetLocalStorageArea(kOrigin.get()));
String long_value(Vector<UChar>(kTestCacheLimit, 'a'));
cached_area2->SetItem(kKey, long_value, source_area);
// Cache is cleared when a new area is opened.
auto cached_area3 = controller.GetLocalStorageArea(kOrigin3.get());
EXPECT_EQ(cached_area2->memory_used(), controller.TotalCacheSize());
}
TEST(StorageControllerTest, CacheLimitSessionStorage) {
base::test::ScopedFeatureList features;
features.InitAndEnableFeature(features::kOnionSoupDOMStorage);
const String kNamespace1 = CreateCanonicalUUIDString();
const String kNamespace2 = CreateCanonicalUUIDString();
const auto kOrigin = SecurityOrigin::CreateFromString("http://dom_storage1/");
const auto kOrigin2 =
SecurityOrigin::CreateFromString("http://dom_storage2/");
const auto kOrigin3 =
SecurityOrigin::CreateFromString("http://dom_storage3/");
const String kKey("key");
const String kValue("value");
const KURL kPageUrl("http://dom_storage/page");
Persistent<FakeAreaSource> source_area =
MakeGarbageCollected<FakeAreaSource>(kPageUrl);
auto task_runner = base::CreateSequencedTaskRunnerWithTraits({});
auto mock_storage_partition_service =
std::make_unique<MockStoragePartitionService>();
MockStoragePartitionService* storage_partition_ptr =
mock_storage_partition_service.get();
mojom::blink::StoragePartitionServicePtr storage_partition_service_ptr;
PostCrossThreadTask(
*task_runner, FROM_HERE,
CrossThreadBind(
[](std::unique_ptr<MockStoragePartitionService> storage_partition_ptr,
mojom::blink::StoragePartitionServiceRequest request) {
mojo::MakeStrongBinding(std::move(storage_partition_ptr),
std::move(request));
},
WTF::Passed(std::move(mock_storage_partition_service)),
WTF::Passed(MakeRequest(&storage_partition_service_ptr))));
StorageController controller(
nullptr, std::move(storage_partition_service_ptr), kTestCacheLimit);
StorageNamespace* ns1 = controller.CreateSessionStorageNamespace(kNamespace1);
StorageNamespace* ns2 = controller.CreateSessionStorageNamespace(kNamespace2);
auto cached_area1 = ns1->GetCachedArea(kOrigin.get());
cached_area1->RegisterSource(source_area);
cached_area1->SetItem(kKey, kValue, source_area);
const auto* area1_ptr = cached_area1.get();
size_t expected_total = (kKey.length() + kValue.length()) * 2;
EXPECT_EQ(expected_total, cached_area1->memory_used());
EXPECT_EQ(expected_total, controller.TotalCacheSize());
cached_area1 = nullptr;
auto cached_area2 = ns2->GetCachedArea(kOrigin2.get());
cached_area2->RegisterSource(source_area);
cached_area2->SetItem(kKey, kValue, source_area);
// Area for kOrigin should still be alive.
EXPECT_EQ(2 * cached_area2->memory_used(), controller.TotalCacheSize());
EXPECT_EQ(area1_ptr, ns1->GetCachedArea(kOrigin.get()));
String long_value(Vector<UChar>(kTestCacheLimit, 'a'));
cached_area2->SetItem(kKey, long_value, source_area);
// Cache is cleared when a new area is opened.
auto cached_area3 = ns1->GetCachedArea(kOrigin3.get());
EXPECT_EQ(cached_area2->memory_used(), controller.TotalCacheSize());
int32_t opens = 0;
{
base::RunLoop loop;
task_runner->PostTaskAndReply(
FROM_HERE,
base::BindOnce(&MockStoragePartitionService::GetSessionStorageUsage,
base::Unretained(storage_partition_ptr), &opens),
loop.QuitClosure());
loop.Run();
}
EXPECT_EQ(opens, 2);
}
} // namespace blink