blob: 3ee7671c6a78d16da124f7b8525b48e737b37e47 [file] [log] [blame]
/*
* Copyright (C) 2010 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:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. 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.
*
* THIS SOFTWARE IS PROVIDED BY APPLE AND ITS 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 APPLE OR ITS 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 "modules/indexeddb/IDBObjectStore.h"
#include <memory>
#include "bindings/core/v8/ExceptionState.h"
#include "bindings/core/v8/ToV8ForCore.h"
#include "bindings/core/v8/serialization/SerializedScriptValueFactory.h"
#include "bindings/modules/v8/ToV8ForModules.h"
#include "bindings/modules/v8/V8BindingForModules.h"
#include "core/dom/DOMStringList.h"
#include "core/dom/ExceptionCode.h"
#include "core/dom/ExecutionContext.h"
#include "modules/indexeddb/IDBAny.h"
#include "modules/indexeddb/IDBCursorWithValue.h"
#include "modules/indexeddb/IDBDatabase.h"
#include "modules/indexeddb/IDBKeyPath.h"
#include "modules/indexeddb/IDBTracing.h"
#include "modules/indexeddb/IDBValueWrapping.h"
#include "platform/Histogram.h"
#include "platform/SharedBuffer.h"
#include "platform/bindings/ScriptState.h"
#include "platform/wtf/RefPtr.h"
#include "public/platform/WebBlobInfo.h"
#include "public/platform/WebData.h"
#include "public/platform/WebVector.h"
#include "public/platform/modules/indexeddb/WebIDBKey.h"
#include "public/platform/modules/indexeddb/WebIDBKeyRange.h"
#include "v8/include/v8.h"
using blink::WebBlobInfo;
using blink::WebIDBCallbacks;
using blink::WebIDBCursor;
using blink::WebIDBDatabase;
using blink::WebVector;
namespace blink {
namespace {
using IndexKeys = HeapVector<Member<IDBKey>>;
}
IDBObjectStore::IDBObjectStore(RefPtr<IDBObjectStoreMetadata> metadata,
IDBTransaction* transaction)
: metadata_(std::move(metadata)), transaction_(transaction) {
DCHECK(transaction_);
DCHECK(metadata_.Get());
}
DEFINE_TRACE(IDBObjectStore) {
visitor->Trace(transaction_);
visitor->Trace(index_map_);
}
void IDBObjectStore::setName(const String& name,
ExceptionState& exception_state) {
IDB_TRACE("IDBObjectStore::setName");
if (!transaction_->IsVersionChange()) {
exception_state.ThrowDOMException(
kInvalidStateError,
IDBDatabase::kNotVersionChangeTransactionErrorMessage);
return;
}
if (IsDeleted()) {
exception_state.ThrowDOMException(
kInvalidStateError, IDBDatabase::kObjectStoreDeletedErrorMessage);
return;
}
if (!transaction_->IsActive()) {
exception_state.ThrowDOMException(kTransactionInactiveError,
transaction_->InactiveErrorMessage());
return;
}
if (this->name() == name)
return;
if (transaction_->db()->ContainsObjectStore(name)) {
exception_state.ThrowDOMException(
kConstraintError, IDBDatabase::kObjectStoreNameTakenErrorMessage);
return;
}
if (!BackendDB()) {
exception_state.ThrowDOMException(kInvalidStateError,
IDBDatabase::kDatabaseClosedErrorMessage);
return;
}
transaction_->db()->RenameObjectStore(Id(), name);
}
ScriptValue IDBObjectStore::keyPath(ScriptState* script_state) const {
return ScriptValue::From(script_state, Metadata().key_path);
}
DOMStringList* IDBObjectStore::indexNames() const {
IDB_TRACE("IDBObjectStore::indexNames");
DOMStringList* index_names = DOMStringList::Create();
for (const auto& it : Metadata().indexes)
index_names->Append(it.value->name);
index_names->Sort();
return index_names;
}
IDBRequest* IDBObjectStore::get(ScriptState* script_state,
const ScriptValue& key,
ExceptionState& exception_state) {
IDB_TRACE("IDBObjectStore::get");
if (IsDeleted()) {
exception_state.ThrowDOMException(
kInvalidStateError, IDBDatabase::kObjectStoreDeletedErrorMessage);
return nullptr;
}
if (!transaction_->IsActive()) {
exception_state.ThrowDOMException(kTransactionInactiveError,
transaction_->InactiveErrorMessage());
return nullptr;
}
IDBKeyRange* key_range = IDBKeyRange::FromScriptValue(
ExecutionContext::From(script_state), key, exception_state);
if (exception_state.HadException())
return nullptr;
if (!key_range) {
exception_state.ThrowDOMException(
kDataError, IDBDatabase::kNoKeyOrKeyRangeErrorMessage);
return nullptr;
}
if (!BackendDB()) {
exception_state.ThrowDOMException(kInvalidStateError,
IDBDatabase::kDatabaseClosedErrorMessage);
return nullptr;
}
IDBRequest* request = IDBRequest::Create(script_state, IDBAny::Create(this),
transaction_.Get());
BackendDB()->Get(transaction_->Id(), Id(), IDBIndexMetadata::kInvalidId,
key_range, /*key_only=*/false,
request->CreateWebCallbacks().release());
return request;
}
IDBRequest* IDBObjectStore::getKey(ScriptState* script_state,
const ScriptValue& key,
ExceptionState& exception_state) {
IDB_TRACE("IDBObjectStore::getKey");
if (IsDeleted()) {
exception_state.ThrowDOMException(
kInvalidStateError, IDBDatabase::kObjectStoreDeletedErrorMessage);
return nullptr;
}
if (!transaction_->IsActive()) {
exception_state.ThrowDOMException(kTransactionInactiveError,
transaction_->InactiveErrorMessage());
return nullptr;
}
IDBKeyRange* key_range = IDBKeyRange::FromScriptValue(
ExecutionContext::From(script_state), key, exception_state);
if (exception_state.HadException())
return nullptr;
if (!key_range) {
exception_state.ThrowDOMException(
kDataError, IDBDatabase::kNoKeyOrKeyRangeErrorMessage);
return nullptr;
}
if (!BackendDB()) {
exception_state.ThrowDOMException(kInvalidStateError,
IDBDatabase::kDatabaseClosedErrorMessage);
return nullptr;
}
IDBRequest* request = IDBRequest::Create(script_state, IDBAny::Create(this),
transaction_.Get());
BackendDB()->Get(transaction_->Id(), Id(), IDBIndexMetadata::kInvalidId,
key_range, /*key_only=*/true,
request->CreateWebCallbacks().release());
return request;
}
IDBRequest* IDBObjectStore::getAll(ScriptState* script_state,
const ScriptValue& key_range,
ExceptionState& exception_state) {
return getAll(script_state, key_range, std::numeric_limits<uint32_t>::max(),
exception_state);
}
IDBRequest* IDBObjectStore::getAll(ScriptState* script_state,
const ScriptValue& key_range,
unsigned long max_count,
ExceptionState& exception_state) {
IDB_TRACE("IDBObjectStore::getAll");
if (!max_count)
max_count = std::numeric_limits<uint32_t>::max();
if (IsDeleted()) {
exception_state.ThrowDOMException(
kInvalidStateError, IDBDatabase::kObjectStoreDeletedErrorMessage);
return nullptr;
}
if (!transaction_->IsActive()) {
exception_state.ThrowDOMException(kTransactionInactiveError,
transaction_->InactiveErrorMessage());
return nullptr;
}
IDBKeyRange* range = IDBKeyRange::FromScriptValue(
ExecutionContext::From(script_state), key_range, exception_state);
if (exception_state.HadException())
return nullptr;
if (!BackendDB()) {
exception_state.ThrowDOMException(kInvalidStateError,
IDBDatabase::kDatabaseClosedErrorMessage);
return nullptr;
}
IDBRequest* request = IDBRequest::Create(script_state, IDBAny::Create(this),
transaction_.Get());
BackendDB()->GetAll(transaction_->Id(), Id(), IDBIndexMetadata::kInvalidId,
range, max_count, false,
request->CreateWebCallbacks().release());
return request;
}
IDBRequest* IDBObjectStore::getAllKeys(ScriptState* script_state,
const ScriptValue& key_range,
ExceptionState& exception_state) {
return getAllKeys(script_state, key_range,
std::numeric_limits<uint32_t>::max(), exception_state);
}
IDBRequest* IDBObjectStore::getAllKeys(ScriptState* script_state,
const ScriptValue& key_range,
unsigned long max_count,
ExceptionState& exception_state) {
IDB_TRACE("IDBObjectStore::getAll");
if (!max_count)
max_count = std::numeric_limits<uint32_t>::max();
if (IsDeleted()) {
exception_state.ThrowDOMException(
kInvalidStateError, IDBDatabase::kObjectStoreDeletedErrorMessage);
return nullptr;
}
if (!transaction_->IsActive()) {
exception_state.ThrowDOMException(kTransactionInactiveError,
transaction_->InactiveErrorMessage());
return nullptr;
}
IDBKeyRange* range = IDBKeyRange::FromScriptValue(
ExecutionContext::From(script_state), key_range, exception_state);
if (exception_state.HadException())
return nullptr;
if (!BackendDB()) {
exception_state.ThrowDOMException(kInvalidStateError,
IDBDatabase::kDatabaseClosedErrorMessage);
return nullptr;
}
IDBRequest* request = IDBRequest::Create(script_state, IDBAny::Create(this),
transaction_.Get());
BackendDB()->GetAll(transaction_->Id(), Id(), IDBIndexMetadata::kInvalidId,
range, max_count, true,
request->CreateWebCallbacks().release());
return request;
}
static void GenerateIndexKeysForValue(v8::Isolate* isolate,
const IDBIndexMetadata& index_metadata,
const ScriptValue& object_value,
IndexKeys* index_keys) {
DCHECK(index_keys);
NonThrowableExceptionState exception_state;
IDBKey* index_key = ScriptValue::To<IDBKey*>(
isolate, object_value, exception_state, index_metadata.key_path);
if (!index_key)
return;
DEFINE_THREAD_SAFE_STATIC_LOCAL(
EnumerationHistogram, key_type_histogram,
("WebCore.IndexedDB.ObjectStore.IndexEntry.KeyType",
static_cast<int>(IDBKey::kTypeEnumMax)));
if (!index_metadata.multi_entry ||
index_key->GetType() != IDBKey::kArrayType) {
if (!index_key->IsValid())
return;
index_keys->push_back(index_key);
key_type_histogram.Count(static_cast<int>(index_key->GetType()));
} else {
DCHECK(index_metadata.multi_entry);
DCHECK_EQ(index_key->GetType(), IDBKey::kArrayType);
IDBKey::KeyArray array = index_key->ToMultiEntryArray();
for (const IDBKey* key : array)
key_type_histogram.Count(static_cast<int>(key->GetType()));
index_keys->AppendVector(array);
}
}
IDBRequest* IDBObjectStore::add(ScriptState* script_state,
const ScriptValue& value,
const ScriptValue& key,
ExceptionState& exception_state) {
IDB_TRACE("IDBObjectStore::add");
return put(script_state, kWebIDBPutModeAddOnly, IDBAny::Create(this), value,
key, exception_state);
}
IDBRequest* IDBObjectStore::put(ScriptState* script_state,
const ScriptValue& value,
const ScriptValue& key,
ExceptionState& exception_state) {
IDB_TRACE("IDBObjectStore::put");
return put(script_state, kWebIDBPutModeAddOrUpdate, IDBAny::Create(this),
value, key, exception_state);
}
IDBRequest* IDBObjectStore::put(ScriptState* script_state,
WebIDBPutMode put_mode,
IDBAny* source,
const ScriptValue& value,
const ScriptValue& key_value,
ExceptionState& exception_state) {
IDBKey* key = key_value.IsUndefined()
? nullptr
: ScriptValue::To<IDBKey*>(script_state->GetIsolate(),
key_value, exception_state);
if (exception_state.HadException())
return nullptr;
return put(script_state, put_mode, source, value, key, exception_state);
}
IDBRequest* IDBObjectStore::put(ScriptState* script_state,
WebIDBPutMode put_mode,
IDBAny* source,
const ScriptValue& value,
IDBKey* key,
ExceptionState& exception_state) {
if (IsDeleted()) {
exception_state.ThrowDOMException(
kInvalidStateError, IDBDatabase::kObjectStoreDeletedErrorMessage);
return nullptr;
}
if (!transaction_->IsActive()) {
exception_state.ThrowDOMException(kTransactionInactiveError,
transaction_->InactiveErrorMessage());
return nullptr;
}
if (transaction_->IsReadOnly()) {
exception_state.ThrowDOMException(
kReadOnlyError, IDBDatabase::kTransactionReadOnlyErrorMessage);
return nullptr;
}
v8::Isolate* isolate = script_state->GetIsolate();
DCHECK(isolate->InContext());
// TODO(crbug.com/719053): This wasm behavior differs from other browsers.
SerializedScriptValue::SerializeOptions::WasmSerializationPolicy wasm_policy =
ExecutionContext::From(script_state)->IsSecureContext()
? SerializedScriptValue::SerializeOptions::kSerialize
: SerializedScriptValue::SerializeOptions::kBlockedInNonSecureContext;
IDBValueWrapper value_wrapper(isolate, value.V8Value(), wasm_policy,
exception_state);
if (exception_state.HadException())
return nullptr;
// Keys that need to be extracted must be taken from a clone so that
// side effects (i.e. getters) are not triggered. Construct the
// clone lazily since the operation may be expensive.
ScriptValue clone;
const IDBKeyPath& key_path = IdbKeyPath();
const bool uses_in_line_keys = !key_path.IsNull();
const bool has_key_generator = autoIncrement();
if (put_mode != kWebIDBPutModeCursorUpdate && uses_in_line_keys && key) {
exception_state.ThrowDOMException(kDataError,
"The object store uses in-line keys and "
"the key parameter was provided.");
return nullptr;
}
// This test logically belongs in IDBCursor, but must operate on the cloned
// value.
if (put_mode == kWebIDBPutModeCursorUpdate && uses_in_line_keys) {
DCHECK(key);
DCHECK(clone.IsEmpty());
value_wrapper.Clone(script_state, &clone);
IDBKey* key_path_key = ScriptValue::To<IDBKey*>(
script_state->GetIsolate(), clone, exception_state, key_path);
if (exception_state.HadException())
return nullptr;
if (!key_path_key || !key_path_key->IsEqual(key)) {
exception_state.ThrowDOMException(
kDataError,
"The effective object store of this cursor uses in-line keys and "
"evaluating the key path of the value parameter results in a "
"different value than the cursor's effective key.");
return nullptr;
}
}
if (!uses_in_line_keys && !has_key_generator && !key) {
exception_state.ThrowDOMException(kDataError,
"The object store uses out-of-line keys "
"and has no key generator and the key "
"parameter was not provided.");
return nullptr;
}
if (uses_in_line_keys) {
if (clone.IsEmpty())
value_wrapper.Clone(script_state, &clone);
IDBKey* key_path_key = ScriptValue::To<IDBKey*>(
script_state->GetIsolate(), clone, exception_state, key_path);
if (exception_state.HadException())
return nullptr;
if (key_path_key && !key_path_key->IsValid()) {
exception_state.ThrowDOMException(
kDataError,
"Evaluating the object store's key path "
"yielded a value that is not a valid "
"key.");
return nullptr;
}
if (!has_key_generator && !key_path_key) {
exception_state.ThrowDOMException(
kDataError,
"Evaluating the object store's key path did not yield a value.");
return nullptr;
}
if (has_key_generator && !key_path_key) {
if (!CanInjectIDBKeyIntoScriptValue(script_state->GetIsolate(), clone,
key_path)) {
exception_state.ThrowDOMException(
kDataError,
"A generated key could not be inserted into the value.");
return nullptr;
}
}
if (key_path_key)
key = key_path_key;
}
if (key && !key->IsValid()) {
exception_state.ThrowDOMException(kDataError,
IDBDatabase::kNotValidKeyErrorMessage);
return nullptr;
}
if (!BackendDB()) {
exception_state.ThrowDOMException(kInvalidStateError,
IDBDatabase::kDatabaseClosedErrorMessage);
return nullptr;
}
if (key && uses_in_line_keys) {
DEFINE_THREAD_SAFE_STATIC_LOCAL(
EnumerationHistogram, key_type_histogram,
("WebCore.IndexedDB.ObjectStore.Record.KeyType",
static_cast<int>(IDBKey::kTypeEnumMax)));
key_type_histogram.Count(static_cast<int>(key->GetType()));
}
Vector<int64_t> index_ids;
HeapVector<IndexKeys> index_keys;
for (const auto& it : Metadata().indexes) {
if (clone.IsEmpty())
value_wrapper.Clone(script_state, &clone);
IndexKeys keys;
GenerateIndexKeysForValue(script_state->GetIsolate(), *it.value, clone,
&keys);
index_ids.push_back(it.key);
index_keys.push_back(keys);
}
IDBRequest* request =
IDBRequest::Create(script_state, source, transaction_.Get());
value_wrapper.ExtractBlobDataHandles(request->transit_blob_handles());
value_wrapper.WrapIfBiggerThan(IDBValueWrapper::kWrapThreshold);
BackendDB()->Put(
transaction_->Id(), Id(), WebData(value_wrapper.ExtractWireBytes()),
value_wrapper.WrappedBlobInfo(), key,
static_cast<WebIDBPutMode>(put_mode),
request->CreateWebCallbacks().release(), index_ids, index_keys);
return request;
}
IDBRequest* IDBObjectStore::deleteFunction(ScriptState* script_state,
const ScriptValue& key,
ExceptionState& exception_state) {
IDB_TRACE("IDBObjectStore::delete");
if (IsDeleted()) {
exception_state.ThrowDOMException(
kInvalidStateError, IDBDatabase::kObjectStoreDeletedErrorMessage);
return nullptr;
}
if (!transaction_->IsActive()) {
exception_state.ThrowDOMException(kTransactionInactiveError,
transaction_->InactiveErrorMessage());
return nullptr;
}
if (transaction_->IsReadOnly()) {
exception_state.ThrowDOMException(
kReadOnlyError, IDBDatabase::kTransactionReadOnlyErrorMessage);
return nullptr;
}
IDBKeyRange* key_range = IDBKeyRange::FromScriptValue(
ExecutionContext::From(script_state), key, exception_state);
if (exception_state.HadException())
return nullptr;
if (!key_range) {
exception_state.ThrowDOMException(
kDataError, IDBDatabase::kNoKeyOrKeyRangeErrorMessage);
return nullptr;
}
if (!BackendDB()) {
exception_state.ThrowDOMException(kInvalidStateError,
IDBDatabase::kDatabaseClosedErrorMessage);
return nullptr;
}
IDBRequest* request = IDBRequest::Create(script_state, IDBAny::Create(this),
transaction_.Get());
BackendDB()->DeleteRange(transaction_->Id(), Id(), key_range,
request->CreateWebCallbacks().release());
return request;
}
IDBRequest* IDBObjectStore::clear(ScriptState* script_state,
ExceptionState& exception_state) {
IDB_TRACE("IDBObjectStore::clear");
if (IsDeleted()) {
exception_state.ThrowDOMException(
kInvalidStateError, IDBDatabase::kObjectStoreDeletedErrorMessage);
return nullptr;
}
if (!transaction_->IsActive()) {
exception_state.ThrowDOMException(kTransactionInactiveError,
transaction_->InactiveErrorMessage());
return nullptr;
}
if (transaction_->IsReadOnly()) {
exception_state.ThrowDOMException(
kReadOnlyError, IDBDatabase::kTransactionReadOnlyErrorMessage);
return nullptr;
}
if (!BackendDB()) {
exception_state.ThrowDOMException(kInvalidStateError,
IDBDatabase::kDatabaseClosedErrorMessage);
return nullptr;
}
IDBRequest* request = IDBRequest::Create(script_state, IDBAny::Create(this),
transaction_.Get());
BackendDB()->Clear(transaction_->Id(), Id(),
request->CreateWebCallbacks().release());
return request;
}
namespace {
// This class creates the index keys for a given index by extracting
// them from the SerializedScriptValue, for all the existing values in
// the object store. It only needs to be kept alive by virtue of being
// a listener on an IDBRequest object, in the same way that JavaScript
// cursor success handlers are kept alive.
class IndexPopulator final : public EventListener {
public:
static IndexPopulator* Create(ScriptState* script_state,
IDBDatabase* database,
int64_t transaction_id,
int64_t object_store_id,
RefPtr<const IDBIndexMetadata> index_metadata) {
return new IndexPopulator(script_state, database, transaction_id,
object_store_id, std::move(index_metadata));
}
bool operator==(const EventListener& other) const override {
return this == &other;
}
DEFINE_INLINE_VIRTUAL_TRACE() {
visitor->Trace(database_);
EventListener::Trace(visitor);
}
private:
IndexPopulator(ScriptState* script_state,
IDBDatabase* database,
int64_t transaction_id,
int64_t object_store_id,
RefPtr<const IDBIndexMetadata> index_metadata)
: EventListener(kCPPEventListenerType),
script_state_(script_state),
database_(database),
transaction_id_(transaction_id),
object_store_id_(object_store_id),
index_metadata_(std::move(index_metadata)) {
DCHECK(index_metadata_.Get());
}
const IDBIndexMetadata& IndexMetadata() const { return *index_metadata_; }
void handleEvent(ExecutionContext* execution_context, Event* event) override {
if (!script_state_->ContextIsValid())
return;
DCHECK_EQ(ExecutionContext::From(script_state_.Get()), execution_context);
DCHECK_EQ(event->type(), EventTypeNames::success);
EventTarget* target = event->target();
IDBRequest* request = static_cast<IDBRequest*>(target);
if (!database_->Backend()) // If database is stopped?
return;
ScriptState::Scope scope(script_state_.Get());
IDBAny* cursor_any = request->ResultAsAny();
IDBCursorWithValue* cursor = nullptr;
if (cursor_any->GetType() == IDBAny::kIDBCursorWithValueType)
cursor = cursor_any->IdbCursorWithValue();
Vector<int64_t> index_ids;
index_ids.push_back(IndexMetadata().id);
if (cursor && !cursor->IsDeleted()) {
cursor->Continue(nullptr, nullptr, ASSERT_NO_EXCEPTION);
IDBKey* primary_key = cursor->IdbPrimaryKey();
ScriptValue value = cursor->value(script_state_.Get());
IndexKeys index_keys;
GenerateIndexKeysForValue(script_state_->GetIsolate(), IndexMetadata(),
value, &index_keys);
HeapVector<IndexKeys> index_keys_list;
index_keys_list.push_back(index_keys);
database_->Backend()->SetIndexKeys(transaction_id_, object_store_id_,
primary_key, index_ids,
index_keys_list);
} else {
// Now that we are done indexing, tell the backend to go
// back to processing tasks of type NormalTask.
database_->Backend()->SetIndexesReady(transaction_id_, object_store_id_,
index_ids);
database_.Clear();
}
}
RefPtr<ScriptState> script_state_;
Member<IDBDatabase> database_;
const int64_t transaction_id_;
const int64_t object_store_id_;
RefPtr<const IDBIndexMetadata> index_metadata_;
};
} // namespace
IDBIndex* IDBObjectStore::createIndex(ScriptState* script_state,
const String& name,
const IDBKeyPath& key_path,
const IDBIndexParameters& options,
ExceptionState& exception_state) {
IDB_TRACE("IDBObjectStore::createIndex");
if (!transaction_->IsVersionChange()) {
exception_state.ThrowDOMException(
kInvalidStateError,
IDBDatabase::kNotVersionChangeTransactionErrorMessage);
return nullptr;
}
if (IsDeleted()) {
exception_state.ThrowDOMException(
kInvalidStateError, IDBDatabase::kObjectStoreDeletedErrorMessage);
return nullptr;
}
if (!transaction_->IsActive()) {
exception_state.ThrowDOMException(kTransactionInactiveError,
transaction_->InactiveErrorMessage());
return nullptr;
}
if (ContainsIndex(name)) {
exception_state.ThrowDOMException(kConstraintError,
IDBDatabase::kIndexNameTakenErrorMessage);
return nullptr;
}
if (!key_path.IsValid()) {
exception_state.ThrowDOMException(
kSyntaxError, "The keyPath argument contains an invalid key path.");
return nullptr;
}
if (key_path.GetType() == IDBKeyPath::kArrayType && options.multiEntry()) {
exception_state.ThrowDOMException(
kInvalidAccessError,
"The keyPath argument was an array and the multiEntry option is true.");
return nullptr;
}
if (!BackendDB()) {
exception_state.ThrowDOMException(kInvalidStateError,
IDBDatabase::kDatabaseClosedErrorMessage);
return nullptr;
}
int64_t index_id = metadata_->max_index_id + 1;
DCHECK_NE(index_id, IDBIndexMetadata::kInvalidId);
BackendDB()->CreateIndex(transaction_->Id(), Id(), index_id, name, key_path,
options.unique(), options.multiEntry());
++metadata_->max_index_id;
RefPtr<IDBIndexMetadata> index_metadata = AdoptRef(new IDBIndexMetadata(
name, index_id, key_path, options.unique(), options.multiEntry()));
IDBIndex* index = IDBIndex::Create(index_metadata, this, transaction_.Get());
index_map_.Set(name, index);
metadata_->indexes.Set(index_id, index_metadata);
DCHECK(!exception_state.HadException());
if (exception_state.HadException())
return nullptr;
IDBRequest* index_request =
openCursor(script_state, nullptr, kWebIDBCursorDirectionNext,
kWebIDBTaskTypePreemptive);
index_request->PreventPropagation();
// This is kept alive by being the success handler of the request, which is in
// turn kept alive by the owning transaction.
IndexPopulator* index_populator = IndexPopulator::Create(
script_state, transaction()->db(), transaction_->Id(), Id(),
std::move(index_metadata));
index_request->setOnsuccess(index_populator);
return index;
}
IDBIndex* IDBObjectStore::index(const String& name,
ExceptionState& exception_state) {
IDB_TRACE("IDBObjectStore::index");
if (IsDeleted()) {
exception_state.ThrowDOMException(
kInvalidStateError, IDBDatabase::kObjectStoreDeletedErrorMessage);
return nullptr;
}
if (transaction_->IsFinished() || transaction_->IsFinishing()) {
exception_state.ThrowDOMException(
kInvalidStateError, IDBDatabase::kTransactionFinishedErrorMessage);
return nullptr;
}
IDBIndexMap::iterator it = index_map_.find(name);
if (it != index_map_.end())
return it->value;
int64_t index_id = FindIndexId(name);
if (index_id == IDBIndexMetadata::kInvalidId) {
exception_state.ThrowDOMException(kNotFoundError,
IDBDatabase::kNoSuchIndexErrorMessage);
return nullptr;
}
DCHECK(Metadata().indexes.Contains(index_id));
RefPtr<IDBIndexMetadata> index_metadata = Metadata().indexes.at(index_id);
DCHECK(index_metadata.Get());
IDBIndex* index =
IDBIndex::Create(std::move(index_metadata), this, transaction_.Get());
index_map_.Set(name, index);
return index;
}
void IDBObjectStore::deleteIndex(const String& name,
ExceptionState& exception_state) {
IDB_TRACE("IDBObjectStore::deleteIndex");
if (!transaction_->IsVersionChange()) {
exception_state.ThrowDOMException(
kInvalidStateError,
IDBDatabase::kNotVersionChangeTransactionErrorMessage);
return;
}
if (IsDeleted()) {
exception_state.ThrowDOMException(
kInvalidStateError, IDBDatabase::kObjectStoreDeletedErrorMessage);
return;
}
if (!transaction_->IsActive()) {
exception_state.ThrowDOMException(kTransactionInactiveError,
transaction_->InactiveErrorMessage());
return;
}
int64_t index_id = FindIndexId(name);
if (index_id == IDBIndexMetadata::kInvalidId) {
exception_state.ThrowDOMException(kNotFoundError,
IDBDatabase::kNoSuchIndexErrorMessage);
return;
}
if (!BackendDB()) {
exception_state.ThrowDOMException(kInvalidStateError,
IDBDatabase::kDatabaseClosedErrorMessage);
return;
}
BackendDB()->DeleteIndex(transaction_->Id(), Id(), index_id);
metadata_->indexes.erase(index_id);
IDBIndexMap::iterator it = index_map_.find(name);
if (it != index_map_.end()) {
transaction_->IndexDeleted(it->value);
it->value->MarkDeleted();
index_map_.erase(name);
}
}
IDBRequest* IDBObjectStore::openCursor(ScriptState* script_state,
const ScriptValue& range,
const String& direction_string,
ExceptionState& exception_state) {
IDB_TRACE("IDBObjectStore::openCursor");
if (IsDeleted()) {
exception_state.ThrowDOMException(
kInvalidStateError, IDBDatabase::kObjectStoreDeletedErrorMessage);
return nullptr;
}
if (!transaction_->IsActive()) {
exception_state.ThrowDOMException(kTransactionInactiveError,
transaction_->InactiveErrorMessage());
return nullptr;
}
WebIDBCursorDirection direction =
IDBCursor::StringToDirection(direction_string);
IDBKeyRange* key_range = IDBKeyRange::FromScriptValue(
ExecutionContext::From(script_state), range, exception_state);
if (exception_state.HadException())
return nullptr;
if (!BackendDB()) {
exception_state.ThrowDOMException(kInvalidStateError,
IDBDatabase::kDatabaseClosedErrorMessage);
return nullptr;
}
return openCursor(script_state, key_range, direction, kWebIDBTaskTypeNormal);
}
IDBRequest* IDBObjectStore::openCursor(ScriptState* script_state,
IDBKeyRange* range,
WebIDBCursorDirection direction,
WebIDBTaskType task_type) {
IDBRequest* request = IDBRequest::Create(script_state, IDBAny::Create(this),
transaction_.Get());
request->SetCursorDetails(IndexedDB::kCursorKeyAndValue, direction);
BackendDB()->OpenCursor(transaction_->Id(), Id(),
IDBIndexMetadata::kInvalidId, range, direction, false,
task_type, request->CreateWebCallbacks().release());
return request;
}
IDBRequest* IDBObjectStore::openKeyCursor(ScriptState* script_state,
const ScriptValue& range,
const String& direction_string,
ExceptionState& exception_state) {
IDB_TRACE("IDBObjectStore::openKeyCursor");
if (IsDeleted()) {
exception_state.ThrowDOMException(
kInvalidStateError, IDBDatabase::kObjectStoreDeletedErrorMessage);
return nullptr;
}
if (!transaction_->IsActive()) {
exception_state.ThrowDOMException(kTransactionInactiveError,
transaction_->InactiveErrorMessage());
return nullptr;
}
WebIDBCursorDirection direction =
IDBCursor::StringToDirection(direction_string);
IDBKeyRange* key_range = IDBKeyRange::FromScriptValue(
ExecutionContext::From(script_state), range, exception_state);
if (exception_state.HadException())
return nullptr;
if (!BackendDB()) {
exception_state.ThrowDOMException(kInvalidStateError,
IDBDatabase::kDatabaseClosedErrorMessage);
return nullptr;
}
IDBRequest* request = IDBRequest::Create(script_state, IDBAny::Create(this),
transaction_.Get());
request->SetCursorDetails(IndexedDB::kCursorKeyOnly, direction);
BackendDB()->OpenCursor(transaction_->Id(), Id(),
IDBIndexMetadata::kInvalidId, key_range, direction,
true, kWebIDBTaskTypeNormal,
request->CreateWebCallbacks().release());
return request;
}
IDBRequest* IDBObjectStore::count(ScriptState* script_state,
const ScriptValue& range,
ExceptionState& exception_state) {
IDB_TRACE("IDBObjectStore::count");
if (IsDeleted()) {
exception_state.ThrowDOMException(
kInvalidStateError, IDBDatabase::kObjectStoreDeletedErrorMessage);
return nullptr;
}
if (!transaction_->IsActive()) {
exception_state.ThrowDOMException(kTransactionInactiveError,
transaction_->InactiveErrorMessage());
return nullptr;
}
IDBKeyRange* key_range = IDBKeyRange::FromScriptValue(
ExecutionContext::From(script_state), range, exception_state);
if (exception_state.HadException())
return nullptr;
if (!BackendDB()) {
exception_state.ThrowDOMException(kInvalidStateError,
IDBDatabase::kDatabaseClosedErrorMessage);
return nullptr;
}
IDBRequest* request = IDBRequest::Create(script_state, IDBAny::Create(this),
transaction_.Get());
BackendDB()->Count(transaction_->Id(), Id(), IDBIndexMetadata::kInvalidId,
key_range, request->CreateWebCallbacks().release());
return request;
}
void IDBObjectStore::MarkDeleted() {
DCHECK(transaction_->IsVersionChange())
<< "An object store got deleted outside a versionchange transaction.";
deleted_ = true;
metadata_->indexes.clear();
for (auto& it : index_map_) {
IDBIndex* index = it.value;
index->MarkDeleted();
}
}
void IDBObjectStore::ClearIndexCache() {
DCHECK(!transaction_->IsActive() || (IsDeleted() && IsNewlyCreated()));
#if DCHECK_IS_ON()
// There is no harm in having ClearIndexCache() happen multiple times for
// the same object. We assert that it is called once to uncover potential
// object store accounting bugs.
DCHECK(!clear_index_cache_called_);
clear_index_cache_called_ = true;
#endif // DCHECK_IS_ON()
index_map_.clear();
}
void IDBObjectStore::RevertMetadata(
RefPtr<IDBObjectStoreMetadata> old_metadata) {
DCHECK(transaction_->IsVersionChange());
DCHECK(!transaction_->IsActive());
DCHECK(old_metadata.Get());
DCHECK(Id() == old_metadata->id);
for (auto& index : index_map_.Values()) {
const int64_t index_id = index->Id();
if (index->IsNewlyCreated(*old_metadata)) {
// The index was created by this transaction. According to the spec,
// its metadata will remain as-is.
DCHECK(!old_metadata->indexes.Contains(index_id));
index->MarkDeleted();
continue;
}
// The index was created in a previous transaction. We need to revert
// its metadata. The index might have been deleted, so we
// unconditionally reset the deletion marker.
DCHECK(old_metadata->indexes.Contains(index_id));
RefPtr<IDBIndexMetadata> old_index_metadata =
old_metadata->indexes.at(index_id);
index->RevertMetadata(std::move(old_index_metadata));
}
metadata_ = std::move(old_metadata);
// An object store's metadata will only get reverted if the index was in the
// database when the versionchange transaction started.
deleted_ = false;
}
void IDBObjectStore::RevertDeletedIndexMetadata(IDBIndex& deleted_index) {
DCHECK(transaction_->IsVersionChange());
DCHECK(!transaction_->IsActive());
DCHECK(deleted_index.objectStore() == this);
DCHECK(deleted_index.IsDeleted());
const int64_t index_id = deleted_index.Id();
DCHECK(metadata_->indexes.Contains(index_id))
<< "The object store's metadata was not correctly reverted";
RefPtr<IDBIndexMetadata> old_index_metadata = metadata_->indexes.at(index_id);
deleted_index.RevertMetadata(std::move(old_index_metadata));
}
void IDBObjectStore::RenameIndex(int64_t index_id, const String& new_name) {
DCHECK(transaction_->IsVersionChange());
DCHECK(transaction_->IsActive());
BackendDB()->RenameIndex(transaction_->Id(), Id(), index_id, new_name);
auto metadata_iterator = metadata_->indexes.find(index_id);
DCHECK_NE(metadata_iterator, metadata_->indexes.end()) << "Invalid index_id";
const String& old_name = metadata_iterator->value->name;
DCHECK(index_map_.Contains(old_name))
<< "The index had to be accessed in order to be renamed.";
DCHECK(!index_map_.Contains(new_name));
index_map_.Set(new_name, index_map_.Take(old_name));
metadata_iterator->value->name = new_name;
}
int64_t IDBObjectStore::FindIndexId(const String& name) const {
for (const auto& it : Metadata().indexes) {
if (it.value->name == name) {
DCHECK_NE(it.key, IDBIndexMetadata::kInvalidId);
return it.key;
}
}
return IDBIndexMetadata::kInvalidId;
}
WebIDBDatabase* IDBObjectStore::BackendDB() const {
return transaction_->BackendDB();
}
} // namespace blink