Files
bigcapital/packages/server/src/modules/SaleEstimates/commands/SendSaleEstimateMail.ts
2025-12-05 00:07:26 +02:00

232 lines
8.0 KiB
TypeScript

import { InjectQueue } from '@nestjs/bullmq';
import { Queue } from 'bull';
import { Inject, Injectable } from '@nestjs/common';
import { EventEmitter2 } from '@nestjs/event-emitter';
import { ContactMailNotification } from '@/modules/MailNotification/ContactMailNotification';
import {
DEFAULT_ESTIMATE_REMINDER_MAIL_CONTENT,
DEFAULT_ESTIMATE_REMINDER_MAIL_SUBJECT,
} from '../constants';
import { GetSaleEstimate } from '../queries/GetSaleEstimate.service';
import { transformEstimateToMailDataArgs } from '../utils';
import { GetSaleEstimatePdf } from '../queries/GetSaleEstimatePdf';
import { events } from '@/common/events/events';
import { SaleEstimate } from '../models/SaleEstimate';
import { mergeAndValidateMailOptions } from '@/modules/MailNotification/utils';
import {
ISaleEstimateMailPresendEvent,
SaleEstimateMailOptionsDTO,
SendSaleEstimateMailJob,
SendSaleEstimateMailQueue,
} from '../types/SaleEstimates.types';
import { SaleEstimateMailOptions } from '../types/SaleEstimates.types';
import { Mail } from '@/modules/Mail/Mail';
import { MailTransporter } from '@/modules/Mail/MailTransporter.service';
import { TenantModelProxy } from '@/modules/System/models/TenantBaseModel';
import { GetSaleEstimateMailTemplateService } from '../queries/GetSaleEstimateMailTemplate.service';
import { TenancyContext } from '@/modules/Tenancy/TenancyContext.service';
@Injectable()
export class SendSaleEstimateMail {
/**
* @param {GetSaleEstimatePdf} estimatePdf - Estimate pdf service.
* @param {GetSaleEstimate} getSaleEstimateService - Get sale estimate service.
* @param {ContactMailNotification} contactMailNotification - Contact mail notification service.
* @param {EventEmitter2} eventPublisher - Event emitter.
* @param {MailTransporter} mailTransporter - Mail transporter service.
* @param {typeof SaleEstimate} saleEstimateModel - Sale estimate model.
*/
constructor(
private readonly estimatePdf: GetSaleEstimatePdf,
private readonly getSaleEstimateService: GetSaleEstimate,
private readonly contactMailNotification: ContactMailNotification,
private readonly getEstimateMailTemplate: GetSaleEstimateMailTemplateService,
private readonly eventPublisher: EventEmitter2,
private readonly mailTransporter: MailTransporter,
private readonly tenancyContext: TenancyContext,
@Inject(SaleEstimate.name)
private readonly saleEstimateModel: TenantModelProxy<typeof SaleEstimate>,
@InjectQueue(SendSaleEstimateMailQueue)
private readonly sendEstimateMailQueue: Queue,
) { }
/**
* Triggers the reminder mail of the given sale estimate.
* @param {number} saleEstimateId - Sale estimate id.
* @param {SaleEstimateMailOptionsDTO} messageOptions - Sale estimate mail options.
* @returns {Promise<void>}
*/
public async triggerMail(
saleEstimateId: number,
messageOptions: SaleEstimateMailOptionsDTO,
): Promise<void> {
const tenant = await this.tenancyContext.getTenant();
const user = await this.tenancyContext.getSystemUser();
const organizationId = tenant.organizationId;
const userId = user.id;
const payload = {
saleEstimateId,
messageOptions,
userId,
organizationId,
};
await this.sendEstimateMailQueue.add(SendSaleEstimateMailJob, payload);
// Triggers `onSaleEstimatePreMailSend` event.
await this.eventPublisher.emitAsync(events.saleEstimate.onPreMailSend, {
saleEstimateId,
messageOptions,
} as ISaleEstimateMailPresendEvent);
}
/**
* Formate the text of the mail.
* @param {number} estimateId - Estimate id.
* @returns {Promise<Record<string, any>>}
*/
public formatterArgs = async (estimateId: number) => {
const estimate = await this.getSaleEstimateService.getEstimate(estimateId);
const commonArgs = await this.contactMailNotification.getCommonFormatArgs();
return {
...commonArgs,
...transformEstimateToMailDataArgs(estimate),
};
};
/**
* Retrieves the mail options.
* @param {number} saleEstimateId - Sale estimate id.
* @param {string} defaultSubject - Default subject.
* @param {string} defaultMessage - Default message.
* @returns {Promise<SaleEstimateMailOptions>}
*/
public getMailOptions = async (
saleEstimateId: number,
defaultSubject: string = DEFAULT_ESTIMATE_REMINDER_MAIL_SUBJECT,
defaultMessage: string = DEFAULT_ESTIMATE_REMINDER_MAIL_CONTENT,
): Promise<SaleEstimateMailOptions> => {
const saleEstimate = await this.saleEstimateModel()
.query()
.findById(saleEstimateId)
.throwIfNotFound();
const formatArgs = await this.formatterArgs(saleEstimateId);
const mailOptions =
await this.contactMailNotification.getDefaultMailOptions(
saleEstimate.customerId,
);
return {
...mailOptions,
message: defaultMessage,
subject: defaultSubject,
attachEstimate: true,
formatArgs,
};
};
/**
* Formats the given mail options.
* @param {number} saleEstimateId - Sale estimate id.
* @param {SaleEstimateMailOptions} mailOptions - Sale estimate mail options.
* @returns {Promise<SaleEstimateMailOptions>}
*/
public formatMailOptions = async (
saleEstimateId: number,
mailOptions: SaleEstimateMailOptions,
): Promise<SaleEstimateMailOptions> => {
const formatterArgs = await this.formatterArgs(saleEstimateId);
const formattedOptions =
await this.contactMailNotification.formatMailOptions(
mailOptions,
formatterArgs,
);
// Retrieves the estimate mail template.
const message = await this.getEstimateMailTemplate.getMailTemplate(
saleEstimateId,
{
message: formattedOptions.message,
preview: formattedOptions.message,
},
);
return { ...formattedOptions, message };
};
/**
* Retrieves the formatted mail options.
* @param {number} saleEstimateId
* @param {SaleEstimateMailOptionsDTO} messageOptions
* @returns {Promise<SaleEstimateMailOptions>}
*/
public async getFormattedMailOptions(
saleEstimateId: number,
messageOptions: SaleEstimateMailOptionsDTO,
): Promise<SaleEstimateMailOptions> {
const defaultMessageOptions = await this.getMailOptions(saleEstimateId);
const parsedMessageOptions = mergeAndValidateMailOptions(
defaultMessageOptions,
messageOptions,
);
return this.formatMailOptions(saleEstimateId, parsedMessageOptions);
}
/**
* Sends the mail notification of the given sale estimate.
* @param {number} saleEstimateId - Sale estimate id.
* @param {SaleEstimateMailOptions} messageOptions - Sale estimate mail options.
* @returns {Promise<void>}
*/
public async sendMail(
saleEstimateId: number,
messageOptions: SaleEstimateMailOptionsDTO,
): Promise<void> {
const formattedOptions = await this.getFormattedMailOptions(
saleEstimateId,
messageOptions,
);
const mail = new Mail()
.setSubject(formattedOptions.subject)
.setTo(formattedOptions.to)
.setCC(formattedOptions.cc)
.setBCC(formattedOptions.bcc)
.setContent(formattedOptions.message);
// Attaches the estimate pdf to the mail.
if (formattedOptions.attachEstimate) {
// Retrieves the estimate pdf and attaches it to the mail.
const [estimatePdfBuffer, estimateFilename] =
await this.estimatePdf.getSaleEstimatePdf(saleEstimateId);
mail.setAttachments([
{
filename: `${estimateFilename}.pdf`,
content: estimatePdfBuffer,
},
]);
}
const eventPayload = {
saleEstimateId,
messageOptions,
formattedOptions,
};
// Triggers `onSaleEstimateMailSend` event.
await this.eventPublisher.emitAsync(
events.saleEstimate.onMailSend,
eventPayload as ISaleEstimateMailPresendEvent,
);
await this.mailTransporter.send(mail);
// Triggers `onSaleEstimateMailSent` event.
await this.eventPublisher.emitAsync(
events.saleEstimate.onMailSent,
eventPayload as ISaleEstimateMailPresendEvent,
);
}
}