import Electron from 'electron'; const { ipcRenderer } = Electron; // tslint:disable: 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 { getSodium } from '../session/crypto'; import { PubKey } from '../session/types'; import { fromArrayBufferToBase64, fromBase64ToArrayBuffer } from '../session/utils/String'; import { ReduxConversationType } from '../state/ducks/conversations'; import { channels } from './channels'; import { channelsToMake as channelstoMakeOpenGroupV2 } from './opengroups'; 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; 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 interface Snode { ip: string; port: number; pubkey_x25519: string; pubkey_ed25519: string; } export type SwarmNode = Snode & { address: string; }; export type ServerToken = { serverUrl: string; token: string; }; export const hasSyncedInitialConfigurationItem = 'hasSyncedInitialConfigurationItem'; export const lastAvatarUploadTimestamp = 'lastAvatarUploadTimestamp'; export const hasLinkPreviewPopupBeenDisplayed = 'hasLinkPreviewPopupBeenDisplayed'; const channelsToMake = { shutdown, close, removeDB, getPasswordHash, getGuardNodes, updateGuardNodes, createOrUpdateItem, getItemById, getAllItems, removeItemById, getSwarmNodesForPubkey, updateSwarmNodesForPubkey, saveConversation, getConversationById, updateConversation, removeConversation, getAllConversations, getAllOpenGroupV1Conversations, getPubkeysInPublicConversation, getAllGroupsInvolvingId, searchConversations, searchMessages, searchMessagesInConversation, saveMessage, cleanSeenMessages, cleanLastHashes, updateLastHash, saveSeenMessageHashes, saveMessages, removeMessage, _removeMessages, getUnreadByConversation, getUnreadCountByConversation, removeAllMessagesInConversation, getMessageBySender, getMessageBySenderAndServerId, getMessageBySenderAndServerTimestamp, getMessageIdsFromServerIds, getMessageById, getMessagesBySentAt, getExpiredMessages, getOutgoingWithoutExpiresAt, getNextExpiringMessage, getMessagesByConversation, getFirstUnreadMessageIdInConversation, 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, removeAllClosedGroupEncryptionKeyPairs, removeOneOpenGroupV1Message, // open group v2 ...channelstoMakeOpenGroupV2, }; 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 (_.isFunction(value)) { // just skip a function which has not a toNumber function. We don't want to save a function to the db. // an attachment comes with a toJson() function // tslint:disable-next-line: no-dynamic-delete delete data[key]; } else if (Array.isArray(value)) { // eslint-disable-next-line no-param-reassign data[key] = value.map(_cleanData); } else if (_.isObject(value) && value instanceof File) { // eslint-disable-next-line no-param-reassign data[key] = { name: value.name, path: value.path, size: value.size, type: value.type }; } else if (_.isObject(value)) { // eslint-disable-next-line no-param-reassign data[key] = _cleanData(value); } else if (_.isBoolean(value)) { // eslint-disable-next-line no-param-reassign data[key] = value ? 1 : 0; } 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); if (_DEBUG) { 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) { global.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(); } // 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 generateAttachmentKeyIfEmpty() { const existingKey = await getItemById('local_attachment_encrypted_key'); if (!existingKey) { const sodium = await getSodium(); const encryptingKey = sodium.to_hex(sodium.randombytes_buf(32)); await createOrUpdateItem({ id: 'local_attachment_encrypted_key', value: encryptingKey, }); // be sure to write the new key to the cache. so we can access it straight away window.textsecure.storage.put('local_attachment_encrypted_key', encryptingKey); } } 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); } // 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 removeAllClosedGroupEncryptionKeyPairs( groupPublicKey: string ): Promise { return channels.removeAllClosedGroupEncryptionKeyPairs(groupPublicKey); } // Conversation export async function saveConversation(data: ReduxConversationType): 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: ReduxConversationType): 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 getAllOpenGroupV1Conversations(): Promise { const conversations = await channels.getAllOpenGroupV1Conversations(); const collection = new ConversationCollection(); collection.add(conversations); return collection; } /** * This returns at most MAX_PUBKEYS_MEMBERS members, the last MAX_PUBKEYS_MEMBERS members who wrote in the chat */ export async function getPubkeysInPublicConversation(id: string): Promise> { return channels.getPubkeysInPublicConversation(id); } 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 cleanedData = _cleanData(data); const id = await channels.saveMessage(cleanedData); 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, true); // 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 | Array, conversationId: string ): Promise | undefined> { return channels.getMessageIdsFromServerIds(serverIds, conversationId); } export async function getMessageById( id: string, skipTimerInit: boolean = false ): Promise { const message = await channels.getMessageById(id); if (!message) { return null; } if (skipTimerInit) { message.skipTimerInit = skipTimerInit; } return new MessageModel(message); } export async function getMessageBySender({ source, sourceDevice, sentAt, }: { source: string; sourceDevice: number; sentAt: number; }): Promise { const messages = await channels.getMessageBySender({ source, sourceDevice, sentAt, }); if (!messages || !messages.length) { return null; } return new MessageModel(messages[0]); } export async function getMessageBySenderAndServerId({ source, serverId, }: { source: string; serverId: number; }): Promise { const messages = await channels.getMessageBySenderAndServerId({ source, serverId, }); if (!messages || !messages.length) { return null; } return new MessageModel(messages[0]); } export async function getMessageBySenderAndServerTimestamp({ source, serverTimestamp, }: { source: string; serverTimestamp: number; }): Promise { const messages = await channels.getMessageBySenderAndServerTimestamp({ source, serverTimestamp, }); 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 = '%', skipTimerInit = false } ): Promise { const messages = await channels.getMessagesByConversation(conversationId, { limit, receivedAt, type, }); if (skipTimerInit) { for (const message of messages) { message.skipTimerInit = skipTimerInit; } } return new MessageCollection(messages); } export async function getFirstUnreadMessageIdInConversation( conversationId: string ): Promise { return channels.getFirstUnreadMessageIdInConversation(conversationId); } 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: 500, }); 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 { 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 ? 1 : 0); } 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, }); } export const SNODE_POOL_ITEM_ID = 'SNODE_POOL_ITEM_ID'; export async function getSnodePoolFromDb(): Promise | null> { // this is currently all stored as a big string as we don't really need to do anything with them (no filtering or anything) // everything is made in memory and written to disk const snodesJson = await exports.getItemById(SNODE_POOL_ITEM_ID); if (!snodesJson || !snodesJson.value) { return null; } return JSON.parse(snodesJson.value); } export async function updateSnodePoolOnDb(snodesAsJsonString: string): Promise { await exports.createOrUpdateItem({ id: SNODE_POOL_ITEM_ID, value: snodesAsJsonString }); } /** Returns the number of message left to remove (opengroupv1) */ export async function removeOneOpenGroupV1Message(): Promise { return channels.removeOneOpenGroupV1Message(); }