2015-02-20 23:57:00 +03:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Adapted from https://github.com/mozilla-b2g/gaia/blob/f09993563fb5fec4393eb71816ce76cb00463190/shared/js/async_storage.js
|
|
|
|
* (converted to use Promises instead of callbacks).
|
|
|
|
*
|
|
|
|
* This file defines an asynchronous version of the localStorage API, backed by
|
|
|
|
* an IndexedDB database. It creates a global asyncStorage object that has
|
|
|
|
* methods like the localStorage object.
|
|
|
|
*
|
|
|
|
* To store a value use setItem:
|
|
|
|
*
|
|
|
|
* asyncStorage.setItem("key", "value");
|
|
|
|
*
|
|
|
|
* This returns a promise in case you want confirmation that the value has been stored.
|
|
|
|
*
|
|
|
|
* asyncStorage.setItem("key", "newvalue").then(function() {
|
|
|
|
* console.log("new value stored");
|
|
|
|
* });
|
|
|
|
*
|
|
|
|
* To read a value, call getItem(), but note that you must wait for a promise
|
|
|
|
* resolution for the value to be retrieved.
|
|
|
|
*
|
|
|
|
* asyncStorage.getItem("key").then(function(value) {
|
|
|
|
* console.log("The value of key is:", value);
|
|
|
|
* });
|
|
|
|
*
|
|
|
|
* Note that unlike localStorage, asyncStorage does not allow you to store and
|
|
|
|
* retrieve values by setting and querying properties directly. You cannot just
|
|
|
|
* write asyncStorage.key; you have to explicitly call setItem() or getItem().
|
|
|
|
*
|
|
|
|
* removeItem(), clear(), length(), and key() are like the same-named methods of
|
|
|
|
* localStorage, and all return a promise.
|
|
|
|
*
|
|
|
|
* The asynchronous nature of getItem() makes it tricky to retrieve multiple
|
|
|
|
* values. But unlike localStorage, asyncStorage does not require the values you
|
|
|
|
* store to be strings. So if you need to save multiple values and want to
|
|
|
|
* retrieve them together, in a single asynchronous operation, just group the
|
|
|
|
* values into a single object. The properties of this object may not include
|
|
|
|
* DOM elements, but they may include things like Blobs and typed arrays.
|
|
|
|
*
|
|
|
|
*/
|
2016-08-03 22:21:11 +03:00
|
|
|
|
|
|
|
"use strict";
|
|
|
|
|
|
|
|
const DBNAME = "devtools-async-storage";
|
|
|
|
const DBVERSION = 1;
|
|
|
|
const STORENAME = "keyvaluepairs";
|
|
|
|
var db = null;
|
2015-02-20 23:57:00 +03:00
|
|
|
|
2016-08-03 22:21:11 +03:00
|
|
|
function withStore(type, onsuccess, onerror) {
|
|
|
|
if (db) {
|
2018-06-01 13:36:09 +03:00
|
|
|
const transaction = db.transaction(STORENAME, type);
|
|
|
|
const store = transaction.objectStore(STORENAME);
|
2016-08-03 22:21:11 +03:00
|
|
|
onsuccess(store);
|
|
|
|
} else {
|
2018-06-01 13:36:09 +03:00
|
|
|
const openreq = indexedDB.open(DBNAME, DBVERSION);
|
2016-08-03 22:21:11 +03:00
|
|
|
openreq.onerror = function withStoreOnError() {
|
|
|
|
onerror();
|
|
|
|
};
|
|
|
|
openreq.onupgradeneeded = function withStoreOnUpgradeNeeded() {
|
|
|
|
// First time setup: create an empty object store
|
|
|
|
openreq.result.createObjectStore(STORENAME);
|
|
|
|
};
|
|
|
|
openreq.onsuccess = function withStoreOnSuccess() {
|
|
|
|
db = openreq.result;
|
2018-06-01 13:36:09 +03:00
|
|
|
const transaction = db.transaction(STORENAME, type);
|
|
|
|
const store = transaction.objectStore(STORENAME);
|
2015-02-20 23:57:00 +03:00
|
|
|
onsuccess(store);
|
2016-08-03 22:21:11 +03:00
|
|
|
};
|
2015-02-20 23:57:00 +03:00
|
|
|
}
|
2016-08-03 22:21:11 +03:00
|
|
|
}
|
2015-02-20 23:57:00 +03:00
|
|
|
|
2016-08-03 22:21:11 +03:00
|
|
|
function getItem(itemKey) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
let req;
|
|
|
|
withStore("readonly", (store) => {
|
|
|
|
store.transaction.oncomplete = function onComplete() {
|
|
|
|
let value = req.result;
|
|
|
|
if (value === undefined) {
|
|
|
|
value = null;
|
|
|
|
}
|
|
|
|
resolve(value);
|
|
|
|
};
|
|
|
|
req = store.get(itemKey);
|
|
|
|
req.onerror = function getItemOnError() {
|
|
|
|
reject("Error in asyncStorage.getItem(): ", req.error.name);
|
|
|
|
};
|
|
|
|
}, reject);
|
|
|
|
});
|
|
|
|
}
|
2015-02-20 23:57:00 +03:00
|
|
|
|
2016-08-03 22:21:11 +03:00
|
|
|
function setItem(itemKey, value) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
withStore("readwrite", (store) => {
|
|
|
|
store.transaction.oncomplete = resolve;
|
2018-06-01 13:36:09 +03:00
|
|
|
const req = store.put(value, itemKey);
|
2016-08-03 22:21:11 +03:00
|
|
|
req.onerror = function setItemOnError() {
|
|
|
|
reject("Error in asyncStorage.setItem(): ", req.error.name);
|
|
|
|
};
|
|
|
|
}, reject);
|
|
|
|
});
|
|
|
|
}
|
2015-02-20 23:57:00 +03:00
|
|
|
|
2016-08-03 22:21:11 +03:00
|
|
|
function removeItem(itemKey) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
withStore("readwrite", (store) => {
|
|
|
|
store.transaction.oncomplete = resolve;
|
2018-06-01 13:36:09 +03:00
|
|
|
const req = store.delete(itemKey);
|
2016-08-03 22:21:11 +03:00
|
|
|
req.onerror = function removeItemOnError() {
|
|
|
|
reject("Error in asyncStorage.removeItem(): ", req.error.name);
|
|
|
|
};
|
|
|
|
}, reject);
|
|
|
|
});
|
|
|
|
}
|
2015-02-20 23:57:00 +03:00
|
|
|
|
2016-08-03 22:21:11 +03:00
|
|
|
function clear() {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
withStore("readwrite", (store) => {
|
|
|
|
store.transaction.oncomplete = resolve;
|
2018-06-01 13:36:09 +03:00
|
|
|
const req = store.clear();
|
2016-08-03 22:21:11 +03:00
|
|
|
req.onerror = function clearOnError() {
|
|
|
|
reject("Error in asyncStorage.clear(): ", req.error.name);
|
|
|
|
};
|
|
|
|
}, reject);
|
|
|
|
});
|
|
|
|
}
|
2015-02-20 23:57:00 +03:00
|
|
|
|
2016-08-03 22:21:11 +03:00
|
|
|
function length() {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
let req;
|
|
|
|
withStore("readonly", (store) => {
|
|
|
|
store.transaction.oncomplete = function onComplete() {
|
|
|
|
resolve(req.result);
|
|
|
|
};
|
|
|
|
req = store.count();
|
|
|
|
req.onerror = function lengthOnError() {
|
|
|
|
reject("Error in asyncStorage.length(): ", req.error.name);
|
|
|
|
};
|
|
|
|
}, reject);
|
|
|
|
});
|
|
|
|
}
|
2015-02-20 23:57:00 +03:00
|
|
|
|
2016-08-03 22:21:11 +03:00
|
|
|
function key(n) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
if (n < 0) {
|
|
|
|
resolve(null);
|
|
|
|
return;
|
|
|
|
}
|
2015-02-20 23:57:00 +03:00
|
|
|
|
2016-08-03 22:21:11 +03:00
|
|
|
let req;
|
|
|
|
withStore("readonly", (store) => {
|
|
|
|
store.transaction.oncomplete = function onComplete() {
|
2018-06-01 13:36:09 +03:00
|
|
|
const cursor = req.result;
|
2016-08-03 22:21:11 +03:00
|
|
|
resolve(cursor ? cursor.key : null);
|
|
|
|
};
|
|
|
|
let advanced = false;
|
|
|
|
req = store.openCursor();
|
|
|
|
req.onsuccess = function keyOnSuccess() {
|
2018-06-01 13:36:09 +03:00
|
|
|
const cursor = req.result;
|
2016-08-03 22:21:11 +03:00
|
|
|
if (!cursor) {
|
|
|
|
// this means there weren"t enough keys
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (n === 0 || advanced) {
|
|
|
|
// Either 1) we have the first key, return it if that's what they
|
|
|
|
// wanted, or 2) we"ve got the nth key.
|
|
|
|
return;
|
|
|
|
}
|
2015-02-20 23:57:00 +03:00
|
|
|
|
2016-08-03 22:21:11 +03:00
|
|
|
// Otherwise, ask the cursor to skip ahead n records
|
|
|
|
advanced = true;
|
|
|
|
cursor.advance(n);
|
|
|
|
};
|
|
|
|
req.onerror = function keyOnError() {
|
|
|
|
reject("Error in asyncStorage.key(): ", req.error.name);
|
|
|
|
};
|
|
|
|
}, reject);
|
|
|
|
});
|
|
|
|
}
|
2015-02-20 23:57:00 +03:00
|
|
|
|
2016-08-03 22:21:11 +03:00
|
|
|
exports.getItem = getItem;
|
|
|
|
exports.setItem = setItem;
|
|
|
|
exports.removeItem = removeItem;
|
|
|
|
exports.clear = clear;
|
|
|
|
exports.length = length;
|
|
|
|
exports.key = key;
|