2018-07-21 19:00:08 +00:00
|
|
|
/* global window */
|
|
|
|
|
2018-04-03 15:37:34 +00:00
|
|
|
const { isString, last } = require('lodash');
|
2018-03-26 23:07:50 +00:00
|
|
|
|
2018-03-26 19:17:19 +00:00
|
|
|
const { runMigrations } = require('./run_migrations');
|
2018-04-11 15:42:06 +00:00
|
|
|
const Migration18 = require('./18');
|
2018-03-26 18:57:10 +00:00
|
|
|
|
|
|
|
// IMPORTANT: The migrations below are run on a database that may be very large
|
|
|
|
// due to attachments being directly stored inside the database. Please avoid
|
|
|
|
// any expensive operations, e.g. modifying all messages / attachments, etc., as
|
|
|
|
// it may cause out-of-memory errors for users with long histories:
|
|
|
|
// https://github.com/signalapp/Signal-Desktop/issues/2163
|
2018-04-02 21:59:35 +00:00
|
|
|
const migrations = [
|
2018-03-26 18:57:10 +00:00
|
|
|
{
|
|
|
|
version: '12.0',
|
|
|
|
migrate(transaction, next) {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info('Migration 12');
|
|
|
|
window.log.info('creating object stores');
|
2018-03-26 18:57:10 +00:00
|
|
|
const messages = transaction.db.createObjectStore('messages');
|
|
|
|
messages.createIndex('conversation', ['conversationId', 'received_at'], {
|
|
|
|
unique: false,
|
|
|
|
});
|
|
|
|
messages.createIndex('receipt', 'sent_at', { unique: false });
|
2018-04-27 21:25:04 +00:00
|
|
|
messages.createIndex('unread', ['conversationId', 'unread'], {
|
|
|
|
unique: false,
|
|
|
|
});
|
2018-03-26 18:57:10 +00:00
|
|
|
messages.createIndex('expires_at', 'expires_at', { unique: false });
|
|
|
|
|
|
|
|
const conversations = transaction.db.createObjectStore('conversations');
|
|
|
|
conversations.createIndex('inbox', 'active_at', { unique: false });
|
|
|
|
conversations.createIndex('group', 'members', {
|
|
|
|
unique: false,
|
|
|
|
multiEntry: true,
|
|
|
|
});
|
|
|
|
conversations.createIndex('type', 'type', {
|
|
|
|
unique: false,
|
|
|
|
});
|
|
|
|
conversations.createIndex('search', 'tokens', {
|
|
|
|
unique: false,
|
|
|
|
multiEntry: true,
|
|
|
|
});
|
|
|
|
|
|
|
|
transaction.db.createObjectStore('groups');
|
|
|
|
|
|
|
|
transaction.db.createObjectStore('sessions');
|
|
|
|
transaction.db.createObjectStore('identityKeys');
|
|
|
|
transaction.db.createObjectStore('preKeys');
|
|
|
|
transaction.db.createObjectStore('signedPreKeys');
|
|
|
|
transaction.db.createObjectStore('items');
|
|
|
|
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info('creating debug log');
|
2018-03-26 18:57:10 +00:00
|
|
|
transaction.db.createObjectStore('debug');
|
|
|
|
|
|
|
|
next();
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
version: '13.0',
|
|
|
|
migrate(transaction, next) {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info('Migration 13');
|
|
|
|
window.log.info('Adding fields to identity keys');
|
2018-03-26 18:57:10 +00:00
|
|
|
const identityKeys = transaction.objectStore('identityKeys');
|
|
|
|
const request = identityKeys.openCursor();
|
|
|
|
const promises = [];
|
2018-04-27 21:25:04 +00:00
|
|
|
request.onsuccess = event => {
|
2018-03-26 18:57:10 +00:00
|
|
|
const cursor = event.target.result;
|
|
|
|
if (cursor) {
|
|
|
|
const attributes = cursor.value;
|
|
|
|
attributes.timestamp = 0;
|
|
|
|
attributes.firstUse = false;
|
|
|
|
attributes.nonblockingApproval = false;
|
|
|
|
attributes.verified = 0;
|
2018-04-27 21:25:04 +00:00
|
|
|
promises.push(
|
|
|
|
new Promise((resolve, reject) => {
|
|
|
|
const putRequest = identityKeys.put(attributes, attributes.id);
|
|
|
|
putRequest.onsuccess = resolve;
|
2018-07-21 19:00:08 +00:00
|
|
|
putRequest.onerror = error => {
|
|
|
|
window.log.error(error && error.stack ? error.stack : error);
|
|
|
|
reject(error);
|
2018-04-27 21:25:04 +00:00
|
|
|
};
|
|
|
|
})
|
|
|
|
);
|
2018-03-26 18:57:10 +00:00
|
|
|
cursor.continue();
|
|
|
|
} else {
|
|
|
|
// no more results
|
|
|
|
// eslint-disable-next-line more/no-then
|
|
|
|
Promise.all(promises).then(() => {
|
|
|
|
next();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
2018-04-27 21:25:04 +00:00
|
|
|
request.onerror = event => {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.error(event);
|
2018-03-26 18:57:10 +00:00
|
|
|
};
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
version: '14.0',
|
|
|
|
migrate(transaction, next) {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info('Migration 14');
|
|
|
|
window.log.info('Adding unprocessed message store');
|
2018-03-26 18:57:10 +00:00
|
|
|
const unprocessed = transaction.db.createObjectStore('unprocessed');
|
|
|
|
unprocessed.createIndex('received', 'timestamp', { unique: false });
|
|
|
|
next();
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
version: '15.0',
|
|
|
|
migrate(transaction, next) {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info('Migration 15');
|
|
|
|
window.log.info('Adding messages index for de-duplication');
|
2018-03-26 18:57:10 +00:00
|
|
|
const messages = transaction.objectStore('messages');
|
|
|
|
messages.createIndex('unique', ['source', 'sourceDevice', 'sent_at'], {
|
|
|
|
unique: true,
|
|
|
|
});
|
|
|
|
next();
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
version: '16.0',
|
|
|
|
migrate(transaction, next) {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info('Migration 16');
|
|
|
|
window.log.info('Dropping log table, since we now log to disk');
|
2018-03-26 18:57:10 +00:00
|
|
|
transaction.db.deleteObjectStore('debug');
|
|
|
|
next();
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
version: 17,
|
|
|
|
async migrate(transaction, next) {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info('Migration 17');
|
2018-03-26 18:57:10 +00:00
|
|
|
|
|
|
|
const start = Date.now();
|
|
|
|
|
|
|
|
const messagesStore = transaction.objectStore('messages');
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info(
|
|
|
|
'Create index from attachment schema version to attachment'
|
|
|
|
);
|
2018-04-27 21:25:04 +00:00
|
|
|
messagesStore.createIndex('schemaVersion', 'schemaVersion', {
|
|
|
|
unique: false,
|
|
|
|
});
|
2018-03-26 18:57:10 +00:00
|
|
|
|
|
|
|
const duration = Date.now() - start;
|
|
|
|
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info(
|
2018-04-11 15:42:06 +00:00
|
|
|
'Complete migration to database version 17',
|
|
|
|
`Duration: ${duration}ms`
|
|
|
|
);
|
|
|
|
next();
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
version: 18,
|
|
|
|
migrate(transaction, next) {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info('Migration 18');
|
2018-04-11 15:42:06 +00:00
|
|
|
|
|
|
|
const start = Date.now();
|
2018-07-21 19:00:08 +00:00
|
|
|
Migration18.run({ transaction, logger: window.log });
|
2018-04-11 15:42:06 +00:00
|
|
|
const duration = Date.now() - start;
|
|
|
|
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info(
|
2018-04-11 15:42:06 +00:00
|
|
|
'Complete migration to database version 18',
|
2018-03-26 18:57:10 +00:00
|
|
|
`Duration: ${duration}ms`
|
|
|
|
);
|
|
|
|
next();
|
|
|
|
},
|
|
|
|
},
|
2018-10-15 19:24:43 +00:00
|
|
|
{
|
|
|
|
version: 19,
|
|
|
|
migrate(transaction, next) {
|
|
|
|
window.log.info('Migration 19');
|
|
|
|
|
2018-10-18 01:01:21 +00:00
|
|
|
// Empty because we don't want to cause incompatibility with beta users who have
|
|
|
|
// already run migration 19 when it was object store removal.
|
|
|
|
|
|
|
|
next();
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
version: 20,
|
|
|
|
migrate(transaction, next) {
|
|
|
|
window.log.info('Migration 20');
|
|
|
|
|
2018-10-15 19:24:43 +00:00
|
|
|
// Empty because we don't want to cause incompatibility with users who have already
|
2018-10-18 01:01:21 +00:00
|
|
|
// run migration 20 when it was object store removal.
|
2018-10-15 19:24:43 +00:00
|
|
|
|
|
|
|
next();
|
|
|
|
},
|
|
|
|
},
|
2018-03-26 18:57:10 +00:00
|
|
|
];
|
|
|
|
|
|
|
|
const database = {
|
|
|
|
id: 'signal',
|
|
|
|
nolog: true,
|
2018-04-02 21:59:35 +00:00
|
|
|
migrations,
|
2018-03-26 18:57:10 +00:00
|
|
|
};
|
|
|
|
|
2018-07-21 19:00:08 +00:00
|
|
|
exports.run = ({ Backbone, databaseName, logger } = {}) =>
|
2018-04-03 15:36:48 +00:00
|
|
|
runMigrations({
|
|
|
|
Backbone,
|
2018-07-21 19:00:08 +00:00
|
|
|
logger,
|
2018-04-03 15:36:48 +00:00
|
|
|
database: Object.assign(
|
|
|
|
{},
|
|
|
|
database,
|
|
|
|
isString(databaseName) ? { id: databaseName } : {}
|
|
|
|
),
|
|
|
|
});
|
2018-03-26 23:07:50 +00:00
|
|
|
|
2018-03-28 14:23:36 +00:00
|
|
|
exports.getDatabase = () => ({
|
|
|
|
name: database.id,
|
2018-04-02 21:59:35 +00:00
|
|
|
version: exports.getLatestVersion(),
|
2018-03-28 14:23:36 +00:00
|
|
|
});
|
2018-03-30 20:31:33 +00:00
|
|
|
|
|
|
|
exports.getLatestVersion = () => {
|
|
|
|
const lastMigration = last(migrations);
|
|
|
|
if (!lastMigration) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return lastMigration.version;
|
|
|
|
};
|