| // Copyright 2021 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include <memory> |
| |
| #include "base/barrier_closure.h" |
| #include "base/files/scoped_temp_dir.h" |
| #include "base/run_loop.h" |
| #include "base/task/sequenced_task_runner.h" |
| #include "base/task/single_thread_task_runner.h" |
| #include "base/test/bind.h" |
| #include "base/test/gmock_callback_support.h" |
| #include "base/test/task_environment.h" |
| #include "base/threading/thread.h" |
| #include "base/time/default_clock.h" |
| #include "components/services/storage/public/cpp/buckets/bucket_info.h" |
| #include "components/services/storage/public/cpp/buckets/bucket_locator.h" |
| #include "components/services/storage/public/cpp/buckets/constants.h" |
| #include "components/services/storage/public/cpp/quota_error_or.h" |
| #include "content/browser/indexed_db/indexed_db_context_impl.h" |
| #include "content/browser/indexed_db/indexed_db_factory.h" |
| #include "content/browser/indexed_db/mock_indexed_db_callbacks.h" |
| #include "content/browser/indexed_db/mock_mojo_indexed_db_callbacks.h" |
| #include "content/browser/indexed_db/mock_mojo_indexed_db_database_callbacks.h" |
| #include "mojo/public/cpp/bindings/associated_receiver.h" |
| #include "mojo/public/cpp/bindings/remote.h" |
| #include "storage/browser/test/mock_quota_manager.h" |
| #include "storage/browser/test/mock_quota_manager_proxy.h" |
| #include "storage/browser/test/mock_special_storage_policy.h" |
| #include "storage/browser/test/quota_manager_proxy_sync.h" |
| #include "testing/gmock/include/gmock/gmock.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| #include "third_party/blink/public/common/storage_key/storage_key.h" |
| |
| namespace content { |
| |
| class IndexedDBContextTest : public testing::Test { |
| public: |
| class MockIndexedDBClientStateChecker |
| : public storage::mojom::IndexedDBClientStateChecker { |
| public: |
| MockIndexedDBClientStateChecker() = default; |
| ~MockIndexedDBClientStateChecker() override = default; |
| |
| // storage::mojom::IndexedDBClientStateChecker overrides |
| void DisallowInactiveClient( |
| storage::mojom::DisallowInactiveClientReason reason, |
| mojo::PendingReceiver<storage::mojom::IndexedDBClientKeepActive> |
| keep_active, |
| storage::mojom::IndexedDBClientStateChecker:: |
| DisallowInactiveClientCallback callback) override {} |
| }; |
| |
| IndexedDBContextTest() |
| : special_storage_policy_( |
| base::MakeRefCounted<storage::MockSpecialStoragePolicy>()) {} |
| ~IndexedDBContextTest() override = default; |
| |
| void SetUp() override { |
| ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); |
| quota_manager_ = base::MakeRefCounted<storage::MockQuotaManager>( |
| /*is_incognito=*/false, temp_dir_.GetPath(), |
| base::SingleThreadTaskRunner::GetCurrentDefault(), |
| special_storage_policy_); |
| quota_manager_proxy_ = base::MakeRefCounted<storage::MockQuotaManagerProxy>( |
| quota_manager_.get(), |
| base::SingleThreadTaskRunner::GetCurrentDefault()); |
| indexed_db_context_ = base::MakeRefCounted<IndexedDBContextImpl>( |
| temp_dir_.GetPath(), quota_manager_proxy_, |
| base::DefaultClock::GetInstance(), |
| /*blob_storage_context=*/mojo::NullRemote(), |
| /*file_system_access_context=*/mojo::NullRemote(), |
| base::SequencedTaskRunner::GetCurrentDefault(), |
| base::SequencedTaskRunner::GetCurrentDefault()); |
| } |
| |
| protected: |
| scoped_refptr<storage::MockSpecialStoragePolicy> special_storage_policy_; |
| |
| base::ScopedTempDir temp_dir_; |
| |
| // These tests need a full TaskEnvironment because IndexedDBContextImpl |
| // uses the thread pool for querying QuotaDatabase. |
| base::test::TaskEnvironment task_environment_; |
| |
| scoped_refptr<IndexedDBContextImpl> indexed_db_context_; |
| scoped_refptr<storage::MockQuotaManager> quota_manager_; |
| scoped_refptr<storage::QuotaManagerProxy> quota_manager_proxy_; |
| |
| MockIndexedDBClientStateChecker example_checker; |
| |
| const blink::StorageKey example_storage_key_ = |
| blink::StorageKey::CreateFromStringForTesting("https://example.com"); |
| const blink::StorageKey google_storage_key_ = |
| blink::StorageKey::CreateFromStringForTesting("https://google.com"); |
| }; |
| |
| TEST_F(IndexedDBContextTest, DefaultBucketCreatedOnBindIndexedDB) { |
| mojo::Remote<blink::mojom::IDBFactory> example_remote; |
| mojo::AssociatedReceiver<storage::mojom::IndexedDBClientStateChecker> |
| example_checker_receiver(&example_checker); |
| indexed_db_context_->BindIndexedDB( |
| example_storage_key_, |
| example_checker_receiver.BindNewEndpointAndPassDedicatedRemote(), |
| example_remote.BindNewPipeAndPassReceiver()); |
| |
| mojo::Remote<blink::mojom::IDBFactory> google_remote; |
| mojo::AssociatedReceiver<storage::mojom::IndexedDBClientStateChecker> |
| google_checker_receiver(&example_checker); |
| indexed_db_context_->BindIndexedDB( |
| google_storage_key_, |
| google_checker_receiver.BindNewEndpointAndPassDedicatedRemote(), |
| google_remote.BindNewPipeAndPassReceiver()); |
| |
| storage::QuotaManagerProxySync quota_manager_proxy_sync( |
| quota_manager_proxy_.get()); |
| |
| // Call a method on both IDBFactory remotes and wait for both replies |
| // to ensure that BindIndexedDB has completed for both storage keys. |
| base::RunLoop loop; |
| auto callbacks = base::MakeRefCounted<MockIndexedDBCallbacks>( |
| /*expect_connection=*/false); |
| callbacks->CallOnInfoSuccess(base::BarrierClosure(2, loop.QuitClosure())); |
| auto example_bucket_locator = storage::BucketLocator(); |
| example_bucket_locator.storage_key = example_storage_key_; |
| indexed_db_context_->GetIDBFactory()->GetDatabaseInfo( |
| callbacks, example_bucket_locator, |
| indexed_db_context_->GetDataPath(example_bucket_locator)); |
| auto google_bucket_locator = storage::BucketLocator(); |
| google_bucket_locator.storage_key = google_storage_key_; |
| indexed_db_context_->GetIDBFactory()->GetDatabaseInfo( |
| callbacks, google_bucket_locator, |
| indexed_db_context_->GetDataPath(google_bucket_locator)); |
| loop.Run(); |
| |
| // Check default bucket exists for https://example.com. |
| storage::QuotaErrorOr<storage::BucketInfo> result = |
| quota_manager_proxy_sync.GetBucket(example_storage_key_, |
| storage::kDefaultBucketName, |
| blink::mojom::StorageType::kTemporary); |
| ASSERT_TRUE(result.ok()); |
| EXPECT_EQ(result->name, storage::kDefaultBucketName); |
| EXPECT_EQ(result->storage_key, example_storage_key_); |
| EXPECT_GT(result->id.value(), 0); |
| |
| // Check default bucket exists for https://google.com. |
| result = quota_manager_proxy_sync.GetBucket( |
| google_storage_key_, storage::kDefaultBucketName, |
| blink::mojom::StorageType::kTemporary); |
| ASSERT_TRUE(result.ok()); |
| EXPECT_EQ(result->name, storage::kDefaultBucketName); |
| EXPECT_EQ(result->storage_key, google_storage_key_); |
| EXPECT_GT(result->id.value(), 0); |
| } |
| |
| TEST_F(IndexedDBContextTest, GetDefaultBucketError) { |
| // Disable database so it will return errors when getting the default bucket. |
| quota_manager_->SetDisableDatabase(true); |
| |
| mojo::Remote<blink::mojom::IDBFactory> example_remote; |
| mojo::AssociatedReceiver<storage::mojom::IndexedDBClientStateChecker> |
| example_checker_receiver(&example_checker); |
| indexed_db_context_->BindIndexedDB( |
| example_storage_key_, |
| example_checker_receiver.BindNewEndpointAndPassDedicatedRemote(), |
| example_remote.BindNewPipeAndPassReceiver()); |
| |
| // IDBFactory::GetDatabaseInfo |
| base::RunLoop loop_1; |
| auto mock_callbacks = |
| std::make_unique<testing::StrictMock<MockMojoIndexedDBCallbacks>>(); |
| EXPECT_CALL(*mock_callbacks, Error(blink::mojom::IDBException::kUnknownError, |
| std::u16string(u"Internal error."))) |
| .Times(1) |
| .WillOnce(base::test::RunClosure(loop_1.QuitClosure())); |
| |
| example_remote->GetDatabaseInfo(mock_callbacks->CreateInterfacePtrAndBind()); |
| loop_1.Run(); |
| |
| testing::Mock::VerifyAndClear(&mock_callbacks); |
| |
| // IDBFactory::Open |
| base::RunLoop loop_2; |
| mock_callbacks = |
| std::make_unique<testing::StrictMock<MockMojoIndexedDBCallbacks>>(); |
| auto database_callbacks = |
| std::make_unique<MockMojoIndexedDBDatabaseCallbacks>(); |
| auto transaction_remote = |
| mojo::AssociatedRemote<blink::mojom::IDBTransaction>(); |
| EXPECT_CALL(*mock_callbacks, Error(blink::mojom::IDBException::kUnknownError, |
| std::u16string(u"Internal error."))) |
| .Times(1) |
| .WillOnce(base::test::RunClosure(loop_2.QuitClosure())); |
| |
| example_remote->Open(mock_callbacks->CreateInterfacePtrAndBind(), |
| database_callbacks->CreateInterfacePtrAndBind(), |
| u"database_name", /*version=*/1, |
| transaction_remote.BindNewEndpointAndPassReceiver(), |
| /*transaction_id=*/0); |
| loop_2.Run(); |
| |
| // IDBFactory::DeleteDatabase |
| base::RunLoop loop_3; |
| mock_callbacks = |
| std::make_unique<testing::StrictMock<MockMojoIndexedDBCallbacks>>(); |
| EXPECT_CALL(*mock_callbacks, Error(blink::mojom::IDBException::kUnknownError, |
| std::u16string(u"Internal error."))) |
| .Times(1) |
| .WillOnce(base::test::RunClosure(loop_3.QuitClosure())); |
| |
| example_remote->DeleteDatabase(mock_callbacks->CreateInterfacePtrAndBind(), |
| u"database_name", /*force_close=*/true); |
| loop_3.Run(); |
| } |
| |
| } // namespace content |