mirror of
https://github.com/Foundry376/Mailspring.git
synced 2025-01-13 03:29:46 +08:00
f2dbb2144b
- This commit also correctly sets the labels on the corresponding thread whenever a message has its labels updated - Fix removing labels from messages
467 lines
15 KiB
JavaScript
467 lines
15 KiB
JavaScript
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;
|