blob: d0b61191edb2a27db1aa3fd667a1edd186d923fa [file] [log] [blame]
/*
* Copyright (C) 2013 Google Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Google Inc. nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "third_party/blink/renderer/modules/indexeddb/idb_transaction.h"
#include <memory>
#include <utility>
#include "base/memory/scoped_refptr.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "third_party/blink/public/platform/web_url_response.h"
#include "third_party/blink/renderer/bindings/core/v8/v8_binding_for_testing.h"
#include "third_party/blink/renderer/core/dom/document.h"
#include "third_party/blink/renderer/core/dom/dom_exception.h"
#include "third_party/blink/renderer/core/dom/events/event_queue.h"
#include "third_party/blink/renderer/core/execution_context/execution_context.h"
#include "third_party/blink/renderer/core/testing/scoped_mock_overlay_scrollbars.h"
#include "third_party/blink/renderer/modules/indexeddb/idb_database.h"
#include "third_party/blink/renderer/modules/indexeddb/idb_key.h"
#include "third_party/blink/renderer/modules/indexeddb/idb_key_path.h"
#include "third_party/blink/renderer/modules/indexeddb/idb_metadata.h"
#include "third_party/blink/renderer/modules/indexeddb/idb_object_store.h"
#include "third_party/blink/renderer/modules/indexeddb/idb_test_helper.h"
#include "third_party/blink/renderer/modules/indexeddb/idb_value.h"
#include "third_party/blink/renderer/modules/indexeddb/idb_value_wrapping.h"
#include "third_party/blink/renderer/modules/indexeddb/mock_idb_database.h"
#include "third_party/blink/renderer/modules/indexeddb/mock_idb_transaction.h"
#include "third_party/blink/renderer/modules/indexeddb/web_idb_database.h"
#include "third_party/blink/renderer/modules/indexeddb/web_idb_transaction.h"
#include "third_party/blink/renderer/platform/bindings/v8_per_isolate_data.h"
#include "third_party/blink/renderer/platform/heap/garbage_collected.h"
#include "third_party/blink/renderer/platform/testing/testing_platform_support.h"
#include "third_party/blink/renderer/platform/testing/url_loader_mock_factory.h"
#include "third_party/blink/renderer/platform/wtf/shared_buffer.h"
#include "v8/include/v8.h"
namespace blink {
namespace {
class IDBTransactionTest : public testing::Test,
public ScopedMockOverlayScrollbars {
protected:
void SetUp() override {
url_loader_mock_factory_ = URLLoaderMockFactory::GetSingletonInstance();
WebURLResponse response;
response.SetCurrentRequestUrl(KURL("blob:"));
url_loader_mock_factory_->RegisterURLProtocol(WebString("blob"), response,
"");
}
void TearDown() override {
url_loader_mock_factory_->UnregisterAllURLsAndClearMemoryCache();
}
void BuildTransaction(V8TestingScope& scope,
MockIDBDatabase& mock_database,
MockIDBTransaction& mock_transaction) {
auto database_backend = std::make_unique<WebIDBDatabase>(
mock_database.BindNewEndpointAndPassDedicatedRemote(),
/*task_runner=*/nullptr);
db_ = MakeGarbageCollected<IDBDatabase>(
scope.GetExecutionContext(), std::move(database_backend),
mojo::NullAssociatedReceiver(), mojo::NullRemote());
auto transaction_backend = std::make_unique<WebIDBTransaction>(
/*task_runner=*/nullptr, kTransactionId);
auto transaction_receiver = transaction_backend->CreateReceiver();
transaction_receiver.EnableUnassociatedUsage();
mock_transaction.Bind(std::move(transaction_receiver));
HashSet<String> transaction_scope = {"store"};
transaction_ = IDBTransaction::CreateNonVersionChange(
scope.GetScriptState(), std::move(transaction_backend), kTransactionId,
transaction_scope, mojom::IDBTransactionMode::ReadOnly,
mojom::IDBTransactionDurability::Relaxed, db_.Get());
IDBKeyPath store_key_path("primaryKey");
scoped_refptr<IDBObjectStoreMetadata> store_metadata = base::AdoptRef(
new IDBObjectStoreMetadata("store", kStoreId, store_key_path, true, 1));
store_ = MakeGarbageCollected<IDBObjectStore>(store_metadata, transaction_);
}
URLLoaderMockFactory* url_loader_mock_factory_;
Persistent<IDBDatabase> db_;
Persistent<IDBTransaction> transaction_;
Persistent<IDBObjectStore> store_;
ScopedTestingPlatformSupport<TestingPlatformSupport> platform_;
static constexpr int64_t kTransactionId = 1234;
static constexpr int64_t kStoreId = 5678;
};
const int64_t IDBTransactionTest::kTransactionId;
TEST_F(IDBTransactionTest, ContextDestroyedEarlyDeath) {
V8TestingScope scope;
MockIDBDatabase database_backend;
MockIDBTransaction transaction_backend;
EXPECT_CALL(database_backend, Close()).Times(1);
BuildTransaction(scope, database_backend, transaction_backend);
Persistent<HeapHashSet<WeakMember<IDBTransaction>>> live_transactions =
MakeGarbageCollected<HeapHashSet<WeakMember<IDBTransaction>>>();
live_transactions->insert(transaction_);
ThreadState::Current()->CollectAllGarbageForTesting();
EXPECT_EQ(1u, live_transactions->size());
Persistent<IDBRequest> request =
IDBRequest::Create(scope.GetScriptState(), store_.Get(),
transaction_.Get(), IDBRequest::AsyncTraceState());
scope.PerformMicrotaskCheckpoint();
request.Clear(); // The transaction is holding onto the request.
ThreadState::Current()->CollectAllGarbageForTesting();
EXPECT_EQ(1u, live_transactions->size());
// This will generate an Abort() call to the back end which is dropped by the
// fake proxy, so an explicit OnAbort call is made.
scope.GetExecutionContext()->NotifyContextDestroyed();
transaction_->OnAbort(MakeGarbageCollected<DOMException>(
DOMExceptionCode::kAbortError, "Aborted"));
transaction_->transaction_backend()->FlushForTesting();
transaction_.Clear();
store_.Clear();
database_backend.Flush();
ThreadState::Current()->CollectAllGarbageForTesting();
EXPECT_EQ(0U, live_transactions->size());
}
TEST_F(IDBTransactionTest, ContextDestroyedAfterDone) {
V8TestingScope scope;
MockIDBDatabase database_backend;
MockIDBTransaction transaction_backend;
EXPECT_CALL(database_backend, Close()).Times(1);
BuildTransaction(scope, database_backend, transaction_backend);
Persistent<HeapHashSet<WeakMember<IDBTransaction>>> live_transactions =
MakeGarbageCollected<HeapHashSet<WeakMember<IDBTransaction>>>();
live_transactions->insert(transaction_);
ThreadState::Current()->CollectAllGarbageForTesting();
EXPECT_EQ(1U, live_transactions->size());
Persistent<IDBRequest> request =
IDBRequest::Create(scope.GetScriptState(), store_.Get(),
transaction_.Get(), IDBRequest::AsyncTraceState());
scope.PerformMicrotaskCheckpoint();
// This response should result in an event being enqueued immediately.
request->HandleResponse(CreateIDBValueForTesting(scope.GetIsolate(), false));
request.Clear(); // The transaction is holding onto the request.
ThreadState::Current()->CollectAllGarbageForTesting();
EXPECT_EQ(1U, live_transactions->size());
// This will generate an Abort() call to the back end which is dropped by the
// fake proxy, so an explicit OnAbort call is made.
scope.GetExecutionContext()->NotifyContextDestroyed();
transaction_->OnAbort(MakeGarbageCollected<DOMException>(
DOMExceptionCode::kAbortError, "Aborted"));
transaction_->transaction_backend()->FlushForTesting();
transaction_.Clear();
store_.Clear();
database_backend.Flush();
// The request completed, so it has enqueued a success event. Discard the
// event, so that the transaction can go away.
EXPECT_EQ(1U, live_transactions->size());
ThreadState::Current()->CollectAllGarbageForTesting();
EXPECT_EQ(0U, live_transactions->size());
}
TEST_F(IDBTransactionTest, ContextDestroyedWithQueuedResult) {
V8TestingScope scope;
MockIDBDatabase database_backend;
MockIDBTransaction transaction_backend;
EXPECT_CALL(database_backend, Close()).Times(1);
BuildTransaction(scope, database_backend, transaction_backend);
Persistent<HeapHashSet<WeakMember<IDBTransaction>>> live_transactions =
MakeGarbageCollected<HeapHashSet<WeakMember<IDBTransaction>>>();
live_transactions->insert(transaction_);
ThreadState::Current()->CollectAllGarbageForTesting();
EXPECT_EQ(1U, live_transactions->size());
Persistent<IDBRequest> request =
IDBRequest::Create(scope.GetScriptState(), store_.Get(),
transaction_.Get(), IDBRequest::AsyncTraceState());
scope.PerformMicrotaskCheckpoint();
request->HandleResponse(CreateIDBValueForTesting(scope.GetIsolate(), true));
request.Clear(); // The transaction is holding onto the request.
ThreadState::Current()->CollectAllGarbageForTesting();
EXPECT_EQ(1U, live_transactions->size());
// This will generate an Abort() call to the back end which is dropped by the
// fake proxy, so an explicit OnAbort call is made.
scope.GetExecutionContext()->NotifyContextDestroyed();
transaction_->OnAbort(MakeGarbageCollected<DOMException>(
DOMExceptionCode::kAbortError, "Aborted"));
transaction_->transaction_backend()->FlushForTesting();
transaction_.Clear();
store_.Clear();
database_backend.Flush();
url_loader_mock_factory_->ServeAsynchronousRequests();
ThreadState::Current()->CollectAllGarbageForTesting();
EXPECT_EQ(0U, live_transactions->size());
}
TEST_F(IDBTransactionTest, ContextDestroyedWithTwoQueuedResults) {
V8TestingScope scope;
MockIDBDatabase database_backend;
MockIDBTransaction transaction_backend;
EXPECT_CALL(database_backend, Close()).Times(1);
BuildTransaction(scope, database_backend, transaction_backend);
Persistent<HeapHashSet<WeakMember<IDBTransaction>>> live_transactions =
MakeGarbageCollected<HeapHashSet<WeakMember<IDBTransaction>>>();
live_transactions->insert(transaction_);
ThreadState::Current()->CollectAllGarbageForTesting();
EXPECT_EQ(1U, live_transactions->size());
Persistent<IDBRequest> request1 =
IDBRequest::Create(scope.GetScriptState(), store_.Get(),
transaction_.Get(), IDBRequest::AsyncTraceState());
Persistent<IDBRequest> request2 =
IDBRequest::Create(scope.GetScriptState(), store_.Get(),
transaction_.Get(), IDBRequest::AsyncTraceState());
scope.PerformMicrotaskCheckpoint();
request1->HandleResponse(CreateIDBValueForTesting(scope.GetIsolate(), true));
request2->HandleResponse(CreateIDBValueForTesting(scope.GetIsolate(), true));
request1.Clear(); // The transaction is holding onto the requests.
request2.Clear();
ThreadState::Current()->CollectAllGarbageForTesting();
EXPECT_EQ(1U, live_transactions->size());
// This will generate an Abort() call to the back end which is dropped by the
// fake proxy, so an explicit OnAbort call is made.
scope.GetExecutionContext()->NotifyContextDestroyed();
transaction_->OnAbort(MakeGarbageCollected<DOMException>(
DOMExceptionCode::kAbortError, "Aborted"));
transaction_->transaction_backend()->FlushForTesting();
transaction_.Clear();
store_.Clear();
database_backend.Flush();
url_loader_mock_factory_->ServeAsynchronousRequests();
ThreadState::Current()->CollectAllGarbageForTesting();
EXPECT_EQ(0U, live_transactions->size());
}
TEST_F(IDBTransactionTest, DocumentShutdownWithQueuedAndBlockedResults) {
// This test covers the conditions of https://crbug.com/733642
MockIDBDatabase database_backend;
MockIDBTransaction transaction_backend;
EXPECT_CALL(database_backend, Close()).Times(1);
{
// The database isn't actually closed until `scope` is destroyed, so create
// this object in a nested scope to allow mock expectations to be verified.
V8TestingScope scope;
BuildTransaction(scope, database_backend, transaction_backend);
Persistent<HeapHashSet<WeakMember<IDBTransaction>>> live_transactions =
MakeGarbageCollected<HeapHashSet<WeakMember<IDBTransaction>>>();
live_transactions->insert(transaction_);
ThreadState::Current()->CollectAllGarbageForTesting();
EXPECT_EQ(1U, live_transactions->size());
Persistent<IDBRequest> request1 =
IDBRequest::Create(scope.GetScriptState(), store_.Get(),
transaction_.Get(), IDBRequest::AsyncTraceState());
Persistent<IDBRequest> request2 =
IDBRequest::Create(scope.GetScriptState(), store_.Get(),
transaction_.Get(), IDBRequest::AsyncTraceState());
scope.PerformMicrotaskCheckpoint();
request1->HandleResponse(
CreateIDBValueForTesting(scope.GetIsolate(), true));
request2->HandleResponse(
CreateIDBValueForTesting(scope.GetIsolate(), false));
request1.Clear(); // The transaction is holding onto the requests.
request2.Clear();
ThreadState::Current()->CollectAllGarbageForTesting();
EXPECT_EQ(1U, live_transactions->size());
// This will generate an Abort() call to the back end which is dropped by
// the fake proxy, so an explicit OnAbort call is made.
scope.GetDocument().Shutdown();
transaction_->OnAbort(MakeGarbageCollected<DOMException>(
DOMExceptionCode::kAbortError, "Aborted"));
transaction_->transaction_backend()->FlushForTesting();
transaction_.Clear();
store_.Clear();
url_loader_mock_factory_->ServeAsynchronousRequests();
ThreadState::Current()->CollectAllGarbageForTesting();
EXPECT_EQ(0U, live_transactions->size());
}
database_backend.Flush();
}
TEST_F(IDBTransactionTest, TransactionFinish) {
V8TestingScope scope;
MockIDBDatabase database_backend;
MockIDBTransaction transaction_backend;
EXPECT_CALL(transaction_backend, Commit(0)).Times(1);
EXPECT_CALL(database_backend, Close()).Times(1);
BuildTransaction(scope, database_backend, transaction_backend);
Persistent<HeapHashSet<WeakMember<IDBTransaction>>> live_transactions =
MakeGarbageCollected<HeapHashSet<WeakMember<IDBTransaction>>>();
live_transactions->insert(transaction_);
ThreadState::Current()->CollectAllGarbageForTesting();
EXPECT_EQ(1U, live_transactions->size());
scope.PerformMicrotaskCheckpoint();
ThreadState::Current()->CollectAllGarbageForTesting();
EXPECT_EQ(1U, live_transactions->size());
transaction_->transaction_backend()->FlushForTesting();
transaction_.Clear();
store_.Clear();
ThreadState::Current()->CollectAllGarbageForTesting();
EXPECT_EQ(1U, live_transactions->size());
// Stop the context, so events don't get queued (which would keep the
// transaction alive).
scope.GetExecutionContext()->NotifyContextDestroyed();
// Fire an abort to make sure this doesn't free the transaction during use.
// The test will not fail if it is, but ASAN would notice the error.
db_->Abort(kTransactionId, mojom::blink::IDBException::kAbortError,
"Aborted");
database_backend.Flush();
// OnAbort() should have cleared the transaction's reference to the database.
ThreadState::Current()->CollectAllGarbageForTesting();
EXPECT_EQ(0U, live_transactions->size());
}
} // namespace
} // namespace blink