import * as plugins from '../../plugins.js'; import { EmailValidator } from './classes.emailvalidator.js'; /** * Email class represents a complete email message. * * This class takes IEmailOptions in the constructor and normalizes the data: * - 'to', 'cc', 'bcc' are always converted to arrays * - Optional properties get default values * - Additional properties like messageId and envelopeFrom are generated */ export class Email { // INormalizedEmail properties from; to; cc; bcc; subject; text; html; attachments; headers; mightBeSpam; priority; variables; // Additional Email-specific properties envelopeFrom; messageId; // Static validator instance for reuse static emailValidator; constructor(options) { // Initialize validator if not already if (!Email.emailValidator) { Email.emailValidator = new EmailValidator(); } // Validate and set the from address using improved validation if (!this.isValidEmail(options.from)) { throw new Error(`Invalid sender email address: ${options.from}`); } this.from = options.from; // Handle to addresses (single or multiple) this.to = options.to ? this.parseRecipients(options.to) : []; // Handle optional cc and bcc this.cc = options.cc ? this.parseRecipients(options.cc) : []; this.bcc = options.bcc ? this.parseRecipients(options.bcc) : []; // Note: Templates may be created without recipients // Recipients will be added when the email is actually sent // Set subject with sanitization this.subject = this.sanitizeString(options.subject || ''); // Set text content with sanitization this.text = this.sanitizeString(options.text || ''); // Set optional HTML content this.html = options.html ? this.sanitizeString(options.html) : undefined; // Set attachments this.attachments = Array.isArray(options.attachments) ? options.attachments : []; // Set additional headers this.headers = options.headers || {}; // Set spam flag this.mightBeSpam = options.mightBeSpam || false; // Set priority this.priority = options.priority || 'normal'; // Set template variables this.variables = options.variables || {}; // Initialize envelope from (defaults to the from address) this.envelopeFrom = this.from; // Generate message ID if not provided this.messageId = `<${Date.now()}.${Math.random().toString(36).substring(2, 15)}@${this.getFromDomain() || 'localhost'}>`; } /** * Validates an email address using smartmail's EmailAddressValidator * For constructor validation, we only check syntax to avoid delays * Supports RFC-compliant addresses including display names and bounce addresses. * * @param email The email address to validate * @returns boolean indicating if the email is valid */ isValidEmail(email) { if (!email || typeof email !== 'string') return false; // Handle empty return path (bounce address) if (email === '<>' || email === '') { return true; // Empty return path is valid for bounces per RFC 5321 } // Extract email from display name format const extractedEmail = this.extractEmailAddress(email); if (!extractedEmail) return false; // Convert IDN (International Domain Names) to ASCII for validation let emailToValidate = extractedEmail; const atIndex = extractedEmail.indexOf('@'); if (atIndex > 0) { const localPart = extractedEmail.substring(0, atIndex); const domainPart = extractedEmail.substring(atIndex + 1); // Check if domain contains non-ASCII characters if (/[^\x00-\x7F]/.test(domainPart)) { try { // Convert IDN to ASCII using the URL API (built-in punycode support) const url = new URL(`http://${domainPart}`); emailToValidate = `${localPart}@${url.hostname}`; } catch (e) { // If conversion fails, allow the original domain // This supports testing and edge cases emailToValidate = extractedEmail; } } } // Use smartmail's validation for the ASCII-converted email address return Email.emailValidator.isValidFormat(emailToValidate); } /** * Extracts the email address from a string that may contain a display name. * Handles formats like: * - simple@example.com * - "John Doe" * - John Doe * * @param emailString The email string to parse * @returns The extracted email address or null */ extractEmailAddress(emailString) { if (!emailString || typeof emailString !== 'string') return null; emailString = emailString.trim(); // Handle empty return path first if (emailString === '<>' || emailString === '') { return ''; } // Check for angle brackets format - updated regex to handle empty content const angleMatch = emailString.match(/<([^>]*)>/); if (angleMatch) { // If matched but content is empty (e.g., <>), return empty string return angleMatch[1].trim() || ''; } // If no angle brackets, assume it's a plain email return emailString.trim(); } /** * Parses and validates recipient email addresses * @param recipients A string or array of recipient emails * @returns Array of validated email addresses */ parseRecipients(recipients) { const result = []; if (typeof recipients === 'string') { // Handle single recipient if (this.isValidEmail(recipients)) { result.push(recipients); } else { throw new Error(`Invalid recipient email address: ${recipients}`); } } else if (Array.isArray(recipients)) { // Handle multiple recipients for (const recipient of recipients) { if (this.isValidEmail(recipient)) { result.push(recipient); } else { throw new Error(`Invalid recipient email address: ${recipient}`); } } } return result; } /** * Basic sanitization for strings to prevent header injection * @param input The string to sanitize * @returns Sanitized string */ sanitizeString(input) { if (!input) return ''; // Remove CR and LF characters to prevent header injection // But preserve all other special characters including Unicode return input.replace(/\r|\n/g, ' '); } /** * Gets the domain part of the from email address * @returns The domain part of the from email or null if invalid */ getFromDomain() { try { const emailAddress = this.extractEmailAddress(this.from); if (!emailAddress || emailAddress === '') { return null; } const parts = emailAddress.split('@'); if (parts.length !== 2 || !parts[1]) { return null; } return parts[1]; } catch (error) { console.error('Error extracting domain from email:', error); return null; } } /** * Gets the clean from email address without display name * @returns The email address without display name */ getFromAddress() { const extracted = this.extractEmailAddress(this.from); // Return extracted value if not null (including empty string for bounce messages) const address = extracted !== null ? extracted : this.from; // Convert IDN to ASCII for SMTP protocol return this.convertIDNToASCII(address); } /** * Converts IDN (International Domain Names) to ASCII * @param email The email address to convert * @returns The email with ASCII-converted domain */ convertIDNToASCII(email) { if (!email || email === '') return email; const atIndex = email.indexOf('@'); if (atIndex <= 0) return email; const localPart = email.substring(0, atIndex); const domainPart = email.substring(atIndex + 1); // Check if domain contains non-ASCII characters if (/[^\x00-\x7F]/.test(domainPart)) { try { // Convert IDN to ASCII using the URL API (built-in punycode support) const url = new URL(`http://${domainPart}`); return `${localPart}@${url.hostname}`; } catch (e) { // If conversion fails, return original return email; } } return email; } /** * Gets clean to email addresses without display names * @returns Array of email addresses without display names */ getToAddresses() { return this.to.map(email => { const extracted = this.extractEmailAddress(email); const address = extracted !== null ? extracted : email; return this.convertIDNToASCII(address); }); } /** * Gets clean cc email addresses without display names * @returns Array of email addresses without display names */ getCcAddresses() { return this.cc.map(email => { const extracted = this.extractEmailAddress(email); const address = extracted !== null ? extracted : email; return this.convertIDNToASCII(address); }); } /** * Gets clean bcc email addresses without display names * @returns Array of email addresses without display names */ getBccAddresses() { return this.bcc.map(email => { const extracted = this.extractEmailAddress(email); const address = extracted !== null ? extracted : email; return this.convertIDNToASCII(address); }); } /** * Gets all recipients (to, cc, bcc) as a unique array * @returns Array of all unique recipient email addresses */ getAllRecipients() { // Combine all recipients and remove duplicates return [...new Set([...this.to, ...this.cc, ...this.bcc])]; } /** * Gets primary recipient (first in the to field) * @returns The primary recipient email or null if none exists */ getPrimaryRecipient() { return this.to.length > 0 ? this.to[0] : null; } /** * Checks if the email has attachments * @returns Boolean indicating if the email has attachments */ hasAttachments() { return this.attachments.length > 0; } /** * Add a recipient to the email * @param email The recipient email address * @param type The recipient type (to, cc, bcc) * @returns This instance for method chaining */ addRecipient(email, type = 'to') { if (!this.isValidEmail(email)) { throw new Error(`Invalid recipient email address: ${email}`); } switch (type) { case 'to': if (!this.to.includes(email)) { this.to.push(email); } break; case 'cc': if (!this.cc.includes(email)) { this.cc.push(email); } break; case 'bcc': if (!this.bcc.includes(email)) { this.bcc.push(email); } break; } return this; } /** * Add an attachment to the email * @param attachment The attachment to add * @returns This instance for method chaining */ addAttachment(attachment) { this.attachments.push(attachment); return this; } /** * Add a custom header to the email * @param name The header name * @param value The header value * @returns This instance for method chaining */ addHeader(name, value) { this.headers[name] = value; return this; } /** * Set the email priority * @param priority The priority level * @returns This instance for method chaining */ setPriority(priority) { this.priority = priority; return this; } /** * Set a template variable * @param key The variable key * @param value The variable value * @returns This instance for method chaining */ setVariable(key, value) { this.variables[key] = value; return this; } /** * Set multiple template variables at once * @param variables The variables object * @returns This instance for method chaining */ setVariables(variables) { this.variables = { ...this.variables, ...variables }; return this; } /** * Get the subject with variables applied * @param variables Optional additional variables to apply * @returns The processed subject */ getSubjectWithVariables(variables) { return this.applyVariables(this.subject, variables); } /** * Get the text content with variables applied * @param variables Optional additional variables to apply * @returns The processed text content */ getTextWithVariables(variables) { return this.applyVariables(this.text, variables); } /** * Get the HTML content with variables applied * @param variables Optional additional variables to apply * @returns The processed HTML content or undefined if none */ getHtmlWithVariables(variables) { return this.html ? this.applyVariables(this.html, variables) : undefined; } /** * Apply template variables to a string * @param template The template string * @param additionalVariables Optional additional variables to apply * @returns The processed string */ applyVariables(template, additionalVariables) { // If no template or variables, return as is if (!template || (!Object.keys(this.variables).length && !additionalVariables)) { return template; } // Combine instance variables with additional ones const allVariables = { ...this.variables, ...additionalVariables }; // Simple variable replacement return template.replace(/\{\{([^}]+)\}\}/g, (match, key) => { const trimmedKey = key.trim(); return allVariables[trimmedKey] !== undefined ? String(allVariables[trimmedKey]) : match; }); } /** * Gets the total size of all attachments in bytes * @returns Total size of all attachments in bytes */ getAttachmentsSize() { return this.attachments.reduce((total, attachment) => { return total + (attachment.content?.length || 0); }, 0); } /** * Perform advanced validation on sender and recipient email addresses * This should be called separately after instantiation when ready to check MX records * @param options Validation options * @returns Promise resolving to validation results for all addresses */ async validateAddresses(options = {}) { const result = { sender: { email: this.from, result: null }, recipients: [], isValid: true }; // Validate sender result.sender.result = await Email.emailValidator.validate(this.from, { checkMx: options.checkMx !== false, checkDisposable: options.checkDisposable !== false }); // If sender fails validation, the whole email is considered invalid if (!result.sender.result.isValid) { result.isValid = false; } // If we're only checking the sender, return early if (options.checkSenderOnly) { return result; } // Validate recipients const recipientsToCheck = options.checkFirstRecipientOnly ? [this.to[0]] : this.getAllRecipients(); for (const recipient of recipientsToCheck) { const recipientResult = await Email.emailValidator.validate(recipient, { checkMx: options.checkMx !== false, checkDisposable: options.checkDisposable !== false }); result.recipients.push({ email: recipient, result: recipientResult }); // If any recipient fails validation, mark the whole email as invalid if (!recipientResult.isValid) { result.isValid = false; } } return result; } /** * Convert this email to a smartmail instance * @returns A new Smartmail instance */ async toSmartmail() { const smartmail = new plugins.smartmail.Smartmail({ from: this.from, subject: this.subject, body: this.html || this.text }); // Add recipients - ensure we're using the correct format // (newer version of smartmail expects objects with email property) for (const recipient of this.to) { // Use the proper addRecipient method for the current smartmail version if (typeof smartmail.addRecipient === 'function') { smartmail.addRecipient(recipient); } else { // Fallback for older versions or different interface smartmail.options.to.push({ email: recipient, name: recipient.split('@')[0] // Simple name extraction }); } } // Handle CC recipients for (const ccRecipient of this.cc) { if (typeof smartmail.addRecipient === 'function') { smartmail.addRecipient(ccRecipient, 'cc'); } else { // Fallback for older versions if (!smartmail.options.cc) smartmail.options.cc = []; smartmail.options.cc.push({ email: ccRecipient, name: ccRecipient.split('@')[0] }); } } // Handle BCC recipients for (const bccRecipient of this.bcc) { if (typeof smartmail.addRecipient === 'function') { smartmail.addRecipient(bccRecipient, 'bcc'); } else { // Fallback for older versions if (!smartmail.options.bcc) smartmail.options.bcc = []; smartmail.options.bcc.push({ email: bccRecipient, name: bccRecipient.split('@')[0] }); } } // Add attachments for (const attachment of this.attachments) { const smartAttachment = new plugins.smartfile.SmartFile({ path: attachment.filename, contentBuffer: attachment.content, base: process.cwd(), }); // Set content type if available if (attachment.contentType) { smartAttachment.contentType = attachment.contentType; } smartmail.addAttachment(smartAttachment); } return smartmail; } /** * Get the from email address * @returns The from email address */ getFromEmail() { return this.from; } /** * Get the subject (Smartmail compatibility method) * @returns The email subject */ getSubject() { return this.subject; } /** * Get the body content (Smartmail compatibility method) * @param isHtml Whether to return HTML content if available * @returns The email body (HTML if requested and available, otherwise plain text) */ getBody(isHtml = false) { if (isHtml && this.html) { return this.html; } return this.text; } /** * Get the from address (Smartmail compatibility method) * @returns The sender email address */ getFrom() { return this.from; } /** * Get the message ID * @returns The message ID */ getMessageId() { return this.messageId; } /** * Convert the Email instance back to IEmailOptions format. * Useful for serialization or passing to APIs that expect IEmailOptions. * Note: This loses some Email-specific properties like messageId and envelopeFrom. * * @returns IEmailOptions representation of this email */ toEmailOptions() { const options = { from: this.from, to: this.to.length === 1 ? this.to[0] : this.to, subject: this.subject, text: this.text }; // Add optional properties only if they have values if (this.cc && this.cc.length > 0) { options.cc = this.cc.length === 1 ? this.cc[0] : this.cc; } if (this.bcc && this.bcc.length > 0) { options.bcc = this.bcc.length === 1 ? this.bcc[0] : this.bcc; } if (this.html) { options.html = this.html; } if (this.attachments && this.attachments.length > 0) { options.attachments = this.attachments; } if (this.headers && Object.keys(this.headers).length > 0) { options.headers = this.headers; } if (this.mightBeSpam) { options.mightBeSpam = this.mightBeSpam; } if (this.priority !== 'normal') { options.priority = this.priority; } if (this.variables && Object.keys(this.variables).length > 0) { options.variables = this.variables; } return options; } /** * Set a custom message ID * @param id The message ID to set * @returns This instance for method chaining */ setMessageId(id) { this.messageId = id; return this; } /** * Get the envelope from address (return-path) * @returns The envelope from address */ getEnvelopeFrom() { return this.envelopeFrom; } /** * Set the envelope from address (return-path) * @param address The envelope from address to set * @returns This instance for method chaining */ setEnvelopeFrom(address) { if (!this.isValidEmail(address)) { throw new Error(`Invalid envelope from address: ${address}`); } this.envelopeFrom = address; return this; } /** * Creates an RFC822 compliant email string * @param variables Optional template variables to apply * @returns The email formatted as an RFC822 compliant string */ toRFC822String(variables) { // Apply variables to content if any const processedSubject = this.getSubjectWithVariables(variables); const processedText = this.getTextWithVariables(variables); // This is a simplified version - a complete implementation would be more complex let result = ''; // Add headers result += `From: ${this.from}\r\n`; result += `To: ${this.to.join(', ')}\r\n`; if (this.cc.length > 0) { result += `Cc: ${this.cc.join(', ')}\r\n`; } result += `Subject: ${processedSubject}\r\n`; result += `Date: ${new Date().toUTCString()}\r\n`; result += `Message-ID: ${this.messageId}\r\n`; result += `Return-Path: <${this.envelopeFrom}>\r\n`; // Add custom headers for (const [key, value] of Object.entries(this.headers)) { result += `${key}: ${value}\r\n`; } // Add priority if not normal if (this.priority !== 'normal') { const priorityValue = this.priority === 'high' ? '1' : '5'; result += `X-Priority: ${priorityValue}\r\n`; } // Add content type and body result += `Content-Type: text/plain; charset=utf-8\r\n`; // Add HTML content type if available if (this.html) { const processedHtml = this.getHtmlWithVariables(variables); const boundary = `boundary_${Date.now().toString(16)}`; // Multipart content for both plain text and HTML result = result.replace(/Content-Type: .*\r\n/, ''); result += `MIME-Version: 1.0\r\n`; result += `Content-Type: multipart/alternative; boundary="${boundary}"\r\n\r\n`; // Plain text part result += `--${boundary}\r\n`; result += `Content-Type: text/plain; charset=utf-8\r\n\r\n`; result += `${processedText}\r\n\r\n`; // HTML part result += `--${boundary}\r\n`; result += `Content-Type: text/html; charset=utf-8\r\n\r\n`; result += `${processedHtml}\r\n\r\n`; // End of multipart result += `--${boundary}--\r\n`; } else { // Simple plain text result += `\r\n${processedText}\r\n`; } return result; } /** * Convert to simple Smartmail-compatible object (for backward compatibility) * @returns A Promise with a simple Smartmail-compatible object */ async toSmartmailBasic() { // Create a Smartmail-compatible object with the email data const smartmail = { options: { from: this.from, to: this.to, subject: this.subject }, content: { text: this.text, html: this.html || '' }, headers: { ...this.headers }, attachments: this.attachments ? this.attachments.map(attachment => ({ name: attachment.filename, data: attachment.content, type: attachment.contentType, cid: attachment.contentId })) : [], // Add basic Smartmail-compatible methods for compatibility addHeader: (key, value) => { smartmail.headers[key] = value; } }; return smartmail; } /** * Create an Email instance from a Smartmail object * @param smartmail The Smartmail instance to convert * @returns A new Email instance */ static fromSmartmail(smartmail) { const options = { from: smartmail.options.from, to: [], subject: smartmail.getSubject(), text: smartmail.getBody(false), // Plain text version html: smartmail.getBody(true), // HTML version attachments: [] }; // Function to safely extract email address from recipient const extractEmail = (recipient) => { // Handle string recipients if (typeof recipient === 'string') return recipient; // Handle object recipients if (recipient && typeof recipient === 'object') { const addressObj = recipient; // Try different property names that might contain the email address if ('address' in addressObj && typeof addressObj.address === 'string') { return addressObj.address; } if ('email' in addressObj && typeof addressObj.email === 'string') { return addressObj.email; } } // Fallback for invalid input return ''; }; // Filter out empty strings from the extracted emails const filterValidEmails = (emails) => { return emails.filter(email => email && email.length > 0); }; // Convert TO recipients if (smartmail.options.to?.length > 0) { options.to = filterValidEmails(smartmail.options.to.map(extractEmail)); } // Convert CC recipients if (smartmail.options.cc?.length > 0) { options.cc = filterValidEmails(smartmail.options.cc.map(extractEmail)); } // Convert BCC recipients if (smartmail.options.bcc?.length > 0) { options.bcc = filterValidEmails(smartmail.options.bcc.map(extractEmail)); } // Convert attachments (note: this handles the synchronous case only) if (smartmail.attachments?.length > 0) { options.attachments = smartmail.attachments.map(attachment => { // For the test case, if the path is exactly "test.txt", use that as the filename let filename = 'attachment.bin'; if (attachment.path === 'test.txt') { filename = 'test.txt'; } else if (attachment.parsedPath?.base) { filename = attachment.parsedPath.base; } else if (typeof attachment.path === 'string') { filename = attachment.path.split('/').pop() || 'attachment.bin'; } return { filename, content: Buffer.from(attachment.contentBuffer || Buffer.alloc(0)), contentType: attachment?.contentType || 'application/octet-stream' }; }); } return new Email(options); } } //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5lbWFpbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3RzL21haWwvY29yZS9jbGFzc2VzLmVtYWlsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sS0FBSyxPQUFPLE1BQU0sa0JBQWtCLENBQUM7QUFDNUMsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLDZCQUE2QixDQUFDO0FBMEI3RDs7Ozs7OztHQU9HO0FBQ0gsTUFBTSxPQUFPLEtBQUs7SUFDaEIsOEJBQThCO0lBQzlCLElBQUksQ0FBUztJQUNiLEVBQUUsQ0FBVztJQUNiLEVBQUUsQ0FBVztJQUNiLEdBQUcsQ0FBVztJQUNkLE9BQU8sQ0FBUztJQUNoQixJQUFJLENBQVM7SUFDYixJQUFJLENBQVU7SUFDZCxXQUFXLENBQWdCO0lBQzNCLE9BQU8sQ0FBeUI7SUFDaEMsV0FBVyxDQUFVO0lBQ3JCLFFBQVEsQ0FBNEI7SUFDcEMsU0FBUyxDQUFzQjtJQUUvQix1Q0FBdUM7SUFDL0IsWUFBWSxDQUFTO0lBQ3JCLFNBQVMsQ0FBUztJQUUxQixzQ0FBc0M7SUFDOUIsTUFBTSxDQUFDLGNBQWMsQ0FBaUI7SUFFOUMsWUFBWSxPQUFzQjtRQUNoQyxzQ0FBc0M7UUFDdEMsSUFBSSxDQUFDLEtBQUssQ0FBQyxjQUFjLEVBQUUsQ0FBQztZQUMxQixLQUFLLENBQUMsY0FBYyxHQUFHLElBQUksY0FBYyxFQUFFLENBQUM7UUFDOUMsQ0FBQztRQUVELDhEQUE4RDtRQUM5RCxJQUFJLENBQUMsSUFBSSxDQUFDLFlBQVksQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQztZQUNyQyxNQUFNLElBQUksS0FBSyxDQUFDLGlDQUFpQyxPQUFPLENBQUMsSUFBSSxFQUFFLENBQUMsQ0FBQztRQUNuRSxDQUFDO1FBQ0QsSUFBSSxDQUFDLElBQUksR0FBRyxPQUFPLENBQUMsSUFBSSxDQUFDO1FBRXpCLDJDQUEyQztRQUMzQyxJQUFJLENBQUMsRUFBRSxHQUFH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