0
Fork 0
mirror of https://github.com/TryGhost/Ghost.git synced 2025-01-06 22:40:14 -05:00

🐛 Increased email timeout when sending to lots of recipients (#19628)

fixes ENG-603

If the database is under load, it is possible that sending batches can
be slow due to connection timouts. In that case, we need to increase the
total timeout when sending an email.

Apart from that, this commit adds more logging.
This commit is contained in:
Simon Backx 2024-01-31 12:46:00 +01:00 committed by GitHub
parent 25a8ab33ca
commit 13c582205c
No known key found for this signature in database
GPG key ID: B5690EEEBB952194

View file

@ -130,8 +130,7 @@ class BatchSendingService {
async emailJob({emailId}) { async emailJob({emailId}) {
logging.info(`Starting email job for email ${emailId}`); logging.info(`Starting email job for email ${emailId}`);
// We'll stop all automatic DB retries after this date const startTime = Date.now();
const retryCutOffTime = new Date(Date.now() + this.#BEFORE_RETRY_CONFIG.maxTime);
// Check if email is 'pending' only + change status to submitting in one transaction. // Check if email is 'pending' only + change status to submitting in one transaction.
// This allows us to have a lock around the email job that makes sure an email can only have one active job. // This allows us to have a lock around the email job that makes sure an email can only have one active job.
@ -146,6 +145,12 @@ class BatchSendingService {
return; return;
} }
// We'll stop all automatic DB retries after this date
const expectedBatchCount = Math.ceil(email.get('email_count') / 1000);
const minimumSecondsPerBatch = 26; // In case of database issues, we make sure we expand the retry window relative to the amount of batches
const stopAfter = Math.max(expectedBatchCount * minimumSecondsPerBatch * 1000, this.#BEFORE_RETRY_CONFIG.maxTime);
const retryCutOffTime = new Date(startTime + stopAfter);
// Save a strict cutoff time for retries // Save a strict cutoff time for retries
email._retryCutOffTime = retryCutOffTime; email._retryCutOffTime = retryCutOffTime;
@ -583,18 +588,28 @@ class BatchSendingService {
options = {...options, stopAfterDate}; options = {...options, stopAfterDate};
} }
} }
const retryCount = (options.retryCount ?? 0);
try { try {
return await func(); if (retryCount > 0) {
logging.info(`[BULK_EMAIL_DB_RETRY] ${options.description} - Retrying ${retryCount + 1}th try`);
} else {
logging.info(`[BULK_EMAIL_DB_RETRY] ${options.description} - Started (1st try)`);
}
const response = await func();
logging.info(`[BULK_EMAIL_DB_RETRY] ${options.description} - Finished (after ${retryCount + 1}${retryCount === 0 ? 'st try' : ' tries'})`);
return response;
} catch (e) { } catch (e) {
const retryCount = (options.retryCount ?? 0);
const sleep = (options.sleep ?? 0); const sleep = (options.sleep ?? 0);
if (retryCount >= options.maxRetries || (options.stopAfterDate && (new Date(Date.now() + sleep)) > options.stopAfterDate)) { if (retryCount >= options.maxRetries || (options.stopAfterDate && (new Date(Date.now() + sleep)) > options.stopAfterDate)) {
if (retryCount > 0) { if (retryCount > 0) {
const ghostError = new errors.EmailError({ const ghostError = new errors.EmailError({
err: e, err: e,
code: 'BULK_EMAIL_DB_RETRY', code: 'BULK_EMAIL_DB_RETRY',
message: `[BULK_EMAIL_DB_RETRY] ${options.description} - Stopped retrying`, message: `[BULK_EMAIL_DB_RETRY] ${options.description} - Failed and stopped retrying: ${retryCount >= options.maxRetries ? 'max retries reached' : 'max time reached'}`,
context: e.message context: e.message
}); });
@ -606,7 +621,7 @@ class BatchSendingService {
const ghostError = new errors.EmailError({ const ghostError = new errors.EmailError({
err: e, err: e,
code: 'BULK_EMAIL_DB_RETRY', code: 'BULK_EMAIL_DB_RETRY',
message: `[BULK_EMAIL_DB_RETRY] ${options.description} - After ${retryCount} retries`, message: `[BULK_EMAIL_DB_RETRY] ${options.description} - Failed (${retryCount + 1}${retryCount === 0 ? 'st' : 'th'} try)`,
context: e.message context: e.message
}); });