358 lines
15 KiB
JavaScript
358 lines
15 KiB
JavaScript
// META: title=IndexedDB: index renaming support
|
|
// META: global=window,worker
|
|
// META: script=resources/support-promises.js
|
|
// META: timeout=long
|
|
|
|
// Spec: https://w3c.github.io/IndexedDB/#dom-idbindex-name
|
|
|
|
'use strict';
|
|
|
|
promise_test(testCase => {
|
|
let authorIndex = null;
|
|
let authorIndex2 = null;
|
|
let renamedAuthorIndex = null;
|
|
let renamedAuthorIndex2 = null;
|
|
return createDatabase(
|
|
testCase,
|
|
(database, transaction) => {
|
|
const store = createBooksStore(testCase, database);
|
|
authorIndex = store.index('by_author');
|
|
})
|
|
.then(database => {
|
|
const transaction = database.transaction('books', 'readonly');
|
|
const store = transaction.objectStore('books');
|
|
assert_array_equals(
|
|
store.indexNames, ['by_author', 'by_title'],
|
|
'Test setup should have created two indexes');
|
|
authorIndex2 = store.index('by_author');
|
|
return checkAuthorIndexContents(
|
|
testCase, authorIndex2,
|
|
'The index should have the expected contents before any renaming')
|
|
.then(() => database.close());
|
|
})
|
|
.then(
|
|
() => migrateDatabase(
|
|
testCase, 2,
|
|
(database, transaction) => {
|
|
const store = transaction.objectStore('books');
|
|
renamedAuthorIndex = store.index('by_author');
|
|
renamedAuthorIndex.name = 'renamed_by_author';
|
|
|
|
assert_equals(
|
|
renamedAuthorIndex.name, 'renamed_by_author',
|
|
'IDBIndex name should change immediately after a rename');
|
|
assert_array_equals(
|
|
store.indexNames, ['by_title', 'renamed_by_author'],
|
|
'IDBObjectStore.indexNames should immediately reflect the rename');
|
|
assert_equals(
|
|
store.index('renamed_by_author'), renamedAuthorIndex,
|
|
'IDBObjectStore.index should return the renamed index store when ' +
|
|
'queried using the new name immediately after the rename');
|
|
assert_throws_dom(
|
|
'NotFoundError', () => store.index('by_author'),
|
|
'IDBObjectStore.index should throw when queried using the ' +
|
|
'renamed index\'s old name immediately after the rename');
|
|
}))
|
|
.then(database => {
|
|
const transaction = database.transaction('books', 'readonly');
|
|
const store = transaction.objectStore('books');
|
|
assert_array_equals(
|
|
store.indexNames, ['by_title', 'renamed_by_author'],
|
|
'IDBObjectStore.indexNames should still reflect the rename after ' +
|
|
'the versionchange transaction commits');
|
|
renamedAuthorIndex2 = store.index('renamed_by_author');
|
|
return checkAuthorIndexContents(
|
|
testCase, renamedAuthorIndex2,
|
|
'Renaming an index should not change its contents')
|
|
.then(() => database.close());
|
|
})
|
|
.then(() => {
|
|
assert_equals(
|
|
authorIndex.name, 'by_author',
|
|
'IDBIndex obtained before the rename transaction should not ' +
|
|
'reflect the rename');
|
|
assert_equals(
|
|
authorIndex2.name, 'by_author',
|
|
'IDBIndex obtained before the rename transaction should not ' +
|
|
'reflect the rename');
|
|
assert_equals(
|
|
renamedAuthorIndex.name, 'renamed_by_author',
|
|
'IDBIndex used in the rename transaction should keep reflecting ' +
|
|
'the new name after the transaction is committed');
|
|
assert_equals(
|
|
renamedAuthorIndex2.name, 'renamed_by_author',
|
|
'IDBIndex obtained after the rename transaction should reflect ' +
|
|
'the new name');
|
|
});
|
|
}, 'IndexedDB index rename in new transaction');
|
|
|
|
promise_test(testCase => {
|
|
let renamedAuthorIndex = null;
|
|
let renamedAuthorIndex2 = null;
|
|
return createDatabase(
|
|
testCase,
|
|
(database, transaction) => {
|
|
const store = createBooksStore(testCase, database);
|
|
renamedAuthorIndex = store.index('by_author');
|
|
renamedAuthorIndex.name = 'renamed_by_author';
|
|
|
|
assert_equals(
|
|
renamedAuthorIndex.name, 'renamed_by_author',
|
|
'IDBIndex name should change immediately after a rename');
|
|
assert_array_equals(
|
|
store.indexNames, ['by_title', 'renamed_by_author'],
|
|
'IDBObjectStore.indexNames should immediately reflect the rename');
|
|
assert_equals(
|
|
store.index('renamed_by_author'), renamedAuthorIndex,
|
|
'IDBObjectStore.index should return the renamed index store when ' +
|
|
'queried using the new name immediately after the rename');
|
|
assert_throws_dom(
|
|
'NotFoundError', () => store.index('by_author'),
|
|
'IDBObjectStore.index should throw when queried using the ' +
|
|
'renamed index\'s old name immediately after the rename');
|
|
})
|
|
.then(database => {
|
|
const transaction = database.transaction('books', 'readonly');
|
|
const store = transaction.objectStore('books');
|
|
assert_array_equals(
|
|
store.indexNames, ['by_title', 'renamed_by_author'],
|
|
'IDBObjectStore.indexNames should still reflect the rename after ' +
|
|
'the versionchange transaction commits');
|
|
renamedAuthorIndex2 = store.index('renamed_by_author');
|
|
return checkAuthorIndexContents(
|
|
testCase, renamedAuthorIndex2,
|
|
'Renaming an index should not change its contents')
|
|
.then(() => database.close());
|
|
})
|
|
.then(() => {
|
|
assert_equals(
|
|
renamedAuthorIndex.name, 'renamed_by_author',
|
|
'IDBIndex used in the rename transaction should keep reflecting ' +
|
|
'the new name after the transaction is committed');
|
|
assert_equals(
|
|
renamedAuthorIndex2.name, 'renamed_by_author',
|
|
'IDBIndex obtained after the rename transaction should reflect ' +
|
|
'the new name');
|
|
});
|
|
}, 'IndexedDB index rename in the transaction where it is created');
|
|
|
|
promise_test(testCase => {
|
|
return createDatabase(
|
|
testCase,
|
|
(database, transaction) => {
|
|
createBooksStore(testCase, database);
|
|
})
|
|
.then(database => {
|
|
database.close();
|
|
})
|
|
.then(
|
|
() => migrateDatabase(
|
|
testCase, 2,
|
|
(database, transaction) => {
|
|
const store = transaction.objectStore('books');
|
|
const index = store.index('by_author');
|
|
index.name = 'by_author';
|
|
assert_array_equals(
|
|
store.indexNames, ['by_author', 'by_title'],
|
|
'Renaming an index to the same name should not change the ' +
|
|
'index\'s IDBObjectStore.indexNames');
|
|
}))
|
|
.then(database => {
|
|
const transaction = database.transaction('books', 'readonly');
|
|
const store = transaction.objectStore('books');
|
|
assert_array_equals(
|
|
store.indexNames, ['by_author', 'by_title'],
|
|
'Committing a transaction that renames a store to the same name ' +
|
|
'should not change the index\'s IDBObjectStore.indexNames');
|
|
const index = store.index('by_author');
|
|
return checkAuthorIndexContents(
|
|
testCase, index,
|
|
'Committing a transaction that renames an index to the same name ' +
|
|
'should not change the index\'s contents')
|
|
.then(() => database.close());
|
|
});
|
|
}, 'IndexedDB index rename to the same name succeeds');
|
|
|
|
promise_test(testCase => {
|
|
return createDatabase(
|
|
testCase,
|
|
(database, transaction) => {
|
|
createBooksStore(testCase, database);
|
|
})
|
|
.then(database => {
|
|
database.close();
|
|
})
|
|
.then(
|
|
() => migrateDatabase(
|
|
testCase, 2,
|
|
(database, transaction) => {
|
|
const store = transaction.objectStore('books');
|
|
const index = store.index('by_author');
|
|
store.deleteIndex('by_title');
|
|
index.name = 'by_title';
|
|
assert_array_equals(
|
|
store.indexNames, ['by_title'],
|
|
'IDBObjectStore.indexNames should immediately reflect the rename');
|
|
}))
|
|
.then(database => {
|
|
const transaction = database.transaction('books', 'readonly');
|
|
const store = transaction.objectStore('books');
|
|
assert_array_equals(
|
|
store.indexNames, ['by_title'],
|
|
'IDBObjectStore.indexNames should still reflect the rename after ' +
|
|
'the versionchange transaction commits');
|
|
const index = store.index('by_title');
|
|
return checkAuthorIndexContents(
|
|
testCase, index,
|
|
'Renaming an index should not change its contents')
|
|
.then(() => database.close());
|
|
});
|
|
}, 'IndexedDB index rename to the name of a deleted index succeeds');
|
|
|
|
promise_test(testCase => {
|
|
return createDatabase(
|
|
testCase,
|
|
(database, transaction) => {
|
|
createBooksStore(testCase, database);
|
|
})
|
|
.then(database => {
|
|
database.close();
|
|
})
|
|
.then(
|
|
() => migrateDatabase(
|
|
testCase, 2,
|
|
(database, transaction) => {
|
|
const store = transaction.objectStore('books');
|
|
store.index('by_author').name = 'tmp';
|
|
store.index('by_title').name = 'by_author';
|
|
store.index('tmp').name = 'by_title';
|
|
assert_array_equals(
|
|
store.indexNames, ['by_author', 'by_title'],
|
|
'IDBObjectStore.indexNames should reflect the swap immediately ' +
|
|
'after the renames');
|
|
return checkTitleIndexContents(
|
|
testCase, store.index('by_author'),
|
|
'Renaming an index should not change its contents');
|
|
}))
|
|
.then(database => {
|
|
const transaction = database.transaction('books', 'readonly');
|
|
const store = transaction.objectStore('books');
|
|
assert_array_equals(
|
|
store.indexNames, ['by_author', 'by_title'],
|
|
'IDBObjectStore.indexNames should still reflect the swap after ' +
|
|
'the versionchange transaction commits');
|
|
const index = store.index('by_title');
|
|
return checkAuthorIndexContents(
|
|
testCase, index,
|
|
'Renaming an index should not change its contents')
|
|
.then(() => database.close());
|
|
});
|
|
}, 'IndexedDB index swapping via renames succeeds');
|
|
|
|
promise_test(testCase => {
|
|
return createDatabase(
|
|
testCase,
|
|
(database, transaction) => {
|
|
createBooksStore(testCase, database);
|
|
})
|
|
.then(database => {
|
|
database.close();
|
|
})
|
|
.then(
|
|
() => migrateDatabase(
|
|
testCase, 2,
|
|
(database, transaction) => {
|
|
const store = transaction.objectStore('books');
|
|
const index = store.index('by_author');
|
|
|
|
index.name = 42;
|
|
assert_equals(
|
|
index.name, '42',
|
|
'IDBIndex name should change immediately after a rename to a ' +
|
|
'number');
|
|
assert_array_equals(
|
|
store.indexNames, ['42', 'by_title'],
|
|
'IDBObjectStore.indexNames should immediately reflect the ' +
|
|
'stringifying rename');
|
|
|
|
index.name = true;
|
|
assert_equals(
|
|
index.name, 'true',
|
|
'IDBIndex name should change immediately after a rename to a ' +
|
|
'boolean');
|
|
|
|
index.name = {};
|
|
assert_equals(
|
|
index.name, '[object Object]',
|
|
'IDBIndex name should change immediately after a rename to an ' +
|
|
'object');
|
|
|
|
index.name = () => null;
|
|
assert_equals(
|
|
index.name, '() => null',
|
|
'IDBIndex name should change immediately after a rename to a ' +
|
|
'function');
|
|
|
|
index.name = undefined;
|
|
assert_equals(
|
|
index.name, 'undefined',
|
|
'IDBIndex name should change immediately after a rename to ' +
|
|
'undefined');
|
|
}))
|
|
.then(database => {
|
|
const transaction = database.transaction('books', 'readonly');
|
|
const store = transaction.objectStore('books');
|
|
assert_array_equals(
|
|
store.indexNames, ['by_title', 'undefined'],
|
|
'IDBObjectStore.indexNames should reflect the last rename ' +
|
|
'after the versionchange transaction commits');
|
|
const index = store.index('undefined');
|
|
return checkAuthorIndexContents(
|
|
testCase, index,
|
|
'Renaming an index should not change its contents')
|
|
.then(() => database.close());
|
|
});
|
|
}, 'IndexedDB index rename stringifies non-string names');
|
|
|
|
for (let escapedName of ['', '\\u0000', '\\uDC00\\uD800'])
|
|
((escapedName) => {
|
|
const name = JSON.parse('"' + escapedName + '"');
|
|
promise_test(testCase => {
|
|
return createDatabase(
|
|
testCase,
|
|
(database, transaction) => {
|
|
createBooksStore(testCase, database);
|
|
})
|
|
.then(database => {
|
|
database.close();
|
|
})
|
|
.then(
|
|
() => migrateDatabase(
|
|
testCase, 2,
|
|
(database, transaction) => {
|
|
const store = transaction.objectStore('books');
|
|
const index = store.index('by_author');
|
|
|
|
index.name = name;
|
|
assert_equals(
|
|
index.name, name,
|
|
'IDBIndex name should change immediately after the rename');
|
|
assert_array_equals(
|
|
store.indexNames, [name, 'by_title'].sort(),
|
|
'IDBObjectStore.indexNames should immediately reflect the rename');
|
|
}))
|
|
.then(database => {
|
|
const transaction = database.transaction('books', 'readonly');
|
|
const store = transaction.objectStore('books');
|
|
assert_array_equals(
|
|
store.indexNames, [name, 'by_title'].sort(),
|
|
'IDBObjectStore.indexNames should reflect the rename ' +
|
|
'after the versionchange transaction commits');
|
|
const index = store.index(name);
|
|
return checkAuthorIndexContents(
|
|
testCase, index,
|
|
'Renaming an index should not change its contents')
|
|
.then(() => database.close());
|
|
});
|
|
}, 'IndexedDB index can be renamed to "' + escapedName + '"');
|
|
})(escapedName);
|