const _ = require('underscore'); const Imap = require('imap'); const {PromiseUtils, IMAPConnection, PubsubConnector} = require('nylas-core'); const {Capabilities} = IMAPConnection; const MessageFlagAttributes = ['id', 'threadId', 'folderImapUID', 'unread', 'starred', 'folderImapXGMLabels'] const SHALLOW_SCAN_UID_COUNT = 1000; const FETCH_MESSAGES_FIRST_COUNT = 100; const FETCH_MESSAGES_COUNT = 200; class FetchMessagesInFolder { constructor(category, options, logger) { this._imap = null this._box = null this._db = null this._category = category; this._options = options; this._logger = logger; if (!this._logger) { throw new Error("FetchMessagesInFolder requires a logger") } if (!this._category) { throw new Error("FetchMessagesInFolder requires a category") } } description() { return `FetchMessagesInFolder (${this._category.name} - ${this._category.id})\n Options: ${JSON.stringify(this._options)}`; } _getLowerBoundUID(count) { return count ? Math.max(1, this._box.uidnext - count) : 1; } _recoverFromUIDInvalidity() { // UID invalidity means the server has asked us to delete all the UIDs for // this folder and start from scratch. Instead of deleting all the messages, // we just remove the category ID and UID. We may re-assign the same message // the same UID. Otherwise they're eventually garbage collected. const {Message} = this._db; return this._db.sequelize.transaction((transaction) => Message.update({ folderImapUID: null, folderId: null, }, { transaction: transaction, where: { folderId: this._category.id, }, }) ) } _updateMessageAttributes(remoteUIDAttributes, localMessageAttributes) { const {sequelize, Label} = this._db; const messageAttributesMap = {}; for (const msg of localMessageAttributes) { messageAttributesMap[msg.folderImapUID] = msg; } const createdUIDs = []; const flagChangeMessages = []; return Label.findAll().then((preloadedLabels) => { Object.keys(remoteUIDAttributes).forEach((uid) => { const msg = messageAttributesMap[uid]; const attrs = remoteUIDAttributes[uid]; if (!msg) { createdUIDs.push(uid); return; } const unread = !attrs.flags.includes('\\Seen'); const starred = attrs.flags.includes('\\Flagged'); const xGmLabels = attrs['x-gm-labels']; const xGmLabelsJSON = xGmLabels ? JSON.stringify(xGmLabels) : null; if (msg.folderImapXGMLabels !== xGmLabelsJSON) { msg.setLabelsFromXGM(xGmLabels, {Label, preloadedLabels}); } if (msg.unread !== unread || msg.starred !== starred) { msg.unread = unread; msg.starred = starred; flagChangeMessages.push(msg); } }) this._logger.info({ flag_changes: flagChangeMessages.length, }, `FetchMessagesInFolder: found flag changes`) if (createdUIDs.length > 0) { this._logger.info({ new_messages: createdUIDs.length, }, `FetchMessagesInFolder: found new messages. These will not be processed because we assume that they will be assigned uid = uidnext, and will be picked up in the next sync when we discover unseen messages.`) } if (flagChangeMessages.length === 0) { return Promise.resolve(); } return sequelize.transaction((transaction) => Promise.all(flagChangeMessages.map(m => m.save({ fields: MessageFlagAttributes, transaction, }))) ); }); } _removeDeletedMessages(remoteUIDAttributes, localMessageAttributes) { const {Message} = this._db; const removedUIDs = localMessageAttributes .filter(msg => !remoteUIDAttributes[msg.folderImapUID]) .map(msg => msg.folderImapUID) this._logger.info({ removed_messages: removedUIDs.length, }, `FetchMessagesInFolder: found messages no longer in the folder`) if (removedUIDs.length === 0) { return Promise.resolve(); } return this._db.sequelize.transaction((transaction) => Message.update({ folderImapUID: null, folderId: null, }, { transaction, where: { folderId: this._category.id, folderImapUID: removedUIDs, }, }) ); } _getDesiredMIMEParts(struct) { const desired = []; const available = []; const unseen = [struct]; while (unseen.length > 0) { const part = unseen.shift(); if (part instanceof Array) { unseen.push(...part); } else { const mimetype = `${part.type}/${part.subtype}`; available.push(mimetype); if (['text/plain', 'text/html', 'application/pgp-encrypted'].includes(mimetype)) { desired.push({id: part.partID, mimetype}); } } } if (desired.length === 0) { this._logger.warn({ available_options: available.join(', '), }, `FetchMessagesInFolder: Could not find good part`) } return desired; } _fetchMessagesAndQueueForProcessing(range) { const uidsByPart = {}; return this._box.fetchEach(range, {struct: true}, ({attributes}) => { const desiredParts = this._getDesiredMIMEParts(attributes.struct); if (desiredParts.length === 0) { return; } const key = JSON.stringify(desiredParts); uidsByPart[key] = uidsByPart[key] || []; uidsByPart[key].push(attributes.uid); }) .then(() => { return PromiseUtils.each(Object.keys(uidsByPart), (key) => { const uids = uidsByPart[key]; const desiredParts = JSON.parse(key); const bodies = ['HEADER'].concat(desiredParts.map(p => p.id)); this._logger.info({ key, num_messages: uids.length, }, `FetchMessagesInFolder: Fetching parts for messages`) // note: the order of UIDs in the array doesn't matter, Gmail always // returns them in ascending (oldest => newest) order. return this._box.fetchEach(uids, {bodies, struct: true}, (msg) => { msg.body = {}; for (const {id, mimetype} of desiredParts) { msg.body[mimetype] = msg.parts[id]; } this._processMessage(msg); }); }); }); } _collectFilesFromStruct(message, struct) { const {File} = this._db; let collected = []; for (const part of struct) { if (part.constructor === Array) { collected = collected.concat(this._collectFilesFromStruct(message, part)); } else if (part.type !== 'text' && part.disposition) { // Only exposes partId for inline attachments const partId = part.disposition.type === 'inline' ? part.partID : null; const filename = part.disposition.params ? part.disposition.params.filename : null; collected.push(File.build({ filename: filename, partId: partId, messageId: message.id, contentType: `${part.type}/${part.subtype}`, accountId: this._db.accountId, size: part.size, })); } } return collected; } _processMessage({attributes, headers, body}) { const {Message, Label, accountId} = this._db; const hash = Message.hashForHeaders(headers); const parsedHeaders = Imap.parseHeader(headers); for (const key of ['x-gm-thrid', 'x-gm-msgid', 'x-gm-labels']) { parsedHeaders[key] = attributes[key]; } const values = { hash: hash, accountId: this._db.accountId, body: body['text/html'] || body['text/plain'] || body['application/pgp-encrypted'] || '', snippet: body['text/plain'] ? body['text/plain'].substr(0, 255) : null, unread: !attributes.flags.includes('\\Seen'), starred: attributes.flags.includes('\\Flagged'), date: attributes.date, folderImapUID: attributes.uid, folderId: this._category.id, headers: parsedHeaders, headerMessageId: parsedHeaders['message-id'] ? parsedHeaders['message-id'][0] : '', subject: parsedHeaders.subject[0], } let created = false; Message.find({where: {hash}}) .then((existing) => { created = !existing; return existing ? existing.update(values) : Message.create(values); }) .then((message) => message.setLabelsFromXGM(attributes['x-gm-labels'], {Label}).thenReturn(message) ) .then((message) => { if (created) { this._logger.info({ message_id: message.id, uid: attributes.uid, }, `FetchMessagesInFolder: Created message`) const files = this._collectFilesFromStruct(message, attributes.struct); if (files.length > 0) { this._db.sequelize.transaction((transaction) => Promise.all(files.map(f => f.save({transaction}))) ) } PubsubConnector.queueProcessMessage({accountId, messageId: message.id}); } else { this._logger.info({ message_id: message.id, uid: attributes.uid, }, `FetchMessagesInFolder: Updated message`) } }) return null; } _openMailboxAndEnsureValidity() { return this._imap.openBox(this._category.name).then((box) => { if (box.persistentUIDs === false) { return Promise.reject(new Error("Mailbox does not support persistentUIDs.")) } const lastUIDValidity = this._category.syncState.uidvalidity; if (lastUIDValidity && (box.uidvalidity !== lastUIDValidity)) { this._logger.info({ boxname: box.name, categoryname: this._category.name, remoteuidvalidity: box.uidvalidity, localuidvalidity: lastUIDValidity, }, `FetchMessagesInFolder: Recovering from UIDInvalidity`); return this._recoverFromUIDInvalidity().thenReturn(box) } return Promise.resolve(box); }); } _fetchUnsyncedMessages() { const savedSyncState = this._category.syncState; const isFirstSync = savedSyncState.fetchedmax === undefined; const boxUidnext = this._box.uidnext; const boxUidvalidity = this._box.uidvalidity; const desiredRanges = []; this._logger.info({ range: `${savedSyncState.fetchedmin}:${savedSyncState.fetchedmax}`, }, `FetchMessagesInFolder: Fetching messages.`) // Todo: In the future, this is where logic should go that limits // sync based on number of messages / age of messages. if (isFirstSync) { const lowerbound = Math.max(1, boxUidnext - FETCH_MESSAGES_FIRST_COUNT); desiredRanges.push({min: lowerbound, max: boxUidnext}) } else { if (savedSyncState.fetchedmax < boxUidnext) { desiredRanges.push({min: savedSyncState.fetchedmax, max: boxUidnext}) } else { this._logger.info('FetchMessagesInFolder: fetchedmax == uidnext, nothing more recent to fetch.') } if (savedSyncState.fetchedmin > 1) { const lowerbound = Math.max(1, savedSyncState.fetchedmin - FETCH_MESSAGES_COUNT); desiredRanges.push({min: lowerbound, max: savedSyncState.fetchedmin}) } else { this._logger.info("FetchMessagesInFolder: fetchedmin == 1, nothing older to fetch.") } } return PromiseUtils.each(desiredRanges, ({min, max}) => { this._logger.info({ range: `${min}:${max}`, }, `FetchMessagesInFolder: Fetching range`); return this._fetchMessagesAndQueueForProcessing(`${min}:${max}`).then(() => { const {fetchedmin, fetchedmax} = this._category.syncState; return this.updateFolderSyncState({ fetchedmin: fetchedmin ? Math.min(fetchedmin, min) : min, fetchedmax: fetchedmax ? Math.max(fetchedmax, max) : max, uidvalidity: boxUidvalidity, timeFetchedUnseen: Date.now(), }); }) }) .then(() => { this._logger.info(`FetchMessagesInFolder: Fetching messages finished`); }); } _runScan() { const {fetchedmin, fetchedmax} = this._category.syncState; if ((fetchedmin === undefined) || (fetchedmax === undefined)) { throw new Error("Unseen messages must be fetched at least once before the first update/delete scan.") } return this._shouldRunDeepScan() ? this._runDeepScan() : this._runShallowScan() } _shouldRunDeepScan() { const {timeDeepScan} = this._category.syncState; return Date.now() - (timeDeepScan || 0) > this._options.deepFolderScan; } _runShallowScan() { const {highestmodseq} = this._category.syncState; const nextHighestmodseq = this._box.highestmodseq; let shallowFetch = null; if (this._imap.serverSupports(Capabilities.Condstore)) { this._logger.info(`FetchMessagesInFolder: Shallow attribute scan (using CONDSTORE)`) if (nextHighestmodseq === highestmodseq) { this._logger.info('FetchMessagesInFolder: highestmodseq matches, nothing more to fetch') return Promise.resolve(); } shallowFetch = this._box.fetchUIDAttributes(`1:*`, {changedsince: highestmodseq}); } else { const range = `${this._getLowerBoundUID(SHALLOW_SCAN_UID_COUNT)}:*`; this._logger.info({range}, `FetchMessagesInFolder: Shallow attribute scan`) shallowFetch = this._box.fetchUIDAttributes(range); } return shallowFetch .then((remoteUIDAttributes) => ( this._db.Message.findAll({ where: {folderId: this._category.id}, attributes: MessageFlagAttributes, }) .then((localMessageAttributes) => ( this._updateMessageAttributes(remoteUIDAttributes, localMessageAttributes) )) .then(() => { this._logger.info(`FetchMessagesInFolder: finished fetching changes to messages`); return this.updateFolderSyncState({ highestmodseq: nextHighestmodseq, timeShallowScan: Date.now(), }) }) )) } _runDeepScan() { const {Message} = this._db; const {fetchedmin, fetchedmax} = this._category.syncState; const range = `${fetchedmin}:${fetchedmax}`; this._logger.info({range}, `FetchMessagesInFolder: Deep attribute scan: fetching attributes in range`) return this._box.fetchUIDAttributes(range) .then((remoteUIDAttributes) => { return Message.findAll({ where: {folderId: this._category.id}, attributes: MessageFlagAttributes, }) .then((localMessageAttributes) => ( PromiseUtils.props({ updates: this._updateMessageAttributes(remoteUIDAttributes, localMessageAttributes), deletes: this._removeDeletedMessages(remoteUIDAttributes, localMessageAttributes), }) )) .then(() => { this._logger.info(`FetchMessagesInFolder: Deep scan finished.`); return this.updateFolderSyncState({ highestmodseq: this._box.highestmodseq, timeDeepScan: Date.now(), timeShallowScan: Date.now(), }) }) }); } updateFolderSyncState(newState) { if (_.isMatch(this._category.syncState, newState)) { return Promise.resolve(); } this._category.syncState = Object.assign(this._category.syncState, newState); return this._category.save(); } run(db, imap) { this._db = db; this._imap = imap; return this._openMailboxAndEnsureValidity().then((box) => { this._box = box return this._fetchUnsyncedMessages().then(() => this._runScan() ) }) } } module.exports = FetchMessagesInFolder;