@bobfrankston/mailx 1.0.411 → 1.0.413
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/package.json
CHANGED
|
@@ -177,6 +177,12 @@ export declare class ImapManager extends EventEmitter {
|
|
|
177
177
|
private reconnectOps;
|
|
178
178
|
/** Handle sync errors — classify and emit appropriate UI events */
|
|
179
179
|
private handleSyncError;
|
|
180
|
+
/** Fetch ONLY new messages above highestUid for one account's INBOX —
|
|
181
|
+
* the IDLE callback's hot path. Skips gap detection, backfill, and the
|
|
182
|
+
* server reconcile (each of which fetches a full UID list — multi-second
|
|
183
|
+
* on a large mailbox). The 5-minute STATUS poll path still runs full
|
|
184
|
+
* `syncFolder` so deletions and gaps eventually reconcile. */
|
|
185
|
+
syncInboxNewOnly(accountId: string): Promise<void>;
|
|
180
186
|
/** Sync just INBOX for each account (fast check for new mail) */
|
|
181
187
|
syncInbox(): Promise<void>;
|
|
182
188
|
/** Quick inbox check — per-account lightweight probe.
|
|
@@ -1540,6 +1540,46 @@ export class ImapManager extends EventEmitter {
|
|
|
1540
1540
|
this.emit("accountError", accountId, errMsg, errMsg, isOAuth);
|
|
1541
1541
|
}
|
|
1542
1542
|
}
|
|
1543
|
+
/** Fetch ONLY new messages above highestUid for one account's INBOX —
|
|
1544
|
+
* the IDLE callback's hot path. Skips gap detection, backfill, and the
|
|
1545
|
+
* server reconcile (each of which fetches a full UID list — multi-second
|
|
1546
|
+
* on a large mailbox). The 5-minute STATUS poll path still runs full
|
|
1547
|
+
* `syncFolder` so deletions and gaps eventually reconcile. */
|
|
1548
|
+
async syncInboxNewOnly(accountId) {
|
|
1549
|
+
if (this.isGmailAccount(accountId))
|
|
1550
|
+
return; // IDLE is IMAP-only
|
|
1551
|
+
const inbox = this.db.getFolders(accountId).find(f => f.specialUse === "inbox");
|
|
1552
|
+
if (!inbox)
|
|
1553
|
+
return;
|
|
1554
|
+
try {
|
|
1555
|
+
await this.withConnection(accountId, async (client) => {
|
|
1556
|
+
const highestUid = this.db.getHighestUid(accountId, inbox.id);
|
|
1557
|
+
if (highestUid === 0) {
|
|
1558
|
+
// First sync — fall through to full path so the date-windowed
|
|
1559
|
+
// backfill runs. `syncFolder` handles the no-highestUid case.
|
|
1560
|
+
await this.syncFolder(accountId, inbox.id, client);
|
|
1561
|
+
return;
|
|
1562
|
+
}
|
|
1563
|
+
const fetched = await client.fetchMessagesSinceUid(inbox.path, highestUid, { source: false });
|
|
1564
|
+
const fresh = fetched.filter((m) => m.uid > highestUid);
|
|
1565
|
+
if (fresh.length === 0)
|
|
1566
|
+
return;
|
|
1567
|
+
const stored = await this.storeMessages(accountId, inbox.id, inbox, fresh, highestUid);
|
|
1568
|
+
if (stored > 0) {
|
|
1569
|
+
this.db.recalcFolderCounts(inbox.id);
|
|
1570
|
+
const updated = this.db.getFolders(accountId).find(f => f.id === inbox.id);
|
|
1571
|
+
this.emit("folderCountsChanged", accountId, {
|
|
1572
|
+
[inbox.id]: { total: updated?.totalCount || 0, unread: updated?.unreadCount || 0 }
|
|
1573
|
+
});
|
|
1574
|
+
this.emit("folderSynced", accountId, inbox.id, Date.now());
|
|
1575
|
+
console.log(` [idle-fast] ${accountId}: stored ${stored} new message(s)`);
|
|
1576
|
+
}
|
|
1577
|
+
});
|
|
1578
|
+
}
|
|
1579
|
+
catch (e) {
|
|
1580
|
+
console.error(` [idle-fast] ${accountId}: ${e.message}`);
|
|
1581
|
+
}
|
|
1582
|
+
}
|
|
1543
1583
|
/** Sync just INBOX for each account (fast check for new mail) */
|
|
1544
1584
|
async syncInbox() {
|
|
1545
1585
|
if (this.inboxSyncing)
|
|
@@ -1783,8 +1823,10 @@ export class ImapManager extends EventEmitter {
|
|
|
1783
1823
|
const watchClient = this.createClient(accountId);
|
|
1784
1824
|
const stop = await watchClient.watchMailbox("INBOX", (newCount) => {
|
|
1785
1825
|
console.log(` [idle] ${accountId}: ${newCount} new message(s)`);
|
|
1786
|
-
//
|
|
1787
|
-
|
|
1826
|
+
// Fetch only the new UIDs — the heavyweight gap/reconcile
|
|
1827
|
+
// path runs on the 5-minute STATUS poll, so EXISTS lands
|
|
1828
|
+
// in the UI in roughly one round-trip.
|
|
1829
|
+
this.syncInboxNewOnly(accountId).catch(e => console.error(` [idle] sync error: ${e.message}`));
|
|
1788
1830
|
});
|
|
1789
1831
|
this.watchers.set(accountId, async () => {
|
|
1790
1832
|
await stop();
|