Commit 5628bbbc authored by preetwinder's avatar preetwinder

use common functions for indexeddb

parent 22094762
...@@ -59,6 +59,11 @@ ${LINTDIR}/queries/query.js: ${SRCDIR}/queries/query.js ...@@ -59,6 +59,11 @@ ${LINTDIR}/queries/query.js: ${SRCDIR}/queries/query.js
${JSLINT} ${LINTOPTS} $< ${JSLINT} ${LINTOPTS} $<
@cat $< > $@ @cat $< > $@
${LINTDIR}/indexeddb/indexeddb.js: ${SRCDIR}/indexeddb/indexeddb.js
@mkdir -p $(@D)
${JSLINT} ${LINTOPTS} $<
@cat $< > $@
${LINTDIR}/${TESTDIR}/jio.storage/%.js: ${TESTDIR}/jio.storage/%.js ${LINTDIR}/${TESTDIR}/jio.storage/%.js: ${TESTDIR}/jio.storage/%.js
@mkdir -p $(@D) @mkdir -p $(@D)
${JSLINT} ${LINTOPTS} --predef QUnit --predef RSVP --predef jIO $< ${JSLINT} ${LINTOPTS} --predef QUnit --predef RSVP --predef jIO $<
...@@ -104,6 +109,7 @@ lint: $(patsubst ${TESTDIR}/jio.storage/%.js, ${LINTDIR}/${TESTDIR}/jio.storage/ ...@@ -104,6 +109,7 @@ lint: $(patsubst ${TESTDIR}/jio.storage/%.js, ${LINTDIR}/${TESTDIR}/jio.storage/
$(patsubst ${EXAMPLEDIR}/%.js, ${LINTDIR}/${EXAMPLEDIR}/%.js, $(wildcard ${EXAMPLEDIR}/*.js)) \ $(patsubst ${EXAMPLEDIR}/%.js, ${LINTDIR}/${EXAMPLEDIR}/%.js, $(wildcard ${EXAMPLEDIR}/*.js)) \
${LINTDIR}/queries/query.js \ ${LINTDIR}/queries/query.js \
${LINTDIR}/jio.date/jiodate.js \ ${LINTDIR}/jio.date/jiodate.js \
${LINTDIR}/indexeddb/indexeddb.js \
${LINTDIR}/jio.js \ ${LINTDIR}/jio.js \
${LINTDIR}/node/jio.js \ ${LINTDIR}/node/jio.js \
${LINTDIR}/${TESTDIR}/node.js \ ${LINTDIR}/${TESTDIR}/node.js \
...@@ -129,6 +135,7 @@ ${JIOVERSION}: ${EXTERNALDIR}/URI.js \ ...@@ -129,6 +135,7 @@ ${JIOVERSION}: ${EXTERNALDIR}/URI.js \
${SRCDIR}/queries/parser-end.js \ ${SRCDIR}/queries/parser-end.js \
${SRCDIR}/queries/query.js \ ${SRCDIR}/queries/query.js \
${SRCDIR}/jio.date/jiodate.js \ ${SRCDIR}/jio.date/jiodate.js \
${SRCDIR}/indexeddb/indexeddb.js \
${SRCDIR}/jio.js \ ${SRCDIR}/jio.js \
${EXTERNALDIR}/rusha.js \ ${EXTERNALDIR}/rusha.js \
${SRCDIR}/jio.storage/replicatestorage.js \ ${SRCDIR}/jio.storage/replicatestorage.js \
......
<!DOCTYPE html>
<!--
Copyright 2019, Nexedi SA
This program is free software: you can Use, Study, Modify and Redistribute
it under the terms of the GNU General Public License version 3, or (at your
option) any later version, as published by the Free Software Foundation.
You can also Link and Combine this program with other software covered by
the terms of any of the Free Software licenses or any of the Open Source
Initiative approved licenses and Convey the resulting work. Corresponding
source of such a combination shall include the source code for all other
software used.
This program is distributed WITHOUT ANY WARRANTY; without even the implied
warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
See COPYING file for full licensing terms.
See https://www.nexedi.com/licensing for rationale and options.
-->
<html lang="en">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>jIO Query Performance test</title>
<script src="../external/rsvp-2.0.4.js"></script>
<script src="../external/renderjs-latest.js"></script>
<script src="../dist/jio-latest.js"></script>
<!-- <link rel="stylesheet" href="../external/qunit.css" type="text/css" media="screen"/> -->
<!-- <script src="../external/qunit.js" type="text/javascript"></script> -->
<script src="index2_benchmark.js"></script>
</head>
<body>
<h1> Testing index2 query </h1>
<a>Test</a>
<p></p>
<div></div>
</body>
</html>
/*global performance, String*/
(function (window, jIO, rJS) {
"use strict";
var test_count = 15;
/*function randomi(limit) {
return Math.floor(Math.random() * Math.floor(limit));
}
function randomSentence(length) {
var alphabet = ['a', 'b', 'c', 'd', 'e', ' ', 'f', 'g', 'h', 'i', 'j', 'k',
' ', 'l', 'm', 'n', 'o', ' ', 'p', 'q', 'r', 's', 't', ' ',
'u', 'v', 'w', ' ', 'x', 'y', 'z', ' '], sentence = '', z;
for (z = 0; z < length; z += 1) {
sentence += alphabet[randomi(alphabet.length - 1)];
}
return sentence;
}
function randomSentenceArray(sentence_length, array_length) {
var y, sentence_array = [];
for (y = 0; y < array_length; y += 1) {
sentence_array.push(randomSentence(sentence_length));
}
return sentence_array;
}*/
function get_fake_data_values2(i) {
if (i === 0 || i === 1 || i === 2) {
return {'url': 'renderjs.com', 'name': 'erp5', 'user': 'preet'};
}
if (i === 3 || i === 4) {
return {'url': 'erp5.com', 'name': 'erp5', 'user': 'test'};
}
if (i === 5 || i === 6 || i === 7) {
return {'url': 'nexedi.com', 'name': 'nexedi', 'user': 'prabetcder'};
}
if (i === 10 || i === 11) {
return {'url': 'vifib.com', 'name': 'renderjs', 'user': 'preetwinder'};
}
if (i === 12 || i === 13) {
return {'url': 'renderjs.com', 'name': 'jio', 'user': 'obscure'};
}
return {'url': 'jio.nexedi.com', 'name': 'jio', 'user': 'praounsteter'};
}
/*function get_fake_data_values(i) {
var data_value = {
'id': i,
'url': 'https://streetsite.com/profiles/' + i,
'pic_url': 'https://cdn.streetsite.com/pictures/saoteuhcu/' + i,
'short_description': randomSentence(10 + randomi(40)),
'description': randomSentence(randomi(250)),
'comments': randomSentenceArray(randomi(500), randomi(20))
};
if (i === 9900) {
data_value.short_description = 'test';
}
if (i === 7500) {
data_value.short_description = 'preet';
}
if (i === 5400) {
data_value.short_description = 'obscure';
}
if (i === 3200) {
data_value.short_description = 'precise';
}
if (i === 1200) {
data_value.short_description = 'environ';
}
return data_value;
}*/
/* function sequential_test(i, storage) {
if (i < test_count) {
var data_value = {
'id': i,
'url': 'https://streetsite.com/profiles/' + i,
'pic_url': 'https://cdn.streetsite.com/pictures/saoteuhcu/' + i,
'short_description': randomSentence(10 + randomi(40)),
'description': randomSentence(randomi(250)),
'comments': randomSentenceArray(randomi(500), randomi(20))
};
if (i === 99000) {
data_value.short_description = 'test';
}
if (i % 100 === 0) {
data_value.short_description = 'preet';
}
if (i % 1000 === 0) {
data_value.short_description = 'obscure';
}
if (i === 32000) {
data_value.short_description = 'precise precise precise';
}
if (i === 120000) {
data_value.short_description = 'environ';
}
return storage.put(String(i), data_value)
.then(function () {
if (i % 1000 === 0) {
console.log(i);
}
data_value = null;
return sequential_test(i + 1, storage);
});
}
return;
}*/
rJS(window)
.declareService(function () {
var storage = jIO.createJIO({
type: "index2",
database: "index2test2",
index_keys: ["user", "name", "url"],
sub_storage: {
type: "indexeddb",
database: "index2testdata2",
}
}), promise_list = [], i, time;
console.log('Staring to write ' + test_count + ' documents');
//sequential_test(0, storage);
for (i = 0; i < test_count; i += 1) {
promise_list.push(storage.put(String(i), get_fake_data_values2(i)));
}
promise_list.push(storage.put('325', get_fake_data_values2(325)));
time = performance.now();
return RSVP.all(promise_list)
.then(function () {
console.log('Time to write - ', (performance.now() - time));
console.log('Starting queries');
console.log('Query 1');
time = performance.now();
return storage.allDocs({query: "user:preetwinder"});
})
.then(function (result) {
console.log('Time to query 1 - ', (performance.now() - time));
console.log(result);
console.log('Query 2');
time = performance.now();
return storage.allDocs({query: 'user:preet'});
})
.then(function (result) {
console.log('Time to query 2 - ', (performance.now() - time));
console.log(result);
console.log('Query 3');
time = performance.now();
return storage.allDocs({query: "(name:jio OR url:nexedi.com" +
") AND user:obscure"});
})
.then(function (result) {
console.log('Time to query 3 - ', performance.now() - time);
console.log(result);
console.log('Query 4');
time = performance.now();
return storage.allDocs({query: 'name:not'});
})
.then(function (result) {
console.log('Time to query 4 - ', performance.now() - time);
console.log(result);
});
});
}(window, jIO, rJS));
\ No newline at end of file
/*
* Copyright 2019, Nexedi SA
*
* This program is free software: you can Use, Study, Modify and Redistribute
* it under the terms of the GNU General Public License version 3, or (at your
* option) any later version, as published by the Free Software Foundation.
*
* You can also Link and Combine this program with other software covered by
* the terms of any of the Free Software licenses or any of the Open Source
* Initiative approved licenses and Convey the resulting work. Corresponding
* source of such a combination shall include the source code for all other
* software used.
*
* This program is distributed WITHOUT ANY WARRANTY; without even the implied
* warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
*
* See COPYING file for full licensing terms.
* See https://www.nexedi.com/licensing for rationale and options.
*/
/*jslint nomen: true */
/*global indexedDB, RSVP, IDBOpenDBRequest, DOMError, Event, Set, DOMException,
window*/
(function (indexedDB, RSVP, IDBOpenDBRequest, DOMError, DOMException) {
"use strict";
function waitForIDBRequest(request) {
return new RSVP.Promise(function (resolve, reject) {
request.onerror = reject;
request.onsuccess = resolve;
});
}
function waitForAllSynchronousCursor(request, callback) {
var force_cancellation = false;
function canceller() {
force_cancellation = true;
}
function resolver(resolve, reject) {
request.onerror = reject;
request.onsuccess = function (evt) {
var cursor = evt.target.result;
if (cursor && !force_cancellation) {
try {
callback(cursor);
} catch (error) {
reject(error);
}
// continue to next iteration
cursor["continue"]();
} else {
resolve();
}
};
}
return new RSVP.Promise(resolver, canceller);
}
function waitForOpenIndexedDB(db_name, version, upgrade_handler, callback) {
var request;
function canceller() {
if ((request !== undefined) && (request.result !== undefined)) {
request.result.close();
}
}
function resolver(resolve, reject) {
// Open DB //
request = indexedDB.open(db_name, version);
request.onerror = function (error) {
canceller();
if ((error !== undefined) &&
(error.target instanceof IDBOpenDBRequest) &&
((error.target.error instanceof DOMError) ||
(error.target.error instanceof DOMException))) {
reject("Connection to: " + db_name + " failed: " +
error.target.error.message);
} else {
reject(error);
}
};
request.onabort = function () {
canceller();
reject("Aborting connection to: " + db_name);
};
request.ontimeout = function () {
reject("Connection to: " + db_name + " timeout");
};
request.onblocked = function () {
canceller();
reject("Connection to: " + db_name + " was blocked");
};
// Create DB if necessary //
request.onupgradeneeded = upgrade_handler;
request.onversionchange = function () {
canceller();
reject(db_name + " was upgraded");
};
request.onsuccess = function () {
var result;
try {
result = callback(request.result);
} catch (error) {
reject(error);
}
return new RSVP.Queue()
.push(function () {
return result;
})
.push(function (final_result) {
canceller();
resolve(final_result);
}, function (error) {
canceller();
reject(error);
});
};
}
return new RSVP.Promise(resolver, canceller);
}
function waitForTransaction(db, stores, flag, callback) {
var tx = db.transaction(stores, flag);
function canceller() {
try {
tx.abort();
} catch (unused) {
// Transaction already finished
return;
}
}
function resolver(resolve, reject) {
var result;
try {
result = callback(tx);
} catch (error) {
reject(error);
}
tx.oncomplete = function () {
return new RSVP.Queue()
.push(function () {
return result;
})
.push(resolve, function (error) {
canceller();
reject(error);
});
};
tx.onerror = function (error) {
canceller();
reject(error);
};
tx.onabort = function (evt) {
reject(evt.target);
};
return tx;
}
return new RSVP.Promise(resolver, canceller);
}
window.waitForTransaction = waitForTransaction;
window.waitForOpenIndexedDB = waitForOpenIndexedDB;
window.waitForIDBRequest = waitForIDBRequest;
window.waitForAllSynchronousCursor = waitForAllSynchronousCursor;
}(indexedDB, RSVP, IDBOpenDBRequest, DOMError, DOMException));
\ No newline at end of file
...@@ -42,11 +42,11 @@ ...@@ -42,11 +42,11 @@
*/ */
/*jslint nomen: true */ /*jslint nomen: true */
/*global indexedDB, jIO, RSVP, Blob, Math, IDBKeyRange, IDBOpenDBRequest, /*global jIO, RSVP, Blob, Math, IDBKeyRange, Event, waitForOpenIndexedDB,
DOMError, Event*/ waitForIDBRequest, waitForAllSynchronousCursor, waitForTransaction*/
(function (indexedDB, jIO, RSVP, Blob, Math, IDBKeyRange, IDBOpenDBRequest, (function (jIO, RSVP, Blob, Math, IDBKeyRange, waitForOpenIndexedDB,
DOMError) { waitForIDBRequest, waitForAllSynchronousCursor, waitForTransaction) {
"use strict"; "use strict";
// Read only as changing it can lead to data corruption // Read only as changing it can lead to data corruption
...@@ -96,143 +96,6 @@ ...@@ -96,143 +96,6 @@
store.createIndex("_id", "_id", {unique: false}); store.createIndex("_id", "_id", {unique: false});
} }
function waitForOpenIndexedDB(db_name, callback) {
var request;
function canceller() {
if ((request !== undefined) && (request.result !== undefined)) {
request.result.close();
}
}
function resolver(resolve, reject) {
// Open DB //
request = indexedDB.open(db_name);
request.onerror = function (error) {
canceller();
if ((error !== undefined) &&
(error.target instanceof IDBOpenDBRequest) &&
(error.target.error instanceof DOMError)) {
reject("Connection to: " + db_name + " failed: " +
error.target.error.message);
} else {
reject(error);
}
};
request.onabort = function () {
canceller();
reject("Aborting connection to: " + db_name);
};
request.ontimeout = function () {
reject("Connection to: " + db_name + " timeout");
};
request.onblocked = function () {
canceller();
reject("Connection to: " + db_name + " was blocked");
};
// Create DB if necessary //
request.onupgradeneeded = handleUpgradeNeeded;
request.onversionchange = function () {
canceller();
reject(db_name + " was upgraded");
};
request.onsuccess = function () {
var result;
try {
result = callback(request.result);
} catch (error) {
reject(error);
}
return new RSVP.Queue()
.push(function () {
return result;
})
.push(function (final_result) {
canceller();
resolve(final_result);
}, function (error) {
canceller();
reject(error);
});
};
}
return new RSVP.Promise(resolver, canceller);
}
function waitForTransaction(db, stores, flag, callback) {
var tx = db.transaction(stores, flag);
function canceller() {
try {
tx.abort();
} catch (unused) {
// Transaction already finished
return;
}
}
function resolver(resolve, reject) {
var result;
try {
result = callback(tx);
} catch (error) {
reject(error);
}
tx.oncomplete = function () {
return new RSVP.Queue()
.push(function () {
return result;
})
.push(resolve, function (error) {
canceller();
reject(error);
});
};
tx.onerror = reject;
tx.onabort = reject;
}
return new RSVP.Promise(resolver, canceller);
}
function waitForIDBRequest(request) {
return new RSVP.Promise(function (resolve, reject) {
request.onerror = reject;
request.onsuccess = resolve;
});
}
function waitForAllSynchronousCursor(request, callback) {
var force_cancellation = false;
function canceller() {
force_cancellation = true;
}
function resolver(resolve, reject) {
request.onerror = reject;
request.onsuccess = function (evt) {
var cursor = evt.target.result;
if (cursor && !force_cancellation) {
try {
callback(cursor);
} catch (error) {
reject(error);
}
// continue to next iteration
cursor["continue"]();
} else {
resolve();
}
};
}
return new RSVP.Promise(resolver, canceller);
}
IndexedDBStorage.prototype.buildQuery = function (options) { IndexedDBStorage.prototype.buildQuery = function (options) {
var result_list = [], var result_list = [],
context = this; context = this;
...@@ -254,21 +117,22 @@ ...@@ -254,21 +117,22 @@
return new RSVP.Queue() return new RSVP.Queue()
.push(function () { .push(function () {
return waitForOpenIndexedDB(context._database_name, function (db) { return waitForOpenIndexedDB(context._database_name, undefined,
return waitForTransaction(db, ["metadata"], "readonly", handleUpgradeNeeded, function (db) {
return waitForTransaction(db, ["metadata"], "readonly",
function (tx) { function (tx) {
if (options.include_docs === true) { if (options.include_docs === true) {
return waitForAllSynchronousCursor(
tx.objectStore("metadata").index("_id").openCursor(),
pushIncludedMetadata
);
}
return waitForAllSynchronousCursor( return waitForAllSynchronousCursor(
tx.objectStore("metadata").index("_id").openCursor(), tx.objectStore("metadata").index("_id").openKeyCursor(),
pushIncludedMetadata pushMetadata
); );
} });
return waitForAllSynchronousCursor( });
tx.objectStore("metadata").index("_id").openKeyCursor(),
pushMetadata
);
});
});
}) })
.push(function () { .push(function () {
return result_list; return result_list;
...@@ -279,12 +143,13 @@ ...@@ -279,12 +143,13 @@
var context = this; var context = this;
return new RSVP.Queue() return new RSVP.Queue()
.push(function () { .push(function () {
return waitForOpenIndexedDB(context._database_name, function (db) { return waitForOpenIndexedDB(context._database_name, undefined,
return waitForTransaction(db, ["metadata"], "readonly", handleUpgradeNeeded, function (db) {
function (tx) { return waitForTransaction(db, ["metadata"], "readonly",
return waitForIDBRequest(tx.objectStore("metadata").get(id)); function (tx) {
}); return waitForIDBRequest(tx.objectStore("metadata").get(id));
}); });
});
}) })
.push(function (evt) { .push(function (evt) {
if (evt.target.result) { if (evt.target.result) {
...@@ -307,19 +172,20 @@ ...@@ -307,19 +172,20 @@
return new RSVP.Queue() return new RSVP.Queue()
.push(function () { .push(function () {
return waitForOpenIndexedDB(context._database_name, function (db) { return waitForOpenIndexedDB(context._database_name, undefined,
return waitForTransaction(db, ["metadata", "attachment"], "readonly", handleUpgradeNeeded, function (db) {
function (tx) { return waitForTransaction(db, ["metadata", "attachment"],
return RSVP.all([ "readonly", function (tx) {
waitForIDBRequest(tx.objectStore("metadata").get(id)), return RSVP.all([
waitForAllSynchronousCursor( waitForIDBRequest(tx.objectStore("metadata").get(id)),
tx.objectStore("attachment").index("_id") waitForAllSynchronousCursor(
.openKeyCursor(IDBKeyRange.only(id)), tx.objectStore("attachment").index("_id")
addEntry .openKeyCursor(IDBKeyRange.only(id)),
) addEntry
]); )
}); ]);
}); });
});
}) })
.push(function (result_list) { .push(function (result_list) {
var evt = result_list[0]; var evt = result_list[0];
...@@ -336,56 +202,58 @@ ...@@ -336,56 +202,58 @@
}; };
IndexedDBStorage.prototype.put = function (id, metadata) { IndexedDBStorage.prototype.put = function (id, metadata) {
return waitForOpenIndexedDB(this._database_name, function (db) { return waitForOpenIndexedDB(this._database_name, undefined,
return waitForTransaction(db, ["metadata"], "readwrite", handleUpgradeNeeded, function (db) {
function (tx) { return waitForTransaction(db, ["metadata"], "readwrite",
return waitForIDBRequest(tx.objectStore("metadata").put({ function (tx) {
"_id": id, return waitForIDBRequest(tx.objectStore("metadata").put({
"doc": metadata "_id": id,
})); "doc": metadata
}); }));
}); });
});
}; };
IndexedDBStorage.prototype.remove = function (id) { IndexedDBStorage.prototype.remove = function (id) {
return waitForOpenIndexedDB(this._database_name, function (db) { return waitForOpenIndexedDB(this._database_name, undefined,
return waitForTransaction(db, ["metadata", "attachment", "blob"], handleUpgradeNeeded, function (db) {
"readwrite", function (tx) { return waitForTransaction(db, ["metadata", "attachment", "blob"],
"readwrite", function (tx) {
var promise_list = [],
metadata_store = tx.objectStore("metadata"), var promise_list = [],
attachment_store = tx.objectStore("attachment"), metadata_store = tx.objectStore("metadata"),
blob_store = tx.objectStore("blob"); attachment_store = tx.objectStore("attachment"),
blob_store = tx.objectStore("blob");
function deleteAttachment(cursor) {
promise_list.push(
waitForIDBRequest(attachment_store.delete(cursor.primaryKey))
);
}
function deleteBlob(cursor) {
promise_list.push(
waitForIDBRequest(blob_store.delete(cursor.primaryKey))
);
}
return RSVP.all([ function deleteAttachment(cursor) {
waitForIDBRequest(metadata_store.delete(id)), promise_list.push(
waitForAllSynchronousCursor( waitForIDBRequest(attachment_store.delete(cursor.primaryKey))
attachment_store.index("_id") );
.openKeyCursor(IDBKeyRange.only(id)), }
deleteAttachment function deleteBlob(cursor) {
), promise_list.push(
waitForAllSynchronousCursor( waitForIDBRequest(blob_store.delete(cursor.primaryKey))
blob_store.index("_id") );
.openKeyCursor(IDBKeyRange.only(id)), }
deleteBlob
), return RSVP.all([
]) waitForIDBRequest(metadata_store.delete(id)),
.then(function () { waitForAllSynchronousCursor(
return RSVP.all(promise_list); attachment_store.index("_id")
}); .openKeyCursor(IDBKeyRange.only(id)),
}); deleteAttachment
}); ),
waitForAllSynchronousCursor(
blob_store.index("_id")
.openKeyCursor(IDBKeyRange.only(id)),
deleteBlob
),
])
.then(function () {
return RSVP.all(promise_list);
});
});
});
}; };
IndexedDBStorage.prototype.getAttachment = function (id, name, options) { IndexedDBStorage.prototype.getAttachment = function (id, name, options) {
...@@ -416,65 +284,66 @@ ...@@ -416,65 +284,66 @@
return new RSVP.Queue() return new RSVP.Queue()
.push(function () { .push(function () {
return waitForOpenIndexedDB(db_name, function (db) { return waitForOpenIndexedDB(db_name, undefined,
return waitForTransaction(db, ["blob"], "readonly", handleUpgradeNeeded, function (db) {
function (tx) { return waitForTransaction(db, ["blob"], "readonly",
var key_path = buildKeyPath([id, name]), function (tx) {
blob_store = tx.objectStore("blob"), var key_path = buildKeyPath([id, name]),
start_index, blob_store = tx.objectStore("blob"),
end_index, start_index,
promise_list = []; end_index,
promise_list = [];
start_index = Math.floor(start / UNITE);
if (end !== undefined) { start_index = Math.floor(start / UNITE);
end_index = Math.floor(end / UNITE); if (end !== undefined) {
if (end % UNITE === 0) { end_index = Math.floor(end / UNITE);
end_index -= 1; if (end % UNITE === 0) {
} end_index -= 1;
} }
function getBlobKey(cursor) {
var index = parseInt(
cursor.primaryKey.slice(key_path.length + 1),
10
),
i;
if ((start !== 0) && (index < start_index)) {
// No need to fetch blobs at the start
return;
}
if ((end !== undefined) && (index > end_index)) {
// No need to fetch blobs at the end
return;
} }
i = index - start_index; function getBlobKey(cursor) {
// Extend array size var index = parseInt(
while (i > promise_list.length) { cursor.primaryKey.slice(key_path.length + 1),
promise_list.push(null); 10
i -= 1; ),
i;
if ((start !== 0) && (index < start_index)) {
// No need to fetch blobs at the start
return;
}
if ((end !== undefined) && (index > end_index)) {
// No need to fetch blobs at the end
return;
}
i = index - start_index;
// Extend array size
while (i > promise_list.length) {
promise_list.push(null);
i -= 1;
}
// Sort the blob by their index
promise_list.splice(
index - start_index,
0,
waitForIDBRequest(blob_store.get(cursor.primaryKey))
);
} }
// Sort the blob by their index
promise_list.splice(
index - start_index,
0,
waitForIDBRequest(blob_store.get(cursor.primaryKey))
);
}
// Get all blob keys to check if they must be fetched // Get all blob keys to check if they must be fetched
return waitForAllSynchronousCursor( return waitForAllSynchronousCursor(
blob_store.index("_id_attachment") blob_store.index("_id_attachment")
.openKeyCursor(IDBKeyRange.only([id, name])), .openKeyCursor(IDBKeyRange.only([id, name])),
getBlobKey getBlobKey
) )
.then(function () { .then(function () {
return RSVP.all(promise_list); return RSVP.all(promise_list);
}); });
}); });
}); });
}) })
.push(function (result_list) { .push(function (result_list) {
// No need to keep the IDB open // No need to keep the IDB open
...@@ -501,46 +370,47 @@ ...@@ -501,46 +370,47 @@
// Request the full blob // Request the full blob
return new RSVP.Queue() return new RSVP.Queue()
.push(function () { .push(function () {
return waitForOpenIndexedDB(db_name, function (db) { return waitForOpenIndexedDB(db_name, undefined, handleUpgradeNeeded,
return waitForTransaction(db, ["attachment", "blob"], "readonly", function (db) {
function (tx) { return waitForTransaction(db, ["attachment", "blob"], "readonly",
var key_path = buildKeyPath([id, name]), function (tx) {
attachment_store = tx.objectStore("attachment"), var key_path = buildKeyPath([id, name]),
blob_store = tx.objectStore("blob"); attachment_store = tx.objectStore("attachment"),
blob_store = tx.objectStore("blob");
function getBlob(cursor) { function getBlob(cursor) {
var index = parseInt( var index = parseInt(
cursor.primaryKey.slice(key_path.length + 1), cursor.primaryKey.slice(key_path.length + 1),
10 10
), ),
i = index; i = index;
// Extend array size // Extend array size
while (i > array_buffer_list.length) { while (i > array_buffer_list.length) {
array_buffer_list.push(null); array_buffer_list.push(null);
i -= 1; i -= 1;
}
// Sort the blob by their index
array_buffer_list.splice(
index,
0,
cursor.value.blob
);
} }
// Sort the blob by their index
array_buffer_list.splice( return RSVP.all([
index, // Get the attachment info (mime type)
0, waitForIDBRequest(attachment_store.get(
cursor.value.blob key_path
); )),
} // Get all needed blobs
waitForAllSynchronousCursor(
return RSVP.all([ blob_store.index("_id_attachment")
// Get the attachment info (mime type) .openCursor(IDBKeyRange.only([id, name])),
waitForIDBRequest(attachment_store.get( getBlob
key_path )
)), ]);
// Get all needed blobs });
waitForAllSynchronousCursor( });
blob_store.index("_id_attachment")
.openCursor(IDBKeyRange.only([id, name])),
getBlob
)
]);
});
});
}) })
.push(function (result_list) { .push(function (result_list) {
...@@ -591,103 +461,106 @@ ...@@ -591,103 +461,106 @@
handled_size += UNITE; handled_size += UNITE;
} }
return waitForOpenIndexedDB(db_name, function (db) { return waitForOpenIndexedDB(db_name, undefined, handleUpgradeNeeded,
return waitForTransaction(db, ["attachment", "blob"], "readwrite", function (db) {
function (tx) { return waitForTransaction(db, ["attachment", "blob"], "readwrite",
var blob_store, function (tx) {
promise_list, var blob_store,
delete_promise_list = [], promise_list,
key_path = buildKeyPath([id, name]), delete_promise_list = [],
i; key_path = buildKeyPath([id, name]),
// First write the attachment info on top of previous i;
promise_list = [ // First write the attachment info on top of previous
waitForIDBRequest(tx.objectStore("attachment").put({ promise_list = [
"_key_path": key_path, waitForIDBRequest(tx.objectStore("attachment").put({
"_id": id, "_key_path": key_path,
"_attachment": name, "_id": id,
"info": { "_attachment": name,
"content_type": blob.type, "info": {
"length": blob.size "content_type": blob.type,
} "length": blob.size
})) }
];
// Then, write all blob parts on top of previous
blob_store = tx.objectStore("blob");
for (i = 0; i < blob_part.length; i += 1) {
promise_list.push(
waitForIDBRequest(blob_store.put({
"_key_path": buildKeyPath([id, name, i]),
"_id" : id,
"_attachment" : name,
"_part" : i,
"blob": blob_part[i]
})) }))
); ];
} // Then, write all blob parts on top of previous
blob_store = tx.objectStore("blob");
for (i = 0; i < blob_part.length; i += 1) {
promise_list.push(
waitForIDBRequest(blob_store.put({
"_key_path": buildKeyPath([id, name, i]),
"_id" : id,
"_attachment" : name,
"_part" : i,
"blob": blob_part[i]
}))
);
}
function deleteEntry(cursor) { function deleteEntry(cursor) {
var index = parseInt( var index = parseInt(
cursor.primaryKey.slice(key_path.length + 1), cursor.primaryKey.slice(key_path.length + 1),
10 10
);
if (index >= blob_part.length) {
delete_promise_list.push(
waitForIDBRequest(blob_store.delete(cursor.primaryKey))
); );
if (index >= blob_part.length) {
delete_promise_list.push(
waitForIDBRequest(blob_store.delete(cursor.primaryKey))
);
}
} }
}
// Finally, remove all remaining blobs // Finally, remove all remaining blobs
promise_list.push( promise_list.push(
waitForAllSynchronousCursor( waitForAllSynchronousCursor(
blob_store.index("_id_attachment") blob_store.index("_id_attachment")
.openKeyCursor(IDBKeyRange.only([id, name])), .openKeyCursor(IDBKeyRange.only([id, name])),
deleteEntry deleteEntry
) )
); );
return RSVP.all(promise_list) return RSVP.all(promise_list)
.then(function () { .then(function () {
if (delete_promise_list.length) { if (delete_promise_list.length) {
return RSVP.all(delete_promise_list); return RSVP.all(delete_promise_list);
} }
}); });
}); });
}); });
}); });
}; };
IndexedDBStorage.prototype.removeAttachment = function (id, name) { IndexedDBStorage.prototype.removeAttachment = function (id, name) {
return waitForOpenIndexedDB(this._database_name, function (db) { return waitForOpenIndexedDB(this._database_name, undefined,
return waitForTransaction(db, ["attachment", "blob"], "readwrite", handleUpgradeNeeded, function (db) {
function (tx) { return waitForTransaction(db, ["attachment", "blob"], "readwrite",
var promise_list = [], function (tx) {
attachment_store = tx.objectStore("attachment"), var promise_list = [],
blob_store = tx.objectStore("blob"); attachment_store = tx.objectStore("attachment"),
blob_store = tx.objectStore("blob");
function deleteEntry(cursor) {
promise_list.push(
waitForIDBRequest(blob_store.delete(cursor.primaryKey))
);
}
return RSVP.all([ function deleteEntry(cursor) {
waitForIDBRequest( promise_list.push(
attachment_store.delete(buildKeyPath([id, name])) waitForIDBRequest(blob_store.delete(cursor.primaryKey))
), );
waitForAllSynchronousCursor( }
blob_store.index("_id_attachment")
.openKeyCursor(IDBKeyRange.only([id, name])), return RSVP.all([
deleteEntry waitForIDBRequest(
) attachment_store.delete(buildKeyPath([id, name]))
]) ),
.then(function () { waitForAllSynchronousCursor(
return RSVP.all(promise_list); blob_store.index("_id_attachment")
}); .openKeyCursor(IDBKeyRange.only([id, name])),
deleteEntry
)
])
.then(function () {
return RSVP.all(promise_list);
});
}); });
}); });
}; };
jIO.addStorage("indexeddb", IndexedDBStorage); jIO.addStorage("indexeddb", IndexedDBStorage);
}(indexedDB, jIO, RSVP, Blob, Math, IDBKeyRange, IDBOpenDBRequest, DOMError)); }(jIO, RSVP, Blob, Math, IDBKeyRange, waitForOpenIndexedDB, waitForIDBRequest,
waitForAllSynchronousCursor, waitForTransaction));
...@@ -18,11 +18,12 @@ ...@@ -18,11 +18,12 @@
* See https://www.nexedi.com/licensing for rationale and options. * See https://www.nexedi.com/licensing for rationale and options.
*/ */
/*jslint nomen: true */ /*jslint nomen: true */
/*global indexedDB, jIO, RSVP, IDBOpenDBRequest, DOMError, Event, /*global indexedDB, jIO, RSVP, Event, parseStringToObject, Set,
parseStringToObject, Set, DOMException*/ waitForTransaction, waitForAllSynchronousCursor, waitForIDBRequest,
waitForOpenIndexedDB*/
(function (indexedDB, jIO, RSVP, IDBOpenDBRequest, DOMError, (function (indexedDB, jIO, RSVP, parseStringToObject, waitForTransaction,
parseStringToObject, DOMException) { waitForAllSynchronousCursor, waitForIDBRequest, waitForOpenIndexedDB) {
"use strict"; "use strict";
function IndexStorage2(description) { function IndexStorage2(description) {
...@@ -74,40 +75,6 @@ ...@@ -74,40 +75,6 @@
return doc; return doc;
} }
function waitForIDBRequest(request) {
return new RSVP.Promise(function (resolve, reject) {
request.onerror = reject;
request.onsuccess = resolve;
});
}
function waitForAllSynchronousCursor(request, callback) {
var force_cancellation = false;
function canceller() {
force_cancellation = true;
}
function resolver(resolve, reject) {
request.onerror = reject;
request.onsuccess = function (evt) {
var cursor = evt.target.result;
if (cursor && !force_cancellation) {
try {
callback(cursor);
} catch (error) {
reject(error);
}
// continue to next iteration
cursor["continue"]();
} else {
resolve();
}
};
}
return new RSVP.Promise(resolver, canceller);
}
function getCursorResult(cursor, limit) { function getCursorResult(cursor, limit) {
var result = [], count = 0; var result = [], count = 0;
function pushLimitedMetadata(cursor) { function pushLimitedMetadata(cursor) {
...@@ -122,21 +89,37 @@ ...@@ -122,21 +89,37 @@
}); });
} }
function waitForRepairableOpenIndexedDB(db_name, version, index_keys,
sub_storage_description, signature_storage_name, upgrade_handler,
callback) {
var handleUpgradeNeeded = function (evt) {
return upgrade_handler(evt, index_keys, sub_storage_description,
signature_storage_name);
};
return waitForOpenIndexedDB(db_name, version, handleUpgradeNeeded,
callback);
}
function VirtualIDB(description) { function VirtualIDB(description) {
this._operations = description.operations; this._operations = description.operations;
} }
function virtualOperation(type, context, function_arguments) { function virtualOperation(type, context, function_arguments) {
var cancel_callback; var cancel_callback, operation = {}, promise;
function resolver(resolve, reject) { function resolver(resolve, reject) {
cancel_callback = reject; cancel_callback = reject;
context._operations.push({type: type, arguments: function_arguments, operation.type = type;
onsuccess: resolve, onerror: reject}); operation.arguments = function_arguments;
operation.onsuccess = resolve;
operation.onerror = reject;
} }
function canceller() { function canceller() {
cancel_callback(); cancel_callback();
} }
return new RSVP.Promise(resolver, canceller); promise = new RSVP.Promise(resolver, canceller);
operation.promise = promise;
context._operations.push(operation);
return promise;
} }
VirtualIDB.prototype.hasCapacity = function (name) { VirtualIDB.prototype.hasCapacity = function (name) {
...@@ -201,6 +184,9 @@ ...@@ -201,6 +184,9 @@
function processVirtualOperation(operation, store, index_keys, disable_get) { function processVirtualOperation(operation, store, index_keys, disable_get) {
var request, get_success_handler; var request, get_success_handler;
if (operation.promise.isRejected) {
return;
}
if (operation.type === "put") { if (operation.type === "put") {
request = store.put({ request = store.put({
id: operation.arguments[0], id: operation.arguments[0],
...@@ -234,8 +220,6 @@ ...@@ -234,8 +220,6 @@
} }
} }
var transaction_failure_reason;
function repairInTransaction(sub_storage_description, transaction, function repairInTransaction(sub_storage_description, transaction,
index_keys, signature_storage_name, clear_storage) { index_keys, signature_storage_name, clear_storage) {
var repair_promise, repeatUntilPromiseFulfilled, store, var repair_promise, repeatUntilPromiseFulfilled, store,
...@@ -269,7 +253,6 @@ ...@@ -269,7 +253,6 @@
} }
if (repair_promise.isRejected) { if (repair_promise.isRejected) {
transaction.abort(); transaction.abort();
transaction_failure_reason = repair_promise.rejectedReason;
return; return;
} }
if (repair_promise.isFulfilled) { if (repair_promise.isFulfilled) {
...@@ -279,7 +262,7 @@ ...@@ -279,7 +262,7 @@
store.get("inexistent"), next_continuation_resolve); store.get("inexistent"), next_continuation_resolve);
}; };
}; };
repeatUntilPromiseFulfilled(store.get("inexistent")); return repeatUntilPromiseFulfilled(store.get("inexistent"));
} }
function handleUpgradeNeeded(evt, index_keys, sub_storage_description, function handleUpgradeNeeded(evt, index_keys, sub_storage_description,
...@@ -319,122 +302,15 @@ ...@@ -319,122 +302,15 @@
} }
} }
function waitForOpenIndexedDB(db_name, version, index_keys,
sub_storage_description, signature_storage_name, callback) {
function resolver(resolve, reject) {
// Open DB //
var request = indexedDB.open(db_name, version);
request.onerror = function (error) {
var error_sub_message;
if (request.result) {
request.result.close();
}
if ((error !== undefined) &&
(error.target instanceof IDBOpenDBRequest) &&
((error.target.error instanceof DOMError) ||
(error.target.error instanceof DOMException))) {
error_sub_message = error.target.error.message;
if (transaction_failure_reason) {
error_sub_message += " " + transaction_failure_reason;
transaction_failure_reason = undefined;
}
reject("Connection to: " + db_name + " failed: " + error_sub_message);
} else {
reject(error);
}
};
request.onabort = function () {
request.result.close();
reject("Aborting connection to: " + db_name);
};
request.ontimeout = function () {
request.result.close();
reject("Connection to: " + db_name + " timeout");
};
request.onblocked = function () {
request.result.close();
reject("Connection to: " + db_name + " was blocked");
};
// Create DB if necessary //
request.onupgradeneeded = function (evt) {
handleUpgradeNeeded(evt, index_keys, sub_storage_description,
signature_storage_name);
};
request.onversionchange = function () {
request.result.close();
reject(db_name + " was upgraded");
};
request.onsuccess = function () {
return new RSVP.Queue()
.push(function () {
return callback(request.result);
})
.push(function (result) {
request.result.close();
resolve(result);
}, function (error) {
request.result.close();
reject(error);
});
};
}
return new RSVP.Promise(resolver);
}
function waitForTransaction(db, stores, flag, callback) {
var tx = db.transaction(stores, flag);
function canceller() {
try {
tx.abort();
} catch (unused) {
// Transaction already finished
return;
}
}
function resolver(resolve, reject) {
var result;
try {
result = callback(tx);
} catch (error) {
reject(error);
}
tx.oncomplete = function () {
return new RSVP.Queue()
.push(function () {
return result;
})
.push(resolve, function (error) {
canceller();
reject(error);
});
};
tx.onerror = function (error) {
canceller();
reject(error);
};
tx.onabort = function (evt) {
reject(evt.target);
};
return tx;
}
return new RSVP.Promise(resolver, canceller);
}
IndexStorage2.prototype._runQuery = function (key, value, limit) { IndexStorage2.prototype._runQuery = function (key, value, limit) {
var context = this; var context = this;
return RSVP.Queue() return RSVP.Queue()
.push(function () { .push(function () {
return waitForOpenIndexedDB(context._database_name, context._version, return waitForRepairableOpenIndexedDB(context._database_name,
context._index_keys, context._sub_storage_description, context._version, context._index_keys,
context._signature_storage_name, function (db) { context._sub_storage_description, context._signature_storage_name,
handleUpgradeNeeded, function (db) {
return waitForTransaction(db, ["index-store"], "readonly", return waitForTransaction(db, ["index-store"], "readonly",
function (tx) { function (tx) {
if (limit) { if (limit) {
...@@ -485,9 +361,9 @@ ...@@ -485,9 +361,9 @@
if (context._index_keys.length === 0) { if (context._index_keys.length === 0) {
return; return;
} }
return waitForOpenIndexedDB(context._database_name, context._version, return waitForRepairableOpenIndexedDB(context._database_name,
context._index_keys, context._sub_storage_description, context._version, context._index_keys, context._sub_storage_description,
context._signature_storage_name, function (db) { context._signature_storage_name, handleUpgradeNeeded, function (db) {
return waitForTransaction(db, ["index-store"], "readwrite", return waitForTransaction(db, ["index-store"], "readwrite",
function (tx) { function (tx) {
return waitForIDBRequest(tx.objectStore("index-store").put({ return waitForIDBRequest(tx.objectStore("index-store").put({
...@@ -521,9 +397,10 @@ ...@@ -521,9 +397,10 @@
var context = this; var context = this;
return context._sub_storage.remove(id) return context._sub_storage.remove(id)
.push(function () { .push(function () {
return waitForOpenIndexedDB(context._database_name, context._version, return waitForRepairableOpenIndexedDB(context._database_name,
context._index_keys, context._sub_storage_description, context._version, context._index_keys,
context._signature_storage_name, function (db) { context._sub_storage_description, context._signature_storage_name,
handleUpgradeNeeded, function (db) {
return waitForTransaction(db, ["index-store"], "readwrite", return waitForTransaction(db, ["index-store"], "readwrite",
function (tx) { function (tx) {
return waitForIDBRequest(tx.objectStore("index-store") return waitForIDBRequest(tx.objectStore("index-store")
...@@ -535,9 +412,9 @@ ...@@ -535,9 +412,9 @@
IndexStorage2.prototype.repair = function () { IndexStorage2.prototype.repair = function () {
var context = this; var context = this;
return waitForOpenIndexedDB(context._database_name, context._version, return waitForRepairableOpenIndexedDB(context._database_name,
context._index_keys, context._sub_storage_description, context._version, context._index_keys, context._sub_storage_description,
context._signature_storage_name, function (db) { context._signature_storage_name, handleUpgradeNeeded, function (db) {
return waitForTransaction(db, ["index-store"], "readwrite", return waitForTransaction(db, ["index-store"], "readwrite",
function (tx) { function (tx) {
return repairInTransaction(context._sub_storage_description, tx, return repairInTransaction(context._sub_storage_description, tx,
...@@ -560,5 +437,5 @@ ...@@ -560,5 +437,5 @@
}; };
jIO.addStorage("index2", IndexStorage2); jIO.addStorage("index2", IndexStorage2);
}(indexedDB, jIO, RSVP, IDBOpenDBRequest, DOMError, parseStringToObject, }(indexedDB, jIO, RSVP, parseStringToObject, waitForTransaction,
DOMException)); waitForAllSynchronousCursor, waitForIDBRequest, waitForOpenIndexedDB));
\ No newline at end of file \ No newline at end of file
...@@ -789,36 +789,6 @@ ...@@ -789,36 +789,6 @@
}); });
}); });
test("Repair fails", function () {
var context = this, chrome_error, firefox_error;
chrome_error = "Connection to: jio:index2_test failed: Version change " +
"transaction was aborted in upgradeneeded event handler. " +
"Error: Capacity 'buildQuery' is not implemented on 'dummystorage3'";
firefox_error = "Connection to: jio:index2_test failed: A request was " +
"aborted, for example through a call to IDBTransaction.abort. " +
"Error: Capacity 'buildQuery' is not implemented on 'dummystorage3'";
context.jio = jIO.createJIO({
type: "index2",
database: "index2_test",
index_keys: ["a", "c"],
sub_storage: {
type: "dummystorage3"
}
});
stop();
expect(1);
DummyStorage3.prototype.buildQuery = undefined;
context.jio.allDocs({query: "c: 'control'"})
.fail(function (error) {
ok(error === chrome_error || error === firefox_error);
})
.always(function () {
start();
});
});
test("Manual repair", function () { test("Manual repair", function () {
var context = this, fake_data; var context = this, fake_data;
context.jio = jIO.createJIO({ context.jio = jIO.createJIO({
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment