You've already forked matrix-js-sdk
mirror of
https://github.com/matrix-org/matrix-js-sdk.git
synced 2025-11-29 16:43:09 +03:00
343 lines
12 KiB
JavaScript
343 lines
12 KiB
JavaScript
/*
|
|
Copyright 2017 Vector Creations Ltd
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
import q from "q";
|
|
import SyncAccumulator from "../sync-accumulator";
|
|
import utils from "../utils";
|
|
|
|
const VERSION = 1;
|
|
|
|
function createDatabase(db) {
|
|
// Make user store, clobber based on user ID. (userId property of User objects)
|
|
db.createObjectStore("users", { keyPath: ["userId"] });
|
|
|
|
// Make account data store, clobber based on event type.
|
|
// (event.type property of MatrixEvent objects)
|
|
db.createObjectStore("accountData", { keyPath: ["type"] });
|
|
|
|
// Make /sync store (sync tokens, room data, etc), always clobber (const key).
|
|
db.createObjectStore("sync", { keyPath: ["clobber"] });
|
|
}
|
|
|
|
/**
|
|
* Helper method to collect results from a Cursor and promiseify it.
|
|
* @param {ObjectStore|Index} store The store to perform openCursor on.
|
|
* @param {IDBKeyRange=} keyRange Optional key range to apply on the cursor.
|
|
* @param {Function} resultMapper A function which is repeatedly called with a
|
|
* Cursor.
|
|
* Return the data you want to keep.
|
|
* @return {Promise<T[]>} Resolves to an array of whatever you returned from
|
|
* resultMapper.
|
|
*/
|
|
function selectQuery(store, keyRange, resultMapper) {
|
|
const query = store.openCursor(keyRange);
|
|
return q.Promise((resolve, reject) => { /*eslint new-cap: 0*/
|
|
const results = [];
|
|
query.onerror = (event) => {
|
|
reject(new Error("Query failed: " + event.target.errorCode));
|
|
};
|
|
// collect results
|
|
query.onsuccess = (event) => {
|
|
const cursor = event.target.result;
|
|
if (!cursor) {
|
|
resolve(results);
|
|
return; // end of results
|
|
}
|
|
results.push(resultMapper(cursor));
|
|
cursor.continue();
|
|
};
|
|
});
|
|
}
|
|
|
|
function promiseifyTxn(txn) {
|
|
return new q.Promise((resolve, reject) => {
|
|
txn.oncomplete = function(event) {
|
|
resolve(event);
|
|
};
|
|
txn.onerror = function(event) {
|
|
reject(event);
|
|
};
|
|
});
|
|
}
|
|
|
|
function promiseifyRequest(req) {
|
|
return new q.Promise((resolve, reject) => {
|
|
req.onsuccess = function(event) {
|
|
resolve(event);
|
|
};
|
|
req.onerror = function(event) {
|
|
reject(event);
|
|
};
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Does the actual reading from and writing to the indexeddb
|
|
*
|
|
* Construct a new Indexed Database store backend. This requires a call to
|
|
* <code>connect()</code> before this store can be used.
|
|
* @constructor
|
|
* @param {Object} indexedDBInterface The Indexed DB interface e.g
|
|
* <code>window.indexedDB</code>
|
|
* @param {string=} dbName Optional database name. The same name must be used
|
|
* to open the same database.
|
|
*/
|
|
const LocalIndexedDBStoreBackend = function LocalIndexedDBStoreBackend(
|
|
indexedDBInterface, dbName,
|
|
) {
|
|
this.indexedDB = indexedDBInterface;
|
|
this._dbName = "matrix-js-sdk:" + (dbName || "default");
|
|
this.db = null;
|
|
this._syncAccumulator = new SyncAccumulator();
|
|
};
|
|
|
|
|
|
LocalIndexedDBStoreBackend.prototype = {
|
|
/**
|
|
* Attempt to connect to the database. This can fail if the user does not
|
|
* grant permission.
|
|
* @return {Promise} Resolves if successfully connected.
|
|
*/
|
|
connect: function() {
|
|
if (this.db) {
|
|
return q();
|
|
}
|
|
const req = this.indexedDB.open(this._dbName, VERSION);
|
|
req.onupgradeneeded = (ev) => {
|
|
const db = ev.target.result;
|
|
const oldVersion = ev.oldVersion;
|
|
if (oldVersion < 1) { // The database did not previously exist.
|
|
createDatabase(db);
|
|
}
|
|
// Expand as needed.
|
|
};
|
|
|
|
return promiseifyRequest(req).then((ev) => {
|
|
this.db = ev.target.result;
|
|
|
|
// add a poorly-named listener for when deleteDatabase is called
|
|
// so we can close our db connections.
|
|
this.db.onversionchange = () => {
|
|
this.db.close();
|
|
};
|
|
|
|
return this._init();
|
|
});
|
|
},
|
|
|
|
/**
|
|
* Having connected, load initial data from the database and prepare for use
|
|
* @return {Promise} Resolves on success
|
|
*/
|
|
_init: function() {
|
|
return q.all([
|
|
this._loadAccountData(),
|
|
this._loadSyncData(),
|
|
]).then(([accountData, syncData]) => {
|
|
this._syncAccumulator.accumulate({
|
|
next_batch: syncData.nextBatch,
|
|
rooms: syncData.roomsData,
|
|
account_data: {
|
|
events: accountData,
|
|
},
|
|
});
|
|
});
|
|
},
|
|
|
|
/**
|
|
* Clear the entire database. This should be used when logging out of a client
|
|
* to prevent mixing data between accounts.
|
|
* @return {Promise} Resolved when the database is cleared.
|
|
*/
|
|
clearDatabase: function() {
|
|
return new q.Promise((resolve, reject) => {
|
|
console.log(`Removing indexeddb instance: ${this._dbName}`);
|
|
const req = this.indexedDB.deleteDatabase(this._dbName);
|
|
|
|
req.onblocked = () => {
|
|
console.log(
|
|
`can't yet delete indexeddb ${this._dbName}` +
|
|
` because it is open elsewhere`,
|
|
);
|
|
};
|
|
|
|
req.onerror = (ev) => {
|
|
// in firefox, with indexedDB disabled, this fails with a
|
|
// DOMError. We treat this as non-fatal, so that we can still
|
|
// use the app.
|
|
console.warn(
|
|
`unable to delete js-sdk store indexeddb: ${ev.target.error}`,
|
|
);
|
|
resolve();
|
|
};
|
|
|
|
req.onsuccess = () => {
|
|
console.log(`Removed indexeddb instance: ${this._dbName}`);
|
|
resolve();
|
|
};
|
|
});
|
|
},
|
|
|
|
/**
|
|
* @param {boolean=} copy If false, the data returned is from internal
|
|
* buffers and must not be muated. Otherwise, a copy is made before
|
|
* returning such that the data can be safely mutated. Default: true.
|
|
*
|
|
* @return {Promise} Resolves with a sync response to restore the
|
|
* client state to where it was at the last save, or null if there
|
|
* is no saved sync data.
|
|
*/
|
|
getSavedSync: function(copy) {
|
|
if (copy === undefined) copy = true;
|
|
|
|
const data = this._syncAccumulator.getJSON();
|
|
if (!data.nextBatch) return q(null);
|
|
if (copy) {
|
|
// We must deep copy the stored data so that the /sync processing code doesn't
|
|
// corrupt the internal state of the sync accumulator (it adds non-clonable keys)
|
|
return q(utils.deepCopy(data));
|
|
} else {
|
|
return q(data);
|
|
}
|
|
},
|
|
|
|
setSyncData: function(syncData) {
|
|
return q().then(() => {
|
|
this._syncAccumulator.accumulate(syncData);
|
|
});
|
|
},
|
|
|
|
syncToDatabase: function(userTuples) {
|
|
const syncData = this._syncAccumulator.getJSON();
|
|
|
|
return q.all([
|
|
this._persistUserPresenceEvents(userTuples),
|
|
this._persistAccountData(syncData.accountData),
|
|
this._persistSyncData(syncData.nextBatch, syncData.roomsData),
|
|
]);
|
|
},
|
|
|
|
/**
|
|
* Persist rooms /sync data along with the next batch token.
|
|
* @param {string} nextBatch The next_batch /sync value.
|
|
* @param {Object} roomsData The 'rooms' /sync data from a SyncAccumulator
|
|
* @return {Promise} Resolves if the data was persisted.
|
|
*/
|
|
_persistSyncData: function(nextBatch, roomsData) {
|
|
console.log("Persisting sync data up to ", nextBatch);
|
|
return q.try(() => {
|
|
const txn = this.db.transaction(["sync"], "readwrite");
|
|
const store = txn.objectStore("sync");
|
|
store.put({
|
|
clobber: "-", // constant key so will always clobber
|
|
nextBatch: nextBatch,
|
|
roomsData: roomsData,
|
|
}); // put == UPSERT
|
|
return promiseifyTxn(txn);
|
|
});
|
|
},
|
|
|
|
/**
|
|
* Persist a list of account data events. Events with the same 'type' will
|
|
* be replaced.
|
|
* @param {Object[]} accountData An array of raw user-scoped account data events
|
|
* @return {Promise} Resolves if the events were persisted.
|
|
*/
|
|
_persistAccountData: function(accountData) {
|
|
return q.try(() => {
|
|
const txn = this.db.transaction(["accountData"], "readwrite");
|
|
const store = txn.objectStore("accountData");
|
|
for (let i = 0; i < accountData.length; i++) {
|
|
store.put(accountData[i]); // put == UPSERT
|
|
}
|
|
return promiseifyTxn(txn);
|
|
});
|
|
},
|
|
|
|
/**
|
|
* Persist a list of [user id, presence event] they are for.
|
|
* Users with the same 'userId' will be replaced.
|
|
* Presence events should be the event in its raw form (not the Event
|
|
* object)
|
|
* @param {Object[]} tuples An array of [userid, event] tuples
|
|
* @return {Promise} Resolves if the users were persisted.
|
|
*/
|
|
_persistUserPresenceEvents: function(tuples) {
|
|
return q.try(() => {
|
|
const txn = this.db.transaction(["users"], "readwrite");
|
|
const store = txn.objectStore("users");
|
|
for (const tuple of tuples) {
|
|
store.put({
|
|
userId: tuple[0],
|
|
event: tuple[1],
|
|
}); // put == UPSERT
|
|
}
|
|
return promiseifyTxn(txn);
|
|
});
|
|
},
|
|
|
|
/**
|
|
* Load all user presence events from the database. This is not cached.
|
|
* FIXME: It would probably be more sensible to store the events in the
|
|
* sync.
|
|
* @return {Promise<Object[]>} A list of presence events in their raw form.
|
|
*/
|
|
getUserPresenceEvents: function() {
|
|
return q.try(() => {
|
|
const txn = this.db.transaction(["users"], "readonly");
|
|
const store = txn.objectStore("users");
|
|
return selectQuery(store, undefined, (cursor) => {
|
|
return [cursor.value.userId, cursor.value.event];
|
|
});
|
|
});
|
|
},
|
|
|
|
/**
|
|
* Load all the account data events from the database. This is not cached.
|
|
* @return {Promise<Object[]>} A list of raw global account events.
|
|
*/
|
|
_loadAccountData: function() {
|
|
return q.try(() => {
|
|
const txn = this.db.transaction(["accountData"], "readonly");
|
|
const store = txn.objectStore("accountData");
|
|
return selectQuery(store, undefined, (cursor) => {
|
|
return cursor.value;
|
|
});
|
|
});
|
|
},
|
|
|
|
/**
|
|
* Load the sync data from the database.
|
|
* @return {Promise<Object>} An object with "roomsData" and "nextBatch" keys.
|
|
*/
|
|
_loadSyncData: function() {
|
|
return q.try(() => {
|
|
const txn = this.db.transaction(["sync"], "readonly");
|
|
const store = txn.objectStore("sync");
|
|
return selectQuery(store, undefined, (cursor) => {
|
|
return cursor.value;
|
|
}).then((results) => {
|
|
if (results.length > 1) {
|
|
console.warn("loadSyncData: More than 1 sync row found.");
|
|
}
|
|
return (results.length > 0 ? results[0] : {});
|
|
});
|
|
});
|
|
},
|
|
};
|
|
|
|
export default LocalIndexedDBStoreBackend;
|