211 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			JavaScript
		
	
	
			
		
		
	
	
			211 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			JavaScript
		
	
	
/*
 | 
						|
Copyright 2018 New Vector 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.
 | 
						|
*/
 | 
						|
 | 
						|
const TARGET_ORIGIN = 'vector://vector';
 | 
						|
const BATCH_SIZE = 500;
 | 
						|
let destFrame;
 | 
						|
 | 
						|
let initResolver = null;
 | 
						|
let getSummaryResolver = null;
 | 
						|
 | 
						|
function onMessage(e) {
 | 
						|
    if (e.origin !== TARGET_ORIGIN) return;
 | 
						|
 | 
						|
    if (e.data.cmd === 'initOK' && initResolver) {
 | 
						|
        initResolver();
 | 
						|
        initResolver = null;
 | 
						|
    } else if (e.data.cmd === 'summary' && getSummaryResolver) {
 | 
						|
        getSummaryResolver(e.data.data);
 | 
						|
        getSummaryResolver = null;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
async function initDestFrame() {
 | 
						|
    return new Promise(resolve => {
 | 
						|
        initResolver = resolve;
 | 
						|
        destFrame.postMessage({
 | 
						|
            cmd: 'init', 
 | 
						|
        }, TARGET_ORIGIN);
 | 
						|
    });
 | 
						|
}
 | 
						|
 | 
						|
async function getSummary() {
 | 
						|
    return new Promise(resolve => {
 | 
						|
        getSummaryResolver = resolve;
 | 
						|
        destFrame.postMessage({
 | 
						|
            cmd: 'getSummary', 
 | 
						|
        }, TARGET_ORIGIN);
 | 
						|
    });
 | 
						|
}
 | 
						|
 | 
						|
async function doMigrate() {
 | 
						|
    let accountSent = 0;
 | 
						|
    let sessionsSent = 0;
 | 
						|
    let inboundGroupSessionsSent = 0;
 | 
						|
    let deviceDataSent = 0;
 | 
						|
    let roomsSent = 0;
 | 
						|
    let localStorageKeysSent = 0;
 | 
						|
 | 
						|
    if (!window.ipcRenderer) {
 | 
						|
        console.error("ipcRenderer not found");
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    if (window.localStorage.getItem('mx_user_id') === null) {
 | 
						|
        window.ipcRenderer.send("origin_migration_nodata");
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    destFrame = window.parent.frames.dest;
 | 
						|
 | 
						|
    await initDestFrame();
 | 
						|
 | 
						|
    const IndexedDBCryptoStore = window.matrixcs.IndexedDBCryptoStore;
 | 
						|
 | 
						|
    const cryptoStore = new IndexedDBCryptoStore(window.indexedDB, 'matrix-js-sdk:crypto');
 | 
						|
 | 
						|
    await cryptoStore.doTxn(
 | 
						|
        'readonly', [IndexedDBCryptoStore.STORE_ACCOUNT],
 | 
						|
        (txn) => {
 | 
						|
            cryptoStore.getAccount(txn, (account) => {
 | 
						|
                destFrame.postMessage({
 | 
						|
                    cmd: 'storeAccount',
 | 
						|
                    data: account,
 | 
						|
                }, TARGET_ORIGIN);
 | 
						|
                ++accountSent;
 | 
						|
            });
 | 
						|
        },
 | 
						|
    );
 | 
						|
 | 
						|
    await cryptoStore.doTxn(
 | 
						|
        'readonly', [IndexedDBCryptoStore.STORE_SESSIONS],
 | 
						|
        (txn) => {
 | 
						|
            let sessBatch = [];
 | 
						|
            cryptoStore.getAllEndToEndSessions(txn, (sessInfo) => {
 | 
						|
                if (sessInfo) {
 | 
						|
                    ++sessionsSent;
 | 
						|
                    sessBatch.push(sessInfo);
 | 
						|
                }
 | 
						|
                if (sessBatch.length >= BATCH_SIZE || sessInfo === null) {
 | 
						|
                    destFrame.postMessage({
 | 
						|
                        cmd: 'storeSessions',
 | 
						|
                        data: sessBatch,
 | 
						|
                    }, TARGET_ORIGIN);
 | 
						|
                    sessBatch = [];
 | 
						|
                }
 | 
						|
            });
 | 
						|
        },
 | 
						|
    );
 | 
						|
 | 
						|
    await cryptoStore.doTxn(
 | 
						|
        'readonly', [IndexedDBCryptoStore.STORE_INBOUND_GROUP_SESSIONS],
 | 
						|
        (txn) => {
 | 
						|
            let sessBatch = [];
 | 
						|
            cryptoStore.getAllEndToEndInboundGroupSessions(txn, (sessInfo) => {
 | 
						|
                if (sessInfo) {
 | 
						|
                    ++inboundGroupSessionsSent;
 | 
						|
                    sessBatch.push(sessInfo);
 | 
						|
                }
 | 
						|
                if (sessBatch.length >= BATCH_SIZE || sessInfo === null) {
 | 
						|
                    destFrame.postMessage({
 | 
						|
                        cmd: 'storeInboundGroupSessions',
 | 
						|
                        data: sessBatch,
 | 
						|
                    }, TARGET_ORIGIN);
 | 
						|
                    sessBatch = [];
 | 
						|
                }
 | 
						|
            });
 | 
						|
        },
 | 
						|
    );
 | 
						|
 | 
						|
    await cryptoStore.doTxn(
 | 
						|
        'readonly', [IndexedDBCryptoStore.STORE_DEVICE_DATA],
 | 
						|
        (txn) => {
 | 
						|
            cryptoStore.getEndToEndDeviceData(txn, (deviceData) => {
 | 
						|
                destFrame.postMessage({
 | 
						|
                    cmd: 'storeDeviceData',
 | 
						|
                    data: deviceData,
 | 
						|
                }, TARGET_ORIGIN);
 | 
						|
                ++deviceDataSent;
 | 
						|
            });
 | 
						|
        },
 | 
						|
    );
 | 
						|
 | 
						|
    await cryptoStore.doTxn(
 | 
						|
        'readonly', [IndexedDBCryptoStore.STORE_ROOMS],
 | 
						|
        (txn) => {
 | 
						|
            cryptoStore.getEndToEndRooms(txn, (rooms) => {
 | 
						|
                destFrame.postMessage({
 | 
						|
                    cmd: 'storeRooms',
 | 
						|
                    data: rooms,
 | 
						|
                }, TARGET_ORIGIN);
 | 
						|
                ++roomsSent;
 | 
						|
            });
 | 
						|
        },
 | 
						|
    );
 | 
						|
 | 
						|
    // we don't bother migrating;
 | 
						|
    // * sync data (we can just initialsync again)
 | 
						|
    // * logs
 | 
						|
    // * key requests (worst case they'll just be re-sent)
 | 
						|
    // * sessions needing backup (feature isn't available on Electron)
 | 
						|
 | 
						|
    for (let i = 0; i < window.localStorage.length; ++i) {
 | 
						|
        const key = window.localStorage.key(i);
 | 
						|
        const val = window.localStorage.getItem(key);
 | 
						|
        
 | 
						|
        destFrame.postMessage({
 | 
						|
            cmd: 'storeLocalStorage',
 | 
						|
            data: { key, val },
 | 
						|
        }, TARGET_ORIGIN);
 | 
						|
        ++localStorageKeysSent;
 | 
						|
    }
 | 
						|
 | 
						|
    const summary = await getSummary();
 | 
						|
    let success = false;
 | 
						|
    if (
 | 
						|
        summary.accountStored === accountSent &&
 | 
						|
        summary.sessionsStored === sessionsSent &&
 | 
						|
        summary.inboundGroupSessionsStored === inboundGroupSessionsSent &&
 | 
						|
        summary.deviceDataStored === deviceDataSent &&
 | 
						|
        summary.roomsStored === roomsSent &&
 | 
						|
        summary.localStorageKeysStored === localStorageKeysSent
 | 
						|
    ) {
 | 
						|
        success = true;
 | 
						|
        window.localStorage.clear();
 | 
						|
        await cryptoStore.deleteAllData();
 | 
						|
 | 
						|
        // we don't bother migrating them, but also blow away the sync & logs db,
 | 
						|
        // otherwise they'll just hang about taking up space
 | 
						|
        await new Promise(resolve => {
 | 
						|
            const req = window.indexedDB.deleteDatabase('matrix-js-sdk:riot-web-sync');
 | 
						|
            req.onsuccess = resolve;
 | 
						|
            req.onerror = resolve;
 | 
						|
        });
 | 
						|
        await new Promise(resolve => {
 | 
						|
            const req = window.indexedDB.deleteDatabase('logs');
 | 
						|
            req.onsuccess = resolve;
 | 
						|
            req.onerror = resolve;
 | 
						|
        });
 | 
						|
    }
 | 
						|
 | 
						|
    window.ipcRenderer.send("origin_migration_complete", success, {
 | 
						|
        accountSent, sessionsSent, inboundGroupSessionsSent,
 | 
						|
        deviceDataSent, roomsSent, localStorageKeysSent,
 | 
						|
    }, summary);
 | 
						|
}
 | 
						|
 | 
						|
window.addEventListener('message', onMessage);
 |