blob: 2a109ede4375d931f31a0b4d423f7a4e3b26d3a6 [file] [log] [blame]
// Copyright 2013 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/indexeddb/web_idb_database_impl.h"
#include <stddef.h>
#include <string>
#include <vector>
#include "base/format_macros.h"
#include "base/memory/ptr_util.h"
#include "mojo/public/cpp/bindings/strong_associated_binding.h"
#include "third_party/blink/public/platform/modules/indexeddb/web_idb_database_exception.h"
#include "third_party/blink/renderer/modules/indexeddb/idb_key_range.h"
#include "third_party/blink/renderer/modules/indexeddb/indexed_db_blink_mojom_traits.h"
#include "third_party/blink/renderer/modules/indexeddb/indexed_db_callbacks_impl.h"
#include "third_party/blink/renderer/modules/indexeddb/indexed_db_dispatcher.h"
#include "third_party/blink/renderer/modules/indexeddb/indexed_db_key_builder.h"
#include "third_party/blink/renderer/modules/indexeddb/web_idb_database_error.h"
#include "third_party/blink/renderer/modules/indexeddb/web_idb_key_path.h"
#include "third_party/blink/renderer/platform/wtf/text/wtf_string.h"
namespace blink {
WebIDBDatabaseImpl::WebIDBDatabaseImpl(
mojom::blink::IDBDatabaseAssociatedPtrInfo database_info)
: database_(std::move(database_info)) {}
WebIDBDatabaseImpl::~WebIDBDatabaseImpl() = default;
void WebIDBDatabaseImpl::CreateObjectStore(long long transaction_id,
long long object_store_id,
const String& name,
const WebIDBKeyPath& key_path,
bool auto_increment) {
database_->CreateObjectStore(transaction_id, object_store_id, name, key_path,
auto_increment);
}
void WebIDBDatabaseImpl::DeleteObjectStore(long long transaction_id,
long long object_store_id) {
database_->DeleteObjectStore(transaction_id, object_store_id);
}
void WebIDBDatabaseImpl::RenameObjectStore(long long transaction_id,
long long object_store_id,
const String& new_name) {
database_->RenameObjectStore(transaction_id, object_store_id, new_name);
}
void WebIDBDatabaseImpl::CreateTransaction(
long long transaction_id,
const Vector<int64_t>& object_store_ids,
mojom::IDBTransactionMode mode) {
database_->CreateTransaction(transaction_id, object_store_ids, mode);
}
void WebIDBDatabaseImpl::Close() {
database_->Close();
}
void WebIDBDatabaseImpl::VersionChangeIgnored() {
database_->VersionChangeIgnored();
}
void WebIDBDatabaseImpl::AddObserver(
long long transaction_id,
int32_t observer_id,
bool include_transaction,
bool no_records,
bool values,
std::bitset<blink::kIDBOperationTypeCount> operation_types) {
static_assert(kIDBOperationTypeCount < sizeof(uint32_t) * CHAR_BIT,
"IDBOperationTypeCount exceeds size of uint32_t");
database_->AddObserver(transaction_id, observer_id, include_transaction,
no_records, values,
static_cast<uint32_t>(operation_types.to_ulong()));
}
void WebIDBDatabaseImpl::RemoveObservers(const Vector<int32_t>& observer_ids) {
database_->RemoveObservers(observer_ids);
}
void WebIDBDatabaseImpl::Get(long long transaction_id,
long long object_store_id,
long long index_id,
const IDBKeyRange* key_range,
bool key_only,
WebIDBCallbacks* callbacks) {
IndexedDBDispatcher::ResetCursorPrefetchCaches(transaction_id, nullptr);
mojom::blink::IDBKeyRangePtr key_range_ptr =
mojom::blink::IDBKeyRange::From(key_range);
auto callbacks_impl = std::make_unique<IndexedDBCallbacksImpl>(
base::WrapUnique(callbacks), transaction_id, nullptr);
database_->Get(transaction_id, object_store_id, index_id,
std::move(key_range_ptr), key_only,
GetCallbacksProxy(std::move(callbacks_impl)));
}
void WebIDBDatabaseImpl::GetAll(long long transaction_id,
long long object_store_id,
long long index_id,
const IDBKeyRange* key_range,
long long max_count,
bool key_only,
WebIDBCallbacks* callbacks) {
IndexedDBDispatcher::ResetCursorPrefetchCaches(transaction_id, nullptr);
mojom::blink::IDBKeyRangePtr key_range_ptr =
mojom::blink::IDBKeyRange::From(key_range);
auto callbacks_impl = std::make_unique<IndexedDBCallbacksImpl>(
base::WrapUnique(callbacks), transaction_id, nullptr);
database_->GetAll(transaction_id, object_store_id, index_id,
std::move(key_range_ptr), key_only, max_count,
GetCallbacksProxy(std::move(callbacks_impl)));
}
void WebIDBDatabaseImpl::Put(long long transaction_id,
long long object_store_id,
std::unique_ptr<IDBValue> value,
std::unique_ptr<IDBKey> primary_key,
mojom::IDBPutMode put_mode,
WebIDBCallbacks* callbacks,
Vector<IDBIndexKeys> index_keys) {
IndexedDBDispatcher::ResetCursorPrefetchCaches(transaction_id, nullptr);
size_t index_keys_size = 0;
for (const auto& index_key : index_keys) {
index_keys_size++; // Account for index_key.first (int64_t).
for (const auto& key : index_key.second) {
index_keys_size += key->SizeEstimate();
}
}
size_t arg_size =
value->DataSize() + primary_key->SizeEstimate() + index_keys_size;
if (arg_size >= max_put_value_size_) {
callbacks->OnError(blink::WebIDBDatabaseError(
blink::kWebIDBDatabaseExceptionUnknownError,
String::Format("The serialized keys and/or value are too large"
" (size=%" PRIuS " bytes, max=%" PRIuS " bytes).",
arg_size, max_put_value_size_)));
return;
}
auto callbacks_impl = std::make_unique<IndexedDBCallbacksImpl>(
base::WrapUnique(callbacks), transaction_id, nullptr);
database_->Put(transaction_id, object_store_id, std::move(value),
std::move(primary_key), put_mode, std::move(index_keys),
GetCallbacksProxy(std::move(callbacks_impl)));
}
void WebIDBDatabaseImpl::SetIndexKeys(long long transaction_id,
long long object_store_id,
std::unique_ptr<IDBKey> primary_key,
Vector<IDBIndexKeys> index_keys) {
database_->SetIndexKeys(transaction_id, object_store_id,
std::move(primary_key), std::move(index_keys));
}
void WebIDBDatabaseImpl::SetIndexesReady(long long transaction_id,
long long object_store_id,
const Vector<int64_t>& index_ids) {
database_->SetIndexesReady(transaction_id, object_store_id,
std::move(index_ids));
}
void WebIDBDatabaseImpl::OpenCursor(long long transaction_id,
long long object_store_id,
long long index_id,
const IDBKeyRange* key_range,
mojom::IDBCursorDirection direction,
bool key_only,
mojom::IDBTaskType task_type,
WebIDBCallbacks* callbacks) {
IndexedDBDispatcher::ResetCursorPrefetchCaches(transaction_id, nullptr);
mojom::blink::IDBKeyRangePtr key_range_ptr =
mojom::blink::IDBKeyRange::From(key_range);
auto callbacks_impl = std::make_unique<IndexedDBCallbacksImpl>(
base::WrapUnique(callbacks), transaction_id, nullptr);
database_->OpenCursor(transaction_id, object_store_id, index_id,
std::move(key_range_ptr), direction, key_only,
task_type,
GetCallbacksProxy(std::move(callbacks_impl)));
}
void WebIDBDatabaseImpl::Count(long long transaction_id,
long long object_store_id,
long long index_id,
const IDBKeyRange* key_range,
WebIDBCallbacks* callbacks) {
IndexedDBDispatcher::ResetCursorPrefetchCaches(transaction_id, nullptr);
mojom::blink::IDBKeyRangePtr key_range_ptr =
mojom::blink::IDBKeyRange::From(key_range);
auto callbacks_impl = std::make_unique<IndexedDBCallbacksImpl>(
base::WrapUnique(callbacks), transaction_id, nullptr);
database_->Count(transaction_id, object_store_id, index_id,
std::move(key_range_ptr),
GetCallbacksProxy(std::move(callbacks_impl)));
}
void WebIDBDatabaseImpl::Delete(long long transaction_id,
long long object_store_id,
const IDBKey* primary_key,
WebIDBCallbacks* callbacks) {
IndexedDBDispatcher::ResetCursorPrefetchCaches(transaction_id, nullptr);
mojom::blink::IDBKeyRangePtr key_range_ptr =
mojom::blink::IDBKeyRange::From(IDBKeyRangeBuilder::Build(primary_key));
auto callbacks_impl = std::make_unique<IndexedDBCallbacksImpl>(
base::WrapUnique(callbacks), transaction_id, nullptr);
database_->DeleteRange(transaction_id, object_store_id,
std::move(key_range_ptr),
GetCallbacksProxy(std::move(callbacks_impl)));
}
void WebIDBDatabaseImpl::DeleteRange(long long transaction_id,
long long object_store_id,
const IDBKeyRange* key_range,
WebIDBCallbacks* callbacks) {
IndexedDBDispatcher::ResetCursorPrefetchCaches(transaction_id, nullptr);
mojom::blink::IDBKeyRangePtr key_range_ptr =
mojom::blink::IDBKeyRange::From(key_range);
auto callbacks_impl = std::make_unique<IndexedDBCallbacksImpl>(
base::WrapUnique(callbacks), transaction_id, nullptr);
database_->DeleteRange(transaction_id, object_store_id,
std::move(key_range_ptr),
GetCallbacksProxy(std::move(callbacks_impl)));
}
void WebIDBDatabaseImpl::Clear(long long transaction_id,
long long object_store_id,
WebIDBCallbacks* callbacks) {
IndexedDBDispatcher::ResetCursorPrefetchCaches(transaction_id, nullptr);
auto callbacks_impl = std::make_unique<IndexedDBCallbacksImpl>(
base::WrapUnique(callbacks), transaction_id, nullptr);
database_->Clear(transaction_id, object_store_id,
GetCallbacksProxy(std::move(callbacks_impl)));
}
void WebIDBDatabaseImpl::CreateIndex(long long transaction_id,
long long object_store_id,
long long index_id,
const String& name,
const WebIDBKeyPath& key_path,
bool unique,
bool multi_entry) {
database_->CreateIndex(transaction_id, object_store_id, index_id, name,
key_path, unique, multi_entry);
}
void WebIDBDatabaseImpl::DeleteIndex(long long transaction_id,
long long object_store_id,
long long index_id) {
database_->DeleteIndex(transaction_id, object_store_id, index_id);
}
void WebIDBDatabaseImpl::RenameIndex(long long transaction_id,
long long object_store_id,
long long index_id,
const String& new_name) {
DCHECK(!new_name.IsNull());
database_->RenameIndex(transaction_id, object_store_id, index_id, new_name);
}
void WebIDBDatabaseImpl::Abort(long long transaction_id) {
database_->Abort(transaction_id);
}
void WebIDBDatabaseImpl::Commit(long long transaction_id) {
database_->Commit(transaction_id);
}
mojom::blink::IDBCallbacksAssociatedPtrInfo
WebIDBDatabaseImpl::GetCallbacksProxy(
std::unique_ptr<IndexedDBCallbacksImpl> callbacks) {
mojom::blink::IDBCallbacksAssociatedPtrInfo ptr_info;
auto request = mojo::MakeRequest(&ptr_info);
mojo::MakeStrongAssociatedBinding(std::move(callbacks), std::move(request));
return ptr_info;
}
} // namespace blink