import Electron from 'electron'; const { ipcRenderer } = Electron; // tslint:disable: function-name no-require-imports no-var-requires one-variable-per-declaration no-void-expression import _ from 'lodash'; import { ConversationCollection, ConversationModel, } from '../models/conversation'; import { MessageCollection, MessageModel } from '../models/message'; import { MessageAttributes } from '../models/messageType'; import { HexKeyPair } from '../receiver/keypairs'; import { PubKey } from '../session/types'; import { fromArrayBufferToBase64, fromBase64ToArrayBuffer, } from '../session/utils/String'; import { ConversationType } from '../state/ducks/conversations'; const DATABASE_UPDATE_TIMEOUT = 2 * 60 * 1000; // two minutes const SQL_CHANNEL_KEY = 'sql-channel'; const ERASE_SQL_KEY = 'erase-sql-key'; const ERASE_ATTACHMENTS_KEY = 'erase-attachments'; const CLEANUP_ORPHANED_ATTACHMENTS_KEY = 'cleanup-orphaned-attachments'; export const _jobs = Object.create(null); const _DEBUG = false; let _jobCounter = 0; let _shuttingDown = false; let _shutdownCallback: any = null; let _shutdownPromise: any = null; const channels = {} as any; export type StorageItem = { id: string; value: any; }; export type IdentityKey = { id: string; publicKey: ArrayBuffer; firstUse: boolean; nonblockingApproval: boolean; secretKey?: string; // found in medium groups }; export type GuardNode = { ed25519PubKey: string; }; export type SwarmNode = { address: string; ip: string; port: string; pubkey_ed25519: string; pubkey_x25519: string; }; export type ServerToken = { serverUrl: string; token: string; }; export const hasSyncedInitialConfigurationItem = 'hasSyncedInitialConfigurationItem'; const channelsToMake = { shutdown, close, removeDB, getPasswordHash, getIdentityKeyById, removeAllPreKeys, removeAllSignedPreKeys, removeAllContactPreKeys, removeAllContactSignedPreKeys, getGuardNodes, updateGuardNodes, createOrUpdateItem, getItemById, getAllItems, removeItemById, removeAllSessions, getSwarmNodesForPubkey, updateSwarmNodesForPubkey, saveConversation, getConversationById, updateConversation, removeConversation, getAllConversations, getAllConversationIds, getAllPublicConversations, getPubkeysInPublicConversation, savePublicServerToken, getPublicServerTokenByServerUrl, getAllGroupsInvolvingId, searchConversations, searchMessages, searchMessagesInConversation, saveMessage, cleanSeenMessages, cleanLastHashes, updateLastHash, saveSeenMessageHashes, saveMessages, removeMessage, _removeMessages, getUnreadByConversation, getUnreadCountByConversation, removeAllMessagesInConversation, getMessageBySender, getMessageIdsFromServerIds, getMessageById, getAllMessages, getAllMessageIds, getMessagesBySentAt, getExpiredMessages, getOutgoingWithoutExpiresAt, getNextExpiringMessage, getMessagesByConversation, getSeenMessagesByHashList, getLastHashBySnode, getUnprocessedCount, getAllUnprocessed, getUnprocessedById, saveUnprocessed, updateUnprocessedAttempts, updateUnprocessedWithData, removeUnprocessed, removeAllUnprocessed, getNextAttachmentDownloadJobs, saveAttachmentDownloadJob, resetAttachmentDownloadPending, setAttachmentDownloadJobPending, removeAttachmentDownloadJob, removeAllAttachmentDownloadJobs, removeAll, removeAllConversations, removeOtherData, cleanupOrphanedAttachments, // Returning plain JSON getMessagesWithVisualMediaAttachments, getMessagesWithFileAttachments, getAllEncryptionKeyPairsForGroup, getLatestClosedGroupEncryptionKeyPair, addClosedGroupEncryptionKeyPair, isKeyPairAlreadySaved, removeAllClosedGroupEncryptionKeyPairs, }; export function init() { // We listen to a lot of events on ipcRenderer, often on the same channel. This prevents // any warnings that might be sent to the console in that case. ipcRenderer.setMaxListeners(0); _.forEach(channelsToMake, fn => { if (_.isFunction(fn)) { makeChannel(fn.name); } }); ipcRenderer.on( `${SQL_CHANNEL_KEY}-done`, (event, jobId, errorForDisplay, result) => { const job = _getJob(jobId); if (!job) { throw new Error( `Received SQL channel reply to job ${jobId}, but did not have it in our registry!` ); } const { resolve, reject, fnName } = job; if (errorForDisplay) { return reject( new Error( `Error received from SQL channel job ${jobId} (${fnName}): ${errorForDisplay}` ) ); } return resolve(result); } ); } // When IPC arguments are prepared for the cross-process send, they are JSON.stringified. // We can't send ArrayBuffers or BigNumbers (what we get from proto library for dates). function _cleanData(data: any): any { const keys = Object.keys(data); for (let index = 0, max = keys.length; index < max; index += 1) { const key = keys[index]; const value = data[key]; if (value === null || value === undefined) { // eslint-disable-next-line no-continue continue; } if (_.isFunction(value.toNumber)) { // eslint-disable-next-line no-param-reassign data[key] = value.toNumber(); } else if (Array.isArray(value)) { // eslint-disable-next-line no-param-reassign data[key] = value.map(_cleanData); } else if (_.isObject(value)) { // eslint-disable-next-line no-param-reassign data[key] = _cleanData(value); } else if ( typeof value !== 'string' && typeof value !== 'number' && typeof value !== 'boolean' ) { window.log.info(`_cleanData: key ${key} had type ${typeof value}`); } } return data; } async function _shutdown() { if (_shutdownPromise) { return _shutdownPromise; } _shuttingDown = true; const jobKeys = Object.keys(_jobs); window.log.info( `data.shutdown: starting process. ${jobKeys.length} jobs outstanding` ); // No outstanding jobs, return immediately if (jobKeys.length === 0) { return null; } // Outstanding jobs; we need to wait until the last one is done _shutdownPromise = new Promise((resolve, reject) => { _shutdownCallback = (error: any) => { window.log.info('data.shutdown: process complete'); if (error) { return reject(error); } return resolve(undefined); }; }); return _shutdownPromise; } function _makeJob(fnName: string) { if (_shuttingDown && fnName !== 'close') { throw new Error( `Rejecting SQL channel job (${fnName}); application is shutting down` ); } _jobCounter += 1; const id = _jobCounter; if (_DEBUG) { window.log.debug(`SQL channel job ${id} (${fnName}) started`); } _jobs[id] = { fnName, start: Date.now(), }; return id; } function _updateJob(id: number, data: any) { const { resolve, reject } = data; const { fnName, start } = _jobs[id]; _jobs[id] = { ..._jobs[id], ...data, resolve: (value: any) => { _removeJob(id); // const end = Date.now(); // const delta = end - start; // if (delta > 10) { // window.log.debug( // `SQL channel job ${id} (${fnName}) succeeded in ${end - start}ms` // ); // } return resolve(value); }, reject: (error: any) => { _removeJob(id); const end = Date.now(); window.log.warn( `SQL channel job ${id} (${fnName}) failed in ${end - start}ms` ); return reject(error); }, }; } function _removeJob(id: number) { if (_DEBUG) { _jobs[id].complete = true; return; } if (_jobs[id].timer) { clearTimeout(_jobs[id].timer); _jobs[id].timer = null; } // tslint:disable-next-line: no-dynamic-delete delete _jobs[id]; if (_shutdownCallback) { const keys = Object.keys(_jobs); if (keys.length === 0) { _shutdownCallback(); } } } function _getJob(id: number) { return _jobs[id]; } function makeChannel(fnName: string) { channels[fnName] = async (...args: any) => { const jobId = _makeJob(fnName); return new Promise((resolve, reject) => { ipcRenderer.send(SQL_CHANNEL_KEY, jobId, fnName, ...args); _updateJob(jobId, { resolve, reject, args: _DEBUG ? args : null, }); _jobs[jobId].timer = setTimeout( () => reject(new Error(`SQL channel job ${jobId} (${fnName}) timed out`)), DATABASE_UPDATE_TIMEOUT ); }); }; } function keysToArrayBuffer(keys: any, data: any) { const updated = _.cloneDeep(data); for (let i = 0, max = keys.length; i < max; i += 1) { const key = keys[i]; const value = _.get(data, key); if (value) { _.set(updated, key, fromBase64ToArrayBuffer(value)); } } return updated; } function keysFromArrayBuffer(keys: any, data: any) { const updated = _.cloneDeep(data); for (let i = 0, max = keys.length; i < max; i += 1) { const key = keys[i]; const value = _.get(data, key); if (value) { _.set(updated, key, fromArrayBufferToBase64(value)); } } return updated; } // Basic export async function shutdown(): Promise { // Stop accepting new SQL jobs, flush outstanding queue await _shutdown(); await close(); } // Note: will need to restart the app after calling this, to set up afresh export async function close(): Promise { await channels.close(); } // Note: will need to restart the app after calling this, to set up afresh export async function removeDB(): Promise { await channels.removeDB(); } // Password hash export async function getPasswordHash(): Promise { return channels.getPasswordHash(); } // Identity Keys const IDENTITY_KEY_KEYS = ['publicKey']; // Identity Keys // TODO: identity key has different shape depending on how it is called, // so we need to come up with a way to make TS work with all of them export async function getIdentityKeyById( id: string ): Promise { const data = await channels.getIdentityKeyById(id); return keysToArrayBuffer(IDENTITY_KEY_KEYS, data); } // Those removeAll are not used anymore except to cleanup the app since we removed all of those tables export async function removeAllPreKeys(): Promise { await channels.removeAllPreKeys(); } const PRE_KEY_KEYS = ['privateKey', 'publicKey', 'signature']; export async function removeAllSignedPreKeys(): Promise { await channels.removeAllSignedPreKeys(); } export async function removeAllContactPreKeys(): Promise { await channels.removeAllContactPreKeys(); } export async function removeAllContactSignedPreKeys(): Promise { await channels.removeAllContactSignedPreKeys(); } // Guard Nodes export async function getGuardNodes(): Promise> { return channels.getGuardNodes(); } export async function updateGuardNodes(nodes: Array): Promise { return channels.updateGuardNodes(nodes); } // Items const ITEM_KEYS: Object = { identityKey: ['value.pubKey', 'value.privKey'], profileKey: ['value'], }; export async function createOrUpdateItem(data: StorageItem): Promise { const { id } = data; if (!id) { throw new Error( 'createOrUpdateItem: Provided data did not have a truthy id' ); } const keys = (ITEM_KEYS as any)[id]; const updated = Array.isArray(keys) ? keysFromArrayBuffer(keys, data) : data; await channels.createOrUpdateItem(updated); } export async function getItemById( id: string ): Promise { const keys = (ITEM_KEYS as any)[id]; const data = await channels.getItemById(id); return Array.isArray(keys) ? keysToArrayBuffer(keys, data) : data; } export async function getAllItems(): Promise> { const items = await channels.getAllItems(); return _.map(items, item => { const { id } = item; const keys = (ITEM_KEYS as any)[id]; return Array.isArray(keys) ? keysToArrayBuffer(keys, item) : item; }); } export async function removeItemById(id: string): Promise { await channels.removeItemById(id); } // Sessions export async function removeAllSessions(): Promise { await channels.removeAllSessions(); } // Swarm nodes export async function getSwarmNodesForPubkey( pubkey: string ): Promise> { return channels.getSwarmNodesForPubkey(pubkey); } export async function updateSwarmNodesForPubkey( pubkey: string, snodeEdKeys: Array ): Promise { await channels.updateSwarmNodesForPubkey(pubkey, snodeEdKeys); } // Closed group /** * The returned array is ordered based on the timestamp, the latest is at the end. */ export async function getAllEncryptionKeyPairsForGroup( groupPublicKey: string | PubKey ): Promise | undefined> { const pubkey = (groupPublicKey as PubKey).key || (groupPublicKey as string); return channels.getAllEncryptionKeyPairsForGroup(pubkey); } export async function getLatestClosedGroupEncryptionKeyPair( groupPublicKey: string ): Promise { return channels.getLatestClosedGroupEncryptionKeyPair(groupPublicKey); } export async function addClosedGroupEncryptionKeyPair( groupPublicKey: string, keypair: HexKeyPair ): Promise { await channels.addClosedGroupEncryptionKeyPair(groupPublicKey, keypair); } export async function isKeyPairAlreadySaved( groupPublicKey: string, keypair: HexKeyPair ): Promise { return channels.isKeyPairAlreadySaved(groupPublicKey, keypair); } export async function removeAllClosedGroupEncryptionKeyPairs( groupPublicKey: string ): Promise { return channels.removeAllClosedGroupEncryptionKeyPairs(groupPublicKey); } // Conversation export async function saveConversation(data: ConversationType): Promise { const cleaned = _.omit(data, 'isOnline'); await channels.saveConversation(cleaned); } export async function getConversationById( id: string ): Promise { const data = await channels.getConversationById(id); if (data) { return new ConversationModel(data); } return undefined; } export async function updateConversation( data: ConversationType ): Promise { await channels.updateConversation(data); } export async function removeConversation(id: string): Promise { const existing = await getConversationById(id); // Note: It's important to have a fully database-hydrated model to delete here because // it needs to delete all associated on-disk files along with the database delete. if (existing) { await channels.removeConversation(id); await existing.cleanup(); } } export async function getAllConversations(): Promise { const conversations = await channels.getAllConversations(); const collection = new ConversationCollection(); collection.add(conversations); return collection; } export async function getAllConversationIds(): Promise> { const ids = await channels.getAllConversationIds(); return ids; } export async function getAllPublicConversations(): Promise< ConversationCollection > { const conversations = await channels.getAllPublicConversations(); const collection = new ConversationCollection(); collection.add(conversations); return collection; } export async function getPubkeysInPublicConversation( id: string ): Promise> { return channels.getPubkeysInPublicConversation(id); } export async function savePublicServerToken(data: ServerToken): Promise { await channels.savePublicServerToken(data); } export async function getPublicServerTokenByServerUrl( serverUrl: string ): Promise { const token = await channels.getPublicServerTokenByServerUrl(serverUrl); return token; } export async function getAllGroupsInvolvingId( id: string ): Promise { const conversations = await channels.getAllGroupsInvolvingId(id); const collection = new ConversationCollection(); collection.add(conversations); return collection; } export async function searchConversations(query: string): Promise> { const conversations = await channels.searchConversations(query); return conversations; } export async function searchMessages( query: string, { limit }: any = {} ): Promise> { const messages = await channels.searchMessages(query, { limit }); return messages; } /** * Returns just json objects not MessageModel */ export async function searchMessagesInConversation( query: string, conversationId: string, options: { limit: number } | undefined ): Promise { const messages = await channels.searchMessagesInConversation( query, conversationId, { limit: options?.limit } ); return messages; } // Message export async function cleanSeenMessages(): Promise { await channels.cleanSeenMessages(); } export async function cleanLastHashes(): Promise { await channels.cleanLastHashes(); } // TODO: Strictly type the following export async function saveSeenMessageHashes( data: Array<{ expiresAt: number; hash: string; }> ): Promise { await channels.saveSeenMessageHashes(_cleanData(data)); } export async function updateLastHash(data: { convoId: string; snode: string; hash: string; expiresAt: number; }): Promise { await channels.updateLastHash(_cleanData(data)); } export async function saveMessage(data: MessageAttributes): Promise { const id = await channels.saveMessage(_cleanData(data)); window.Whisper.ExpiringMessagesListener.update(); return id; } export async function saveMessages( arrayOfMessages: Array ): Promise { await channels.saveMessages(_cleanData(arrayOfMessages)); } export async function removeMessage(id: string): Promise { const message = await getMessageById(id); // Note: It's important to have a fully database-hydrated model to delete here because // it needs to delete all associated on-disk files along with the database delete. if (message) { await channels.removeMessage(id); await message.cleanup(); } } // Note: this method will not clean up external files, just delete from SQL export async function _removeMessages(ids: Array): Promise { await channels.removeMessage(ids); } export async function getMessageIdsFromServerIds( serverIds: Array, conversationId: string ) { return channels.getMessageIdsFromServerIds(serverIds, conversationId); } export async function getMessageById(id: string): Promise { const message = await channels.getMessageById(id); if (!message) { return null; } return new MessageModel(message); } // For testing only export async function getAllMessages(): Promise { const messages = await channels.getAllMessages(); return new MessageCollection(messages); } export async function getAllMessageIds(): Promise> { const ids = await channels.getAllMessageIds(); return ids; } export async function getMessageBySender( // eslint-disable-next-line camelcase { source, sourceDevice, sent_at, }: { source: string; sourceDevice: number; sent_at: number } ): Promise { const messages = await channels.getMessageBySender({ source, sourceDevice, sent_at, }); if (!messages || !messages.length) { return null; } return new MessageModel(messages[0]); } export async function getUnreadByConversation( conversationId: string ): Promise { const messages = await channels.getUnreadByConversation(conversationId); return new MessageCollection(messages); } // might throw export async function getUnreadCountByConversation( conversationId: string ): Promise { return channels.getUnreadCountByConversation(conversationId); } export async function getMessagesByConversation( conversationId: string, { limit = 100, receivedAt = Number.MAX_VALUE, type = '%' } ): Promise { const messages = await channels.getMessagesByConversation(conversationId, { limit, receivedAt, type, }); return new MessageCollection(messages); } export async function getLastHashBySnode( convoId: string, snode: string ): Promise { return channels.getLastHashBySnode(convoId, snode); } export async function getSeenMessagesByHashList( hashes: Array ): Promise { return channels.getSeenMessagesByHashList(hashes); } export async function removeAllMessagesInConversation( conversationId: string ): Promise { let messages; do { // Yes, we really want the await in the loop. We're deleting 100 at a // time so we don't use too much memory. // eslint-disable-next-line no-await-in-loop messages = await getMessagesByConversation(conversationId, { limit: 100, }); if (!messages.length) { return; } const ids = messages.map(message => message.id); // Note: It's very important that these models are fully hydrated because // we need to delete all associated on-disk files along with the database delete. // eslint-disable-next-line no-await-in-loop await Promise.all(messages.map(message => message.cleanup())); // eslint-disable-next-line no-await-in-loop await channels.removeMessage(ids); } while (messages.length > 0); } export async function getMessagesBySentAt( sentAt: number ): Promise { const messages = await channels.getMessagesBySentAt(sentAt); return new MessageCollection(messages); } export async function getExpiredMessages(): Promise { const messages = await channels.getExpiredMessages(); return new MessageCollection(messages); } export async function getOutgoingWithoutExpiresAt(): Promise< MessageCollection > { const messages = await channels.getOutgoingWithoutExpiresAt(); return new MessageCollection(messages); } export async function getNextExpiringMessage(): Promise { const messages = await channels.getNextExpiringMessage(); return new MessageCollection(messages); } // Unprocessed export async function getUnprocessedCount(): Promise { return channels.getUnprocessedCount(); } export async function getAllUnprocessed(): Promise { return channels.getAllUnprocessed(); } export async function getUnprocessedById(id: string): Promise { return channels.getUnprocessedById(id); } export type UnprocessedParameter = { id: string; version: number; envelope: string; timestamp: number; attempts: number; senderIdentity?: string; }; export async function saveUnprocessed( data: UnprocessedParameter ): Promise { const id = await channels.saveUnprocessed(_cleanData(data)); return id; } export async function updateUnprocessedAttempts( id: string, attempts: number ): Promise { await channels.updateUnprocessedAttempts(id, attempts); } export async function updateUnprocessedWithData( id: string, data: any ): Promise { await channels.updateUnprocessedWithData(id, data); } export async function removeUnprocessed(id: string): Promise { await channels.removeUnprocessed(id); } export async function removeAllUnprocessed(): Promise { await channels.removeAllUnprocessed(); } // Attachment downloads export async function getNextAttachmentDownloadJobs( limit: number ): Promise { return channels.getNextAttachmentDownloadJobs(limit); } export async function saveAttachmentDownloadJob(job: any): Promise { await channels.saveAttachmentDownloadJob(job); } export async function setAttachmentDownloadJobPending( id: string, pending: boolean ): Promise { await channels.setAttachmentDownloadJobPending(id, pending); } export async function resetAttachmentDownloadPending(): Promise { await channels.resetAttachmentDownloadPending(); } export async function removeAttachmentDownloadJob(id: string): Promise { await channels.removeAttachmentDownloadJob(id); } export async function removeAllAttachmentDownloadJobs(): Promise { await channels.removeAllAttachmentDownloadJobs(); } // Other export async function removeAll(): Promise { await channels.removeAll(); } export async function removeAllConversations(): Promise { await channels.removeAllConversations(); } export async function cleanupOrphanedAttachments(): Promise { await callChannel(CLEANUP_ORPHANED_ATTACHMENTS_KEY); } // Note: will need to restart the app after calling this, to set up afresh export async function removeOtherData(): Promise { await Promise.all([ callChannel(ERASE_SQL_KEY), callChannel(ERASE_ATTACHMENTS_KEY), ]); } async function callChannel(name: string): Promise { return new Promise((resolve, reject) => { ipcRenderer.send(name); ipcRenderer.once(`${name}-done`, (event, error) => { if (error) { return reject(error); } return resolve(undefined); }); setTimeout( () => reject(new Error(`callChannel call to ${name} timed out`)), DATABASE_UPDATE_TIMEOUT ); }); } // Functions below here return plain JSON instead of Backbone Models export async function getMessagesWithVisualMediaAttachments( conversationId: string, options?: { limit: number } ): Promise { return channels.getMessagesWithVisualMediaAttachments(conversationId, { limit: options?.limit, }); } export async function getMessagesWithFileAttachments( conversationId: string, options?: { limit: number } ): Promise { return channels.getMessagesWithFileAttachments(conversationId, { limit: options?.limit, }); }