You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			257 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			JavaScript
		
	
			
		
		
	
	
			257 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			JavaScript
		
	
/* global Whisper: false */
 | 
						|
/* global Backbone: false */
 | 
						|
/* global _: false */
 | 
						|
 | 
						|
/* eslint-disable more/no-then */
 | 
						|
 | 
						|
// eslint-disable-next-line func-names
 | 
						|
(function () {
 | 
						|
  'use strict';
 | 
						|
 | 
						|
  const { Migrations } = window.Signal;
 | 
						|
 | 
						|
  window.Whisper = window.Whisper || {};
 | 
						|
  window.Whisper.Database = window.Whisper.Database || {};
 | 
						|
  window.Whisper.Database.id = window.Whisper.Database.id || 'signal';
 | 
						|
  window.Whisper.Database.nolog = true;
 | 
						|
 | 
						|
  Whisper.Database.handleDOMException = (prefix, error, reject) => {
 | 
						|
    console.log(
 | 
						|
      `${prefix}:`,
 | 
						|
      error && error.name,
 | 
						|
      error && error.message,
 | 
						|
      error && error.code
 | 
						|
    );
 | 
						|
    reject(error || new Error(prefix));
 | 
						|
  };
 | 
						|
 | 
						|
  function clearStores(db, names) {
 | 
						|
    return new Promise(((resolve, reject) => {
 | 
						|
      const storeNames = names || db.objectStoreNames;
 | 
						|
      console.log('Clearing these indexeddb stores:', storeNames);
 | 
						|
      const transaction = db.transaction(storeNames, 'readwrite');
 | 
						|
 | 
						|
      let finished = false;
 | 
						|
      const finish = (via) => {
 | 
						|
        console.log('clearing all stores done via', via);
 | 
						|
        if (finished) {
 | 
						|
          resolve();
 | 
						|
        }
 | 
						|
        finished = true;
 | 
						|
      };
 | 
						|
 | 
						|
      transaction.oncomplete = finish.bind(null, 'transaction complete');
 | 
						|
      transaction.onerror = () => {
 | 
						|
        Whisper.Database.handleDOMException(
 | 
						|
          'clearStores transaction error',
 | 
						|
          transaction.error,
 | 
						|
          reject
 | 
						|
        );
 | 
						|
      };
 | 
						|
 | 
						|
      let count = 0;
 | 
						|
 | 
						|
      // can't use built-in .forEach because db.objectStoreNames is not a plain array
 | 
						|
      _.forEach(storeNames, (storeName) => {
 | 
						|
        const store = transaction.objectStore(storeName);
 | 
						|
        const request = store.clear();
 | 
						|
 | 
						|
        request.onsuccess = () => {
 | 
						|
          count += 1;
 | 
						|
          console.log('Done clearing store', storeName);
 | 
						|
 | 
						|
          if (count >= storeNames.length) {
 | 
						|
            console.log('Done clearing indexeddb stores');
 | 
						|
            finish('clears complete');
 | 
						|
          }
 | 
						|
        };
 | 
						|
 | 
						|
        request.onerror = () => {
 | 
						|
          Whisper.Database.handleDOMException(
 | 
						|
            'clearStores request error',
 | 
						|
            request.error,
 | 
						|
            reject
 | 
						|
          );
 | 
						|
        };
 | 
						|
      });
 | 
						|
    }));
 | 
						|
  }
 | 
						|
 | 
						|
  Whisper.Database.open = () => {
 | 
						|
    const { migrations } = Whisper.Database;
 | 
						|
    const { version } = migrations[migrations.length - 1];
 | 
						|
    const DBOpenRequest = window.indexedDB.open(Whisper.Database.id, version);
 | 
						|
 | 
						|
    return new Promise(((resolve, reject) => {
 | 
						|
      // these two event handlers act on the IDBDatabase object,
 | 
						|
      // when the database is opened successfully, or not
 | 
						|
      DBOpenRequest.onerror = reject;
 | 
						|
      DBOpenRequest.onsuccess = () => resolve(DBOpenRequest.result);
 | 
						|
 | 
						|
      // This event handles the event whereby a new version of
 | 
						|
      // the database needs to be created Either one has not
 | 
						|
      // been created before, or a new version number has been
 | 
						|
      // submitted via the window.indexedDB.open line above
 | 
						|
      DBOpenRequest.onupgradeneeded = reject;
 | 
						|
    }));
 | 
						|
  };
 | 
						|
 | 
						|
  Whisper.Database.clear = async () => {
 | 
						|
    const db = await Whisper.Database.open();
 | 
						|
    return clearStores(db);
 | 
						|
  };
 | 
						|
 | 
						|
  Whisper.Database.clearStores = async (storeNames) => {
 | 
						|
    const db = await Whisper.Database.open();
 | 
						|
    return clearStores(db, storeNames);
 | 
						|
  };
 | 
						|
 | 
						|
  Whisper.Database.close = () => window.wrapDeferred(Backbone.sync('closeall'));
 | 
						|
 | 
						|
  Whisper.Database.drop = () =>
 | 
						|
    new Promise(((resolve, reject) => {
 | 
						|
      const request = window.indexedDB.deleteDatabase(Whisper.Database.id);
 | 
						|
 | 
						|
      request.onblocked = () => {
 | 
						|
        reject(new Error('Error deleting database: Blocked.'));
 | 
						|
      };
 | 
						|
      request.onupgradeneeded = () => {
 | 
						|
        reject(new Error('Error deleting database: Upgrade needed.'));
 | 
						|
      };
 | 
						|
      request.onerror = () => {
 | 
						|
        reject(new Error('Error deleting database.'));
 | 
						|
      };
 | 
						|
 | 
						|
      request.onsuccess = resolve;
 | 
						|
    }));
 | 
						|
 | 
						|
  Whisper.Database.migrations = [
 | 
						|
    {
 | 
						|
      version: '12.0',
 | 
						|
      migrate(transaction, next) {
 | 
						|
        console.log('migration 12.0');
 | 
						|
        console.log('creating object stores');
 | 
						|
        const messages = transaction.db.createObjectStore('messages');
 | 
						|
        messages.createIndex('conversation', ['conversationId', 'received_at'], {
 | 
						|
          unique: false,
 | 
						|
        });
 | 
						|
        messages.createIndex('receipt', 'sent_at', { unique: false });
 | 
						|
        messages.createIndex('unread', ['conversationId', 'unread'], { unique: false });
 | 
						|
        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');
 | 
						|
 | 
						|
        console.log('creating debug log');
 | 
						|
        transaction.db.createObjectStore('debug');
 | 
						|
 | 
						|
        next();
 | 
						|
      },
 | 
						|
    },
 | 
						|
    {
 | 
						|
      version: '13.0',
 | 
						|
      migrate(transaction, next) {
 | 
						|
        console.log('migration 13.0');
 | 
						|
        console.log('Adding fields to identity keys');
 | 
						|
        const identityKeys = transaction.objectStore('identityKeys');
 | 
						|
        const request = identityKeys.openCursor();
 | 
						|
        const promises = [];
 | 
						|
        request.onsuccess = (event) => {
 | 
						|
          const cursor = event.target.result;
 | 
						|
          if (cursor) {
 | 
						|
            const attributes = cursor.value;
 | 
						|
            attributes.timestamp = 0;
 | 
						|
            attributes.firstUse = false;
 | 
						|
            attributes.nonblockingApproval = false;
 | 
						|
            attributes.verified = 0;
 | 
						|
            promises.push(new Promise(((resolve, reject) => {
 | 
						|
              const putRequest = identityKeys.put(attributes, attributes.id);
 | 
						|
              putRequest.onsuccess = resolve;
 | 
						|
              putRequest.onerror = (e) => {
 | 
						|
                console.log(e);
 | 
						|
                reject(e);
 | 
						|
              };
 | 
						|
            })));
 | 
						|
            cursor.continue();
 | 
						|
          } else {
 | 
						|
            // no more results
 | 
						|
            Promise.all(promises).then(() => {
 | 
						|
              next();
 | 
						|
            });
 | 
						|
          }
 | 
						|
        };
 | 
						|
        request.onerror = (event) => {
 | 
						|
          console.log(event);
 | 
						|
        };
 | 
						|
      },
 | 
						|
    },
 | 
						|
    {
 | 
						|
      version: '14.0',
 | 
						|
      migrate(transaction, next) {
 | 
						|
        console.log('migration 14.0');
 | 
						|
        console.log('Adding unprocessed message store');
 | 
						|
        const unprocessed = transaction.db.createObjectStore('unprocessed');
 | 
						|
        unprocessed.createIndex('received', 'timestamp', { unique: false });
 | 
						|
        next();
 | 
						|
      },
 | 
						|
    },
 | 
						|
    {
 | 
						|
      version: '15.0',
 | 
						|
      migrate(transaction, next) {
 | 
						|
        console.log('migration 15.0');
 | 
						|
        console.log('Adding messages index for de-duplication');
 | 
						|
        const messages = transaction.objectStore('messages');
 | 
						|
        messages.createIndex('unique', ['source', 'sourceDevice', 'sent_at'], {
 | 
						|
          unique: true,
 | 
						|
        });
 | 
						|
        next();
 | 
						|
      },
 | 
						|
    },
 | 
						|
    {
 | 
						|
      version: '16.0',
 | 
						|
      migrate(transaction, next) {
 | 
						|
        console.log('migration 16.0');
 | 
						|
        console.log('Dropping log table, since we now log to disk');
 | 
						|
        transaction.db.deleteObjectStore('debug');
 | 
						|
        next();
 | 
						|
      },
 | 
						|
    },
 | 
						|
    {
 | 
						|
      version: 17,
 | 
						|
      async migrate(transaction, next) {
 | 
						|
        console.log('migration 17');
 | 
						|
        console.log('Start migration to database version 17');
 | 
						|
 | 
						|
        const start = Date.now();
 | 
						|
        await Migrations.V17.run(transaction);
 | 
						|
        const duration = Date.now() - start;
 | 
						|
 | 
						|
        console.log(
 | 
						|
          'Complete migration to database version 17.',
 | 
						|
          `Duration: ${duration}ms`
 | 
						|
        );
 | 
						|
        next();
 | 
						|
      },
 | 
						|
    },
 | 
						|
  ];
 | 
						|
}());
 |