mirror of
https://github.com/xtr-dev/payload-mailing.git
synced 2025-12-10 16:23:23 +00:00
Compare commits
12 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
9811d63a92 | ||
| 0fa10164bf | |||
| ada13d27cb | |||
|
|
53ab62ed10 | ||
| de57dd4102 | |||
|
|
4633ead274 | ||
| d69f7c1f98 | |||
|
|
57984e8633 | ||
| d15fa454a0 | |||
| f431786907 | |||
|
|
63a5c5f982 | ||
| 107f67e22b |
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@xtr-dev/payload-mailing",
|
"name": "@xtr-dev/payload-mailing",
|
||||||
"version": "0.4.16",
|
"version": "0.4.21",
|
||||||
"description": "Template-based email system with scheduling and job processing for PayloadCMS",
|
"description": "Template-based email system with scheduling and job processing for PayloadCMS",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"main": "dist/index.js",
|
"main": "dist/index.js",
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ const Emails: CollectionConfig = {
|
|||||||
description: 'Email delivery and status tracking',
|
description: 'Email delivery and status tracking',
|
||||||
},
|
},
|
||||||
defaultPopulate: {
|
defaultPopulate: {
|
||||||
template: true,
|
templateSlug: true,
|
||||||
to: true,
|
to: true,
|
||||||
cc: true,
|
cc: true,
|
||||||
bcc: true,
|
bcc: true,
|
||||||
@@ -40,6 +40,14 @@ const Emails: CollectionConfig = {
|
|||||||
description: 'Email template used (optional if custom content provided)',
|
description: 'Email template used (optional if custom content provided)',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
name: 'templateSlug',
|
||||||
|
type: 'text',
|
||||||
|
admin: {
|
||||||
|
description: 'Slug of the email template (auto-populated from template relationship)',
|
||||||
|
readOnly: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
{
|
{
|
||||||
name: 'to',
|
name: 'to',
|
||||||
type: 'text',
|
type: 'text',
|
||||||
@@ -198,7 +206,7 @@ const Emails: CollectionConfig = {
|
|||||||
readOnly: true,
|
readOnly: true,
|
||||||
},
|
},
|
||||||
filterOptions: ({ id }) => {
|
filterOptions: ({ id }) => {
|
||||||
const emailId = resolveID({ id })
|
const emailId = resolveID(id)
|
||||||
return {
|
return {
|
||||||
'input.emailId': {
|
'input.emailId': {
|
||||||
equals: emailId ? String(emailId) : '',
|
equals: emailId ? String(emailId) : '',
|
||||||
@@ -208,6 +216,27 @@ const Emails: CollectionConfig = {
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
hooks: {
|
hooks: {
|
||||||
|
beforeChange: [
|
||||||
|
async ({ data, req }) => {
|
||||||
|
// Auto-populate templateSlug from template relationship
|
||||||
|
if (data.template) {
|
||||||
|
try {
|
||||||
|
const template = await req.payload.findByID({
|
||||||
|
collection: 'email-templates',
|
||||||
|
id: typeof data.template === 'string' ? data.template : data.template.id,
|
||||||
|
})
|
||||||
|
data.templateSlug = template.slug
|
||||||
|
} catch (error) {
|
||||||
|
// If template lookup fails, clear the slug
|
||||||
|
data.templateSlug = undefined
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Clear templateSlug if template is removed
|
||||||
|
data.templateSlug = undefined
|
||||||
|
}
|
||||||
|
return data
|
||||||
|
}
|
||||||
|
],
|
||||||
// Simple approach: Only use afterChange hook for job management
|
// Simple approach: Only use afterChange hook for job management
|
||||||
// This avoids complex interaction between hooks and ensures document ID is always available
|
// This avoids complex interaction between hooks and ensures document ID is always available
|
||||||
afterChange: [
|
afterChange: [
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { Payload } from 'payload'
|
import { Payload } from 'payload'
|
||||||
import { getMailing, renderTemplate, parseAndValidateEmails, sanitizeFromName } from './utils/helpers.js'
|
import { getMailing, renderTemplateWithId, parseAndValidateEmails, sanitizeFromName } from './utils/helpers.js'
|
||||||
import { BaseEmailDocument } from './types/index.js'
|
import { BaseEmailDocument } from './types/index.js'
|
||||||
import { processJobById } from './utils/emailProcessor.js'
|
import { processJobById } from './utils/emailProcessor.js'
|
||||||
import { createContextLogger } from './utils/logger.js'
|
import { createContextLogger } from './utils/logger.js'
|
||||||
@@ -50,7 +50,8 @@ export const sendEmail = async <TEmail extends BaseEmailDocument = BaseEmailDocu
|
|||||||
let emailData: Partial<TEmail> = { ...options.data } as Partial<TEmail>
|
let emailData: Partial<TEmail> = { ...options.data } as Partial<TEmail>
|
||||||
|
|
||||||
if (options.template) {
|
if (options.template) {
|
||||||
const { html, text, subject } = await renderTemplate(
|
// Look up and render the template in a single operation to avoid duplicate lookups
|
||||||
|
const { html, text, subject, templateId } = await renderTemplateWithId(
|
||||||
payload,
|
payload,
|
||||||
options.template.slug,
|
options.template.slug,
|
||||||
options.template.variables || {}
|
options.template.variables || {}
|
||||||
@@ -58,6 +59,7 @@ export const sendEmail = async <TEmail extends BaseEmailDocument = BaseEmailDocu
|
|||||||
|
|
||||||
emailData = {
|
emailData = {
|
||||||
...emailData,
|
...emailData,
|
||||||
|
template: templateId,
|
||||||
subject,
|
subject,
|
||||||
html,
|
html,
|
||||||
text,
|
text,
|
||||||
|
|||||||
@@ -1,10 +1,10 @@
|
|||||||
import { Payload } from 'payload'
|
import {CollectionSlug, EmailAdapter, Payload, SendEmailOptions} from 'payload'
|
||||||
import { Liquid } from 'liquidjs'
|
import { Liquid } from 'liquidjs'
|
||||||
import {
|
import {
|
||||||
MailingPluginConfig,
|
MailingPluginConfig,
|
||||||
TemplateVariables,
|
TemplateVariables,
|
||||||
MailingService as IMailingService,
|
MailingService as IMailingService,
|
||||||
BaseEmail, BaseEmailTemplate, BaseEmailDocument, BaseEmailTemplateDocument
|
BaseEmailDocument, BaseEmailTemplateDocument
|
||||||
} from '../types/index.js'
|
} from '../types/index.js'
|
||||||
import { serializeRichTextToHTML, serializeRichTextToText } from '../utils/richTextSerializer.js'
|
import { serializeRichTextToHTML, serializeRichTextToText } from '../utils/richTextSerializer.js'
|
||||||
import { sanitizeDisplayName } from '../utils/helpers.js'
|
import { sanitizeDisplayName } from '../utils/helpers.js'
|
||||||
@@ -12,7 +12,6 @@ import { sanitizeDisplayName } from '../utils/helpers.js'
|
|||||||
export class MailingService implements IMailingService {
|
export class MailingService implements IMailingService {
|
||||||
public payload: Payload
|
public payload: Payload
|
||||||
private config: MailingPluginConfig
|
private config: MailingPluginConfig
|
||||||
private emailAdapter: any
|
|
||||||
private templatesCollection: string
|
private templatesCollection: string
|
||||||
private emailsCollection: string
|
private emailsCollection: string
|
||||||
private liquid: Liquid | null | false = null
|
private liquid: Liquid | null | false = null
|
||||||
@@ -31,14 +30,13 @@ export class MailingService implements IMailingService {
|
|||||||
if (!this.payload.email) {
|
if (!this.payload.email) {
|
||||||
throw new Error('Payload email configuration is required. Please configure email in your Payload config.')
|
throw new Error('Payload email configuration is required. Please configure email in your Payload config.')
|
||||||
}
|
}
|
||||||
this.emailAdapter = this.payload.email
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private ensureInitialized(): void {
|
private ensureInitialized(): void {
|
||||||
if (!this.payload || !this.payload.db) {
|
if (!this.payload || !this.payload.db) {
|
||||||
throw new Error('MailingService payload not properly initialized')
|
throw new Error('MailingService payload not properly initialized')
|
||||||
}
|
}
|
||||||
if (!this.emailAdapter) {
|
if (!this.payload.email) {
|
||||||
throw new Error('Email adapter not configured. Please ensure Payload has email configured.')
|
throw new Error('Email adapter not configured. Please ensure Payload has email configured.')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -127,6 +125,17 @@ export class MailingService implements IMailingService {
|
|||||||
throw new Error(`Email template not found: ${templateSlug}`)
|
throw new Error(`Email template not found: ${templateSlug}`)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return this.renderTemplateDocument(template, variables)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Render a template document (for when you already have the template loaded)
|
||||||
|
* This avoids duplicate template lookups
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
async renderTemplateDocument(template: BaseEmailTemplateDocument, variables: TemplateVariables): Promise<{ html: string; text: string; subject: string }> {
|
||||||
|
this.ensureInitialized()
|
||||||
|
|
||||||
const emailContent = await this.renderEmailTemplate(template, variables)
|
const emailContent = await this.renderEmailTemplate(template, variables)
|
||||||
const subject = await this.renderTemplateString(template.subject || '', variables)
|
const subject = await this.renderTemplateString(template.subject || '', variables)
|
||||||
|
|
||||||
@@ -142,7 +151,7 @@ export class MailingService implements IMailingService {
|
|||||||
const currentTime = new Date().toISOString()
|
const currentTime = new Date().toISOString()
|
||||||
|
|
||||||
const { docs: pendingEmails } = await this.payload.find({
|
const { docs: pendingEmails } = await this.payload.find({
|
||||||
collection: this.emailsCollection as any,
|
collection: this.emailsCollection as CollectionSlug,
|
||||||
where: {
|
where: {
|
||||||
and: [
|
and: [
|
||||||
{
|
{
|
||||||
@@ -182,7 +191,7 @@ export class MailingService implements IMailingService {
|
|||||||
const retryTime = new Date(Date.now() - retryDelay).toISOString()
|
const retryTime = new Date(Date.now() - retryDelay).toISOString()
|
||||||
|
|
||||||
const { docs: failedEmails } = await this.payload.find({
|
const { docs: failedEmails } = await this.payload.find({
|
||||||
collection: this.emailsCollection as any,
|
collection: this.emailsCollection as CollectionSlug,
|
||||||
where: {
|
where: {
|
||||||
and: [
|
and: [
|
||||||
{
|
{
|
||||||
@@ -222,7 +231,7 @@ export class MailingService implements IMailingService {
|
|||||||
async processEmailItem(emailId: string): Promise<void> {
|
async processEmailItem(emailId: string): Promise<void> {
|
||||||
try {
|
try {
|
||||||
await this.payload.update({
|
await this.payload.update({
|
||||||
collection: this.emailsCollection as any,
|
collection: this.emailsCollection as CollectionSlug,
|
||||||
id: emailId,
|
id: emailId,
|
||||||
data: {
|
data: {
|
||||||
status: 'processing',
|
status: 'processing',
|
||||||
@@ -231,7 +240,7 @@ export class MailingService implements IMailingService {
|
|||||||
})
|
})
|
||||||
|
|
||||||
const email = await this.payload.findByID({
|
const email = await this.payload.findByID({
|
||||||
collection: this.emailsCollection as any,
|
collection: this.emailsCollection as CollectionSlug,
|
||||||
id: emailId,
|
id: emailId,
|
||||||
depth: 1,
|
depth: 1,
|
||||||
}) as BaseEmailDocument
|
}) as BaseEmailDocument
|
||||||
@@ -244,7 +253,7 @@ export class MailingService implements IMailingService {
|
|||||||
fromField = this.getDefaultFrom()
|
fromField = this.getDefaultFrom()
|
||||||
}
|
}
|
||||||
|
|
||||||
let mailOptions: any = {
|
let mailOptions: SendEmailOptions = {
|
||||||
from: fromField,
|
from: fromField,
|
||||||
to: email.to,
|
to: email.to,
|
||||||
cc: email.cc || undefined,
|
cc: email.cc || undefined,
|
||||||
@@ -255,6 +264,19 @@ export class MailingService implements IMailingService {
|
|||||||
text: email.text || undefined,
|
text: email.text || undefined,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!mailOptions.from) {
|
||||||
|
throw new Error('Email from field is required')
|
||||||
|
}
|
||||||
|
if (!mailOptions.to || (Array.isArray(mailOptions.to) && mailOptions.to.length === 0)) {
|
||||||
|
throw new Error('Email to field is required')
|
||||||
|
}
|
||||||
|
if (!mailOptions.subject) {
|
||||||
|
throw new Error('Email subject is required')
|
||||||
|
}
|
||||||
|
if (!mailOptions.html && !mailOptions.text) {
|
||||||
|
throw new Error('Email content is required')
|
||||||
|
}
|
||||||
|
|
||||||
// Call beforeSend hook if configured
|
// Call beforeSend hook if configured
|
||||||
if (this.config.beforeSend) {
|
if (this.config.beforeSend) {
|
||||||
try {
|
try {
|
||||||
@@ -280,10 +302,10 @@ export class MailingService implements IMailingService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Send email using Payload's email adapter
|
// Send email using Payload's email adapter
|
||||||
await this.emailAdapter.sendEmail(mailOptions)
|
await this.payload.email.sendEmail(mailOptions)
|
||||||
|
|
||||||
await this.payload.update({
|
await this.payload.update({
|
||||||
collection: this.emailsCollection as any,
|
collection: this.emailsCollection as CollectionSlug,
|
||||||
id: emailId,
|
id: emailId,
|
||||||
data: {
|
data: {
|
||||||
status: 'sent',
|
status: 'sent',
|
||||||
@@ -297,7 +319,7 @@ export class MailingService implements IMailingService {
|
|||||||
const maxAttempts = this.config.retryAttempts || 3
|
const maxAttempts = this.config.retryAttempts || 3
|
||||||
|
|
||||||
await this.payload.update({
|
await this.payload.update({
|
||||||
collection: this.emailsCollection as any,
|
collection: this.emailsCollection as CollectionSlug,
|
||||||
id: emailId,
|
id: emailId,
|
||||||
data: {
|
data: {
|
||||||
status: attempts >= maxAttempts ? 'failed' : 'pending',
|
status: attempts >= maxAttempts ? 'failed' : 'pending',
|
||||||
@@ -314,14 +336,14 @@ export class MailingService implements IMailingService {
|
|||||||
|
|
||||||
private async incrementAttempts(emailId: string): Promise<number> {
|
private async incrementAttempts(emailId: string): Promise<number> {
|
||||||
const email = await this.payload.findByID({
|
const email = await this.payload.findByID({
|
||||||
collection: this.emailsCollection as any,
|
collection: this.emailsCollection as CollectionSlug,
|
||||||
id: emailId,
|
id: emailId,
|
||||||
})
|
})
|
||||||
|
|
||||||
const newAttempts = ((email as any).attempts || 0) + 1
|
const newAttempts = ((email as any).attempts || 0) + 1
|
||||||
|
|
||||||
await this.payload.update({
|
await this.payload.update({
|
||||||
collection: this.emailsCollection as any,
|
collection: this.emailsCollection as CollectionSlug,
|
||||||
id: emailId,
|
id: emailId,
|
||||||
data: {
|
data: {
|
||||||
attempts: newAttempts,
|
attempts: newAttempts,
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { Payload } from 'payload'
|
import {Payload, SendEmailOptions} from 'payload'
|
||||||
import type { CollectionConfig, RichTextField } from 'payload'
|
import type { CollectionConfig, RichTextField } from 'payload'
|
||||||
|
|
||||||
// Payload ID type (string or number)
|
// Payload ID type (string or number)
|
||||||
@@ -14,6 +14,7 @@ export type JSONValue = string | number | boolean | { [k: string]: unknown } | u
|
|||||||
export interface BaseEmailDocument {
|
export interface BaseEmailDocument {
|
||||||
id: string | number
|
id: string | number
|
||||||
template?: any
|
template?: any
|
||||||
|
templateSlug?: string | null
|
||||||
to: string[]
|
to: string[]
|
||||||
cc?: string[] | null
|
cc?: string[] | null
|
||||||
bcc?: string[] | null
|
bcc?: string[] | null
|
||||||
@@ -45,28 +46,11 @@ export interface BaseEmailTemplateDocument {
|
|||||||
updatedAt?: string | Date | null
|
updatedAt?: string | Date | null
|
||||||
}
|
}
|
||||||
|
|
||||||
export type BaseEmail<TEmail extends BaseEmailDocument = BaseEmailDocument, TEmailTemplate extends BaseEmailTemplateDocument = BaseEmailTemplateDocument> = Omit<TEmail, 'id' | 'template'> & {template: Omit<TEmailTemplate, 'id'> | TEmailTemplate['id'] | undefined | null}
|
|
||||||
|
|
||||||
export type BaseEmailTemplate<TEmailTemplate extends BaseEmailTemplateDocument = BaseEmailTemplateDocument> = Omit<TEmailTemplate, 'id'>
|
|
||||||
|
|
||||||
export type TemplateRendererHook = (template: string, variables: Record<string, any>) => string | Promise<string>
|
export type TemplateRendererHook = (template: string, variables: Record<string, any>) => string | Promise<string>
|
||||||
|
|
||||||
export type TemplateEngine = 'liquidjs' | 'mustache' | 'simple'
|
export type TemplateEngine = 'liquidjs' | 'mustache' | 'simple'
|
||||||
|
|
||||||
export interface BeforeSendMailOptions {
|
export type BeforeSendHook = (options: SendEmailOptions, email: BaseEmailDocument) => SendEmailOptions | Promise<SendEmailOptions>
|
||||||
from: string
|
|
||||||
to: string[]
|
|
||||||
cc?: string[]
|
|
||||||
bcc?: string[]
|
|
||||||
replyTo?: string
|
|
||||||
subject: string
|
|
||||||
html: string
|
|
||||||
text?: string
|
|
||||||
attachments?: any[]
|
|
||||||
[key: string]: any
|
|
||||||
}
|
|
||||||
|
|
||||||
export type BeforeSendHook = (options: BeforeSendMailOptions, email: BaseEmailDocument) => BeforeSendMailOptions | Promise<BeforeSendMailOptions>
|
|
||||||
|
|
||||||
export interface JobPollingConfig {
|
export interface JobPollingConfig {
|
||||||
maxAttempts?: number // Maximum number of polling attempts (default: 5)
|
maxAttempts?: number // Maximum number of polling attempts (default: 5)
|
||||||
|
|||||||
@@ -130,6 +130,53 @@ export const renderTemplate = async (payload: Payload, templateSlug: string, var
|
|||||||
return mailing.service.renderTemplate(templateSlug, variables)
|
return mailing.service.renderTemplate(templateSlug, variables)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Render a template and return both rendered content and template ID
|
||||||
|
* This is used by sendEmail to avoid duplicate template lookups
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
export const renderTemplateWithId = async (
|
||||||
|
payload: Payload,
|
||||||
|
templateSlug: string,
|
||||||
|
variables: TemplateVariables
|
||||||
|
): Promise<{ html: string; text: string; subject: string; templateId: PayloadID }> => {
|
||||||
|
const mailing = getMailing(payload)
|
||||||
|
const templatesCollection = mailing.config.collections?.templates || 'email-templates'
|
||||||
|
|
||||||
|
// Runtime validation: Ensure the collection exists in Payload
|
||||||
|
if (!payload.collections[templatesCollection]) {
|
||||||
|
throw new Error(
|
||||||
|
`Templates collection '${templatesCollection}' not found. ` +
|
||||||
|
`Available collections: ${Object.keys(payload.collections).join(', ')}`
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Look up the template document once
|
||||||
|
const { docs: templateDocs } = await payload.find({
|
||||||
|
collection: templatesCollection as any,
|
||||||
|
where: {
|
||||||
|
slug: {
|
||||||
|
equals: templateSlug,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
limit: 1,
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!templateDocs || templateDocs.length === 0) {
|
||||||
|
throw new Error(`Template not found: ${templateSlug}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
const templateDoc = templateDocs[0]
|
||||||
|
|
||||||
|
// Render using the document directly to avoid duplicate lookup
|
||||||
|
const rendered = await mailing.service.renderTemplateDocument(templateDoc, variables)
|
||||||
|
|
||||||
|
return {
|
||||||
|
...rendered,
|
||||||
|
templateId: templateDoc.id,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export const processEmails = async (payload: Payload): Promise<void> => {
|
export const processEmails = async (payload: Payload): Promise<void> => {
|
||||||
const mailing = getMailing(payload)
|
const mailing = getMailing(payload)
|
||||||
return mailing.service.processEmails()
|
return mailing.service.processEmails()
|
||||||
|
|||||||
@@ -129,7 +129,11 @@ export async function updateEmailJobRelationship(
|
|||||||
id: normalizedEmailId,
|
id: normalizedEmailId,
|
||||||
})
|
})
|
||||||
|
|
||||||
const currentJobs = (currentEmail.jobs || []).map((job: any) => String(job))
|
// Extract IDs from job objects or use the value directly if it's already an ID
|
||||||
|
// Jobs can be populated (objects with id field) or just IDs (strings/numbers)
|
||||||
|
const currentJobs = (currentEmail.jobs || []).map((job: any) =>
|
||||||
|
typeof job === 'object' && job !== null && job.id ? String(job.id) : String(job)
|
||||||
|
)
|
||||||
const allJobs = [...new Set([...currentJobs, ...normalizedJobIds])] // Deduplicate with normalized strings
|
const allJobs = [...new Set([...currentJobs, ...normalizedJobIds])] // Deduplicate with normalized strings
|
||||||
|
|
||||||
await payload.update({
|
await payload.update({
|
||||||
|
|||||||
Reference in New Issue
Block a user