Source code

Revision control

Copy as Markdown

Other Tools

Test Info:

// META: title=IndexedDB: clone before key path evaluation
// META: global=window,worker
// META: script=resources/support.js
'use strict';
function ProbeObject() {
this.id_count = 0;
this.invalid_id_count = 0;
this.prop_count = 0;
Object.defineProperties(this, {
id: {
enumerable: true,
get() {
++this.id_count;
return 1000 + this.id_count;
},
},
invalid_id: {
enumerable: true,
get() {
++this.invalid_id_count;
return {};
},
},
prop: {
enumerable: true,
get() {
++this.prop_count;
return 2000 + this.prop_count;
},
},
});
}
function createObjectStoreWithKeyPath(
storeName, keyPath, autoIncrement = false) {
return (t, db) => {
db.createObjectStore(storeName, {keyPath, autoIncrement});
};
}
function createObjectStoreWithIndex(
storeName, keyPath, indexName, indexKeyPath) {
return (t, db) => {
const storeOptions = keyPath ? {keyPath} : {};
const store = db.createObjectStore(storeName, storeOptions);
// If index parameters are provided, create the index.
if (indexName && indexKeyPath) {
store.createIndex(indexName, indexKeyPath);
}
};
}
function createTransactionAndReturnObjectStore(db, storeName) {
const tx = db.transaction(storeName, 'readwrite', {durability: 'relaxed'});
const store = tx.objectStore(storeName);
return {tx, store};
}
indexeddb_test(createObjectStoreWithKeyPath('store', 'id', true), (t, db) => {
const {store} = createTransactionAndReturnObjectStore(db, 'store');
const obj = new ProbeObject();
store.put(obj);
assert_equals(
obj.id_count, 1,
'put() operation should access primary key property once');
assert_equals(
obj.prop_count, 1, 'put() operation should access other properties once');
t.done();
}, 'Key generator and key path validity check operates on a clone');
indexeddb_test(
createObjectStoreWithKeyPath('store', 'invalid_id', true), (t, db) => {
const {store} = createTransactionAndReturnObjectStore(db, 'store');
const obj = new ProbeObject();
assert_throws_dom('DataError', () => {
store.put(obj);
}, 'put() should throw if primary key cannot be injected');
assert_equals(
obj.invalid_id_count, 1,
'put() operation should access primary key property once');
assert_equals(
obj.prop_count, 1,
'put() operation should access other properties once');
t.done();
}, 'Failing key path validity check operates on a clone');
indexeddb_test(
createObjectStoreWithIndex('store', null, 'index', 'prop'), (t, db) => {
const {store} = createTransactionAndReturnObjectStore(db, 'store');
const obj = new ProbeObject();
store.put(obj, 'key');
assert_equals(
obj.prop_count, 1, 'put() should access index key property once');
assert_equals(
obj.id_count, 1,
'put() operation should access other properties once');
t.done();
}, 'Index key path evaluations operate on a clone');
indexeddb_test(
createObjectStoreWithIndex('store', 'id', 'index', 'prop'), (t, db) => {
const {store} = createTransactionAndReturnObjectStore(db, 'store');
const obj = new ProbeObject();
store.put(obj);
assert_equals(
obj.id_count, 1, 'put() should access primary key property once');
assert_equals(
obj.prop_count, 1, 'put() should access index key property once');
t.done();
}, 'Store and index key path evaluations operate on the same clone');
indexeddb_test(
createObjectStoreWithIndex('store', 'id', 'index', 'prop'), (t, db) => {
const {store} = createTransactionAndReturnObjectStore(db, 'store');
store.put(new ProbeObject());
store.openCursor().onsuccess = t.step_func((event) => {
const cursor = event.target.result;
const obj = new ProbeObject();
cursor.update(obj);
assert_equals(
obj.id_count, 1, 'put() should access primary key property once');
assert_equals(
obj.prop_count, 1, 'put() should access index key property once');
t.done();
});
}, 'Cursor update checks and keypath evaluations operate on a clone');