10 changed files with 274 additions and 113 deletions
@ -0,0 +1,66 @@ |
|||
// chat.js
|
|||
// Copyright (C) 2022 DTP Technologies, LLC
|
|||
// License: Apache-2.0
|
|||
|
|||
'use strict'; |
|||
|
|||
const path = require('path'); |
|||
require('dotenv').config({ path: path.resolve(__dirname, '..', '..', '.env') }); |
|||
|
|||
const mongoose = require('mongoose'); |
|||
|
|||
const { SiteLog, SiteWorker, SiteAsync } = require(path.join(__dirname, '..', '..', 'lib', 'site-lib')); |
|||
|
|||
module.rootPath = path.resolve(__dirname, '..', '..'); |
|||
module.pkg = require(path.resolve(__dirname, '..', '..', 'package.json')); |
|||
|
|||
module.config = { |
|||
environment: process.env.NODE_ENV, |
|||
root: module.rootPath, |
|||
component: { name: 'chatWorker', slug: 'chat-worker' }, |
|||
}; |
|||
|
|||
module.config.site = require(path.join(module.rootPath, 'config', 'site')); |
|||
|
|||
class ChatWorker extends SiteWorker { |
|||
|
|||
constructor (dtp) { |
|||
super(dtp, dtp.config.component); |
|||
} |
|||
|
|||
async start ( ) { |
|||
await super.start(); |
|||
|
|||
await this.loadProcessor(path.join(__dirname, 'chat', 'job', 'chat-room-clear.js')); |
|||
await this.loadProcessor(path.join(__dirname, 'chat', 'job', 'chat-room-delete.js')); |
|||
|
|||
await this.startProcessors(); |
|||
} |
|||
|
|||
async stop ( ) { |
|||
await super.stop(); |
|||
} |
|||
|
|||
async deleteChatMessage (message) { |
|||
const { attachment: attachmentService } = this.dtp.services; |
|||
const ChatMessage = mongoose.model('ChatMessage'); |
|||
|
|||
await SiteAsync.each(message.attachments, attachmentService.remove.bind(attachmentService), 2); |
|||
await ChatMessage.deleteOne({ _id: message._id }); |
|||
} |
|||
} |
|||
|
|||
(async ( ) => { |
|||
try { |
|||
module.log = new SiteLog(module, module.config.component); |
|||
|
|||
module.worker = new ChatWorker(module); |
|||
await module.worker.start(); |
|||
|
|||
module.log.info(`${module.pkg.name} v${module.pkg.version} ${module.config.component.name} started`); |
|||
} catch (error) { |
|||
module.log.error('failed to start worker', { component: module.config.component, error }); |
|||
process.exit(-1); |
|||
} |
|||
|
|||
})(); |
@ -0,0 +1,63 @@ |
|||
// newsletter/job/email-send.js
|
|||
// Copyright (C) 2022 DTP Technologies, LLC
|
|||
// License: Apache-2.0
|
|||
|
|||
'use strict'; |
|||
|
|||
const path = require('path'); |
|||
|
|||
const { SiteWorkerProcess } = require(path.join(__dirname, '..', '..', '..', '..', 'lib', 'site-lib')); |
|||
|
|||
class NewsletterEmailSendJob extends SiteWorkerProcess { |
|||
|
|||
static get COMPONENT ( ) { |
|||
return { |
|||
name: 'newsletterEmailSendJob', |
|||
slug: 'newsletter-email-send-job', |
|||
}; |
|||
} |
|||
|
|||
constructor (worker) { |
|||
super(worker, NewsletterEmailSendJob.COMPONENT); |
|||
} |
|||
|
|||
async start ( ) { |
|||
await super.start(); |
|||
|
|||
this.queue = await this.getJobQueue('newsletter'); |
|||
|
|||
this.log.info('registering job processor', { queue: this.queue.name, name: 'email-send' }); |
|||
this.queue.process('email-send', this.processEmailSend.bind(this)); |
|||
} |
|||
|
|||
async stop ( ) { |
|||
await super.stop(); |
|||
} |
|||
|
|||
async processEmailSend (job) { |
|||
const { email: emailService } = this.dtp.services; |
|||
const { newsletterId, recipient } = job.data; |
|||
|
|||
try { |
|||
let newsletter = await this.worker.loadNewsletter(newsletterId); |
|||
if (!newsletter) { |
|||
throw new Error('newsletter not found'); |
|||
} |
|||
|
|||
const result = await emailService.send({ |
|||
from: process.env.DTP_EMAIL_SMTP_FROM || `noreply@${this.dtp.config.site.domainKey}`, |
|||
to: recipient, |
|||
subject: newsletter.title, |
|||
html: newsletter.content.html, |
|||
text: newsletter.content.text, |
|||
}); |
|||
|
|||
this.jobLog(job, 'newsletter email sent', { result }); |
|||
} catch (error) { |
|||
this.log.error('failed to send newsletter email', { newsletterId, recipient, error }); |
|||
throw error; // throw error to Bull so it can report in job reports
|
|||
} |
|||
} |
|||
} |
|||
|
|||
module.exports = NewsletterEmailSendJob; |
@ -0,0 +1,100 @@ |
|||
// newsletter/job/transmit.js
|
|||
// Copyright (C) 2022 DTP Technologies, LLC
|
|||
// License: Apache-2.0
|
|||
|
|||
'use strict'; |
|||
|
|||
const path = require('path'); |
|||
|
|||
const mongoose = require('mongoose'); |
|||
|
|||
const User = mongoose.model('User'); |
|||
const NewsletterRecipient = mongoose.model('NewsletterRecipient'); |
|||
|
|||
const { SiteWorkerProcess } = require(path.join(__dirname, '..', '..', '..', '..', 'lib', 'site-lib')); |
|||
|
|||
class NewsletterTransmitJob extends SiteWorkerProcess { |
|||
|
|||
static get COMPONENT ( ) { |
|||
return { |
|||
name: 'newsletterTransmitJob', |
|||
slug: 'newsletter-transmit-job', |
|||
}; |
|||
} |
|||
|
|||
constructor (worker) { |
|||
super(worker, NewsletterTransmitJob.COMPONENT); |
|||
} |
|||
|
|||
async start ( ) { |
|||
await super.start(); |
|||
|
|||
this.queue = await this.getJobQueue('newsletter'); |
|||
|
|||
this.log.info('registering job processor', { queue: this.queue.name, name: 'transmit' }); |
|||
this.queue.process('transmit', this.processTransmit.bind(this)); |
|||
} |
|||
|
|||
async stop ( ) { |
|||
await super.stop(); |
|||
} |
|||
|
|||
async processTransmit (job) { |
|||
const { newsletterId } = job.data; |
|||
this.log.info('newsletter email job received', { id: job.id, newsletterId }); |
|||
|
|||
try { |
|||
/* |
|||
* Transmit first to all local user accounts with verified email who've |
|||
* opted in for receiving marketing email. |
|||
*/ |
|||
await User |
|||
.find({ |
|||
'flags.isEmailVerified': true, |
|||
'optIn.marketing': true, |
|||
}) |
|||
.select('email displayName username username_lc') |
|||
.lean() |
|||
.cursor() |
|||
.eachAsync(async (user) => { |
|||
try { |
|||
const jobData = { |
|||
newsletterId: newsletterId, |
|||
recipient: user.email, |
|||
recipientName: user.displayName || user.username, |
|||
}; |
|||
const jobOptions = { attempts: 3 }; |
|||
await this.queue.add('email-send', jobData, jobOptions); |
|||
} catch (error) { |
|||
this.log.error('failed to create newsletter email job', { error }); |
|||
} |
|||
}, { parallel: 4 }); |
|||
|
|||
/* |
|||
* Transmit to all newsletter recipients on file who've joined through the |
|||
* widget on the site w/o signing up for an account. |
|||
*/ |
|||
await NewsletterRecipient |
|||
.find({ 'flags.isVerified': true, 'flags.isOptIn': true, 'flags.isRejected': false }) |
|||
.lean() |
|||
.cursor() |
|||
.eachAsync(async (recipient) => { |
|||
try { |
|||
const jobData = { |
|||
newsletterId: newsletterId, |
|||
recipient: recipient.address, |
|||
}; |
|||
const jobOptions = { attempts: 3 }; |
|||
await this.queue.add('email-send', jobData, jobOptions); |
|||
} catch (error) { |
|||
this.log.error('failed to create newsletter email job', { error }); |
|||
} |
|||
}, { parallel: 4 }); |
|||
} catch (error) { |
|||
this.log.error('failed to send newsletter', { newsletterId, error }); |
|||
throw error; |
|||
} |
|||
} |
|||
} |
|||
|
|||
module.exports = NewsletterTransmitJob; |
Loading…
Reference in new issue