parent
e3142c1271
commit
5fd0b22f3f
@ -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