2021-09-29 11:49:58 +02:00
|
|
|
import {IDOMStorage} from "./types";
|
2021-09-29 19:39:26 +02:00
|
|
|
import {ITransaction} from "./QueryTarget";
|
2021-08-12 13:28:36 -07:00
|
|
|
import {iterateCursor, NOT_DONE, reqAsPromise} from "./utils";
|
2020-08-19 16:28:09 +02:00
|
|
|
import {RoomMember, EVENT_TYPE as MEMBER_EVENT_TYPE} from "../../room/members/RoomMember.js";
|
2021-08-27 19:39:24 +02:00
|
|
|
import {addRoomToIdentity} from "../../e2ee/DeviceTracker.js";
|
2021-09-29 19:20:27 +02:00
|
|
|
import {SESSION_E2EE_KEY_PREFIX} from "../../e2ee/common.js";
|
2021-08-31 15:32:33 -07:00
|
|
|
import {SummaryData} from "../../room/RoomSummary";
|
|
|
|
import {RoomMemberStore, MemberData} from "./stores/RoomMemberStore";
|
2021-08-12 13:28:36 -07:00
|
|
|
import {RoomStateEntry} from "./stores/RoomStateStore";
|
2021-08-10 16:10:55 -07:00
|
|
|
import {SessionStore} from "./stores/SessionStore";
|
2021-09-29 19:39:26 +02:00
|
|
|
import {Store} from "./Store";
|
2021-08-12 11:05:55 -07:00
|
|
|
import {encodeScopeTypeKey} from "./stores/OperationStore";
|
2021-08-31 11:41:07 -07:00
|
|
|
import {MAX_UNICODE} from "./stores/common";
|
2021-11-15 17:29:08 +05:30
|
|
|
import {ILogItem} from "../../../logging/LogItem";
|
2020-06-26 23:26:24 +02:00
|
|
|
|
2021-09-29 11:49:58 +02:00
|
|
|
|
2021-11-15 17:29:08 +05:30
|
|
|
export type MigrationFunc = (db: IDBDatabase, txn: IDBTransaction, localStorage: IDOMStorage, log: ILogItem) => Promise<void> | void;
|
2020-06-26 23:26:24 +02:00
|
|
|
// FUNCTIONS SHOULD ONLY BE APPENDED!!
|
|
|
|
// the index in the array is the database version
|
2021-09-29 11:49:58 +02:00
|
|
|
export const schema: MigrationFunc[] = [
|
2020-06-26 23:26:24 +02:00
|
|
|
createInitialStores,
|
|
|
|
createMemberStore,
|
2020-08-27 14:28:40 +02:00
|
|
|
migrateSession,
|
2020-09-11 16:48:04 +02:00
|
|
|
createE2EEStores,
|
2020-09-17 10:39:51 +02:00
|
|
|
migrateEncryptionFlag,
|
2021-04-20 13:02:50 +02:00
|
|
|
createAccountDataStore,
|
2021-05-04 13:34:42 +02:00
|
|
|
createInviteStore,
|
|
|
|
createArchivedRoomSummaryStore,
|
2021-05-12 15:36:48 +02:00
|
|
|
migrateOperationScopeIndex,
|
2021-06-03 16:44:35 +02:00
|
|
|
createTimelineRelationsStore,
|
2021-09-29 19:20:27 +02:00
|
|
|
fixMissingRoomsInUserIdentities,
|
|
|
|
changeSSSSKeyPrefix,
|
2021-09-30 08:37:33 +02:00
|
|
|
backupAndRestoreE2EEAccountToLocalStorage,
|
|
|
|
clearAllStores
|
2020-06-26 23:26:24 +02:00
|
|
|
];
|
|
|
|
// TODO: how to deal with git merge conflicts of this array?
|
|
|
|
|
2021-08-12 13:28:36 -07:00
|
|
|
// TypeScript note: for now, do not bother introducing interfaces / alias
|
|
|
|
// for old schemas. Just take them as `any`.
|
2020-06-26 23:26:24 +02:00
|
|
|
|
|
|
|
// how do we deal with schema updates vs existing data migration in a way that
|
|
|
|
//v1
|
2021-08-12 13:28:36 -07:00
|
|
|
function createInitialStores(db: IDBDatabase): void {
|
2020-06-26 23:26:24 +02:00
|
|
|
db.createObjectStore("session", {keyPath: "key"});
|
|
|
|
// any way to make keys unique here? (just use put?)
|
|
|
|
db.createObjectStore("roomSummary", {keyPath: "roomId"});
|
|
|
|
|
|
|
|
// need index to find live fragment? prooobably ok without for now
|
|
|
|
//key = room_id | fragment_id
|
|
|
|
db.createObjectStore("timelineFragments", {keyPath: "key"});
|
|
|
|
//key = room_id | fragment_id | event_index
|
|
|
|
const timelineEvents = db.createObjectStore("timelineEvents", {keyPath: "key"});
|
|
|
|
//eventIdKey = room_id | event_id
|
|
|
|
timelineEvents.createIndex("byEventId", "eventIdKey", {unique: true});
|
|
|
|
//key = room_id | event.type | event.state_key,
|
|
|
|
db.createObjectStore("roomState", {keyPath: "key"});
|
|
|
|
db.createObjectStore("pendingEvents", {keyPath: "key"});
|
|
|
|
}
|
|
|
|
//v2
|
2021-08-12 13:28:36 -07:00
|
|
|
async function createMemberStore(db: IDBDatabase, txn: IDBTransaction): Promise<void> {
|
|
|
|
// Cast ok here because only "set" is used
|
|
|
|
const roomMembers = new RoomMemberStore(db.createObjectStore("roomMembers", {keyPath: "key"}) as any);
|
2020-06-26 23:26:24 +02:00
|
|
|
// migrate existing member state events over
|
|
|
|
const roomState = txn.objectStore("roomState");
|
2021-08-12 13:28:36 -07:00
|
|
|
await iterateCursor<RoomStateEntry>(roomState.openCursor(), entry => {
|
2020-06-26 23:26:24 +02:00
|
|
|
if (entry.event.type === MEMBER_EVENT_TYPE) {
|
|
|
|
roomState.delete(entry.key);
|
|
|
|
const member = RoomMember.fromMemberEvent(entry.roomId, entry.event);
|
|
|
|
if (member) {
|
2020-08-19 16:07:58 +02:00
|
|
|
roomMembers.set(member.serialize());
|
2020-06-26 23:26:24 +02:00
|
|
|
}
|
|
|
|
}
|
2021-08-12 13:28:36 -07:00
|
|
|
return NOT_DONE;
|
2020-06-26 23:26:24 +02:00
|
|
|
});
|
|
|
|
}
|
2020-08-31 14:38:03 +02:00
|
|
|
//v3
|
2021-09-29 11:49:58 +02:00
|
|
|
async function migrateSession(db: IDBDatabase, txn: IDBTransaction, localStorage: IDOMStorage): Promise<void> {
|
2020-08-27 14:28:40 +02:00
|
|
|
const session = txn.objectStore("session");
|
|
|
|
try {
|
|
|
|
const PRE_MIGRATION_KEY = 1;
|
|
|
|
const entry = await reqAsPromise(session.get(PRE_MIGRATION_KEY));
|
|
|
|
if (entry) {
|
|
|
|
session.delete(PRE_MIGRATION_KEY);
|
2020-08-27 14:36:50 +02:00
|
|
|
const {syncToken, syncFilterId, serverVersions} = entry.value;
|
2021-08-12 13:28:36 -07:00
|
|
|
// Cast ok here because only "set" is used and we don't look into return
|
2021-09-29 11:49:58 +02:00
|
|
|
const store = new SessionStore(session as any, localStorage);
|
2020-08-27 14:36:50 +02:00
|
|
|
store.set("sync", {token: syncToken, filterId: syncFilterId});
|
|
|
|
store.set("serverVersions", serverVersions);
|
2020-08-27 14:28:40 +02:00
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
txn.abort();
|
|
|
|
console.error("could not migrate session", err.stack);
|
|
|
|
}
|
|
|
|
}
|
2020-08-31 14:38:03 +02:00
|
|
|
//v4
|
2021-08-12 13:28:36 -07:00
|
|
|
function createE2EEStores(db: IDBDatabase): void {
|
2020-08-31 14:38:03 +02:00
|
|
|
db.createObjectStore("userIdentities", {keyPath: "userId"});
|
2020-09-11 08:40:43 +02:00
|
|
|
const deviceIdentities = db.createObjectStore("deviceIdentities", {keyPath: "key"});
|
|
|
|
deviceIdentities.createIndex("byCurve25519Key", "curve25519Key", {unique: true});
|
2020-09-01 17:59:59 +02:00
|
|
|
db.createObjectStore("olmSessions", {keyPath: "key"});
|
2020-09-02 14:24:38 +02:00
|
|
|
db.createObjectStore("inboundGroupSessions", {keyPath: "key"});
|
2020-09-03 17:49:20 +02:00
|
|
|
db.createObjectStore("outboundGroupSessions", {keyPath: "roomId"});
|
2020-09-04 15:31:00 +02:00
|
|
|
db.createObjectStore("groupSessionDecryptions", {keyPath: "key"});
|
2020-09-11 14:40:05 +02:00
|
|
|
const operations = db.createObjectStore("operations", {keyPath: "id"});
|
|
|
|
operations.createIndex("byTypeAndScope", "typeScopeKey", {unique: false});
|
2020-09-04 15:31:00 +02:00
|
|
|
}
|
2020-09-11 16:48:04 +02:00
|
|
|
|
|
|
|
// v5
|
2021-08-12 13:28:36 -07:00
|
|
|
async function migrateEncryptionFlag(db: IDBDatabase, txn: IDBTransaction): Promise<void> {
|
2020-09-11 16:48:04 +02:00
|
|
|
// migrate room summary isEncrypted -> encryption prop
|
|
|
|
const roomSummary = txn.objectStore("roomSummary");
|
|
|
|
const roomState = txn.objectStore("roomState");
|
2021-08-12 13:28:36 -07:00
|
|
|
const summaries: any[] = [];
|
|
|
|
await iterateCursor<any>(roomSummary.openCursor(), summary => {
|
2020-09-11 16:48:04 +02:00
|
|
|
summaries.push(summary);
|
2021-08-12 13:28:36 -07:00
|
|
|
return NOT_DONE;
|
2020-09-11 16:48:04 +02:00
|
|
|
});
|
|
|
|
for (const summary of summaries) {
|
|
|
|
const encryptionEntry = await reqAsPromise(roomState.get(`${summary.roomId}|m.room.encryption|`));
|
|
|
|
if (encryptionEntry) {
|
|
|
|
summary.encryption = encryptionEntry?.event?.content;
|
|
|
|
delete summary.isEncrypted;
|
|
|
|
roomSummary.put(summary);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-09-17 10:39:51 +02:00
|
|
|
|
|
|
|
// v6
|
2021-08-12 13:28:36 -07:00
|
|
|
function createAccountDataStore(db: IDBDatabase): void {
|
2020-09-17 10:39:51 +02:00
|
|
|
db.createObjectStore("accountData", {keyPath: "type"});
|
|
|
|
}
|
2021-04-20 13:02:50 +02:00
|
|
|
|
|
|
|
// v7
|
2021-08-12 13:28:36 -07:00
|
|
|
function createInviteStore(db: IDBDatabase): void {
|
2021-04-20 13:02:50 +02:00
|
|
|
db.createObjectStore("invites", {keyPath: "roomId"});
|
|
|
|
}
|
2021-05-04 13:34:42 +02:00
|
|
|
|
|
|
|
// v8
|
2021-08-12 13:28:36 -07:00
|
|
|
function createArchivedRoomSummaryStore(db: IDBDatabase): void {
|
2021-05-11 13:01:19 +02:00
|
|
|
db.createObjectStore("archivedRoomSummary", {keyPath: "summary.roomId"});
|
2021-05-12 15:36:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// v9
|
2021-08-12 13:28:36 -07:00
|
|
|
async function migrateOperationScopeIndex(db: IDBDatabase, txn: IDBTransaction): Promise<void> {
|
2021-05-12 15:36:48 +02:00
|
|
|
try {
|
|
|
|
const operations = txn.objectStore("operations");
|
|
|
|
operations.deleteIndex("byTypeAndScope");
|
2021-08-12 13:28:36 -07:00
|
|
|
await iterateCursor<any>(operations.openCursor(), (op, key, cur) => {
|
2021-05-12 15:36:48 +02:00
|
|
|
const {typeScopeKey} = op;
|
|
|
|
delete op.typeScopeKey;
|
|
|
|
const [type, scope] = typeScopeKey.split("|");
|
|
|
|
op.scopeTypeKey = encodeScopeTypeKey(scope, type);
|
|
|
|
cur.update(op);
|
2021-08-12 13:28:36 -07:00
|
|
|
return NOT_DONE;
|
2021-05-12 15:36:48 +02:00
|
|
|
});
|
|
|
|
operations.createIndex("byScopeAndType", "scopeTypeKey", {unique: false});
|
|
|
|
} catch (err) {
|
|
|
|
txn.abort();
|
|
|
|
console.error("could not migrate operations", err.stack);
|
|
|
|
}
|
2021-06-03 16:44:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
//v10
|
2021-08-12 13:28:36 -07:00
|
|
|
function createTimelineRelationsStore(db: IDBDatabase) : void {
|
2021-06-24 16:16:15 +02:00
|
|
|
db.createObjectStore("timelineRelations", {keyPath: "key"});
|
|
|
|
}
|
2021-08-27 19:39:24 +02:00
|
|
|
|
|
|
|
//v11 doesn't change the schema, but ensures all userIdentities have all the roomIds they should (see #470)
|
2021-11-15 17:29:08 +05:30
|
|
|
async function fixMissingRoomsInUserIdentities(db: IDBDatabase, txn: IDBTransaction, localStorage: IDOMStorage, log: ILogItem) {
|
2021-08-27 19:39:24 +02:00
|
|
|
const roomSummaryStore = txn.objectStore("roomSummary");
|
2021-08-31 15:32:33 -07:00
|
|
|
const trackedRoomIds: string[] = [];
|
|
|
|
await iterateCursor<SummaryData>(roomSummaryStore.openCursor(), roomSummary => {
|
2021-08-27 19:39:24 +02:00
|
|
|
if (roomSummary.isTrackingMembers) {
|
|
|
|
trackedRoomIds.push(roomSummary.roomId);
|
|
|
|
}
|
2021-08-31 15:32:33 -07:00
|
|
|
return NOT_DONE;
|
2021-08-27 19:39:24 +02:00
|
|
|
});
|
|
|
|
const outboundGroupSessionsStore = txn.objectStore("outboundGroupSessions");
|
2021-08-31 15:32:33 -07:00
|
|
|
const userIdentitiesStore: IDBObjectStore = txn.objectStore("userIdentities");
|
2021-08-27 19:39:24 +02:00
|
|
|
const roomMemberStore = txn.objectStore("roomMembers");
|
|
|
|
for (const roomId of trackedRoomIds) {
|
|
|
|
let foundMissing = false;
|
2021-08-31 15:32:33 -07:00
|
|
|
const joinedUserIds: string[] = [];
|
2021-08-27 19:39:24 +02:00
|
|
|
const memberRange = IDBKeyRange.bound(roomId, `${roomId}|${MAX_UNICODE}`, true, true);
|
|
|
|
await log.wrap({l: "room", id: roomId}, async log => {
|
2021-08-31 15:32:33 -07:00
|
|
|
await iterateCursor<MemberData>(roomMemberStore.openCursor(memberRange), member => {
|
2021-08-27 19:39:24 +02:00
|
|
|
if (member.membership === "join") {
|
|
|
|
joinedUserIds.push(member.userId);
|
|
|
|
}
|
2021-08-31 15:32:33 -07:00
|
|
|
return NOT_DONE;
|
2021-08-27 19:39:24 +02:00
|
|
|
});
|
|
|
|
log.set("joinedUserIds", joinedUserIds.length);
|
|
|
|
for (const userId of joinedUserIds) {
|
|
|
|
const identity = await reqAsPromise(userIdentitiesStore.get(userId));
|
|
|
|
const originalRoomCount = identity?.roomIds?.length;
|
|
|
|
const updatedIdentity = addRoomToIdentity(identity, userId, roomId);
|
|
|
|
if (updatedIdentity) {
|
|
|
|
log.log({l: `fixing up`, id: userId,
|
2021-11-15 17:29:08 +05:30
|
|
|
roomsBefore: originalRoomCount, roomsAfter: updatedIdentity.roomIds.length}, null);
|
2021-08-27 19:39:24 +02:00
|
|
|
userIdentitiesStore.put(updatedIdentity);
|
|
|
|
foundMissing = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
log.set("foundMissing", foundMissing);
|
|
|
|
if (foundMissing) {
|
|
|
|
// clear outbound megolm session,
|
|
|
|
// so we'll create a new one on the next message that will be properly shared
|
|
|
|
outboundGroupSessionsStore.delete(roomId);
|
|
|
|
}
|
2021-11-15 17:29:08 +05:30
|
|
|
}, null, null);
|
2021-08-27 19:39:24 +02:00
|
|
|
}
|
|
|
|
}
|
2021-09-29 19:20:27 +02:00
|
|
|
|
|
|
|
// v12 move ssssKey to e2ee:ssssKey so it will get backed up in the next step
|
|
|
|
async function changeSSSSKeyPrefix(db: IDBDatabase, txn: IDBTransaction) {
|
|
|
|
const session = txn.objectStore("session");
|
|
|
|
const ssssKey = await reqAsPromise(session.get("ssssKey"));
|
|
|
|
if (ssssKey) {
|
2021-09-30 10:18:03 +02:00
|
|
|
session.put({key: `${SESSION_E2EE_KEY_PREFIX}ssssKey`, value: ssssKey.value});
|
2021-09-29 19:20:27 +02:00
|
|
|
}
|
|
|
|
}
|
2021-09-29 19:39:26 +02:00
|
|
|
// v13
|
2021-11-15 17:29:08 +05:30
|
|
|
async function backupAndRestoreE2EEAccountToLocalStorage(db: IDBDatabase, txn: IDBTransaction, localStorage: IDOMStorage, log: ILogItem) {
|
2021-09-29 19:39:26 +02:00
|
|
|
const session = txn.objectStore("session");
|
|
|
|
// the Store object gets passed in several things through the Transaction class (a wrapper around IDBTransaction),
|
|
|
|
// the only thing we should need here is the databaseName though, so we mock it out.
|
|
|
|
// ideally we should have an easier way to go from the idb primitive layer to the specific store classes where
|
|
|
|
// we implement logic, but for now we need this.
|
|
|
|
const databaseNameHelper: ITransaction = {
|
|
|
|
databaseName: db.name,
|
|
|
|
get idbFactory(): IDBFactory { throw new Error("unused");},
|
|
|
|
get IDBKeyRange(): typeof IDBKeyRange { throw new Error("unused");},
|
|
|
|
addWriteError() {},
|
|
|
|
};
|
|
|
|
const sessionStore = new SessionStore(new Store(session, databaseNameHelper), localStorage);
|
|
|
|
// if we already have an e2ee identity, write a backup to local storage.
|
|
|
|
// further updates to e2ee keys in the session store will also write to local storage from 0.2.15 on,
|
|
|
|
// but here we make sure a backup is immediately created after installing the update and we don't wait until
|
|
|
|
// the olm account needs to change
|
|
|
|
sessionStore.writeE2EEIdentityToLocalStorage();
|
|
|
|
// and if we already have a backup, restore it now for any missing key in idb.
|
|
|
|
// this will restore the backup every time the idb database is dropped as it will
|
|
|
|
// run through all the migration steps when recreating it.
|
|
|
|
const restored = await sessionStore.tryRestoreE2EEIdentityFromLocalStorage(log);
|
|
|
|
log.set("restored", restored);
|
|
|
|
}
|
2021-09-30 08:37:33 +02:00
|
|
|
// v14 clear all stores apart from e2ee keys because of possible timeline corruption in #515, will trigger initial sync
|
|
|
|
async function clearAllStores(db: IDBDatabase, txn: IDBTransaction) {
|
|
|
|
for (const storeName of db.objectStoreNames) {
|
|
|
|
const store = txn.objectStore(storeName);
|
|
|
|
switch (storeName) {
|
|
|
|
case "inboundGroupSessions":
|
|
|
|
case "outboundGroupSessions":
|
|
|
|
case "olmSessions":
|
|
|
|
case "operations":
|
|
|
|
continue;
|
|
|
|
case "session": {
|
|
|
|
await iterateCursor(store.openCursor(), (value, key, cursor) => {
|
|
|
|
if (!(key as string).startsWith(SESSION_E2EE_KEY_PREFIX)) {
|
|
|
|
cursor.delete();
|
|
|
|
}
|
|
|
|
return NOT_DONE;
|
|
|
|
})
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default: {
|
|
|
|
store.clear();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|