зеркало из https://github.com/mozilla/gecko-dev.git
Bug 1078438
- Change the way IndexedDB uses SQLite table locks. r=janv.
This commit is contained in:
Родитель
68b8173ea8
Коммит
88f558f496
|
@ -2198,7 +2198,13 @@ SetDefaultPragmas(mozIStorageConnection* aConnection)
|
|||
// refcount function. This behavior changes with enabled recursive triggers,
|
||||
// so the statement fires the delete trigger first and then the insert
|
||||
// trigger.
|
||||
"PRAGMA recursive_triggers = ON;";
|
||||
"PRAGMA recursive_triggers = ON;"
|
||||
// We don't need SQLite's table locks because we manage transaction ordering
|
||||
// ourselves and we know we will never allow a write transaction to modify
|
||||
// an object store that a read transaction is in the process of using.
|
||||
"PRAGMA read_uncommitted = TRUE;"
|
||||
// No more PRAGMAs.
|
||||
;
|
||||
|
||||
nsresult rv = aConnection->ExecuteSimpleSQL(NS_LITERAL_CSTRING(query));
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) {
|
||||
|
|
|
@ -78,6 +78,8 @@ support-files =
|
|||
unit/test_setVersion_events.js
|
||||
unit/test_setVersion_exclusion.js
|
||||
unit/test_success_events_after_abort.js
|
||||
unit/test_table_locks.js
|
||||
unit/test_table_rollback.js
|
||||
unit/test_temporary_storage.js
|
||||
unit/test_traffic_jam.js
|
||||
unit/test_transaction_abort.js
|
||||
|
@ -339,6 +341,10 @@ skip-if = (buildapp == 'b2g' && toolkit != 'gonk') # Bug 931116
|
|||
skip-if = (buildapp == 'b2g' && toolkit != 'gonk') # Bug 931116
|
||||
[test_success_events_after_abort.html]
|
||||
skip-if = (buildapp == 'b2g' && toolkit != 'gonk') # Bug 931116
|
||||
[test_table_locks.html]
|
||||
skip-if = (buildapp == 'b2g' && toolkit != 'gonk') # Bug 931116
|
||||
[test_table_rollback.html]
|
||||
skip-if = (buildapp == 'b2g' && toolkit != 'gonk') # Bug 931116
|
||||
[test_third_party.html]
|
||||
skip-if = (buildapp == 'b2g' && toolkit != 'gonk') # Bug 931116
|
||||
[test_traffic_jam.html]
|
||||
|
|
|
@ -0,0 +1,18 @@
|
|||
<!--
|
||||
Any copyright is dedicated to the Public Domain.
|
||||
http://creativecommons.org/publicdomain/zero/1.0/
|
||||
-->
|
||||
<html>
|
||||
<head>
|
||||
<title>IndexedDB Test</title>
|
||||
|
||||
<script type="text/javascript" src="/tests/SimpleTest/SimpleTest.js"></script>
|
||||
<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
|
||||
|
||||
<script type="text/javascript;version=1.7" src="unit/test_table_locks.js"></script>
|
||||
<script type="text/javascript;version=1.7" src="helpers.js"></script>
|
||||
</head>
|
||||
|
||||
<body onload="runTest();"></body>
|
||||
|
||||
</html>
|
|
@ -0,0 +1,19 @@
|
|||
<!--
|
||||
Any copyright is dedicated to the Public Domain.
|
||||
http://creativecommons.org/publicdomain/zero/1.0/
|
||||
-->
|
||||
<html>
|
||||
<head>
|
||||
<title>Indexed Database Test</title>
|
||||
|
||||
<script type="text/javascript" src="/tests/SimpleTest/SimpleTest.js"></script>
|
||||
<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
|
||||
|
||||
<script type="text/javascript;version=1.7" src="unit/test_table_rollback.js"></script>
|
||||
<script type="text/javascript;version=1.7" src="helpers.js"></script>
|
||||
|
||||
</head>
|
||||
|
||||
<body onload="runTest();"></body>
|
||||
|
||||
</html>
|
|
@ -0,0 +1,116 @@
|
|||
/**
|
||||
* Any copyright is dedicated to the Public Domain.
|
||||
* http://creativecommons.org/publicdomain/zero/1.0/
|
||||
*/
|
||||
|
||||
const dbName = ("window" in this) ? window.location.pathname : "test";
|
||||
const dbVersion = 1;
|
||||
const objName1 = "o1";
|
||||
const objName2 = "o2";
|
||||
const idxName1 = "i1";
|
||||
const idxName2 = "i2";
|
||||
const idxKeyPathProp = "idx";
|
||||
const objDataProp = "data";
|
||||
const objData = "1234567890";
|
||||
const objDataCount = 5;
|
||||
const loopCount = 100;
|
||||
|
||||
let testGenerator = testSteps();
|
||||
|
||||
function testSteps()
|
||||
{
|
||||
let req = indexedDB.open(dbName, dbVersion);
|
||||
req.onerror = errorHandler;
|
||||
req.onupgradeneeded = grabEventAndContinueHandler;
|
||||
req.onsuccess = grabEventAndContinueHandler;
|
||||
|
||||
let event = yield undefined;
|
||||
|
||||
is(event.type, "upgradeneeded", "Got upgradeneeded event");
|
||||
|
||||
let db = event.target.result;
|
||||
|
||||
let objectStore1 = db.createObjectStore(objName1);
|
||||
objectStore1.createIndex(idxName1, idxKeyPathProp);
|
||||
|
||||
let objectStore2 = db.createObjectStore(objName2);
|
||||
objectStore2.createIndex(idxName2, idxKeyPathProp);
|
||||
|
||||
for (let i = 0; i < objDataCount; i++) {
|
||||
var data = { };
|
||||
data[objDataProp] = objData;
|
||||
data[idxKeyPathProp] = objDataCount - i - 1;
|
||||
|
||||
objectStore1.add(data, i);
|
||||
objectStore2.add(data, i);
|
||||
}
|
||||
|
||||
event = yield undefined;
|
||||
|
||||
is(event.type, "success", "Got success event");
|
||||
|
||||
doReadOnlyTransaction(db, 0, loopCount);
|
||||
doReadWriteTransaction(db, 0, loopCount);
|
||||
|
||||
// Wait for readonly and readwrite transaction loops to complete.
|
||||
yield undefined;
|
||||
yield undefined;
|
||||
|
||||
finishTest();
|
||||
yield undefined;
|
||||
}
|
||||
|
||||
function doReadOnlyTransaction(db, key, remaining)
|
||||
{
|
||||
if (!remaining) {
|
||||
info("Finished all readonly transactions");
|
||||
continueToNextStep();
|
||||
return;
|
||||
}
|
||||
|
||||
info("Starting readonly transaction for key " + key + ", " + remaining +
|
||||
" loops left");
|
||||
|
||||
let objectStore = db.transaction(objName1, "readonly").objectStore(objName1);
|
||||
let index = objectStore.index(idxName1);
|
||||
|
||||
index.openKeyCursor(key, "prev").onsuccess = function(event) {
|
||||
let cursor = event.target.result;
|
||||
ok(cursor, "Got readonly cursor");
|
||||
|
||||
objectStore.get(cursor.primaryKey).onsuccess = function(event) {
|
||||
if (++key == objDataCount) {
|
||||
key = 0;
|
||||
}
|
||||
doReadOnlyTransaction(db, key, remaining - 1);
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
function doReadWriteTransaction(db, key, remaining)
|
||||
{
|
||||
if (!remaining) {
|
||||
info("Finished all readwrite transactions");
|
||||
continueToNextStep();
|
||||
return;
|
||||
}
|
||||
|
||||
info("Starting readwrite transaction for key " + key + ", " + remaining +
|
||||
" loops left");
|
||||
|
||||
let objectStore = db.transaction(objName2, "readwrite").objectStore(objName2);
|
||||
objectStore.openCursor(key).onsuccess = function(event) {
|
||||
let cursor = event.target.result;
|
||||
ok(cursor, "Got readwrite cursor");
|
||||
|
||||
let value = cursor.value;
|
||||
value[idxKeyPathProp]++;
|
||||
|
||||
cursor.update(value).onsuccess = function(event) {
|
||||
if (++key == objDataCount) {
|
||||
key = 0;
|
||||
}
|
||||
doReadWriteTransaction(db, key, remaining - 1);
|
||||
}
|
||||
};
|
||||
}
|
|
@ -0,0 +1,115 @@
|
|||
/**
|
||||
* Any copyright is dedicated to the Public Domain.
|
||||
* http://creativecommons.org/publicdomain/zero/1.0/
|
||||
*/
|
||||
|
||||
let testGenerator = testSteps();
|
||||
|
||||
function testSteps()
|
||||
{
|
||||
const dbName = ("window" in this) ? window.location.pathname : "test";
|
||||
const objName1 = "foo";
|
||||
const objName2 = "bar";
|
||||
const data1 = "1234567890";
|
||||
const data2 = "0987654321";
|
||||
const dataCount = 500;
|
||||
|
||||
let request = indexedDB.open(dbName, 1);
|
||||
request.onerror = errorHandler;
|
||||
request.onupgradeneeded = grabEventAndContinueHandler;
|
||||
|
||||
let event = yield undefined;
|
||||
|
||||
is(event.type, "upgradeneeded", "Got upgradeneeded");
|
||||
|
||||
request.onupgradeneeded = errorHandler;
|
||||
request.onsuccess = grabEventAndContinueHandler;
|
||||
|
||||
let db = request.result;
|
||||
|
||||
let objectStore1 = db.createObjectStore(objName1, { autoIncrement: true });
|
||||
let objectStore2 = db.createObjectStore(objName2, { autoIncrement: true });
|
||||
|
||||
info("Created object stores, adding data");
|
||||
|
||||
for (let i = 0; i < dataCount; i++) {
|
||||
objectStore1.add(data1);
|
||||
objectStore2.add(data2);
|
||||
}
|
||||
|
||||
info("Done adding data");
|
||||
|
||||
event = yield undefined;
|
||||
|
||||
is(event.type, "success", "Got success");
|
||||
|
||||
let readResult = null;
|
||||
let readError = null;
|
||||
let writeAborted = false;
|
||||
|
||||
info("Creating readwrite transaction");
|
||||
|
||||
objectStore1 = db.transaction(objName1, "readwrite").objectStore(objName1);
|
||||
objectStore1.openCursor().onsuccess = grabEventAndContinueHandler;
|
||||
|
||||
event = yield undefined;
|
||||
|
||||
let cursor = event.target.result;
|
||||
is(cursor.value, data1, "Got correct data for readwrite transaction");
|
||||
|
||||
info("Modifying object store on readwrite transaction");
|
||||
|
||||
cursor.update(data2);
|
||||
cursor.continue();
|
||||
|
||||
event = yield undefined;
|
||||
|
||||
info("Done modifying object store on readwrite transaction, creating " +
|
||||
"readonly transaction");
|
||||
|
||||
objectStore2 = db.transaction(objName2, "readonly").objectStore(objName2);
|
||||
request = objectStore2.getAll();
|
||||
request.onsuccess = function(event) {
|
||||
readResult = event.target.result;
|
||||
is(readResult.length,
|
||||
dataCount,
|
||||
"Got correct number of results on readonly transaction");
|
||||
for (let i = 0; i < readResult.length; i++) {
|
||||
is(readResult[i], data2, "Got correct data for readonly transaction");
|
||||
}
|
||||
if (writeAborted) {
|
||||
continueToNextStep();
|
||||
}
|
||||
};
|
||||
request.onerror = function(event) {
|
||||
readResult = null;
|
||||
readError = event.target.error;
|
||||
|
||||
ok(false, "Got read error: " + readError.name);
|
||||
event.preventDefault();
|
||||
|
||||
if (writeAborted) {
|
||||
continueToNextStep();
|
||||
}
|
||||
}
|
||||
|
||||
cursor = event.target.result;
|
||||
is(cursor.value, data1, "Got correct data for readwrite transaction");
|
||||
|
||||
info("Aborting readwrite transaction");
|
||||
|
||||
cursor.source.transaction.abort();
|
||||
writeAborted = true;
|
||||
|
||||
if (!readError && !readResult) {
|
||||
info("Waiting for readonly transaction to complete");
|
||||
yield undefined;
|
||||
}
|
||||
|
||||
ok(readResult, "Got result from readonly transaction");
|
||||
is(readError, null, "No read error");
|
||||
is(writeAborted, true, "Aborted readwrite transaction");
|
||||
|
||||
finishTest();
|
||||
yield undefined;
|
||||
}
|
|
@ -64,6 +64,8 @@ skip-if = toolkit == 'android' # bug 1079278
|
|||
[test_setVersion_events.js]
|
||||
[test_setVersion_exclusion.js]
|
||||
[test_success_events_after_abort.js]
|
||||
[test_table_locks.js]
|
||||
[test_table_rollback.js]
|
||||
[test_traffic_jam.js]
|
||||
[test_transaction_abort.js]
|
||||
[test_transaction_abort_hang.js]
|
||||
|
|
Загрузка…
Ссылка в новой задаче