diff --git a/.gitignore b/.gitignore index 54c1397..dbb120d 100644 --- a/.gitignore +++ b/.gitignore @@ -1,9 +1,24 @@ -node_modules/ .nogit/ + +# artifacts +coverage/ +public/ + +# installs +node_modules/ + +# caches +.yarn/ +.cache/ +.rpt2_cache + +# builds dist/ -dist_rust/ -rust/target/ -deno.lock -*.log -.env -.DS_Store +dist_*/ + +# AI +.claude/ +.serena/ + +#------# custom +rust/target \ No newline at end of file diff --git a/changelog.md b/changelog.md index af61137..0e975e2 100644 --- a/changelog.md +++ b/changelog.md @@ -1,5 +1,12 @@ # Changelog +## 2026-02-10 - 2.3.1 - fix(npmextra) +update .gitignore and npmextra.json to add ignore patterns, registries, and module metadata + +- .gitignore: expanded ignore list for artifacts, installs, caches, builds, AI dirs, and rust target path +- npmextra.json: added npmjs registry alongside internal Verdaccio registry for @git.zone/cli release settings +- npmextra.json: added projectType and module metadata (githost, gitscope, gitrepo, description, npmPackagename, license) for @git.zone/cli and added empty @ship.zone/szci entry + ## 2026-02-10 - 2.3.0 - feat(mailer-smtp) add in-process security pipeline for SMTP delivery (DKIM/SPF/DMARC, content scanning, IP reputation) diff --git a/dist_ts/00_commitinfo_data.d.ts b/dist_ts/00_commitinfo_data.d.ts deleted file mode 100644 index 931e8fc..0000000 --- a/dist_ts/00_commitinfo_data.d.ts +++ /dev/null @@ -1,8 +0,0 @@ -/** - * autocreated commitinfo by @push.rocks/commitinfo - */ -export declare const commitinfo: { - name: string; - version: string; - description: string; -}; diff --git a/dist_ts/00_commitinfo_data.js b/dist_ts/00_commitinfo_data.js deleted file mode 100644 index 5c9a05c..0000000 --- a/dist_ts/00_commitinfo_data.js +++ /dev/null @@ -1,9 +0,0 @@ -/** - * autocreated commitinfo by @push.rocks/commitinfo - */ -export const commitinfo = { - name: '@push.rocks/smartmta', - version: '2.2.1', - description: 'A high-performance, enterprise-grade Mail Transfer Agent (MTA) built from scratch in TypeScript with Rust acceleration.' -}; -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiMDBfY29tbWl0aW5mb19kYXRhLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vdHMvMDBfY29tbWl0aW5mb19kYXRhLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sVUFBVSxHQUFHO0lBQ3hCLElBQUksRUFBRSxzQkFBc0I7SUFDNUIsT0FBTyxFQUFFLE9BQU87SUFDaEIsV0FBVyxFQUFFLHlIQUF5SDtDQUN2SSxDQUFBIn0= \ No newline at end of file diff --git a/dist_ts/errors/index.d.ts b/dist_ts/errors/index.d.ts deleted file mode 100644 index b862d2c..0000000 --- a/dist_ts/errors/index.d.ts +++ /dev/null @@ -1,41 +0,0 @@ -/** - * MTA error classes for SMTP client operations - */ -export declare class MtaConnectionError extends Error { - code: string; - details?: any; - constructor(message: string, detailsOrCode?: any); - static timeout(host: string, port: number, timeoutMs?: number): MtaConnectionError; - static refused(host: string, port: number): MtaConnectionError; - static dnsError(host: string, err?: any): MtaConnectionError; -} -export declare class MtaAuthenticationError extends Error { - code: string; - details?: any; - constructor(message: string, detailsOrCode?: any); - static invalidCredentials(host?: string, user?: string): MtaAuthenticationError; -} -export declare class MtaDeliveryError extends Error { - code: string; - responseCode?: number; - details?: any; - constructor(message: string, detailsOrCode?: any, responseCode?: number); - static temporary(message: string, ...args: any[]): MtaDeliveryError; - static permanent(message: string, ...args: any[]): MtaDeliveryError; -} -export declare class MtaConfigurationError extends Error { - code: string; - details?: any; - constructor(message: string, detailsOrCode?: any); -} -export declare class MtaTimeoutError extends Error { - code: string; - details?: any; - constructor(message: string, detailsOrCode?: any); - static commandTimeout(command: string, hostOrTimeout?: any, timeoutMs?: number): MtaTimeoutError; -} -export declare class MtaProtocolError extends Error { - code: string; - details?: any; - constructor(message: string, detailsOrCode?: any); -} diff --git a/dist_ts/errors/index.js b/dist_ts/errors/index.js deleted file mode 100644 index 4594555..0000000 --- a/dist_ts/errors/index.js +++ /dev/null @@ -1,120 +0,0 @@ -/** - * MTA error classes for SMTP client operations - */ -export class MtaConnectionError extends Error { - code; - details; - constructor(message, detailsOrCode) { - super(message); - this.name = 'MtaConnectionError'; - if (typeof detailsOrCode === 'string') { - this.code = detailsOrCode; - } - else { - this.code = 'CONNECTION_ERROR'; - this.details = detailsOrCode; - } - } - static timeout(host, port, timeoutMs) { - return new MtaConnectionError(`Connection to ${host}:${port} timed out${timeoutMs ? ` after ${timeoutMs}ms` : ''}`, 'TIMEOUT'); - } - static refused(host, port) { - return new MtaConnectionError(`Connection to ${host}:${port} refused`, 'REFUSED'); - } - static dnsError(host, err) { - const errMsg = typeof err === 'string' ? err : err?.message || ''; - return new MtaConnectionError(`DNS resolution failed for ${host}${errMsg ? `: ${errMsg}` : ''}`, 'DNS_ERROR'); - } -} -export class MtaAuthenticationError extends Error { - code; - details; - constructor(message, detailsOrCode) { - super(message); - this.name = 'MtaAuthenticationError'; - if (typeof detailsOrCode === 'string') { - this.code = detailsOrCode; - } - else { - this.code = 'AUTH_ERROR'; - this.details = detailsOrCode; - } - } - static invalidCredentials(host, user) { - const detail = host && user ? `${user}@${host}` : host || user || ''; - return new MtaAuthenticationError(`Authentication failed${detail ? `: ${detail}` : ''}`, 'INVALID_CREDENTIALS'); - } -} -export class MtaDeliveryError extends Error { - code; - responseCode; - details; - constructor(message, detailsOrCode, responseCode) { - super(message); - this.name = 'MtaDeliveryError'; - if (typeof detailsOrCode === 'string') { - this.code = detailsOrCode; - this.responseCode = responseCode; - } - else { - this.code = 'DELIVERY_ERROR'; - this.details = detailsOrCode; - } - } - static temporary(message, ...args) { - return new MtaDeliveryError(message, 'TEMPORARY'); - } - static permanent(message, ...args) { - return new MtaDeliveryError(message, 'PERMANENT'); - } -} -export class MtaConfigurationError extends Error { - code; - details; - constructor(message, detailsOrCode) { - super(message); - this.name = 'MtaConfigurationError'; - if (typeof detailsOrCode === 'string') { - this.code = detailsOrCode; - } - else { - this.code = 'CONFIG_ERROR'; - this.details = detailsOrCode; - } - } -} -export class MtaTimeoutError extends Error { - code; - details; - constructor(message, detailsOrCode) { - super(message); - this.name = 'MtaTimeoutError'; - if (typeof detailsOrCode === 'string') { - this.code = detailsOrCode; - } - else { - this.code = 'TIMEOUT'; - this.details = detailsOrCode; - } - } - static commandTimeout(command, hostOrTimeout, timeoutMs) { - const timeout = typeof hostOrTimeout === 'number' ? hostOrTimeout : timeoutMs; - return new MtaTimeoutError(`Command '${command}' timed out${timeout ? ` after ${timeout}ms` : ''}`, 'COMMAND_TIMEOUT'); - } -} -export class MtaProtocolError extends Error { - code; - details; - constructor(message, detailsOrCode) { - super(message); - this.name = 'MtaProtocolError'; - if (typeof detailsOrCode === 'string') { - this.code = detailsOrCode; - } - else { - this.code = 'PROTOCOL_ERROR'; - this.details = detailsOrCode; - } - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi90cy9lcnJvcnMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxNQUFNLE9BQU8sa0JBQW1CLFNBQVEsS0FBSztJQUNwQyxJQUFJLENBQVM7SUFDYixPQUFPLENBQU87SUFDckIsWUFBWSxPQUFlLEVBQUUsYUFBbUI7UUFDOUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBQ2YsSUFBSSxDQUFDLElBQUksR0FBRyxvQkFBb0IsQ0FBQztRQUNqQyxJQUFJLE9BQU8sYUFBYSxLQUFLLFFBQVEsRUFBRSxDQUFDO1lBQ3RDLElBQUksQ0FBQyxJQUFJLEdBQUcsYUFBYSxDQUFDO1FBQzVCLENBQUM7YUFBTSxDQUFDO1lBQ04sSUFBSSxDQUFDLElBQUksR0FBRyxrQkFBa0IsQ0FBQztZQUMvQixJQUFJLENBQUMsT0FBTyxHQUFHLGFBQWEsQ0FBQztRQUMvQixDQUFDO0lBQ0gsQ0FBQztJQUNELE1BQU0sQ0FBQyxPQUFPLENBQUMsSUFBWSxFQUFFLElBQVksRUFBRSxTQUFrQjtRQUMzRCxPQUFPLElBQUksa0JBQWtCLENBQUMsaUJBQWlCLElBQUksSUFBSSxJQUFJLGFBQWEsU0FBUyxDQUFDLENBQUMsQ0FBQyxVQUFVLFNBQVMsSUFBSSxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUUsRUFBRSxTQUFTLENBQUMsQ0FBQztJQUNqSSxDQUFDO0lBQ0QsTUFBTSxDQUFDLE9BQU8sQ0FBQyxJQUFZLEVBQUUsSUFBWTtRQUN2QyxPQUFPLElBQUksa0JBQWtCLENBQUMsaUJBQWlCLElBQUksSUFBSSxJQUFJLFVBQVUsRUFBRSxTQUFTLENBQUMsQ0FBQztJQUNwRixDQUFDO0lBQ0QsTUFBTSxDQUFDLFFBQVEsQ0FBQyxJQUFZLEVBQUUsR0FBUztRQUNyQyxNQUFNLE1BQU0sR0FBRyxPQUFPLEdBQUcsS0FBSyxRQUFRLENBQUMsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxD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 \ No newline at end of file diff --git a/dist_ts/logger.d.ts b/dist_ts/logger.d.ts deleted file mode 100644 index 2b5554e..0000000 --- a/dist_ts/logger.d.ts +++ /dev/null @@ -1,17 +0,0 @@ -declare class StandardLogger { - private defaultContext; - private correlationId; - constructor(); - log(level: 'error' | 'warn' | 'info' | 'success' | 'debug', message: string, context?: Record): void; - error(message: string, context?: Record): void; - warn(message: string, context?: Record): void; - info(message: string, context?: Record): void; - success(message: string, context?: Record): void; - debug(message: string, context?: Record): void; - setContext(context: Record, overwrite?: boolean): void; - setCorrelationId(id?: string | null): string; - getCorrelationId(): string | null; - clearCorrelationId(): void; -} -export declare const logger: StandardLogger; -export {}; diff --git a/dist_ts/logger.js b/dist_ts/logger.js deleted file mode 100644 index 680e9d1..0000000 --- a/dist_ts/logger.js +++ /dev/null @@ -1,76 +0,0 @@ -import * as plugins from './plugins.js'; -import { randomUUID } from 'node:crypto'; -// Map NODE_ENV to valid TEnvironment -const nodeEnv = process.env.NODE_ENV || 'production'; -const envMap = { - 'development': 'local', - 'test': 'test', - 'staging': 'staging', - 'production': 'production' -}; -// Default Smartlog instance -const baseLogger = new plugins.smartlog.Smartlog({ - logContext: { - environment: envMap[nodeEnv] || 'production', - runtime: 'node', - zone: 'serve.zone', - } -}); -// Extended logger compatible with the original enhanced logger API -class StandardLogger { - defaultContext = {}; - correlationId = null; - constructor() { } - // Log methods - log(level, message, context = {}) { - const combinedContext = { - ...this.defaultContext, - ...context - }; - if (this.correlationId) { - combinedContext.correlation_id = this.correlationId; - } - baseLogger.log(level, message, combinedContext); - } - error(message, context = {}) { - this.log('error', message, context); - } - warn(message, context = {}) { - this.log('warn', message, context); - } - info(message, context = {}) { - this.log('info', message, context); - } - success(message, context = {}) { - this.log('success', message, context); - } - debug(message, context = {}) { - this.log('debug', message, context); - } - // Context management - setContext(context, overwrite = false) { - if (overwrite) { - this.defaultContext = context; - } - else { - this.defaultContext = { - ...this.defaultContext, - ...context - }; - } - } - // Correlation ID management - setCorrelationId(id = null) { - this.correlationId = id || randomUUID(); - return this.correlationId; - } - getCorrelationId() { - return this.correlationId; - } - clearCorrelationId() { - this.correlationId = null; - } -} -// Export a singleton instance -export const logger = new StandardLogger(); -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9nZ2VyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vdHMvbG9nZ2VyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sS0FBSyxPQUFPLE1BQU0sY0FBYyxDQUFDO0FBQ3hDLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxhQUFhLENBQUM7QUFFekMscUNBQXFDO0FBQ3JDLE1BQU0sT0FBTyxHQUFHLE9BQU8sQ0FBQyxHQUFHLENBQUMsUUFBUSxJQUFJLFlBQVksQ0FBQztBQUNyRCxNQUFNLE1BQU0sR0FBZ0U7SUFDMUUsYUFBYSxFQUFFLE9BQU87SUFDdEIsTUFBTSxFQUFFLE1BQU07SUFDZCxTQUFTLEVBQUUsU0FBUztJQUNwQixZQUFZLEVBQUUsWUFBWTtDQUMzQixDQUFDO0FBRUYsNEJBQTRCO0FBQzVCLE1BQU0sVUFBVSxHQUFHLElBQUksT0FBTyxDQUFDLFFBQVEsQ0FBQyxRQUFRLENBQUM7SUFDL0MsVUFBVSxFQUFFO1FBQ1YsV0FBVyxFQUFFLE1BQU0sQ0FBQyxPQUFPLENBQUMsSUFBSSxZQUFZO1FBQzVDLE9BQU8sRUFBRSxNQUFNO1FBQ2YsSUFBSSxFQUFFLFlBQVk7S0FDbkI7Q0FDRixDQUFDLENBQUM7QUFFSCxtRUFBbUU7QUFDbkUsTUFBTSxjQUFjO0lBQ1YsY0FBYyxHQUF3QixFQUFFLENBQUM7SUFDekMsYUFBYSxHQUFrQixJQUFJLENBQUM7SUFFNUMsZ0JBQWUsQ0FBQztJQUVoQixjQUFjO0lBQ1AsR0FBRyxDQUFDLEtBQXNELEVBQUUsT0FBZSxFQUFFLFVBQStCLEVBQUU7UUFDbkgsTUFBTSxlQUFlLEdBQUc7WUFDdEIsR0FBRyxJQUFJLENBQUMsY0FBYztZQUN0QixHQUFHLE9BQU87U0FDWCxDQUFDO1FBRUYsSUFBSSxJQUFJLENBQUMsYUFBYSxFQUFFLENBQUM7WUFDdkIsZUFBZSxDQUFDLGNBQWMsR0FBRyxJQUFJLENBQUMsYUFBYSxDQUFDO1FBQ3RELENBQUM7UUFFRCxVQUFVLENBQUMsR0FBRyxDQUFDLEtBQUssRUFBRSxPQUFPLEVBQUUsZUFBZSxDQUFDLENBQUM7SUFDbEQsQ0FBQztJQUVNLEtBQUssQ0FBQyxPQUFlLEVBQUUsVUFBK0IsRUFBRTtRQUM3RCxJQUFJLENBQUMsR0FBRyxDQUFDLE9BQU8sRUFBRSxPQUFPLEVBQUUsT0FBTyxDQUFDLENBQUM7SUFDdEMsQ0FBQztJQUVNLElBQUksQ0FBQyxPQUFlLEVBQUUsVUFBK0IsRUFBRTtRQUM1RCxJQUFJLENBQUMsR0FBRyxDQUFDLE1BQU0sRUFBRSxPQUFPLEVBQUUsT0FBTyxDQUFDLENBQUM7SUFDckMsQ0FBQztJQUVNLElBQUksQ0FBQyxPQUFlLEVBQUUsVUFBK0IsRUFBRTtRQUM1RCxJQUFJLENBQUMsR0FBRyxDQUFDLE1BQU0sRUFBRSxPQUFPLEVBQUUsT0FBTyxDQUFDLENBQUM7SUFDckMsQ0FBQztJQUVNLE9BQU8sQ0FBQyxPQUFlLEVBQUUsVUFBK0IsRUFBRTtRQUMvRCxJQUFJLENBQUMsR0FBRyxDQUFDLFNBQVMsRUFBRSxPQUFPLEVBQUUsT0FBTyxDQUFDLENBQUM7SUFDeEMsQ0FBQztJQUVNLEtBQUssQ0FBQyxPQUFlLEVBQUUsVUFBK0IsRUFBRTtRQUM3RCxJQUFJLENBQUMsR0FBRyxDQUFDLE9BQU8sRUFBRSxPQUFPLEVBQUUsT0FBTyxDQUFDLENBQUM7SUFDdEMsQ0FBQztJQUVELHFCQUFxQjtJQUNkLFVBQVUsQ0FBQyxPQUE0QixFQUFFLFlBQXFCLEtBQUs7UUFDeEUsSUFBSSxTQUFTLEVBQUUsQ0FBQztZQUNkLElBQUksQ0FBQyxjQUFjLEdBQUcsT0FBTyxDQUFDO1FBQ2hDLENBQUM7YUFBTSxDQUFDO1lBQ04sSUFBSSxDQUFDLGNBQWMsR0FBRztnQkFDcEIsR0FBRyxJQUFJLENBQUMsY0FBYztnQkFDdEIsR0FBRyxPQUFPO2FBQ1gsQ0FBQztRQUNKLENBQUM7SUFDSCxDQUFDO0lBRUQsNEJBQTRCO0lBQ3JCLGdCQUFnQixDQUFDLEtBQW9CLElBQUk7UUFDOUMsSUFBSSxDQUFDLGFBQWEsR0FBRyxFQUFFLElBQUksVUFBVSxFQUFFLENBQUM7UUFDeEMsT0FBTyxJQUFJLENBQUMsYUFBYSxDQUFDO0lBQzVCLENBQUM7SUFFTSxnQkFBZ0I7UUFDckIsT0FBTyxJQUFJLENBQUMsYUFBYSxDQUFDO0lBQzVCLENBQUM7SUFFTSxrQkFBa0I7UUFDdkIsSUFBSSxDQUFDLGFBQWEsR0FBRyxJQUFJLENBQUM7SUFDNUIsQ0FBQztDQUNGO0FBRUQsOEJBQThCO0FBQzlCLE1BQU0sQ0FBQyxNQUFNLE1BQU0sR0FBRyxJQUFJLGNBQWMsRUFBRSxDQUFDIn0= \ No newline at end of file diff --git a/dist_ts/mail/core/classes.bouncemanager.d.ts b/dist_ts/mail/core/classes.bouncemanager.d.ts deleted file mode 100644 index 33d671d..0000000 --- a/dist_ts/mail/core/classes.bouncemanager.d.ts +++ /dev/null @@ -1,185 +0,0 @@ -import type { Email } from './classes.email.js'; -/** - * Bounce types for categorizing the reasons for bounces - */ -export declare enum BounceType { - INVALID_RECIPIENT = "invalid_recipient", - DOMAIN_NOT_FOUND = "domain_not_found", - MAILBOX_FULL = "mailbox_full", - MAILBOX_INACTIVE = "mailbox_inactive", - BLOCKED = "blocked", - SPAM_RELATED = "spam_related", - POLICY_RELATED = "policy_related", - SERVER_UNAVAILABLE = "server_unavailable", - TEMPORARY_FAILURE = "temporary_failure", - QUOTA_EXCEEDED = "quota_exceeded", - NETWORK_ERROR = "network_error", - TIMEOUT = "timeout", - AUTO_RESPONSE = "auto_response", - CHALLENGE_RESPONSE = "challenge_response", - UNKNOWN = "unknown" -} -/** - * Hard vs soft bounce classification - */ -export declare enum BounceCategory { - HARD = "hard", - SOFT = "soft", - AUTO_RESPONSE = "auto_response", - UNKNOWN = "unknown" -} -/** - * Bounce data structure - */ -export interface BounceRecord { - id: string; - originalEmailId?: string; - recipient: string; - sender: string; - domain: string; - subject?: string; - bounceType: BounceType; - bounceCategory: BounceCategory; - timestamp: number; - smtpResponse?: string; - diagnosticCode?: string; - statusCode?: string; - headers?: Record; - processed: boolean; - retryCount?: number; - nextRetryTime?: number; -} -/** - * Retry strategy configuration for soft bounces - */ -interface RetryStrategy { - maxRetries: number; - initialDelay: number; - maxDelay: number; - backoffFactor: number; -} -/** - * Manager for handling email bounces - */ -export declare class BounceManager { - private retryStrategy; - private bounceStore; - private bounceCache; - private suppressionList; - private storageManager?; - constructor(options?: { - retryStrategy?: Partial; - maxCacheSize?: number; - cacheTTL?: number; - storageManager?: any; - }); - /** - * Process a bounce notification - * @param bounceData Bounce data to process - * @returns Processed bounce record - */ - processBounce(bounceData: Partial): Promise; - /** - * Process an SMTP failure as a bounce - * @param recipient Recipient email - * @param smtpResponse SMTP error response - * @param options Additional options - * @returns Processed bounce record - */ - processSmtpFailure(recipient: string, smtpResponse: string, options?: { - sender?: string; - originalEmailId?: string; - statusCode?: string; - headers?: Record; - }): Promise; - /** - * Process a bounce notification email - * @param bounceEmail The email containing bounce information - * @returns Processed bounce record or null if not a bounce - */ - processBounceEmail(bounceEmail: Email): Promise; - /** - * Handle a hard bounce by adding to suppression list - * @param bounce The bounce record - */ - private handleHardBounce; - /** - * Handle a soft bounce by scheduling a retry if eligible - * @param bounce The bounce record - */ - private handleSoftBounce; - /** - * Add an email address to the suppression list - * @param email Email address to suppress - * @param reason Reason for suppression - * @param expiresAt Expiration timestamp (undefined for permanent) - */ - addToSuppressionList(email: string, reason: string, expiresAt?: number): void; - /** - * Remove an email address from the suppression list - * @param email Email address to remove - */ - removeFromSuppressionList(email: string): void; - /** - * Check if an email is on the suppression list - * @param email Email address to check - * @returns Whether the email is suppressed - */ - isEmailSuppressed(email: string): boolean; - /** - * Get suppression information for an email - * @param email Email address to check - * @returns Suppression information or null if not suppressed - */ - getSuppressionInfo(email: string): { - reason: string; - timestamp: number; - expiresAt?: number; - } | null; - /** - * Save suppression list to disk - */ - private saveSuppressionList; - /** - * Load suppression list from disk - */ - private loadSuppressionList; - /** - * Save bounce record to disk - * @param bounce Bounce record to save - */ - private saveBounceRecord; - /** - * Update bounce cache with new bounce information - * @param bounce Bounce record to update cache with - */ - private updateBounceCache; - /** - * Check bounce history for an email address - * @param email Email address to check - * @returns Bounce information or null if no bounces - */ - getBounceInfo(email: string): { - lastBounce: number; - count: number; - type: BounceType; - category: BounceCategory; - } | null; - /** - * Get all known hard bounced addresses - * @returns Array of hard bounced email addresses - */ - getHardBouncedAddresses(): string[]; - /** - * Get suppression list - * @returns Array of suppressed email addresses - */ - getSuppressionList(): string[]; - /** - * Clear old bounce records (for maintenance) - * @param olderThan Timestamp to remove records older than - * @returns Number of records removed - */ - clearOldBounceRecords(olderThan: number): number; -} -export {}; diff --git a/dist_ts/mail/core/classes.bouncemanager.js b/dist_ts/mail/core/classes.bouncemanager.js deleted file mode 100644 index fce89fd..0000000 --- a/dist_ts/mail/core/classes.bouncemanager.js +++ /dev/null @@ -1,569 +0,0 @@ -import * as plugins from '../../plugins.js'; -import * as paths from '../../paths.js'; -import { logger } from '../../logger.js'; -import { SecurityLogger, SecurityLogLevel, SecurityEventType } from '../../security/index.js'; -import { RustSecurityBridge } from '../../security/classes.rustsecuritybridge.js'; -import { LRUCache } from 'lru-cache'; -/** - * Bounce types for categorizing the reasons for bounces - */ -export var BounceType; -(function (BounceType) { - // Hard bounces (permanent failures) - BounceType["INVALID_RECIPIENT"] = "invalid_recipient"; - BounceType["DOMAIN_NOT_FOUND"] = "domain_not_found"; - BounceType["MAILBOX_FULL"] = "mailbox_full"; - BounceType["MAILBOX_INACTIVE"] = "mailbox_inactive"; - BounceType["BLOCKED"] = "blocked"; - BounceType["SPAM_RELATED"] = "spam_related"; - BounceType["POLICY_RELATED"] = "policy_related"; - // Soft bounces (temporary failures) - BounceType["SERVER_UNAVAILABLE"] = "server_unavailable"; - BounceType["TEMPORARY_FAILURE"] = "temporary_failure"; - BounceType["QUOTA_EXCEEDED"] = "quota_exceeded"; - BounceType["NETWORK_ERROR"] = "network_error"; - BounceType["TIMEOUT"] = "timeout"; - // Special cases - BounceType["AUTO_RESPONSE"] = "auto_response"; - BounceType["CHALLENGE_RESPONSE"] = "challenge_response"; - BounceType["UNKNOWN"] = "unknown"; -})(BounceType || (BounceType = {})); -/** - * Hard vs soft bounce classification - */ -export var BounceCategory; -(function (BounceCategory) { - BounceCategory["HARD"] = "hard"; - BounceCategory["SOFT"] = "soft"; - BounceCategory["AUTO_RESPONSE"] = "auto_response"; - BounceCategory["UNKNOWN"] = "unknown"; -})(BounceCategory || (BounceCategory = {})); -/** - * Manager for handling email bounces - */ -export class BounceManager { - // Retry strategy with exponential backoff - retryStrategy = { - maxRetries: 5, - initialDelay: 15 * 60 * 1000, // 15 minutes - maxDelay: 24 * 60 * 60 * 1000, // 24 hours - backoffFactor: 2 - }; - // Store of bounced emails - bounceStore = []; - // Cache of recently bounced email addresses to avoid sending to known bad addresses - bounceCache; - // Suppression list for addresses that should not receive emails - suppressionList = new Map(); - storageManager; // StorageManager instance - constructor(options) { - // Set retry strategy with defaults - if (options?.retryStrategy) { - this.retryStrategy = { - ...this.retryStrategy, - ...options.retryStrategy - }; - } - // Initialize bounce cache with LRU (least recently used) caching - this.bounceCache = new LRUCache({ - max: options?.maxCacheSize || 10000, - ttl: options?.cacheTTL || 30 * 24 * 60 * 60 * 1000, // 30 days default - }); - // Store storage manager reference - this.storageManager = options?.storageManager; - // Load suppression list from storage - // Note: This is async but we can't await in constructor - // The suppression list will be loaded asynchronously - this.loadSuppressionList().catch(error => { - logger.log('error', `Failed to load suppression list on startup: ${error.message}`); - }); - } - /** - * Process a bounce notification - * @param bounceData Bounce data to process - * @returns Processed bounce record - */ - async processBounce(bounceData) { - try { - // Add required fields if missing - const bounce = { - id: bounceData.id || plugins.uuid.v4(), - recipient: bounceData.recipient, - sender: bounceData.sender, - domain: bounceData.domain || bounceData.recipient.split('@')[1], - subject: bounceData.subject, - bounceType: bounceData.bounceType || BounceType.UNKNOWN, - bounceCategory: bounceData.bounceCategory || BounceCategory.UNKNOWN, - timestamp: bounceData.timestamp || Date.now(), - smtpResponse: bounceData.smtpResponse, - diagnosticCode: bounceData.diagnosticCode, - statusCode: bounceData.statusCode, - headers: bounceData.headers, - processed: false, - originalEmailId: bounceData.originalEmailId, - retryCount: bounceData.retryCount || 0, - nextRetryTime: bounceData.nextRetryTime - }; - // Determine bounce type and category via Rust bridge if not provided - if (!bounceData.bounceType || bounceData.bounceType === BounceType.UNKNOWN) { - const bridge = RustSecurityBridge.getInstance(); - const rustResult = await bridge.detectBounce({ - smtpResponse: bounce.smtpResponse, - diagnosticCode: bounce.diagnosticCode, - statusCode: bounce.statusCode, - }); - bounce.bounceType = rustResult.bounce_type; - bounce.bounceCategory = rustResult.category; - } - // Process the bounce based on category - switch (bounce.bounceCategory) { - case BounceCategory.HARD: - // Handle hard bounce - add to suppression list - await this.handleHardBounce(bounce); - break; - case BounceCategory.SOFT: - // Handle soft bounce - schedule retry if eligible - await this.handleSoftBounce(bounce); - break; - case BounceCategory.AUTO_RESPONSE: - // Handle auto-response - typically no action needed - logger.log('info', `Auto-response detected for ${bounce.recipient}`); - break; - default: - // Unknown bounce type - log for investigation - logger.log('warn', `Unknown bounce type for ${bounce.recipient}`, { - bounceType: bounce.bounceType, - smtpResponse: bounce.smtpResponse - }); - break; - } - // Store the bounce record - bounce.processed = true; - this.bounceStore.push(bounce); - // Update the bounce cache - this.updateBounceCache(bounce); - // Log the bounce - logger.log(bounce.bounceCategory === BounceCategory.HARD ? 'warn' : 'info', `Email bounce processed: ${bounce.bounceCategory} bounce for ${bounce.recipient}`, { - bounceType: bounce.bounceType, - domain: bounce.domain, - category: bounce.bounceCategory - }); - // Enhanced security logging - SecurityLogger.getInstance().logEvent({ - level: bounce.bounceCategory === BounceCategory.HARD - ? SecurityLogLevel.WARN - : SecurityLogLevel.INFO, - type: SecurityEventType.EMAIL_VALIDATION, - message: `Email bounce detected: ${bounce.bounceCategory} bounce for recipient`, - domain: bounce.domain, - details: { - recipient: bounce.recipient, - bounceType: bounce.bounceType, - smtpResponse: bounce.smtpResponse, - diagnosticCode: bounce.diagnosticCode, - statusCode: bounce.statusCode - }, - success: false - }); - return bounce; - } - catch (error) { - logger.log('error', `Error processing bounce: ${error.message}`, { - error: error.message, - bounceData - }); - throw error; - } - } - /** - * Process an SMTP failure as a bounce - * @param recipient Recipient email - * @param smtpResponse SMTP error response - * @param options Additional options - * @returns Processed bounce record - */ - async processSmtpFailure(recipient, smtpResponse, options = {}) { - // Create bounce data from SMTP failure - const bounceData = { - recipient, - sender: options.sender || '', - domain: recipient.split('@')[1], - smtpResponse, - statusCode: options.statusCode, - headers: options.headers, - originalEmailId: options.originalEmailId, - timestamp: Date.now() - }; - // Process as a regular bounce - return this.processBounce(bounceData); - } - /** - * Process a bounce notification email - * @param bounceEmail The email containing bounce information - * @returns Processed bounce record or null if not a bounce - */ - async processBounceEmail(bounceEmail) { - try { - // Check if this is a bounce notification - const subject = bounceEmail.getSubject(); - const body = bounceEmail.getBody(); - // Check for common bounce notification subject patterns - const isBounceSubject = /mail delivery|delivery (failed|status|notification)|failure notice|returned mail|undeliverable|delivery problem/i.test(subject); - if (!isBounceSubject) { - // Not a bounce notification based on subject - return null; - } - // Extract original recipient from the body or headers - let recipient = ''; - let originalMessageId = ''; - // Extract recipient from common bounce formats - const recipientMatch = body.match(/(?:failed recipient|to[:=]\s*|recipient:|delivery failed:)\s*?/i); - if (recipientMatch && recipientMatch[1]) { - recipient = recipientMatch[1]; - } - // Extract diagnostic code - let diagnosticCode = ''; - const diagnosticMatch = body.match(/diagnostic(?:-|\\s+)code:\s*(.+)(?:\n|$)/i); - if (diagnosticMatch && diagnosticMatch[1]) { - diagnosticCode = diagnosticMatch[1].trim(); - } - // Extract SMTP status code - let statusCode = ''; - const statusMatch = body.match(/status(?:-|\\s+)code:\s*([0-9.]+)/i); - if (statusMatch && statusMatch[1]) { - statusCode = statusMatch[1].trim(); - } - // If recipient not found in standard patterns, try DSN (Delivery Status Notification) format - if (!recipient) { - // Look for DSN format with Original-Recipient or Final-Recipient fields - const originalRecipientMatch = body.match(/original-recipient:.*?([a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,})/i); - const finalRecipientMatch = body.match(/final-recipient:.*?([a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,})/i); - if (originalRecipientMatch && originalRecipientMatch[1]) { - recipient = originalRecipientMatch[1]; - } - else if (finalRecipientMatch && finalRecipientMatch[1]) { - recipient = finalRecipientMatch[1]; - } - } - // If still no recipient, can't process as bounce - if (!recipient) { - logger.log('warn', 'Could not extract recipient from bounce notification', { - subject, - sender: bounceEmail.from - }); - return null; - } - // Extract original message ID if available - const messageIdMatch = body.match(/original[ -]message[ -]id:[ \t]*]+)>?/i); - if (messageIdMatch && messageIdMatch[1]) { - originalMessageId = messageIdMatch[1].trim(); - } - // Create bounce data - const bounceData = { - recipient, - sender: bounceEmail.from, - domain: recipient.split('@')[1], - subject: bounceEmail.getSubject(), - diagnosticCode, - statusCode, - timestamp: Date.now(), - headers: {} - }; - // Process as a regular bounce - return this.processBounce(bounceData); - } - catch (error) { - logger.log('error', `Error processing bounce email: ${error.message}`); - return null; - } - } - /** - * Handle a hard bounce by adding to suppression list - * @param bounce The bounce record - */ - async handleHardBounce(bounce) { - // Add to suppression list permanently (no expiry) - this.addToSuppressionList(bounce.recipient, `Hard bounce: ${bounce.bounceType}`, undefined); - // Increment bounce count in cache - this.updateBounceCache(bounce); - // Save to permanent storage - await this.saveBounceRecord(bounce); - // Log hard bounce for monitoring - logger.log('warn', `Hard bounce for ${bounce.recipient}: ${bounce.bounceType}`, { - domain: bounce.domain, - smtpResponse: bounce.smtpResponse, - diagnosticCode: bounce.diagnosticCode - }); - } - /** - * Handle a soft bounce by scheduling a retry if eligible - * @param bounce The bounce record - */ - async handleSoftBounce(bounce) { - // Check if we've exceeded max retries - if (bounce.retryCount >= this.retryStrategy.maxRetries) { - logger.log('warn', `Max retries exceeded for ${bounce.recipient}, treating as hard bounce`); - // Convert to hard bounce after max retries - bounce.bounceCategory = BounceCategory.HARD; - await this.handleHardBounce(bounce); - return; - } - // Calculate next retry time with exponential backoff - const delay = Math.min(this.retryStrategy.initialDelay * Math.pow(this.retryStrategy.backoffFactor, bounce.retryCount), this.retryStrategy.maxDelay); - bounce.retryCount++; - bounce.nextRetryTime = Date.now() + delay; - // Add to suppression list temporarily (with expiry) - this.addToSuppressionList(bounce.recipient, `Soft bounce: ${bounce.bounceType}`, bounce.nextRetryTime); - // Log the retry schedule - logger.log('info', `Scheduled retry ${bounce.retryCount} for ${bounce.recipient} at ${new Date(bounce.nextRetryTime).toISOString()}`, { - bounceType: bounce.bounceType, - retryCount: bounce.retryCount, - nextRetry: bounce.nextRetryTime - }); - } - /** - * Add an email address to the suppression list - * @param email Email address to suppress - * @param reason Reason for suppression - * @param expiresAt Expiration timestamp (undefined for permanent) - */ - addToSuppressionList(email, reason, expiresAt) { - this.suppressionList.set(email.toLowerCase(), { - reason, - timestamp: Date.now(), - expiresAt - }); - // Save asynchronously without blocking - this.saveSuppressionList().catch(error => { - logger.log('error', `Failed to save suppression list after adding ${email}: ${error.message}`); - }); - logger.log('info', `Added ${email} to suppression list`, { - reason, - expiresAt: expiresAt ? new Date(expiresAt).toISOString() : 'permanent' - }); - } - /** - * Remove an email address from the suppression list - * @param email Email address to remove - */ - removeFromSuppressionList(email) { - const wasRemoved = this.suppressionList.delete(email.toLowerCase()); - if (wasRemoved) { - // Save asynchronously without blocking - this.saveSuppressionList().catch(error => { - logger.log('error', `Failed to save suppression list after removing ${email}: ${error.message}`); - }); - logger.log('info', `Removed ${email} from suppression list`); - } - } - /** - * Check if an email is on the suppression list - * @param email Email address to check - * @returns Whether the email is suppressed - */ - isEmailSuppressed(email) { - const lowercaseEmail = email.toLowerCase(); - const suppression = this.suppressionList.get(lowercaseEmail); - if (!suppression) { - return false; - } - // Check if suppression has expired - if (suppression.expiresAt && Date.now() > suppression.expiresAt) { - this.suppressionList.delete(lowercaseEmail); - // Save asynchronously without blocking - this.saveSuppressionList().catch(error => { - logger.log('error', `Failed to save suppression list after expiry cleanup: ${error.message}`); - }); - return false; - } - return true; - } - /** - * Get suppression information for an email - * @param email Email address to check - * @returns Suppression information or null if not suppressed - */ - getSuppressionInfo(email) { - const lowercaseEmail = email.toLowerCase(); - const suppression = this.suppressionList.get(lowercaseEmail); - if (!suppression) { - return null; - } - // Check if suppression has expired - if (suppression.expiresAt && Date.now() > suppression.expiresAt) { - this.suppressionList.delete(lowercaseEmail); - // Save asynchronously without blocking - this.saveSuppressionList().catch(error => { - logger.log('error', `Failed to save suppression list after expiry cleanup: ${error.message}`); - }); - return null; - } - return suppression; - } - /** - * Save suppression list to disk - */ - async saveSuppressionList() { - try { - const suppressionData = JSON.stringify(Array.from(this.suppressionList.entries())); - if (this.storageManager) { - // Use storage manager - await this.storageManager.set('/email/bounces/suppression-list.json', suppressionData); - } - else { - // Fall back to filesystem - await plugins.smartfs.file(plugins.path.join(paths.dataDir, 'emails', 'suppression_list.json')).write(suppressionData); - } - } - catch (error) { - logger.log('error', `Failed to save suppression list: ${error.message}`); - } - } - /** - * Load suppression list from disk - */ - async loadSuppressionList() { - try { - let entries = null; - let needsMigration = false; - if (this.storageManager) { - // Try to load from storage manager first - const suppressionData = await this.storageManager.get('/email/bounces/suppression-list.json'); - if (suppressionData) { - entries = JSON.parse(suppressionData); - } - else { - // Check if data exists in filesystem and migrate - const suppressionPath = plugins.path.join(paths.dataDir, 'emails', 'suppression_list.json'); - if (plugins.fs.existsSync(suppressionPath)) { - const data = plugins.fs.readFileSync(suppressionPath, 'utf8'); - entries = JSON.parse(data); - needsMigration = true; - logger.log('info', 'Migrating suppression list from filesystem to StorageManager'); - } - } - } - else { - // No storage manager, use filesystem directly - const suppressionPath = plugins.path.join(paths.dataDir, 'emails', 'suppression_list.json'); - if (plugins.fs.existsSync(suppressionPath)) { - const data = plugins.fs.readFileSync(suppressionPath, 'utf8'); - entries = JSON.parse(data); - } - } - if (entries) { - this.suppressionList = new Map(entries); - // Clean expired entries - const now = Date.now(); - let expiredCount = 0; - for (const [email, info] of this.suppressionList.entries()) { - if (info.expiresAt && now > info.expiresAt) { - this.suppressionList.delete(email); - expiredCount++; - } - } - if (expiredCount > 0 || needsMigration) { - logger.log('info', `Cleaned ${expiredCount} expired entries from suppression list`); - await this.saveSuppressionList(); - } - logger.log('info', `Loaded ${this.suppressionList.size} entries from suppression list`); - } - } - catch (error) { - logger.log('error', `Failed to load suppression list: ${error.message}`); - } - } - /** - * Save bounce record to disk - * @param bounce Bounce record to save - */ - async saveBounceRecord(bounce) { - try { - const bounceData = JSON.stringify(bounce, null, 2); - if (this.storageManager) { - // Use storage manager - await this.storageManager.set(`/email/bounces/records/${bounce.id}.json`, bounceData); - } - else { - // Fall back to filesystem - const bouncePath = plugins.path.join(paths.dataDir, 'emails', 'bounces', `${bounce.id}.json`); - // Ensure directory exists - const bounceDir = plugins.path.join(paths.dataDir, 'emails', 'bounces'); - await plugins.smartfs.directory(bounceDir).recursive().create(); - await plugins.smartfs.file(bouncePath).write(bounceData); - } - } - catch (error) { - logger.log('error', `Failed to save bounce record: ${error.message}`); - } - } - /** - * Update bounce cache with new bounce information - * @param bounce Bounce record to update cache with - */ - updateBounceCache(bounce) { - const email = bounce.recipient.toLowerCase(); - const existing = this.bounceCache.get(email); - if (existing) { - // Update existing cache entry - existing.lastBounce = bounce.timestamp; - existing.count++; - existing.type = bounce.bounceType; - existing.category = bounce.bounceCategory; - } - else { - // Create new cache entry - this.bounceCache.set(email, { - lastBounce: bounce.timestamp, - count: 1, - type: bounce.bounceType, - category: bounce.bounceCategory - }); - } - } - /** - * Check bounce history for an email address - * @param email Email address to check - * @returns Bounce information or null if no bounces - */ - getBounceInfo(email) { - return this.bounceCache.get(email.toLowerCase()) || null; - } - /** - * Get all known hard bounced addresses - * @returns Array of hard bounced email addresses - */ - getHardBouncedAddresses() { - const hardBounced = []; - for (const [email, info] of this.bounceCache.entries()) { - if (info.category === BounceCategory.HARD) { - hardBounced.push(email); - } - } - return hardBounced; - } - /** - * Get suppression list - * @returns Array of suppressed email addresses - */ - getSuppressionList() { - return Array.from(this.suppressionList.keys()); - } - /** - * Clear old bounce records (for maintenance) - * @param olderThan Timestamp to remove records older than - * @returns Number of records removed - */ - clearOldBounceRecords(olderThan) { - let removed = 0; - this.bounceStore = this.bounceStore.filter(bounce => { - if (bounce.timestamp < olderThan) { - removed++; - return false; - } - return true; - }); - return removed; - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5ib3VuY2VtYW5hZ2VyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vdHMvbWFpbC9jb3JlL2NsYXNzZXMuYm91bmNlbWFuYWdlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssT0FBTyxNQUFNLGtCQUFrQixDQUFDO0FBQzVDLE9BQU8sS0FBSyxLQUFLLE1BQU0sZ0JBQWdCLENBQUM7QUFDeEMsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQ3pDLE9BQU8sRUFBRSxjQUFjLEVBQUUsZ0JBQWdCLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUM5RixPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSw4Q0FBOEMsQ0FBQztBQUNsRixPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sV0FBVyxDQUFDO0FBR3JDOztHQUVHO0FBQ0gsTUFBTSxDQUFOLElBQVksVUFxQlg7QUFyQkQsV0FBWSxVQUFVO0lBQ3BCLG9DQUFvQztJQUNwQyxxREFBdUMsQ0FBQTtJQUN2QyxtREFBcUMsQ0FBQTtJQUNyQywyQ0FBNkIsQ0FBQTtJQUM3QixtREFBcUMsQ0FBQTtJQUNyQyxpQ0FBbUIsQ0FBQTtJQUNuQiwyQ0FBNkIsQ0FBQTtJQUM3QiwrQ0FBaUMsQ0FBQTtJQUVqQyxvQ0FBb0M7SUFDcEMsdURBQXlDLENBQUE7SUFDekMscURBQXVDLENBQUE7SUFDdkMsK0NBQWlDLENBQUE7SUFDakMsNkNBQStCLENBQUE7SUFDL0IsaUNBQW1CLENBQUE7SUFFbkIsZ0JBQWdCO0lBQ2hCLDZDQUErQixDQUFBO0lBQy9CLHVEQUF5QyxDQUFBO0lBQ3pDLGlDQUFtQixDQUFBO0FBQ3JCLENBQUMsRUFyQlcsVUFBVSxLQUFWLFVBQVUsUUFxQnJCO0FBRUQ7O0dBRUc7QUFDSCxNQUFNLENBQU4sSUFBWSxjQUtYO0FBTEQsV0FBWSxjQUFjO0lBQ3hCLCtCQUFhLENBQUE7SUFDYiwrQkFBYSxDQUFBO0lBQ2IsaURBQStCLENBQUE7SUFDL0IscUNBQW1CLENBQUE7QUFDckIsQ0FBQyxFQUxXLGNBQWMsS0FBZCxjQUFjLFFBS3pCO0FBa0NEOztHQUVHO0FBQ0gsTUFBTSxPQUFPLGFBQWE7SUFDeEIsMENBQTBDO0lBQ2xDLGFBQWEsR0FBa0I7UUFDckMsVUFBVSxFQUFFLENBQUM7UUFDYixZQUFZLEVBQUUsRUFBRSxHQUFHLEVBQUUsR0FBRyxJQUFJLEVBQUUsYUFBYTtRQUMzQyxRQUFRLEVBQUUsRUFBRSxHQUFHLEVBQUUsR0FBRyxFQUFFLEdBQUcsSUFBSSxFQUFFLFdBQVc7UUFDMUMsYUFBYSxFQUFFLENBQUM7S0FDakIsQ0FBQztJQUVGLDBCQUEwQjtJQUNsQixXQUFXLEdBQW1CLEVBQUUsQ0FBQztJQUV6QyxvRkFBb0Y7SUFDNUUsV0FBVyxDQUtoQjtJQUVILGdFQUFnRTtJQUN4RCxlQUFlLEdBSWxCLElBQUksR0FBRyxFQUFFLENBQUM7SUFFUCxjQUFjLENBQU8sQ0FBQywwQkFBMEI7SUFFeEQsWUFBWSxPQUtYO1FBQ0MsbUNBQW1DO1FBQ25DLElBQUksT0FBTyxFQUFFLGFBQWEsRUFBRSxDQUFDO1lBQzNCLElBQUksQ0FBQyxhQUFhLEdBQUc7Z0JBQ25CLEdBQUcsSUFBSSxDQUFDLGFBQWE7Z0JBQ3JCLEdBQUcsT0FBTyxDQUFDLGFBQWE7YUFDekIsQ0FBQztRQUNKLENBQUM7UUFFRCxpRUFBaUU7UUFDakUsSUFBSSxDQUFDLFdBQVcsR0FBRyxJQUFJLFFBQVEsQ0FBYztZQUMzQyxHQUFHLEVBQUUsT0FBTyxFQUFFLFlBQVksSUFBSSxLQUFLO1lBQ25DLEdBQUcsRUFBRSxPQUFPLEVBQUUsUUFBUSxJQUFJLEVBQUUsR0FBRyxFQUFFLEdBQUcsRUFBRSxHQUFHLEVBQUUsR0FBRyxJQUFJLEVBQUUsa0JBQWtCO1NBQ3ZFLENBQUMsQ0FBQztRQUVILGtDQUFrQztRQUNsQyxJQUFJLENBQUMsY0FBYyxHQUFHLE9BQU8sRUFBRSxjQUFjLENBQUM7UUFFOUMscUNBQXFDO1FBQ3JDLHdEQUF3RDtRQUN4RCxxREFBcUQ7UUFDckQsSUFBSSxDQUFDLG1CQUFtQixFQUFFLENBQUMsS0FBSyxDQUFDLEtBQUssQ0FBQyxFQUFFO1lBQ3ZDLE1BQU0sQ0FBQyxHQUFHLENBQUMsT0FBTyxFQUFFLCtDQUErQyxLQUFLLENBQUMsT0FBTyxFQUFFLENBQUMsQ0FBQztRQUN0RixDQUFDLENBQUMsQ0FBQztJQUNMLENBQUM7SUFFRDs7OztPQUlHO0lBQ0ksS0FBSyxDQUFDLGFBQWEsQ0FBQyxVQUFpQztRQUMxRCxJQUFJLENBQUM7WUFDSCxpQ0FBaUM7WUFDakMsTUFBTSxNQUFNLEdBQWlCO2dCQUMzQixFQUFFLEVBQUUsVUFBVSxDQUFDLEVBQUUsSUFBSSxPQUFPLENBQUMsSUFBSSxDQUFDLEVBQUUsRUFBRTtnQkFDdEMsU0FBUyxFQUFFLFVBQVUsQ0FBQyxTQUFTO2dCQUMvQixNQUFNLEVBQUUsVUFBVSxDQUFDLE1BQU07Z0JBQ3pCLE1BQU0sRUFBRSxVQUFVLENBQUMsTUFBTSxJQUFJLFVBQVUsQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQztnQkFDL0QsT0FBTyxFQUFFLFVBQVUsQ0FBQyxPQUFPO2dCQUMzQixVQUFVLEVBQUUsVUFBVSxDQUFDLFVBQVUsSUFBSSxVQUFVLENBQUMsT0FBTztnQkFDdkQsY0FBYyxFQUFFLFVBQVUsQ0FBQyxjQUFjLElBQUksY0FBYyxDQUFDLE9BQU87Z0JBQ25FLFNBQVMsRUFBRSxVQUFVLENBQUMsU0FBUyxJQUFJLElBQUksQ0FBQyxHQUFHLEVBQUU7Z0JBQzdDLFlBQVksRUFBRSxVQUFVLENBQUMsWUFBWTtnQkFDckMsY0FBYyxFQUFFLFVBQVUsQ0FBQyxjQUFjO2dCQUN6QyxVQUFVLEVBQUUsVUFBVSxDQUFDLFVBQVU7Z0JBQ2pDLE9BQU8sRUFBRSxVQUFVLENBQUMsT0FBTztnQkFDM0IsU0FBUyxFQUFFLEtBQUs7Z0JBQ2hCLGVBQWUsRUFBRSxVQUFVLENBQUMsZUFBZTtnQkFDM0MsVUFBVSxFQUFFLFVBQVUsQ0FBQyxVQUFVLElBQUksQ0FBQztnQkFDdEMsYUFBYSxFQUFFLFVBQVUsQ0FBQyxhQUFhO2FBQ3hDLENBQUM7WUFFRixxRUFBcUU7WUFDckUsSUFBSSxDQUFDLFVBQVUsQ0FBQyxVQUFVLElBQUksVUFBVSxDQUFDLFVBQVUsS0FBSyxVQUFVLENBQUMsT0FBTyxFQUFFLENBQUM7Z0JBQzNFLE1BQU0sTUFBTSxHQUFHLGtCQUFrQixDQUFDLFdBQVcsRUFBRSxDQUFDO2dCQUNoRCxNQUFNLFVBQVUsR0FBRyxNQUFNLE1BQU0sQ0FBQyxZQUFZLENBQUM7b0JBQzNDLFlBQVksRUFBRSxNQUFNLENBQUMsWUFBWTtvQkFDakMsY0FBYyxFQUFFLE1BQU0sQ0FBQyxjQUFjO29CQUNyQyxVQUFVLEVBQUUsTUFBTSxDQUFDLFVBQVU7aUJBQzlCLENBQUMsQ0FBQztnQkFDSCxNQUFNLENBQUMsVUFBVSxHQUFHLFVBQVUsQ0FBQyxXQUF5QixDQUFDO2dCQUN6RCxNQUFNLENBQUMsY0FBYyxHQUFHLFVBQVUsQ0FBQyxRQUEwQixDQUFDO1lBQ2hFLENBQUM7WUFFRCx1Q0FBdUM7WUFDdkMsUUFBUSxNQUFNLENBQUMsY0FBYyxFQUFFLENBQUM7Z0JBQzlCLEtBQUssY0FBYyxDQUFDLElBQUk7b0JBQ3RCLCtDQUErQztvQkFDL0MsTUFBTSxJQUFJLENBQUMsZ0JBQWdCLENBQUMsTUFBTSxDQUFDLENBQUM7b0JBQ3BDLE1BQU07Z0JBRVIsS0FBSyxjQUFjLENBQUMsSUFBSTtvQkFDdEIsa0RBQWtEO29CQUNsRCxNQUFNLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxNQUFNLENBQUMsQ0FBQztvQkFDcEMsTUFBTTtnQkFFUixLQUFLLGNBQWMsQ0FBQyxhQUFhO29CQUMvQixvREFBb0Q7b0JBQ3BELE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLDhCQUE4QixNQUFNLENBQUMsU0FBUyxFQUFFLENBQUMsQ0FBQztvQkFDckUsTUFBTTtnQkFFUjtvQkFDRSw4Q0FBOEM7b0JBQzlDLE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLDJCQUEyQixNQUFNLENBQUMsU0FBUyxFQUFFLEVBQUU7d0JBQ2hFLFVBQVUsRUFBRSxNQUFNLENBQUMsVUFBVTt3QkFDN0IsWUFBWSxFQUFFLE1BQU0sQ0FBQyxZQUFZO3FCQUNsQyxDQUFDLENBQUM7b0JBQ0gsTUFBTTtZQUNWLENBQUM7WUFFRCwwQkFBMEI7WUFDMUIsTUFBTSxDQUFDLFNBQVMsR0FBRyxJQUFJLENBQUM7WUFDeEIsSUFBSSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUM7WUFFOUIsMEJBQTBCO1lBQzFCLElBQUksQ0FBQyxpQkFBaUIsQ0FBQyxNQUFNLENBQUMsQ0FBQztZQUUvQixpQkFBaUI7WUFDakIsTUFBTSxDQUFDLEdBQUcsQ0FDUixNQUFNLENBQUMsY0FBYyxLQUFLLGNBQWMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsTUFBTSxFQUMvRCwyQkFBMkIsTUFBTSxDQUFDLGNBQWMsZUFBZSxNQUFNLENBQUMsU0FBUyxFQUFFLEVBQ2pGO2dCQUNFLFVBQVUsRUFBRSxNQUFNLENBQUMsVUFBVTtnQkFDN0IsTUFBTSxFQUFFLE1BQU0sQ0FBQyxNQUFNO2dCQUNyQixRQUFRLEVBQUUsTUFBTSxDQUFDLGNBQWM7YUFDaEMsQ0FDRixDQUFDO1lBRUYsNEJBQTRCO1lBQzVCLGNBQWMsQ0FBQyxXQUFXLEVBQUUsQ0FBQyxRQUFRLENBQUM7Z0JBQ3BDLEtBQUssRUFBRSxNQUFNLENBQUMsY0FBYyxLQUFLLGNBQWMsQ0FBQyxJQUFJO29CQUNsRCxDQUFDLENBQUMsZ0JBQWdCLENBQUMsSUFBSTtvQkFDdkIsQ0FBQyxDQUFDLGdCQUFnQixDQUFDLElBQUk7Z0JBQ3pCLElBQUksRUFBRSxpQkFBaUIsQ0FBQyxnQkFBZ0I7Z0JBQ3hDLE9BQU8sRUFBRSwwQkFBMEIsTUFBTSxDQUFDLGNBQWMsdUJBQXVCO2dCQUMvRSxNQUFNLEVBQUUsTUFBTSxDQUFDLE1BQU07Z0JBQ3JCLE9BQU8sRUFBRTtvQkFDUCxTQUFTLEVBQUUsTUFBTSxDQUFDLFNBQVM7b0JBQzNCLFVBQVUsRUFBRSxNQUFNLENBQUMsVUFBVTtvQkFDN0IsWUFBWSxFQUFFLE1BQU0sQ0FBQyxZQUFZO29CQUNqQyxjQUFjLEVBQUUsTUFBTSxDQUFDLGNBQWM7b0JBQ3JDLFVBQVUsRUFBRSxNQUFNLENBQUMsVUFBVTtpQkFDOUI7Z0JBQ0QsT0FBTyxFQUFFLEtBQUs7YUFDZixDQUFDLENBQUM7WUFFSCxPQUFPLE1BQU0sQ0FBQztRQUNoQixDQUFDO1FBQUMsT0FBTyxLQUFLLEVBQUUsQ0FBQztZQUNmLE1BQU0sQ0FBQyxHQUFHLENBQUMsT0FBTyxFQUFFLDRCQUE0QixLQUFLLENBQUMsT0FBTyxFQUFFLEVBQUU7Z0JBQy9ELEtBQUssRUFBRSxLQUFLLENBQUMsT0FBTztnQkFDcEIsVUFBVTthQUNYLENBQUMsQ0FBQztZQUNILE1BQU0sS0FBSyxDQUFDO1FBQ2QsQ0FBQztJQUNILENBQUM7SUFFRDs7Ozs7O09BTUc7SUFDSSxLQUFLLENBQUMsa0JBQWtCLENBQzdCLFNBQWlCLEVBQ2pCLFlBQW9CLEVBQ3BCLFVBS0ksRUFBRTtRQUVOLHVDQUF1QztRQUN2QyxNQUFNLFVBQVUsR0FBMEI7WUFDeEMsU0FBUztZQUNULE1BQU0sRUFBRSxPQUFPLENBQUMsTUFBTSxJQUFJLEVBQUU7WUFDNUIsTUFBTSxFQUFFLFNBQVMsQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQy9CLFlBQVk7WUFDWixVQUFVLEVBQUUsT0FBTyxDQUFDLFVBQVU7WUFDOUIsT0FBTyxFQUFFLE9BQU8sQ0FBQyxPQUFPO1lBQ3hCLGVBQWUsRUFBRSxPQUFPLENBQUMsZUFBZTtZQUN4QyxTQUFTLEVBQUUsSUFBSSxDQUFDLEdBQUcsRUFBRTtTQUN0QixDQUFDO1FBRUYsOEJBQThCO1FBQzlCLE9BQU8sSUFBSSxDQUFDLGFBQWEsQ0FBQyxVQUFVLENBQUMsQ0FBQztJQUN4QyxDQUFDO0lBRUQ7Ozs7T0FJRztJQUNJLEtBQUssQ0FBQyxrQkFBa0IsQ0FBQyxXQUFrQjtRQUNoRCxJQUFJLENBQUM7WUFDSCx5Q0FBeUM7WUFDekMsTUFBTSxPQUFPLEdBQUcsV0FBVyxDQUFDLFVBQVUsRUFBRSxDQUFDO1lBQ3pDLE1BQU0sSUFBSSxHQUFHLFdBQVcsQ0FBQyxPQUFPLEVBQUUsQ0FBQztZQUVuQyx3REFBd0Q7WUFDeEQsTUFBTSxlQUFlLEdBQUcsa0hBQWtILENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDO1lBRXpKLElBQUksQ0FBQyxlQUFlLEVBQUUsQ0FBQztnQkFDckIsNkNBQTZDO2dCQUM3QyxPQUFPLElBQUksQ0FBQztZQUNkLENBQUM7WUFFRCxzREFBc0Q7WUFDdEQsSUFBSSxTQUFTLEdBQUcsRUFBRSxDQUFDO1lBQ25CLElBQUksaUJBQWlCLEdBQUcsRUFBRSxDQUFDO1lBRTNCLCtDQUErQztZQUMvQyxNQUFNLGNBQWMsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLG9IQUFvSCxDQUFDLENBQUM7WUFDeEosSUFBSSxjQUFjLElBQUksY0FBYyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUM7Z0JBQ3hDLFNBQVMsR0FBRyxjQUFjLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDaEMsQ0FBQztZQUVELDBCQUEwQjtZQUMxQixJQUFJLGNBQWMsR0FBRyxFQUFFLENBQUM7WUFDeEIsTUFBTSxlQUFlLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQywyQ0FBMkMsQ0FBQyxDQUFDO1lBQ2hGLElBQUksZUFBZSxJQUFJLGVBQWUsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDO2dCQUMxQyxjQUFjLEdBQUcsZUFBZSxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksRUFBRSxDQUFDO1lBQzdDLENBQUM7WUFFRCwyQkFBMkI7WUFDM0IsSUFBSSxVQUFVLEdBQUcsRUFBRSxDQUFDO1lBQ3BCLE1BQU0sV0FBVyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsb0NBQW9DLENBQUMsQ0FBQztZQUNyRSxJQUFJLFdBQVcsSUFBSSxXQUFXLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQztnQkFDbEMsVUFBVSxHQUFHLFdBQVcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLEVBQUUsQ0FBQztZQUNyQyxDQUFDO1lBRUQsNkZBQTZGO1lBQzdGLElBQUksQ0FBQyxTQUFTLEVBQUUsQ0FBQztnQkFDZix3RUFBd0U7Z0JBQ3hFLE1BQU0sc0JBQXNCLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyx5RUFBeUUsQ0FBQyxDQUFDO2dCQUNySCxNQUFNLG1CQUFtQixHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsc0VBQXNFLENBQUMsQ0FBQztnQkFFL0csSUFBSSxzQkFBc0IsSUFBSSxzQkFBc0IsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDO29CQUN4RCxTQUFTLEdBQUcsc0JBQXNCLENBQUMsQ0FBQyxDQUFDLENBQUM7Z0JBQ3hDLENBQUM7cUJBQU0sSUFBSSxtQkFBbUIsSUFBSSxtQkFBbUIsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDO29CQUN6RCxTQUFTLEdBQUcsbUJBQW1CLENBQUMsQ0FBQyxDQUFDLENBQUM7Z0JBQ3JDLENBQUM7WUFDSCxDQUFDO1lBRUQsaURBQWlEO1lBQ2pELElBQUksQ0FBQyxTQUFTLEVBQUUsQ0FBQztnQkFDZixNQUFNLENBQUMsR0FBRyxDQUFDLE1BQU0sRUFBRSxzREFBc0QsRUFBRTtvQkFDekUsT0FBTztvQkFDUCxNQUFNLEVBQUUsV0FBVyxDQUFDLElBQUk7aUJBQ3pCLENBQUMsQ0FBQztnQkFDSCxPQUFPLElBQUksQ0FBQztZQUNkLENBQUM7WUFFRCwyQ0FBMkM7WUFDM0MsTUFBTSxjQUFjLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyw4Q0FBOEMsQ0FBQyxDQUFDO1lBQ2xGLElBQUksY0FBYyxJQUFJLGNBQWMsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDO2dCQUN4QyxpQkFBaUIsR0FBRyxjQUFjLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxFQUFFLENBQUM7WUFDL0MsQ0FBQztZQUVELHFCQUFxQjtZQUNyQixNQUFNLFVBQVUsR0FBMEI7Z0JBQ3hDLFNBQVM7Z0JBQ1QsTUFBTSxFQUFFLFdBQVcsQ0FBQyxJQUFJO2dCQUN4QixNQUFNLEVBQUUsU0FBUyxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUM7Z0JBQy9CLE9BQU8sRUFBRSxXQUFXLENBQUMsVUFBVSxFQUFFO2dCQUNqQyxjQUFjO2dCQUNkLFVBQVU7Z0JBQ1YsU0FBUyxFQUFFLElBQUksQ0FBQyxHQUFHLEVBQUU7Z0JBQ3JCLE9BQU8sRUFBRSxFQUFFO2FBQ1osQ0FBQztZQUVGLDhCQUE4QjtZQUM5QixPQUFPLElBQUksQ0FBQyxhQUFhLENBQUMsVUFBVSxDQUFDLENBQUM7UUFDeEMsQ0FBQztRQUFDLE9BQU8sS0FBSyxFQUFFLENBQUM7WUFDZixNQUFNLENBQUMsR0FBRyxDQUFDLE9BQU8sRUFBRSxrQ0FBa0MsS0FBSyxDQUFDLE9BQU8sRUFBRSxDQUFDLENBQUM7WUFDdkUsT0FBTyxJQUFJLENBQUM7UUFDZCxDQUFDO0lBQ0gsQ0FBQztJQUVEOzs7T0FHRztJQUNLLEtBQUssQ0FBQyxnQkFBZ0IsQ0FBQyxNQUFvQjtRQUNqRCxrREFBa0Q7UUFDbEQsSUFBSSxDQUFDLG9CQUFvQixDQUFDLE1BQU0sQ0FBQyxTQUFTLEVBQUUsZ0JBQWdCLE1BQU0sQ0FBQyxVQUFVLEVBQUUsRUFBRSxTQUFTLENBQUMsQ0FBQztRQUU1RixrQ0FBa0M7UUFDbEMsSUFBSSxDQUFDLGlCQUFpQixDQUFDLE1BQU0sQ0FBQyxDQUFDO1FBRS9CLDRCQUE0QjtRQUM1QixNQUFNLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxNQUFNLENBQUMsQ0FBQztRQUVwQyxpQ0FBaUM7UUFDakMsTUFBTSxDQUFDLEdBQUcsQ0FBQyxNQUFNLEVBQUUsbUJBQW1CLE1BQU0sQ0FBQyxTQUFTLEtBQUssTUFBTSxDQUFDLFVBQVUsRUFBRSxFQUFFO1lBQzlFLE1BQU0sRUFBRSxNQUFNLENBQUMsTUFBTTtZQUNyQixZQUFZLEVBQUUsTUFBTSxDQUFDLFlBQVk7WUFDakMsY0FBYyxFQUFFLE1BQU0sQ0FBQyxjQUFjO1NBQ3RDLENBQUMsQ0FBQztJQUNMLENBQUM7SUFFRDs7O09BR0c7SUFDSyxLQUFLLENBQUMsZ0JBQWdCLENBQUMsTUFBb0I7UUFDakQsc0NBQXNDO1FBQ3RDLElBQUksTUFBTSxDQUFDLFVBQVUsSUFBSSxJQUFJLENBQUMsYUFBYSxDQUFDLFVBQVUsRUFBRSxDQUFDO1lBQ3ZELE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLDRCQUE0QixNQUFNLENBQUMsU0FBUywyQkFBMkIsQ0FBQyxDQUFDO1lBRTVGLDJDQUEyQztZQUMzQyxNQUFNLENBQUMsY0FBYyxHQUFHLGNBQWMsQ0FBQyxJQUFJLENBQUM7WUFDNUMsTUFBTSxJQUFJLENBQUMsZ0JBQWdCLENBQUMsTUFBTSxDQUFDLENBQUM7WUFDcEMsT0FBTztRQUNULENBQUM7UUFFRCxxREFBcUQ7UUFDckQsTUFBTSxLQUFLLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FDcEIsSUFBSSxDQUFDLGFBQWEsQ0FBQyxZQUFZLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsYUFBYSxDQUFDLGFBQWEsRUFBRSxNQUFNLENBQUMsVUFBVSxDQUFDLEVBQy9GLElBQUksQ0FBQyxhQUFhLENBQUMsUUFBUSxDQUM1QixDQUFDO1FBRUYsTUFBTSxDQUFDLFVBQVUsRUFBRSxDQUFDO1FBQ3BCLE1BQU0sQ0FBQyxhQUFhLEdBQUcsSUFBSSxDQUFDLEdBQUcsRUFBRSxHQUFHLEtBQUssQ0FBQztRQUUxQyxvREFBb0Q7UUFDcEQsSUFBSSxDQUFDLG9CQUFvQixDQUN2QixNQUFNLENBQUMsU0FBUyxFQUNoQixnQkFBZ0IsTUFBTSxDQUFDLFVBQVUsRUFBRSxFQUNuQyxNQUFNLENBQUMsYUFBYSxDQUNyQixDQUFDO1FBRUYseUJBQXlCO1FBQ3pCLE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLG1CQUFtQixNQUFNLENBQUMsVUFBVSxRQUFRLE1BQU0sQ0FBQyxTQUFTLE9BQU8sSUFBSSxJQUFJLENBQUMsTUFBTSxDQUFDLGFBQWEsQ0FBQyxDQUFDLFdBQVcsRUFBRSxFQUFFLEVBQUU7WUFDcEksVUFBVSxFQUFFLE1BQU0sQ0FBQyxVQUFVO1lBQzdCLFVBQVUsRUFBRSxNQUFNLENBQUMsVUFBVTtZQUM3QixTQUFTLEVBQUUsTUFBTSxDQUFDLGFBQWE7U0FDaEMsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztJQUVEOzs7OztPQUtHO0lBQ0ksb0JBQW9CLENBQ3pCLEtBQWEsRUFDYixNQUFjLEVBQ2QsU0FBa0I7UUFFbEIsSUFBSSxDQUFDLGVBQWUsQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLFdBQVcsRUFBRSxFQUFFO1lBQzVDLE1BQU07WUFDTixTQUFTLEVBQUUsSUFBSSxDQUFDLEdBQUcsRUFBRTtZQUNyQixTQUFTO1NBQ1YsQ0FBQyxDQUFDO1FBRUgsdUNBQXVDO1FBQ3ZDLElBQUksQ0FBQyxtQkFBbUIsRUFBRSxDQUFDLEtBQUssQ0FBQyxLQUFLLENBQUMsRUFBRTtZQUN2QyxNQUFNLENBQUMsR0FBRyxDQUFDLE9BQU8sRUFBRSxnREFBZ0QsS0FBSyxLQUFLLEtBQUssQ0FBQyxPQUFPLEVBQUUsQ0FBQyxDQUFDO1FBQ2pHLENBQUMsQ0FBQyxDQUFDO1FBRUgsTUFBTSxDQUFDLEdBQUcsQ0FBQyxNQUFNLEVBQUUsU0FBUyxLQUFLLHNCQUFzQixFQUFFO1lBQ3ZELE1BQU07WUFDTixTQUFTLEVBQUUsU0FBUyxDQUFDLENBQUMsQ0FBQyxJQUFJLElBQUksQ0FBQyxTQUFTLENBQUMsQ0FBQyxXQUFXLEVBQUUsQ0FBQyxDQUFDLENBQUMsV0FBVztTQUN2RSxDQUFDLENBQUM7SUFDTCxDQUFDO0lBRUQ7OztPQUdHO0lBQ0kseUJBQXlCLENBQUMsS0FBYTtRQUM1QyxNQUFNLFVBQVUsR0FBRyxJQUFJLENBQUMsZUFBZSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsV0FBVyxFQUFFLENBQUMsQ0FBQztRQUVwRSxJQUFJLFVBQVUsRUFBRSxDQUFDO1lBQ2YsdUNBQXVDO1lBQ3ZDLElBQUksQ0FBQyxtQkFBbUIsRUFBRSxDQUFDLEtBQUssQ0FBQyxLQUFLLENBQUMsRUFBRTtnQkFDdkMsTUFBTSxDQUFDLEdBQUcsQ0FBQyxPQUFPLEVBQUUsa0RBQWtELEtBQUssS0FBSyxLQUFLLENBQUMsT0FBTyxFQUFFLENBQUMsQ0FBQztZQUNuRyxDQUFDLENBQUMsQ0FBQztZQUNILE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLFdBQVcsS0FBSyx3QkFBd0IsQ0FBQyxDQUFDO1FBQy9ELENBQUM7SUFDSCxDQUFDO0lBRUQ7Ozs7T0FJRztJQUNJLGlCQUFpQixDQUFDLEtBQWE7UUFDcEMsTUFBTSxjQUFjLEdBQUcsS0FBSyxDQUFDLFdBQVcsRUFBRSxDQUFDO1FBQzNDLE1BQU0sV0FBVyxHQUFHLElBQUksQ0FBQyxlQUFlLENBQUMsR0FBRyxDQUFDLGNBQWMsQ0FBQyxDQUFDO1FBRTdELElBQUksQ0FBQyxXQUFXLEVBQUUsQ0FBQztZQUNqQixPQUFPLEtBQUssQ0FBQztRQUNmLENBQUM7UUFFRCxtQ0FBbUM7UUFDbkMsSUFBSSxXQUFXLENBQUMsU0FBUyxJQUFJLElBQUksQ0FBQyxHQUFHLEVBQUUsR0FBRyxXQUFXLENBQUMsU0FBUyxFQUFFLENBQUM7WUFDaEUsSUFBSSxDQUFDLGVBQWUsQ0FBQyxNQUFNLENBQUMsY0FBYyxDQUFDLENBQUM7WUFDNUMsdUNBQXVDO1lBQ3ZDLElBQUksQ0FBQyxtQkFBbUIsRUFBRSxDQUFDLEtBQUssQ0FBQyxLQUFLLENBQUMsRUFBRTtnQkFDdkMsTUFBTSxDQUFDLEdBQUcsQ0FBQyxPQUFPLEVBQUUseURBQXlELEtBQUssQ0FBQyxPQUFPLEVBQUUsQ0FBQyxDQUFDO1lBQ2hHLENBQUMsQ0FBQyxDQUFDO1lBQ0gsT0FBTyxLQUFLLENBQUM7UUFDZixDQUFDO1FBRUQsT0FBTyxJQUFJLENBQUM7SUFDZCxDQUFDO0lBRUQ7Ozs7T0FJRztJQUNJLGtCQUFrQixDQUFDLEtBQWE7UUFLckMsTUFBTSxjQUFjLEdBQUcsS0FBSyxDQUFDLFdBQVcsRUFBRSxDQUFDO1FBQzNDLE1BQU0sV0FBVyxHQUFHLElBQUksQ0FBQyxlQUFlLENBQUMsR0FBRyxDQUFDLGNBQWMsQ0FBQyxDQUFDO1FBRTdELElBQUksQ0FBQyxXQUFXLEVBQUUsQ0FBQztZQUNqQixPQUFPLElBQUksQ0FBQztRQUNkLENBQUM7UUFFRCxtQ0FBbUM7UUFDbkMsSUFBSSxXQUFXLENBQUMsU0FBUyxJQUFJLElBQUksQ0FBQyxHQUFHLEVBQUUsR0FBRyxXQUFXLENBQUMsU0FBUyxFQUFFLENBQUM7WUFDaEUsSUFBSSxDQUFDLGVBQWUsQ0FBQyxNQUFNLENBQUMsY0FBYyxDQUFDLENBQUM7WUFDNUMsdUNBQXVDO1lBQ3ZDLElBQUksQ0FBQyxtQkFBbUIsRUFBRSxDQUFDLEtBQUssQ0FBQyxLQUFLLENBQUMsRUFBRTtnQkFDdkMsTUFBTSxDQUFDLEdBQUcsQ0FBQyxPQUFPLEVBQUUseURBQXlELEtBQUssQ0FBQyxPQUFPLEVBQUUsQ0FBQyxDQUFDO1lBQ2hHLENBQUMsQ0FBQyxDQUFDO1lBQ0gsT0FBTyxJQUFJLENBQUM7UUFDZCxDQUFDO1FBRUQsT0FBTyxXQUFXLENBQUM7SUFDckIsQ0FBQztJQUVEOztPQUVHO0lBQ0ssS0FBSyxDQUFDLG1CQUFtQjtRQUMvQixJQUFJLENBQUM7WUFDSCxNQUFNLGVBQWUsR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLGVBQWUsQ0FBQyxPQUFPLEVBQUUsQ0FBQyxDQUFDLENBQUM7WUFFbkYsSUFBSSxJQUFJLENBQUMsY0FBYyxFQUFFLENBQUM7Z0JBQ3hCLHNCQUFzQjtnQkFDdEIsTUFBTSxJQUFJLENBQUMsY0FBYyxDQUFDLEdBQUcsQ0FBQyxzQ0FBc0MsRUFBRSxlQUFlLENBQUMsQ0FBQztZQUN6RixDQUFDO2lCQUFNLENBQUM7Z0JBQ04sMEJBQTBCO2dCQUMxQixNQUFNLE9BQU8sQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUN4QixPQUFPLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTyxFQUFFLFFBQVEsRUFBRSx1QkFBdUIsQ0FBQyxDQUNwRSxDQUFDLEtBQUssQ0FBQyxlQUFlLENBQUMsQ0FBQztZQUMzQixDQUFDO1FBQ0gsQ0FBQztRQUFDLE9BQU8sS0FBSyxFQUFFLENBQUM7WUFDZixNQUFNLENBQUMsR0FBRyxDQUFDLE9BQU8sRUFBRSxvQ0FBb0MsS0FBSyxDQUFDLE9BQU8sRUFBRSxDQUFDLENBQUM7UUFDM0UsQ0FBQztJQUNILENBQUM7SUFFRDs7T0FFRztJQUNLLEtBQUssQ0FBQyxtQkFBbUI7UUFDL0IsSUFBSSxDQUFDO1lBQ0gsSUFBSSxPQUFPLEdBQUcsSUFBSSxDQUFDO1lBQ25CLElBQUksY0FBYyxHQUFHLEtBQUssQ0FBQztZQUUzQixJQUFJLElBQUksQ0FBQyxjQUFjLEVBQUUsQ0FBQztnQkFDeEIseUNBQXlDO2dCQUN6QyxNQUFNLGVBQWUsR0FBRyxNQUFNLElBQUksQ0FBQyxjQUFjLENBQUMsR0FBRyxDQUFDLHNDQUFzQyxDQUFDLENBQUM7Z0JBRTlGLElBQUksZUFBZSxFQUFFLENBQUM7b0JBQ3BCLE9BQU8sR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLGVBQWUsQ0FBQyxDQUFDO2dCQUN4QyxDQUFDO3FCQUFNLENBQUM7b0JBQ04saURBQWlEO29CQUNqRCxNQUFNLGVBQWUsR0FBRyxPQUFPLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTyxFQUFFLFFBQVEsRUFBRSx1QkFBdUIsQ0FBQyxDQUFDO29CQUU1RixJQUFJLE9BQU8sQ0FBQyxFQUFFLENBQUMsVUFBVSxDQUFDLGVBQWUsQ0FBQyxFQUFFLENBQUM7d0JBQzNDLE1BQU0sSUFBSSxHQUFHLE9BQU8sQ0FBQyxFQUFFLENBQUMsWUFBWSxDQUFDLGVBQWUsRUFBRSxNQUFNLENBQUMsQ0FBQzt3QkFDOUQsT0FBTyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUM7d0JBQzNCLGNBQWMsR0FBRyxJQUFJLENBQUM7d0JBRXRCLE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLDhEQUE4RCxDQUFDLENBQUM7b0JBQ3JGLENBQUM7Z0JBQ0gsQ0FBQztZQUNILENBQUM7aUJBQU0sQ0FBQztnQkFDTiw4Q0FBOEM7Z0JBQzlDLE1BQU0sZUFBZSxHQUFHLE9BQU8sQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLEVBQUUsUUFBUSxFQUFFLHVCQUF1QixDQUFDLENBQUM7Z0JBRTVGLElBQUksT0FBTyxDQUFDLEVBQUUsQ0FBQyxVQUFVLENBQUMsZUFBZSxDQUFDLEVBQUUsQ0FBQztvQkFDM0MsTUFBTSxJQUFJLEdBQUcsT0FBTyxDQUFDLEVBQUUsQ0FBQyxZQUFZLENBQUMsZUFBZSxFQUFFLE1BQU0sQ0FBQyxDQUFDO29CQUM5RCxPQUFPLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQztnQkFDN0IsQ0FBQztZQUNILENBQUM7WUFFRCxJQUFJLE9BQU8sRUFBRSxDQUFDO2dCQUNaLElBQUksQ0FBQyxlQUFlLEdBQUcsSUFBSSxHQUFHLENBQUMsT0FBTyxDQUFDLENBQUM7Z0JBRXhDLHdCQUF3QjtnQkFDeEIsTUFBTSxHQUFHLEdBQUcsSUFBSSxDQUFDLEdBQUcsRUFBRSxDQUFDO2dCQUN2QixJQUFJLFlBQVksR0FBRyxDQUFDLENBQUM7Z0JBRXJCLEtBQUssTUFBTSxDQUFDLEtBQUssRUFBRSxJQUFJLENBQUMsSUFBSSxJQUFJLENBQUMsZUFBZSxDQUFDLE9BQU8sRUFBRSxFQUFFLENBQUM7b0JBQzNELElBQUksSUFBSSxDQUFDLFNBQVMsSUFBSSxHQUFHLEdBQUcsSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDO3dCQUMzQyxJQUFJLENBQUMsZUFBZSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQzt3QkFDbkMsWUFBWSxFQUFFLENBQUM7b0JBQ2pCLENBQUM7Z0JBQ0gsQ0FBQztnQkFFRCxJQUFJLFlBQVksR0FBRyxDQUFDLElBQUksY0FBYyxFQUFFLENBQUM7b0JBQ3ZDLE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLFdBQVcsWUFBWSx3Q0FBd0MsQ0FBQyxDQUFDO29CQUNwRixNQUFNLElBQUksQ0FBQyxtQkFBbUIsRUFBRSxDQUFDO2dCQUNuQyxDQUFDO2dCQUVELE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLFVBQVUsSUFBSSxDQUFDLGVBQWUsQ0FBQyxJQUFJLGdDQUFnQyxDQUFDLENBQUM7WUFDMUYsQ0FBQztRQUNILENBQUM7UUFBQyxPQUFPLEtBQUssRUFBRSxDQUFDO1lBQ2YsTUFBTSxDQUFDLEdBQUcsQ0FBQyxPQUFPLEVBQUUsb0NBQW9DLEtBQUssQ0FBQyxPQUFPLEVBQUUsQ0FBQyxDQUFDO1FBQzNFLENBQUM7SUFDSCxDQUFDO0lBRUQ7OztPQUdHO0lBQ0ssS0FBSyxDQUFDLGdCQUFnQixDQUFDLE1BQW9CO1FBQ2pELElBQUksQ0FBQztZQUNILE1BQU0sVUFBVSxHQUFHLElBQUksQ0FBQyxTQUFTLENBQUMsTUFBTSxFQUFFLElBQUksRUFBRSxDQUFDLENBQUMsQ0FBQztZQUVuRCxJQUFJLElBQUksQ0FBQyxjQUFjLEVBQUUsQ0FBQztnQkFDeEIsc0JBQXNCO2dCQUN0QixNQUFNLElBQUksQ0FBQyxjQUFjLENBQUMsR0FBRyxDQUFDLDBCQUEwQixNQUFNLENBQUMsRUFBRSxPQUFPLEVBQUUsVUFBVSxDQUFDLENBQUM7WUFDeEYsQ0FBQztpQkFBTSxDQUFDO2dCQUNOLDBCQUEwQjtnQkFDMUIsTUFBTSxVQUFVLEdBQUcsT0FBTyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQ2xDLEtBQUssQ0FBQyxPQUFPLEVBQ2IsUUFBUSxFQUNSLFNBQVMsRUFDVCxHQUFHLE1BQU0sQ0FBQyxFQUFFLE9BQU8sQ0FDcEIsQ0FBQztnQkFFRiwwQkFBMEI7Z0JBQzFCLE1BQU0sU0FBUyxHQUFHLE9BQU8sQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLEVBQUUsUUFBUSxFQUFFLFNBQVMsQ0FBQyxDQUFDO2dCQUN4RSxNQUFNLE9BQU8sQ0FBQyxPQUFPLENBQUMsU0FBUyxDQUFDLFNBQVMsQ0FBQyxDQUFDLFNBQVMsRUFBRSxDQUFDLE1BQU0sRUFBRSxDQUFDO2dCQUVoRSxNQUFNLE9BQU8sQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxDQUFDLEtBQUssQ0FBQyxVQUFVLENBQUMsQ0FBQztZQUMzRCxDQUFDO1FBQ0gsQ0FBQztRQUFDLE9BQU8sS0FBSyxFQUFFLENBQUM7WUFDZixNQUFNLENBQUMsR0FBRyxDQUFDLE9BQU8sRUFBRSxpQ0FBaUMsS0FBSyxDQUFDLE9BQU8sRUFBRSxDQUFDLENBQUM7UUFDeEUsQ0FBQztJQUNILENBQUM7SUFFRDs7O09BR0c7SUFDSyxpQkFBaUIsQ0FBQyxNQUFvQjtRQUM1QyxNQUFNLEtBQUssR0FBRyxNQUFNLENBQUMsU0FBUyxDQUFDLFdBQVcsRUFBRSxDQUFDO1FBQzdDLE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQyxXQUFXLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBRTdDLElBQUksUUFBUSxFQUFFLENBQUM7WUFDYiw4QkFBOEI7WUFDOUIsUUFBUSxDQUFDLFVBQVUsR0FBRyxNQUFNLENBQUMsU0FBUyxDQUFDO1lBQ3ZDLFFBQVEsQ0FBQyxLQUFLLEVBQUUsQ0FBQztZQUNqQixRQUFRLENBQUMsSUFBSSxHQUFHLE1BQU0sQ0FBQyxVQUFVLENBQUM7WUFDbEMsUUFBUSxDQUFDLFFBQVEsR0FBRyxNQUFNLENBQUMsY0FBYyxDQUFDO1FBQzVDLENBQUM7YUFBTSxDQUFDO1lBQ04seUJBQXlCO1lBQ3pCLElBQUksQ0FBQyxXQUFXLENBQUMsR0FBRyxDQUFDLEtBQUssRUFBRTtnQkFDMUIsVUFBVSxFQUFFLE1BQU0sQ0FBQyxTQUFTO2dCQUM1QixLQUFLLEVBQUUsQ0FBQztnQkFDUixJQUFJLEVBQUUsTUFBTSxDQUFDLFVBQVU7Z0JBQ3ZCLFFBQVEsRUFBRSxNQUFNLENBQUMsY0FBYzthQUNoQyxDQUFDLENBQUM7UUFDTCxDQUFDO0lBQ0gsQ0FBQztJQUVEOzs7O09BSUc7SUFDSSxhQUFhLENBQUMsS0FBYTtRQU1oQyxPQUFPLElBQUksQ0FBQyxXQUFXLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxXQUFXLEVBQUUsQ0FBQyxJQUFJLElBQUksQ0FBQztJQUMzRCxDQUFDO0lBRUQ7OztPQUdHO0lBQ0ksdUJBQXVCO1FBQzVCLE1BQU0sV0FBVyxHQUFhLEVBQUUsQ0FBQztRQUVqQyxLQUFLLE1BQU0sQ0FBQyxLQUFLLEVBQUUsSUFBSSxDQUFDLElBQUksSUFBSSxDQUFDLFdBQVcsQ0FBQyxPQUFPLEVBQUUsRUFBRSxDQUFDO1lBQ3ZELElBQUksSUFBSSxDQUFDLFFBQVEsS0FBSyxjQUFjLENBQUMsSUFBSSxFQUFFLENBQUM7Z0JBQzFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7WUFDMUIsQ0FBQztRQUNILENBQUM7UUFFRCxPQUFPLFdBQVcsQ0FBQztJQUNyQixDQUFDO0lBRUQ7OztPQUdHO0lBQ0ksa0JBQWtCO1FBQ3ZCLE9BQU8sS0FBSyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsZUFBZSxDQUFDLElBQUksRUFBRSxDQUFDLENBQUM7SUFDakQsQ0FBQztJQUVEOzs7O09BSUc7SUFDSSxxQkFBcUIsQ0FBQyxTQUFpQjtRQUM1QyxJQUFJLE9BQU8sR0FBRyxDQUFDLENBQUM7UUFFaEIsSUFBSSxDQUFDLFdBQVcsR0FBRyxJQUFJLENBQUMsV0FBVyxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsRUFBRTtZQUNsRCxJQUFJLE1BQU0sQ0FBQyxTQUFTLEdBQUcsU0FBUyxFQUFFLENBQUM7Z0JBQ2pDLE9BQU8sRUFBRSxDQUFDO2dCQUNWLE9BQU8sS0FBSyxDQUFDO1lBQ2YsQ0FBQztZQUNELE9BQU8sSUFBSSxDQUFDO1FBQ2QsQ0FBQyxDQUFDLENBQUM7UUFFSCxPQUFPLE9BQU8sQ0FBQztJQUNqQixDQUFDO0NBQ0YifQ== \ No newline at end of file diff --git a/dist_ts/mail/core/classes.email.d.ts b/dist_ts/mail/core/classes.email.d.ts deleted file mode 100644 index 713de9e..0000000 --- a/dist_ts/mail/core/classes.email.d.ts +++ /dev/null @@ -1,291 +0,0 @@ -import * as plugins from '../../plugins.js'; -export interface IAttachment { - filename: string; - content: Buffer; - contentType: string; - contentId?: string; - encoding?: string; -} -export interface IEmailOptions { - from: string; - to?: string | string[]; - cc?: string | string[]; - bcc?: string | string[]; - subject: string; - text: string; - html?: string; - attachments?: IAttachment[]; - headers?: Record; - mightBeSpam?: boolean; - priority?: 'high' | 'normal' | 'low'; - skipAdvancedValidation?: boolean; - variables?: Record; -} -/** - * 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 declare class Email { - from: string; - to: string[]; - cc: string[]; - bcc: string[]; - subject: string; - text: string; - html?: string; - attachments: IAttachment[]; - headers: Record; - mightBeSpam: boolean; - priority: 'high' | 'normal' | 'low'; - variables: Record; - private envelopeFrom; - private messageId; - private static emailValidator; - constructor(options: IEmailOptions); - /** - * 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 - */ - private isValidEmail; - /** - * 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 - */ - private extractEmailAddress; - /** - * Parses and validates recipient email addresses - * @param recipients A string or array of recipient emails - * @returns Array of validated email addresses - */ - private parseRecipients; - /** - * Basic sanitization for strings to prevent header injection - * @param input The string to sanitize - * @returns Sanitized string - */ - private sanitizeString; - /** - * Gets the domain part of the from email address - * @returns The domain part of the from email or null if invalid - */ - getFromDomain(): string | null; - /** - * Gets the clean from email address without display name - * @returns The email address without display name - */ - getFromAddress(): string; - /** - * Converts IDN (International Domain Names) to ASCII - * @param email The email address to convert - * @returns The email with ASCII-converted domain - */ - private convertIDNToASCII; - /** - * Gets clean to email addresses without display names - * @returns Array of email addresses without display names - */ - getToAddresses(): string[]; - /** - * Gets clean cc email addresses without display names - * @returns Array of email addresses without display names - */ - getCcAddresses(): string[]; - /** - * Gets clean bcc email addresses without display names - * @returns Array of email addresses without display names - */ - getBccAddresses(): string[]; - /** - * Gets all recipients (to, cc, bcc) as a unique array - * @returns Array of all unique recipient email addresses - */ - getAllRecipients(): string[]; - /** - * Gets primary recipient (first in the to field) - * @returns The primary recipient email or null if none exists - */ - getPrimaryRecipient(): string | null; - /** - * Checks if the email has attachments - * @returns Boolean indicating if the email has attachments - */ - hasAttachments(): boolean; - /** - * 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: string, type?: 'to' | 'cc' | 'bcc'): this; - /** - * Add an attachment to the email - * @param attachment The attachment to add - * @returns This instance for method chaining - */ - addAttachment(attachment: IAttachment): 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: string, value: string): this; - /** - * Set the email priority - * @param priority The priority level - * @returns This instance for method chaining - */ - setPriority(priority: 'high' | 'normal' | 'low'): this; - /** - * Set a template variable - * @param key The variable key - * @param value The variable value - * @returns This instance for method chaining - */ - setVariable(key: string, value: any): this; - /** - * Set multiple template variables at once - * @param variables The variables object - * @returns This instance for method chaining - */ - setVariables(variables: Record): this; - /** - * Get the subject with variables applied - * @param variables Optional additional variables to apply - * @returns The processed subject - */ - getSubjectWithVariables(variables?: Record): string; - /** - * Get the text content with variables applied - * @param variables Optional additional variables to apply - * @returns The processed text content - */ - getTextWithVariables(variables?: Record): string; - /** - * 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?: Record): string | undefined; - /** - * Apply template variables to a string - * @param template The template string - * @param additionalVariables Optional additional variables to apply - * @returns The processed string - */ - private applyVariables; - /** - * Gets the total size of all attachments in bytes - * @returns Total size of all attachments in bytes - */ - getAttachmentsSize(): number; - /** - * 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 - */ - validateAddresses(options?: { - checkMx?: boolean; - checkDisposable?: boolean; - checkSenderOnly?: boolean; - checkFirstRecipientOnly?: boolean; - }): Promise<{ - sender: { - email: string; - result: any; - }; - recipients: Array<{ - email: string; - result: any; - }>; - isValid: boolean; - }>; - /** - * Convert this email to a smartmail instance - * @returns A new Smartmail instance - */ - toSmartmail(): Promise>; - /** - * Get the from email address - * @returns The from email address - */ - getFromEmail(): string; - /** - * Get the subject (Smartmail compatibility method) - * @returns The email subject - */ - getSubject(): string; - /** - * 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?: boolean): string; - /** - * Get the from address (Smartmail compatibility method) - * @returns The sender email address - */ - getFrom(): string; - /** - * Get the message ID - * @returns The message ID - */ - getMessageId(): string; - /** - * 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(): IEmailOptions; - /** - * Set a custom message ID - * @param id The message ID to set - * @returns This instance for method chaining - */ - setMessageId(id: string): this; - /** - * Get the envelope from address (return-path) - * @returns The envelope from address - */ - getEnvelopeFrom(): string; - /** - * Set the envelope from address (return-path) - * @param address The envelope from address to set - * @returns This instance for method chaining - */ - setEnvelopeFrom(address: string): 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?: Record): string; - /** - * Convert to simple Smartmail-compatible object (for backward compatibility) - * @returns A Promise with a simple Smartmail-compatible object - */ - toSmartmailBasic(): Promise; - /** - * Create an Email instance from a Smartmail object - * @param smartmail The Smartmail instance to convert - * @returns A new Email instance - */ - static fromSmartmail(smartmail: plugins.smartmail.Smartmail): Email; -} diff --git a/dist_ts/mail/core/classes.email.js b/dist_ts/mail/core/classes.email.js deleted file mode 100644 index fadf057..0000000 --- a/dist_ts/mail/core/classes.email.js +++ /dev/null @@ -1,802 +0,0 @@ -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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 \ No newline at end of file diff --git a/dist_ts/mail/core/classes.emailvalidator.d.ts b/dist_ts/mail/core/classes.emailvalidator.d.ts deleted file mode 100644 index 08138f8..0000000 --- a/dist_ts/mail/core/classes.emailvalidator.d.ts +++ /dev/null @@ -1,61 +0,0 @@ -export interface IEmailValidationResult { - isValid: boolean; - hasMx: boolean; - hasSpamMarkings: boolean; - score: number; - details?: { - formatValid?: boolean; - mxRecords?: string[]; - disposable?: boolean; - role?: boolean; - spamIndicators?: string[]; - errorMessage?: string; - }; -} -/** - * Advanced email validator class using smartmail's capabilities - */ -export declare class EmailValidator { - private validator; - private dnsCache; - constructor(options?: { - maxCacheSize?: number; - cacheTTL?: number; - }); - /** - * Validates an email address using comprehensive checks - * @param email The email to validate - * @param options Validation options - * @returns Validation result with details - */ - validate(email: string, options?: { - checkMx?: boolean; - checkDisposable?: boolean; - checkRole?: boolean; - checkSyntaxOnly?: boolean; - }): Promise; - /** - * Gets MX records for a domain with caching - * @param domain Domain to check - * @returns Array of MX records - */ - private getMxRecords; - /** - * Validates multiple email addresses in batch - * @param emails Array of emails to validate - * @param options Validation options - * @returns Object with email addresses as keys and validation results as values - */ - validateBatch(emails: string[], options?: { - checkMx?: boolean; - checkDisposable?: boolean; - checkRole?: boolean; - checkSyntaxOnly?: boolean; - }): Promise>; - /** - * Quick check if an email format is valid (synchronous, no DNS checks) - * @param email Email to check - * @returns Boolean indicating if format is valid - */ - isValidFormat(email: string): boolean; -} diff --git a/dist_ts/mail/core/classes.emailvalidator.js b/dist_ts/mail/core/classes.emailvalidator.js deleted file mode 100644 index db273f4..0000000 --- a/dist_ts/mail/core/classes.emailvalidator.js +++ /dev/null @@ -1,184 +0,0 @@ -import * as plugins from '../../plugins.js'; -import { logger } from '../../logger.js'; -import { LRUCache } from 'lru-cache'; -/** - * Advanced email validator class using smartmail's capabilities - */ -export class EmailValidator { - validator; - dnsCache; - constructor(options) { - this.validator = new plugins.smartmail.EmailAddressValidator(); - // Initialize LRU cache for DNS records - this.dnsCache = new LRUCache({ - // Default to 1000 entries (reasonable for most applications) - max: options?.maxCacheSize || 1000, - // Default TTL of 1 hour (DNS records don't change frequently) - ttl: options?.cacheTTL || 60 * 60 * 1000, - // Optional cache monitoring - allowStale: false, - updateAgeOnGet: true, - // Add logging for cache events in production environments - disposeAfter: (value, key) => { - logger.log('debug', `DNS cache entry expired for domain: ${key}`); - }, - }); - } - /** - * Validates an email address using comprehensive checks - * @param email The email to validate - * @param options Validation options - * @returns Validation result with details - */ - async validate(email, options = {}) { - try { - const result = { - isValid: false, - hasMx: false, - hasSpamMarkings: false, - score: 0, - details: { - formatValid: false, - spamIndicators: [] - } - }; - // Always check basic format - result.details.formatValid = this.validator.isValidEmailFormat(email); - if (!result.details.formatValid) { - result.details.errorMessage = 'Invalid email format'; - return result; - } - // If syntax-only check is requested, return early - if (options.checkSyntaxOnly) { - result.isValid = true; - result.score = 0.5; - return result; - } - // Get domain for additional checks - const domain = email.split('@')[1]; - // Check MX records - if (options.checkMx !== false) { - try { - const mxRecords = await this.getMxRecords(domain); - result.details.mxRecords = mxRecords; - result.hasMx = mxRecords && mxRecords.length > 0; - if (!result.hasMx) { - result.details.spamIndicators.push('No MX records'); - result.details.errorMessage = 'Domain has no MX records'; - } - } - catch (error) { - logger.log('error', `Error checking MX records: ${error.message}`); - result.details.errorMessage = 'Unable to check MX records'; - } - } - // Check if domain is disposable - if (options.checkDisposable !== false) { - result.details.disposable = await this.validator.isDisposableEmail(email); - if (result.details.disposable) { - result.details.spamIndicators.push('Disposable email'); - } - } - // Check if email is a role account - if (options.checkRole !== false) { - result.details.role = this.validator.isRoleAccount(email); - if (result.details.role) { - result.details.spamIndicators.push('Role account'); - } - } - // Calculate spam score and final validity - result.hasSpamMarkings = result.details.spamIndicators.length > 0; - // Calculate a score between 0-1 based on checks - let scoreFactors = 0; - let scoreTotal = 0; - // Format check (highest weight) - scoreFactors += 0.4; - if (result.details.formatValid) - scoreTotal += 0.4; - // MX check (high weight) - if (options.checkMx !== false) { - scoreFactors += 0.3; - if (result.hasMx) - scoreTotal += 0.3; - } - // Disposable check (medium weight) - if (options.checkDisposable !== false) { - scoreFactors += 0.2; - if (!result.details.disposable) - scoreTotal += 0.2; - } - // Role account check (low weight) - if (options.checkRole !== false) { - scoreFactors += 0.1; - if (!result.details.role) - scoreTotal += 0.1; - } - // Normalize score based on factors actually checked - result.score = scoreFactors > 0 ? scoreTotal / scoreFactors : 0; - // Email is valid if score is above 0.7 (configurable threshold) - result.isValid = result.score >= 0.7; - return result; - } - catch (error) { - logger.log('error', `Email validation error: ${error.message}`); - return { - isValid: false, - hasMx: false, - hasSpamMarkings: true, - score: 0, - details: { - formatValid: false, - errorMessage: `Validation error: ${error.message}`, - spamIndicators: ['Validation error'] - } - }; - } - } - /** - * Gets MX records for a domain with caching - * @param domain Domain to check - * @returns Array of MX records - */ - async getMxRecords(domain) { - // Check cache first - const cachedRecords = this.dnsCache.get(domain); - if (cachedRecords) { - logger.log('debug', `Using cached MX records for domain: ${domain}`); - return cachedRecords; - } - try { - // Use smartmail's getMxRecords method - const records = await this.validator.getMxRecords(domain); - // Store in cache (TTL is handled by the LRU cache configuration) - this.dnsCache.set(domain, records); - logger.log('debug', `Cached MX records for domain: ${domain}`); - return records; - } - catch (error) { - logger.log('error', `Error fetching MX records for ${domain}: ${error.message}`); - return []; - } - } - /** - * Validates multiple email addresses in batch - * @param emails Array of emails to validate - * @param options Validation options - * @returns Object with email addresses as keys and validation results as values - */ - async validateBatch(emails, options = {}) { - const results = {}; - for (const email of emails) { - results[email] = await this.validate(email, options); - } - return results; - } - /** - * Quick check if an email format is valid (synchronous, no DNS checks) - * @param email Email to check - * @returns Boolean indicating if format is valid - */ - isValidFormat(email) { - return this.validator.isValidEmailFormat(email); - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5lbWFpbHZhbGlkYXRvci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3RzL21haWwvY29yZS9jbGFzc2VzLmVtYWlsdmFsaWRhdG9yLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sS0FBSyxPQUFPLE1BQU0sa0JBQWtCLENBQUM7QUFDNUMsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQ3pDLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxXQUFXLENBQUM7QUFpQnJDOztHQUVHO0FBQ0gsTUFBTSxPQUFPLGNBQWM7SUFDakIsU0FBUyxDQUEwQztJQUNuRCxRQUFRLENBQTZCO0lBRTdDLFlBQVksT0FHWDtRQUNDLElBQUksQ0FBQyxTQUFTLEdBQUcsSUFBSSxPQUFPLENBQUMsU0FBUyxDQUFDLHFCQUFxQixFQUFFLENBQUM7UUFFL0QsdUNBQXVDO1FBQ3ZDLElBQUksQ0FBQyxRQUFRLEdBQUcsSUFBSSxRQUFRLENBQW1CO1lBQzdDLDZEQUE2RDtZQUM3RCxHQUFHLEVBQUUsT0FBTyxFQUFFLFlBQVksSUFBSSxJQUFJO1lBQ2xDLDhEQUE4RDtZQUM5RCxHQUFHLEVBQUUsT0FBTyxFQUFFLFFBQVEsSUFBSSxFQUFFLEdBQUcsRUFBRSxHQUFHLElBQUk7WUFDeEMsNEJBQTRCO1lBQzVCLFVBQVUsRUFBRSxLQUFLO1lBQ2pCLGNBQWMsRUFBRSxJQUFJO1lBQ3BCLDBEQUEwRDtZQUMxRCxZQUFZLEVBQUUsQ0FBQyxLQUFLLEVBQUUsR0FBRyxFQUFFLEVBQUU7Z0JBQzNCLE1BQU0sQ0FBQyxHQUFHLENBQUMsT0FBTyxFQUFFLHVDQUF1QyxHQUFHLEVBQUUsQ0FBQyxDQUFDO1lBQ3BFLENBQUM7U0FDRixDQUFDLENBQUM7SUFDTCxDQUFDO0lBRUQ7Ozs7O09BS0c7SUFDSSxLQUFLLENBQUMsUUFBUSxDQUNuQixLQUFhLEVBQ2IsVUFLSSxFQUFFO1FBRU4sSUFBSSxDQUFDO1lBQ0gsTUFB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 \ No newline at end of file diff --git a/dist_ts/mail/core/classes.templatemanager.d.ts b/dist_ts/mail/core/classes.templatemanager.d.ts deleted file mode 100644 index 14a0d1a..0000000 --- a/dist_ts/mail/core/classes.templatemanager.d.ts +++ /dev/null @@ -1,95 +0,0 @@ -import { Email } from './classes.email.js'; -/** - * Email template type definition - */ -export interface IEmailTemplate { - id: string; - name: string; - description: string; - from: string; - subject: string; - bodyHtml: string; - bodyText?: string; - category?: string; - sampleData?: T; - attachments?: Array<{ - name: string; - path: string; - contentType?: string; - }>; -} -/** - * Email template context - data used to render the template - */ -export interface ITemplateContext { - [key: string]: any; -} -/** - * Template category definitions - */ -export declare enum TemplateCategory { - NOTIFICATION = "notification", - TRANSACTIONAL = "transactional", - MARKETING = "marketing", - SYSTEM = "system" -} -/** - * Enhanced template manager using Email class for template rendering - */ -export declare class TemplateManager { - private templates; - private defaultConfig; - constructor(defaultConfig?: { - from?: string; - replyTo?: string; - footerHtml?: string; - footerText?: string; - }); - /** - * Register built-in email templates - */ - private registerBuiltinTemplates; - /** - * Register a new email template - * @param template The email template to register - */ - registerTemplate(template: IEmailTemplate): void; - /** - * Get an email template by ID - * @param templateId The template ID - * @returns The template or undefined if not found - */ - getTemplate(templateId: string): IEmailTemplate | undefined; - /** - * List all available templates - * @param category Optional category filter - * @returns Array of email templates - */ - listTemplates(category?: TemplateCategory): IEmailTemplate[]; - /** - * Create an Email instance from a template - * @param templateId The template ID - * @param context The template context data - * @returns A configured Email instance - */ - createEmail(templateId: string, context?: ITemplateContext): Promise; - /** - * Create and completely process an Email instance from a template - * @param templateId The template ID - * @param context The template context data - * @returns A complete, processed Email instance ready to send - */ - prepareEmail(templateId: string, context?: ITemplateContext): Promise; - /** - * Create a MIME-formatted email from a template - * @param templateId The template ID - * @param context The template context data - * @returns A MIME-formatted email string - */ - createMimeEmail(templateId: string, context?: ITemplateContext): Promise; - /** - * Load templates from a directory - * @param directory The directory containing template JSON files - */ - loadTemplatesFromDirectory(directory: string): Promise; -} diff --git a/dist_ts/mail/core/classes.templatemanager.js b/dist_ts/mail/core/classes.templatemanager.js deleted file mode 100644 index 21748b3..0000000 --- a/dist_ts/mail/core/classes.templatemanager.js +++ /dev/null @@ -1,240 +0,0 @@ -import * as plugins from '../../plugins.js'; -import * as paths from '../../paths.js'; -import { logger } from '../../logger.js'; -import { Email } from './classes.email.js'; -/** - * Template category definitions - */ -export var TemplateCategory; -(function (TemplateCategory) { - TemplateCategory["NOTIFICATION"] = "notification"; - TemplateCategory["TRANSACTIONAL"] = "transactional"; - TemplateCategory["MARKETING"] = "marketing"; - TemplateCategory["SYSTEM"] = "system"; -})(TemplateCategory || (TemplateCategory = {})); -/** - * Enhanced template manager using Email class for template rendering - */ -export class TemplateManager { - templates = new Map(); - defaultConfig; - constructor(defaultConfig) { - // Set default configuration - this.defaultConfig = { - from: defaultConfig?.from || 'noreply@mail.lossless.com', - replyTo: defaultConfig?.replyTo, - footerHtml: defaultConfig?.footerHtml || '', - footerText: defaultConfig?.footerText || '' - }; - // Initialize with built-in templates - this.registerBuiltinTemplates(); - } - /** - * Register built-in email templates - */ - registerBuiltinTemplates() { - // Welcome email - this.registerTemplate({ - id: 'welcome', - name: 'Welcome Email', - description: 'Sent to users when they first sign up', - from: this.defaultConfig.from, - subject: 'Welcome to {{serviceName}}!', - category: TemplateCategory.TRANSACTIONAL, - bodyHtml: ` -

Welcome, {{firstName}}!

-

Thank you for joining {{serviceName}}. We're excited to have you on board.

-

To get started, visit your account.

- `, - bodyText: `Welcome, {{firstName}}! - - Thank you for joining {{serviceName}}. We're excited to have you on board. - - To get started, visit your account: {{accountUrl}} - `, - sampleData: { - firstName: 'John', - accountUrl: 'https://example.com/account' - } - }); - // Password reset - this.registerTemplate({ - id: 'password-reset', - name: 'Password Reset', - description: 'Sent when a user requests a password reset', - from: this.defaultConfig.from, - subject: 'Password Reset Request', - category: TemplateCategory.TRANSACTIONAL, - bodyHtml: ` -

Password Reset Request

-

You recently requested to reset your password. Click the link below to reset it:

-

Reset Password

-

This link will expire in {{expiryHours}} hours.

-

If you didn't request a password reset, please ignore this email.

- `, - sampleData: { - resetUrl: 'https://example.com/reset-password?token=abc123', - expiryHours: 24 - } - }); - // System notification - this.registerTemplate({ - id: 'system-notification', - name: 'System Notification', - description: 'General system notification template', - from: this.defaultConfig.from, - subject: '{{subject}}', - category: TemplateCategory.SYSTEM, - bodyHtml: ` -

{{title}}

-
{{message}}
- `, - sampleData: { - subject: 'Important System Notification', - title: 'System Maintenance', - message: 'The system will be undergoing maintenance on Saturday from 2-4am UTC.' - } - }); - } - /** - * Register a new email template - * @param template The email template to register - */ - registerTemplate(template) { - if (this.templates.has(template.id)) { - logger.log('warn', `Template with ID '${template.id}' already exists and will be overwritten`); - } - // Add footer to templates if configured - if (this.defaultConfig.footerHtml && template.bodyHtml) { - template.bodyHtml += this.defaultConfig.footerHtml; - } - if (this.defaultConfig.footerText && template.bodyText) { - template.bodyText += this.defaultConfig.footerText; - } - this.templates.set(template.id, template); - logger.log('info', `Registered email template: ${template.id}`); - } - /** - * Get an email template by ID - * @param templateId The template ID - * @returns The template or undefined if not found - */ - getTemplate(templateId) { - return this.templates.get(templateId); - } - /** - * List all available templates - * @param category Optional category filter - * @returns Array of email templates - */ - listTemplates(category) { - const templates = Array.from(this.templates.values()); - if (category) { - return templates.filter(template => template.category === category); - } - return templates; - } - /** - * Create an Email instance from a template - * @param templateId The template ID - * @param context The template context data - * @returns A configured Email instance - */ - async createEmail(templateId, context) { - const template = this.getTemplate(templateId); - if (!template) { - throw new Error(`Template with ID '${templateId}' not found`); - } - // Build attachments array for Email - const attachments = []; - if (template.attachments && template.attachments.length > 0) { - for (const attachment of template.attachments) { - try { - const attachmentPath = plugins.path.isAbsolute(attachment.path) - ? attachment.path - : plugins.path.join(paths.MtaAttachmentsDir, attachment.path); - // Read the file - const fileBuffer = await plugins.fs.promises.readFile(attachmentPath); - attachments.push({ - filename: attachment.name, - content: fileBuffer, - contentType: attachment.contentType || 'application/octet-stream' - }); - } - catch (error) { - logger.log('error', `Failed to add attachment '${attachment.name}': ${error.message}`); - } - } - } - // Create Email instance with template content - const emailOptions = { - from: template.from || this.defaultConfig.from, - subject: template.subject, - text: template.bodyText || '', - html: template.bodyHtml, - // Note: 'to' is intentionally omitted for templates - attachments, - variables: context || {} - }; - return new Email(emailOptions); - } - /** - * Create and completely process an Email instance from a template - * @param templateId The template ID - * @param context The template context data - * @returns A complete, processed Email instance ready to send - */ - async prepareEmail(templateId, context = {}) { - const email = await this.createEmail(templateId, context); - // Email class processes variables when needed, no pre-compilation required - return email; - } - /** - * Create a MIME-formatted email from a template - * @param templateId The template ID - * @param context The template context data - * @returns A MIME-formatted email string - */ - async createMimeEmail(templateId, context = {}) { - const email = await this.prepareEmail(templateId, context); - return email.toRFC822String(context); - } - /** - * Load templates from a directory - * @param directory The directory containing template JSON files - */ - async loadTemplatesFromDirectory(directory) { - try { - // Ensure directory exists - if (!plugins.fs.existsSync(directory)) { - logger.log('error', `Template directory does not exist: ${directory}`); - return; - } - // Get all JSON files - const files = plugins.fs.readdirSync(directory) - .filter(file => file.endsWith('.json')); - for (const file of files) { - try { - const filePath = plugins.path.join(directory, file); - const content = plugins.fs.readFileSync(filePath, 'utf8'); - const template = JSON.parse(content); - // Validate template - if (!template.id || !template.subject || (!template.bodyHtml && !template.bodyText)) { - logger.log('warn', `Invalid template in ${file}: missing required fields`); - continue; - } - this.registerTemplate(template); - } - catch (error) { - logger.log('error', `Error loading template from ${file}: ${error.message}`); - } - } - logger.log('info', `Loaded ${this.templates.size} email templates`); - } - catch (error) { - logger.log('error', `Failed to load templates from directory: ${error.message}`); - throw error; - } - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy50ZW1wbGF0ZW1hbmFnZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi90cy9tYWlsL2NvcmUvY2xhc3Nlcy50ZW1wbGF0ZW1hbmFnZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxLQUFLLE9BQU8sTUFBTSxrQkFBa0IsQ0FBQztBQUM1QyxPQUFPLEtBQUssS0FBSyxNQUFNLGdCQUFnQixDQUFDO0FBQ3hDLE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUN6QyxPQUFPLEVBQUUsS0FBSyxFQUF3QyxNQUFNLG9CQUFvQixDQUFDO0FBNkJqRjs7R0FFRztBQUNILE1BQU0sQ0FBTixJQUFZLGdCQUtYO0FBTEQsV0FBWSxnQkFBZ0I7SUFDMUIsaURBQTZCLENBQUE7SUFDN0IsbURBQStCLENBQUE7SUFDL0IsMkNBQXVCLENBQUE7SUFDdkIscUNBQWlCLENBQUE7QUFDbkIsQ0FBQyxFQUxXLGdCQUFnQixLQUFoQixnQkFBZ0IsUUFLM0I7QUFFRDs7R0FFRztBQUNILE1BQU0sT0FBTyxlQUFlO0lBQ2xCLFNBQVMsR0FBZ0MsSUFBSSxHQUFHLEVBQUUsQ0FBQztJQUNuRCxhQUFhLENBS25CO0lBRUYsWUFBWSxhQUtYO1FBQ0MsNEJBQTRCO1FBQzVCLElBQUksQ0FBQyxhQUFhLEdBQUc7WUFDbkIsSUFBSSxFQUFFLGFBQWEsRUFBRSxJQUFJLElBQUksMkJBQTJCO1lBQ3hELE9BQU8sRUFBRSxhQUFhLEVBQUUsT0FBTztZQUMvQixVQUFVLEVBQUUsYUFBYSxFQUFFLFVBQVUsSUFBSSxFQUFFO1lBQzNDLFVBQVUsRUFBRSxhQUFhLEVBQUUsVUFBVSxJQUFJLEVBQUU7U0FDNUMsQ0FBQztRQUVGLHFDQUFxQztRQUNyQyxJQUFJLENBQUMsd0JBQXdCLEVBQUUsQ0FBQztJQUNsQyxDQUFDO0lBRUQ7O09BRUc7SUFDSyx3QkFBd0I7UUFDOUIsZ0JBQWdCO1FBQ2hCLElBQUksQ0FBQyxnQkFB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 \ No newline at end of file diff --git a/dist_ts/mail/core/index.d.ts b/dist_ts/mail/core/index.d.ts deleted file mode 100644 index a624fe1..0000000 --- a/dist_ts/mail/core/index.d.ts +++ /dev/null @@ -1,4 +0,0 @@ -export * from './classes.email.js'; -export * from './classes.emailvalidator.js'; -export * from './classes.templatemanager.js'; -export * from './classes.bouncemanager.js'; diff --git a/dist_ts/mail/core/index.js b/dist_ts/mail/core/index.js deleted file mode 100644 index 28734a1..0000000 --- a/dist_ts/mail/core/index.js +++ /dev/null @@ -1,6 +0,0 @@ -// Core email components -export * from './classes.email.js'; -export * from './classes.emailvalidator.js'; -export * from './classes.templatemanager.js'; -export * from './classes.bouncemanager.js'; -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi90cy9tYWlsL2NvcmUvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsd0JBQXdCO0FBQ3hCLGNBQWMsb0JBQW9CLENBQUM7QUFDbkMsY0FBYyw2QkFBNkIsQ0FBQztBQUM1QyxjQUFjLDhCQUE4QixDQUFDO0FBQzdDLGNBQWMsNEJBQTRCLENBQUMifQ== \ No newline at end of file diff --git a/dist_ts/mail/delivery/classes.delivery.queue.d.ts b/dist_ts/mail/delivery/classes.delivery.queue.d.ts deleted file mode 100644 index 937487b..0000000 --- a/dist_ts/mail/delivery/classes.delivery.queue.d.ts +++ /dev/null @@ -1,163 +0,0 @@ -import { EventEmitter } from 'node:events'; -import { type EmailProcessingMode } from '../routing/classes.email.config.js'; -import type { IEmailRoute } from '../routing/interfaces.js'; -/** - * Queue item status - */ -export type QueueItemStatus = 'pending' | 'processing' | 'delivered' | 'failed' | 'deferred'; -/** - * Queue item interface - */ -export interface IQueueItem { - id: string; - processingMode: EmailProcessingMode; - processingResult: any; - route: IEmailRoute; - status: QueueItemStatus; - attempts: number; - nextAttempt: Date; - lastError?: string; - createdAt: Date; - updatedAt: Date; - deliveredAt?: Date; -} -/** - * Queue options interface - */ -export interface IQueueOptions { - storageType?: 'memory' | 'disk'; - persistentPath?: string; - checkInterval?: number; - maxQueueSize?: number; - maxPerDestination?: number; - maxRetries?: number; - baseRetryDelay?: number; - maxRetryDelay?: number; -} -/** - * Queue statistics interface - */ -export interface IQueueStats { - queueSize: number; - status: { - pending: number; - processing: number; - delivered: number; - failed: number; - deferred: number; - }; - modes: { - forward: number; - mta: number; - process: number; - }; - oldestItem?: Date; - newestItem?: Date; - averageAttempts: number; - totalProcessed: number; - processingActive: boolean; -} -/** - * A unified queue for all email modes - */ -export declare class UnifiedDeliveryQueue extends EventEmitter { - private options; - private queue; - private checkTimer?; - private stats; - private processing; - private totalProcessed; - /** - * Create a new unified delivery queue - * @param options Queue options - */ - constructor(options: IQueueOptions); - /** - * Initialize the queue - */ - initialize(): Promise; - /** - * Start queue processing - */ - private startProcessing; - /** - * Stop queue processing - */ - private stopProcessing; - /** - * Check for items that need to be processed - */ - private processQueue; - /** - * Add an item to the queue - * @param processingResult Processing result to queue - * @param mode Processing mode - * @param route Email route - */ - enqueue(processingResult: any, mode: EmailProcessingMode, route: IEmailRoute): Promise; - /** - * Get an item from the queue - * @param id Item ID - */ - getItem(id: string): IQueueItem | undefined; - /** - * Mark an item as being processed - * @param id Item ID - */ - markProcessing(id: string): Promise; - /** - * Mark an item as delivered - * @param id Item ID - */ - markDelivered(id: string): Promise; - /** - * Mark an item as failed - * @param id Item ID - * @param error Error message - */ - markFailed(id: string, error: string): Promise; - /** - * Remove an item from the queue - * @param id Item ID - */ - removeItem(id: string): Promise; - /** - * Persist an item to disk - * @param item Item to persist - */ - private persistItem; - /** - * Remove an item from disk - * @param id Item ID - */ - private removeItemFromDisk; - /** - * Load queue items from disk - */ - private loadFromDisk; - /** - * Update queue statistics - */ - private updateStats; - /** - * Get queue statistics - */ - getStats(): IQueueStats; - /** - * Pause queue processing - */ - pause(): void; - /** - * Resume queue processing - */ - resume(): void; - /** - * Clean up old delivered and failed items - * @param maxAge Maximum age in milliseconds (default: 7 days) - */ - cleanupOldItems(maxAge?: number): Promise; - /** - * Shutdown the queue - */ - shutdown(): Promise; -} diff --git a/dist_ts/mail/delivery/classes.delivery.queue.js b/dist_ts/mail/delivery/classes.delivery.queue.js deleted file mode 100644 index b29ce72..0000000 --- a/dist_ts/mail/delivery/classes.delivery.queue.js +++ /dev/null @@ -1,488 +0,0 @@ -import * as plugins from '../../plugins.js'; -import { EventEmitter } from 'node:events'; -import * as fs from 'node:fs'; -import * as path from 'node:path'; -import { logger } from '../../logger.js'; -import {} from '../routing/classes.email.config.js'; -/** - * A unified queue for all email modes - */ -export class UnifiedDeliveryQueue extends EventEmitter { - options; - queue = new Map(); - checkTimer; - stats; - processing = false; - totalProcessed = 0; - /** - * Create a new unified delivery queue - * @param options Queue options - */ - constructor(options) { - super(); - // Set default options - this.options = { - storageType: options.storageType || 'memory', - persistentPath: options.persistentPath || path.join(process.cwd(), 'email-queue'), - checkInterval: options.checkInterval || 30000, // 30 seconds - maxQueueSize: options.maxQueueSize || 10000, - maxPerDestination: options.maxPerDestination || 100, - maxRetries: options.maxRetries || 5, - baseRetryDelay: options.baseRetryDelay || 60000, // 1 minute - maxRetryDelay: options.maxRetryDelay || 3600000 // 1 hour - }; - // Initialize statistics - this.stats = { - queueSize: 0, - status: { - pending: 0, - processing: 0, - delivered: 0, - failed: 0, - deferred: 0 - }, - modes: { - forward: 0, - mta: 0, - process: 0 - }, - averageAttempts: 0, - totalProcessed: 0, - processingActive: false - }; - } - /** - * Initialize the queue - */ - async initialize() { - logger.log('info', 'Initializing UnifiedDeliveryQueue'); - try { - // Create persistent storage directory if using disk storage - if (this.options.storageType === 'disk') { - if (!fs.existsSync(this.options.persistentPath)) { - fs.mkdirSync(this.options.persistentPath, { recursive: true }); - } - // Load existing items from disk - await this.loadFromDisk(); - } - // Start the queue processing timer - this.startProcessing(); - // Emit initialized event - this.emit('initialized'); - logger.log('info', 'UnifiedDeliveryQueue initialized successfully'); - } - catch (error) { - logger.log('error', `Failed to initialize queue: ${error.message}`); - throw error; - } - } - /** - * Start queue processing - */ - startProcessing() { - if (this.checkTimer) { - clearInterval(this.checkTimer); - } - this.checkTimer = setInterval(() => this.processQueue(), this.options.checkInterval); - this.processing = true; - this.stats.processingActive = true; - this.emit('processingStarted'); - logger.log('info', 'Queue processing started'); - } - /** - * Stop queue processing - */ - stopProcessing() { - if (this.checkTimer) { - clearInterval(this.checkTimer); - this.checkTimer = undefined; - } - this.processing = false; - this.stats.processingActive = false; - this.emit('processingStopped'); - logger.log('info', 'Queue processing stopped'); - } - /** - * Check for items that need to be processed - */ - async processQueue() { - try { - const now = new Date(); - let readyItems = []; - // Find items ready for processing - for (const item of this.queue.values()) { - if (item.status === 'pending' || (item.status === 'deferred' && item.nextAttempt <= now)) { - readyItems.push(item); - } - } - if (readyItems.length === 0) { - return; - } - // Sort by oldest first - readyItems.sort((a, b) => a.createdAt.getTime() - b.createdAt.getTime()); - // Emit event for ready items - this.emit('itemsReady', readyItems); - logger.log('info', `Found ${readyItems.length} items ready for processing`); - // Update statistics - this.updateStats(); - } - catch (error) { - logger.log('error', `Error processing queue: ${error.message}`); - this.emit('error', error); - } - } - /** - * Add an item to the queue - * @param processingResult Processing result to queue - * @param mode Processing mode - * @param route Email route - */ - async enqueue(processingResult, mode, route) { - // Check if queue is full - if (this.queue.size >= this.options.maxQueueSize) { - throw new Error('Queue is full'); - } - // Generate a unique ID - const id = `${Date.now()}-${Math.random().toString(36).substring(2, 15)}`; - // Create queue item - const item = { - id, - processingMode: mode, - processingResult, - route, - status: 'pending', - attempts: 0, - nextAttempt: new Date(), - createdAt: new Date(), - updatedAt: new Date() - }; - // Add to queue - this.queue.set(id, item); - // Persist to disk if using disk storage - if (this.options.storageType === 'disk') { - await this.persistItem(item); - } - // Update statistics - this.updateStats(); - // Emit event - this.emit('itemEnqueued', item); - logger.log('info', `Item enqueued with ID ${id}, mode: ${mode}`); - return id; - } - /** - * Get an item from the queue - * @param id Item ID - */ - getItem(id) { - return this.queue.get(id); - } - /** - * Mark an item as being processed - * @param id Item ID - */ - async markProcessing(id) { - const item = this.queue.get(id); - if (!item) { - return false; - } - // Update status - item.status = 'processing'; - item.attempts++; - item.updatedAt = new Date(); - // Persist changes if using disk storage - if (this.options.storageType === 'disk') { - await this.persistItem(item); - } - // Update statistics - this.updateStats(); - // Emit event - this.emit('itemProcessing', item); - logger.log('info', `Item ${id} marked as processing, attempt ${item.attempts}`); - return true; - } - /** - * Mark an item as delivered - * @param id Item ID - */ - async markDelivered(id) { - const item = this.queue.get(id); - if (!item) { - return false; - } - // Update status - item.status = 'delivered'; - item.updatedAt = new Date(); - item.deliveredAt = new Date(); - // Persist changes if using disk storage - if (this.options.storageType === 'disk') { - await this.persistItem(item); - } - // Update statistics - this.totalProcessed++; - this.updateStats(); - // Emit event - this.emit('itemDelivered', item); - logger.log('info', `Item ${id} marked as delivered after ${item.attempts} attempts`); - return true; - } - /** - * Mark an item as failed - * @param id Item ID - * @param error Error message - */ - async markFailed(id, error) { - const item = this.queue.get(id); - if (!item) { - return false; - } - // Determine if we should retry - if (item.attempts < this.options.maxRetries) { - // Calculate next retry time with exponential backoff - const delay = Math.min(this.options.baseRetryDelay * Math.pow(2, item.attempts - 1), this.options.maxRetryDelay); - // Update status - item.status = 'deferred'; - item.lastError = error; - item.nextAttempt = new Date(Date.now() + delay); - item.updatedAt = new Date(); - // Persist changes if using disk storage - if (this.options.storageType === 'disk') { - await this.persistItem(item); - } - // Emit event - this.emit('itemDeferred', item); - logger.log('info', `Item ${id} deferred for ${delay}ms, attempt ${item.attempts}, error: ${error}`); - } - else { - // Mark as permanently failed - item.status = 'failed'; - item.lastError = error; - item.updatedAt = new Date(); - // Persist changes if using disk storage - if (this.options.storageType === 'disk') { - await this.persistItem(item); - } - // Update statistics - this.totalProcessed++; - // Emit event - this.emit('itemFailed', item); - logger.log('warn', `Item ${id} permanently failed after ${item.attempts} attempts, error: ${error}`); - } - // Update statistics - this.updateStats(); - return true; - } - /** - * Remove an item from the queue - * @param id Item ID - */ - async removeItem(id) { - const item = this.queue.get(id); - if (!item) { - return false; - } - // Remove from queue - this.queue.delete(id); - // Remove from disk if using disk storage - if (this.options.storageType === 'disk') { - await this.removeItemFromDisk(id); - } - // Update statistics - this.updateStats(); - // Emit event - this.emit('itemRemoved', item); - logger.log('info', `Item ${id} removed from queue`); - return true; - } - /** - * Persist an item to disk - * @param item Item to persist - */ - async persistItem(item) { - try { - const filePath = path.join(this.options.persistentPath, `${item.id}.json`); - await fs.promises.writeFile(filePath, JSON.stringify(item, null, 2), 'utf8'); - } - catch (error) { - logger.log('error', `Failed to persist item ${item.id}: ${error.message}`); - this.emit('error', error); - } - } - /** - * Remove an item from disk - * @param id Item ID - */ - async removeItemFromDisk(id) { - try { - const filePath = path.join(this.options.persistentPath, `${id}.json`); - if (fs.existsSync(filePath)) { - await fs.promises.unlink(filePath); - } - } - catch (error) { - logger.log('error', `Failed to remove item ${id} from disk: ${error.message}`); - this.emit('error', error); - } - } - /** - * Load queue items from disk - */ - async loadFromDisk() { - try { - // Check if directory exists - if (!fs.existsSync(this.options.persistentPath)) { - return; - } - // Get all JSON files - const files = fs.readdirSync(this.options.persistentPath).filter(file => file.endsWith('.json')); - // Load each file - for (const file of files) { - try { - const filePath = path.join(this.options.persistentPath, file); - const data = await fs.promises.readFile(filePath, 'utf8'); - const item = JSON.parse(data); - // Convert date strings to Date objects - item.createdAt = new Date(item.createdAt); - item.updatedAt = new Date(item.updatedAt); - item.nextAttempt = new Date(item.nextAttempt); - if (item.deliveredAt) { - item.deliveredAt = new Date(item.deliveredAt); - } - // Add to queue - this.queue.set(item.id, item); - } - catch (error) { - logger.log('error', `Failed to load item from ${file}: ${error.message}`); - } - } - // Update statistics - this.updateStats(); - logger.log('info', `Loaded ${this.queue.size} items from disk`); - } - catch (error) { - logger.log('error', `Failed to load items from disk: ${error.message}`); - throw error; - } - } - /** - * Update queue statistics - */ - updateStats() { - // Reset counters - this.stats.queueSize = this.queue.size; - this.stats.status = { - pending: 0, - processing: 0, - delivered: 0, - failed: 0, - deferred: 0 - }; - this.stats.modes = { - forward: 0, - mta: 0, - process: 0 - }; - let totalAttempts = 0; - let oldestTime = Date.now(); - let newestTime = 0; - // Count by status and mode - for (const item of this.queue.values()) { - // Count by status - this.stats.status[item.status]++; - // Count by mode - this.stats.modes[item.processingMode]++; - // Track total attempts - totalAttempts += item.attempts; - // Track oldest and newest - const itemTime = item.createdAt.getTime(); - if (itemTime < oldestTime) { - oldestTime = itemTime; - } - if (itemTime > newestTime) { - newestTime = itemTime; - } - } - // Calculate average attempts - this.stats.averageAttempts = this.queue.size > 0 ? totalAttempts / this.queue.size : 0; - // Set oldest and newest - this.stats.oldestItem = this.queue.size > 0 ? new Date(oldestTime) : undefined; - this.stats.newestItem = this.queue.size > 0 ? new Date(newestTime) : undefined; - // Set total processed - this.stats.totalProcessed = this.totalProcessed; - // Set processing active - this.stats.processingActive = this.processing; - // Emit statistics event - this.emit('statsUpdated', this.stats); - } - /** - * Get queue statistics - */ - getStats() { - return { ...this.stats }; - } - /** - * Pause queue processing - */ - pause() { - if (this.processing) { - this.stopProcessing(); - logger.log('info', 'Queue processing paused'); - } - } - /** - * Resume queue processing - */ - resume() { - if (!this.processing) { - this.startProcessing(); - logger.log('info', 'Queue processing resumed'); - } - } - /** - * Clean up old delivered and failed items - * @param maxAge Maximum age in milliseconds (default: 7 days) - */ - async cleanupOldItems(maxAge = 7 * 24 * 60 * 60 * 1000) { - const cutoff = new Date(Date.now() - maxAge); - let removedCount = 0; - // Find old items - for (const item of this.queue.values()) { - if (['delivered', 'failed'].includes(item.status) && item.updatedAt < cutoff) { - // Remove item - await this.removeItem(item.id); - removedCount++; - } - } - logger.log('info', `Cleaned up ${removedCount} old items`); - return removedCount; - } - /** - * Shutdown the queue - */ - async shutdown() { - logger.log('info', 'Shutting down UnifiedDeliveryQueue'); - // Stop processing - this.stopProcessing(); - // Clear the check timer to prevent memory leaks - if (this.checkTimer) { - clearInterval(this.checkTimer); - this.checkTimer = undefined; - } - // If using disk storage, make sure all items are persisted - if (this.options.storageType === 'disk') { - const pendingWrites = []; - for (const item of this.queue.values()) { - pendingWrites.push(this.persistItem(item)); - } - // Wait for all writes to complete - await Promise.all(pendingWrites); - } - // Clear the queue (memory only) - this.queue.clear(); - // Update statistics - this.updateStats(); - // Emit shutdown event - this.emit('shutdown'); - logger.log('info', 'UnifiedDeliveryQueue shut down successfully'); - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5kZWxpdmVyeS5xdWV1ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3RzL21haWwvZGVsaXZlcnkvY2xhc3Nlcy5kZWxpdmVyeS5xdWV1ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssT0FBTyxNQUFNLGtCQUFrQixDQUFDO0FBQzVDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxhQUFhLENBQUM7QUFDM0MsT0FBTyxLQUFLLEVBQUUsTUFBTSxTQUFTLENBQUM7QUFDOUIsT0FBTyxLQUFLLElBQUksTUFBTSxXQUFXLENBQUM7QUFDbEMsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQ3pDLE9BQU8sRUFBNEIsTUFBTSxvQ0FBb0MsQ0FBQztBQW9FOUU7O0dBRUc7QUFDSCxNQUFNLE9BQU8sb0JBQXFCLFNBQVEsWUFBWTtJQUM1QyxPQUFPLENBQTBCO0lBQ2pDLEtBQUssR0FBNEIsSUFBSSxHQUFHLEVBQUUsQ0FBQztJQUMzQyxVQUFVLENBQWtCO0lBQzVCLEtBQUssQ0FBYztJQUNuQixVQUFVLEdBQVksS0FBSyxDQUFDO0lBQzVCLGNBQWMsR0FBVyxDQUFDLENBQUM7SUFFbkM7OztPQUdHO0lBQ0gsWUFBWSxPQUFzQjtRQUNoQyxLQUFLLEVBQUUsQ0FBQztRQUVSLHNCQUFzQjtRQUN0QixJQUFJLENBQUMsT0FBTyxHQUFHO1lBQ2IsV0FBVyxFQUFFLE9BQU8sQ0FBQyxXQUFXLElBQUksUUFBUTtZQUM1QyxjQUFjLEVBQUUsT0FBTyxDQUFDLGNBQWMsSUFBSSxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxHQUFHLEVBQUUsRUFBRSxhQUFhLENBQUM7WUFDakYsYUFBYSxFQUFFLE9BQU8sQ0FBQyxhQUFhLElBQUksS0FBSyxFQUFFLGFBQWE7WUFDNUQsWUFBWSxFQUFFLE9BQU8sQ0FBQyxZQUFZLElBQUksS0FBSztZQUMzQyxpQkFBaUIsRUFBRSxPQUFPLENBQUMsaUJBQWlCLElBQUksR0FBRztZQUNuRCxVQUFVLEVBQUUsT0FBTyxDQUFDLFVBQVUsSUFBSSxDQUFDO1lBQ25DLGNBQWMsRUFBRSxPQUFPLENBQUMsY0FBYyxJQUFJLEtBQUssRUFBRSxXQUFXO1lBQzVELGFBQWEsRUFBRSxPQUFPLENBQUMsYUFBYSxJQUFJLE9BQU8sQ0FBQyxTQUFTO1NBQzFELENBQUM7UUFFRix3QkFBd0I7UUFDeEIsSUFBSSxDQUFDLEtBQUssR0FBRztZQUNYLFNBQVMsRUFBRSxDQUFDO1lBQ1osTUFBTSxFQUFFO2dCQUNOLE9BQU8sRUFBRSxDQUFDO2dCQUNWLFVBQVUsRUFBRSxDQUFDO2dCQUNiLFNBQVMsRUFBRSxDQUFDO2dCQUNaLE1BQU0sRUFBRSxDQUFDO2dCQUNULFFBQVEsRUFBRSxDQUFDO2FBQ1o7WUFDRCxLQUFLLEVBQUU7Z0JBQ0wsT0FBTyxFQUFFLENBQUM7Z0JBQ1YsR0FBRyxFQUFFLENBQUM7Z0JBQ04sT0FBTyxFQUFFLENBQUM7YUFDWDtZQUNELGVBQWUsRUFBRSxDQUFDO1lBQ2xCLGNBQWMsRUFBRSxDQUFDO1lBQ2pCLGdCQUFnQixFQUFFLEtBQUs7U0FDeEIsQ0FBQztJQUNKLENBQUM7SUFFRDs7T0FFRztJQUNJLEtBQUssQ0FBQyxVQUFVO1FBQ3JCLE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLG1DQUFtQyxDQUFDLENBQUM7UUFFeEQsSUFBSSxDQUFDO1lBQ0gsNERBQTREO1lBQzVELElBQUksSUFBSSxDQUFDLE9BQU8sQ0FBQyxXQUFXLEtBQUssTUFBTSxFQUFFLENBQUM7Z0JBQ3hDLElBQUksQ0FBQyxFQUFFLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsY0FBYyxDQUFDLEVBQUUsQ0FBQztvQkFDaEQsRUFBRSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLGNBQWMsRUFBRSxFQUFFLFNBQVMsRUFBRSxJQUFJLEVBQUUsQ0FBQyxDQUFDO2dCQUNqRSxDQUFDO2dCQUVELGdDQUFnQztnQkFDaEMsTUFBTSxJQUFJLENBQUMsWUFBWSxFQUFFLENBQUM7WUFDNUIsQ0FBQztZQUVELG1DQUFtQztZQUNuQyxJQUFJLENBQUMsZUFBZSxFQUFFLENBQUM7WUFFdkIseUJBQXlCO1lBQ3pCLElBQUksQ0FBQyxJQUFJLENBQUMsYUFBYSxDQUFDLENBQUM7WUFDekIsTUFBTSxDQUFDLEdBQUcsQ0FBQyxNQUFNLEVBQUUsK0NBQStDLENBQUMsQ0FBQztRQUN0RSxDQUFDO1FBQUMsT0FBTyxLQUFLLEVBQUUsQ0FBQztZQUNmLE1BQU0sQ0FBQyxHQUFHLENBQUMsT0FBTyxFQUFFLCtCQUErQixLQUFLLENBQUMsT0FBTyxFQUFFLENBQUMsQ0FBQztZQUNwRSxNQUFNLEtBQUssQ0FBQztRQUNkLENBQUM7SUFDSCxDQUFDO0lBRUQ7O09BRUc7SUFDSyxlQUFlO1FBQ3JCLElBQUksSUFBSSxDQUFDLFVBQVUsRUFBRSxDQUFDO1lBQ3BCLGFBQWEsQ0FBQyxJQUFJLENBQUMsVUFBVSxDQUFDLENBQUM7UUFDakMsQ0FBQztRQUVELElBQUksQ0FBQyxVQUFVLEdBQUcsV0FBVyxDQUFDLEdBQUcsRUFBRSxDQUFDLElBQUksQ0FBQyxZQUFZLEVBQUUsRUFBRSxJQUFJLENBQUMsT0FBTyxDQUFDLGFBQWEsQ0FBQyxDQUFDO1FBQ3JGLElBQUksQ0FBQyxVQUFVLEdBQUcsSUFBSSxDQUFDO1FBQ3ZCLElBQUksQ0FBQyxLQUFLLENBQUMsZ0JBQWdCLEdBQUcsSUFBSSxDQUFDO1FBQ25DLElBQUksQ0FBQyxJQUFJLENBQUMsbUJBQW1CLENBQUMsQ0FBQztRQUMvQixNQUFNLENBQUMsR0FBRyxDQUFDLE1BQU0sRUFBRSwwQkFBMEIsQ0FBQyxDQUFDO0lBQ2pELENBQUM7SUFFRDs7T0FFRztJQUNLLGNBQWM7UUFDcEIsSUFBSSxJQUFJLENBQUMsVUFBVSxFQUFFLENBQUM7WUFDcEIsYUFBYSxDQUFDLElBQUksQ0FBQyxVQUFVLENBQUMsQ0FBQztZQUMvQixJQUFJLENBQUMsVUFBVSxHQUFHLFNBQVMsQ0FBQztRQUM5QixDQUFDO1FBRUQsSUFBSSxDQUFDLFVBQVUsR0FBRyxLQUFLLENBQUM7UUFDeEIsSUFBSSxDQUFDLEtBQUssQ0FBQyxnQkFBZ0IsR0FBRyxLQUFLLENBQUM7UUFDcEMsSUFBSSxDQUFDLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxDQUFDO1FBQy9CLE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLDBCQUEwQixDQUFDLENBQUM7SUFDakQsQ0FBQztJQUVEOztPQUVHO0lBQ0ssS0FBSyxDQUFDLFlBQVk7UUFDeEIsSUFBSSxDQUFDO1lBQ0gsTUFBTSxHQUFHLEdBQUcsSUFBSSxJQUFJLEVBQUUsQ0FBQztZQUN2QixJQUFJLFVBQVUsR0FBaUIsRUFBRSxDQUFDO1lBRWxDLGtDQUFrQztZQUNsQyxLQUFLLE1BQU0sSUFBSSxJQUFJLElBQUksQ0FBQyxLQUFLLENBQUMsTUFBTSxFQUFFLEVBQUUsQ0FBQztnQkFDdkMsSUFBSSxJQUFJLENBQUMsTUFBTSxLQUFLLFNBQVMsSUFBSSxDQUFDLElBQUksQ0FBQyxNQUFNLEtBQUssVUFBVSxJQUFJLElBQUksQ0FBQyxXQUFXLElBQUksR0FBRyxDQUFDLEVBQUUsQ0FBQztvQkFDekYsVUFBVSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQztnQkFDeEIsQ0FBQztZQUNILENBQUM7WUFFRCxJQUFJLFVBQVUsQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFLENBQUM7Z0JBQzVCLE9BQU87WUFDVCxDQUFDO1lBRUQsdUJBQXVCO1lBQ3ZCLFVBQVUsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsU0FBUyxDQUFDLE9BQU8sRUFBRSxHQUFHLENBQUMsQ0FBQyxTQUFTLENBQUMsT0FBTyxFQUFFLENBQUMsQ0FBQztZQUV6RSw2QkFBNkI7WUFDN0IsSUFBSSxDQUFDLElBQUksQ0FBQyxZQUFZLEVBQUUsVUFBVSxDQUFDLENBQUM7WUFDcEMsTUFBTSxDQUFDLEdBQUcsQ0FBQyxNQUFNLEVBQUUsU0FBUyxVQUFVLENBQUMsTUFBTSw2QkFBNkIsQ0FBQyxDQUFDO1lBRTVFLG9CQUFvQjtZQUNwQixJQUFJLENBQUMsV0FBVyxFQUFFLENBQUM7UUFDckIsQ0FBQztRQUFDLE9BQU8sS0FBSyxFQUFFLENBQUM7WUFDZixNQUFNLENBQUMsR0FBRyxDQUFDLE9BQU8sRUFBRSwyQkFBMkIsS0FBSyxDQUFDLE9BQU8sRUFBRSxDQUFDLENBQUM7WUFDaEUsSUFBSSxDQUFDLElBQUksQ0FBQyxPQUFPLEVBQUUsS0FBSyxDQUFDLENBQUM7UUFDNUIsQ0FBQztJQUNILENBQUM7SUFFRDs7Ozs7T0FLRztJQUNJLEtBQUssQ0FBQyxPQUFPLENBQUMsZ0JBQXFCLEVBQUUsSUFBeUIsRUFBRSxLQUFrQjtRQUN2Rix5QkFBeUI7UUFDekIsSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksSUFBSSxJQUFJLENBQUMsT0FBTyxDQUFDLFlBQVksRUFBRSxDQUFDO1lBQ2pELE1BQU0sSUFBSSxLQUFLLENBQUMsZUFBZSxDQUFDLENBQUM7UUFDbkMsQ0FBQztRQUVELHVCQUF1QjtRQUN2QixNQUFNLEVBQUUsR0FBRyxHQUFHLElBQUksQ0FBQyxHQUFHLEVBQUUsSUFBSSxJQUFJLENBQUMsTUFBTSxFQUFFLENBQUMsUUFBUSxDQUFDLEVBQUUsQ0FBQyxDQUFDLFNBQVMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLEVBQUUsQ0FBQztRQUUxRSxvQkFBb0I7UUFDcEIsTUFBTSxJQUFJLEdBQWU7WUFDdkIsRUFBRTtZQUNGLGNBQWMsRUFBRSxJQUFJO1lBQ3BCLGdCQUFnQjtZQUNoQixLQUFLO1lBQ0wsTUFBTSxFQUFFLFNBQVM7WUFDakIsUUFBUSxFQUFFLENBQUM7WUFDWCxXQUFXLEVBQUUsSUFBSSxJQUFJLEVBQUU7WUFDdkIsU0FBUyxFQUFFLElBQUksSUFBSSxFQUFFO1lBQ3JCLFNBQVMsRUFBRSxJQUFJLElBQUksRUFBRTtTQUN0QixDQUFDO1FBRUYsZUFBZTtRQUNmLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLEVBQUUsRUFBRSxJQUFJLENBQUMsQ0FBQztRQUV6Qix3Q0FBd0M7UUFDeEMsSUFBSSxJQUFJLENBQUMsT0FBTyxDQUFDLFdBQVcsS0FBSyxNQUFNLEVBQUUsQ0FBQztZQUN4QyxNQUFNLElBQUksQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDL0IsQ0FBQztRQUVELG9CQUFvQjtRQUNwQixJQUFJLENBQUMsV0FBVyxFQUFFLENBQUM7UUFFbkIsYUFBYTtRQUNiLElBQUksQ0FBQyxJQUFJLENBQUMsY0FBYyxFQUFFLElBQUksQ0FBQyxDQUFDO1FBQ2hDLE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLHlCQUF5QixFQUFFLFdBQVcsSUFBSSxFQUFFLENBQUMsQ0FBQztRQUVqRSxPQUFPLEVBQUUsQ0FBQztJQUNaLENBQUM7SUFFRDs7O09BR0c7SUFDSSxPQUFPLENBQUMsRUFBVTtRQUN2QixPQUFPLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxDQUFDO0lBQzVCLENBQUM7SUFFRDs7O09BR0c7SUFDSSxLQUFLLENBQUMsY0FBYyxDQUFDLEVBQVU7UUFDcEMsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLENBQUM7UUFFaEMsSUFBSSxDQUFDLElBQUksRUFBRSxDQUFDO1lBQ1YsT0FBTyxLQUFLLENBQUM7UUFDZixDQUFDO1FBRUQsZ0JBQWdCO1FBQ2hCLElBQUksQ0FBQyxNQUFNLEdBQUcsWUFBWSxDQUFDO1FBQzNCLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBQztRQUNoQixJQUFJLENBQUMsU0FBUyxHQUFHLElBQUksSUFBSSxFQUFFLENBQUM7UUFFNUIsd0NBQXdDO1FBQ3hDLElBQUksSUFBSSxDQUFDLE9BQU8sQ0FBQyxXQUFXLEtBQUssTUFBTSxFQUFFLENBQUM7WUFDeEMsTUFBTSxJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxDQUFDO1FBQy9CLENBQUM7UUFFRCxvQkFBb0I7UUFDcEIsSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDO1FBRW5CLGFBQWE7UUFDYixJQUFJLENBQUMsSUFBSSxDQUFDLGdCQUFnQixFQUFFLElBQUksQ0FBQyxDQUFDO1FBQ2xDLE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLFFBQVEsRUFBRSxrQ0FBa0MsSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDLENBQUM7UUFFaEYsT0FBTyxJQUFJLENBQUM7SUFDZCxDQUFDO0lBRUQ7OztPQUdHO0lBQ0ksS0FBSyxDQUFDLGFBQWEsQ0FBQyxFQUFVO1FBQ25DLE1BQU0sSUFBSSxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxDQUFDO1FBRWhDLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQztZQUNWLE9BQU8sS0FBSyxDQUFDO1FBQ2YsQ0FBQztRQUVELGdCQUFnQjtRQUNoQixJQUFJLENBQUMsTUFBTSxHQUFHLFdBQVcsQ0FBQztRQUMxQixJQUFJLENBQUMsU0FBUyxHQUFHLElBQUksSUFBSSxFQUFFLENBQUM7UUFDNUIsSUFBSSxDQUFDLFdBQVcsR0FBRyxJQUFJLElBQUksRUFBRSxDQUFDO1FBRTlCLHdDQUF3QztRQUN4QyxJQUFJLElBQUksQ0FBQyxPQUFPLENBQUMsV0FBVyxLQUFLLE1BQU0sRUFBRSxDQUFDO1lBQ3hDLE1BQU0sSUFBSSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUMvQixDQUFDO1FBRUQsb0JBQW9CO1FBQ3BCLElBQUksQ0FBQyxjQUFjLEVBQUUsQ0FBQztRQUN0QixJQUFJLENBQUMsV0FBVyxFQUFFLENBQUM7UUFFbkIsYUFBYTtRQUNiLElBQUksQ0FBQyxJQUFJLENBQUMsZUFBZSxFQUFFLElBQUksQ0FBQyxDQUFDO1FBQ2pDLE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLFFBQVEsRUFBRSw4QkFBOEIsSUFBSSxDQUFDLFFBQVEsV0FBVyxDQUFDLENBQUM7UUFFckYsT0FBTyxJQUFJLENBQUM7SUFDZCxDQUFDO0lBRUQ7Ozs7T0FJRztJQUNJLEtBQUssQ0FBQyxVQUFVLENBQUMsRUFBVSxFQUFFLEtBQWE7UUFDL0MsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLENBQUM7UUFFaEMsSUFBSSxDQUFDLElBQUksRUFBRSxDQUFDO1lBQ1YsT0FBTyxLQUFLLENBQUM7UUFDZixDQUFDO1FBRUQsK0JBQStCO1FBQy9CLElBQUksSUFBSSxDQUFDLFFBQVEsR0FBRyxJQUFJLENBQUMsT0FBTyxDQUFDLFVBQVUsRUFBRSxDQUFDO1lBQzVDLHFEQUFxRDtZQUNyRCxNQUFNLEtBQUssR0FBRyxJQUFJLENBQUMsR0FBRyxDQUNwQixJQUFJLENBQUMsT0FBTyxDQUFDLGNBQWMsR0FBRyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUMsRUFBRSxJQUFJLENBQUMsUUFBUSxHQUFHLENBQUMsQ0FBQyxFQUM1RCxJQUFJLENBQUMsT0FBTyxDQUFDLGFBQWEsQ0FDM0IsQ0FBQztZQUVGLGdCQUFnQjtZQUNoQixJQUFJLENBQUMsTUFBTSxHQUFHLFVBQVUsQ0FBQztZQUN6QixJQUFJLENBQUMsU0FBUyxHQUFHLEtBQUssQ0FBQztZQUN2QixJQUFJLENBQUMsV0FBVyxHQUFHLElBQUksSUFBSSxDQUFDLElBQUksQ0FBQyxHQUFHLEVBQUUsR0FBRyxLQUFLLENBQUMsQ0FBQztZQUNoRCxJQUFJLENBQUMsU0FBUyxHQUFHLElBQUksSUFBSSxFQUFFLENBQUM7WUFFNUIsd0NBQXdDO1lBQ3hDLElBQUksSUFBSSxDQUFDLE9BQU8sQ0FBQyxXQUFXLEtBQUssTUFBTSxFQUFFLENBQUM7Z0JBQ3hDLE1BQU0sSUFBSSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsQ0FBQztZQUMvQixDQUFDO1lBRUQsYUFBYTtZQUNiLElBQUksQ0FBQyxJQUFJLENBQUMsY0FBYyxFQUFFLElBQUksQ0FBQyxDQUFDO1lBQ2hDLE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLFFBQVEsRUFBRSxpQkFBaUIsS0FBSyxlQUFlLElBQUksQ0FBQyxRQUFRLFlBQVksS0FBSyxFQUFFLENBQUMsQ0FBQztRQUN0RyxDQUFDO2FBQU0sQ0FBQztZQUNOLDZCQUE2QjtZQUM3QixJQUFJLENBQUMsTUFBTSxHQUFHLFFBQVEsQ0FBQztZQUN2QixJQUFJLENBQUMsU0FBUyxHQUFHLEtBQUssQ0FBQztZQUN2QixJQUFJLENBQUMsU0FBUyxHQUFHLElBQUksSUFBSSxFQUFFLENBQUM7WUFFNUIsd0NBQXdDO1lBQ3hDLElBQUksSUFBSSxDQUFDLE9BQU8sQ0FBQyxXQUFXLEtBQUssTUFBTSxFQUFFLENBQUM7Z0JBQ3hDLE1BQU0sSUFBSSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsQ0FBQztZQUMvQixDQUFDO1lBRUQsb0JBQW9CO1lBQ3BCLElBQUksQ0FBQyxjQUFjLEVBQUUsQ0FBQztZQUV0QixhQUFhO1lBQ2IsSUFBSSxDQUFDLElBQUksQ0FBQyxZQUFZLEVBQUUsSUFBSSxDQUFDLENBQUM7WUFDOUIsTUFBTSxDQUFDLEdBQUcsQ0FBQyxNQUFNLEVBQUUsUUFBUSxFQUFFLDZCQUE2QixJQUFJLENBQUMsUUFBUSxxQkFBcUIsS0FBSyxFQUFFLENBQUMsQ0FBQztRQUN2RyxDQUFDO1FBRUQsb0JBQW9CO1FBQ3BCLElBQUksQ0FBQyxXQUFXLEVBQUUsQ0FBQztRQUVuQixPQUFPLElBQUksQ0FBQztJQUNkLENBQUM7SUFFRDs7O09BR0c7SUFDSSxLQUFLLENBQUMsVUFBVSxDQUFDLEVBQVU7UUFDaEMsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLENBQUM7UUFFaEMsSUFBSSxDQUFDLElBQUksRUFBRSxDQUFDO1lBQ1YsT0FBTyxLQUFLLENBQUM7UUFDZixDQUFDO1FBRUQsb0JBQW9CO1FBQ3BCLElBQUksQ0FBQyxLQUFLLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQyxDQUFDO1FBRXRCLHlDQUF5QztRQUN6QyxJQUFJLElBQUksQ0FBQyxPQUFPLENBQUMsV0FBVyxLQUFLLE1BQU0sRUFBRSxDQUFDO1lBQ3hDLE1BQU0sSUFBSSxDQUFDLGtCQUFrQixDQUFDLEVBQUUsQ0FBQyxDQUFDO1FBQ3BDLENBQUM7UUFFRCxvQkFBb0I7UUFDcEIsSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDO1FBRW5CLGFBQWE7UUFDYixJQUFJLENBQUMsSUFBSSxDQUFDLGFBQWEsRUFBRSxJQUFJLENBQUMsQ0FBQztRQUMvQixNQUFNLENBQUMsR0FBRyxDQUFDLE1BQU0sRUFBRSxRQUFRLEVBQUUscUJBQXFCLENBQUMsQ0FBQztRQUVwRCxPQUFPLElBQUksQ0FBQztJQUNkLENBQUM7SUFFRDs7O09BR0c7SUFDSyxLQUFLLENBQUMsV0FBVyxDQUFDLElBQWdCO1FBQ3hDLElBQUksQ0FBQztZQUNILE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxjQUFjLEVBQUUsR0FBRyxJQUFJLENBQUMsRUFBRSxPQUFPLENBQUMsQ0FBQztZQUMzRSxNQUFNLEVBQUUsQ0FBQyxRQUFRLENBQUMsU0FBUyxDQUFDLFFBQVEsRUFBRSxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksRUFBRSxJQUFJLEVBQUUsQ0FBQyxDQUFDLEVBQUUsTUFBTSxDQUFDLENBQUM7UUFDL0UsQ0FBQztRQUFDLE9BQU8sS0FBSyxFQUFFLENBQUM7WUFDZixNQUFNLENBQUMsR0FBRyxDQUFDLE9BQU8sRUFBRSwwQkFBMEIsSUFBSSxDQUFDLEVBQUUsS0FBSyxLQUFLLENBQUMsT0FBTyxFQUFFLENBQUMsQ0FBQztZQUMzRSxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFBRSxLQUFLLENBQUMsQ0FBQztRQUM1QixDQUFDO0lBQ0gsQ0FBQztJQUVEOzs7T0FHRztJQUNLLEtBQUssQ0FBQyxrQkFBa0IsQ0FBQyxFQUFVO1FBQ3pDLElBQUksQ0FBQztZQUNILE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxjQUFjLEVBQUUsR0FBRyxFQUFFLE9BQU8sQ0FBQyxDQUFDO1lBRXRFLElBQUksRUFBRSxDQUFDLFVBQVUsQ0FBQyxRQUFRLENBQUMsRUFBRSxDQUFDO2dCQUM1QixNQUFNLEVBQUUsQ0FBQyxRQUFRLENBQUMsTUFBTSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1lBQ3JDLENBQUM7UUFDSCxDQUFDO1FBQUMsT0FBTyxLQUFLLEVBQUUsQ0FBQztZQUNmLE1BQU0sQ0FBQyxHQUFHLENBQUMsT0FBTyxFQUFFLHlCQUF5QixFQUFFLGVBQWUsS0FBSyxDQUFDLE9BQU8sRUFBRSxDQUFDLENBQUM7WUFDL0UsSUFBSSxDQUFDLElBQUksQ0FBQyxPQUFPLEVBQUUsS0FBSyxDQUFDLENBQUM7UUFDNUIsQ0FBQztJQUNILENBQUM7SUFFRDs7T0FFRztJQUNLLEtBQUssQ0FBQyxZQUFZO1FBQ3hCLElBQUksQ0FBQztZQUNILDRCQUE0QjtZQUM1QixJQUFJLENBQUMsRUFBRSxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLGNBQWMsQ0FBQyxFQUFFLENBQUM7Z0JBQ2hELE9BQU87WUFDVCxDQUFDO1lBRUQscUJBQXFCO1lBQ3JCLE1BQU0sS0FBSyxHQUFHLEVBQUUsQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxjQUFjLENBQUMsQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUM7WUFFakcsaUJBQWlCO1lBQ2pCLEtBQUssTUFBTSxJQUFJLElBQUksS0FBSyxFQUFFLENBQUM7Z0JBQ3pCLElBQUksQ0FBQztvQkFDSCxNQUFNLFFBQVEsR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsY0FBYyxFQUFFLElBQUksQ0FBQyxDQUFDO29CQUM5RCxNQUFNLElBQUksR0FBRyxNQUFNLEVBQUUsQ0FBQyxRQUFRLENBQUMsUUFBUSxDQUFDLFFBQVEsRUFBRSxNQUFNLENBQUMsQ0FBQztvQkFDMUQsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQWUsQ0FBQztvQkFFNUMsdUNBQXVDO29CQUN2QyxJQUFJLENBQUMsU0FBUyxHQUFHLElBQUksSUFBSSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsQ0FBQztvQkFDMUMsSUFBSSxDQUFDLFNBQVMsR0FBRyxJQUFJLElBQUksQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLENBQUM7b0JBQzFDLElBQUksQ0FBQyxXQUFXLEdBQUcsSUFBSSxJQUFJLENBQUMsSUFBSSxDQUFDLFdBQVcsQ0FBQyxDQUFDO29CQUM5QyxJQUFJLElBQUksQ0FBQyxXQUFXLEVBQUUsQ0FBQzt3QkFDckIsSUFBSSxDQUFDLFdBQVcsR0FBRyxJQUFJLElBQUksQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLENBQUM7b0JBQ2hELENBQUM7b0JBRUQsZUFBZTtvQkFDZixJQUFJLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsRUFBRSxFQUFFLElBQUksQ0FBQyxDQUFDO2dCQUNoQyxDQUFDO2dCQUFDLE9BQU8sS0FBSyxFQUFFLENBQUM7b0JBQ2YsTUFBTSxDQUFDLEdBQUcsQ0FBQyxPQUFPLEVBQUUsNEJBQTRCLElBQUksS0FBSyxLQUFLLENBQUMsT0FBTyxFQUFFLENBQUMsQ0FBQztnQkFDNUUsQ0FBQztZQUNILENBQUM7WUFFRCxvQkFBb0I7WUFDcEIsSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDO1lBRW5CLE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLFVBQVUsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLGtCQUFrQixDQUFDLENBQUM7UUFDbEUsQ0FBQztRQUFDLE9BQU8sS0FBSyxFQUFFLENBQUM7WUFDZixNQUFNLENBQUMsR0FBRyxDQUFDLE9BQU8sRUFBRSxtQ0FBbUMsS0FBSyxDQUFDLE9BQU8sRUFBRSxDQUFDLENBQUM7WUFDeEUsTUFBTSxLQUFLLENBQUM7UUFDZCxDQUFDO0lBQ0gsQ0FBQztJQUVEOztPQUVHO0lBQ0ssV0FBVztRQUNqQixpQkFBaUI7UUFDakIsSUFBSSxDQUFDLEtBQUssQ0FBQyxTQUFTLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUM7UUFDdkMsSUFBSSxDQUFDLEtBQUssQ0FBQyxNQUFNLEdBQUc7WUFDbEIsT0FBTyxFQUFFLENBQUM7WUFDVixVQUFVLEVBQUUsQ0FBQztZQUNiLFNBQVMsRUFBRSxDQUFDO1lBQ1osTUFBTSxFQUFFLENBQUM7WUFDVCxRQUFRLEVBQUUsQ0FBQztTQUNaLENBQUM7UUFDRixJQUFJLENBQUMsS0FBSyxDQUFDLEtBQUssR0FBRztZQUNqQixPQUFPLEVBQUUsQ0FBQztZQUNWLEdBQUcsRUFBRSxDQUFDO1lBQ04sT0FBTyxFQUFFLENBQUM7U0FDWCxDQUFDO1FBRUYsSUFBSSxhQUFhLEdBQUcsQ0FBQyxDQUFDO1FBQ3RCLElBQUksVUFBVSxHQUFHLElBQUksQ0FBQyxHQUFHLEVBQUUsQ0FBQztRQUM1QixJQUFJLFVBQVUsR0FBRyxDQUFDLENBQUM7UUFFbkIsMkJBQTJCO1FBQzNCLEtBQUssTUFBTSxJQUFJLElBQUksSUFBSSxDQUFDLEtBQUssQ0FBQyxNQUFNLEVBQUUsRUFBRSxDQUFDO1lBQ3ZDLGtCQUFrQjtZQUNsQixJQUFJLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQztZQUVqQyxnQkFBZ0I7WUFDaEIsSUFBSSxDQUFDLEtBQUssQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLGNBQWMsQ0FBQyxFQUFFLENBQUM7WUFFeEMsdUJBQXVCO1lBQ3ZCLGFBQWEsSUFBSSxJQUFJLENBQUMsUUFBUSxDQUFDO1lBRS9CLDBCQUEwQjtZQUMxQixNQUFNLFFBQVEsR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDLE9BQU8sRUFBRSxDQUFDO1lBQzFDLElBQUksUUFBUSxHQUFHLFVBQVUsRUFBRSxDQUFDO2dCQUMxQixVQUFVLEdBQUcsUUFBUSxDQUFDO1lBQ3hCLENBQUM7WUFDRCxJQUFJLFFBQVEsR0FBRyxVQUFVLEVBQUUsQ0FBQztnQkFDMUIsVUFBVSxHQUFHLFFBQVEsQ0FBQztZQUN4QixDQUFDO1FBQ0gsQ0FBQztRQUVELDZCQUE2QjtRQUM3QixJQUFJLENBQUMsS0FBSyxDQUFDLGVBQWUsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLGFBQWEsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBRXZGLHdCQUF3QjtRQUN4QixJQUFJLENBQUMsS0FBSyxDQUFDLFVBQVUsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksSUFBSSxDQUFDLFVBQVUsQ0FBQyxDQUFDLENBQUMsQ0FBQyxTQUFTLENBQUM7UUFDL0UsSUFBSSxDQUFDLEtBQUssQ0FBQyxVQUFVLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLElBQUksQ0FBQyxVQUFVLENBQUMsQ0FBQyxDQUFDLENBQUMsU0FBUyxDQUFDO1FBRS9FLHNCQUFzQjtRQUN0QixJQUFJLENBQUMsS0FBSyxDQUFDLGNBQWMsR0FBRyxJQUFJLENBQUMsY0FBYyxDQUFDO1FBRWhELHdCQUF3QjtRQUN4QixJQUFJLENBQUMsS0FBSyxDQUFDLGdCQUFnQixHQUFHLElBQUksQ0FBQyxVQUFVLENBQUM7UUFFOUMsd0JBQXdCO1FBQ3hCLElBQUksQ0FBQyxJQUFJLENBQUMsY0FBYyxFQUFFLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUN4QyxDQUFDO0lBRUQ7O09BRUc7SUFDSSxRQUFRO1FBQ2IsT0FBTyxFQUFFLEdBQUcsSUFBSSxDQUFDLEtBQUssRUFBRSxDQUFDO0lBQzNCLENBQUM7SUFFRDs7T0FFRztJQUNJLEtBQUs7UUFDVixJQUFJLElBQUksQ0FBQyxVQUFVLEVBQUUsQ0FBQztZQUNwQixJQUFJLENBQUMsY0FBYyxFQUFFLENBQUM7WUFDdEIsTUFBTSxDQUFDLEdBQUcsQ0FBQyxNQUFNLEVBQUUseUJBQXlCLENBQUMsQ0FBQztRQUNoRCxDQUFDO0lBQ0gsQ0FBQztJQUVEOztPQUVHO0lBQ0ksTUFBTTtRQUNYLElBQUksQ0FBQyxJQUFJLENBQUMsVUFBVSxFQUFFLENBQUM7WUFDckIsSUFBSSxDQUFDLGVBQWUsRUFBRSxDQUFDO1lBQ3ZCLE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLDBCQUEwQixDQUFDLENBQUM7UUFDakQsQ0FBQztJQUNILENBQUM7SUFFRDs7O09BR0c7SUFDSSxLQUFLLENBQUMsZUFBZSxDQUFDLFNBQWlCLENBQUMsR0FBRyxFQUFFLEdBQUcsRUFBRSxHQUFHLEVBQUUsR0FBRyxJQUFJO1FBQ25FLE1BQU0sTUFBTSxHQUFHLElBQUksSUFBSSxDQUFDLElBQUksQ0FBQyxHQUFHLEVBQUUsR0FBRyxNQUFNLENBQUMsQ0FBQztRQUM3QyxJQUFJLFlBQVksR0FBRyxDQUFDLENBQUM7UUFFckIsaUJBQWlCO1FBQ2pCLEtBQUssTUFBTSxJQUFJLElBQUksSUFBSSxDQUFDLEtBQUssQ0FBQyxNQUFNLEVBQUUsRUFBRSxDQUFDO1lBQ3ZDLElBQUksQ0FBQyxXQUFXLEVBQUUsUUFBUSxDQUFDLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsSUFBSSxJQUFJLENBQUMsU0FBUyxHQUFHLE1BQU0sRUFBRSxDQUFDO2dCQUM3RSxjQUFjO2dCQUNkLE1BQU0sSUFBSSxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLENBQUM7Z0JBQy9CLFlBQVksRUFBRSxDQUFDO1lBQ2pCLENBQUM7UUFDSCxDQUFDO1FBRUQsTUFBTSxDQUFDLEdBQUcsQ0FBQyxNQUFNLEVBQUUsY0FBYyxZQUFZLFlBQVksQ0FBQyxDQUFDO1FBQzNELE9BQU8sWUFBWSxDQUFDO0lBQ3RCLENBQUM7SUFFRDs7T0FFRztJQUNJLEtBQUssQ0FBQyxRQUFRO1FBQ25CLE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLG9DQUFvQyxDQUFDLENBQUM7UUFFekQsa0JBQWtCO1FBQ2xCLElBQUksQ0FBQyxjQUFjLEVBQUUsQ0FBQztRQUV0QixnREFBZ0Q7UUFDaEQsSUFBSSxJQUFJLENBQUMsVUFBVSxFQUFFLENBQUM7WUFDcEIsYUFBYSxDQUFDLElBQUksQ0FBQyxVQUFVLENBQUMsQ0FBQztZQUMvQixJQUFJLENBQUMsVUFBVSxHQUFHLFNBQVMsQ0FBQztRQUM5QixDQUFDO1FBRUQsMkRBQTJEO1FBQzNELElBQUksSUFBSSxDQUFDLE9BQU8sQ0FBQyxXQUFXLEtBQUssTUFBTSxFQUFFLENBQUM7WUFDeEMsTUFBTSxhQUFhLEdBQW9CLEVBQUUsQ0FBQztZQUUxQyxLQUFLLE1BQU0sSUFBSSxJQUFJLElBQUksQ0FBQyxLQUFLLENBQUMsTUFBTSxFQUFFLEVBQUUsQ0FBQztnQkFDdkMsYUFBYSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUM7WUFDN0MsQ0FBQztZQUVELGtDQUFrQztZQUNsQyxNQUFNLE9BQU8sQ0FBQyxHQUFHLENBQUMsYUFBYSxDQUFDLENBQUM7UUFDbkMsQ0FBQztRQUVELGdDQUFnQztRQUNoQyxJQUFJLENBQUMsS0FBSyxDQUFDLEtBQUssRUFBRSxDQUFDO1FBRW5CLG9CQUFvQjtRQUNwQixJQUFJLENBQUMsV0FBVyxFQUFFLENBQUM7UUFFbkIsc0JBQXNCO1FBQ3RCLElBQUksQ0FBQyxJQUFJLENBQUMsVUFBVSxDQUFDLENBQUM7UUFDdEIsTUFBTSxDQUFDLEdBQUcsQ0FBQyxNQUFNLEVBQUUsNkNBQTZDLENBQUMsQ0FBQztJQUNwRSxDQUFDO0NBQ0YifQ== \ No newline at end of file diff --git a/dist_ts/mail/delivery/classes.delivery.system.d.ts b/dist_ts/mail/delivery/classes.delivery.system.d.ts deleted file mode 100644 index 5f896c0..0000000 --- a/dist_ts/mail/delivery/classes.delivery.system.d.ts +++ /dev/null @@ -1,186 +0,0 @@ -import { EventEmitter } from 'node:events'; -import { UnifiedDeliveryQueue, type IQueueItem } from './classes.delivery.queue.js'; -import type { UnifiedEmailServer } from '../routing/classes.unified.email.server.js'; -/** - * Delivery status enumeration - */ -export declare enum DeliveryStatus { - PENDING = "pending", - DELIVERING = "delivering", - DELIVERED = "delivered", - DEFERRED = "deferred", - FAILED = "failed" -} -/** - * Delivery handler interface - */ -export interface IDeliveryHandler { - deliver(item: IQueueItem): Promise; -} -/** - * Delivery options - */ -export interface IMultiModeDeliveryOptions { - connectionPoolSize?: number; - socketTimeout?: number; - concurrentDeliveries?: number; - sendTimeout?: number; - verifyCertificates?: boolean; - tlsMinVersion?: string; - forwardHandler?: IDeliveryHandler; - deliveryHandler?: IDeliveryHandler; - processHandler?: IDeliveryHandler; - globalRateLimit?: number; - perPatternRateLimit?: Record; - processBounces?: boolean; - bounceHandler?: { - processSmtpFailure: (recipient: string, smtpResponse: string, options: any) => Promise; - }; - onDeliveryStart?: (item: IQueueItem) => Promise; - onDeliverySuccess?: (item: IQueueItem, result: any) => Promise; - onDeliveryFailed?: (item: IQueueItem, error: string) => Promise; -} -/** - * Delivery system statistics - */ -export interface IDeliveryStats { - activeDeliveries: number; - totalSuccessful: number; - totalFailed: number; - avgDeliveryTime: number; - byMode: { - forward: { - successful: number; - failed: number; - }; - mta: { - successful: number; - failed: number; - }; - process: { - successful: number; - failed: number; - }; - }; - rateLimiting: { - currentRate: number; - globalLimit: number; - throttled: number; - }; -} -/** - * Handles delivery for all email processing modes - */ -export declare class MultiModeDeliverySystem extends EventEmitter { - private queue; - private options; - private stats; - private deliveryTimes; - private activeDeliveries; - private running; - private throttled; - private rateLimitLastCheck; - private rateLimitCounter; - private emailServer?; - /** - * Create a new multi-mode delivery system - * @param queue Unified delivery queue - * @param options Delivery options - * @param emailServer Optional reference to unified email server for SmtpClient access - */ - constructor(queue: UnifiedDeliveryQueue, options: IMultiModeDeliveryOptions, emailServer?: UnifiedEmailServer); - /** - * Start the delivery system - */ - start(): Promise; - /** - * Stop the delivery system - */ - stop(): Promise; - /** - * Process ready items from the queue - * @param items Queue items ready for processing - */ - private processItems; - /** - * Deliver an item from the queue - * @param item Queue item to deliver - */ - private deliverItem; - /** - * Default handler for forward mode delivery - * @param item Queue item - */ - private handleForwardDelivery; - /** - * Legacy forward delivery using raw sockets (fallback) - * @param item Queue item - */ - private handleForwardDeliveryLegacy; - /** - * Complete the SMTP exchange after connection and initial setup - * @param socket Network socket - * @param email Email to send - * @param rule Domain rule - */ - private completeSMTPExchange; - /** - * Default handler for MTA mode delivery - * @param item Queue item - */ - private handleMtaDelivery; - /** - * Default handler for process mode delivery - * @param item Queue item - */ - private handleProcessDelivery; - /** - * Get file extension from filename - */ - private getFileExtension; - /** - * Apply DKIM signing to an email - */ - private applyDkimSigning; - /** - * Format email for SMTP transmission - * @param email Email to format - */ - private getFormattedEmail; - /** - * Send SMTP command and wait for response - * @param socket Socket connection - * @param command SMTP command to send - */ - private smtpCommand; - /** - * Send SMTP DATA command with content - * @param socket Socket connection - * @param data Email content to send - */ - private smtpData; - /** - * Upgrade socket to TLS - * @param socket Socket connection - * @param hostname Target hostname for TLS - */ - private upgradeTls; - /** - * Update delivery time statistics - */ - private updateDeliveryTimeStats; - /** - * Check if rate limit is exceeded - * @returns True if rate limited, false otherwise - */ - private checkRateLimit; - /** - * Update delivery options - * @param options New options - */ - updateOptions(options: Partial): void; - /** - * Get delivery statistics - */ - getStats(): IDeliveryStats; -} diff --git a/dist_ts/mail/delivery/classes.delivery.system.js b/dist_ts/mail/delivery/classes.delivery.system.js deleted file mode 100644 index e9d3aba..0000000 --- a/dist_ts/mail/delivery/classes.delivery.system.js +++ /dev/null @@ -1,846 +0,0 @@ -import * as plugins from '../../plugins.js'; -import { EventEmitter } from 'node:events'; -import * as net from 'node:net'; -import * as tls from 'node:tls'; -import { logger } from '../../logger.js'; -import { SecurityLogger, SecurityLogLevel, SecurityEventType } from '../../security/index.js'; -import { UnifiedDeliveryQueue } from './classes.delivery.queue.js'; -import { RustSecurityBridge } from '../../security/classes.rustsecuritybridge.js'; -/** - * Delivery status enumeration - */ -export var DeliveryStatus; -(function (DeliveryStatus) { - DeliveryStatus["PENDING"] = "pending"; - DeliveryStatus["DELIVERING"] = "delivering"; - DeliveryStatus["DELIVERED"] = "delivered"; - DeliveryStatus["DEFERRED"] = "deferred"; - DeliveryStatus["FAILED"] = "failed"; -})(DeliveryStatus || (DeliveryStatus = {})); -/** - * Handles delivery for all email processing modes - */ -export class MultiModeDeliverySystem extends EventEmitter { - queue; - options; - stats; - deliveryTimes = []; - activeDeliveries = new Set(); - running = false; - throttled = false; - rateLimitLastCheck = Date.now(); - rateLimitCounter = 0; - emailServer; - /** - * Create a new multi-mode delivery system - * @param queue Unified delivery queue - * @param options Delivery options - * @param emailServer Optional reference to unified email server for SmtpClient access - */ - constructor(queue, options, emailServer) { - super(); - this.queue = queue; - this.emailServer = emailServer; - // Set default options - this.options = { - connectionPoolSize: options.connectionPoolSize || 10, - socketTimeout: options.socketTimeout || 30000, // 30 seconds - concurrentDeliveries: options.concurrentDeliveries || 10, - sendTimeout: options.sendTimeout || 60000, // 1 minute - verifyCertificates: options.verifyCertificates !== false, // Default to true - tlsMinVersion: options.tlsMinVersion || 'TLSv1.2', - forwardHandler: options.forwardHandler || { - deliver: this.handleForwardDelivery.bind(this) - }, - deliveryHandler: options.deliveryHandler || { - deliver: this.handleMtaDelivery.bind(this) - }, - processHandler: options.processHandler || { - deliver: this.handleProcessDelivery.bind(this) - }, - globalRateLimit: options.globalRateLimit || 100, // 100 emails per minute - perPatternRateLimit: options.perPatternRateLimit || {}, - processBounces: options.processBounces !== false, // Default to true - bounceHandler: options.bounceHandler || null, - onDeliveryStart: options.onDeliveryStart || (async () => { }), - onDeliverySuccess: options.onDeliverySuccess || (async () => { }), - onDeliveryFailed: options.onDeliveryFailed || (async () => { }) - }; - // Initialize statistics - this.stats = { - activeDeliveries: 0, - totalSuccessful: 0, - totalFailed: 0, - avgDeliveryTime: 0, - byMode: { - forward: { - successful: 0, - failed: 0 - }, - mta: { - successful: 0, - failed: 0 - }, - process: { - successful: 0, - failed: 0 - } - }, - rateLimiting: { - currentRate: 0, - globalLimit: this.options.globalRateLimit, - throttled: 0 - } - }; - // Set up event listeners - this.queue.on('itemsReady', this.processItems.bind(this)); - } - /** - * Start the delivery system - */ - async start() { - logger.log('info', 'Starting MultiModeDeliverySystem'); - if (this.running) { - logger.log('warn', 'MultiModeDeliverySystem is already running'); - return; - } - this.running = true; - // Emit started event - this.emit('started'); - logger.log('info', 'MultiModeDeliverySystem started successfully'); - } - /** - * Stop the delivery system - */ - async stop() { - logger.log('info', 'Stopping MultiModeDeliverySystem'); - if (!this.running) { - logger.log('warn', 'MultiModeDeliverySystem is already stopped'); - return; - } - this.running = false; - // Wait for active deliveries to complete - if (this.activeDeliveries.size > 0) { - logger.log('info', `Waiting for ${this.activeDeliveries.size} active deliveries to complete`); - // Wait for a maximum of 30 seconds - await new Promise(resolve => { - const checkInterval = setInterval(() => { - if (this.activeDeliveries.size === 0) { - clearInterval(checkInterval); - clearTimeout(forceTimeout); - resolve(); - } - }, 1000); - // Force resolve after 30 seconds - const forceTimeout = setTimeout(() => { - clearInterval(checkInterval); - resolve(); - }, 30000); - }); - } - // Emit stopped event - this.emit('stopped'); - logger.log('info', 'MultiModeDeliverySystem stopped successfully'); - } - /** - * Process ready items from the queue - * @param items Queue items ready for processing - */ - async processItems(items) { - if (!this.running) { - return; - } - // Check if we're already at max concurrent deliveries - if (this.activeDeliveries.size >= this.options.concurrentDeliveries) { - logger.log('debug', `Already at max concurrent deliveries (${this.activeDeliveries.size})`); - return; - } - // Check rate limiting - if (this.checkRateLimit()) { - logger.log('debug', 'Rate limit exceeded, throttling deliveries'); - return; - } - // Calculate how many more deliveries we can start - const availableSlots = this.options.concurrentDeliveries - this.activeDeliveries.size; - const itemsToProcess = items.slice(0, availableSlots); - if (itemsToProcess.length === 0) { - return; - } - logger.log('info', `Processing ${itemsToProcess.length} items for delivery`); - // Process each item - for (const item of itemsToProcess) { - // Mark as processing - await this.queue.markProcessing(item.id); - // Add to active deliveries - this.activeDeliveries.add(item.id); - this.stats.activeDeliveries = this.activeDeliveries.size; - // Deliver asynchronously - this.deliverItem(item).catch(err => { - logger.log('error', `Unhandled error in delivery: ${err.message}`); - }); - } - // Update statistics - this.emit('statsUpdated', this.stats); - } - /** - * Deliver an item from the queue - * @param item Queue item to deliver - */ - async deliverItem(item) { - const startTime = Date.now(); - try { - // Call delivery start hook - await this.options.onDeliveryStart(item); - // Emit delivery start event - this.emit('deliveryStart', item); - logger.log('info', `Starting delivery of item ${item.id}, mode: ${item.processingMode}`); - // Choose the appropriate handler based on mode - let result; - switch (item.processingMode) { - case 'forward': - result = await this.options.forwardHandler.deliver(item); - break; - case 'mta': - result = await this.options.deliveryHandler.deliver(item); - break; - case 'process': - result = await this.options.processHandler.deliver(item); - break; - default: - throw new Error(`Unknown processing mode: ${item.processingMode}`); - } - // Mark as delivered - await this.queue.markDelivered(item.id); - // Update statistics - this.stats.totalSuccessful++; - this.stats.byMode[item.processingMode].successful++; - // Calculate delivery time - const deliveryTime = Date.now() - startTime; - this.deliveryTimes.push(deliveryTime); - this.updateDeliveryTimeStats(); - // Call delivery success hook - await this.options.onDeliverySuccess(item, result); - // Emit delivery success event - this.emit('deliverySuccess', item, result); - logger.log('info', `Item ${item.id} delivered successfully in ${deliveryTime}ms`); - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.INFO, - type: SecurityEventType.EMAIL_DELIVERY, - message: 'Email delivery successful', - details: { - itemId: item.id, - mode: item.processingMode, - routeName: item.route?.name || 'unknown', - deliveryTime - }, - success: true - }); - } - catch (error) { - // Calculate delivery attempt time even for failures - const deliveryTime = Date.now() - startTime; - // Mark as failed - await this.queue.markFailed(item.id, error.message); - // Update statistics - this.stats.totalFailed++; - this.stats.byMode[item.processingMode].failed++; - // Call delivery failed hook - await this.options.onDeliveryFailed(item, error.message); - // Process as bounce if enabled and we have a bounce handler - if (this.options.processBounces && this.options.bounceHandler) { - try { - const email = item.processingResult; - // Extract recipient and error message - // For multiple recipients, we'd need more sophisticated parsing - const recipient = email.to.length > 0 ? email.to[0] : ''; - if (recipient) { - logger.log('info', `Processing delivery failure as bounce for recipient ${recipient}`); - // Process SMTP failure through bounce handler - await this.options.bounceHandler.processSmtpFailure(recipient, error.message, { - sender: email.from, - originalEmailId: item.id, - headers: email.headers - }); - logger.log('info', `Bounce record created for failed delivery to ${recipient}`); - } - } - catch (bounceError) { - logger.log('error', `Failed to process bounce: ${bounceError.message}`); - } - } - // Emit delivery failed event - this.emit('deliveryFailed', item, error); - logger.log('error', `Item ${item.id} delivery failed: ${error.message}`); - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.ERROR, - type: SecurityEventType.EMAIL_DELIVERY, - message: 'Email delivery failed', - details: { - itemId: item.id, - mode: item.processingMode, - routeName: item.route?.name || 'unknown', - error: error.message, - deliveryTime - }, - success: false - }); - } - finally { - // Remove from active deliveries - this.activeDeliveries.delete(item.id); - this.stats.activeDeliveries = this.activeDeliveries.size; - // Update statistics - this.emit('statsUpdated', this.stats); - } - } - /** - * Default handler for forward mode delivery - * @param item Queue item - */ - async handleForwardDelivery(item) { - logger.log('info', `Forward delivery for item ${item.id}`); - const email = item.processingResult; - const route = item.route; - // Get target server information - const targetServer = route?.action.forward?.host; - const targetPort = route?.action.forward?.port || 25; - const useTls = false; // TLS configuration can be enhanced later - if (!targetServer) { - throw new Error('No target server configured for forward mode'); - } - logger.log('info', `Forwarding email to ${targetServer}:${targetPort}, TLS: ${useTls}`); - try { - // Get SMTP client from email server if available - if (!this.emailServer) { - // Fall back to raw socket implementation if no email server - logger.log('warn', 'No email server available, falling back to raw socket implementation'); - return this.handleForwardDeliveryLegacy(item); - } - // Get SMTP client from UnifiedEmailServer - const smtpClient = this.emailServer.getSmtpClient(targetServer, targetPort); - // Apply DKIM signing if configured in the route - if (item.route?.action.options?.mtaOptions?.dkimSign) { - await this.applyDkimSigning(email, item.route.action.options.mtaOptions); - } - // Send the email using SmtpClient - const result = await smtpClient.sendMail(email); - if (result.success) { - logger.log('info', `Email forwarded successfully to ${targetServer}:${targetPort}`); - return { - targetServer: targetServer, - targetPort: targetPort, - recipients: result.acceptedRecipients.length, - messageId: result.messageId, - rejectedRecipients: result.rejectedRecipients - }; - } - else { - throw new Error(result.error?.message || 'Failed to forward email'); - } - } - catch (error) { - logger.log('error', `Failed to forward email: ${error.message}`); - throw error; - } - } - /** - * Legacy forward delivery using raw sockets (fallback) - * @param item Queue item - */ - async handleForwardDeliveryLegacy(item) { - const email = item.processingResult; - const route = item.route; - // Get target server information - const targetServer = route?.action.forward?.host; - const targetPort = route?.action.forward?.port || 25; - const useTls = false; // TLS configuration can be enhanced later - if (!targetServer) { - throw new Error('No target server configured for forward mode'); - } - // Create a socket connection to the target server - const socket = new net.Socket(); - // Set timeout - socket.setTimeout(this.options.socketTimeout); - try { - // Connect to the target server - await new Promise((resolve, reject) => { - // Handle connection events - socket.on('connect', () => { - logger.log('debug', `Connected to ${targetServer}:${targetPort}`); - resolve(); - }); - socket.on('timeout', () => { - reject(new Error(`Connection timeout to ${targetServer}:${targetPort}`)); - }); - socket.on('error', (err) => { - reject(new Error(`Connection error to ${targetServer}:${targetPort}: ${err.message}`)); - }); - // Connect to the server - socket.connect({ - host: targetServer, - port: targetPort - }); - }); - // Send EHLO - await this.smtpCommand(socket, `EHLO ${route?.action.options?.mtaOptions?.domain || 'localhost'}`); - // Start TLS if required - if (useTls) { - await this.smtpCommand(socket, 'STARTTLS'); - // Upgrade to TLS - const tlsSocket = await this.upgradeTls(socket, targetServer); - // Send EHLO again after STARTTLS - await this.smtpCommand(tlsSocket, `EHLO ${route?.action.options?.mtaOptions?.domain || 'localhost'}`); - // Use tlsSocket for remaining commands - return this.completeSMTPExchange(tlsSocket, email, route); - } - // Complete the SMTP exchange - return this.completeSMTPExchange(socket, email, route); - } - catch (error) { - logger.log('error', `Failed to forward email: ${error.message}`); - // Close the connection - socket.destroy(); - throw error; - } - } - /** - * Complete the SMTP exchange after connection and initial setup - * @param socket Network socket - * @param email Email to send - * @param rule Domain rule - */ - async completeSMTPExchange(socket, email, route) { - try { - // Authenticate if credentials provided - if (route?.action?.forward?.auth?.user && route?.action?.forward?.auth?.pass) { - // Send AUTH LOGIN - await this.smtpCommand(socket, 'AUTH LOGIN'); - // Send username (base64) - const username = Buffer.from(route.action.forward.auth.user).toString('base64'); - await this.smtpCommand(socket, username); - // Send password (base64) - const password = Buffer.from(route.action.forward.auth.pass).toString('base64'); - await this.smtpCommand(socket, password); - } - // Send MAIL FROM - await this.smtpCommand(socket, `MAIL FROM:<${email.from}>`); - // Send RCPT TO for each recipient - for (const recipient of email.getAllRecipients()) { - await this.smtpCommand(socket, `RCPT TO:<${recipient}>`); - } - // Send DATA - await this.smtpCommand(socket, 'DATA'); - // Send email content (simplified) - const emailContent = await this.getFormattedEmail(email); - await this.smtpData(socket, emailContent); - // Send QUIT - await this.smtpCommand(socket, 'QUIT'); - // Close the connection - socket.end(); - logger.log('info', `Email forwarded successfully to ${route?.action?.forward?.host}:${route?.action?.forward?.port || 25}`); - return { - targetServer: route?.action?.forward?.host, - targetPort: route?.action?.forward?.port || 25, - recipients: email.getAllRecipients().length - }; - } - catch (error) { - logger.log('error', `Failed to forward email: ${error.message}`); - // Close the connection - socket.destroy(); - throw error; - } - } - /** - * Default handler for MTA mode delivery - * @param item Queue item - */ - async handleMtaDelivery(item) { - logger.log('info', `MTA delivery for item ${item.id}`); - const email = item.processingResult; - const route = item.route; - try { - // Apply DKIM signing if configured in the route - if (item.route?.action.options?.mtaOptions?.dkimSign) { - await this.applyDkimSigning(email, item.route.action.options.mtaOptions); - } - // In a full implementation, this would use the MTA service - // For now, we'll simulate a successful delivery - logger.log('info', `Email processed by MTA: ${email.subject} to ${email.getAllRecipients().join(', ')}`); - // Note: The MTA implementation would handle actual local delivery - // Simulate successful delivery - return { - recipients: email.getAllRecipients().length, - subject: email.subject, - dkimSigned: !!item.route?.action.options?.mtaOptions?.dkimSign - }; - } - catch (error) { - logger.log('error', `Failed to process email in MTA mode: ${error.message}`); - throw error; - } - } - /** - * Default handler for process mode delivery - * @param item Queue item - */ - async handleProcessDelivery(item) { - logger.log('info', `Process delivery for item ${item.id}`); - const email = item.processingResult; - const route = item.route; - try { - // Apply content scanning if enabled - if (route?.action.options?.contentScanning && route?.action.options?.scanners && route.action.options.scanners.length > 0) { - logger.log('info', 'Performing content scanning'); - // Apply each scanner - for (const scanner of route.action.options.scanners) { - switch (scanner.type) { - case 'spam': - logger.log('info', 'Scanning for spam content'); - // Implement spam scanning - break; - case 'virus': - logger.log('info', 'Scanning for virus content'); - // Implement virus scanning - break; - case 'attachment': - logger.log('info', 'Scanning attachments'); - // Check for blocked extensions - if (scanner.blockedExtensions && scanner.blockedExtensions.length > 0) { - for (const attachment of email.attachments) { - const ext = this.getFileExtension(attachment.filename); - if (scanner.blockedExtensions.includes(ext)) { - if (scanner.action === 'reject') { - throw new Error(`Blocked attachment type: ${ext}`); - } - else { // tag - email.addHeader('X-Attachment-Warning', `Potentially unsafe attachment: ${attachment.filename}`); - } - } - } - } - break; - } - } - } - // Apply transformations if defined - if (route?.action.options?.transformations && route?.action.options?.transformations.length > 0) { - logger.log('info', 'Applying email transformations'); - for (const transform of route.action.options.transformations) { - switch (transform.type) { - case 'addHeader': - if (transform.header && transform.value) { - email.addHeader(transform.header, transform.value); - } - break; - } - } - } - // Apply DKIM signing if configured (after all transformations) - if (item.route?.action.options?.mtaOptions?.dkimSign || item.route?.action.process?.dkim) { - await this.applyDkimSigning(email, item.route.action.options?.mtaOptions || {}); - } - logger.log('info', `Email successfully processed in store-and-forward mode`); - // Simulate successful delivery - return { - recipients: email.getAllRecipients().length, - subject: email.subject, - scanned: !!route?.action.options?.contentScanning, - transformed: !!(route?.action.options?.transformations && route?.action.options?.transformations.length > 0), - dkimSigned: !!(item.route?.action.options?.mtaOptions?.dkimSign || item.route?.action.process?.dkim) - }; - } - catch (error) { - logger.log('error', `Failed to process email: ${error.message}`); - throw error; - } - } - /** - * Get file extension from filename - */ - getFileExtension(filename) { - return filename.substring(filename.lastIndexOf('.')).toLowerCase(); - } - /** - * Apply DKIM signing to an email - */ - async applyDkimSigning(email, mtaOptions) { - if (!this.emailServer) { - logger.log('warn', 'Cannot apply DKIM signing without email server reference'); - return; - } - const domainName = mtaOptions.dkimOptions?.domainName || email.from.split('@')[1]; - const keySelector = mtaOptions.dkimOptions?.keySelector || 'default'; - try { - // Ensure DKIM keys exist for the domain - await this.emailServer.dkimCreator.handleDKIMKeysForDomain(domainName); - // Get the private key - const dkimPrivateKey = (await this.emailServer.dkimCreator.readDKIMKeys(domainName)).privateKey; - // Convert Email to raw format for signing - const rawEmail = email.toRFC822String(); - // Sign via Rust bridge - const bridge = RustSecurityBridge.getInstance(); - const signResult = await bridge.signDkim({ - rawMessage: rawEmail, - domain: domainName, - selector: keySelector, - privateKey: dkimPrivateKey, - }); - if (signResult.header) { - email.addHeader('DKIM-Signature', signResult.header); - logger.log('info', `Successfully added DKIM signature for ${domainName}`); - } - } - catch (error) { - logger.log('error', `Failed to apply DKIM signature: ${error.message}`); - // Don't throw - allow email to be sent without DKIM if signing fails - } - } - /** - * Format email for SMTP transmission - * @param email Email to format - */ - async getFormattedEmail(email) { - // This is a simplified implementation - // In a full implementation, this would use proper MIME formatting - let content = ''; - // Add headers - content += `From: ${email.from}\r\n`; - content += `To: ${email.to.join(', ')}\r\n`; - content += `Subject: ${email.subject}\r\n`; - // Add additional headers - for (const [name, value] of Object.entries(email.headers || {})) { - content += `${name}: ${value}\r\n`; - } - // Add content type for multipart - if (email.attachments && email.attachments.length > 0) { - const boundary = `----_=_NextPart_${Math.random().toString(36).substr(2)}`; - content += `MIME-Version: 1.0\r\n`; - content += `Content-Type: multipart/mixed; boundary="${boundary}"\r\n`; - content += `\r\n`; - // Add text part - content += `--${boundary}\r\n`; - content += `Content-Type: text/plain; charset="UTF-8"\r\n`; - content += `\r\n`; - content += `${email.text}\r\n`; - // Add HTML part if present - if (email.html) { - content += `--${boundary}\r\n`; - content += `Content-Type: text/html; charset="UTF-8"\r\n`; - content += `\r\n`; - content += `${email.html}\r\n`; - } - // Add attachments - for (const attachment of email.attachments) { - content += `--${boundary}\r\n`; - content += `Content-Type: ${attachment.contentType || 'application/octet-stream'}; name="${attachment.filename}"\r\n`; - content += `Content-Disposition: attachment; filename="${attachment.filename}"\r\n`; - content += `Content-Transfer-Encoding: base64\r\n`; - content += `\r\n`; - // Add base64 encoded content - const base64Content = attachment.content.toString('base64'); - // Split into lines of 76 characters - for (let i = 0; i < base64Content.length; i += 76) { - content += base64Content.substring(i, i + 76) + '\r\n'; - } - } - // End boundary - content += `--${boundary}--\r\n`; - } - else { - // Simple email with just text - content += `Content-Type: text/plain; charset="UTF-8"\r\n`; - content += `\r\n`; - content += `${email.text}\r\n`; - } - return content; - } - /** - * Send SMTP command and wait for response - * @param socket Socket connection - * @param command SMTP command to send - */ - async smtpCommand(socket, command) { - return new Promise((resolve, reject) => { - const onData = (data) => { - const response = data.toString().trim(); - // Clean up listeners - socket.removeListener('data', onData); - socket.removeListener('error', onError); - socket.removeListener('timeout', onTimeout); - // Check response code - if (response.charAt(0) === '2' || response.charAt(0) === '3') { - resolve(response); - } - else { - reject(new Error(`SMTP error: ${response}`)); - } - }; - const onError = (err) => { - // Clean up listeners - socket.removeListener('data', onData); - socket.removeListener('error', onError); - socket.removeListener('timeout', onTimeout); - reject(err); - }; - const onTimeout = () => { - // Clean up listeners - socket.removeListener('data', onData); - socket.removeListener('error', onError); - socket.removeListener('timeout', onTimeout); - reject(new Error('SMTP command timeout')); - }; - // Set up listeners - socket.once('data', onData); - socket.once('error', onError); - socket.once('timeout', onTimeout); - // Send command - socket.write(command + '\r\n'); - }); - } - /** - * Send SMTP DATA command with content - * @param socket Socket connection - * @param data Email content to send - */ - async smtpData(socket, data) { - return new Promise((resolve, reject) => { - const onData = (responseData) => { - const response = responseData.toString().trim(); - // Clean up listeners - socket.removeListener('data', onData); - socket.removeListener('error', onError); - socket.removeListener('timeout', onTimeout); - // Check response code - if (response.charAt(0) === '2') { - resolve(response); - } - else { - reject(new Error(`SMTP error: ${response}`)); - } - }; - const onError = (err) => { - // Clean up listeners - socket.removeListener('data', onData); - socket.removeListener('error', onError); - socket.removeListener('timeout', onTimeout); - reject(err); - }; - const onTimeout = () => { - // Clean up listeners - socket.removeListener('data', onData); - socket.removeListener('error', onError); - socket.removeListener('timeout', onTimeout); - reject(new Error('SMTP data timeout')); - }; - // Set up listeners - socket.once('data', onData); - socket.once('error', onError); - socket.once('timeout', onTimeout); - // Send data and end with CRLF.CRLF - socket.write(data + '\r\n.\r\n'); - }); - } - /** - * Upgrade socket to TLS - * @param socket Socket connection - * @param hostname Target hostname for TLS - */ - async upgradeTls(socket, hostname) { - return new Promise((resolve, reject) => { - const tlsOptions = { - socket, - servername: hostname, - rejectUnauthorized: this.options.verifyCertificates, - minVersion: this.options.tlsMinVersion - }; - const tlsSocket = tls.connect(tlsOptions); - tlsSocket.once('secureConnect', () => { - resolve(tlsSocket); - }); - tlsSocket.once('error', (err) => { - reject(new Error(`TLS error: ${err.message}`)); - }); - tlsSocket.setTimeout(this.options.socketTimeout); - tlsSocket.once('timeout', () => { - reject(new Error('TLS connection timeout')); - }); - }); - } - /** - * Update delivery time statistics - */ - updateDeliveryTimeStats() { - if (this.deliveryTimes.length === 0) - return; - // Keep only the last 1000 delivery times - if (this.deliveryTimes.length > 1000) { - this.deliveryTimes = this.deliveryTimes.slice(-1000); - } - // Calculate average - const sum = this.deliveryTimes.reduce((acc, time) => acc + time, 0); - this.stats.avgDeliveryTime = sum / this.deliveryTimes.length; - } - /** - * Check if rate limit is exceeded - * @returns True if rate limited, false otherwise - */ - checkRateLimit() { - const now = Date.now(); - const elapsed = now - this.rateLimitLastCheck; - // Reset counter if more than a minute has passed - if (elapsed >= 60000) { - this.rateLimitLastCheck = now; - this.rateLimitCounter = 0; - this.throttled = false; - this.stats.rateLimiting.currentRate = 0; - return false; - } - // Check if we're already throttled - if (this.throttled) { - return true; - } - // Increment counter - this.rateLimitCounter++; - // Calculate current rate (emails per minute) - const rate = (this.rateLimitCounter / elapsed) * 60000; - this.stats.rateLimiting.currentRate = rate; - // Check if rate limit is exceeded - if (rate > this.options.globalRateLimit) { - this.throttled = true; - this.stats.rateLimiting.throttled++; - // Schedule throttle reset - const resetDelay = 60000 - elapsed; - setTimeout(() => { - this.throttled = false; - this.rateLimitLastCheck = Date.now(); - this.rateLimitCounter = 0; - this.stats.rateLimiting.currentRate = 0; - }, resetDelay); - return true; - } - return false; - } - /** - * Update delivery options - * @param options New options - */ - updateOptions(options) { - this.options = { - ...this.options, - ...options - }; - // Update rate limit statistics - if (options.globalRateLimit) { - this.stats.rateLimiting.globalLimit = options.globalRateLimit; - } - logger.log('info', 'MultiModeDeliverySystem options updated'); - } - /** - * Get delivery statistics - */ - getStats() { - return { ...this.stats }; - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5kZWxpdmVyeS5zeXN0ZW0uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi90cy9tYWlsL2RlbGl2ZXJ5L2NsYXNzZXMuZGVsaXZlcnkuc3lzdGVtLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sS0FBSyxPQUFPLE1BQU0sa0JBQWtCLENBQUM7QUFDNUMsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGFBQWEsQ0FBQztBQUMzQyxPQUFPLEtBQUssR0FBRyxNQUFNLFVBQVUsQ0FBQztBQUNoQyxPQUFPLEtBQUssR0FBRyxNQUFNLFVBQVUsQ0FBQztBQUNoQyxPQUFPLEVBQUUsTUFBTSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDekMsT0FBTyxFQUNMLGNBQWMsRUFDZCxnQkFBZ0IsRUFDaEIsaUJBQWlCLEVBQ2xCLE1BQU0seUJBQXlCLENBQUM7QUFDakMsT0FBTyxFQUFFLG9CQUFvQixFQUFtQixNQUFNLDZCQUE2QixDQUFDO0FBSXBGLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLDhDQUE4QyxDQUFDO0FBRWxGOztHQUVHO0FBQ0gsTUFBTSxDQUFOLElBQVksY0FNWDtBQU5ELFdBQVksY0FBYztJQUN4QixxQ0FBbUIsQ0FBQTtJQUNuQiwyQ0FBeUIsQ0FBQTtJQUN6Qix5Q0FBdUIsQ0FBQTtJQUN2Qix1Q0FBcUIsQ0FBQTtJQUNyQixtQ0FBaUIsQ0FBQTtBQUNuQixDQUFDLEVBTlcsY0FBYyxLQUFkLGNBQWMsUUFNekI7QUEyRUQ7O0dBRUc7QUFDSCxNQUFNLE9BQU8sdUJBQXdCLFNBQVEsWUFBWTtJQUMvQyxLQUFLLENBQXVCO0lBQzVCLE9BQU8sQ0FBc0M7SUFDN0MsS0FBSyxDQUFpQjtJQUN0QixhQUFhLEdBQWEsRUFBRSxDQUFDO0lBQzdCLGdCQUFnQixHQUFnQixJQUFJLEdBQUcsRUFBRSxDQUFDO0lBQzFDLE9BQU8sR0FBWSxLQUFLLENBQUM7SUFDekIsU0FBUyxHQUFZLEtBQUssQ0FBQztJ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 \ No newline at end of file diff --git a/dist_ts/mail/delivery/classes.emailsendjob.d.ts b/dist_ts/mail/delivery/classes.emailsendjob.d.ts deleted file mode 100644 index 0d16b8f..0000000 --- a/dist_ts/mail/delivery/classes.emailsendjob.d.ts +++ /dev/null @@ -1,84 +0,0 @@ -import * as plugins from '../../plugins.js'; -import { Email } from '../core/classes.email.js'; -import type { UnifiedEmailServer } from '../routing/classes.unified.email.server.js'; -export interface IEmailSendOptions { - maxRetries?: number; - retryDelay?: number; - connectionTimeout?: number; - tlsOptions?: plugins.tls.ConnectionOptions; - debugMode?: boolean; -} -export declare enum DeliveryStatus { - PENDING = "pending", - SENDING = "sending", - DELIVERED = "delivered", - FAILED = "failed", - DEFERRED = "deferred" -} -export interface DeliveryInfo { - status: DeliveryStatus; - attempts: number; - error?: Error; - lastAttempt?: Date; - nextAttempt?: Date; - mxServer?: string; - deliveryTime?: Date; - logs: string[]; -} -export declare class EmailSendJob { - emailServerRef: UnifiedEmailServer; - private email; - private mxServers; - private currentMxIndex; - private options; - deliveryInfo: DeliveryInfo; - constructor(emailServerRef: UnifiedEmailServer, emailArg: Email, options?: IEmailSendOptions); - /** - * Send the email to its recipients - */ - send(): Promise; - /** - * Validate the email before sending - */ - private validateEmail; - /** - * Resolve MX records for the recipient domain - */ - private resolveMxRecords; - /** - * Attempt to deliver the email with retries - */ - private attemptDelivery; - /** - * Connect to a specific MX server and send the email using SmtpClient - */ - private connectAndSend; - /** - * Record delivery event for monitoring - */ - private recordDeliveryEvent; - /** - * Check if an error represents a permanent failure - */ - private isPermanentFailure; - /** - * Resolve MX records for a domain - */ - private resolveMx; - /** - * Log a message with timestamp - */ - private log; - /** - * Save successful email to storage - */ - private saveSuccess; - /** - * Save failed email to storage - */ - private saveFailed; - /** - * Delay for specified milliseconds - */ - private delay; -} diff --git a/dist_ts/mail/delivery/classes.emailsendjob.js b/dist_ts/mail/delivery/classes.emailsendjob.js deleted file mode 100644 index 7aaa425..0000000 --- a/dist_ts/mail/delivery/classes.emailsendjob.js +++ /dev/null @@ -1,366 +0,0 @@ -import * as plugins from '../../plugins.js'; -import * as paths from '../../paths.js'; -import { Email } from '../core/classes.email.js'; -import { EmailSignJob } from './classes.emailsignjob.js'; -// Email delivery status -export var DeliveryStatus; -(function (DeliveryStatus) { - DeliveryStatus["PENDING"] = "pending"; - DeliveryStatus["SENDING"] = "sending"; - DeliveryStatus["DELIVERED"] = "delivered"; - DeliveryStatus["FAILED"] = "failed"; - DeliveryStatus["DEFERRED"] = "deferred"; // Temporary failure, will retry -})(DeliveryStatus || (DeliveryStatus = {})); -export class EmailSendJob { - emailServerRef; - email; - mxServers = []; - currentMxIndex = 0; - options; - deliveryInfo; - constructor(emailServerRef, emailArg, options = {}) { - this.email = emailArg; - this.emailServerRef = emailServerRef; - // Set default options - this.options = { - maxRetries: options.maxRetries || 3, - retryDelay: options.retryDelay || 30000, // 30 seconds - connectionTimeout: options.connectionTimeout || 60000, // 60 seconds - tlsOptions: options.tlsOptions || {}, - debugMode: options.debugMode || false - }; - // Initialize delivery info - this.deliveryInfo = { - status: DeliveryStatus.PENDING, - attempts: 0, - logs: [] - }; - } - /** - * Send the email to its recipients - */ - async send() { - try { - // Check if the email is valid before attempting to send - this.validateEmail(); - // Resolve MX records for the recipient domain - await this.resolveMxRecords(); - // Try to send the email - return await this.attemptDelivery(); - } - catch (error) { - this.log(`Critical error in send process: ${error.message}`); - this.deliveryInfo.status = DeliveryStatus.FAILED; - this.deliveryInfo.error = error; - // Save failed email for potential future retry or analysis - await this.saveFailed(); - return DeliveryStatus.FAILED; - } - } - /** - * Validate the email before sending - */ - validateEmail() { - if (!this.email.to || this.email.to.length === 0) { - throw new Error('No recipients specified'); - } - if (!this.email.from) { - throw new Error('No sender specified'); - } - const fromDomain = this.email.getFromDomain(); - if (!fromDomain) { - throw new Error('Invalid sender domain'); - } - } - /** - * Resolve MX records for the recipient domain - */ - async resolveMxRecords() { - const domain = this.email.getPrimaryRecipient()?.split('@')[1]; - if (!domain) { - throw new Error('Invalid recipient domain'); - } - this.log(`Resolving MX records for domain: ${domain}`); - try { - const addresses = await this.resolveMx(domain); - // Sort by priority (lowest number = highest priority) - addresses.sort((a, b) => a.priority - b.priority); - this.mxServers = addresses.map(mx => mx.exchange); - this.log(`Found ${this.mxServers.length} MX servers: ${this.mxServers.join(', ')}`); - if (this.mxServers.length === 0) { - throw new Error(`No MX records found for domain: ${domain}`); - } - } - catch (error) { - this.log(`Failed to resolve MX records: ${error.message}`); - throw new Error(`MX lookup failed for ${domain}: ${error.message}`); - } - } - /** - * Attempt to deliver the email with retries - */ - async attemptDelivery() { - while (this.deliveryInfo.attempts < this.options.maxRetries) { - this.deliveryInfo.attempts++; - this.deliveryInfo.lastAttempt = new Date(); - this.deliveryInfo.status = DeliveryStatus.SENDING; - try { - this.log(`Delivery attempt ${this.deliveryInfo.attempts} of ${this.options.maxRetries}`); - // Try each MX server in order of priority - while (this.currentMxIndex < this.mxServers.length) { - const currentMx = this.mxServers[this.currentMxIndex]; - this.deliveryInfo.mxServer = currentMx; - try { - this.log(`Attempting delivery to MX server: ${currentMx}`); - await this.connectAndSend(currentMx); - // If we get here, email was sent successfully - this.deliveryInfo.status = DeliveryStatus.DELIVERED; - this.deliveryInfo.deliveryTime = new Date(); - this.log(`Email delivered successfully to ${currentMx}`); - // Record delivery for sender reputation monitoring - this.recordDeliveryEvent('delivered'); - // Save successful email record - await this.saveSuccess(); - return DeliveryStatus.DELIVERED; - } - catch (error) { - this.log(`Failed to deliver to ${currentMx}: ${error.message}`); - this.currentMxIndex++; - // If this MX server failed, try the next one - if (this.currentMxIndex >= this.mxServers.length) { - throw error; // No more MX servers to try - } - } - } - throw new Error('All MX servers failed'); - } - catch (error) { - this.deliveryInfo.error = error; - // Check if this is a permanent failure - if (this.isPermanentFailure(error)) { - this.log('Permanent failure detected, not retrying'); - this.deliveryInfo.status = DeliveryStatus.FAILED; - // Record permanent failure for bounce management - this.recordDeliveryEvent('bounced', true); - await this.saveFailed(); - return DeliveryStatus.FAILED; - } - // This is a temporary failure - if (this.deliveryInfo.attempts < this.options.maxRetries) { - this.log(`Temporary failure, will retry in ${this.options.retryDelay}ms`); - this.deliveryInfo.status = DeliveryStatus.DEFERRED; - this.deliveryInfo.nextAttempt = new Date(Date.now() + this.options.retryDelay); - // Record temporary failure for monitoring - this.recordDeliveryEvent('deferred'); - // Reset MX server index for next retry - this.currentMxIndex = 0; - // Wait before retrying - await this.delay(this.options.retryDelay); - } - } - } - // If we get here, all retries failed - this.deliveryInfo.status = DeliveryStatus.FAILED; - await this.saveFailed(); - return DeliveryStatus.FAILED; - } - /** - * Connect to a specific MX server and send the email using SmtpClient - */ - async connectAndSend(mxServer) { - this.log(`Connecting to ${mxServer}:25`); - try { - // Check if IP warmup is enabled and get an IP to use - let localAddress = undefined; - try { - const fromDomain = this.email.getFromDomain(); - const bestIP = this.emailServerRef.getBestIPForSending({ - from: this.email.from, - to: this.email.getAllRecipients(), - domain: fromDomain, - isTransactional: this.email.priority === 'high' - }); - if (bestIP) { - this.log(`Using warmed-up IP ${bestIP} for sending`); - localAddress = bestIP; - // Record the send for warm-up tracking - this.emailServerRef.recordIPSend(bestIP); - } - } - catch (error) { - this.log(`Error selecting IP address: ${error.message}`); - } - // Get SMTP client from UnifiedEmailServer - const smtpClient = this.emailServerRef.getSmtpClient(mxServer, 25); - // Sign the email with DKIM if available - let signedEmail = this.email; - try { - const fromDomain = this.email.getFromDomain(); - if (fromDomain && this.emailServerRef.hasDkimKey(fromDomain)) { - // Convert email to RFC822 format for signing - const emailMessage = this.email.toRFC822String(); - // Create sign job with proper options - const emailSignJob = new EmailSignJob(this.emailServerRef, { - domain: fromDomain, - selector: 'default', // Using default selector - headers: {}, // Headers will be extracted from emailMessage - body: emailMessage - }); - // Get the DKIM signature header - const signatureHeader = await emailSignJob.getSignatureHeader(emailMessage); - // Add the signature to the email - if (signatureHeader) { - // For now, we'll use the email as-is since SmtpClient will handle DKIM - this.log(`Email ready for DKIM signing for domain: ${fromDomain}`); - } - } - } - catch (error) { - this.log(`Failed to prepare DKIM: ${error.message}`); - } - // Send the email using SmtpClient - const result = await smtpClient.sendMail(signedEmail); - if (result.success) { - this.log(`Email sent successfully: ${result.response}`); - // Record the send for reputation monitoring - this.recordDeliveryEvent('delivered'); - } - else { - throw new Error(result.error?.message || 'Failed to send email'); - } - } - catch (error) { - this.log(`Failed to send email via ${mxServer}: ${error.message}`); - throw error; - } - } - /** - * Record delivery event for monitoring - */ - recordDeliveryEvent(eventType, isHardBounce = false) { - try { - const domain = this.email.getFromDomain(); - if (domain) { - if (eventType === 'delivered') { - this.emailServerRef.recordDelivery(domain); - } - else if (eventType === 'bounced') { - // Get the receiving domain for bounce recording - let receivingDomain = null; - const primaryRecipient = this.email.getPrimaryRecipient(); - if (primaryRecipient) { - receivingDomain = primaryRecipient.split('@')[1]; - } - if (receivingDomain) { - this.emailServerRef.recordBounce(domain, receivingDomain, isHardBounce ? 'hard' : 'soft', this.deliveryInfo.error?.message || 'Unknown error'); - } - } - } - } - catch (error) { - this.log(`Failed to record delivery event: ${error.message}`); - } - } - /** - * Check if an error represents a permanent failure - */ - isPermanentFailure(error) { - const permanentFailurePatterns = [ - 'User unknown', - 'No such user', - 'Mailbox not found', - 'Invalid recipient', - 'Account disabled', - 'Account suspended', - 'Domain not found', - 'No such domain', - 'Invalid domain', - 'Relay access denied', - 'Access denied', - 'Blacklisted', - 'Blocked', - '550', // Permanent failure SMTP code - '551', - '552', - '553', - '554' - ]; - const errorMessage = error.message.toLowerCase(); - return permanentFailurePatterns.some(pattern => errorMessage.includes(pattern.toLowerCase())); - } - /** - * Resolve MX records for a domain - */ - resolveMx(domain) { - return new Promise((resolve, reject) => { - plugins.dns.resolveMx(domain, (err, addresses) => { - if (err) { - reject(err); - } - else { - resolve(addresses || []); - } - }); - }); - } - /** - * Log a message with timestamp - */ - log(message) { - const timestamp = new Date().toISOString(); - const logEntry = `[${timestamp}] ${message}`; - this.deliveryInfo.logs.push(logEntry); - if (this.options.debugMode) { - console.log(`[EmailSendJob] ${logEntry}`); - } - } - /** - * Save successful email to storage - */ - async saveSuccess() { - try { - // Use the existing email storage path - const emailContent = this.email.toRFC822String(); - const fileName = `${Date.now()}_${this.email.from}_to_${this.email.to[0]}_success.eml`; - const filePath = plugins.path.join(paths.sentEmailsDir, fileName); - await plugins.smartfs.directory(paths.sentEmailsDir).recursive().create(); - await plugins.smartfs.file(filePath).write(emailContent); - // Also save delivery info - const infoFileName = `${Date.now()}_${this.email.from}_to_${this.email.to[0]}_info.json`; - const infoPath = plugins.path.join(paths.sentEmailsDir, infoFileName); - await plugins.smartfs.file(infoPath).write(JSON.stringify(this.deliveryInfo, null, 2)); - this.log(`Email saved to ${fileName}`); - } - catch (error) { - this.log(`Failed to save email: ${error.message}`); - } - } - /** - * Save failed email to storage - */ - async saveFailed() { - try { - // Use the existing email storage path - const emailContent = this.email.toRFC822String(); - const fileName = `${Date.now()}_${this.email.from}_to_${this.email.to[0]}_failed.eml`; - const filePath = plugins.path.join(paths.failedEmailsDir, fileName); - await plugins.smartfs.directory(paths.failedEmailsDir).recursive().create(); - await plugins.smartfs.file(filePath).write(emailContent); - // Also save delivery info with error details - const infoFileName = `${Date.now()}_${this.email.from}_to_${this.email.to[0]}_error.json`; - const infoPath = plugins.path.join(paths.failedEmailsDir, infoFileName); - await plugins.smartfs.file(infoPath).write(JSON.stringify(this.deliveryInfo, null, 2)); - this.log(`Failed email saved to ${fileName}`); - } - catch (error) { - this.log(`Failed to save failed email: ${error.message}`); - } - } - /** - * Delay for specified milliseconds - */ - delay(ms) { - return new Promise(resolve => setTimeout(resolve, ms)); - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5lbWFpbHNlbmRqb2IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi90cy9tYWlsL2RlbGl2ZXJ5L2NsYXNzZXMuZW1haWxzZW5kam9iLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sS0FBSyxPQUFPLE1BQU0sa0JBQWtCLENBQUM7QUFDNUMsT0FBTyxLQUFLLEtBQUssTUFBTSxnQkFBZ0IsQ0FBQztBQUN4QyxPQUFPLEVBQUUsS0FBSyxFQUFFLE1BQU0sMEJBQTBCLENBQUM7QUFDakQsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBY3pELHdCQUF3QjtBQUN4QixNQUFNLENBQU4sSUFBWSxjQU1YO0FBTkQsV0FBWSxjQUFjO0lBQ3hCLHFDQUFtQixDQUFBO0lBQ25CLHFDQUFtQixDQUFBO0lBQ25CLHlDQUF1QixDQUFBO0lBQ3ZCLG1DQUFpQixDQUFBO0lBQ2pCLHVDQUFxQixDQUFBLENBQUMsZ0NBQWdDO0FBQ3hELENBQUMsRUFOVyxjQUFjLEtBQWQsY0FBYyxRQU16QjtBQWNELE1BQU0sT0FBTyxZQUFZO0lBQ3ZCLGNBQWMsQ0FBcUI7SUFDM0IsS0FBSyxDQUFRO0lBQ2IsU0FBUyxHQUFhLEVBQUUsQ0FBQztJQUN6QixjQUFjLEdBQUcsQ0FBQyxDQUFDO0lBQ25CLE9BQU8sQ0FBb0I7SUFDNUIsWUFBWSxDQUFlO0lBRWxDLFlBQVksY0FBa0MsRUFBRSxRQUFlLEVBQUUsVUFBNkIsRUFBRTtRQUM5RixJQUFJLENBQUMsS0FBSyxHQUFHLFFBQVEsQ0FBQztRQUN0QixJQUFJLENBQUMsY0FBYyxHQUFHLGNBQWMsQ0FBQztRQUVyQyxzQkFBc0I7UUFDdEIsSUFBSSxDQUFDLE9BQU8sR0FBRztZQUNiLFVBQVUsRUFBRSxPQUFPLENBQUMsVUFBVSxJQUFJLENBQUM7WUFDbkMsVUFBVSxFQUFFLE9BQU8sQ0FBQyxVQUFVLElBQUksS0FBSyxFQUFFLGFBQWE7WUFDdEQsaUJBQWlCLEVB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 \ No newline at end of file diff --git a/dist_ts/mail/delivery/classes.emailsignjob.d.ts b/dist_ts/mail/delivery/classes.emailsignjob.d.ts deleted file mode 100644 index 159f4a2..0000000 --- a/dist_ts/mail/delivery/classes.emailsignjob.d.ts +++ /dev/null @@ -1,18 +0,0 @@ -import type { UnifiedEmailServer } from '../routing/classes.unified.email.server.js'; -interface Headers { - [key: string]: string; -} -interface IEmailSignJobOptions { - domain: string; - selector: string; - headers: Headers; - body: string; -} -export declare class EmailSignJob { - emailServerRef: UnifiedEmailServer; - jobOptions: IEmailSignJobOptions; - constructor(emailServerRef: UnifiedEmailServer, options: IEmailSignJobOptions); - loadPrivateKey(): Promise; - getSignatureHeader(emailMessage: string): Promise; -} -export {}; diff --git a/dist_ts/mail/delivery/classes.emailsignjob.js b/dist_ts/mail/delivery/classes.emailsignjob.js deleted file mode 100644 index 869c681..0000000 --- a/dist_ts/mail/delivery/classes.emailsignjob.js +++ /dev/null @@ -1,26 +0,0 @@ -import * as plugins from '../../plugins.js'; -import { RustSecurityBridge } from '../../security/classes.rustsecuritybridge.js'; -export class EmailSignJob { - emailServerRef; - jobOptions; - constructor(emailServerRef, options) { - this.emailServerRef = emailServerRef; - this.jobOptions = options; - } - async loadPrivateKey() { - const keyInfo = await this.emailServerRef.dkimCreator.readDKIMKeys(this.jobOptions.domain); - return keyInfo.privateKey; - } - async getSignatureHeader(emailMessage) { - const privateKey = await this.loadPrivateKey(); - const bridge = RustSecurityBridge.getInstance(); - const signResult = await bridge.signDkim({ - rawMessage: emailMessage, - domain: this.jobOptions.domain, - selector: this.jobOptions.selector, - privateKey, - }); - return signResult.header; - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5lbWFpbHNpZ25qb2IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi90cy9tYWlsL2RlbGl2ZXJ5L2NsYXNzZXMuZW1haWxzaWduam9iLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sS0FBSyxPQUFPLE1BQU0sa0JBQWtCLENBQUM7QUFFNUMsT0FBTyxFQUFFLGtCQUFrQixFQUFFLE1BQU0sOENBQThDLENBQUM7QUFhbEYsTUFBTSxPQUFPLFlBQVk7SUFDdkIsY0FBYyxDQUFxQjtJQUNuQyxVQUFVLENBQXVCO0lBRWpDLFlBQVksY0FBa0MsRUFBRSxPQUE2QjtRQUMzRSxJQUFJLENBQUMsY0FBYyxHQUFHLGNBQWMsQ0FBQztRQUNyQyxJQUFJLENBQUMsVUFBVSxHQUFHLE9BQU8sQ0FBQztJQUM1QixDQUFDO0lBRUQsS0FBSyxDQUFDLGNBQWM7UUFDbEIsTUFBTSxPQUFPLEdBQUcsTUFBTSxJQUFJLENBQUMsY0FBYyxDQUFDLFdBQVcsQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxNQUFNLENBQUMsQ0FBQztRQUMzRixPQUFPLE9BQU8sQ0FBQyxVQUFVLENBQUM7SUFDNUIsQ0FBQztJQUVNLEtBQUssQ0FBQyxrQkFBa0IsQ0FBQyxZQUFvQjtRQUNsRCxNQUFNLFVBQVUsR0FBRyxNQUFNLElBQUksQ0FBQyxjQUFjLEVBQUUsQ0FBQztRQUMvQyxNQUFNLE1BQU0sR0FBRyxrQkFBa0IsQ0FBQyxXQUFXLEVBQUUsQ0FBQztRQUNoRCxNQUFNLFVBQVUsR0FBRyxNQUFNLE1BQU0sQ0FBQyxRQUFRLENBQUM7WUFDdkMsVUFBVSxFQUFFLFlBQVk7WUFDeEIsTUFBTSxFQUFFLElBQUksQ0FBQyxVQUFVLENBQUMsTUFBTTtZQUM5QixRQUFRLEVBQUUsSUFBSSxDQUFDLFVBQVUsQ0FBQyxRQUFRO1lBQ2xDLFVBQVU7U0FDWCxDQUFDLENBQUM7UUFDSCxPQUFPLFVBQVUsQ0FBQyxNQUFNLENBQUM7SUFDM0IsQ0FBQztDQUNGIn0= \ No newline at end of file diff --git a/dist_ts/mail/delivery/classes.mta.config.d.ts b/dist_ts/mail/delivery/classes.mta.config.d.ts deleted file mode 100644 index 503aca0..0000000 --- a/dist_ts/mail/delivery/classes.mta.config.d.ts +++ /dev/null @@ -1,22 +0,0 @@ -import type { UnifiedEmailServer } from '../routing/classes.unified.email.server.js'; -/** - * Configures email server storage settings - * @param emailServer Reference to the unified email server - * @param options Configuration options containing storage paths - */ -export declare function configureEmailStorage(emailServer: UnifiedEmailServer, options: any): Promise; -/** - * Configure email server with port and storage settings - * @param emailServer Reference to the unified email server - * @param config Configuration settings for email server - */ -export declare function configureEmailServer(emailServer: UnifiedEmailServer, config: { - ports?: number[]; - hostname?: string; - tls?: { - certPath?: string; - keyPath?: string; - caPath?: string; - }; - storagePath?: string; -}): Promise; diff --git a/dist_ts/mail/delivery/classes.mta.config.js b/dist_ts/mail/delivery/classes.mta.config.js deleted file mode 100644 index 294248b..0000000 --- a/dist_ts/mail/delivery/classes.mta.config.js +++ /dev/null @@ -1,51 +0,0 @@ -import * as plugins from '../../plugins.js'; -import * as paths from '../../paths.js'; -/** - * Configures email server storage settings - * @param emailServer Reference to the unified email server - * @param options Configuration options containing storage paths - */ -export async function configureEmailStorage(emailServer, options) { - // Extract the receivedEmailsPath if available - if (options?.emailPortConfig?.receivedEmailsPath) { - const receivedEmailsPath = options.emailPortConfig.receivedEmailsPath; - // Ensure the directory exists - await plugins.smartfs.directory(receivedEmailsPath).recursive().create(); - // Set path for received emails - if (emailServer) { - // Storage paths are now handled by the unified email server system - await plugins.smartfs.directory(paths.receivedEmailsDir).recursive().create(); - console.log(`Configured email server to store received emails to: ${receivedEmailsPath}`); - } - } -} -/** - * Configure email server with port and storage settings - * @param emailServer Reference to the unified email server - * @param config Configuration settings for email server - */ -export async function configureEmailServer(emailServer, config) { - if (!emailServer) { - console.error('Email server not available'); - return false; - } - // Configure the email server with updated options - const serverOptions = { - ports: config.ports || [25, 587, 465], - hostname: config.hostname || 'localhost', - tls: config.tls - }; - // Update the email server options - emailServer.updateOptions(serverOptions); - console.log(`Configured email server on ports ${serverOptions.ports.join(', ')}`); - // Set up storage path if provided - if (config.storagePath) { - await configureEmailStorage(emailServer, { - emailPortConfig: { - receivedEmailsPath: config.storagePath - } - }); - } - return true; -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5tdGEuY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vdHMvbWFpbC9kZWxpdmVyeS9jbGFzc2VzLm10YS5jb25maWcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxLQUFLLE9BQU8sTUFBTSxrQkFBa0IsQ0FBQztBQUM1QyxPQUFPLEtBQUssS0FBSyxNQUFNLGdCQUFnQixDQUFDO0FBR3hDOzs7O0dBSUc7QUFDSCxNQUFNLENBQUMsS0FBSyxVQUFVLHFCQUFxQixDQUFDLFdBQStCLEVBQUUsT0FBWTtJQUN2Riw4Q0FBOEM7SUFDOUMsSUFBSSxPQUFPLEVBQUUsZUFBZSxFQUFFLGtCQUFrQixFQUFFLENBQUM7UUFDakQsTUFBTSxrQkFBa0IsR0FBRyxPQUFPLENBQUMsZUFBZSxDQUFDLGtCQUFrQixDQUFDO1FBRXRFLDhCQUE4QjtRQUM5QixNQUFNLE9BQU8sQ0FBQyxPQUFPLENBQUMsU0FBUyxDQUFDLGtCQUFrQixDQUFDLENBQUMsU0FBUyxFQUFFLENBQUMsTUFBTSxFQUFFLENBQUM7UUFFekUsK0JBQStCO1FBQy9CLElBQUksV0FBVyxFQUFFLENBQUM7WUFDaEIsbUVBQW1FO1lBQ25FLE1BQU0sT0FBTyxDQUFDLE9BQU8sQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLGlCQUFpQixDQUFDLENBQUMsU0FBUyxFQUFFLENBQUMsTUFBTSxFQUFFLENBQUM7WUFFOUUsT0FBTyxDQUFDLEdBQUcsQ0FBQyx3REFBd0Qsa0JBQWtCLEVBQUUsQ0FBQyxDQUFDO1FBQzVGLENBQUM7SUFDSCxDQUFDO0FBQ0gsQ0FBQztBQUVEOzs7O0dBSUc7QUFDSCxNQUFNLENBQUMsS0FBSyxVQUFVLG9CQUFvQixDQUN4QyxXQUErQixFQUMvQixNQVNDO0lBRUQsSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDO1FBQ2pCLE9BQU8sQ0FBQyxLQUFLLENBQUMsNEJBQTRCLENBQUMsQ0FBQztRQUM1QyxPQUFPLEtBQUssQ0FBQztJQUNmLENBQUM7SUFFRCxrREFBa0Q7SUFDbEQsTUFBTSxhQUFhLEdBQUc7UUFDcEIsS0FBSyxFQUFFLE1BQU0sQ0FBQyxLQUFLLElBQUksQ0FBQyxFQUFFLEVBQUUsR0FBRyxFQUFFLEdBQUcsQ0FBQztRQUNyQyxRQUFRLEVBQUUsTUFBTSxDQUFDLFFBQVEsSUFBSSxXQUFXO1FBQ3hDLEdBQUcsRUFBRSxNQUFNLENBQUMsR0FBRztLQUNoQixDQUFDO0lBRUYsa0NBQWtDO0lBQ2xDLFdBQVcsQ0FBQyxhQUFhLENBQUMsYUFBYSxDQUFDLENBQUM7SUFFekMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxvQ0FBb0MsYUFBYSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQyxDQUFDO0lBRWxGLGtDQUFrQztJQUNsQyxJQUFJLE1BQU0sQ0FBQyxXQUFXLEVBQUUsQ0FBQztRQUN2QixNQUFNLHFCQUFxQixDQUFDLFdBQVcsRUFBRTtZQUN2QyxlQUFlLEVBQUU7Z0JBQ2Ysa0JBQWtCLEVBQUUsTUFBTSxDQUFDLFdBQVc7YUFDdkM7U0FDRixDQUFDLENBQUM7SUFDTCxDQUFDO0lBRUQsT0FBTyxJQUFJLENBQUM7QUFDZCxDQUFDIn0= \ No newline at end of file diff --git a/dist_ts/mail/delivery/classes.ratelimiter.d.ts b/dist_ts/mail/delivery/classes.ratelimiter.d.ts deleted file mode 100644 index 2d02504..0000000 --- a/dist_ts/mail/delivery/classes.ratelimiter.d.ts +++ /dev/null @@ -1,108 +0,0 @@ -/** - * Configuration options for rate limiter - */ -export interface IRateLimitConfig { - /** Maximum tokens per period */ - maxPerPeriod: number; - /** Time period in milliseconds */ - periodMs: number; - /** Whether to apply per domain/key (vs globally) */ - perKey: boolean; - /** Initial token count (defaults to max) */ - initialTokens?: number; - /** Grace tokens to allow occasional bursts */ - burstTokens?: number; - /** Apply global limit in addition to per-key limits */ - useGlobalLimit?: boolean; -} -/** - * Rate limiter using token bucket algorithm - * Provides more sophisticated rate limiting with burst handling - */ -export declare class RateLimiter { - /** Rate limit configuration */ - private config; - /** Token buckets per key */ - private buckets; - /** Global bucket for non-keyed rate limiting */ - private globalBucket; - /** - * Create a new rate limiter - * @param config Rate limiter configuration - */ - constructor(config: IRateLimitConfig); - /** - * Check if a request is allowed under rate limits - * @param key Key to check rate limit for (e.g. domain, user, IP) - * @param cost Token cost (defaults to 1) - * @returns Whether the request is allowed - */ - isAllowed(key?: string, cost?: number): boolean; - /** - * Check if a bucket has enough tokens and consume them - * @param bucket The token bucket to check - * @param cost Token cost - * @returns Whether tokens were consumed - */ - private checkBucket; - /** - * Consume tokens for a request (if available) - * @param key Key to consume tokens for - * @param cost Token cost (defaults to 1) - * @returns Whether tokens were consumed - */ - consume(key?: string, cost?: number): boolean; - /** - * Get the remaining tokens for a key - * @param key Key to check - * @returns Number of remaining tokens - */ - getRemainingTokens(key?: string): number; - /** - * Get stats for a specific key - * @param key Key to get stats for - * @returns Rate limit statistics - */ - getStats(key?: string): { - remaining: number; - limit: number; - resetIn: number; - allowed: number; - denied: number; - }; - /** - * Get or create a token bucket for a key - * @param key The rate limit key - * @returns Token bucket - */ - private getBucket; - /** - * Refill tokens in a bucket based on elapsed time - * @param bucket Token bucket to refill - */ - private refillBucket; - /** - * Reset rate limits for a specific key - * @param key Key to reset - */ - reset(key?: string): void; - /** - * Reset all rate limiters - */ - resetAll(): void; - /** - * Cleanup old buckets to prevent memory leaks - * @param maxAge Maximum age in milliseconds - */ - cleanup(maxAge?: number): void; - /** - * Record an error for a key (e.g., IP address) and determine if blocking is needed - * RFC 5321 Section 4.5.4.1 suggests limiting errors to prevent abuse - * - * @param key Key to record error for (typically an IP address) - * @param errorWindow Time window for error tracking in ms (default: 5 minutes) - * @param errorThreshold Maximum errors before blocking (default: 10) - * @returns true if the key should be blocked due to excessive errors - */ - recordError(key: string, errorWindow?: number, errorThreshold?: number): boolean; -} diff --git a/dist_ts/mail/delivery/classes.ratelimiter.js b/dist_ts/mail/delivery/classes.ratelimiter.js deleted file mode 100644 index 63adfac..0000000 --- a/dist_ts/mail/delivery/classes.ratelimiter.js +++ /dev/null @@ -1,241 +0,0 @@ -import { logger } from '../../logger.js'; -/** - * Rate limiter using token bucket algorithm - * Provides more sophisticated rate limiting with burst handling - */ -export class RateLimiter { - /** Rate limit configuration */ - config; - /** Token buckets per key */ - buckets = new Map(); - /** Global bucket for non-keyed rate limiting */ - globalBucket; - /** - * Create a new rate limiter - * @param config Rate limiter configuration - */ - constructor(config) { - // Set defaults - this.config = { - maxPerPeriod: config.maxPerPeriod, - periodMs: config.periodMs, - perKey: config.perKey ?? true, - initialTokens: config.initialTokens ?? config.maxPerPeriod, - burstTokens: config.burstTokens ?? 0, - useGlobalLimit: config.useGlobalLimit ?? false - }; - // Initialize global bucket - this.globalBucket = { - tokens: this.config.initialTokens, - lastRefill: Date.now(), - allowed: 0, - denied: 0, - errors: 0, - firstErrorTime: 0 - }; - // Log initialization - logger.log('info', `Rate limiter initialized: ${this.config.maxPerPeriod} per ${this.config.periodMs}ms${this.config.perKey ? ' per key' : ''}`); - } - /** - * Check if a request is allowed under rate limits - * @param key Key to check rate limit for (e.g. domain, user, IP) - * @param cost Token cost (defaults to 1) - * @returns Whether the request is allowed - */ - isAllowed(key = 'global', cost = 1) { - // If using global bucket directly, just check that - if (key === 'global' || !this.config.perKey) { - return this.checkBucket(this.globalBucket, cost); - } - // Get the key-specific bucket - const bucket = this.getBucket(key); - // If we also need to check global limit - if (this.config.useGlobalLimit) { - // Both key bucket and global bucket must have tokens - return this.checkBucket(bucket, cost) && this.checkBucket(this.globalBucket, cost); - } - else { - // Only need to check the key-specific bucket - return this.checkBucket(bucket, cost); - } - } - /** - * Check if a bucket has enough tokens and consume them - * @param bucket The token bucket to check - * @param cost Token cost - * @returns Whether tokens were consumed - */ - checkBucket(bucket, cost) { - // Refill tokens based on elapsed time - this.refillBucket(bucket); - // Check if we have enough tokens - if (bucket.tokens >= cost) { - // Use tokens - bucket.tokens -= cost; - bucket.allowed++; - return true; - } - else { - // Rate limit exceeded - bucket.denied++; - return false; - } - } - /** - * Consume tokens for a request (if available) - * @param key Key to consume tokens for - * @param cost Token cost (defaults to 1) - * @returns Whether tokens were consumed - */ - consume(key = 'global', cost = 1) { - const isAllowed = this.isAllowed(key, cost); - return isAllowed; - } - /** - * Get the remaining tokens for a key - * @param key Key to check - * @returns Number of remaining tokens - */ - getRemainingTokens(key = 'global') { - const bucket = this.getBucket(key); - this.refillBucket(bucket); - return bucket.tokens; - } - /** - * Get stats for a specific key - * @param key Key to get stats for - * @returns Rate limit statistics - */ - getStats(key = 'global') { - const bucket = this.getBucket(key); - this.refillBucket(bucket); - // Calculate time until next token - const resetIn = bucket.tokens < this.config.maxPerPeriod ? - Math.ceil(this.config.periodMs / this.config.maxPerPeriod) : - 0; - return { - remaining: bucket.tokens, - limit: this.config.maxPerPeriod, - resetIn, - allowed: bucket.allowed, - denied: bucket.denied - }; - } - /** - * Get or create a token bucket for a key - * @param key The rate limit key - * @returns Token bucket - */ - getBucket(key) { - if (!this.config.perKey || key === 'global') { - return this.globalBucket; - } - if (!this.buckets.has(key)) { - // Create new bucket - this.buckets.set(key, { - tokens: this.config.initialTokens, - lastRefill: Date.now(), - allowed: 0, - denied: 0, - errors: 0, - firstErrorTime: 0 - }); - } - return this.buckets.get(key); - } - /** - * Refill tokens in a bucket based on elapsed time - * @param bucket Token bucket to refill - */ - refillBucket(bucket) { - const now = Date.now(); - const elapsedMs = now - bucket.lastRefill; - // Calculate how many tokens to add - const rate = this.config.maxPerPeriod / this.config.periodMs; - const tokensToAdd = elapsedMs * rate; - if (tokensToAdd >= 0.1) { // Allow for partial token refills - // Add tokens, but don't exceed the normal maximum (without burst) - // This ensures burst tokens are only used for bursts and don't refill - const normalMax = this.config.maxPerPeriod; - bucket.tokens = Math.min( - // Don't exceed max + burst - this.config.maxPerPeriod + (this.config.burstTokens || 0), - // Don't exceed normal max when refilling - Math.min(normalMax, bucket.tokens + tokensToAdd)); - // Update last refill time - bucket.lastRefill = now; - } - } - /** - * Reset rate limits for a specific key - * @param key Key to reset - */ - reset(key = 'global') { - if (key === 'global' || !this.config.perKey) { - this.globalBucket.tokens = this.config.initialTokens; - this.globalBucket.lastRefill = Date.now(); - } - else if (this.buckets.has(key)) { - const bucket = this.buckets.get(key); - bucket.tokens = this.config.initialTokens; - bucket.lastRefill = Date.now(); - } - } - /** - * Reset all rate limiters - */ - resetAll() { - this.globalBucket.tokens = this.config.initialTokens; - this.globalBucket.lastRefill = Date.now(); - for (const bucket of this.buckets.values()) { - bucket.tokens = this.config.initialTokens; - bucket.lastRefill = Date.now(); - } - } - /** - * Cleanup old buckets to prevent memory leaks - * @param maxAge Maximum age in milliseconds - */ - cleanup(maxAge = 24 * 60 * 60 * 1000) { - const now = Date.now(); - let removed = 0; - for (const [key, bucket] of this.buckets.entries()) { - if (now - bucket.lastRefill > maxAge) { - this.buckets.delete(key); - removed++; - } - } - if (removed > 0) { - logger.log('debug', `Cleaned up ${removed} stale rate limit buckets`); - } - } - /** - * Record an error for a key (e.g., IP address) and determine if blocking is needed - * RFC 5321 Section 4.5.4.1 suggests limiting errors to prevent abuse - * - * @param key Key to record error for (typically an IP address) - * @param errorWindow Time window for error tracking in ms (default: 5 minutes) - * @param errorThreshold Maximum errors before blocking (default: 10) - * @returns true if the key should be blocked due to excessive errors - */ - recordError(key, errorWindow = 5 * 60 * 1000, errorThreshold = 10) { - const bucket = this.getBucket(key); - const now = Date.now(); - // Reset error count if the time window has expired - if (bucket.firstErrorTime === 0 || now - bucket.firstErrorTime > errorWindow) { - bucket.errors = 0; - bucket.firstErrorTime = now; - } - // Increment error count - bucket.errors++; - // Log error tracking - logger.log('debug', `Error recorded for ${key}: ${bucket.errors}/${errorThreshold} in window`); - // Check if threshold exceeded - if (bucket.errors >= errorThreshold) { - logger.log('warn', `Error threshold exceeded for ${key}: ${bucket.errors} errors`); - return true; // Should block - } - return false; // Continue allowing - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5yYXRlbGltaXRlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3RzL21haWwvZGVsaXZlcnkvY2xhc3Nlcy5yYXRlbGltaXRlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsTUFBTSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFnRHpDOzs7R0FHRztBQUNILE1BQU0sT0FBTyxXQUFXO0lBQ3RCLCtCQUErQjtJQUN2QixNQUFNLENBQW1CO0lBRWpDLDRCQUE0QjtJQUNwQixPQUFPLEdBQTZCLElBQUksR0FBRyxFQUFFLENBQUM7SUFFdEQsZ0RBQWdEO0lBQ3hDLFlBQVksQ0FBYztJQUVsQzs7O09BR0c7SUFDSCxZQUFZLE1BQXdCO1FBQ2xDLGVBQWU7UUFDZixJQUFJLENBQUMsTUFBTSxHQUFHO1lBQ1osWUFBWSxFQUFFLE1BQU0sQ0FBQyxZQUFZO1lBQ2pDLFFBQVEsRUFBRSxNQUFNLENBQUMsUUFBUTtZQUN6QixNQUFNLEVBQUUsTUFBTSxDQUFDLE1BQU0sSUFBSSxJQUFJO1lBQzdCLGFBQWEsRUFBRSxNQUFNLENBQUMsYUFBYSxJQUFJLE1BQU0sQ0FBQyxZQUFZO1lBQzFELFdBQVcsRUFBRSxNQUFNLENBQUMsV0FBVyxJQUFJLENBQUM7WUFDcEMsY0FBYyxFQUFFLE1BQU0sQ0FBQyxjQUFjLElBQUksS0FBSztTQUMvQyxDQUFDO1FBRUYsMkJBQTJCO1FBQzNCLElBQUksQ0FBQyxZQUFZLEdBQUc7WUFDbEIsTUFBTSxFQUFFLElBQUksQ0FBQyxNQUFNLENBQUMsYUFBYTtZQUNqQyxVQUFVLEVBQUUsSUFBSSxDQUFDLEdBQUcsRUFBRTtZQUN0QixPQUFPLEVBQUUsQ0FBQztZQUNWLE1BQU0sRUFBRSxDQUFDO1lBQ1QsTUFBTSxFQUFFLENBQUM7WUFDVCxjQUFjLEVBQUUsQ0FBQztTQUNsQixDQUFDO1FBRUYscUJBQXFCO1FBQ3JCLE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLDZCQUE2QixJQUFJLENBQUMsTUFBTSxDQUFDLFlBQVksUUFBUSxJQUFJLENBQUMsTUFBTSxDQUFDLFFBQVEsS0FBSyxJQUFJLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsVUFBVSxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDO0lBQ25KLENBQUM7SUFFRDs7Ozs7T0FLRztJQUNJLFNBQVMsQ0FBQyxNQUFjLFFBQVEsRUFBRSxPQUFlLENBQUM7UUFDdkQsbURBQW1EO1FBQ25ELElBQUksR0FBRyxLQUFLLFFBQVEsSUFBSSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsTUFBTSxFQUFFLENBQUM7WUFDNUMsT0FBTyxJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxZQUFZLEVBQUUsSUFBSSxDQUFDLENBQUM7UUFDbkQsQ0FBQztRQUVELDhCQUE4QjtRQUM5QixNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDLEdBQUcsQ0FBQyxDQUFDO1FBRW5DLHdDQUF3QztRQUN4QyxJQUFJLElBQUksQ0FBQyxNQUFNLENBQUMsY0FBYyxFQUFFLENBQUM7WUFDL0IscURBQXFEO1lBQ3JELE9BQU8sSUFBSSxDQUFDLFdBQVcsQ0FBQyxNQUFNLEVBQUUsSUFBSSxDQUFDLElBQUksSUFBSSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsWUFBWSxFQUFFLElBQUksQ0FBQyxDQUFDO1FBQ3JGLENBQUM7YUFBTSxDQUFDO1lBQ04sNkNBQTZDO1lBQzdDLE9BQU8sSUFBSSxDQUFDLFdBQVcsQ0FBQyxNQUFNLEVBQUUsSUFBSSxDQUFDLENBQUM7UUFDeEMsQ0FBQztJQUNILENBQUM7SUFFRDs7Ozs7T0FLRztJQUNLLFdBQVcsQ0FBQyxNQUFtQixFQUFFLElBQVk7UUFDbkQsc0NBQXNDO1FBQ3RDLElBQUksQ0FBQyxZQUFZLENBQUMsTUFBTSxDQUFDLENBQUM7UUFFMUIsaUNBQWlDO1FBQ2pDLElBQUksTUFBTSxDQUFDLE1BQU0sSUFBSSxJQUFJLEVBQUUsQ0FBQztZQUMxQixhQUFhO1lBQ2IsTUFBTSxDQUFDLE1BQU0sSUFBSSxJQUFJLENBQUM7WUFDdEIsTUFBTSxDQUFDLE9BQU8sRUFBRSxDQUFDO1lBQ2pCLE9BQU8sSUFBSSxDQUFDO1FBQ2QsQ0FBQzthQUFNLENBQUM7WUFDTixzQkFBc0I7WUFDdEIsTUFBTSxDQUFDLE1BQU0sRUFBRSxDQUFDO1lBQ2hCLE9BQU8sS0FBSyxDQUFDO1FBQ2YsQ0FBQztJQUNILENBQUM7SUFFRDs7Ozs7T0FLRztJQUNJLE9BQU8sQ0FBQyxNQUFjLFFBQVEsRUFBRSxPQUFlLENBQUM7UUFDckQsTUFBTSxTQUFTLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxHQUFHLEVBQUUsSUFBSSxDQUFDLENBQUM7UUFDNUMsT0FBTyxTQUFTLENBQUM7SUFDbkIsQ0FBQztJQUVEOzs7O09BSUc7SUFDSSxrQkFBa0IsQ0FBQyxNQUFjLFFBQVE7UUFDOUMsTUFBTSxNQUFNLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUNuQyxJQUFJLENBQUMsWUFBWSxDQUFDLE1BQU0sQ0FBQyxDQUFDO1FBQzFCLE9BQU8sTUFBTSxDQUFDLE1BQU0sQ0FBQztJQUN2QixDQUFDO0lBRUQ7Ozs7T0FJRztJQUNJLFFBQVEsQ0FBQyxNQUFjLFFBQVE7UUFPcEMsTUFBTSxNQUFNLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUNuQyxJQUFJLENBQUMsWUFBWSxDQUFDLE1BQU0sQ0FBQyxDQUFDO1FBRTFCLGtDQUFrQztRQUNsQyxNQUFNLE9BQU8sR0FBRyxNQUFNLENBQUMsTUFBTSxHQUFHLElBQUksQ0FBQyxNQUFNLENBQUMsWUFBWSxDQUFDLENBQUM7WUFDeEQsSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLFFBQVEsR0FBRyxJQUFJLENBQUMsTUFBTSxDQUFDLFlBQVksQ0FBQyxDQUFDLENBQUM7WUFDNUQsQ0FBQyxDQUFDO1FBRUosT0FBTztZQUNMLFNBQVMsRUFBRSxNQUFNLENBQUMsTUFBTTtZQUN4QixLQUFLLEVBQUUsSUFBSSxDQUFDLE1BQU0sQ0FBQyxZQUFZO1lBQy9CLE9BQU87WUFDUCxPQUFPLEVBQUUsTUFBTSxDQUFDLE9BQU87WUFDdkIsTUFBTSxFQUFFLE1BQU0sQ0FBQyxNQUFNO1NBQ3RCLENBQUM7SUFDSixDQUFDO0lBRUQ7Ozs7T0FJRztJQUNLLFNBQVMsQ0FBQyxHQUFXO1FBQzNCLElBQUksQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLE1BQU0sSUFBSSxHQUFHLEtBQUssUUFBUSxFQUFFLENBQUM7WUFDNUMsT0FBTyxJQUFJLENBQUMsWUFBWSxDQUFDO1FBQzNCLENBQUM7UUFFRCxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQztZQUMzQixvQkFBb0I7WUFDcEIsSUFBSSxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsR0FBRyxFQUFFO2dCQUNwQixNQUFNLEVBQUUsSUFBSSxDQUFDLE1BQU0sQ0FBQyxhQUFhO2dCQUNqQyxVQUFVLEVBQUUsSUFBSSxDQUFDLEdBQUcsRUFBRTtnQkFDdEIsT0FBTyxFQUFFLENBQUM7Z0JBQ1YsTUFBTSxFQUFFLENBQUM7Z0JBQ1QsTUFBTSxFQUFFLENBQUM7Z0JBQ1QsY0FBYyxFQUFFLENBQUM7YUFDbEIsQ0FBQyxDQUFDO1FBQ0wsQ0FBQztRQUVELE9BQU8sSUFBSSxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDL0IsQ0FBQztJQUVEOzs7T0FHRztJQUNLLFlBQVksQ0FBQyxNQUFtQjtRQUN0QyxNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsR0FBRyxFQUFFLENBQUM7UUFDdkIsTUFBTSxTQUFTLEdBQUcsR0FBRyxHQUFHLE1BQU0sQ0FBQyxVQUFVLENBQUM7UUFFMUMsbUNBQW1DO1FBQ25DLE1BQU0sSUFBSSxHQUFHLElBQUksQ0FBQyxNQUFNLENBQUMsWUFBWSxHQUFHLElBQUksQ0FBQyxNQUFNLENBQUMsUUFBUSxDQUFDO1FBQzdELE1BQU0sV0FBVyxHQUFHLFNBQVMsR0FBRyxJQUFJLENBQUM7UUFFckMsSUFBSSxXQUFXLElBQUksR0FBRyxFQUFFLENBQUMsQ0FBQyxrQ0FBa0M7WUFDMUQsa0VBQWtFO1lBQ2xFLHNFQUFzRTtZQUN0RSxNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsTUFBTSxDQUFDLFlBQVksQ0FBQztZQUMzQyxNQUFNLENBQUMsTUFBTSxHQUFHLElBQUksQ0FBQyxHQUFHO1lBQ3RCLDJCQUEyQjtZQUMzQixJQUFJLENBQUMsTUFBTSxDQUFDLFlBQVksR0FBRyxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsV0FBVyxJQUFJLENBQUMsQ0FBQztZQUN6RCx5Q0FBeUM7WUFDekMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxTQUFTLEVBQUUsTUFBTSxDQUFDLE1BQU0sR0FBRyxXQUFXLENBQUMsQ0FDakQsQ0FBQztZQUVGLDBCQUEwQjtZQUMxQixNQUFNLENBQUMsVUFBVSxHQUFHLEdBQUcsQ0FBQztRQUMxQixDQUFDO0lBQ0gsQ0FBQztJQUVEOzs7T0FHRztJQUNJLEtBQUssQ0FBQyxNQUFjLFFBQVE7UUFDakMsSUFBSSxHQUFHLEtBQUssUUFBUSxJQUFJLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxNQUFNLEVBQUUsQ0FBQztZQUM1QyxJQUFJLENBQUMsWUFBWSxDQUFDLE1BQU0sR0FBRyxJQUFJLENBQUMsTUFBTSxDQUFDLGFBQWEsQ0FBQztZQUNyRCxJQUFJLENBQUMsWUFBWSxDQUFDLFVBQVUsR0FBRyxJQUFJLENBQUMsR0FBRyxFQUFFLENBQUM7UUFDNUMsQ0FBQzthQUFNLElBQUksSUFBSSxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQztZQUNqQyxNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxHQUFHLENBQUMsQ0FBQztZQUNyQyxNQUFNLENBQUMsTUFBTSxHQUFHLElBQUksQ0FBQyxNQUFNLENBQUMsYUFBYSxDQUFDO1lBQzFDLE1BQU0sQ0FBQyxVQUFVLEdBQUcsSUFBSSxDQUFDLEdBQUcsRUFBRSxDQUFDO1FBQ2pDLENBQUM7SUFDSCxDQUFDO0lBRUQ7O09BRUc7SUFDSSxRQUFRO1FBQ2IsSUFBSSxDQUFDLFlBQVksQ0FBQyxNQUFNLEdBQUcsSUFBSSxDQUFDLE1BQU0sQ0FBQyxhQUFhLENBQUM7UUFDckQsSUFBSSxDQUFDLFlBQVksQ0FBQyxVQUFVLEdBQUcsSUFBSSxDQUFDLEdBQUcsRUFBRSxDQUFDO1FBRTFDLEtBQUssTUFBTSxNQUFNLElBQUksSUFBSSxDQUFDLE9BQU8sQ0FBQyxNQUFNLEVBQUUsRUFBRSxDQUFDO1lBQzNDLE1BQU0sQ0FBQyxNQUFNLEdBQUcsSUFBSSxDQUFDLE1BQU0sQ0FBQyxhQUFhLENBQUM7WUFDMUMsTUFBTSxDQUFDLFVBQVUsR0FBRyxJQUFJLENBQUMsR0FBRyxFQUFFLENBQUM7UUFDakMsQ0FBQztJQUNILENBQUM7SUFFRDs7O09BR0c7SUFDSSxPQUFPLENBQUMsU0FBaUIsRUFBRSxHQUFHLEVBQUUsR0FBRyxFQUFFLEdBQUcsSUFBSTtRQUNqRCxNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsR0FBRyxFQUFFLENBQUM7UUFDdkIsSUFBSSxPQUFPLEdBQUcsQ0FBQyxDQUFDO1FBRWhCLEtBQUssTUFBTSxDQUFDLEdBQUcsRUFBRSxNQUFNLENBQUMsSUFBSSxJQUFJLENBQUMsT0FBTyxDQUFDLE9BQU8sRUFBRSxFQUFFLENBQUM7WUFDbkQsSUFBSSxHQUFHLEdBQUcsTUFBTSxDQUFDLFVBQVUsR0FBRyxNQUFNLEVBQUUsQ0FBQztnQkFDckMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLENBQUM7Z0JBQ3pCLE9BQU8sRUFBRSxDQUFDO1lBQ1osQ0FBQztRQUNILENBQUM7UUFFRCxJQUFJLE9BQU8sR0FBRyxDQUFDLEVBQUUsQ0FBQztZQUNoQixNQUFNLENBQUMsR0FBRyxDQUFDLE9BQU8sRUFBRSxjQUFjLE9BQU8sMkJBQTJCLENBQUMsQ0FBQztRQUN4RSxDQUFDO0lBQ0gsQ0FBQztJQUVEOzs7Ozs7OztPQVFHO0lBQ0ksV0FBVyxDQUFDLEdBQVcsRUFBRSxjQUFzQixDQUFDLEdBQUcsRUFBRSxHQUFHLElBQUksRUFBRSxpQkFBeUIsRUFBRTtRQUM5RixNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDLEdBQUcsQ0FBQyxDQUFDO1FBQ25DLE1BQU0sR0FBRyxHQUFHLElBQUksQ0FBQyxHQUFHLEVBQUUsQ0FBQztRQUV2QixtREFBbUQ7UUFDbkQsSUFBSSxNQUFNLENBQUMsY0FBYyxLQUFLLENBQUMsSUFBSSxHQUFHLEdBQUcsTUFBTSxDQUFDLGNBQWMsR0FBRyxXQUFXLEVBQUUsQ0FBQztZQUM3RSxNQUFNLENBQUMsTUFBTSxHQUFHLENBQUMsQ0FBQztZQUNsQixNQUFNLENBQUMsY0FBYyxHQUFHLEdBQUcsQ0FBQztRQUM5QixDQUFDO1FBRUQsd0JBQXdCO1FBQ3hCLE1BQU0sQ0FBQyxNQUFNLEVBQUUsQ0FBQztRQUVoQixxQkFBcUI7UUFDckIsTUFBTSxDQUFDLEdBQUcsQ0FBQyxPQUFPLEVBQUUsc0JBQXNCLEdBQUcsS0FBSyxNQUFNLENBQUMsTUFBTSxJQUFJLGNBQWMsWUFBWSxDQUFDLENBQUM7UUFFL0YsOEJBQThCO1FBQzlCLElBQUksTUFBTSxDQUFDLE1BQU0sSUFBSSxjQUFjLEVBQUUsQ0FBQztZQUNwQyxNQUFNLENBQUMsR0FBRyxDQUFDLE1BQU0sRUFBRSxnQ0FBZ0MsR0FBRyxLQUFLLE1BQU0sQ0FBQyxNQUFNLFNBQVMsQ0FBQyxDQUFDO1lBQ25GLE9BQU8sSUFBSSxDQUFDLENBQUMsZUFBZTtRQUM5QixDQUFDO1FBRUQsT0FBTyxLQUFLLENBQUMsQ0FBQyxvQkFBb0I7SUFDcEMsQ0FBQztDQUNGIn0= \ No newline at end of file diff --git a/dist_ts/mail/delivery/classes.smtp.client.legacy.d.ts b/dist_ts/mail/delivery/classes.smtp.client.legacy.d.ts deleted file mode 100644 index 200253f..0000000 --- a/dist_ts/mail/delivery/classes.smtp.client.legacy.d.ts +++ /dev/null @@ -1,275 +0,0 @@ -import { Email } from '../core/classes.email.js'; -import type { EmailProcessingMode } from './interfaces.js'; -/** - * SMTP client connection options - */ -export type ISmtpClientOptions = { - /** - * Hostname of the SMTP server - */ - host: string; - /** - * Port to connect to - */ - port: number; - /** - * Whether to use TLS for the connection - */ - secure?: boolean; - /** - * Connection timeout in milliseconds - */ - connectionTimeout?: number; - /** - * Socket timeout in milliseconds - */ - socketTimeout?: number; - /** - * Command timeout in milliseconds - */ - commandTimeout?: number; - /** - * TLS options - */ - tls?: { - /** - * Whether to verify certificates - */ - rejectUnauthorized?: boolean; - /** - * Minimum TLS version - */ - minVersion?: string; - /** - * CA certificate path - */ - ca?: string; - }; - /** - * Authentication options - */ - auth?: { - /** - * Authentication user - */ - user: string; - /** - * Authentication password - */ - pass: string; - /** - * Authentication method - */ - method?: 'PLAIN' | 'LOGIN' | 'OAUTH2'; - }; - /** - * Domain name for EHLO - */ - domain?: string; - /** - * DKIM options for signing outgoing emails - */ - dkim?: { - /** - * Whether to sign emails with DKIM - */ - enabled: boolean; - /** - * Domain name for DKIM - */ - domain: string; - /** - * Selector for DKIM - */ - selector: string; - /** - * Private key for DKIM signing - */ - privateKey: string; - /** - * Headers to sign - */ - headers?: string[]; - }; -}; -/** - * SMTP delivery result - */ -export type ISmtpDeliveryResult = { - /** - * Whether the delivery was successful - */ - success: boolean; - /** - * Message ID if successful - */ - messageId?: string; - /** - * Error message if failed - */ - error?: string; - /** - * SMTP response code - */ - responseCode?: string; - /** - * Recipients successfully delivered to - */ - acceptedRecipients: string[]; - /** - * Recipients rejected during delivery - */ - rejectedRecipients: string[]; - /** - * Server response - */ - response?: string; - /** - * Timestamp of the delivery attempt - */ - timestamp: number; - /** - * Whether DKIM signing was applied - */ - dkimSigned?: boolean; - /** - * Whether this was a TLS secured delivery - */ - secure?: boolean; - /** - * Whether authentication was used - */ - authenticated?: boolean; -}; -/** - * SMTP client for sending emails to remote mail servers - */ -export declare class SmtpClient { - private options; - private connected; - private socket?; - private supportedExtensions; - /** - * Create a new SMTP client instance - * @param options SMTP client connection options - */ - constructor(options: ISmtpClientOptions); - /** - * Connect to the SMTP server - */ - connect(): Promise; - /** - * Send EHLO command to the server - */ - private sendEhlo; - /** - * Start TLS negotiation - */ - private startTls; - /** - * Upgrade socket to TLS - * @param socket Original socket - */ - private upgradeTls; - /** - * Authenticate with the server - */ - private authenticate; - /** - * Authenticate using PLAIN method - * @param user Username - * @param pass Password - */ - private authPlain; - /** - * Authenticate using LOGIN method - * @param user Username - * @param pass Password - */ - private authLogin; - /** - * Authenticate using OAuth2 method - * @param user Username - * @param token OAuth2 token - */ - private authOAuth2; - /** - * Send an email through the SMTP client - * @param email Email to send - * @param processingMode Optional processing mode - */ - sendMail(email: Email, processingMode?: EmailProcessingMode): Promise; - /** - * Apply DKIM signature to email - * @param email Email to sign - */ - private applyDkimSignature; - /** - * Format email for SMTP transmission - * @param email Email to format - */ - private getFormattedEmail; - /** - * Get size of email in bytes - * @param email Email to measure - */ - private getEmailSize; - /** - * Send SMTP command and wait for response - * @param command SMTP command to send - */ - private commandQueue; - private processingCommands; - private supportsPipelining; - /** - * Send an SMTP command and wait for response - * @param command SMTP command to send - * @param allowPipelining Whether this command can be pipelined - */ - private sendCommand; - /** - * Process the command queue - either one by one or pipelined if supported - */ - private processCommandQueue; - /** - * Process the next command in the queue (non-pipelined mode) - */ - private processNextCommand; - /** - * Process responses for pipelined commands - */ - private processResponses; - /** - * Read response from the server - */ - private readResponse; - /** - * Check if the response is complete - * @param response Response to check - */ - private isCompleteResponse; - /** - * Check if the response is an error - * @param response Response to check - */ - private isErrorResponse; - /** - * Create appropriate error from response - * @param response Error response - * @param code SMTP status code - */ - private createErrorFromResponse; - /** - * Close the connection to the server - */ - close(): Promise; - /** - * Checks if the connection is active - */ - isConnected(): boolean; - /** - * Update SMTP client options - * @param options New options - */ - updateOptions(options: Partial): void; -} diff --git a/dist_ts/mail/delivery/classes.smtp.client.legacy.js b/dist_ts/mail/delivery/classes.smtp.client.legacy.js deleted file mode 100644 index b091f9f..0000000 --- a/dist_ts/mail/delivery/classes.smtp.client.legacy.js +++ /dev/null @@ -1,969 +0,0 @@ -import * as plugins from '../../plugins.js'; -import { logger } from '../../logger.js'; -import { SecurityLogger, SecurityLogLevel, SecurityEventType } from '../../security/index.js'; -import { RustSecurityBridge } from '../../security/classes.rustsecuritybridge.js'; -import { MtaConnectionError, MtaAuthenticationError, MtaDeliveryError, MtaConfigurationError, MtaTimeoutError, MtaProtocolError } from '../../errors/index.js'; -import { Email } from '../core/classes.email.js'; -/** - * SMTP client for sending emails to remote mail servers - */ -export class SmtpClient { - options; - connected = false; - socket; - supportedExtensions = new Set(); - /** - * Create a new SMTP client instance - * @param options SMTP client connection options - */ - constructor(options) { - // Set default options - this.options = { - ...options, - connectionTimeout: options.connectionTimeout || 30000, // 30 seconds - socketTimeout: options.socketTimeout || 60000, // 60 seconds - commandTimeout: options.commandTimeout || 30000, // 30 seconds - secure: options.secure || false, - domain: options.domain || 'localhost', - tls: { - rejectUnauthorized: options.tls?.rejectUnauthorized !== false, // Default to true - minVersion: options.tls?.minVersion || 'TLSv1.2' - } - }; - } - /** - * Connect to the SMTP server - */ - async connect() { - if (this.connected && this.socket) { - return; - } - try { - logger.log('info', `Connecting to SMTP server ${this.options.host}:${this.options.port}`); - // Create socket - const socket = new plugins.net.Socket(); - // Set timeouts - socket.setTimeout(this.options.socketTimeout); - // Connect to the server - await new Promise((resolve, reject) => { - // Handle connection events - socket.once('connect', () => { - logger.log('debug', `Connected to ${this.options.host}:${this.options.port}`); - resolve(); - }); - socket.once('timeout', () => { - reject(MtaConnectionError.timeout(this.options.host, this.options.port, this.options.connectionTimeout)); - }); - socket.once('error', (err) => { - if (err.code === 'ECONNREFUSED') { - reject(MtaConnectionError.refused(this.options.host, this.options.port)); - } - else if (err.code === 'ENOTFOUND') { - reject(MtaConnectionError.dnsError(this.options.host, err)); - } - else { - reject(new MtaConnectionError(`Connection error to ${this.options.host}:${this.options.port}: ${err.message}`, { - data: { - host: this.options.host, - port: this.options.port, - error: err.message, - code: err.code - } - })); - } - }); - // Connect to the server - const connectOptions = { - host: this.options.host, - port: this.options.port - }; - // For direct TLS connections - if (this.options.secure) { - const tlsSocket = plugins.tls.connect({ - ...connectOptions, - rejectUnauthorized: this.options.tls.rejectUnauthorized, - minVersion: this.options.tls.minVersion, - ca: this.options.tls.ca ? [this.options.tls.ca] : undefined - }); - tlsSocket.once('secureConnect', () => { - logger.log('debug', `Secure connection established to ${this.options.host}:${this.options.port}`); - this.socket = tlsSocket; - resolve(); - }); - tlsSocket.once('error', (err) => { - reject(new MtaConnectionError(`TLS connection error to ${this.options.host}:${this.options.port}: ${err.message}`, { - data: { - host: this.options.host, - port: this.options.port, - error: err.message, - code: err.code - } - })); - }); - tlsSocket.setTimeout(this.options.socketTimeout); - tlsSocket.once('timeout', () => { - reject(MtaConnectionError.timeout(this.options.host, this.options.port, this.options.connectionTimeout)); - }); - } - else { - socket.connect(connectOptions); - this.socket = socket; - } - }); - // Wait for server greeting - const greeting = await this.readResponse(); - if (!greeting.startsWith('220')) { - throw new MtaConnectionError(`Unexpected greeting from server: ${greeting}`, { - data: { - host: this.options.host, - port: this.options.port, - greeting - } - }); - } - // Send EHLO - await this.sendEhlo(); - // Start TLS if not secure and supported - if (!this.options.secure && this.supportedExtensions.has('STARTTLS')) { - await this.startTls(); - // Send EHLO again after STARTTLS - await this.sendEhlo(); - } - // Authenticate if credentials provided - if (this.options.auth) { - await this.authenticate(); - } - this.connected = true; - logger.log('info', `Successfully connected to SMTP server ${this.options.host}:${this.options.port}`); - // Set up error handling for the socket - this.socket.on('error', (err) => { - logger.log('error', `Socket error: ${err.message}`); - this.connected = false; - this.socket = undefined; - }); - this.socket.on('close', () => { - logger.log('debug', 'Socket closed'); - this.connected = false; - this.socket = undefined; - }); - this.socket.on('timeout', () => { - logger.log('error', 'Socket timeout'); - this.connected = false; - if (this.socket) { - this.socket.destroy(); - this.socket = undefined; - } - }); - } - catch (error) { - // Clean up socket if connection failed - if (this.socket) { - this.socket.destroy(); - this.socket = undefined; - } - logger.log('error', `Failed to connect to SMTP server: ${error.message}`); - throw error; - } - } - /** - * Send EHLO command to the server - */ - async sendEhlo() { - // Clear previous extensions - this.supportedExtensions.clear(); - // Send EHLO - don't allow pipelining for this command - const response = await this.sendCommand(`EHLO ${this.options.domain}`, false); - // Parse supported extensions - const lines = response.split('\r\n'); - for (let i = 1; i < lines.length; i++) { - const line = lines[i]; - if (line.startsWith('250-') || line.startsWith('250 ')) { - const extension = line.substring(4).split(' ')[0]; - this.supportedExtensions.add(extension); - } - } - // Check if server supports pipelining - this.supportsPipelining = this.supportedExtensions.has('PIPELINING'); - logger.log('debug', `Server supports extensions: ${Array.from(this.supportedExtensions).join(', ')}`); - if (this.supportsPipelining) { - logger.log('info', 'Server supports PIPELINING - will use for improved performance'); - } - } - /** - * Start TLS negotiation - */ - async startTls() { - logger.log('debug', 'Starting TLS negotiation'); - // Send STARTTLS command - const response = await this.sendCommand('STARTTLS'); - if (!response.startsWith('220')) { - throw new MtaConnectionError(`Failed to start TLS: ${response}`, { - data: { - host: this.options.host, - port: this.options.port, - response - } - }); - } - if (!this.socket) { - throw new MtaConnectionError('No socket available for TLS upgrade', { - data: { - host: this.options.host, - port: this.options.port - } - }); - } - // Upgrade socket to TLS - const currentSocket = this.socket; - this.socket = await this.upgradeTls(currentSocket); - } - /** - * Upgrade socket to TLS - * @param socket Original socket - */ - async upgradeTls(socket) { - return new Promise((resolve, reject) => { - const tlsOptions = { - socket, - servername: this.options.host, - rejectUnauthorized: this.options.tls.rejectUnauthorized, - minVersion: this.options.tls.minVersion, - ca: this.options.tls.ca ? [this.options.tls.ca] : undefined - }; - const tlsSocket = plugins.tls.connect(tlsOptions); - tlsSocket.once('secureConnect', () => { - logger.log('debug', 'TLS negotiation successful'); - resolve(tlsSocket); - }); - tlsSocket.once('error', (err) => { - reject(new MtaConnectionError(`TLS error: ${err.message}`, { - data: { - host: this.options.host, - port: this.options.port, - error: err.message, - code: err.code - } - })); - }); - tlsSocket.setTimeout(this.options.socketTimeout); - tlsSocket.once('timeout', () => { - reject(MtaTimeoutError.commandTimeout('STARTTLS', this.options.host, this.options.socketTimeout)); - }); - }); - } - /** - * Authenticate with the server - */ - async authenticate() { - if (!this.options.auth) { - return; - } - const { user, pass, method = 'LOGIN' } = this.options.auth; - logger.log('debug', `Authenticating as ${user} using ${method}`); - try { - switch (method) { - case 'PLAIN': - await this.authPlain(user, pass); - break; - case 'LOGIN': - await this.authLogin(user, pass); - break; - case 'OAUTH2': - await this.authOAuth2(user, pass); - break; - default: - throw new MtaAuthenticationError(`Authentication method ${method} not supported by client`, { - data: { - method - } - }); - } - logger.log('info', `Successfully authenticated as ${user}`); - } - catch (error) { - logger.log('error', `Authentication failed: ${error.message}`); - throw error; - } - } - /** - * Authenticate using PLAIN method - * @param user Username - * @param pass Password - */ - async authPlain(user, pass) { - // PLAIN authentication format: \0username\0password - const authString = Buffer.from(`\0${user}\0${pass}`).toString('base64'); - const response = await this.sendCommand(`AUTH PLAIN ${authString}`); - if (!response.startsWith('235')) { - throw MtaAuthenticationError.invalidCredentials(this.options.host, user); - } - } - /** - * Authenticate using LOGIN method - * @param user Username - * @param pass Password - */ - async authLogin(user, pass) { - // Start LOGIN authentication - const response = await this.sendCommand('AUTH LOGIN'); - if (!response.startsWith('334')) { - throw new MtaAuthenticationError(`Server did not accept AUTH LOGIN: ${response}`, { - data: { - host: this.options.host, - response - } - }); - } - // Send username (base64) - const userResponse = await this.sendCommand(Buffer.from(user).toString('base64')); - if (!userResponse.startsWith('334')) { - throw MtaAuthenticationError.invalidCredentials(this.options.host, user); - } - // Send password (base64) - const passResponse = await this.sendCommand(Buffer.from(pass).toString('base64')); - if (!passResponse.startsWith('235')) { - throw MtaAuthenticationError.invalidCredentials(this.options.host, user); - } - } - /** - * Authenticate using OAuth2 method - * @param user Username - * @param token OAuth2 token - */ - async authOAuth2(user, token) { - // XOAUTH2 format - const authString = `user=${user}\x01auth=Bearer ${token}\x01\x01`; - const response = await this.sendCommand(`AUTH XOAUTH2 ${Buffer.from(authString).toString('base64')}`); - if (!response.startsWith('235')) { - throw MtaAuthenticationError.invalidCredentials(this.options.host, user); - } - } - /** - * Send an email through the SMTP client - * @param email Email to send - * @param processingMode Optional processing mode - */ - async sendMail(email, processingMode) { - // Ensure we're connected - if (!this.connected || !this.socket) { - await this.connect(); - } - const startTime = Date.now(); - const result = { - success: false, - acceptedRecipients: [], - rejectedRecipients: [], - timestamp: startTime, - secure: this.options.secure || this.socket instanceof plugins.tls.TLSSocket, - authenticated: !!this.options.auth - }; - try { - logger.log('info', `Sending email to ${email.getAllRecipients().join(', ')}`); - // Apply DKIM signing if configured - if (this.options.dkim?.enabled) { - await this.applyDkimSignature(email); - result.dkimSigned = true; - } - // Get envelope and recipients - const envelope_from = email.getEnvelopeFrom() || email.from; - const recipients = email.getAllRecipients(); - // Check if we can use pipelining for MAIL FROM and RCPT TO commands - if (this.supportsPipelining && recipients.length > 0) { - logger.log('debug', 'Using SMTP pipelining for sending'); - // Send MAIL FROM command first (always needed) - const mailFromCmd = `MAIL FROM:<${envelope_from}> SIZE=${this.getEmailSize(email)}`; - let mailFromResponse; - try { - mailFromResponse = await this.sendCommand(mailFromCmd); - if (!mailFromResponse.startsWith('250')) { - throw new MtaDeliveryError(`MAIL FROM command failed: ${mailFromResponse}`, { - data: { - command: mailFromCmd, - response: mailFromResponse - } - }); - } - } - catch (error) { - logger.log('error', `MAIL FROM failed: ${error.message}`); - throw error; - } - // Pipeline all RCPT TO commands - const rcptPromises = recipients.map(recipient => { - return this.sendCommand(`RCPT TO:<${recipient}>`) - .then(response => { - if (response.startsWith('250')) { - result.acceptedRecipients.push(recipient); - return { recipient, accepted: true, response }; - } - else { - result.rejectedRecipients.push(recipient); - logger.log('warn', `Recipient ${recipient} rejected: ${response}`); - return { recipient, accepted: false, response }; - } - }) - .catch(error => { - result.rejectedRecipients.push(recipient); - logger.log('warn', `Recipient ${recipient} rejected with error: ${error.message}`); - return { recipient, accepted: false, error: error.message }; - }); - }); - // Wait for all RCPT TO commands to complete - await Promise.all(rcptPromises); - } - else { - // Fall back to sequential commands if pipelining not supported - logger.log('debug', 'Using sequential SMTP commands for sending'); - // Send MAIL FROM - await this.sendCommand(`MAIL FROM:<${envelope_from}> SIZE=${this.getEmailSize(email)}`); - // Send RCPT TO for each recipient - for (const recipient of recipients) { - try { - await this.sendCommand(`RCPT TO:<${recipient}>`); - result.acceptedRecipients.push(recipient); - } - catch (error) { - logger.log('warn', `Recipient ${recipient} rejected: ${error.message}`); - result.rejectedRecipients.push(recipient); - } - } - } - // Check if at least one recipient was accepted - if (result.acceptedRecipients.length === 0) { - throw new MtaDeliveryError('All recipients were rejected', { - data: { - recipients, - rejectedRecipients: result.rejectedRecipients - } - }); - } - // Send DATA - const dataResponse = await this.sendCommand('DATA'); - if (!dataResponse.startsWith('354')) { - throw new MtaProtocolError(`Failed to start DATA phase: ${dataResponse}`, { - data: { - response: dataResponse - } - }); - } - // Format email content efficiently - const emailContent = await this.getFormattedEmail(email); - // Send email content - const finalResponse = await this.sendCommand(emailContent + '\r\n.'); - // Extract message ID if available - const messageIdMatch = finalResponse.match(/\[(.*?)\]/); - if (messageIdMatch) { - result.messageId = messageIdMatch[1]; - } - result.success = true; - result.response = finalResponse; - logger.log('info', `Email sent successfully to ${result.acceptedRecipients.join(', ')}`); - // Log security event - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.INFO, - type: SecurityEventType.EMAIL_DELIVERY, - message: 'Email sent successfully', - details: { - recipients: result.acceptedRecipients, - rejectedRecipients: result.rejectedRecipients, - messageId: result.messageId, - secure: result.secure, - authenticated: result.authenticated, - server: `${this.options.host}:${this.options.port}`, - dkimSigned: result.dkimSigned - }, - success: true - }); - return result; - } - catch (error) { - logger.log('error', `Failed to send email: ${error.message}`); - // Format error for result - result.error = error.message; - // Extract SMTP code if available - if (error.context?.data?.statusCode) { - result.responseCode = error.context.data.statusCode; - } - // Log security event - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.ERROR, - type: SecurityEventType.EMAIL_DELIVERY, - message: 'Email delivery failed', - details: { - error: error.message, - server: `${this.options.host}:${this.options.port}`, - recipients: email.getAllRecipients(), - acceptedRecipients: result.acceptedRecipients, - rejectedRecipients: result.rejectedRecipients, - secure: result.secure, - authenticated: result.authenticated - }, - success: false - }); - return result; - } - } - /** - * Apply DKIM signature to email - * @param email Email to sign - */ - async applyDkimSignature(email) { - if (!this.options.dkim?.enabled || !this.options.dkim?.privateKey) { - return; - } - try { - logger.log('debug', `Signing email with DKIM for domain ${this.options.dkim.domain}`); - const emailContent = await this.getFormattedEmail(email); - // Sign via Rust bridge - const bridge = RustSecurityBridge.getInstance(); - const signResult = await bridge.signDkim({ - rawMessage: emailContent, - domain: this.options.dkim.domain, - selector: this.options.dkim.selector, - privateKey: this.options.dkim.privateKey, - }); - if (signResult.header) { - email.addHeader('DKIM-Signature', signResult.header); - } - logger.log('debug', 'DKIM signature applied successfully'); - } - catch (error) { - logger.log('error', `Failed to apply DKIM signature: ${error.message}`); - throw error; - } - } - /** - * Format email for SMTP transmission - * @param email Email to format - */ - async getFormattedEmail(email) { - // This is a simplified implementation - // In a full implementation, this would use proper MIME formatting - let content = ''; - // Add headers - content += `From: ${email.from}\r\n`; - content += `To: ${email.to.join(', ')}\r\n`; - content += `Subject: ${email.subject}\r\n`; - content += `Date: ${new Date().toUTCString()}\r\n`; - content += `Message-ID: <${plugins.uuid.v4()}@${this.options.domain}>\r\n`; - // Add additional headers - for (const [name, value] of Object.entries(email.headers || {})) { - content += `${name}: ${value}\r\n`; - } - // Add content type for multipart - if (email.attachments && email.attachments.length > 0) { - const boundary = `----_=_NextPart_${Math.random().toString(36).substr(2)}`; - content += `MIME-Version: 1.0\r\n`; - content += `Content-Type: multipart/mixed; boundary="${boundary}"\r\n`; - content += `\r\n`; - // Add text part - content += `--${boundary}\r\n`; - content += `Content-Type: text/plain; charset="UTF-8"\r\n`; - content += `\r\n`; - content += `${email.text}\r\n`; - // Add HTML part if present - if (email.html) { - content += `--${boundary}\r\n`; - content += `Content-Type: text/html; charset="UTF-8"\r\n`; - content += `\r\n`; - content += `${email.html}\r\n`; - } - // Add attachments - for (const attachment of email.attachments) { - content += `--${boundary}\r\n`; - content += `Content-Type: ${attachment.contentType || 'application/octet-stream'}; name="${attachment.filename}"\r\n`; - content += `Content-Disposition: attachment; filename="${attachment.filename}"\r\n`; - content += `Content-Transfer-Encoding: base64\r\n`; - content += `\r\n`; - // Add base64 encoded content - const base64Content = attachment.content.toString('base64'); - // Split into lines of 76 characters - for (let i = 0; i < base64Content.length; i += 76) { - content += base64Content.substring(i, i + 76) + '\r\n'; - } - } - // End boundary - content += `--${boundary}--\r\n`; - } - else { - // Simple email with just text - content += `Content-Type: text/plain; charset="UTF-8"\r\n`; - content += `\r\n`; - content += `${email.text}\r\n`; - } - return content; - } - /** - * Get size of email in bytes - * @param email Email to measure - */ - getEmailSize(email) { - // Simplified size estimation - let size = 0; - // Headers - size += `From: ${email.from}\r\n`.length; - size += `To: ${email.to.join(', ')}\r\n`.length; - size += `Subject: ${email.subject}\r\n`.length; - // Body - size += (email.text?.length || 0) + 2; // +2 for CRLF - // HTML part if present - if (email.html) { - size += email.html.length + 2; - } - // Attachments - for (const attachment of email.attachments || []) { - size += attachment.content.length; - } - // Add overhead for MIME boundaries and headers - const overhead = email.attachments?.length ? 1000 + (email.attachments.length * 200) : 200; - return size + overhead; - } - /** - * Send SMTP command and wait for response - * @param command SMTP command to send - */ - // Queue for command pipelining - commandQueue = []; - // Flag to indicate if we're currently processing commands - processingCommands = false; - // Flag to indicate if server supports pipelining - supportsPipelining = false; - /** - * Send an SMTP command and wait for response - * @param command SMTP command to send - * @param allowPipelining Whether this command can be pipelined - */ - async sendCommand(command, allowPipelining = true) { - if (!this.socket) { - throw new MtaConnectionError('Not connected to server', { - data: { - host: this.options.host, - port: this.options.port - } - }); - } - // Log command if not sensitive - if (!command.startsWith('AUTH')) { - logger.log('debug', `> ${command}`); - } - else { - logger.log('debug', '> AUTH ***'); - } - return new Promise((resolve, reject) => { - // Set up timeout for command - const timeout = setTimeout(() => { - // Remove this command from the queue if it times out - const index = this.commandQueue.findIndex(item => item.command === command); - if (index !== -1) { - this.commandQueue.splice(index, 1); - } - reject(MtaTimeoutError.commandTimeout(command.split(' ')[0], this.options.host, this.options.commandTimeout)); - }, this.options.commandTimeout); - // Add command to the queue - this.commandQueue.push({ - command, - resolve, - reject, - timeout - }); - // Process command queue if we can pipeline or if not currently processing commands - if ((this.supportsPipelining && allowPipelining) || !this.processingCommands) { - this.processCommandQueue(); - } - }); - } - /** - * Process the command queue - either one by one or pipelined if supported - */ - processCommandQueue() { - if (this.processingCommands || this.commandQueue.length === 0 || !this.socket) { - return; - } - this.processingCommands = true; - try { - // If pipelining is supported, send all commands at once - if (this.supportsPipelining) { - // Send all commands in queue at once - const commands = this.commandQueue.map(item => item.command).join('\r\n') + '\r\n'; - this.socket.write(commands, (err) => { - if (err) { - // Handle write error for all commands - const error = new MtaConnectionError(`Failed to send commands: ${err.message}`, { - data: { - error: err.message - } - }); - // Fail all pending commands - while (this.commandQueue.length > 0) { - const item = this.commandQueue.shift(); - clearTimeout(item.timeout); - item.reject(error); - } - this.processingCommands = false; - } - }); - // Process responses one by one in order - this.processResponses(); - } - else { - // Process commands one by one if pipelining not supported - this.processNextCommand(); - } - } - catch (error) { - logger.log('error', `Error processing command queue: ${error.message}`); - this.processingCommands = false; - } - } - /** - * Process the next command in the queue (non-pipelined mode) - */ - processNextCommand() { - if (this.commandQueue.length === 0 || !this.socket) { - this.processingCommands = false; - return; - } - const currentCommand = this.commandQueue[0]; - this.socket.write(currentCommand.command + '\r\n', (err) => { - if (err) { - // Handle write error - const error = new MtaConnectionError(`Failed to send command: ${err.message}`, { - data: { - command: currentCommand.command.split(' ')[0], - error: err.message - } - }); - // Remove from queue - this.commandQueue.shift(); - clearTimeout(currentCommand.timeout); - currentCommand.reject(error); - // Continue with next command - this.processNextCommand(); - return; - } - // Read response - this.readResponse() - .then((response) => { - // Remove from queue and resolve - this.commandQueue.shift(); - clearTimeout(currentCommand.timeout); - currentCommand.resolve(response); - // Process next command - this.processNextCommand(); - }) - .catch((err) => { - // Remove from queue and reject - this.commandQueue.shift(); - clearTimeout(currentCommand.timeout); - currentCommand.reject(err); - // Process next command - this.processNextCommand(); - }); - }); - } - /** - * Process responses for pipelined commands - */ - async processResponses() { - try { - // Process responses for each command in order - while (this.commandQueue.length > 0) { - const currentCommand = this.commandQueue[0]; - try { - // Wait for response - const response = await this.readResponse(); - // Remove from queue and resolve - this.commandQueue.shift(); - clearTimeout(currentCommand.timeout); - currentCommand.resolve(response); - } - catch (error) { - // Remove from queue and reject - this.commandQueue.shift(); - clearTimeout(currentCommand.timeout); - currentCommand.reject(error); - // Stop processing if this is a critical error - if (error instanceof MtaConnectionError && - (error.message.includes('Connection closed') || error.message.includes('Not connected'))) { - break; - } - } - } - } - catch (error) { - logger.log('error', `Error processing responses: ${error.message}`); - } - finally { - this.processingCommands = false; - } - } - /** - * Read response from the server - */ - async readResponse() { - if (!this.socket) { - throw new MtaConnectionError('Not connected to server', { - data: { - host: this.options.host, - port: this.options.port - } - }); - } - return new Promise((resolve, reject) => { - // Use an array to collect response chunks instead of string concatenation - const responseChunks = []; - // Single function to clean up all listeners - const cleanupListeners = () => { - if (!this.socket) - return; - this.socket.removeListener('data', onData); - this.socket.removeListener('error', onError); - this.socket.removeListener('close', onClose); - this.socket.removeListener('end', onEnd); - }; - const onData = (data) => { - // Store buffer directly, avoiding unnecessary string conversion - responseChunks.push(data); - // Convert to string only for response checking - const responseData = Buffer.concat(responseChunks).toString(); - // Check if this is a complete response - if (this.isCompleteResponse(responseData)) { - // Clean up listeners - cleanupListeners(); - const trimmedResponse = responseData.trim(); - logger.log('debug', `< ${trimmedResponse}`); - // Check if this is an error response - if (this.isErrorResponse(responseData)) { - const code = responseData.substring(0, 3); - reject(this.createErrorFromResponse(trimmedResponse, code)); - } - else { - resolve(trimmedResponse); - } - } - }; - const onError = (err) => { - cleanupListeners(); - reject(new MtaConnectionError(`Socket error while waiting for response: ${err.message}`, { - data: { - error: err.message - } - })); - }; - const onClose = () => { - cleanupListeners(); - const responseData = Buffer.concat(responseChunks).toString(); - reject(new MtaConnectionError('Connection closed while waiting for response', { - data: { - partialResponse: responseData - } - })); - }; - const onEnd = () => { - cleanupListeners(); - const responseData = Buffer.concat(responseChunks).toString(); - reject(new MtaConnectionError('Connection ended while waiting for response', { - data: { - partialResponse: responseData - } - })); - }; - // Set up listeners - this.socket.on('data', onData); - this.socket.once('error', onError); - this.socket.once('close', onClose); - this.socket.once('end', onEnd); - }); - } - /** - * Check if the response is complete - * @param response Response to check - */ - isCompleteResponse(response) { - // Check if it's a multi-line response - const lines = response.split('\r\n'); - const lastLine = lines[lines.length - 2]; // Second to last because of the trailing CRLF - // Check if the last line starts with a code followed by a space - // If it does, this is a complete response - if (lastLine && /^\d{3} /.test(lastLine)) { - return true; - } - // For single line responses - if (lines.length === 2 && lines[0].length >= 3 && /^\d{3} /.test(lines[0])) { - return true; - } - return false; - } - /** - * Check if the response is an error - * @param response Response to check - */ - isErrorResponse(response) { - // Get the status code (first 3 characters) - const code = response.substring(0, 3); - // 4xx and 5xx are error codes - return code.startsWith('4') || code.startsWith('5'); - } - /** - * Create appropriate error from response - * @param response Error response - * @param code SMTP status code - */ - createErrorFromResponse(response, code) { - // Extract message part - const message = response.substring(4).trim(); - switch (code.charAt(0)) { - case '4': // Temporary errors - return MtaDeliveryError.temporary(message, 'recipient', code, response); - case '5': // Permanent errors - return MtaDeliveryError.permanent(message, 'recipient', code, response); - default: - return new MtaDeliveryError(`Unexpected error response: ${response}`, { - data: { - response, - code - } - }); - } - } - /** - * Close the connection to the server - */ - async close() { - if (!this.connected || !this.socket) { - return; - } - try { - // Send QUIT - await this.sendCommand('QUIT'); - } - catch (error) { - logger.log('warn', `Error sending QUIT command: ${error.message}`); - } - finally { - // Close socket - this.socket.destroy(); - this.socket = undefined; - this.connected = false; - logger.log('info', 'SMTP connection closed'); - } - } - /** - * Checks if the connection is active - */ - isConnected() { - return this.connected && !!this.socket; - } - /** - * Update SMTP client options - * @param options New options - */ - updateOptions(options) { - this.options = { - ...this.options, - ...options - }; - logger.log('info', 'SMTP client options updated'); - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5zbXRwLmNsaWVudC5sZWdhY3kuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi90cy9tYWlsL2RlbGl2ZXJ5L2NsYXNzZXMuc210cC5jbGllbnQubGVnYWN5LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sS0FBSyxPQUFPLE1BQU0sa0JBQWtCLENBQUM7QUFDNUMsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQ3pDLE9BQU8sRUFDTCxjQUFjLEVBQ2QsZ0JBQWdCLEVBQ2hCLGlCQUFpQixFQUNsQixNQUFNLHlCQUF5QixDQUFDO0FBQ2pDLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLDhDQUE4QyxDQUFDO0FBRWxGLE9BQU8sRUFDTCxrQkFBa0IsRUFDbEIsc0JBQXNCLEVBQ3RCLGdCQUFnQixFQUNoQixxQkFBcUIsRUFDckIsZUFBZSxFQUNmLGdCQUFnQixFQUNqQixNQUFNLHVCQUF1QixDQUFDO0FBRS9CLE9BQU8sRUFBRSxLQUFLLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQztBQWtMakQ7O0dBRUc7QUFDSCxNQUFNLE9BQU8sVUFBVTtJQUNiLE9BQU8sQ0FBcUI7SUFDNUIsU0FBUyxHQUFZLEtBQUssQ0FBQztJQUMzQixNQUFNLENBQThDO0lBQ3BELG1CQUFtQixHQUFnQixJQUFJLEdBQUcsRUFBRSxDQUFDO0lBRXJEOzs7T0FHRztJQUNILFlBQVksT0FBMkI7UUFDckMsc0JBQXNCO1FBQ3RCLElBQUksQ0FBQyxPQUFPLEdBQUc7WUFDYixHQUFHLE9BQU87WUFDVixpQkFBaUIsRUFBRSxPQUFPLENBQUMsaUJBQWlCLElBQUksS0FBSyxFQUFFLGFBQWE7WUFDcEUsYUFBYSxFQUFFLE9BQU8sQ0FBQyxhQUFhLElBQUksS0FBSyxFQUFFLGFBQWE7WUFDNUQsY0FBYyxFQUFFLE9BQU8sQ0FBQyxjQUFjLElBQUksS0FBSyxFQUFFLGFBQWE7WUFDOUQsTUFBTSxFQUFFLE9BQU8sQ0FB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 \ No newline at end of file diff --git a/dist_ts/mail/delivery/classes.unified.rate.limiter.d.ts b/dist_ts/mail/delivery/classes.unified.rate.limiter.d.ts deleted file mode 100644 index ef3d2f7..0000000 --- a/dist_ts/mail/delivery/classes.unified.rate.limiter.d.ts +++ /dev/null @@ -1,200 +0,0 @@ -import { EventEmitter } from 'node:events'; -/** - * Interface for rate limit configuration - */ -export interface IRateLimitConfig { - maxMessagesPerMinute?: number; - maxRecipientsPerMessage?: number; - maxConnectionsPerIP?: number; - maxErrorsPerIP?: number; - maxAuthFailuresPerIP?: number; - blockDuration?: number; -} -/** - * Interface for hierarchical rate limits - */ -export interface IHierarchicalRateLimits { - global: IRateLimitConfig; - patterns?: Record; - ips?: Record; - domains?: Record; - blocks?: Record; -} -/** - * Rate limiter statistics - */ -export interface IRateLimiterStats { - activeCounters: number; - totalBlocked: number; - currentlyBlocked: number; - byPattern: Record; - byIp: Record; -} -/** - * Result of a rate limit check - */ -export interface IRateLimitResult { - allowed: boolean; - reason?: string; - limit?: number; - current?: number; - resetIn?: number; -} -/** - * Unified rate limiter for all email processing modes - */ -export declare class UnifiedRateLimiter extends EventEmitter { - private config; - private counters; - private patternCounters; - private ipCounters; - private domainCounters; - private cleanupInterval?; - private stats; - /** - * Create a new unified rate limiter - * @param config Rate limit configuration - */ - constructor(config: IHierarchicalRateLimits); - /** - * Start the cleanup interval - */ - private startCleanupInterval; - /** - * Stop the cleanup interval - */ - stop(): void; - /** - * Destroy the rate limiter and clean up all resources - */ - destroy(): void; - /** - * Clean up expired counters and blocks - */ - private cleanup; - /** - * Check if a message is allowed by rate limits - * @param email Email address - * @param ip IP address - * @param recipients Number of recipients - * @param pattern Matched pattern - * @param domain Domain name for domain-specific limits - * @returns Result of rate limit check - */ - checkMessageLimit(email: string, ip: string, recipients: number, pattern?: string, domain?: string): IRateLimitResult; - /** - * Check global message rate limit - * @param email Email address - */ - private checkGlobalMessageLimit; - /** - * Check pattern-specific message rate limit - * @param pattern Pattern to check - */ - private checkPatternMessageLimit; - /** - * Check domain-specific message rate limit - * @param domain Domain to check - */ - private checkDomainMessageLimit; - /** - * Check IP-specific message rate limit - * @param ip IP address - */ - private checkIpMessageLimit; - /** - * Check recipient limit - * @param email Email address - * @param recipients Number of recipients - * @param pattern Matched pattern - * @param domain Domain name - */ - private checkRecipientLimit; - /** - * Record a connection from an IP - * @param ip IP address - * @returns Result of rate limit check - */ - recordConnection(ip: string): IRateLimitResult; - /** - * Record an error from an IP - * @param ip IP address - * @returns True if IP should be blocked - */ - recordError(ip: string): boolean; - /** - * Record an authentication failure from an IP - * @param ip IP address - * @returns True if IP should be blocked - */ - recordAuthFailure(ip: string): boolean; - /** - * Block an IP address - * @param ip IP address to block - * @param duration Override the default block duration (milliseconds) - */ - blockIp(ip: string, duration?: number): void; - /** - * Unblock an IP address - * @param ip IP address to unblock - */ - unblockIp(ip: string): void; - /** - * Check if an IP is blocked - * @param ip IP address to check - */ - isIpBlocked(ip: string): boolean; - /** - * Get the time until a block is released - * @param ip IP address - * @returns Milliseconds until release or 0 if not blocked - */ - getBlockReleaseTime(ip: string): number; - /** - * Update rate limiter statistics - */ - private updateStats; - /** - * Get rate limiter statistics - */ - getStats(): IRateLimiterStats; - /** - * Update rate limiter configuration - * @param config New configuration - */ - updateConfig(config: Partial): void; - /** - * Get configuration for debugging - */ - getConfig(): IHierarchicalRateLimits; - /** - * Apply domain-specific rate limits - * Merges domain limits with existing configuration - * @param domain Domain name - * @param limits Rate limit configuration for the domain - */ - applyDomainLimits(domain: string, limits: IRateLimitConfig): void; - /** - * Remove domain-specific rate limits - * @param domain Domain name - */ - removeDomainLimits(domain: string): void; - /** - * Get domain-specific rate limits - * @param domain Domain name - * @returns Domain rate limit config or undefined - */ - getDomainLimits(domain: string): IRateLimitConfig | undefined; -} diff --git a/dist_ts/mail/delivery/classes.unified.rate.limiter.js b/dist_ts/mail/delivery/classes.unified.rate.limiter.js deleted file mode 100644 index c17bee5..0000000 --- a/dist_ts/mail/delivery/classes.unified.rate.limiter.js +++ /dev/null @@ -1,820 +0,0 @@ -import * as plugins from '../../plugins.js'; -import { EventEmitter } from 'node:events'; -import { logger } from '../../logger.js'; -import { SecurityLogger, SecurityLogLevel, SecurityEventType } from '../../security/index.js'; -/** - * Unified rate limiter for all email processing modes - */ -export class UnifiedRateLimiter extends EventEmitter { - config; - counters = new Map(); - patternCounters = new Map(); - ipCounters = new Map(); - domainCounters = new Map(); - cleanupInterval; - stats; - /** - * Create a new unified rate limiter - * @param config Rate limit configuration - */ - constructor(config) { - super(); - // Set default configuration - this.config = { - global: { - maxMessagesPerMinute: config.global.maxMessagesPerMinute || 100, - maxRecipientsPerMessage: config.global.maxRecipientsPerMessage || 100, - maxConnectionsPerIP: config.global.maxConnectionsPerIP || 20, - maxErrorsPerIP: config.global.maxErrorsPerIP || 10, - maxAuthFailuresPerIP: config.global.maxAuthFailuresPerIP || 5, - blockDuration: config.global.blockDuration || 3600000 // 1 hour - }, - patterns: config.patterns || {}, - ips: config.ips || {}, - blocks: config.blocks || {} - }; - // Initialize statistics - this.stats = { - activeCounters: 0, - totalBlocked: 0, - currentlyBlocked: 0, - byPattern: {}, - byIp: {} - }; - // Start cleanup interval - this.startCleanupInterval(); - } - /** - * Start the cleanup interval - */ - startCleanupInterval() { - if (this.cleanupInterval) { - clearInterval(this.cleanupInterval); - } - // Run cleanup every minute - this.cleanupInterval = setInterval(() => this.cleanup(), 60000); - } - /** - * Stop the cleanup interval - */ - stop() { - if (this.cleanupInterval) { - clearInterval(this.cleanupInterval); - this.cleanupInterval = undefined; - } - } - /** - * Destroy the rate limiter and clean up all resources - */ - destroy() { - // Stop the cleanup interval - this.stop(); - // Clear all maps to free memory - this.counters.clear(); - this.ipCounters.clear(); - this.patternCounters.clear(); - // Clear blocks - if (this.config.blocks) { - this.config.blocks = {}; - } - // Clear statistics - this.stats = { - activeCounters: 0, - totalBlocked: 0, - currentlyBlocked: 0, - byPattern: {}, - byIp: {} - }; - logger.log('info', 'UnifiedRateLimiter destroyed'); - } - /** - * Clean up expired counters and blocks - */ - cleanup() { - const now = Date.now(); - // Clean up expired blocks - if (this.config.blocks) { - for (const [ip, expiry] of Object.entries(this.config.blocks)) { - if (expiry <= now) { - delete this.config.blocks[ip]; - logger.log('info', `Rate limit block expired for IP ${ip}`); - // Update statistics - if (this.stats.byIp[ip]) { - this.stats.byIp[ip].blocked = false; - } - this.stats.currentlyBlocked--; - } - } - } - // Clean up old counters (older than 10 minutes) - const cutoff = now - 600000; - // Clean global counters - for (const [key, counter] of this.counters.entries()) { - if (counter.lastReset < cutoff) { - this.counters.delete(key); - } - } - // Clean pattern counters - for (const [key, counter] of this.patternCounters.entries()) { - if (counter.lastReset < cutoff) { - this.patternCounters.delete(key); - } - } - // Clean IP counters - for (const [key, counter] of this.ipCounters.entries()) { - if (counter.lastReset < cutoff) { - this.ipCounters.delete(key); - } - } - // Clean domain counters - for (const [key, counter] of this.domainCounters.entries()) { - if (counter.lastReset < cutoff) { - this.domainCounters.delete(key); - } - } - // Update statistics - this.updateStats(); - } - /** - * Check if a message is allowed by rate limits - * @param email Email address - * @param ip IP address - * @param recipients Number of recipients - * @param pattern Matched pattern - * @param domain Domain name for domain-specific limits - * @returns Result of rate limit check - */ - checkMessageLimit(email, ip, recipients, pattern, domain) { - // Check if IP is blocked - if (this.isIpBlocked(ip)) { - return { - allowed: false, - reason: 'IP is blocked', - resetIn: this.getBlockReleaseTime(ip) - }; - } - // Check global message rate limit - const globalResult = this.checkGlobalMessageLimit(email); - if (!globalResult.allowed) { - return globalResult; - } - // Check pattern-specific limit if pattern is provided - if (pattern) { - const patternResult = this.checkPatternMessageLimit(pattern); - if (!patternResult.allowed) { - return patternResult; - } - } - // Check domain-specific limit if domain is provided - if (domain) { - const domainResult = this.checkDomainMessageLimit(domain); - if (!domainResult.allowed) { - return domainResult; - } - } - // Check IP-specific limit - const ipResult = this.checkIpMessageLimit(ip); - if (!ipResult.allowed) { - return ipResult; - } - // Check recipient limit - const recipientResult = this.checkRecipientLimit(email, recipients, pattern, domain); - if (!recipientResult.allowed) { - return recipientResult; - } - // All checks passed - return { allowed: true }; - } - /** - * Check global message rate limit - * @param email Email address - */ - checkGlobalMessageLimit(email) { - const now = Date.now(); - const limit = this.config.global.maxMessagesPerMinute; - if (!limit) { - return { allowed: true }; - } - // Get or create counter - const key = 'global'; - let counter = this.counters.get(key); - if (!counter) { - counter = { - count: 0, - lastReset: now, - recipients: 0, - errors: 0, - authFailures: 0, - connections: 0 - }; - this.counters.set(key, counter); - } - // Check if counter needs to be reset - if (now - counter.lastReset >= 60000) { - counter.count = 0; - counter.lastReset = now; - } - // Check if limit is exceeded - if (counter.count >= limit) { - // Calculate reset time - const resetIn = 60000 - (now - counter.lastReset); - return { - allowed: false, - reason: 'Global message rate limit exceeded', - limit, - current: counter.count, - resetIn - }; - } - // Increment counter - counter.count++; - // Update statistics - this.updateStats(); - return { allowed: true }; - } - /** - * Check pattern-specific message rate limit - * @param pattern Pattern to check - */ - checkPatternMessageLimit(pattern) { - const now = Date.now(); - // Get pattern-specific limit or use global - const patternConfig = this.config.patterns?.[pattern]; - const limit = patternConfig?.maxMessagesPerMinute || this.config.global.maxMessagesPerMinute; - if (!limit) { - return { allowed: true }; - } - // Get or create counter - let counter = this.patternCounters.get(pattern); - if (!counter) { - counter = { - count: 0, - lastReset: now, - recipients: 0, - errors: 0, - authFailures: 0, - connections: 0 - }; - this.patternCounters.set(pattern, counter); - // Initialize pattern stats if needed - if (!this.stats.byPattern[pattern]) { - this.stats.byPattern[pattern] = { - messagesPerMinute: 0, - totalMessages: 0, - totalBlocked: 0 - }; - } - } - // Check if counter needs to be reset - if (now - counter.lastReset >= 60000) { - counter.count = 0; - counter.lastReset = now; - } - // Check if limit is exceeded - if (counter.count >= limit) { - // Calculate reset time - const resetIn = 60000 - (now - counter.lastReset); - // Update statistics - this.stats.byPattern[pattern].totalBlocked++; - this.stats.totalBlocked++; - return { - allowed: false, - reason: `Pattern "${pattern}" message rate limit exceeded`, - limit, - current: counter.count, - resetIn - }; - } - // Increment counter - counter.count++; - // Update statistics - this.stats.byPattern[pattern].messagesPerMinute = counter.count; - this.stats.byPattern[pattern].totalMessages++; - return { allowed: true }; - } - /** - * Check domain-specific message rate limit - * @param domain Domain to check - */ - checkDomainMessageLimit(domain) { - const now = Date.now(); - // Get domain-specific limit or use global - const domainConfig = this.config.domains?.[domain]; - const limit = domainConfig?.maxMessagesPerMinute || this.config.global.maxMessagesPerMinute; - if (!limit) { - return { allowed: true }; - } - // Get or create counter - let counter = this.domainCounters.get(domain); - if (!counter) { - counter = { - count: 0, - lastReset: now, - recipients: 0, - errors: 0, - authFailures: 0, - connections: 0 - }; - this.domainCounters.set(domain, counter); - } - // Check if counter needs to be reset - if (now - counter.lastReset >= 60000) { - counter.count = 0; - counter.lastReset = now; - } - // Check if limit is exceeded - if (counter.count >= limit) { - // Calculate reset time - const resetIn = 60000 - (now - counter.lastReset); - logger.log('warn', `Domain ${domain} rate limit exceeded: ${counter.count}/${limit} messages per minute`); - return { - allowed: false, - reason: `Domain "${domain}" message rate limit exceeded`, - limit, - current: counter.count, - resetIn - }; - } - // Increment counter - counter.count++; - return { allowed: true }; - } - /** - * Check IP-specific message rate limit - * @param ip IP address - */ - checkIpMessageLimit(ip) { - const now = Date.now(); - // Get IP-specific limit or use global - const ipConfig = this.config.ips?.[ip]; - const limit = ipConfig?.maxMessagesPerMinute || this.config.global.maxMessagesPerMinute; - if (!limit) { - return { allowed: true }; - } - // Get or create counter - let counter = this.ipCounters.get(ip); - if (!counter) { - counter = { - count: 0, - lastReset: now, - recipients: 0, - errors: 0, - authFailures: 0, - connections: 0 - }; - this.ipCounters.set(ip, counter); - // Initialize IP stats if needed - if (!this.stats.byIp[ip]) { - this.stats.byIp[ip] = { - messagesPerMinute: 0, - totalMessages: 0, - totalBlocked: 0, - connections: 0, - errors: 0, - authFailures: 0, - blocked: false - }; - } - } - // Check if counter needs to be reset - if (now - counter.lastReset >= 60000) { - counter.count = 0; - counter.lastReset = now; - } - // Check if limit is exceeded - if (counter.count >= limit) { - // Calculate reset time - const resetIn = 60000 - (now - counter.lastReset); - // Update statistics - this.stats.byIp[ip].totalBlocked++; - this.stats.totalBlocked++; - return { - allowed: false, - reason: `IP ${ip} message rate limit exceeded`, - limit, - current: counter.count, - resetIn - }; - } - // Increment counter - counter.count++; - // Update statistics - this.stats.byIp[ip].messagesPerMinute = counter.count; - this.stats.byIp[ip].totalMessages++; - return { allowed: true }; - } - /** - * Check recipient limit - * @param email Email address - * @param recipients Number of recipients - * @param pattern Matched pattern - * @param domain Domain name - */ - checkRecipientLimit(email, recipients, pattern, domain) { - // Get the most specific limit available - let limit = this.config.global.maxRecipientsPerMessage; - // Check pattern-specific limit - if (pattern && this.config.patterns?.[pattern]?.maxRecipientsPerMessage) { - limit = this.config.patterns[pattern].maxRecipientsPerMessage; - } - // Check domain-specific limit (overrides pattern if present) - if (domain && this.config.domains?.[domain]?.maxRecipientsPerMessage) { - limit = this.config.domains[domain].maxRecipientsPerMessage; - } - if (!limit) { - return { allowed: true }; - } - // Check if limit is exceeded - if (recipients > limit) { - return { - allowed: false, - reason: 'Recipient limit exceeded', - limit, - current: recipients - }; - } - return { allowed: true }; - } - /** - * Record a connection from an IP - * @param ip IP address - * @returns Result of rate limit check - */ - recordConnection(ip) { - const now = Date.now(); - // Check if IP is blocked - if (this.isIpBlocked(ip)) { - return { - allowed: false, - reason: 'IP is blocked', - resetIn: this.getBlockReleaseTime(ip) - }; - } - // Get IP-specific limit or use global - const ipConfig = this.config.ips?.[ip]; - const limit = ipConfig?.maxConnectionsPerIP || this.config.global.maxConnectionsPerIP; - if (!limit) { - return { allowed: true }; - } - // Get or create counter - let counter = this.ipCounters.get(ip); - if (!counter) { - counter = { - count: 0, - lastReset: now, - recipients: 0, - errors: 0, - authFailures: 0, - connections: 0 - }; - this.ipCounters.set(ip, counter); - // Initialize IP stats if needed - if (!this.stats.byIp[ip]) { - this.stats.byIp[ip] = { - messagesPerMinute: 0, - totalMessages: 0, - totalBlocked: 0, - connections: 0, - errors: 0, - authFailures: 0, - blocked: false - }; - } - } - // Check if counter needs to be reset - if (now - counter.lastReset >= 60000) { - counter.connections = 0; - counter.lastReset = now; - } - // Check if limit is exceeded - if (counter.connections >= limit) { - // Calculate reset time - const resetIn = 60000 - (now - counter.lastReset); - // Update statistics - this.stats.byIp[ip].totalBlocked++; - this.stats.totalBlocked++; - return { - allowed: false, - reason: `IP ${ip} connection rate limit exceeded`, - limit, - current: counter.connections, - resetIn - }; - } - // Increment counter - counter.connections++; - // Update statistics - this.stats.byIp[ip].connections = counter.connections; - return { allowed: true }; - } - /** - * Record an error from an IP - * @param ip IP address - * @returns True if IP should be blocked - */ - recordError(ip) { - const now = Date.now(); - // Get IP-specific limit or use global - const ipConfig = this.config.ips?.[ip]; - const limit = ipConfig?.maxErrorsPerIP || this.config.global.maxErrorsPerIP; - if (!limit) { - return false; - } - // Get or create counter - let counter = this.ipCounters.get(ip); - if (!counter) { - counter = { - count: 0, - lastReset: now, - recipients: 0, - errors: 0, - authFailures: 0, - connections: 0 - }; - this.ipCounters.set(ip, counter); - // Initialize IP stats if needed - if (!this.stats.byIp[ip]) { - this.stats.byIp[ip] = { - messagesPerMinute: 0, - totalMessages: 0, - totalBlocked: 0, - connections: 0, - errors: 0, - authFailures: 0, - blocked: false - }; - } - } - // Check if counter needs to be reset - if (now - counter.lastReset >= 60000) { - counter.errors = 0; - counter.lastReset = now; - } - // Increment counter - counter.errors++; - // Update statistics - this.stats.byIp[ip].errors = counter.errors; - // Check if limit is exceeded - if (counter.errors >= limit) { - // Block the IP - this.blockIp(ip); - logger.log('warn', `IP ${ip} blocked due to excessive errors (${counter.errors}/${limit})`); - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.WARN, - type: SecurityEventType.RATE_LIMITING, - message: 'IP blocked due to excessive errors', - ipAddress: ip, - details: { - errors: counter.errors, - limit - }, - success: false - }); - return true; - } - return false; - } - /** - * Record an authentication failure from an IP - * @param ip IP address - * @returns True if IP should be blocked - */ - recordAuthFailure(ip) { - const now = Date.now(); - // Get IP-specific limit or use global - const ipConfig = this.config.ips?.[ip]; - const limit = ipConfig?.maxAuthFailuresPerIP || this.config.global.maxAuthFailuresPerIP; - if (!limit) { - return false; - } - // Get or create counter - let counter = this.ipCounters.get(ip); - if (!counter) { - counter = { - count: 0, - lastReset: now, - recipients: 0, - errors: 0, - authFailures: 0, - connections: 0 - }; - this.ipCounters.set(ip, counter); - // Initialize IP stats if needed - if (!this.stats.byIp[ip]) { - this.stats.byIp[ip] = { - messagesPerMinute: 0, - totalMessages: 0, - totalBlocked: 0, - connections: 0, - errors: 0, - authFailures: 0, - blocked: false - }; - } - } - // Check if counter needs to be reset - if (now - counter.lastReset >= 60000) { - counter.authFailures = 0; - counter.lastReset = now; - } - // Increment counter - counter.authFailures++; - // Update statistics - this.stats.byIp[ip].authFailures = counter.authFailures; - // Check if limit is exceeded - if (counter.authFailures >= limit) { - // Block the IP - this.blockIp(ip); - logger.log('warn', `IP ${ip} blocked due to excessive authentication failures (${counter.authFailures}/${limit})`); - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.WARN, - type: SecurityEventType.AUTHENTICATION, - message: 'IP blocked due to excessive authentication failures', - ipAddress: ip, - details: { - authFailures: counter.authFailures, - limit - }, - success: false - }); - return true; - } - return false; - } - /** - * Block an IP address - * @param ip IP address to block - * @param duration Override the default block duration (milliseconds) - */ - blockIp(ip, duration) { - if (!this.config.blocks) { - this.config.blocks = {}; - } - // Set block expiry time - const expiry = Date.now() + (duration || this.config.global.blockDuration || 3600000); - this.config.blocks[ip] = expiry; - // Update statistics - if (!this.stats.byIp[ip]) { - this.stats.byIp[ip] = { - messagesPerMinute: 0, - totalMessages: 0, - totalBlocked: 0, - connections: 0, - errors: 0, - authFailures: 0, - blocked: false - }; - } - this.stats.byIp[ip].blocked = true; - this.stats.currentlyBlocked++; - // Emit event - this.emit('ipBlocked', { - ip, - expiry, - duration: duration || this.config.global.blockDuration - }); - logger.log('warn', `IP ${ip} blocked until ${new Date(expiry).toISOString()}`); - } - /** - * Unblock an IP address - * @param ip IP address to unblock - */ - unblockIp(ip) { - if (!this.config.blocks) { - return; - } - // Remove block - delete this.config.blocks[ip]; - // Update statistics - if (this.stats.byIp[ip]) { - this.stats.byIp[ip].blocked = false; - this.stats.currentlyBlocked--; - } - // Emit event - this.emit('ipUnblocked', { ip }); - logger.log('info', `IP ${ip} unblocked`); - } - /** - * Check if an IP is blocked - * @param ip IP address to check - */ - isIpBlocked(ip) { - if (!this.config.blocks) { - return false; - } - // Check if IP is in blocks - if (!(ip in this.config.blocks)) { - return false; - } - // Check if block has expired - const expiry = this.config.blocks[ip]; - if (expiry <= Date.now()) { - // Remove expired block - delete this.config.blocks[ip]; - // Update statistics - if (this.stats.byIp[ip]) { - this.stats.byIp[ip].blocked = false; - this.stats.currentlyBlocked--; - } - return false; - } - return true; - } - /** - * Get the time until a block is released - * @param ip IP address - * @returns Milliseconds until release or 0 if not blocked - */ - getBlockReleaseTime(ip) { - if (!this.config.blocks || !(ip in this.config.blocks)) { - return 0; - } - const expiry = this.config.blocks[ip]; - const now = Date.now(); - return expiry > now ? expiry - now : 0; - } - /** - * Update rate limiter statistics - */ - updateStats() { - // Update active counters count - this.stats.activeCounters = this.counters.size + this.patternCounters.size + this.ipCounters.size; - // Emit statistics update - this.emit('statsUpdated', this.stats); - } - /** - * Get rate limiter statistics - */ - getStats() { - return { ...this.stats }; - } - /** - * Update rate limiter configuration - * @param config New configuration - */ - updateConfig(config) { - if (config.global) { - this.config.global = { - ...this.config.global, - ...config.global - }; - } - if (config.patterns) { - this.config.patterns = { - ...this.config.patterns, - ...config.patterns - }; - } - if (config.ips) { - this.config.ips = { - ...this.config.ips, - ...config.ips - }; - } - logger.log('info', 'Rate limiter configuration updated'); - } - /** - * Get configuration for debugging - */ - getConfig() { - return { ...this.config }; - } - /** - * Apply domain-specific rate limits - * Merges domain limits with existing configuration - * @param domain Domain name - * @param limits Rate limit configuration for the domain - */ - applyDomainLimits(domain, limits) { - if (!this.config.domains) { - this.config.domains = {}; - } - // Merge the limits with any existing domain config - this.config.domains[domain] = { - ...this.config.domains[domain], - ...limits - }; - logger.log('info', `Applied rate limits for domain ${domain}:`, limits); - } - /** - * Remove domain-specific rate limits - * @param domain Domain name - */ - removeDomainLimits(domain) { - if (this.config.domains && this.config.domains[domain]) { - delete this.config.domains[domain]; - // Also remove the counter - this.domainCounters.delete(domain); - logger.log('info', `Removed rate limits for domain ${domain}`); - } - } - /** - * Get domain-specific rate limits - * @param domain Domain name - * @returns Domain rate limit config or undefined - */ - getDomainLimits(domain) { - return this.config.domains?.[domain]; - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy51bmlmaWVkLnJhdGUubGltaXRlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3RzL21haWwvZGVsaXZlcnkvY2xhc3Nlcy51bmlmaWVkLnJhdGUubGltaXRlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssT0FBTyxNQUFNLGtCQUFrQixDQUFDO0FBQzVDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxhQUFhLENBQUM7QUFDM0MsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQ3pDLE9BQU8sRUFBRSxjQUFjLEVBQUUsZ0JBQWdCLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQWdGOUY7O0dBRUc7QUFDSCxNQUFNLE9BQU8sa0JBQW1CLFNBQVEsWUFBWTtJQUMxQyxNQUFNLENBQTBCO0lBQ2hDLFFBQVEsR0FBK0IsSUFBSSxHQUFHLEVBQUUsQ0FBQztJQUNqRCxlQUFlLEdBQStCLElBQUksR0FBRyxFQUFFLENBQUM7SUFDeEQsVUFBVSxHQUErQixJQUFJLEdBQUcsRUFBRSxDQUFDO0lBQ25ELGNBQWMsR0FBK0IsSUFBSSxHQUFHLEVBQUUsQ0FBQztJQUN2RCxlQUFlLENBQWtCO0lBQ2pDLEtBQUssQ0FBb0I7SUFFakM7OztPQUdHO0lBQ0gsWUFBWSxNQUErQjtRQUN6QyxLQUFLLEVBQUUsQ0FBQztRQUVSLDRCQUE0QjtRQUM1QixJQUFJLENBQUMsTUFBTSxHQUFHO1lBQ1osTUFBTSxFQUFFO2dCQUNOLG9CQUFvQixFQUFFLE1BQU0sQ0FBQyxNQUFNLENBQUMsb0JBQW9CLElBQUksR0FBRztnQkFDL0QsdUJBQXVCLEVBQUUsTUFBTSxDQUFDLE1BQU0sQ0FBQyx1QkFBdUIsSUFBSSxHQUFHO2dCQUNyRSxtQkFBbUIsRUFBRSxNQUFNLENBQUMsTUFBTSxDQUFDLG1CQUFtQixJQUFJLEVBQUU7Z0JBQzVELGNBQWMsRUFBRSxNQUFNLENBQUMsTUFBTSxD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 \ No newline at end of file diff --git a/dist_ts/mail/delivery/index.d.ts b/dist_ts/mail/delivery/index.d.ts deleted file mode 100644 index 7faca12..0000000 --- a/dist_ts/mail/delivery/index.d.ts +++ /dev/null @@ -1,11 +0,0 @@ -export * from './classes.emailsignjob.js'; -export * from './classes.delivery.queue.js'; -export * from './classes.delivery.system.js'; -export { EmailSendJob } from './classes.emailsendjob.js'; -export { DeliveryStatus } from './classes.delivery.system.js'; -export { RateLimiter } from './classes.ratelimiter.js'; -export type { IRateLimitConfig } from './classes.ratelimiter.js'; -export * from './classes.unified.rate.limiter.js'; -export * from './classes.mta.config.js'; -import * as smtpClientMod from './smtpclient/index.js'; -export { smtpClientMod }; diff --git a/dist_ts/mail/delivery/index.js b/dist_ts/mail/delivery/index.js deleted file mode 100644 index 7a2abdf..0000000 --- a/dist_ts/mail/delivery/index.js +++ /dev/null @@ -1,17 +0,0 @@ -// Email delivery components -export * from './classes.emailsignjob.js'; -export * from './classes.delivery.queue.js'; -export * from './classes.delivery.system.js'; -// Handle exports with naming conflicts -export { EmailSendJob } from './classes.emailsendjob.js'; -export { DeliveryStatus } from './classes.delivery.system.js'; -// Rate limiter exports - fix naming conflict -export { RateLimiter } from './classes.ratelimiter.js'; -// Unified rate limiter -export * from './classes.unified.rate.limiter.js'; -// SMTP client and configuration -export * from './classes.mta.config.js'; -// Import and export SMTP modules as namespaces to avoid conflicts -import * as smtpClientMod from './smtpclient/index.js'; -export { smtpClientMod }; -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi90cy9tYWlsL2RlbGl2ZXJ5L2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLDRCQUE0QjtBQUM1QixjQUFjLDJCQUEyQixDQUFDO0FBQzFDLGNBQWMsNkJBQTZCLENBQUM7QUFDNUMsY0FBYyw4QkFBOEIsQ0FBQztBQUU3Qyx1Q0FBdUM7QUFDdkMsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBQ3pELE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSw4QkFBOEIsQ0FBQztBQUU5RCw2Q0FBNkM7QUFDN0MsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBR3ZELHVCQUF1QjtBQUN2QixjQUFjLG1DQUFtQyxDQUFDO0FBRWxELGdDQUFnQztBQUNoQyxjQUFjLHlCQUF5QixDQUFDO0FBRXhDLGtFQUFrRTtBQUNsRSxPQUFPLEtBQUssYUFBYSxNQUFNLHVCQUF1QixDQUFDO0FBRXZELE9BQU8sRUFBRSxhQUFhLEVBQUUsQ0FBQyJ9 \ No newline at end of file diff --git a/dist_ts/mail/delivery/interfaces.d.ts b/dist_ts/mail/delivery/interfaces.d.ts deleted file mode 100644 index 59f7be4..0000000 --- a/dist_ts/mail/delivery/interfaces.d.ts +++ /dev/null @@ -1,243 +0,0 @@ -/** - * SMTP and email delivery interface definitions - */ -import type { Email } from '../core/classes.email.js'; -/** - * SMTP session state enumeration - */ -export declare enum SmtpState { - GREETING = "GREETING", - AFTER_EHLO = "AFTER_EHLO", - MAIL_FROM = "MAIL_FROM", - RCPT_TO = "RCPT_TO", - DATA = "DATA", - DATA_RECEIVING = "DATA_RECEIVING", - FINISHED = "FINISHED" -} -/** - * Email processing mode type - */ -export type EmailProcessingMode = 'forward' | 'mta' | 'process'; -/** - * Envelope recipient information - */ -export interface IEnvelopeRecipient { - /** - * Email address of the recipient - */ - address: string; - /** - * Additional SMTP command arguments - */ - args: Record; -} -/** - * SMTP session envelope information - */ -export interface ISmtpEnvelope { - /** - * Envelope sender (MAIL FROM) information - */ - mailFrom: { - /** - * Email address of the sender - */ - address: string; - /** - * Additional SMTP command arguments - */ - args: Record; - }; - /** - * Envelope recipients (RCPT TO) information - */ - rcptTo: IEnvelopeRecipient[]; -} -/** - * SMTP Session interface - represents an active SMTP connection - */ -export interface ISmtpSession { - /** - * Unique session identifier - */ - id: string; - /** - * Current session state in the SMTP conversation - */ - state: SmtpState; - /** - * Hostname provided by the client in EHLO/HELO command - */ - clientHostname: string; - /** - * MAIL FROM email address (legacy format) - */ - mailFrom: string; - /** - * RCPT TO email addresses (legacy format) - */ - rcptTo: string[]; - /** - * Raw email data being received - */ - emailData: string; - /** - * Chunks of email data for more efficient buffer management - */ - emailDataChunks?: string[]; - /** - * Whether the connection is using TLS - */ - useTLS: boolean; - /** - * Whether the connection has ended - */ - connectionEnded: boolean; - /** - * Remote IP address of the client - */ - remoteAddress: string; - /** - * Whether the connection is secure (TLS) - */ - secure: boolean; - /** - * Whether the client has been authenticated - */ - authenticated: boolean; - /** - * SMTP envelope information (structured format) - */ - envelope: ISmtpEnvelope; - /** - * Email processing mode to use for this session - */ - processingMode?: EmailProcessingMode; - /** - * Timestamp of last activity for session timeout tracking - */ - lastActivity?: number; - /** - * Timeout ID for DATA command timeout - */ - dataTimeoutId?: NodeJS.Timeout; -} -/** - * SMTP authentication data - */ -export interface ISmtpAuth { - /** - * Authentication method used - */ - method: 'PLAIN' | 'LOGIN' | 'OAUTH2' | string; - /** - * Username for authentication - */ - username: string; - /** - * Password or token for authentication - */ - password: string; -} -/** - * SMTP server options - */ -export interface ISmtpServerOptions { - /** - * Port to listen on - */ - port: number; - /** - * TLS private key (PEM format) - */ - key: string; - /** - * TLS certificate (PEM format) - */ - cert: string; - /** - * Server hostname for SMTP banner - */ - hostname?: string; - /** - * Host address to bind to (defaults to all interfaces) - */ - host?: string; - /** - * Secure port for dedicated TLS connections - */ - securePort?: number; - /** - * CA certificates for TLS (PEM format) - */ - ca?: string; - /** - * Maximum size of messages in bytes - */ - maxSize?: number; - /** - * Maximum number of concurrent connections - */ - maxConnections?: number; - /** - * Authentication options - */ - auth?: { - /** - * Whether authentication is required - */ - required: boolean; - /** - * Allowed authentication methods - */ - methods: ('PLAIN' | 'LOGIN' | 'OAUTH2')[]; - }; - /** - * Socket timeout in milliseconds (default: 5 minutes / 300000ms) - */ - socketTimeout?: number; - /** - * Initial connection timeout in milliseconds (default: 30 seconds / 30000ms) - */ - connectionTimeout?: number; - /** - * Interval for checking idle sessions in milliseconds (default: 5 seconds / 5000ms) - * For testing, can be set lower (e.g. 1000ms) to detect timeouts more quickly - */ - cleanupInterval?: number; - /** - * Maximum number of recipients allowed per message (default: 100) - */ - maxRecipients?: number; - /** - * Maximum message size in bytes (default: 10MB / 10485760 bytes) - * This is advertised in the EHLO SIZE extension - */ - size?: number; - /** - * Timeout for the DATA command in milliseconds (default: 60000ms / 1 minute) - * This controls how long to wait for the complete email data - */ - dataTimeout?: number; -} -/** - * Result of SMTP transaction - */ -export interface ISmtpTransactionResult { - /** - * Whether the transaction was successful - */ - success: boolean; - /** - * Error message if failed - */ - error?: string; - /** - * Message ID if successful - */ - messageId?: string; - /** - * Resulting email if successful - */ - email?: Email; -} diff --git a/dist_ts/mail/delivery/interfaces.js b/dist_ts/mail/delivery/interfaces.js deleted file mode 100644 index 7eccd4f..0000000 --- a/dist_ts/mail/delivery/interfaces.js +++ /dev/null @@ -1,17 +0,0 @@ -/** - * SMTP and email delivery interface definitions - */ -/** - * SMTP session state enumeration - */ -export var SmtpState; -(function (SmtpState) { - SmtpState["GREETING"] = "GREETING"; - SmtpState["AFTER_EHLO"] = "AFTER_EHLO"; - SmtpState["MAIL_FROM"] = "MAIL_FROM"; - SmtpState["RCPT_TO"] = "RCPT_TO"; - SmtpState["DATA"] = "DATA"; - SmtpState["DATA_RECEIVING"] = "DATA_RECEIVING"; - SmtpState["FINISHED"] = "FINISHED"; -})(SmtpState || (SmtpState = {})); -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW50ZXJmYWNlcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3RzL21haWwvZGVsaXZlcnkvaW50ZXJmYWNlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUlIOztHQUVHO0FBQ0gsTUFBTSxDQUFOLElBQVksU0FRWDtBQVJELFdBQVksU0FBUztJQUNuQixrQ0FBcUIsQ0FBQTtJQUNyQixzQ0FBeUIsQ0FBQTtJQUN6QixvQ0FBdUIsQ0FBQTtJQUN2QixnQ0FBbUIsQ0FBQTtJQUNuQiwwQkFBYSxDQUFBO0lBQ2IsOENBQWlDLENBQUE7SUFDakMsa0NBQXFCLENBQUE7QUFDdkIsQ0FBQyxFQVJXLFNBQVMsS0FBVCxTQUFTLFFBUXBCIn0= \ No newline at end of file diff --git a/dist_ts/mail/delivery/smtpclient/auth-handler.d.ts b/dist_ts/mail/delivery/smtpclient/auth-handler.d.ts deleted file mode 100644 index ab7e010..0000000 --- a/dist_ts/mail/delivery/smtpclient/auth-handler.d.ts +++ /dev/null @@ -1,43 +0,0 @@ -/** - * SMTP Client Authentication Handler - * Authentication mechanisms implementation - */ -import type { ISmtpConnection, ISmtpAuthOptions, ISmtpClientOptions } from './interfaces.js'; -import type { CommandHandler } from './command-handler.js'; -export declare class AuthHandler { - private options; - private commandHandler; - constructor(options: ISmtpClientOptions, commandHandler: CommandHandler); - /** - * Authenticate using the configured method - */ - authenticate(connection: ISmtpConnection): Promise; - /** - * Authenticate using AUTH PLAIN - */ - private authenticatePlain; - /** - * Authenticate using AUTH LOGIN - */ - private authenticateLogin; - /** - * Authenticate using OAuth2 - */ - private authenticateOAuth2; - /** - * Select appropriate authentication method - */ - private selectAuthMethod; - /** - * Check if OAuth2 token is expired - */ - private isTokenExpired; - /** - * Refresh OAuth2 access token - */ - private refreshOAuth2Token; - /** - * Validate authentication configuration - */ - validateAuthConfig(auth: ISmtpAuthOptions): string[]; -} diff --git a/dist_ts/mail/delivery/smtpclient/auth-handler.js b/dist_ts/mail/delivery/smtpclient/auth-handler.js deleted file mode 100644 index 2722fc8..0000000 --- a/dist_ts/mail/delivery/smtpclient/auth-handler.js +++ /dev/null @@ -1,190 +0,0 @@ -/** - * SMTP Client Authentication Handler - * Authentication mechanisms implementation - */ -import { AUTH_METHODS } from './constants.js'; -import { encodeAuthPlain, encodeAuthLogin, generateOAuth2String, isSuccessCode } from './utils/helpers.js'; -import { logAuthentication, logDebug } from './utils/logging.js'; -export class AuthHandler { - options; - commandHandler; - constructor(options, commandHandler) { - this.options = options; - this.commandHandler = commandHandler; - } - /** - * Authenticate using the configured method - */ - async authenticate(connection) { - if (!this.options.auth) { - logDebug('No authentication configured', this.options); - return; - } - const authOptions = this.options.auth; - const capabilities = connection.capabilities; - if (!capabilities || capabilities.authMethods.size === 0) { - throw new Error('Server does not support authentication'); - } - // Determine authentication method - const method = this.selectAuthMethod(authOptions, capabilities.authMethods); - logAuthentication('start', method, this.options); - try { - switch (method) { - case AUTH_METHODS.PLAIN: - await this.authenticatePlain(connection, authOptions); - break; - case AUTH_METHODS.LOGIN: - await this.authenticateLogin(connection, authOptions); - break; - case AUTH_METHODS.OAUTH2: - await this.authenticateOAuth2(connection, authOptions); - break; - default: - throw new Error(`Unsupported authentication method: ${method}`); - } - logAuthentication('success', method, this.options); - } - catch (error) { - logAuthentication('failure', method, this.options, { error }); - throw error; - } - } - /** - * Authenticate using AUTH PLAIN - */ - async authenticatePlain(connection, auth) { - if (!auth.user || !auth.pass) { - throw new Error('Username and password required for PLAIN authentication'); - } - const credentials = encodeAuthPlain(auth.user, auth.pass); - const response = await this.commandHandler.sendAuth(connection, AUTH_METHODS.PLAIN, credentials); - if (!isSuccessCode(response.code)) { - throw new Error(`PLAIN authentication failed: ${response.message}`); - } - } - /** - * Authenticate using AUTH LOGIN - */ - async authenticateLogin(connection, auth) { - if (!auth.user || !auth.pass) { - throw new Error('Username and password required for LOGIN authentication'); - } - // Step 1: Send AUTH LOGIN - let response = await this.commandHandler.sendAuth(connection, AUTH_METHODS.LOGIN); - if (response.code !== 334) { - throw new Error(`LOGIN authentication initiation failed: ${response.message}`); - } - // Step 2: Send username - const encodedUser = encodeAuthLogin(auth.user); - response = await this.commandHandler.sendCommand(connection, encodedUser); - if (response.code !== 334) { - throw new Error(`LOGIN username failed: ${response.message}`); - } - // Step 3: Send password - const encodedPass = encodeAuthLogin(auth.pass); - response = await this.commandHandler.sendCommand(connection, encodedPass); - if (!isSuccessCode(response.code)) { - throw new Error(`LOGIN password failed: ${response.message}`); - } - } - /** - * Authenticate using OAuth2 - */ - async authenticateOAuth2(connection, auth) { - if (!auth.oauth2) { - throw new Error('OAuth2 configuration required for OAUTH2 authentication'); - } - let accessToken = auth.oauth2.accessToken; - // Refresh token if needed - if (!accessToken || this.isTokenExpired(auth.oauth2)) { - accessToken = await this.refreshOAuth2Token(auth.oauth2); - } - const authString = generateOAuth2String(auth.oauth2.user, accessToken); - const response = await this.commandHandler.sendAuth(connection, AUTH_METHODS.OAUTH2, authString); - if (!isSuccessCode(response.code)) { - throw new Error(`OAUTH2 authentication failed: ${response.message}`); - } - } - /** - * Select appropriate authentication method - */ - selectAuthMethod(auth, serverMethods) { - // If method is explicitly specified, use it - if (auth.method && auth.method !== 'AUTO') { - const method = auth.method === 'OAUTH2' ? AUTH_METHODS.OAUTH2 : auth.method; - if (serverMethods.has(method)) { - return method; - } - throw new Error(`Requested authentication method ${auth.method} not supported by server`); - } - // Auto-select based on available credentials and server support - if (auth.oauth2 && serverMethods.has(AUTH_METHODS.OAUTH2)) { - return AUTH_METHODS.OAUTH2; - } - if (auth.user && auth.pass) { - // Prefer PLAIN over LOGIN for simplicity - if (serverMethods.has(AUTH_METHODS.PLAIN)) { - return AUTH_METHODS.PLAIN; - } - if (serverMethods.has(AUTH_METHODS.LOGIN)) { - return AUTH_METHODS.LOGIN; - } - } - throw new Error('No compatible authentication method found'); - } - /** - * Check if OAuth2 token is expired - */ - isTokenExpired(oauth2) { - if (!oauth2.expires) { - return false; // No expiry information, assume valid - } - const now = Date.now(); - const buffer = 300000; // 5 minutes buffer - return oauth2.expires < (now + buffer); - } - /** - * Refresh OAuth2 access token - */ - async refreshOAuth2Token(oauth2) { - // This is a simplified implementation - // In a real implementation, you would make an HTTP request to the OAuth2 provider - logDebug('OAuth2 token refresh required', this.options); - if (!oauth2.refreshToken) { - throw new Error('Refresh token required for OAuth2 token refresh'); - } - // TODO: Implement actual OAuth2 token refresh - // For now, throw an error to indicate this needs to be implemented - throw new Error('OAuth2 token refresh not implemented. Please provide a valid access token.'); - } - /** - * Validate authentication configuration - */ - validateAuthConfig(auth) { - const errors = []; - if (auth.method === 'OAUTH2' || auth.oauth2) { - if (!auth.oauth2) { - errors.push('OAuth2 configuration required when using OAUTH2 method'); - } - else { - if (!auth.oauth2.user) - errors.push('OAuth2 user required'); - if (!auth.oauth2.clientId) - errors.push('OAuth2 clientId required'); - if (!auth.oauth2.clientSecret) - errors.push('OAuth2 clientSecret required'); - if (!auth.oauth2.refreshToken && !auth.oauth2.accessToken) { - errors.push('OAuth2 refreshToken or accessToken required'); - } - } - } - else if (auth.method === 'PLAIN' || auth.method === 'LOGIN' || (!auth.method && (auth.user || auth.pass))) { - if (!auth.user) - errors.push('Username required for basic authentication'); - if (!auth.pass) - errors.push('Password required for basic authentication'); - } - return errors; - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXV0aC1oYW5kbGVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vdHMvbWFpbC9kZWxpdmVyeS9zbXRwY2xpZW50L2F1dGgtaGFuZGxlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFROUMsT0FBTyxFQUNMLGVBQWUsRUFDZixlQUFlLEVBQ2Ysb0JBQW9CLEVBQ3BCLGFBQWEsRUFDZCxNQUFNLG9CQUFvQixDQUFDO0FBQzVCLE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxRQUFRLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQUdqRSxNQUFNLE9BQU8sV0FBVztJQUNkLE9BQU8sQ0FBcUI7SUFDNUIsY0FBYyxDQUFpQjtJQUV2QyxZQUFZLE9BQTJCLEVBQUUsY0FBOEI7UUFDckUsSUFBSSxDQUFDLE9BQU8sR0FBRyxPQUFPLENBQUM7UUFDdkIsSUFBSSxDQUFDLGNBQWMsR0FBRyxjQUFjLENBQUM7SUFDdkMsQ0FBQztJQUVEOztPQUVHO0lBQ0ksS0FBSyxDQUFDLFlBQVksQ0FBQyxVQUEyQjtRQUNuRCxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxJQUFJLEVBQUUsQ0FBQztZQUN2QixRQUFRLENBQUMsOEJBQThCLEVBQUUsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDO1lBQ3ZELE9BQU87UUFDVCxDQUFDO1FBRUQsTUFBTSxXQUFXLEdBQUcsSUFBSSxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUM7UUFDdEMsTUFBTSxZQUFZLEdBQUcsVUFBVSxDQUFDLFlBQVksQ0FBQztRQUU3QyxJQUFJLENBQUMsWUFBWSxJQUFJLFlBQVksQ0FBQyxXQUFXLENBQUMsSUFBSSxLQUFLLENBQUMsRUFBRSxDQUFDO1lBQ3pELE1BQU0sSUFBSSxLQUFLLENBQUMsd0NBQXdDLENBQUMsQ0FBQztRQUM1RCxDQUFDO1FBRUQsa0NBQWtDO1FBQ2xDLE1B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 \ No newline at end of file diff --git a/dist_ts/mail/delivery/smtpclient/command-handler.d.ts b/dist_ts/mail/delivery/smtpclient/command-handler.d.ts deleted file mode 100644 index f07cb1d..0000000 --- a/dist_ts/mail/delivery/smtpclient/command-handler.d.ts +++ /dev/null @@ -1,67 +0,0 @@ -/** - * SMTP Client Command Handler - * SMTP command sending and response parsing - */ -import { EventEmitter } from 'node:events'; -import type { ISmtpConnection, ISmtpResponse, ISmtpClientOptions, ISmtpCapabilities } from './interfaces.js'; -export declare class CommandHandler extends EventEmitter { - private options; - private responseBuffer; - private pendingCommand; - private commandTimeout; - constructor(options: ISmtpClientOptions); - /** - * Send EHLO command and parse capabilities - */ - sendEhlo(connection: ISmtpConnection, domain?: string): Promise; - /** - * Send MAIL FROM command - */ - sendMailFrom(connection: ISmtpConnection, fromAddress: string): Promise; - /** - * Send RCPT TO command - */ - sendRcptTo(connection: ISmtpConnection, toAddress: string): Promise; - /** - * Send DATA command - */ - sendData(connection: ISmtpConnection): Promise; - /** - * Send email data content - */ - sendDataContent(connection: ISmtpConnection, emailData: string): Promise; - /** - * Send RSET command - */ - sendRset(connection: ISmtpConnection): Promise; - /** - * Send NOOP command - */ - sendNoop(connection: ISmtpConnection): Promise; - /** - * Send QUIT command - */ - sendQuit(connection: ISmtpConnection): Promise; - /** - * Send STARTTLS command - */ - sendStartTls(connection: ISmtpConnection): Promise; - /** - * Send AUTH command - */ - sendAuth(connection: ISmtpConnection, method: string, credentials?: string): Promise; - /** - * Send a generic SMTP command - */ - sendCommand(connection: ISmtpConnection, command: string): Promise; - /** - * Send raw data without command formatting - */ - sendRawData(connection: ISmtpConnection, data: string): Promise; - /** - * Wait for server greeting - */ - waitForGreeting(connection: ISmtpConnection): Promise; - private handleIncomingData; - private isCompleteResponse; -} diff --git a/dist_ts/mail/delivery/smtpclient/command-handler.js b/dist_ts/mail/delivery/smtpclient/command-handler.js deleted file mode 100644 index d45c7aa..0000000 --- a/dist_ts/mail/delivery/smtpclient/command-handler.js +++ /dev/null @@ -1,277 +0,0 @@ -/** - * SMTP Client Command Handler - * SMTP command sending and response parsing - */ -import { EventEmitter } from 'node:events'; -import { SMTP_COMMANDS, SMTP_CODES, LINE_ENDINGS } from './constants.js'; -import { parseSmtpResponse, parseEhloResponse, formatCommand, isSuccessCode } from './utils/helpers.js'; -import { logCommand, logDebug } from './utils/logging.js'; -export class CommandHandler extends EventEmitter { - options; - responseBuffer = ''; - pendingCommand = null; - commandTimeout = null; - constructor(options) { - super(); - this.options = options; - } - /** - * Send EHLO command and parse capabilities - */ - async sendEhlo(connection, domain) { - const hostname = domain || this.options.domain || 'localhost'; - const command = `${SMTP_COMMANDS.EHLO} ${hostname}`; - const response = await this.sendCommand(connection, command); - if (!isSuccessCode(response.code)) { - throw new Error(`EHLO failed: ${response.message}`); - } - const capabilities = parseEhloResponse(response.raw); - connection.capabilities = capabilities; - logDebug('EHLO capabilities parsed', this.options, { capabilities }); - return capabilities; - } - /** - * Send MAIL FROM command - */ - async sendMailFrom(connection, fromAddress) { - // Handle empty return path for bounce messages - const command = fromAddress === '' - ? `${SMTP_COMMANDS.MAIL_FROM}:<>` - : `${SMTP_COMMANDS.MAIL_FROM}:<${fromAddress}>`; - return this.sendCommand(connection, command); - } - /** - * Send RCPT TO command - */ - async sendRcptTo(connection, toAddress) { - const command = `${SMTP_COMMANDS.RCPT_TO}:<${toAddress}>`; - return this.sendCommand(connection, command); - } - /** - * Send DATA command - */ - async sendData(connection) { - return this.sendCommand(connection, SMTP_COMMANDS.DATA); - } - /** - * Send email data content - */ - async sendDataContent(connection, emailData) { - // Normalize line endings to CRLF - let data = emailData.replace(/\r\n/g, '\n').replace(/\r/g, '\n').replace(/\n/g, '\r\n'); - // Ensure email data ends with CRLF - if (!data.endsWith(LINE_ENDINGS.CRLF)) { - data += LINE_ENDINGS.CRLF; - } - // Perform dot stuffing (escape lines starting with a dot) - data = data.replace(/\r\n\./g, '\r\n..'); - // Add termination sequence - data += '.' + LINE_ENDINGS.CRLF; - return this.sendRawData(connection, data); - } - /** - * Send RSET command - */ - async sendRset(connection) { - return this.sendCommand(connection, SMTP_COMMANDS.RSET); - } - /** - * Send NOOP command - */ - async sendNoop(connection) { - return this.sendCommand(connection, SMTP_COMMANDS.NOOP); - } - /** - * Send QUIT command - */ - async sendQuit(connection) { - return this.sendCommand(connection, SMTP_COMMANDS.QUIT); - } - /** - * Send STARTTLS command - */ - async sendStartTls(connection) { - return this.sendCommand(connection, SMTP_COMMANDS.STARTTLS); - } - /** - * Send AUTH command - */ - async sendAuth(connection, method, credentials) { - const command = credentials ? - `${SMTP_COMMANDS.AUTH} ${method} ${credentials}` : - `${SMTP_COMMANDS.AUTH} ${method}`; - return this.sendCommand(connection, command); - } - /** - * Send a generic SMTP command - */ - async sendCommand(connection, command) { - return new Promise((resolve, reject) => { - if (this.pendingCommand) { - reject(new Error('Another command is already pending')); - return; - } - this.pendingCommand = { resolve, reject, command }; - // Set command timeout - const timeout = 30000; // 30 seconds - this.commandTimeout = setTimeout(() => { - this.pendingCommand = null; - this.commandTimeout = null; - reject(new Error(`Command timeout: ${command}`)); - }, timeout); - // Set up data handler - const dataHandler = (data) => { - this.handleIncomingData(data.toString()); - }; - connection.socket.on('data', dataHandler); - // Clean up function - const cleanup = () => { - connection.socket.removeListener('data', dataHandler); - if (this.commandTimeout) { - clearTimeout(this.commandTimeout); - this.commandTimeout = null; - } - }; - // Send command - const formattedCommand = command.endsWith(LINE_ENDINGS.CRLF) ? command : formatCommand(command); - logCommand(command, undefined, this.options); - logDebug(`Sending command: ${command}`, this.options); - connection.socket.write(formattedCommand, (error) => { - if (error) { - cleanup(); - this.pendingCommand = null; - reject(error); - } - }); - // Override resolve/reject to include cleanup - const originalResolve = resolve; - const originalReject = reject; - this.pendingCommand.resolve = (response) => { - cleanup(); - this.pendingCommand = null; - logCommand(command, response, this.options); - originalResolve(response); - }; - this.pendingCommand.reject = (error) => { - cleanup(); - this.pendingCommand = null; - originalReject(error); - }; - }); - } - /** - * Send raw data without command formatting - */ - async sendRawData(connection, data) { - return new Promise((resolve, reject) => { - if (this.pendingCommand) { - reject(new Error('Another command is already pending')); - return; - } - this.pendingCommand = { resolve, reject, command: 'DATA_CONTENT' }; - // Set data timeout - const timeout = 60000; // 60 seconds for data - this.commandTimeout = setTimeout(() => { - this.pendingCommand = null; - this.commandTimeout = null; - reject(new Error('Data transmission timeout')); - }, timeout); - // Set up data handler - const dataHandler = (chunk) => { - this.handleIncomingData(chunk.toString()); - }; - connection.socket.on('data', dataHandler); - // Clean up function - const cleanup = () => { - connection.socket.removeListener('data', dataHandler); - if (this.commandTimeout) { - clearTimeout(this.commandTimeout); - this.commandTimeout = null; - } - }; - // Override resolve/reject to include cleanup - const originalResolve = resolve; - const originalReject = reject; - this.pendingCommand.resolve = (response) => { - cleanup(); - this.pendingCommand = null; - originalResolve(response); - }; - this.pendingCommand.reject = (error) => { - cleanup(); - this.pendingCommand = null; - originalReject(error); - }; - // Send data - connection.socket.write(data, (error) => { - if (error) { - cleanup(); - this.pendingCommand = null; - reject(error); - } - }); - }); - } - /** - * Wait for server greeting - */ - async waitForGreeting(connection) { - return new Promise((resolve, reject) => { - const timeout = 30000; // 30 seconds - let timeoutHandler; - const dataHandler = (data) => { - this.responseBuffer += data.toString(); - if (this.isCompleteResponse(this.responseBuffer)) { - clearTimeout(timeoutHandler); - connection.socket.removeListener('data', dataHandler); - const response = parseSmtpResponse(this.responseBuffer); - this.responseBuffer = ''; - if (isSuccessCode(response.code)) { - resolve(response); - } - else { - reject(new Error(`Server greeting failed: ${response.message}`)); - } - } - }; - timeoutHandler = setTimeout(() => { - connection.socket.removeListener('data', dataHandler); - reject(new Error('Greeting timeout')); - }, timeout); - connection.socket.on('data', dataHandler); - }); - } - handleIncomingData(data) { - if (!this.pendingCommand) { - return; - } - this.responseBuffer += data; - if (this.isCompleteResponse(this.responseBuffer)) { - const response = parseSmtpResponse(this.responseBuffer); - this.responseBuffer = ''; - if (isSuccessCode(response.code) || (response.code >= 300 && response.code < 400) || response.code >= 400) { - this.pendingCommand.resolve(response); - } - else { - this.pendingCommand.reject(new Error(`Command failed: ${response.message}`)); - } - } - } - isCompleteResponse(buffer) { - // Check if we have a complete response - const lines = buffer.split(/\r?\n/); - if (lines.length < 1) { - return false; - } - // Check the last non-empty line - for (let i = lines.length - 1; i >= 0; i--) { - const line = lines[i].trim(); - if (line.length > 0) { - // Response is complete if line starts with "XXX " (space after code) - return /^\d{3} /.test(line); - } - } - return false; - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tbWFuZC1oYW5kbGVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vdHMvbWFpbC9kZWxpdmVyeS9zbXRwY2xpZW50L2NvbW1hbmQtaGFuZGxlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sYUFBYSxDQUFDO0FBQzNDLE9BQU8sRUFBRSxhQUFhLEVBQUUsVUFBVSxFQUFFLFlBQVksRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBT3pFLE9BQU8sRUFDTCxpQkFBaUIsRUFDakIsaUJBQWlCLEVBQ2pCLGFBQWEsRUFDYixhQUFhLEVBQ2QsTUFBTSxvQkFBb0IsQ0FBQztBQUM1QixPQUFPLEVBQUUsVUFBVSxFQUFFLFFBQVEsRUFBRSxNQUFNLG9CQUFvQixDQUFDO0FBRTFELE1BQU0sT0FBTyxjQUFlLFNBQVEsWUFBWTtJQUN0QyxPQUFPLENBQXFCO0lBQzVCLGNBQWMsR0FBVyxFQUFFLENBQUM7SUFDNUIsY0FBYyxHQUFvRSxJQUFJLENBQUM7SUFDdkYsY0FBYyxHQUEwQixJQUFJLENBQUM7SUFFckQsWUFBWSxPQUEyQjtRQUNyQyxLQUFLLEVBQUUsQ0FBQztRQUNSLElBQUksQ0FBQyxPQUFPLEdBQUcsT0FBTyxDQUFDO0lBQ3pCLENBQUM7SUFFRDs7T0FFRztJQUNJLEtBQUssQ0FBQyxRQUFRLENBQUMsVUFBMkIsRUFBRSxNQUFlO1FBQ2hFLE1BQU0sUUFBUSxHQUFHLE1BQU0sSUFBSSxJQUFJLENBQUMsT0FBTyxDQUFDLE1BQU0sSUFBSSxXQUFXLENBQUM7UUFDOUQsTUFBTSxPQUFPLEdBQUcsR0FBRyxhQUFhLENBQUMsSUFBSSxJQUFJLFFBQVEsRUFBRSxDQUFDO1FBRXBELE1BQU0sUUFBUSxHQUFHLE1BQU0sSUFBSSxDQUFDLFdBQVcsQ0FBQyxVQUFVLEVBQUUsT0FBTyxDQUFDLENBQUM7UUFFN0QsSUFBSSxDQUFDLGFBQWEsQ0FBQyxRQUFRLENB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 \ No newline at end of file diff --git a/dist_ts/mail/delivery/smtpclient/connection-manager.d.ts b/dist_ts/mail/delivery/smtpclient/connection-manager.d.ts deleted file mode 100644 index 45eb4af..0000000 --- a/dist_ts/mail/delivery/smtpclient/connection-manager.d.ts +++ /dev/null @@ -1,48 +0,0 @@ -/** - * SMTP Client Connection Manager - * Connection pooling and lifecycle management - */ -import { EventEmitter } from 'node:events'; -import type { ISmtpClientOptions, ISmtpConnection, IConnectionPoolStatus } from './interfaces.js'; -export declare class ConnectionManager extends EventEmitter { - private options; - private connections; - private pendingConnections; - private idleTimeout; - constructor(options: ISmtpClientOptions); - /** - * Get or create a connection - */ - getConnection(): Promise; - /** - * Create a new connection - */ - createConnection(): Promise; - /** - * Release a connection back to the pool or close it - */ - releaseConnection(connection: ISmtpConnection): void; - /** - * Close a specific connection - */ - closeConnection(connection: ISmtpConnection): void; - /** - * Close all connections - */ - closeAllConnections(): void; - /** - * Get connection pool status - */ - getPoolStatus(): IConnectionPoolStatus; - /** - * Update connection activity timestamp - */ - updateActivity(connection: ISmtpConnection): void; - private establishSocket; - private setupSocketHandlers; - private findIdleConnection; - private shouldReuseConnection; - private getActiveConnectionCount; - private getConnectionId; - private setupIdleCleanup; -} diff --git a/dist_ts/mail/delivery/smtpclient/connection-manager.js b/dist_ts/mail/delivery/smtpclient/connection-manager.js deleted file mode 100644 index e9dca6b..0000000 --- a/dist_ts/mail/delivery/smtpclient/connection-manager.js +++ /dev/null @@ -1,239 +0,0 @@ -/** - * SMTP Client Connection Manager - * Connection pooling and lifecycle management - */ -import * as net from 'node:net'; -import * as tls from 'node:tls'; -import { EventEmitter } from 'node:events'; -import { DEFAULTS, CONNECTION_STATES } from './constants.js'; -import { logConnection, logDebug } from './utils/logging.js'; -import { generateConnectionId } from './utils/helpers.js'; -export class ConnectionManager extends EventEmitter { - options; - connections = new Map(); - pendingConnections = new Set(); - idleTimeout = null; - constructor(options) { - super(); - this.options = options; - this.setupIdleCleanup(); - } - /** - * Get or create a connection - */ - async getConnection() { - // Try to reuse an idle connection if pooling is enabled - if (this.options.pool) { - const idleConnection = this.findIdleConnection(); - if (idleConnection) { - const connectionId = this.getConnectionId(idleConnection) || 'unknown'; - logDebug('Reusing idle connection', this.options, { connectionId }); - return idleConnection; - } - // Check if we can create a new connection - if (this.getActiveConnectionCount() >= (this.options.maxConnections || DEFAULTS.MAX_CONNECTIONS)) { - throw new Error('Maximum number of connections reached'); - } - } - return this.createConnection(); - } - /** - * Create a new connection - */ - async createConnection() { - const connectionId = generateConnectionId(); - try { - this.pendingConnections.add(connectionId); - logConnection('connecting', this.options, { connectionId }); - const socket = await this.establishSocket(); - const connection = { - socket, - state: CONNECTION_STATES.CONNECTED, - options: this.options, - secure: this.options.secure || false, - createdAt: new Date(), - lastActivity: new Date(), - messageCount: 0 - }; - this.setupSocketHandlers(socket, connectionId); - this.connections.set(connectionId, connection); - this.pendingConnections.delete(connectionId); - logConnection('connected', this.options, { connectionId }); - this.emit('connection', connection); - return connection; - } - catch (error) { - this.pendingConnections.delete(connectionId); - logConnection('error', this.options, { connectionId, error }); - throw error; - } - } - /** - * Release a connection back to the pool or close it - */ - releaseConnection(connection) { - const connectionId = this.getConnectionId(connection); - if (!connectionId || !this.connections.has(connectionId)) { - return; - } - if (this.options.pool && this.shouldReuseConnection(connection)) { - // Return to pool - connection.state = CONNECTION_STATES.READY; - connection.lastActivity = new Date(); - logDebug('Connection returned to pool', this.options, { connectionId }); - } - else { - // Close connection - this.closeConnection(connection); - } - } - /** - * Close a specific connection - */ - closeConnection(connection) { - const connectionId = this.getConnectionId(connection); - if (connectionId) { - this.connections.delete(connectionId); - } - connection.state = CONNECTION_STATES.CLOSING; - try { - if (!connection.socket.destroyed) { - connection.socket.destroy(); - } - } - catch (error) { - logDebug('Error closing connection', this.options, { error }); - } - logConnection('disconnected', this.options, { connectionId }); - this.emit('disconnect', connection); - } - /** - * Close all connections - */ - closeAllConnections() { - logDebug('Closing all connections', this.options); - for (const connection of this.connections.values()) { - this.closeConnection(connection); - } - this.connections.clear(); - this.pendingConnections.clear(); - if (this.idleTimeout) { - clearInterval(this.idleTimeout); - this.idleTimeout = null; - } - } - /** - * Get connection pool status - */ - getPoolStatus() { - const total = this.connections.size; - const active = Array.from(this.connections.values()) - .filter(conn => conn.state === CONNECTION_STATES.BUSY).length; - const idle = total - active; - const pending = this.pendingConnections.size; - return { total, active, idle, pending }; - } - /** - * Update connection activity timestamp - */ - updateActivity(connection) { - connection.lastActivity = new Date(); - } - async establishSocket() { - return new Promise((resolve, reject) => { - const timeout = this.options.connectionTimeout || DEFAULTS.CONNECTION_TIMEOUT; - let socket; - if (this.options.secure) { - // Direct TLS connection - socket = tls.connect({ - host: this.options.host, - port: this.options.port, - ...this.options.tls - }); - } - else { - // Plain connection - socket = new net.Socket(); - socket.connect(this.options.port, this.options.host); - } - const timeoutHandler = setTimeout(() => { - socket.destroy(); - reject(new Error(`Connection timeout after ${timeout}ms`)); - }, timeout); - // For TLS connections, we need to wait for 'secureConnect' instead of 'connect' - const successEvent = this.options.secure ? 'secureConnect' : 'connect'; - socket.once(successEvent, () => { - clearTimeout(timeoutHandler); - resolve(socket); - }); - socket.once('error', (error) => { - clearTimeout(timeoutHandler); - reject(error); - }); - }); - } - setupSocketHandlers(socket, connectionId) { - const socketTimeout = this.options.socketTimeout || DEFAULTS.SOCKET_TIMEOUT; - socket.setTimeout(socketTimeout); - socket.on('timeout', () => { - logDebug('Socket timeout', this.options, { connectionId }); - socket.destroy(); - }); - socket.on('error', (error) => { - logConnection('error', this.options, { connectionId, error }); - this.connections.delete(connectionId); - }); - socket.on('close', () => { - this.connections.delete(connectionId); - logDebug('Socket closed', this.options, { connectionId }); - }); - } - findIdleConnection() { - for (const connection of this.connections.values()) { - if (connection.state === CONNECTION_STATES.READY) { - return connection; - } - } - return null; - } - shouldReuseConnection(connection) { - const maxMessages = this.options.maxMessages || DEFAULTS.MAX_MESSAGES; - const maxAge = 300000; // 5 minutes - const age = Date.now() - connection.createdAt.getTime(); - return connection.messageCount < maxMessages && - age < maxAge && - !connection.socket.destroyed; - } - getActiveConnectionCount() { - return this.connections.size + this.pendingConnections.size; - } - getConnectionId(connection) { - for (const [id, conn] of this.connections.entries()) { - if (conn === connection) { - return id; - } - } - return null; - } - setupIdleCleanup() { - if (!this.options.pool) { - return; - } - const cleanupInterval = DEFAULTS.POOL_IDLE_TIMEOUT; - this.idleTimeout = setInterval(() => { - const now = Date.now(); - const connectionsToClose = []; - for (const connection of this.connections.values()) { - const idleTime = now - connection.lastActivity.getTime(); - if (connection.state === CONNECTION_STATES.READY && idleTime > cleanupInterval) { - connectionsToClose.push(connection); - } - } - for (const connection of connectionsToClose) { - logDebug('Closing idle connection', this.options); - this.closeConnection(connection); - } - }, cleanupInterval); - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29ubmVjdGlvbi1tYW5hZ2VyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vdHMvbWFpbC9kZWxpdmVyeS9zbXRwY2xpZW50L2Nvbm5lY3Rpb24tbWFuYWdlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxPQUFPLEtBQUssR0FBRyxNQUFNLFVBQVUsQ0FBQztBQUNoQyxPQUFPLEtBQUssR0FBRyxNQUFNLFVBQVUsQ0FBQztBQUNoQyxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sYUFBYSxDQUFDO0FBQzNDLE9BQU8sRUFBRSxRQUFRLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQU83RCxPQUFPLEVBQUUsYUFBYSxFQUFFLFFBQVEsRUFBRSxNQUFNLG9CQUFvQixDQUFDO0FBQzdELE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLG9CQUFvQixDQUFDO0FBRTFELE1BQU0sT0FBTyxpQkFBa0IsU0FBUSxZQUFZO0lBQ3pDLE9BQU8sQ0FBcUI7SUFDNUIsV0FBVyxHQUFpQyxJQUFJLEdBQUcsRUFBRSxDQUFDO0lBQ3RELGtCQUFrQixHQUFnQixJQUFJLEdBQUcsRUFBRSxDQUFDO0lBQzVDLFdBQVcsR0FBMEIsSUFBSSxDQUFDO0lBRWxELFlBQVksT0FBMkI7UUFDckMsS0FBSyxFQUFFLENBQUM7UUFDUixJQUFJLENBQUMsT0FBTyxHQUFHLE9BQU8sQ0FBQztRQUN2QixJQUFJLENBQUMsZ0JBQWdCLEVBQUUsQ0FBQztJQUMxQixDQUFDO0lBRUQ7O09BRUc7SUFDSSxLQUFLLENBQUMsYUFBYTtRQUN4Qix3REFBd0Q7UUFDeEQsSUFBSSxJQUFJLENBQUMsT0FBTyxDQUFDLElBQUksRUFBRSxDQUFDO1lBQ3RCLE1BQU0sY0FBYyxHQUFHLElBQUksQ0FBQyxrQkFBa0IsRUFBRSxDQUFDO1lBQ2pELElBQUksY0FBYyxFQUFFLENBQUM7Z0JBQ25CLE1BQU0sWUFBWSxHQUFHLElBQUksQ0FB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 \ No newline at end of file diff --git a/dist_ts/mail/delivery/smtpclient/constants.d.ts b/dist_ts/mail/delivery/smtpclient/constants.d.ts deleted file mode 100644 index a55bc19..0000000 --- a/dist_ts/mail/delivery/smtpclient/constants.d.ts +++ /dev/null @@ -1,129 +0,0 @@ -/** - * SMTP Client Constants and Error Codes - * All constants, error codes, and enums for SMTP client operations - */ -/** - * SMTP response codes - */ -export declare const SMTP_CODES: { - readonly SERVICE_READY: 220; - readonly SERVICE_CLOSING: 221; - readonly AUTHENTICATION_SUCCESSFUL: 235; - readonly REQUESTED_ACTION_OK: 250; - readonly USER_NOT_LOCAL: 251; - readonly CANNOT_VERIFY_USER: 252; - readonly START_MAIL_INPUT: 354; - readonly SERVICE_NOT_AVAILABLE: 421; - readonly MAILBOX_BUSY: 450; - readonly LOCAL_ERROR: 451; - readonly INSUFFICIENT_STORAGE: 452; - readonly UNABLE_TO_ACCOMMODATE: 455; - readonly SYNTAX_ERROR: 500; - readonly SYNTAX_ERROR_PARAMETERS: 501; - readonly COMMAND_NOT_IMPLEMENTED: 502; - readonly BAD_SEQUENCE: 503; - readonly PARAMETER_NOT_IMPLEMENTED: 504; - readonly MAILBOX_UNAVAILABLE: 550; - readonly USER_NOT_LOCAL_TRY_FORWARD: 551; - readonly EXCEEDED_STORAGE: 552; - readonly MAILBOX_NAME_NOT_ALLOWED: 553; - readonly TRANSACTION_FAILED: 554; -}; -/** - * SMTP command names - */ -export declare const SMTP_COMMANDS: { - readonly HELO: "HELO"; - readonly EHLO: "EHLO"; - readonly MAIL_FROM: "MAIL FROM"; - readonly RCPT_TO: "RCPT TO"; - readonly DATA: "DATA"; - readonly RSET: "RSET"; - readonly NOOP: "NOOP"; - readonly QUIT: "QUIT"; - readonly STARTTLS: "STARTTLS"; - readonly AUTH: "AUTH"; -}; -/** - * Authentication methods - */ -export declare const AUTH_METHODS: { - readonly PLAIN: "PLAIN"; - readonly LOGIN: "LOGIN"; - readonly OAUTH2: "XOAUTH2"; - readonly CRAM_MD5: "CRAM-MD5"; -}; -/** - * Common SMTP extensions - */ -export declare const SMTP_EXTENSIONS: { - readonly PIPELINING: "PIPELINING"; - readonly SIZE: "SIZE"; - readonly STARTTLS: "STARTTLS"; - readonly AUTH: "AUTH"; - readonly EIGHT_BIT_MIME: "8BITMIME"; - readonly CHUNKING: "CHUNKING"; - readonly ENHANCED_STATUS_CODES: "ENHANCEDSTATUSCODES"; - readonly DSN: "DSN"; -}; -/** - * Default configuration values - */ -export declare const DEFAULTS: { - readonly CONNECTION_TIMEOUT: 60000; - readonly SOCKET_TIMEOUT: 300000; - readonly COMMAND_TIMEOUT: 30000; - readonly MAX_CONNECTIONS: 5; - readonly MAX_MESSAGES: 100; - readonly PORT_SMTP: 25; - readonly PORT_SUBMISSION: 587; - readonly PORT_SMTPS: 465; - readonly RETRY_ATTEMPTS: 3; - readonly RETRY_DELAY: 1000; - readonly POOL_IDLE_TIMEOUT: 30000; -}; -/** - * Error types for classification - */ -export declare enum SmtpErrorType { - CONNECTION_ERROR = "CONNECTION_ERROR", - AUTHENTICATION_ERROR = "AUTHENTICATION_ERROR", - PROTOCOL_ERROR = "PROTOCOL_ERROR", - TIMEOUT_ERROR = "TIMEOUT_ERROR", - TLS_ERROR = "TLS_ERROR", - SYNTAX_ERROR = "SYNTAX_ERROR", - MAILBOX_ERROR = "MAILBOX_ERROR", - QUOTA_ERROR = "QUOTA_ERROR", - UNKNOWN_ERROR = "UNKNOWN_ERROR" -} -/** - * Regular expressions for parsing - */ -export declare const REGEX_PATTERNS: { - readonly EMAIL_ADDRESS: RegExp; - readonly RESPONSE_CODE: RegExp; - readonly ENHANCED_STATUS: RegExp; - readonly AUTH_CAPABILITIES: RegExp; - readonly SIZE_EXTENSION: RegExp; -}; -/** - * Line endings and separators - */ -export declare const LINE_ENDINGS: { - readonly CRLF: "\r\n"; - readonly LF: "\n"; - readonly CR: "\r"; -}; -/** - * Connection states for internal use - */ -export declare const CONNECTION_STATES: { - readonly DISCONNECTED: "disconnected"; - readonly CONNECTING: "connecting"; - readonly CONNECTED: "connected"; - readonly AUTHENTICATED: "authenticated"; - readonly READY: "ready"; - readonly BUSY: "busy"; - readonly CLOSING: "closing"; - readonly ERROR: "error"; -}; diff --git a/dist_ts/mail/delivery/smtpclient/constants.js b/dist_ts/mail/delivery/smtpclient/constants.js deleted file mode 100644 index 9caab70..0000000 --- a/dist_ts/mail/delivery/smtpclient/constants.js +++ /dev/null @@ -1,135 +0,0 @@ -/** - * SMTP Client Constants and Error Codes - * All constants, error codes, and enums for SMTP client operations - */ -/** - * SMTP response codes - */ -export const SMTP_CODES = { - // Positive completion replies - SERVICE_READY: 220, - SERVICE_CLOSING: 221, - AUTHENTICATION_SUCCESSFUL: 235, - REQUESTED_ACTION_OK: 250, - USER_NOT_LOCAL: 251, - CANNOT_VERIFY_USER: 252, - // Positive intermediate replies - START_MAIL_INPUT: 354, - // Transient negative completion replies - SERVICE_NOT_AVAILABLE: 421, - MAILBOX_BUSY: 450, - LOCAL_ERROR: 451, - INSUFFICIENT_STORAGE: 452, - UNABLE_TO_ACCOMMODATE: 455, - // Permanent negative completion replies - SYNTAX_ERROR: 500, - SYNTAX_ERROR_PARAMETERS: 501, - COMMAND_NOT_IMPLEMENTED: 502, - BAD_SEQUENCE: 503, - PARAMETER_NOT_IMPLEMENTED: 504, - MAILBOX_UNAVAILABLE: 550, - USER_NOT_LOCAL_TRY_FORWARD: 551, - EXCEEDED_STORAGE: 552, - MAILBOX_NAME_NOT_ALLOWED: 553, - TRANSACTION_FAILED: 554 -}; -/** - * SMTP command names - */ -export const SMTP_COMMANDS = { - HELO: 'HELO', - EHLO: 'EHLO', - MAIL_FROM: 'MAIL FROM', - RCPT_TO: 'RCPT TO', - DATA: 'DATA', - RSET: 'RSET', - NOOP: 'NOOP', - QUIT: 'QUIT', - STARTTLS: 'STARTTLS', - AUTH: 'AUTH' -}; -/** - * Authentication methods - */ -export const AUTH_METHODS = { - PLAIN: 'PLAIN', - LOGIN: 'LOGIN', - OAUTH2: 'XOAUTH2', - CRAM_MD5: 'CRAM-MD5' -}; -/** - * Common SMTP extensions - */ -export const SMTP_EXTENSIONS = { - PIPELINING: 'PIPELINING', - SIZE: 'SIZE', - STARTTLS: 'STARTTLS', - AUTH: 'AUTH', - EIGHT_BIT_MIME: '8BITMIME', - CHUNKING: 'CHUNKING', - ENHANCED_STATUS_CODES: 'ENHANCEDSTATUSCODES', - DSN: 'DSN' -}; -/** - * Default configuration values - */ -export const DEFAULTS = { - CONNECTION_TIMEOUT: 60000, // 60 seconds - SOCKET_TIMEOUT: 300000, // 5 minutes - COMMAND_TIMEOUT: 30000, // 30 seconds - MAX_CONNECTIONS: 5, - MAX_MESSAGES: 100, - PORT_SMTP: 25, - PORT_SUBMISSION: 587, - PORT_SMTPS: 465, - RETRY_ATTEMPTS: 3, - RETRY_DELAY: 1000, - POOL_IDLE_TIMEOUT: 30000 // 30 seconds -}; -/** - * Error types for classification - */ -export var SmtpErrorType; -(function (SmtpErrorType) { - SmtpErrorType["CONNECTION_ERROR"] = "CONNECTION_ERROR"; - SmtpErrorType["AUTHENTICATION_ERROR"] = "AUTHENTICATION_ERROR"; - SmtpErrorType["PROTOCOL_ERROR"] = "PROTOCOL_ERROR"; - SmtpErrorType["TIMEOUT_ERROR"] = "TIMEOUT_ERROR"; - SmtpErrorType["TLS_ERROR"] = "TLS_ERROR"; - SmtpErrorType["SYNTAX_ERROR"] = "SYNTAX_ERROR"; - SmtpErrorType["MAILBOX_ERROR"] = "MAILBOX_ERROR"; - SmtpErrorType["QUOTA_ERROR"] = "QUOTA_ERROR"; - SmtpErrorType["UNKNOWN_ERROR"] = "UNKNOWN_ERROR"; -})(SmtpErrorType || (SmtpErrorType = {})); -/** - * Regular expressions for parsing - */ -export const REGEX_PATTERNS = { - EMAIL_ADDRESS: /^[^\s@]+@[^\s@]+\.[^\s@]+$/, - RESPONSE_CODE: /^(\d{3})([ -])(.*)/, - ENHANCED_STATUS: /^(\d\.\d\.\d)\s/, - AUTH_CAPABILITIES: /AUTH\s+(.+)/i, - SIZE_EXTENSION: /SIZE\s+(\d+)/i -}; -/** - * Line endings and separators - */ -export const LINE_ENDINGS = { - CRLF: '\r\n', - LF: '\n', - CR: '\r' -}; -/** - * Connection states for internal use - */ -export const CONNECTION_STATES = { - DISCONNECTED: 'disconnected', - CONNECTING: 'connecting', - CONNECTED: 'connected', - AUTHENTICATED: 'authenticated', - READY: 'ready', - BUSY: 'busy', - CLOSING: 'closing', - ERROR: 'error' -}; -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vdHMvbWFpbC9kZWxpdmVyeS9zbXRwY2xpZW50L2NvbnN0YW50cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSDs7R0FFRztBQUNILE1BQU0sQ0FBQyxNQUFNLFVBQVUsR0FBRztJQUN4Qiw4QkFBOEI7SUFDOUIsYUFBYSxFQUFFLEdBQUc7SUFDbEIsZUFBZSxFQUFFLEdBQUc7SUFDcEIseUJBQXlCLEVBQUUsR0FBRztJQUM5QixtQkFBbUIsRUFBRSxHQUFHO0lBQ3hCLGNBQWMsRUFBRSxHQUFHO0lBQ25CLGtCQUFrQixFQUFFLEdBQUc7SUFFdkIsZ0NBQWdDO0lBQ2hDLGdCQUFnQixFQUFFLEdBQUc7SUFFckIsd0NBQXdDO0lBQ3hDLHFCQUFxQixFQUFFLEdBQUc7SUFDMUIsWUFBWSxFQUFFLEdBQUc7SUFDakIsV0FBVyxFQUFFLEdBQUc7SUFDaEIsb0JBQW9CLEVBQUUsR0FBRztJQUN6QixxQkFBcUIsRUFBRSxHQUFHO0lBRTFCLHdDQUF3QztJQUN4QyxZQUFZLEVBQUUsR0FBRztJQUNqQix1QkFBdUIsRUFBRSxHQUFHO0lBQzVCLHVCQUF1QixFQUFFLEdBQUc7SUFDNUIsWUFBWSxFQUFFLEdBQUc7SUFDakIseUJBQXlCLEVBQUUsR0FBRztJQUM5QixtQkFBbUIsRUFBRSxHQUFHO0lBQ3hCLDBCQUEwQixFQUFFLEdBQUc7SUFDL0IsZ0JBQWdCLEVBQUUsR0FBRztJQUNyQix3QkFBd0IsRUFBRSxHQUFHO0lBQzdCLGtCQUFrQixFQUFFLEdBQUc7Q0FDZixDQUFDO0FBRVg7O0dBRUc7QUFDSCxNQUFNLENBQUMsTUFBTSxhQUFhLEdBQUc7SUFDM0IsSUFBSSxFQUFFLE1BQU07SUFDWixJQUFJLEVBQUUsTUFBTTtJQUNaLFNBQVMsRUFBRSxXQUFXO0lBQ3RCLE9BQU8sRUFBRSxTQUFTO0lBQ2xCLElBQUksRUFBRSxNQUFNO0lBQ1osSUFB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 \ No newline at end of file diff --git a/dist_ts/mail/delivery/smtpclient/create-client.d.ts b/dist_ts/mail/delivery/smtpclient/create-client.d.ts deleted file mode 100644 index 3f6f87d..0000000 --- a/dist_ts/mail/delivery/smtpclient/create-client.d.ts +++ /dev/null @@ -1,22 +0,0 @@ -/** - * SMTP Client Factory - * Factory function for client creation and dependency injection - */ -import { SmtpClient } from './smtp-client.js'; -import type { ISmtpClientOptions } from './interfaces.js'; -/** - * Create a complete SMTP client with all components - */ -export declare function createSmtpClient(options: ISmtpClientOptions): SmtpClient; -/** - * Create SMTP client with connection pooling enabled - */ -export declare function createPooledSmtpClient(options: ISmtpClientOptions): SmtpClient; -/** - * Create SMTP client for high-volume sending - */ -export declare function createBulkSmtpClient(options: ISmtpClientOptions): SmtpClient; -/** - * Create SMTP client for transactional emails - */ -export declare function createTransactionalSmtpClient(options: ISmtpClientOptions): SmtpClient; diff --git a/dist_ts/mail/delivery/smtpclient/create-client.js b/dist_ts/mail/delivery/smtpclient/create-client.js deleted file mode 100644 index 46b60f3..0000000 --- a/dist_ts/mail/delivery/smtpclient/create-client.js +++ /dev/null @@ -1,86 +0,0 @@ -/** - * SMTP Client Factory - * Factory function for client creation and dependency injection - */ -import { SmtpClient } from './smtp-client.js'; -import { ConnectionManager } from './connection-manager.js'; -import { CommandHandler } from './command-handler.js'; -import { AuthHandler } from './auth-handler.js'; -import { TlsHandler } from './tls-handler.js'; -import { SmtpErrorHandler } from './error-handler.js'; -import { validateClientOptions } from './utils/validation.js'; -import { DEFAULTS } from './constants.js'; -/** - * Create a complete SMTP client with all components - */ -export function createSmtpClient(options) { - // Validate options - const errors = validateClientOptions(options); - if (errors.length > 0) { - throw new Error(`Invalid client options: ${errors.join(', ')}`); - } - // Apply defaults - const clientOptions = { - connectionTimeout: DEFAULTS.CONNECTION_TIMEOUT, - socketTimeout: DEFAULTS.SOCKET_TIMEOUT, - maxConnections: DEFAULTS.MAX_CONNECTIONS, - maxMessages: DEFAULTS.MAX_MESSAGES, - pool: false, - secure: false, - debug: false, - ...options - }; - // Create handlers - const errorHandler = new SmtpErrorHandler(clientOptions); - const connectionManager = new ConnectionManager(clientOptions); - const commandHandler = new CommandHandler(clientOptions); - const authHandler = new AuthHandler(clientOptions, commandHandler); - const tlsHandler = new TlsHandler(clientOptions, commandHandler); - // Create and return SMTP client - return new SmtpClient({ - options: clientOptions, - connectionManager, - commandHandler, - authHandler, - tlsHandler, - errorHandler - }); -} -/** - * Create SMTP client with connection pooling enabled - */ -export function createPooledSmtpClient(options) { - return createSmtpClient({ - ...options, - pool: true, - maxConnections: options.maxConnections || DEFAULTS.MAX_CONNECTIONS, - maxMessages: options.maxMessages || DEFAULTS.MAX_MESSAGES - }); -} -/** - * Create SMTP client for high-volume sending - */ -export function createBulkSmtpClient(options) { - return createSmtpClient({ - ...options, - pool: true, - maxConnections: Math.max(options.maxConnections || 10, 10), - maxMessages: Math.max(options.maxMessages || 1000, 1000), - connectionTimeout: options.connectionTimeout || 30000, - socketTimeout: options.socketTimeout || 120000 - }); -} -/** - * Create SMTP client for transactional emails - */ -export function createTransactionalSmtpClient(options) { - return createSmtpClient({ - ...options, - pool: false, // Use fresh connections for transactional emails - maxConnections: 1, - maxMessages: 1, - connectionTimeout: options.connectionTimeout || 10000, - socketTimeout: options.socketTimeout || 30000 - }); -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY3JlYXRlLWNsaWVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3RzL21haWwvZGVsaXZlcnkvc210cGNsaWVudC9jcmVhdGUtY2xpZW50LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7R0FHRztBQUVILE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxrQkFBa0IsQ0FBQztBQUM5QyxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUM1RCxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sc0JBQXNCLENBQUM7QUFDdEQsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLG1CQUFtQixDQUFDO0FBQ2hELE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxrQkFBa0IsQ0FBQztBQUM5QyxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQUV0RCxPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQUM5RCxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFFMUM7O0dBRUc7QUFDSCxNQUFNLFVBQVUsZ0JBQWdCLENBQUMsT0FBMkI7SUFDMUQsbUJBQW1CO0lBQ25CLE1BQU0sTUFBTSxHQUFHLHFCQUFxQixDQUFDLE9BQU8sQ0FBQyxDQUFDO0lBQzlDLElBQUksTUFBTSxDQUFDLE1BQU0sR0FBRyxDQUFDLEVBQUUsQ0FBQztRQUN0QixNQUFNLElBQUksS0FBSyxDQUFDLDJCQUEyQixNQUFNLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsQ0FBQztJQUNsRSxDQUFDO0lBRUQsaUJBQWlCO0lBQ2pCLE1BQU0sYUFBYSxHQUF1QjtRQUN4QyxpQkFBaUIsRUFBRSxRQUFRLENBQUMsa0JBQWtCO1FBQzlDLGFBQWEsRUFBRSxRQUFRLENBQUMsY0FBYztRQUN0QyxjQUFjLEVBQUUsUUFBUSxDQUFDLGVBQWU7UUFDeEMsV0FBVyxFQUFFLFFBQVEsQ0FBQyxZQUFZO1FBQ2xDLElBQUksRUFBRSxLQUFLO1FBQ1gsTUFBTSxFQUFFLEtBQUs7UUFDYixLQUFLLEVBQUUsS0FBSztRQUNaLEdBQUcsT0FBTztLQUNYLENBQUM7SUFFRixrQkFBa0I7SUFDbEIsTUFBTSxZQUFZLEdBQUcsSUFBSSxnQkFBZ0IsQ0FBQyxhQUFhLENBQUMsQ0FBQztJQUN6RCxNQUFNLGlCQUFpQixHQUFHLElBQUksaUJBQWlCLENBQUMsYUFBYSxDQUFDLENBQUM7SUFDL0QsTUFBTSxjQUFjLEdBQUcsSUFBSSxjQUFjLENBQUMsYUFBYSxDQUFDLENBQUM7SUFDekQsTUFBTSxXQUFXLEdBQUcsSUFBSSxXQUFXLENBQUMsYUFBYSxFQUFFLGNBQWMsQ0FBQyxDQUFDO0lBQ25FLE1BQU0sVUFBVSxHQUFHLElBQUksVUFBVSxDQUFDLGFBQWEsRUFBRSxjQUFjLENBQUMsQ0FBQztJQUVqRSxnQ0FBZ0M7SUFDaEMsT0FBTyxJQUFJLFVBQVUsQ0FBQztRQUNwQixPQUFPLEVBQUUsYUFBYTtRQUN0QixpQkFBaUI7UUFDakIsY0FBYztRQUNkLFdBQVc7UUFDWCxVQUFVO1FBQ1YsWUFBWTtLQUNiLENBQUMsQ0FBQztBQUNMLENBQUM7QUFFRDs7R0FFRztBQUNILE1BQU0sVUFBVSxzQkFBc0IsQ0FBQyxPQUEyQjtJQUNoRSxPQUFPLGdCQUFnQixDQUFDO1FBQ3RCLEdBQUcsT0FBTztRQUNWLElBQUksRUFBRSxJQUFJO1FBQ1YsY0FBYyxFQUFFLE9BQU8sQ0FBQyxjQUFjLElBQUksUUFBUSxDQUFDLGVBQWU7UUFDbEUsV0FBVyxFQUFFLE9BQU8sQ0FBQyxXQUFXLElBQUksUUFBUSxDQUFDLFlBQVk7S0FDMUQsQ0FBQyxDQUFDO0FBQ0wsQ0FBQztBQUVEOztHQUVHO0FBQ0gsTUFBTSxVQUFVLG9CQUFvQixDQUFDLE9BQTJCO0lBQzlELE9BQU8sZ0JBQWdCLENBQUM7UUFDdEIsR0FBRyxPQUFPO1FBQ1YsSUFBSSxFQUFFLElBQUk7UUFDVixjQUFjLEVBQUUsSUFBSSxDQUFDLEdBQUcsQ0FBQyxPQUFPLENBQUMsY0FBYyxJQUFJLEVBQUUsRUFBRSxFQUFFLENBQUM7UUFDMUQsV0FBVyxFQUFFLElBQUksQ0FBQyxHQUFHLENBQUMsT0FBTyxDQUFDLFdBQVcsSUFBSSxJQUFJLEVBQUUsSUFBSSxDQUFDO1FBQ3hELGlCQUFpQixFQUFFLE9BQU8sQ0FBQyxpQkFBaUIsSUFBSSxLQUFLO1FBQ3JELGFBQWEsRUFBRSxPQUFPLENBQUMsYUFBYSxJQUFJLE1BQU07S0FDL0MsQ0FBQyxDQUFDO0FBQ0wsQ0FBQztBQUVEOztHQUVHO0FBQ0gsTUFBTSxVQUFVLDZCQUE2QixDQUFDLE9BQTJCO0lBQ3ZFLE9BQU8sZ0JBQWdCLENBQUM7UUFDdEIsR0FBRyxPQUFPO1FBQ1YsSUFBSSxFQUFFLEtBQUssRUFBRSxpREFBaUQ7UUFDOUQsY0FBYyxFQUFFLENBQUM7UUFDakIsV0FBVyxFQUFFLENBQUM7UUFDZCxpQkFBaUIsRUFBRSxPQUFPLENBQUMsaUJBQWlCLElBQUksS0FBSztRQUNyRCxhQUFhLEVBQUUsT0FBTyxDQUFDLGFBQWEsSUFBSSxLQUFLO0tBQzlDLENBQUMsQ0FBQztBQUNMLENBQUMifQ== \ No newline at end of file diff --git a/dist_ts/mail/delivery/smtpclient/error-handler.d.ts b/dist_ts/mail/delivery/smtpclient/error-handler.d.ts deleted file mode 100644 index 91ad403..0000000 --- a/dist_ts/mail/delivery/smtpclient/error-handler.d.ts +++ /dev/null @@ -1,28 +0,0 @@ -/** - * SMTP Client Error Handler - * Error classification and recovery strategies - */ -import { SmtpErrorType } from './constants.js'; -import type { ISmtpResponse, ISmtpErrorContext, ISmtpClientOptions } from './interfaces.js'; -export declare class SmtpErrorHandler { - private options; - constructor(options: ISmtpClientOptions); - /** - * Classify error type based on response or error - */ - classifyError(error: Error | ISmtpResponse, context?: ISmtpErrorContext): SmtpErrorType; - /** - * Determine if error is retryable - */ - isRetryable(errorType: SmtpErrorType, response?: ISmtpResponse): boolean; - /** - * Get retry delay for error type - */ - getRetryDelay(attempt: number, errorType: SmtpErrorType): number; - /** - * Create enhanced error with context - */ - createError(message: string, errorType: SmtpErrorType, context?: ISmtpErrorContext, originalError?: Error): Error; - private classifyErrorByMessage; - private classifyErrorByCode; -} diff --git a/dist_ts/mail/delivery/smtpclient/error-handler.js b/dist_ts/mail/delivery/smtpclient/error-handler.js deleted file mode 100644 index e961fcf..0000000 --- a/dist_ts/mail/delivery/smtpclient/error-handler.js +++ /dev/null @@ -1,111 +0,0 @@ -/** - * SMTP Client Error Handler - * Error classification and recovery strategies - */ -import { SmtpErrorType } from './constants.js'; -import { logDebug } from './utils/logging.js'; -export class SmtpErrorHandler { - options; - constructor(options) { - this.options = options; - } - /** - * Classify error type based on response or error - */ - classifyError(error, context) { - logDebug('Classifying error', this.options, { errorMessage: error instanceof Error ? error.message : String(error), context }); - // Handle Error objects - if (error instanceof Error) { - return this.classifyErrorByMessage(error); - } - // Handle SMTP response codes - if (typeof error === 'object' && 'code' in error) { - return this.classifyErrorByCode(error.code); - } - return SmtpErrorType.UNKNOWN_ERROR; - } - /** - * Determine if error is retryable - */ - isRetryable(errorType, response) { - switch (errorType) { - case SmtpErrorType.CONNECTION_ERROR: - case SmtpErrorType.TIMEOUT_ERROR: - return true; - case SmtpErrorType.PROTOCOL_ERROR: - // Only retry on temporary failures (4xx codes) - return response ? response.code >= 400 && response.code < 500 : false; - case SmtpErrorType.AUTHENTICATION_ERROR: - case SmtpErrorType.TLS_ERROR: - case SmtpErrorType.SYNTAX_ERROR: - case SmtpErrorType.MAILBOX_ERROR: - case SmtpErrorType.QUOTA_ERROR: - return false; - default: - return false; - } - } - /** - * Get retry delay for error type - */ - getRetryDelay(attempt, errorType) { - const baseDelay = 1000; // 1 second - const maxDelay = 30000; // 30 seconds - // Exponential backoff with jitter - const delay = Math.min(baseDelay * Math.pow(2, attempt - 1), maxDelay); - const jitter = Math.random() * 0.1 * delay; // 10% jitter - return Math.floor(delay + jitter); - } - /** - * Create enhanced error with context - */ - createError(message, errorType, context, originalError) { - const error = new Error(message); - error.type = errorType; - error.context = context; - error.originalError = originalError; - return error; - } - classifyErrorByMessage(error) { - const message = error.message.toLowerCase(); - if (message.includes('timeout') || message.includes('etimedout')) { - return SmtpErrorType.TIMEOUT_ERROR; - } - if (message.includes('connect') || message.includes('econnrefused') || - message.includes('enotfound') || message.includes('enetunreach')) { - return SmtpErrorType.CONNECTION_ERROR; - } - if (message.includes('tls') || message.includes('ssl') || - message.includes('certificate') || message.includes('handshake')) { - return SmtpErrorType.TLS_ERROR; - } - if (message.includes('auth')) { - return SmtpErrorType.AUTHENTICATION_ERROR; - } - return SmtpErrorType.UNKNOWN_ERROR; - } - classifyErrorByCode(code) { - if (code >= 500) { - // Permanent failures - if (code === 550 || code === 551 || code === 553) { - return SmtpErrorType.MAILBOX_ERROR; - } - if (code === 552) { - return SmtpErrorType.QUOTA_ERROR; - } - if (code === 500 || code === 501 || code === 502 || code === 504) { - return SmtpErrorType.SYNTAX_ERROR; - } - return SmtpErrorType.PROTOCOL_ERROR; - } - if (code >= 400) { - // Temporary failures - if (code === 450 || code === 451 || code === 452) { - return SmtpErrorType.QUOTA_ERROR; - } - return SmtpErrorType.PROTOCOL_ERROR; - } - return SmtpErrorType.UNKNOWN_ERROR; - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXJyb3ItaGFuZGxlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3RzL21haWwvZGVsaXZlcnkvc210cGNsaWVudC9lcnJvci1oYW5kbGVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7R0FHRztBQUVILE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUUvQyxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sb0JBQW9CLENBQUM7QUFFOUMsTUFBTSxPQUFPLGdCQUFnQjtJQUNuQixPQUFPLENBQXFCO0lBRXBDLFlBQVksT0FBMkI7UUFDckMsSUFBSSxDQUFDLE9BQU8sR0FBRyxPQUFPLENBQUM7SUFDekIsQ0FBQztJQUVEOztPQUVHO0lBQ0ksYUFBYSxDQUFDLEtBQTRCLEVBQUUsT0FBMkI7UUFDNUUsUUFBUSxDQUFDLG1CQUFtQixFQUFFLElBQUksQ0FBQyxPQUFPLEVBQUUsRUFBRSxZQUFZLEVBQUUsS0FBSyxZQUFZLEtBQUssQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxFQUFFLE9BQU8sRUFBRSxDQUFDLENBQUM7UUFFL0gsdUJBQXVCO1FBQ3ZCLElBQUksS0FBSyxZQUFZLEtBQUssRUFBRSxDQUFDO1lBQzNCLE9BQU8sSUFBSSxDQUFDLHNCQUFzQixDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQzVDLENBQUM7UUFFRCw2QkFBNkI7UUFDN0IsSUFBSSxPQUFPLEtBQUssS0FBSyxRQUFRLElBQUksTUFBTSxJQUFJLEtBQUssRUFBRSxDQUFDO1lBQ2pELE9BQU8sSUFBSSxDQUFDLG1CQUFtQixDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUM5QyxDQUFDO1FBRUQsT0FBTyxhQUFhLENBQUMsYUFBYSxDQUFDO0lBQ3JDLENBQUM7SUFFRDs7T0FFRztJQUNJLFdBQVcsQ0FBQyxTQUF3QixFQUFFLFFBQXdCO1FBQ25F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 \ No newline at end of file diff --git a/dist_ts/mail/delivery/smtpclient/index.d.ts b/dist_ts/mail/delivery/smtpclient/index.d.ts deleted file mode 100644 index e582807..0000000 --- a/dist_ts/mail/delivery/smtpclient/index.d.ts +++ /dev/null @@ -1,16 +0,0 @@ -/** - * SMTP Client Module Exports - * Modular SMTP client implementation for robust email delivery - */ -export * from './smtp-client.js'; -export * from './create-client.js'; -export * from './connection-manager.js'; -export * from './command-handler.js'; -export * from './auth-handler.js'; -export * from './tls-handler.js'; -export * from './error-handler.js'; -export * from './interfaces.js'; -export * from './constants.js'; -export * from './utils/validation.js'; -export * from './utils/logging.js'; -export * from './utils/helpers.js'; diff --git a/dist_ts/mail/delivery/smtpclient/index.js b/dist_ts/mail/delivery/smtpclient/index.js deleted file mode 100644 index 9d61526..0000000 --- a/dist_ts/mail/delivery/smtpclient/index.js +++ /dev/null @@ -1,21 +0,0 @@ -/** - * SMTP Client Module Exports - * Modular SMTP client implementation for robust email delivery - */ -// Main client class and factory -export * from './smtp-client.js'; -export * from './create-client.js'; -// Core handlers -export * from './connection-manager.js'; -export * from './command-handler.js'; -export * from './auth-handler.js'; -export * from './tls-handler.js'; -export * from './error-handler.js'; -// Interfaces and types -export * from './interfaces.js'; -export * from './constants.js'; -// Utilities -export * from './utils/validation.js'; -export * from './utils/logging.js'; -export * from './utils/helpers.js'; -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi90cy9tYWlsL2RlbGl2ZXJ5L3NtdHBjbGllbnQvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7OztHQUdHO0FBRUgsZ0NBQWdDO0FBQ2hDLGNBQWMsa0JBQWtCLENBQUM7QUFDakMsY0FBYyxvQkFBb0IsQ0FBQztBQUVuQyxnQkFBZ0I7QUFDaEIsY0FBYyx5QkFBeUIsQ0FBQztBQUN4QyxjQUFjLHNCQUFzQixDQUFDO0FBQ3JDLGNBQWMsbUJBQW1CLENBQUM7QUFDbEMsY0FBYyxrQkFBa0IsQ0FBQztBQUNqQyxjQUFjLG9CQUFvQixDQUFDO0FBRW5DLHVCQUF1QjtBQUN2QixjQUFjLGlCQUFpQixDQUFDO0FBQ2hDLGNBQWMsZ0JBQWdCLENBQUM7QUFFL0IsWUFBWTtBQUNaLGNBQWMsdUJBQXVCLENBQUM7QUFDdEMsY0FBYyxvQkFBb0IsQ0FBQztBQUNuQyxjQUFjLG9CQUFvQixDQUFDIn0= \ No newline at end of file diff --git a/dist_ts/mail/delivery/smtpclient/interfaces.d.ts b/dist_ts/mail/delivery/smtpclient/interfaces.d.ts deleted file mode 100644 index a1fbf84..0000000 --- a/dist_ts/mail/delivery/smtpclient/interfaces.d.ts +++ /dev/null @@ -1,183 +0,0 @@ -/** - * SMTP Client Interfaces and Types - * All interface definitions for the modular SMTP client - */ -import type * as tls from 'node:tls'; -import type * as net from 'node:net'; -/** - * SMTP client connection options - */ -export interface ISmtpClientOptions { - /** Hostname of the SMTP server */ - host: string; - /** Port to connect to */ - port: number; - /** Whether to use TLS for the connection */ - secure?: boolean; - /** Connection timeout in milliseconds */ - connectionTimeout?: number; - /** Socket timeout in milliseconds */ - socketTimeout?: number; - /** Domain name for EHLO command */ - domain?: string; - /** Authentication options */ - auth?: ISmtpAuthOptions; - /** TLS options */ - tls?: tls.ConnectionOptions; - /** Maximum number of connections in pool */ - pool?: boolean; - maxConnections?: number; - maxMessages?: number; - /** Enable debug logging */ - debug?: boolean; - /** Proxy settings */ - proxy?: string; -} -/** - * Authentication options for SMTP - */ -export interface ISmtpAuthOptions { - /** Username */ - user?: string; - /** Password */ - pass?: string; - /** OAuth2 settings */ - oauth2?: IOAuth2Options; - /** Authentication method preference */ - method?: 'PLAIN' | 'LOGIN' | 'OAUTH2' | 'AUTO'; -} -/** - * OAuth2 authentication options - */ -export interface IOAuth2Options { - /** OAuth2 user identifier */ - user: string; - /** OAuth2 client ID */ - clientId: string; - /** OAuth2 client secret */ - clientSecret: string; - /** OAuth2 refresh token */ - refreshToken: string; - /** OAuth2 access token */ - accessToken?: string; - /** Token expiry time */ - expires?: number; -} -/** - * Result of an email send operation - */ -export interface ISmtpSendResult { - /** Whether the send was successful */ - success: boolean; - /** Message ID from server */ - messageId?: string; - /** List of accepted recipients */ - acceptedRecipients: string[]; - /** List of rejected recipients */ - rejectedRecipients: string[]; - /** Error information if failed */ - error?: Error; - /** Server response */ - response?: string; - /** Envelope information */ - envelope?: ISmtpEnvelope; -} -/** - * SMTP envelope information - */ -export interface ISmtpEnvelope { - /** Sender address */ - from: string; - /** Recipient addresses */ - to: string[]; -} -/** - * Connection pool status - */ -export interface IConnectionPoolStatus { - /** Total connections in pool */ - total: number; - /** Active connections */ - active: number; - /** Idle connections */ - idle: number; - /** Pending connection requests */ - pending: number; -} -/** - * SMTP command response - */ -export interface ISmtpResponse { - /** Response code */ - code: number; - /** Response message */ - message: string; - /** Enhanced status code */ - enhancedCode?: string; - /** Raw response */ - raw: string; -} -/** - * Connection state - */ -export declare enum ConnectionState { - DISCONNECTED = "disconnected", - CONNECTING = "connecting", - CONNECTED = "connected", - AUTHENTICATED = "authenticated", - READY = "ready", - BUSY = "busy", - CLOSING = "closing", - ERROR = "error" -} -/** - * SMTP capabilities - */ -export interface ISmtpCapabilities { - /** Supported extensions */ - extensions: Set; - /** Maximum message size */ - maxSize?: number; - /** Supported authentication methods */ - authMethods: Set; - /** Support for pipelining */ - pipelining: boolean; - /** Support for STARTTLS */ - starttls: boolean; - /** Support for 8BITMIME */ - eightBitMime: boolean; -} -/** - * Internal connection interface - */ -export interface ISmtpConnection { - /** Socket connection */ - socket: net.Socket | tls.TLSSocket; - /** Connection state */ - state: ConnectionState; - /** Server capabilities */ - capabilities?: ISmtpCapabilities; - /** Connection options */ - options: ISmtpClientOptions; - /** Whether connection is secure */ - secure: boolean; - /** Connection creation time */ - createdAt: Date; - /** Last activity time */ - lastActivity: Date; - /** Number of messages sent */ - messageCount: number; -} -/** - * Error context for detailed error reporting - */ -export interface ISmtpErrorContext { - /** Command that caused the error */ - command?: string; - /** Server response */ - response?: ISmtpResponse; - /** Connection state */ - connectionState?: ConnectionState; - /** Additional context data */ - data?: Record; -} diff --git a/dist_ts/mail/delivery/smtpclient/interfaces.js b/dist_ts/mail/delivery/smtpclient/interfaces.js deleted file mode 100644 index bf08dca..0000000 --- a/dist_ts/mail/delivery/smtpclient/interfaces.js +++ /dev/null @@ -1,19 +0,0 @@ -/** - * SMTP Client Interfaces and Types - * All interface definitions for the modular SMTP client - */ -/** - * Connection state - */ -export var ConnectionState; -(function (ConnectionState) { - ConnectionState["DISCONNECTED"] = "disconnected"; - ConnectionState["CONNECTING"] = "connecting"; - ConnectionState["CONNECTED"] = "connected"; - ConnectionState["AUTHENTICATED"] = "authenticated"; - ConnectionState["READY"] = "ready"; - ConnectionState["BUSY"] = "busy"; - ConnectionState["CLOSING"] = "closing"; - ConnectionState["ERROR"] = "error"; -})(ConnectionState || (ConnectionState = {})); -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW50ZXJmYWNlcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3RzL21haWwvZGVsaXZlcnkvc210cGNsaWVudC9pbnRlcmZhY2VzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7R0FHRztBQTZKSDs7R0FFRztBQUNILE1BQU0sQ0FBTixJQUFZLGVBU1g7QUFURCxXQUFZLGVBQWU7SUFDekIsZ0RBQTZCLENBQUE7SUFDN0IsNENBQXlCLENBQUE7SUFDekIsMENBQXVCLENBQUE7SUFDdkIsa0RBQStCLENBQUE7SUFDL0Isa0NBQWUsQ0FBQTtJQUNmLGdDQUFhLENBQUE7SUFDYixzQ0FBbUIsQ0FBQTtJQUNuQixrQ0FBZSxDQUFBO0FBQ2pCLENBQUMsRUFUVyxlQUFlLEtBQWYsZUFBZSxRQVMxQiJ9 \ No newline at end of file diff --git a/dist_ts/mail/delivery/smtpclient/smtp-client.d.ts b/dist_ts/mail/delivery/smtpclient/smtp-client.d.ts deleted file mode 100644 index 67486d5..0000000 --- a/dist_ts/mail/delivery/smtpclient/smtp-client.d.ts +++ /dev/null @@ -1,58 +0,0 @@ -/** - * SMTP Client Core Implementation - * Main client class with delegation to handlers - */ -import { EventEmitter } from 'node:events'; -import type { Email } from '../../core/classes.email.js'; -import type { ISmtpClientOptions, ISmtpSendResult, IConnectionPoolStatus } from './interfaces.js'; -import type { ConnectionManager } from './connection-manager.js'; -import type { CommandHandler } from './command-handler.js'; -import type { AuthHandler } from './auth-handler.js'; -import type { TlsHandler } from './tls-handler.js'; -import type { SmtpErrorHandler } from './error-handler.js'; -interface ISmtpClientDependencies { - options: ISmtpClientOptions; - connectionManager: ConnectionManager; - commandHandler: CommandHandler; - authHandler: AuthHandler; - tlsHandler: TlsHandler; - errorHandler: SmtpErrorHandler; -} -export declare class SmtpClient extends EventEmitter { - private options; - private connectionManager; - private commandHandler; - private authHandler; - private tlsHandler; - private errorHandler; - private isShuttingDown; - constructor(dependencies: ISmtpClientDependencies); - /** - * Send an email - */ - sendMail(email: Email): Promise; - /** - * Test connection to SMTP server - */ - verify(): Promise; - /** - * Check if client is connected - */ - isConnected(): boolean; - /** - * Get connection pool status - */ - getPoolStatus(): IConnectionPoolStatus; - /** - * Update client options - */ - updateOptions(newOptions: Partial): void; - /** - * Close all connections and shutdown client - */ - close(): Promise; - private formatEmailData; - private extractMessageId; - private setupEventForwarding; -} -export {}; diff --git a/dist_ts/mail/delivery/smtpclient/smtp-client.js b/dist_ts/mail/delivery/smtpclient/smtp-client.js deleted file mode 100644 index c6cfaec..0000000 --- a/dist_ts/mail/delivery/smtpclient/smtp-client.js +++ /dev/null @@ -1,285 +0,0 @@ -/** - * SMTP Client Core Implementation - * Main client class with delegation to handlers - */ -import { EventEmitter } from 'node:events'; -import { CONNECTION_STATES, SmtpErrorType } from './constants.js'; -import { validateSender, validateRecipients } from './utils/validation.js'; -import { logEmailSend, logPerformance, logDebug } from './utils/logging.js'; -export class SmtpClient extends EventEmitter { - options; - connectionManager; - commandHandler; - authHandler; - tlsHandler; - errorHandler; - isShuttingDown = false; - constructor(dependencies) { - super(); - this.options = dependencies.options; - this.connectionManager = dependencies.connectionManager; - this.commandHandler = dependencies.commandHandler; - this.authHandler = dependencies.authHandler; - this.tlsHandler = dependencies.tlsHandler; - this.errorHandler = dependencies.errorHandler; - this.setupEventForwarding(); - } - /** - * Send an email - */ - async sendMail(email) { - const startTime = Date.now(); - // Extract clean email addresses without display names for SMTP operations - const fromAddress = email.getFromAddress(); - const recipients = email.getToAddresses(); - const ccRecipients = email.getCcAddresses(); - const bccRecipients = email.getBccAddresses(); - // Combine all recipients for SMTP operations - const allRecipients = [...recipients, ...ccRecipients, ...bccRecipients]; - // Validate email addresses - if (!validateSender(fromAddress)) { - throw new Error(`Invalid sender address: ${fromAddress}`); - } - const recipientErrors = validateRecipients(allRecipients); - if (recipientErrors.length > 0) { - throw new Error(`Invalid recipients: ${recipientErrors.join(', ')}`); - } - logEmailSend('start', allRecipients, this.options); - let connection = null; - const result = { - success: false, - acceptedRecipients: [], - rejectedRecipients: [], - envelope: { - from: fromAddress, - to: allRecipients - } - }; - try { - // Get connection - connection = await this.connectionManager.getConnection(); - connection.state = CONNECTION_STATES.BUSY; - // Wait for greeting if new connection - if (!connection.capabilities) { - await this.commandHandler.waitForGreeting(connection); - } - // Perform EHLO - await this.commandHandler.sendEhlo(connection, this.options.domain); - // Upgrade to TLS if needed - if (this.tlsHandler.shouldUseTLS(connection)) { - await this.tlsHandler.upgradeToTLS(connection); - // Re-send EHLO after TLS upgrade - await this.commandHandler.sendEhlo(connection, this.options.domain); - } - // Authenticate if needed - if (this.options.auth) { - await this.authHandler.authenticate(connection); - } - // Send MAIL FROM - const mailFromResponse = await this.commandHandler.sendMailFrom(connection, fromAddress); - if (mailFromResponse.code >= 400) { - throw new Error(`MAIL FROM failed: ${mailFromResponse.message}`); - } - // Send RCPT TO for each recipient (includes TO, CC, and BCC) - for (const recipient of allRecipients) { - try { - const rcptResponse = await this.commandHandler.sendRcptTo(connection, recipient); - if (rcptResponse.code >= 400) { - result.rejectedRecipients.push(recipient); - logDebug(`Recipient rejected: ${recipient}`, this.options, { response: rcptResponse }); - } - else { - result.acceptedRecipients.push(recipient); - } - } - catch (error) { - result.rejectedRecipients.push(recipient); - logDebug(`Recipient error: ${recipient}`, this.options, { error }); - } - } - // Check if we have any accepted recipients - if (result.acceptedRecipients.length === 0) { - throw new Error('All recipients were rejected'); - } - // Send DATA command - const dataResponse = await this.commandHandler.sendData(connection); - if (dataResponse.code !== 354) { - throw new Error(`DATA command failed: ${dataResponse.message}`); - } - // Send email content - const emailData = await this.formatEmailData(email); - const sendResponse = await this.commandHandler.sendDataContent(connection, emailData); - if (sendResponse.code >= 400) { - throw new Error(`Email data rejected: ${sendResponse.message}`); - } - // Success - result.success = true; - result.messageId = this.extractMessageId(sendResponse.message); - result.response = sendResponse.message; - connection.messageCount++; - logEmailSend('success', recipients, this.options, { - messageId: result.messageId, - duration: Date.now() - startTime - }); - } - catch (error) { - result.success = false; - result.error = error instanceof Error ? error : new Error(String(error)); - // Classify error and determine if we should retry - const errorType = this.errorHandler.classifyError(result.error); - result.error = this.errorHandler.createError(result.error.message, errorType, { command: 'SEND_MAIL' }, result.error); - logEmailSend('failure', recipients, this.options, { - error: result.error, - duration: Date.now() - startTime - }); - } - finally { - // Release connection - if (connection) { - connection.state = CONNECTION_STATES.READY; - this.connectionManager.updateActivity(connection); - this.connectionManager.releaseConnection(connection); - } - logPerformance('sendMail', Date.now() - startTime, this.options); - } - return result; - } - /** - * Test connection to SMTP server - */ - async verify() { - let connection = null; - try { - connection = await this.connectionManager.createConnection(); - await this.commandHandler.waitForGreeting(connection); - await this.commandHandler.sendEhlo(connection, this.options.domain); - if (this.tlsHandler.shouldUseTLS(connection)) { - await this.tlsHandler.upgradeToTLS(connection); - await this.commandHandler.sendEhlo(connection, this.options.domain); - } - if (this.options.auth) { - await this.authHandler.authenticate(connection); - } - await this.commandHandler.sendQuit(connection); - return true; - } - catch (error) { - logDebug('Connection verification failed', this.options, { error }); - return false; - } - finally { - if (connection) { - this.connectionManager.closeConnection(connection); - } - } - } - /** - * Check if client is connected - */ - isConnected() { - const status = this.connectionManager.getPoolStatus(); - return status.total > 0; - } - /** - * Get connection pool status - */ - getPoolStatus() { - return this.connectionManager.getPoolStatus(); - } - /** - * Update client options - */ - updateOptions(newOptions) { - this.options = { ...this.options, ...newOptions }; - logDebug('Client options updated', this.options); - } - /** - * Close all connections and shutdown client - */ - async close() { - if (this.isShuttingDown) { - return; - } - this.isShuttingDown = true; - logDebug('Shutting down SMTP client', this.options); - try { - this.connectionManager.closeAllConnections(); - this.emit('close'); - } - catch (error) { - logDebug('Error during client shutdown', this.options, { error }); - } - } - async formatEmailData(email) { - // Convert Email object to raw SMTP data - const headers = []; - // Required headers - headers.push(`From: ${email.from}`); - headers.push(`To: ${Array.isArray(email.to) ? email.to.join(', ') : email.to}`); - headers.push(`Subject: ${email.subject || ''}`); - headers.push(`Date: ${new Date().toUTCString()}`); - headers.push(`Message-ID: <${Date.now()}.${Math.random().toString(36)}@${this.options.host}>`); - // Optional headers - if (email.cc) { - const cc = Array.isArray(email.cc) ? email.cc.join(', ') : email.cc; - headers.push(`Cc: ${cc}`); - } - if (email.bcc) { - const bcc = Array.isArray(email.bcc) ? email.bcc.join(', ') : email.bcc; - headers.push(`Bcc: ${bcc}`); - } - // Content headers - if (email.html && email.text) { - // Multipart message - const boundary = `boundary_${Date.now()}_${Math.random().toString(36)}`; - headers.push(`Content-Type: multipart/alternative; boundary="${boundary}"`); - headers.push('MIME-Version: 1.0'); - const body = [ - `--${boundary}`, - 'Content-Type: text/plain; charset=utf-8', - 'Content-Transfer-Encoding: quoted-printable', - '', - email.text, - '', - `--${boundary}`, - 'Content-Type: text/html; charset=utf-8', - 'Content-Transfer-Encoding: quoted-printable', - '', - email.html, - '', - `--${boundary}--` - ].join('\r\n'); - return headers.join('\r\n') + '\r\n\r\n' + body; - } - else if (email.html) { - headers.push('Content-Type: text/html; charset=utf-8'); - headers.push('MIME-Version: 1.0'); - return headers.join('\r\n') + '\r\n\r\n' + email.html; - } - else { - headers.push('Content-Type: text/plain; charset=utf-8'); - headers.push('MIME-Version: 1.0'); - return headers.join('\r\n') + '\r\n\r\n' + (email.text || ''); - } - } - extractMessageId(response) { - // Try to extract message ID from server response - const match = response.match(/queued as ([^\s]+)/i) || - response.match(/id=([^\s]+)/i) || - response.match(/Message-ID: <([^>]+)>/i); - return match ? match[1] : undefined; - } - setupEventForwarding() { - // Forward events from connection manager - this.connectionManager.on('connection', (connection) => { - this.emit('connection', connection); - }); - this.connectionManager.on('disconnect', (connection) => { - this.emit('disconnect', connection); - }); - this.connectionManager.on('error', (error) => { - this.emit('error', error); - }); - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic210cC1jbGllbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi90cy9tYWlsL2RlbGl2ZXJ5L3NtdHBjbGllbnQvc210cC1jbGllbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7OztHQUdHO0FBRUgsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGFBQWEsQ0FBQztBQVMzQyxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsYUFBYSxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFNbEUsT0FBTyxFQUFFLGNBQWMsRUFBRSxrQkFBa0IsRUFBRSxNQUFNLHVCQUF1QixDQUFDO0FBQzNFLE9BQU8sRUFBRSxZQUFZLEVBQUUsY0FBYyxFQUFFLFFBQVEsRUFBRSxNQUFNLG9CQUFvQixDQUFDO0FBVzVFLE1BQU0sT0FBTyxVQUFXLFNBQVEsWUFBWTtJQUNsQyxPQUFPLENBQXFCO0lBQzVCLGlCQUFpQixDQUFvQjtJQUNyQyxjQUFjLENBQWlCO0lBQy9CLFdBQVcsQ0FBYztJQUN6QixVQUFVLENBQWE7SUFDdkIsWUFBWSxDQUFtQjtJQUMvQixjQUFjLEdBQVksS0FBSyxDQUFDO0lBRXhDLFlBQVksWUFBcUM7UUFDL0MsS0FBSyxFQUFFLENBQUM7UUFFUixJQUFJLENBQUMsT0FBTyxHQUFHLFlBQVksQ0FBQyxPQUFPLENBQUM7UUFDcEMsSUFBSSxDQUFDLGlCQUFpQixHQUFHLFlBQVksQ0FBQyxpQkFBaUIsQ0FBQztRQUN4RCxJQUFJLENBQUMsY0FBYyxHQUFHLFlBQVksQ0FBQyxjQUFjLENBQUM7UUFDbEQsSUFBSSxDQUFDLFdBQVcsR0FBRyxZQUFZLENBQUMsV0FBVyxDQUFDO1FBQzVDLElBQUksQ0FBQyxVQUFVLEdBQUcsWUFBWSxDQUFDLFVBQVUsQ0FBQztRQUMxQyxJQUFJLENBQUMsWUFBWSxHQUFHLFlBQVksQ0FBQyxZQUFZLENBQUM7UUFFOUMsSUFBSSxDQUFDLG9CQUFvQixFQUFFLENBQUM7SUFDOUIsQ0FBQztJQUVEOztPQUVH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 \ No newline at end of file diff --git a/dist_ts/mail/delivery/smtpclient/tls-handler.d.ts b/dist_ts/mail/delivery/smtpclient/tls-handler.d.ts deleted file mode 100644 index 684b493..0000000 --- a/dist_ts/mail/delivery/smtpclient/tls-handler.d.ts +++ /dev/null @@ -1,33 +0,0 @@ -/** - * SMTP Client TLS Handler - * TLS and STARTTLS client functionality - */ -import * as tls from 'node:tls'; -import type { ISmtpConnection, ISmtpClientOptions } from './interfaces.js'; -import type { CommandHandler } from './command-handler.js'; -export declare class TlsHandler { - private options; - private commandHandler; - constructor(options: ISmtpClientOptions, commandHandler: CommandHandler); - /** - * Upgrade connection to TLS using STARTTLS - */ - upgradeToTLS(connection: ISmtpConnection): Promise; - /** - * Create a direct TLS connection - */ - createTLSConnection(host: string, port: number): Promise; - /** - * Validate TLS certificate - */ - validateCertificate(socket: tls.TLSSocket): boolean; - /** - * Get TLS connection information - */ - getTLSInfo(socket: tls.TLSSocket): any; - /** - * Check if TLS upgrade is required or recommended - */ - shouldUseTLS(connection: ISmtpConnection): boolean; - private performTLSUpgrade; -} diff --git a/dist_ts/mail/delivery/smtpclient/tls-handler.js b/dist_ts/mail/delivery/smtpclient/tls-handler.js deleted file mode 100644 index 8fbad1e..0000000 --- a/dist_ts/mail/delivery/smtpclient/tls-handler.js +++ /dev/null @@ -1,204 +0,0 @@ -/** - * SMTP Client TLS Handler - * TLS and STARTTLS client functionality - */ -import * as tls from 'node:tls'; -import * as net from 'node:net'; -import { DEFAULTS } from './constants.js'; -import { CONNECTION_STATES } from './constants.js'; -import { logTLS, logDebug } from './utils/logging.js'; -import { isSuccessCode } from './utils/helpers.js'; -export class TlsHandler { - options; - commandHandler; - constructor(options, commandHandler) { - this.options = options; - this.commandHandler = commandHandler; - } - /** - * Upgrade connection to TLS using STARTTLS - */ - async upgradeToTLS(connection) { - if (connection.secure) { - logDebug('Connection already secure', this.options); - return; - } - // Check if STARTTLS is supported - if (!connection.capabilities?.starttls) { - throw new Error('Server does not support STARTTLS'); - } - logTLS('starttls_start', this.options); - try { - // Send STARTTLS command - const response = await this.commandHandler.sendStartTls(connection); - if (!isSuccessCode(response.code)) { - throw new Error(`STARTTLS command failed: ${response.message}`); - } - // Upgrade the socket to TLS - await this.performTLSUpgrade(connection); - // Clear capabilities as they may have changed after TLS - connection.capabilities = undefined; - connection.secure = true; - logTLS('starttls_success', this.options); - } - catch (error) { - logTLS('starttls_failure', this.options, { error }); - throw error; - } - } - /** - * Create a direct TLS connection - */ - async createTLSConnection(host, port) { - return new Promise((resolve, reject) => { - const timeout = this.options.connectionTimeout || DEFAULTS.CONNECTION_TIMEOUT; - const tlsOptions = { - host, - port, - ...this.options.tls, - // Default TLS options for email - secureProtocol: 'TLS_method', - ciphers: 'HIGH:!aNULL:!eNULL:!EXPORT:!DES:!RC4:!MD5:!PSK:!SRP:!CAMELLIA', - rejectUnauthorized: this.options.tls?.rejectUnauthorized !== false - }; - logTLS('tls_connected', this.options, { host, port }); - const socket = tls.connect(tlsOptions); - const timeoutHandler = setTimeout(() => { - socket.destroy(); - reject(new Error(`TLS connection timeout after ${timeout}ms`)); - }, timeout); - socket.once('secureConnect', () => { - clearTimeout(timeoutHandler); - if (!socket.authorized && this.options.tls?.rejectUnauthorized !== false) { - socket.destroy(); - reject(new Error(`TLS certificate verification failed: ${socket.authorizationError}`)); - return; - } - logDebug('TLS connection established', this.options, { - authorized: socket.authorized, - protocol: socket.getProtocol(), - cipher: socket.getCipher() - }); - resolve(socket); - }); - socket.once('error', (error) => { - clearTimeout(timeoutHandler); - reject(error); - }); - }); - } - /** - * Validate TLS certificate - */ - validateCertificate(socket) { - if (!socket.authorized) { - logDebug('TLS certificate not authorized', this.options, { - error: socket.authorizationError - }); - // Allow self-signed certificates if explicitly configured - if (this.options.tls?.rejectUnauthorized === false) { - logDebug('Accepting unauthorized certificate (rejectUnauthorized: false)', this.options); - return true; - } - return false; - } - const cert = socket.getPeerCertificate(); - if (!cert) { - logDebug('No peer certificate available', this.options); - return false; - } - // Additional certificate validation - const now = new Date(); - if (cert.valid_from && new Date(cert.valid_from) > now) { - logDebug('Certificate not yet valid', this.options, { validFrom: cert.valid_from }); - return false; - } - if (cert.valid_to && new Date(cert.valid_to) < now) { - logDebug('Certificate expired', this.options, { validTo: cert.valid_to }); - return false; - } - logDebug('TLS certificate validated', this.options, { - subject: cert.subject, - issuer: cert.issuer, - validFrom: cert.valid_from, - validTo: cert.valid_to - }); - return true; - } - /** - * Get TLS connection information - */ - getTLSInfo(socket) { - if (!(socket instanceof tls.TLSSocket)) { - return null; - } - return { - authorized: socket.authorized, - authorizationError: socket.authorizationError, - protocol: socket.getProtocol(), - cipher: socket.getCipher(), - peerCertificate: socket.getPeerCertificate(), - alpnProtocol: socket.alpnProtocol - }; - } - /** - * Check if TLS upgrade is required or recommended - */ - shouldUseTLS(connection) { - // Already secure - if (connection.secure) { - return false; - } - // Direct TLS connection configured - if (this.options.secure) { - return false; // Already handled in connection establishment - } - // STARTTLS available and not explicitly disabled - if (connection.capabilities?.starttls) { - return this.options.tls !== null && this.options.tls !== undefined; // Use TLS if configured - } - return false; - } - async performTLSUpgrade(connection) { - return new Promise((resolve, reject) => { - const plainSocket = connection.socket; - const timeout = this.options.connectionTimeout || DEFAULTS.CONNECTION_TIMEOUT; - const tlsOptions = { - socket: plainSocket, - host: this.options.host, - ...this.options.tls, - // Default TLS options for STARTTLS - secureProtocol: 'TLS_method', - ciphers: 'HIGH:!aNULL:!eNULL:!EXPORT:!DES:!RC4:!MD5:!PSK:!SRP:!CAMELLIA', - rejectUnauthorized: this.options.tls?.rejectUnauthorized !== false - }; - const timeoutHandler = setTimeout(() => { - reject(new Error(`TLS upgrade timeout after ${timeout}ms`)); - }, timeout); - // Create TLS socket from existing connection - const tlsSocket = tls.connect(tlsOptions); - tlsSocket.once('secureConnect', () => { - clearTimeout(timeoutHandler); - // Validate certificate if required - if (!this.validateCertificate(tlsSocket)) { - tlsSocket.destroy(); - reject(new Error('TLS certificate validation failed')); - return; - } - // Replace the socket in the connection - connection.socket = tlsSocket; - connection.secure = true; - logDebug('STARTTLS upgrade completed', this.options, { - protocol: tlsSocket.getProtocol(), - cipher: tlsSocket.getCipher() - }); - resolve(); - }); - tlsSocket.once('error', (error) => { - clearTimeout(timeoutHandler); - reject(error); - }); - }); - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGxzLWhhbmRsZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi90cy9tYWlsL2RlbGl2ZXJ5L3NtdHBjbGllbnQvdGxzLWhhbmRsZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7OztHQUdHO0FBRUgsT0FBTyxLQUFLLEdBQUcsTUFBTSxVQUFVLENBQUM7QUFDaEMsT0FBTyxLQUFLLEdBQUcsTUFBTSxVQUFVLENBQUM7QUFDaEMsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBTTFDLE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQ25ELE9BQU8sRUFBRSxNQUFNLEVBQUUsUUFBUSxFQUFFLE1BQU0sb0JBQW9CLENBQUM7QUFDdEQsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLG9CQUFvQixDQUFDO0FBR25ELE1BQU0sT0FBTyxVQUFVO0lBQ2IsT0FBTyxDQUFxQjtJQUM1QixjQUFjLENBQWlCO0lBRXZDLFlBQVksT0FBMkIsRUFBRSxjQUE4QjtRQUNyRSxJQUFJLENBQUMsT0FBTyxHQUFHLE9BQU8sQ0FBQztRQUN2QixJQUFJLENBQUMsY0FBYyxHQUFHLGNBQWMsQ0FBQztJQUN2QyxDQUFDO0lBRUQ7O09BRUc7SUFDSSxLQUFLLENBQUMsWUFBWSxDQUFDLFVBQTJCO1FBQ25ELElBQUksVUFBVSxDQUFDLE1BQU0sRUFBRSxDQUFDO1lBQ3RCLFFBQVEsQ0FBQywyQkFBMkIsRUFBRSxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUM7WUFDcEQsT0FBTztRQUNULENBQUM7UUFFRCxpQ0FBaUM7UUFDakMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxZQUFZLEVBQUUsUUFBUSxFQUFFLENBQUM7WUFDdkMsTUFBTSxJQUFJLEtBQUssQ0FBQyxrQ0FBa0MsQ0FBQyxDQUFDO1FBQ3RELENBQUM7UUFFRCxNQUFNLENBQUMsZ0JBQWdCLEVBQUUsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FB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 \ No newline at end of file diff --git a/dist_ts/mail/delivery/smtpclient/utils/helpers.d.ts b/dist_ts/mail/delivery/smtpclient/utils/helpers.d.ts deleted file mode 100644 index 68a7e3d..0000000 --- a/dist_ts/mail/delivery/smtpclient/utils/helpers.d.ts +++ /dev/null @@ -1,77 +0,0 @@ -/** - * SMTP Client Helper Functions - * Protocol helper functions and utilities - */ -import type { ISmtpResponse, ISmtpCapabilities } from '../interfaces.js'; -/** - * Parse SMTP server response - */ -export declare function parseSmtpResponse(data: string): ISmtpResponse; -/** - * Parse EHLO response and extract capabilities - */ -export declare function parseEhloResponse(response: string): ISmtpCapabilities; -/** - * Format SMTP command with proper line ending - */ -export declare function formatCommand(command: string, ...args: string[]): string; -/** - * Encode authentication string for AUTH PLAIN - */ -export declare function encodeAuthPlain(username: string, password: string): string; -/** - * Encode authentication string for AUTH LOGIN - */ -export declare function encodeAuthLogin(value: string): string; -/** - * Generate OAuth2 authentication string - */ -export declare function generateOAuth2String(username: string, accessToken: string): string; -/** - * Check if response code indicates success - */ -export declare function isSuccessCode(code: number): boolean; -/** - * Check if response code indicates temporary failure - */ -export declare function isTemporaryFailure(code: number): boolean; -/** - * Check if response code indicates permanent failure - */ -export declare function isPermanentFailure(code: number): boolean; -/** - * Escape email address for SMTP commands - */ -export declare function escapeEmailAddress(email: string): string; -/** - * Extract email address from angle brackets - */ -export declare function extractEmailAddress(email: string): string; -/** - * Generate unique connection ID - */ -export declare function generateConnectionId(): string; -/** - * Format timeout duration for human readability - */ -export declare function formatTimeout(milliseconds: number): string; -/** - * Validate and normalize email data size - */ -export declare function validateEmailSize(emailData: string, maxSize?: number): boolean; -/** - * Clean sensitive data from logs - */ -export declare function sanitizeForLogging(data: any): any; -/** - * Calculate exponential backoff delay - */ -export declare function calculateBackoffDelay(attempt: number, baseDelay?: number): number; -/** - * Parse enhanced status code - */ -export declare function parseEnhancedStatusCode(code: string): { - class: number; - subject: number; - detail: number; -} | null; diff --git a/dist_ts/mail/delivery/smtpclient/utils/helpers.js b/dist_ts/mail/delivery/smtpclient/utils/helpers.js deleted file mode 100644 index a3cfc1e..0000000 --- a/dist_ts/mail/delivery/smtpclient/utils/helpers.js +++ /dev/null @@ -1,196 +0,0 @@ -/** - * SMTP Client Helper Functions - * Protocol helper functions and utilities - */ -import { SMTP_CODES, REGEX_PATTERNS, LINE_ENDINGS } from '../constants.js'; -/** - * Parse SMTP server response - */ -export function parseSmtpResponse(data) { - const lines = data.trim().split(/\r?\n/); - const firstLine = lines[0]; - const match = firstLine.match(REGEX_PATTERNS.RESPONSE_CODE); - if (!match) { - return { - code: 500, - message: 'Invalid server response', - raw: data - }; - } - const code = parseInt(match[1], 10); - const separator = match[2]; - const message = lines.map(line => line.substring(4)).join(' '); - // Check for enhanced status code - const enhancedMatch = message.match(REGEX_PATTERNS.ENHANCED_STATUS); - const enhancedCode = enhancedMatch ? enhancedMatch[1] : undefined; - return { - code, - message: enhancedCode ? message.substring(enhancedCode.length + 1) : message, - enhancedCode, - raw: data - }; -} -/** - * Parse EHLO response and extract capabilities - */ -export function parseEhloResponse(response) { - const lines = response.trim().split(/\r?\n/); - const capabilities = { - extensions: new Set(), - authMethods: new Set(), - pipelining: false, - starttls: false, - eightBitMime: false - }; - for (const line of lines.slice(1)) { // Skip first line (greeting) - const extensionLine = line.substring(4); // Remove "250-" or "250 " - const parts = extensionLine.split(/\s+/); - const extension = parts[0].toUpperCase(); - capabilities.extensions.add(extension); - switch (extension) { - case 'PIPELINING': - capabilities.pipelining = true; - break; - case 'STARTTLS': - capabilities.starttls = true; - break; - case '8BITMIME': - capabilities.eightBitMime = true; - break; - case 'SIZE': - if (parts[1]) { - capabilities.maxSize = parseInt(parts[1], 10); - } - break; - case 'AUTH': - // Parse authentication methods - for (let i = 1; i < parts.length; i++) { - capabilities.authMethods.add(parts[i].toUpperCase()); - } - break; - } - } - return capabilities; -} -/** - * Format SMTP command with proper line ending - */ -export function formatCommand(command, ...args) { - const fullCommand = args.length > 0 ? `${command} ${args.join(' ')}` : command; - return fullCommand + LINE_ENDINGS.CRLF; -} -/** - * Encode authentication string for AUTH PLAIN - */ -export function encodeAuthPlain(username, password) { - const authString = `\0${username}\0${password}`; - return Buffer.from(authString, 'utf8').toString('base64'); -} -/** - * Encode authentication string for AUTH LOGIN - */ -export function encodeAuthLogin(value) { - return Buffer.from(value, 'utf8').toString('base64'); -} -/** - * Generate OAuth2 authentication string - */ -export function generateOAuth2String(username, accessToken) { - const authString = `user=${username}\x01auth=Bearer ${accessToken}\x01\x01`; - return Buffer.from(authString, 'utf8').toString('base64'); -} -/** - * Check if response code indicates success - */ -export function isSuccessCode(code) { - return code >= 200 && code < 300; -} -/** - * Check if response code indicates temporary failure - */ -export function isTemporaryFailure(code) { - return code >= 400 && code < 500; -} -/** - * Check if response code indicates permanent failure - */ -export function isPermanentFailure(code) { - return code >= 500; -} -/** - * Escape email address for SMTP commands - */ -export function escapeEmailAddress(email) { - return `<${email.trim()}>`; -} -/** - * Extract email address from angle brackets - */ -export function extractEmailAddress(email) { - const match = email.match(/^<(.+)>$/); - return match ? match[1] : email.trim(); -} -/** - * Generate unique connection ID - */ -export function generateConnectionId() { - return `smtp-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`; -} -/** - * Format timeout duration for human readability - */ -export function formatTimeout(milliseconds) { - if (milliseconds < 1000) { - return `${milliseconds}ms`; - } - else if (milliseconds < 60000) { - return `${Math.round(milliseconds / 1000)}s`; - } - else { - return `${Math.round(milliseconds / 60000)}m`; - } -} -/** - * Validate and normalize email data size - */ -export function validateEmailSize(emailData, maxSize) { - const size = Buffer.byteLength(emailData, 'utf8'); - return !maxSize || size <= maxSize; -} -/** - * Clean sensitive data from logs - */ -export function sanitizeForLogging(data) { - if (typeof data !== 'object' || data === null) { - return data; - } - const sanitized = { ...data }; - const sensitiveFields = ['password', 'pass', 'accessToken', 'refreshToken', 'clientSecret']; - for (const field of sensitiveFields) { - if (field in sanitized) { - sanitized[field] = '[REDACTED]'; - } - } - return sanitized; -} -/** - * Calculate exponential backoff delay - */ -export function calculateBackoffDelay(attempt, baseDelay = 1000) { - return Math.min(baseDelay * Math.pow(2, attempt - 1), 30000); // Max 30 seconds -} -/** - * Parse enhanced status code - */ -export function parseEnhancedStatusCode(code) { - const match = code.match(/^(\d)\.(\d)\.(\d)$/); - if (!match) { - return null; - } - return { - class: parseInt(match[1], 10), - subject: parseInt(match[2], 10), - detail: parseInt(match[3], 10) - }; -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGVscGVycy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3RzL21haWwvZGVsaXZlcnkvc210cGNsaWVudC91dGlscy9oZWxwZXJzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7R0FHRztBQUVILE9BQU8sRUFBRSxVQUFVLEVBQUUsY0FBYyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBRzNFOztHQUVHO0FBQ0gsTUFBTSxVQUFVLGlCQUFpQixDQUFDLElBQVk7SUFDNUMsTUFBTSxLQUFLLEdBQUcsSUFBSSxDQUFDLElBQUksRUFBRSxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQztJQUN6QyxNQUFNLFNBQVMsR0FBRyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDM0IsTUFBTSxLQUFLLEdBQUcsU0FBUyxDQUFDLEtBQUssQ0FBQyxjQUFjLENBQUMsYUFBYSxDQUFDLENBQUM7SUFFNUQsSUFBSSxDQUFDLEtBQUssRUFBRSxDQUFDO1FBQ1gsT0FBTztZQUNMLElBQUksRUFBRSxHQUFHO1lBQ1QsT0FBTyxFQUFFLHlCQUF5QjtZQUNsQyxHQUFHLEVBQUUsSUFBSTtTQUNWLENBQUM7SUFDSixDQUFDO0lBRUQsTUFBTSxJQUFJLEdBQUcsUUFBUSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQztJQUNwQyxNQUFNLFNBQVMsR0FBRyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDM0IsTUFBTSxPQUFPLEdBQUcsS0FBSyxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUM7SUFFL0QsaUNBQWlDO0lBQ2pDLE1BQU0sYUFBYSxHQUFHLE9BQU8sQ0FBQyxLQUFLLENBQUMsY0FBYyxDQUFDLGVBQWUsQ0FBQyxDQUFDO0lBQ3BFLE1B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 \ No newline at end of file diff --git a/dist_ts/mail/delivery/smtpclient/utils/logging.d.ts b/dist_ts/mail/delivery/smtpclient/utils/logging.d.ts deleted file mode 100644 index 951a1f8..0000000 --- a/dist_ts/mail/delivery/smtpclient/utils/logging.d.ts +++ /dev/null @@ -1,46 +0,0 @@ -/** - * SMTP Client Logging Utilities - * Client-side logging utilities for SMTP operations - */ -import type { ISmtpResponse, ISmtpClientOptions } from '../interfaces.js'; -export interface ISmtpClientLogData { - component: string; - host?: string; - port?: number; - secure?: boolean; - command?: string; - response?: ISmtpResponse; - error?: Error; - connectionId?: string; - messageId?: string; - duration?: number; - [key: string]: any; -} -/** - * Log SMTP client connection events - */ -export declare function logConnection(event: 'connecting' | 'connected' | 'disconnected' | 'error', options: ISmtpClientOptions, data?: Partial): void; -/** - * Log SMTP command execution - */ -export declare function logCommand(command: string, response?: ISmtpResponse, options?: ISmtpClientOptions, data?: Partial): void; -/** - * Log authentication events - */ -export declare function logAuthentication(event: 'start' | 'success' | 'failure', method: string, options: ISmtpClientOptions, data?: Partial): void; -/** - * Log TLS/STARTTLS events - */ -export declare function logTLS(event: 'starttls_start' | 'starttls_success' | 'starttls_failure' | 'tls_connected', options: ISmtpClientOptions, data?: Partial): void; -/** - * Log email sending events - */ -export declare function logEmailSend(event: 'start' | 'success' | 'failure', recipients: string[], options: ISmtpClientOptions, data?: Partial): void; -/** - * Log performance metrics - */ -export declare function logPerformance(operation: string, duration: number, options: ISmtpClientOptions, data?: Partial): void; -/** - * Log debug information (only when debug is enabled) - */ -export declare function logDebug(message: string, options: ISmtpClientOptions, data?: Partial): void; diff --git a/dist_ts/mail/delivery/smtpclient/utils/logging.js b/dist_ts/mail/delivery/smtpclient/utils/logging.js deleted file mode 100644 index 0fdc7ce..0000000 --- a/dist_ts/mail/delivery/smtpclient/utils/logging.js +++ /dev/null @@ -1,153 +0,0 @@ -/** - * SMTP Client Logging Utilities - * Client-side logging utilities for SMTP operations - */ -import { logger } from '../../../../logger.js'; -/** - * Log SMTP client connection events - */ -export function logConnection(event, options, data) { - const logData = { - component: 'smtp-client', - event, - host: options.host, - port: options.port, - secure: options.secure, - ...data - }; - switch (event) { - case 'connecting': - logger.info('SMTP client connecting', logData); - break; - case 'connected': - logger.info('SMTP client connected', logData); - break; - case 'disconnected': - logger.info('SMTP client disconnected', logData); - break; - case 'error': - logger.error('SMTP client connection error', logData); - break; - } -} -/** - * Log SMTP command execution - */ -export function logCommand(command, response, options, data) { - const logData = { - component: 'smtp-client', - command, - response, - host: options?.host, - port: options?.port, - ...data - }; - if (response && response.code >= 400) { - logger.warn('SMTP command failed', logData); - } - else { - logger.debug('SMTP command executed', logData); - } -} -/** - * Log authentication events - */ -export function logAuthentication(event, method, options, data) { - const logData = { - component: 'smtp-client', - event: `auth_${event}`, - authMethod: method, - host: options.host, - port: options.port, - ...data - }; - switch (event) { - case 'start': - logger.debug('SMTP authentication started', logData); - break; - case 'success': - logger.info('SMTP authentication successful', logData); - break; - case 'failure': - logger.error('SMTP authentication failed', logData); - break; - } -} -/** - * Log TLS/STARTTLS events - */ -export function logTLS(event, options, data) { - const logData = { - component: 'smtp-client', - event, - host: options.host, - port: options.port, - ...data - }; - if (event.includes('failure')) { - logger.error('SMTP TLS operation failed', logData); - } - else { - logger.info('SMTP TLS operation', logData); - } -} -/** - * Log email sending events - */ -export function logEmailSend(event, recipients, options, data) { - const logData = { - component: 'smtp-client', - event: `send_${event}`, - recipientCount: recipients.length, - recipients: recipients.slice(0, 5), // Only log first 5 recipients for privacy - host: options.host, - port: options.port, - ...data - }; - switch (event) { - case 'start': - logger.info('SMTP email send started', logData); - break; - case 'success': - logger.info('SMTP email send successful', logData); - break; - case 'failure': - logger.error('SMTP email send failed', logData); - break; - } -} -/** - * Log performance metrics - */ -export function logPerformance(operation, duration, options, data) { - const logData = { - component: 'smtp-client', - operation, - duration, - host: options.host, - port: options.port, - ...data - }; - if (duration > 10000) { // Log slow operations (>10s) - logger.warn('SMTP slow operation detected', logData); - } - else { - logger.debug('SMTP operation performance', logData); - } -} -/** - * Log debug information (only when debug is enabled) - */ -export function logDebug(message, options, data) { - if (!options.debug) { - return; - } - const logData = { - component: 'smtp-client-debug', - host: options.host, - port: options.port, - ...data - }; - logger.debug(`[SMTP Client Debug] ${message}`, logData); -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9nZ2luZy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3RzL21haWwvZGVsaXZlcnkvc210cGNsaWVudC91dGlscy9sb2dnaW5nLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7R0FHRztBQUVILE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQWlCL0M7O0dBRUc7QUFDSCxNQUFNLFVBQVUsYUFBYSxDQUMzQixLQUE0RCxFQUM1RCxPQUEyQixFQUMzQixJQUFrQztJQUVsQyxNQUFNLE9BQU8sR0FBdUI7UUFDbEMsU0FBUyxFQUFFLGFBQWE7UUFDeEIsS0FBSztRQUNMLElBQUksRUFBRSxPQUFPLENBQUMsSUFBSTtRQUNsQixJQUFJLEVBQUUsT0FBTyxDQUFDLElBQUk7UUFDbEIsTUFBTSxFQUFFLE9BQU8sQ0FBQyxNQUFNO1FBQ3RCLEdBQUcsSUFBSTtLQUNSLENBQUM7SUFFRixRQUFRLEtBQUssRUFBRSxDQUFDO1FBQ2QsS0FBSyxZQUFZO1lBQ2YsTUFBTSxDQUFDLElBQUksQ0FBQyx3QkFBd0IsRUFBRSxPQUFPLENBQUMsQ0FBQztZQUMvQyxNQUFNO1FBQ1IsS0FBSyxXQUFXO1lBQ2QsTUFBTSxDQUFDLElBQUksQ0FBQyx1QkFBdUIsRUFBRSxPQUFPLENBQUMsQ0FBQztZQUM5QyxNQUFNO1FBQ1IsS0FBSyxjQUFjO1lBQ2pCLE1BQU0sQ0FBQyxJQUFJLENBQUMsMEJBQTBCLEVBQUUsT0FBTyxDQUFDLENBQUM7WUFDakQsTUFBTTtRQUNSLEtBQUssT0FBTztZQUNWLE1BQU0sQ0FBQyxLQUFLLENBQUMsOEJBQThCLEVBQUUsT0FBTyxDQUFDLENBQUM7WUFDdEQsTUFBTTtJQUNWLENBQUM7QUFDSCxDQUFDO0FBRUQ7O0dBRUc7QUFDSCxNQUFNLFVBQVUsVUFBVSxDQUN4QixPQUFlLEVBQ2YsUUFBd0IsRUFDeEIsT0FBNEIsRUFDNUIsSUFBa0M7SUFFbEMsTUFBTSxPQUFPLEdBQXVCO1FB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 \ No newline at end of file diff --git a/dist_ts/mail/delivery/smtpclient/utils/validation.d.ts b/dist_ts/mail/delivery/smtpclient/utils/validation.d.ts deleted file mode 100644 index 33ba3e6..0000000 --- a/dist_ts/mail/delivery/smtpclient/utils/validation.d.ts +++ /dev/null @@ -1,38 +0,0 @@ -/** - * SMTP Client Validation Utilities - * Input validation functions for SMTP client operations - */ -import type { ISmtpClientOptions, ISmtpAuthOptions } from '../interfaces.js'; -/** - * Validate email address format - * Supports RFC-compliant addresses including empty return paths for bounces - */ -export declare function validateEmailAddress(email: string): boolean; -/** - * Validate SMTP client options - */ -export declare function validateClientOptions(options: ISmtpClientOptions): string[]; -/** - * Validate authentication options - */ -export declare function validateAuthOptions(auth: ISmtpAuthOptions): string[]; -/** - * Validate hostname format - */ -export declare function validateHostname(hostname: string): boolean; -/** - * Validate port number - */ -export declare function validatePort(port: number): boolean; -/** - * Sanitize and validate domain name for EHLO - */ -export declare function validateAndSanitizeDomain(domain: string): string; -/** - * Validate recipient list - */ -export declare function validateRecipients(recipients: string | string[]): string[]; -/** - * Validate sender address - */ -export declare function validateSender(sender: string): boolean; diff --git a/dist_ts/mail/delivery/smtpclient/utils/validation.js b/dist_ts/mail/delivery/smtpclient/utils/validation.js deleted file mode 100644 index 3b777b4..0000000 --- a/dist_ts/mail/delivery/smtpclient/utils/validation.js +++ /dev/null @@ -1,139 +0,0 @@ -/** - * SMTP Client Validation Utilities - * Input validation functions for SMTP client operations - */ -import { REGEX_PATTERNS } from '../constants.js'; -/** - * Validate email address format - * Supports RFC-compliant addresses including empty return paths for bounces - */ -export function validateEmailAddress(email) { - if (typeof email !== 'string') { - return false; - } - const trimmed = email.trim(); - // Handle empty return path for bounce messages (RFC 5321) - if (trimmed === '' || trimmed === '<>') { - return true; - } - // Handle display name formats - const angleMatch = trimmed.match(/<([^>]+)>/); - if (angleMatch) { - return REGEX_PATTERNS.EMAIL_ADDRESS.test(angleMatch[1]); - } - // Regular email validation - return REGEX_PATTERNS.EMAIL_ADDRESS.test(trimmed); -} -/** - * Validate SMTP client options - */ -export function validateClientOptions(options) { - const errors = []; - // Required fields - if (!options.host || typeof options.host !== 'string') { - errors.push('Host is required and must be a string'); - } - if (!options.port || typeof options.port !== 'number' || options.port < 1 || options.port > 65535) { - errors.push('Port must be a number between 1 and 65535'); - } - // Optional field validation - if (options.connectionTimeout !== undefined) { - if (typeof options.connectionTimeout !== 'number' || options.connectionTimeout < 1000) { - errors.push('Connection timeout must be a number >= 1000ms'); - } - } - if (options.socketTimeout !== undefined) { - if (typeof options.socketTimeout !== 'number' || options.socketTimeout < 1000) { - errors.push('Socket timeout must be a number >= 1000ms'); - } - } - if (options.maxConnections !== undefined) { - if (typeof options.maxConnections !== 'number' || options.maxConnections < 1) { - errors.push('Max connections must be a positive number'); - } - } - if (options.maxMessages !== undefined) { - if (typeof options.maxMessages !== 'number' || options.maxMessages < 1) { - errors.push('Max messages must be a positive number'); - } - } - // Validate authentication options - if (options.auth) { - errors.push(...validateAuthOptions(options.auth)); - } - return errors; -} -/** - * Validate authentication options - */ -export function validateAuthOptions(auth) { - const errors = []; - if (auth.method && !['PLAIN', 'LOGIN', 'OAUTH2', 'AUTO'].includes(auth.method)) { - errors.push('Invalid authentication method'); - } - // For basic auth, require user and pass - if ((auth.user || auth.pass) && (!auth.user || !auth.pass)) { - errors.push('Both user and pass are required for basic authentication'); - } - // For OAuth2, validate required fields - if (auth.oauth2) { - const oauth = auth.oauth2; - if (!oauth.user || !oauth.clientId || !oauth.clientSecret || !oauth.refreshToken) { - errors.push('OAuth2 requires user, clientId, clientSecret, and refreshToken'); - } - if (oauth.user && !validateEmailAddress(oauth.user)) { - errors.push('OAuth2 user must be a valid email address'); - } - } - return errors; -} -/** - * Validate hostname format - */ -export function validateHostname(hostname) { - if (!hostname || typeof hostname !== 'string') { - return false; - } - // Basic hostname validation (allow IP addresses and domain names) - const hostnameRegex = /^(?:[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?\.)*[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?$|^(?:\d{1,3}\.){3}\d{1,3}$|^\[(?:[0-9a-fA-F]{1,4}:){7}[0-9a-fA-F]{1,4}\]$/; - return hostnameRegex.test(hostname); -} -/** - * Validate port number - */ -export function validatePort(port) { - return typeof port === 'number' && port >= 1 && port <= 65535; -} -/** - * Sanitize and validate domain name for EHLO - */ -export function validateAndSanitizeDomain(domain) { - if (!domain || typeof domain !== 'string') { - return 'localhost'; - } - const sanitized = domain.trim().toLowerCase(); - if (validateHostname(sanitized)) { - return sanitized; - } - return 'localhost'; -} -/** - * Validate recipient list - */ -export function validateRecipients(recipients) { - const errors = []; - const recipientList = Array.isArray(recipients) ? recipients : [recipients]; - for (const recipient of recipientList) { - if (!validateEmailAddress(recipient)) { - errors.push(`Invalid email address: ${recipient}`); - } - } - return errors; -} -/** - * Validate sender address - */ -export function validateSender(sender) { - return validateEmailAddress(sender); -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmFsaWRhdGlvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3RzL21haWwvZGVsaXZlcnkvc210cGNsaWVudC91dGlscy92YWxpZGF0aW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7R0FHRztBQUVILE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUdqRDs7O0dBR0c7QUFDSCxNQUFNLFVBQVUsb0JBQW9CLENBQUMsS0FBYTtJQUNoRCxJQUFJLE9BQU8sS0FBSyxLQUFLLFFBQVEsRUFBRSxDQUFDO1FBQzlCLE9BQU8sS0FBSyxDQUFDO0lBQ2YsQ0FBQztJQUVELE1BQU0sT0FBTyxHQUFHLEtBQUssQ0FBQyxJQUFJLEVBQUUsQ0FBQztJQUU3QiwwREFBMEQ7SUFDMUQsSUFBSSxPQUFPLEtBQUssRUFBRSxJQUFJLE9BQU8sS0FBSyxJQUFJLEVBQUUsQ0FBQztRQUN2QyxPQUFPLElBQUksQ0FBQztJQUNkLENBQUM7SUFFRCw4QkFBOEI7SUFDOUIsTUFBTSxVQUFVLEdBQUcsT0FBTyxDQUFDLEtBQUssQ0FBQyxXQUFXLENBQUMsQ0FBQztJQUM5QyxJQUFJLFVBQVUsRUFBRSxDQUFDO1FBQ2YsT0FBTyxjQUFjLENBQUMsYUFBYSxDQUFDLElBQUksQ0FBQyxVQUFVLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUMxRCxDQUFDO0lBRUQsMkJBQTJCO0lBQzNCLE9BQU8sY0FBYyxDQUFDLGFBQWEsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUM7QUFDcEQsQ0FBQztBQUVEOztHQUVHO0FBQ0gsTUFBTSxVQUFVLHFCQUFxQixDQUFDLE9BQTJCO0lBQy9ELE1BQU0sTUFBTSxHQUFhLEVBQUUsQ0FBQztJQUU1QixrQkFBa0I7SUFDbEIsSUFBSSxDQUFDLE9BQU8sQ0FBQyxJQUFJLElBQUksT0FBTyxPQUFPLENBQUMsSUFBSSxLQUFLLFFBQVEsRUFBRSxDQUFDO1FBQ3RELE1BQU0sQ0FB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 \ No newline at end of file diff --git a/dist_ts/mail/index.d.ts b/dist_ts/mail/index.d.ts deleted file mode 100644 index 555810f..0000000 --- a/dist_ts/mail/index.d.ts +++ /dev/null @@ -1,7 +0,0 @@ -export * from './routing/index.js'; -export * from './security/index.js'; -import * as Core from './core/index.js'; -import * as Delivery from './delivery/index.js'; -export { Core, Delivery }; -import { Email } from './core/classes.email.js'; -export { Email, }; diff --git a/dist_ts/mail/index.js b/dist_ts/mail/index.js deleted file mode 100644 index 802dc02..0000000 --- a/dist_ts/mail/index.js +++ /dev/null @@ -1,12 +0,0 @@ -// Export all mail modules for simplified imports -export * from './routing/index.js'; -export * from './security/index.js'; -// Make the core and delivery modules accessible -import * as Core from './core/index.js'; -import * as Delivery from './delivery/index.js'; -export { Core, Delivery }; -// For direct imports -import { Email } from './core/classes.email.js'; -// Re-export commonly used classes -export { Email, }; -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi90cy9tYWlsL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGlEQUFpRDtBQUNqRCxjQUFjLG9CQUFvQixDQUFDO0FBQ25DLGNBQWMscUJBQXFCLENBQUM7QUFFcEMsZ0RBQWdEO0FBQ2hELE9BQU8sS0FBSyxJQUFJLE1BQU0saUJBQWlCLENBQUM7QUFDeEMsT0FBTyxLQUFLLFFBQVEsTUFBTSxxQkFBcUIsQ0FBQztBQUVoRCxPQUFPLEVBQUUsSUFBSSxFQUFFLFFBQVEsRUFBRSxDQUFDO0FBRTFCLHFCQUFxQjtBQUNyQixPQUFPLEVBQUUsS0FBSyxFQUFFLE1BQU0seUJBQXlCLENBQUM7QUFFaEQsa0NBQWtDO0FBQ2xDLE9BQU8sRUFDTCxLQUFLLEdBQ04sQ0FBQyJ9 \ No newline at end of file diff --git a/dist_ts/mail/routing/classes.dns.manager.d.ts b/dist_ts/mail/routing/classes.dns.manager.d.ts deleted file mode 100644 index a94072d..0000000 --- a/dist_ts/mail/routing/classes.dns.manager.d.ts +++ /dev/null @@ -1,79 +0,0 @@ -import type { IEmailDomainConfig } from './interfaces.js'; -/** External DcRouter interface shape used by DnsManager */ -interface IDcRouterLike { - storageManager: IStorageManagerLike; - dnsServer?: any; - options?: { - dnsNsDomains?: string[]; - dnsScopes?: string[]; - }; -} -/** External StorageManager interface shape used by DnsManager */ -interface IStorageManagerLike { - get(key: string): Promise; - set(key: string, value: string): Promise; -} -/** - * DNS validation result - */ -export interface IDnsValidationResult { - valid: boolean; - errors: string[]; - warnings: string[]; - requiredChanges: string[]; -} -/** - * Manages DNS configuration for email domains - * Handles both validation and creation of DNS records - */ -export declare class DnsManager { - private dcRouter; - private storageManager; - constructor(dcRouter: IDcRouterLike); - /** - * Validate all domain configurations - */ - validateAllDomains(domainConfigs: IEmailDomainConfig[]): Promise>; - /** - * Validate a single domain configuration - */ - validateDomain(config: IEmailDomainConfig): Promise; - /** - * Validate forward mode configuration - */ - private validateForwardMode; - /** - * Validate internal DNS mode configuration - */ - private validateInternalDnsMode; - /** - * Validate external DNS mode configuration - */ - private validateExternalDnsMode; - /** - * Check DNS records for a domain - */ - private checkDnsRecords; - /** - * Resolve NS records for a domain - */ - private resolveNs; - /** - * Get base domain from email domain (e.g., mail.example.com -> example.com) - */ - private getBaseDomain; - /** - * Ensure all DNS records are created for configured domains - * This is the main entry point for DNS record management - */ - ensureDnsRecords(domainConfigs: IEmailDomainConfig[], dkimCreator?: any): Promise; - /** - * Create DNS records for internal-dns mode domains - */ - private createInternalDnsRecords; - /** - * Create DKIM DNS records for all domains - */ - private createDkimRecords; -} -export {}; diff --git a/dist_ts/mail/routing/classes.dns.manager.js b/dist_ts/mail/routing/classes.dns.manager.js deleted file mode 100644 index fcee375..0000000 --- a/dist_ts/mail/routing/classes.dns.manager.js +++ /dev/null @@ -1,415 +0,0 @@ -import * as plugins from '../../plugins.js'; -import { logger } from '../../logger.js'; -/** - * Manages DNS configuration for email domains - * Handles both validation and creation of DNS records - */ -export class DnsManager { - dcRouter; - storageManager; - constructor(dcRouter) { - this.dcRouter = dcRouter; - this.storageManager = dcRouter.storageManager; - } - /** - * Validate all domain configurations - */ - async validateAllDomains(domainConfigs) { - const results = new Map(); - for (const config of domainConfigs) { - const result = await this.validateDomain(config); - results.set(config.domain, result); - } - return results; - } - /** - * Validate a single domain configuration - */ - async validateDomain(config) { - switch (config.dnsMode) { - case 'forward': - return this.validateForwardMode(config); - case 'internal-dns': - return this.validateInternalDnsMode(config); - case 'external-dns': - return this.validateExternalDnsMode(config); - default: - return { - valid: false, - errors: [`Unknown DNS mode: ${config.dnsMode}`], - warnings: [], - requiredChanges: [] - }; - } - } - /** - * Validate forward mode configuration - */ - async validateForwardMode(config) { - const result = { - valid: true, - errors: [], - warnings: [], - requiredChanges: [] - }; - // Forward mode doesn't require DNS validation by default - if (!config.dns?.forward?.skipDnsValidation) { - logger.log('info', `DNS validation skipped for forward mode domain: ${config.domain}`); - } - // DKIM keys are still generated for consistency - result.warnings.push(`Domain "${config.domain}" uses forward mode. DKIM keys will be generated but signing only happens if email is processed.`); - return result; - } - /** - * Validate internal DNS mode configuration - */ - async validateInternalDnsMode(config) { - const result = { - valid: true, - errors: [], - warnings: [], - requiredChanges: [] - }; - // Check if DNS configuration is set up - const dnsNsDomains = this.dcRouter.options?.dnsNsDomains; - const dnsScopes = this.dcRouter.options?.dnsScopes; - if (!dnsNsDomains || dnsNsDomains.length === 0) { - result.valid = false; - result.errors.push(`Domain "${config.domain}" is configured to use internal DNS, but dnsNsDomains is not set in DcRouter configuration.`); - console.error(`❌ ERROR: Domain "${config.domain}" is configured to use internal DNS,\n` + - ' but dnsNsDomains is not set in DcRouter configuration.\n' + - ' Please configure dnsNsDomains to enable the DNS server.\n' + - ' Example: dnsNsDomains: ["ns1.myservice.com", "ns2.myservice.com"]'); - return result; - } - if (!dnsScopes || dnsScopes.length === 0) { - result.valid = false; - result.errors.push(`Domain "${config.domain}" is configured to use internal DNS, but dnsScopes is not set in DcRouter configuration.`); - console.error(`❌ ERROR: Domain "${config.domain}" is configured to use internal DNS,\n` + - ' but dnsScopes is not set in DcRouter configuration.\n' + - ' Please configure dnsScopes to define authoritative domains.\n' + - ' Example: dnsScopes: ["myservice.com", "mail.myservice.com"]'); - return result; - } - // Check if the email domain is in dnsScopes - if (!dnsScopes.includes(config.domain)) { - result.valid = false; - result.errors.push(`Domain "${config.domain}" is configured to use internal DNS, but is not included in dnsScopes.`); - console.error(`❌ ERROR: Domain "${config.domain}" is configured to use internal DNS,\n` + - ` but is not included in dnsScopes: [${dnsScopes.join(', ')}].\n` + - ' Please add this domain to dnsScopes to enable internal DNS.\n' + - ` Example: dnsScopes: [..., "${config.domain}"]`); - return result; - } - const primaryNameserver = dnsNsDomains[0]; - // Check NS delegation - try { - const nsRecords = await this.resolveNs(config.domain); - const delegatedNameservers = dnsNsDomains.filter(ns => nsRecords.includes(ns)); - const isDelegated = delegatedNameservers.length > 0; - if (!isDelegated) { - result.warnings.push(`NS delegation not found for ${config.domain}. Please add NS records at your registrar.`); - dnsNsDomains.forEach(ns => { - result.requiredChanges.push(`Add NS record: ${config.domain}. NS ${ns}.`); - }); - console.log(`📋 DNS Delegation Required for ${config.domain}:\n` + - '━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\n' + - 'Please add these NS records at your domain registrar:\n' + - dnsNsDomains.map(ns => ` ${config.domain}. NS ${ns}.`).join('\n') + '\n' + - '━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\n' + - 'This delegation is required for internal DNS mode to work.'); - } - else { - console.log(`✅ NS delegation verified: ${config.domain} -> [${delegatedNameservers.join(', ')}]`); - } - } - catch (error) { - result.warnings.push(`Could not verify NS delegation for ${config.domain}: ${error.message}`); - } - return result; - } - /** - * Validate external DNS mode configuration - */ - async validateExternalDnsMode(config) { - const result = { - valid: true, - errors: [], - warnings: [], - requiredChanges: [] - }; - try { - // Get current DNS records - const records = await this.checkDnsRecords(config); - const requiredRecords = config.dns?.external?.requiredRecords || ['MX', 'SPF', 'DKIM', 'DMARC']; - // Check MX record - if (requiredRecords.includes('MX') && !records.mx?.length) { - result.requiredChanges.push(`Add MX record: ${this.getBaseDomain(config.domain)} -> ${config.domain} (priority 10)`); - } - // Check SPF record - if (requiredRecords.includes('SPF') && !records.spf) { - result.requiredChanges.push(`Add TXT record: ${this.getBaseDomain(config.domain)} -> "v=spf1 a mx ~all"`); - } - // Check DKIM record - if (requiredRecords.includes('DKIM') && !records.dkim) { - const selector = config.dkim?.selector || 'default'; - const dkimPublicKey = await this.storageManager.get(`/email/dkim/${config.domain}/public.key`); - if (dkimPublicKey) { - const publicKeyBase64 = dkimPublicKey - .replace(/-----BEGIN PUBLIC KEY-----/g, '') - .replace(/-----END PUBLIC KEY-----/g, '') - .replace(/\s/g, ''); - result.requiredChanges.push(`Add TXT record: ${selector}._domainkey.${config.domain} -> "v=DKIM1; k=rsa; p=${publicKeyBase64}"`); - } - else { - result.warnings.push(`DKIM public key not found for ${config.domain}. It will be generated on first use.`); - } - } - // Check DMARC record - if (requiredRecords.includes('DMARC') && !records.dmarc) { - result.requiredChanges.push(`Add TXT record: _dmarc.${this.getBaseDomain(config.domain)} -> "v=DMARC1; p=none; rua=mailto:dmarc@${config.domain}"`); - } - // Show setup instructions if needed - if (result.requiredChanges.length > 0) { - console.log(`📋 DNS Configuration Required for ${config.domain}:\n` + - '━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\n' + - result.requiredChanges.map((change, i) => `${i + 1}. ${change}`).join('\n') + - '\n━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━'); - } - } - catch (error) { - result.errors.push(`DNS validation failed: ${error.message}`); - result.valid = false; - } - return result; - } - /** - * Check DNS records for a domain - */ - async checkDnsRecords(config) { - const records = {}; - const baseDomain = this.getBaseDomain(config.domain); - const selector = config.dkim?.selector || 'default'; - // Use custom DNS servers if specified - const resolver = new plugins.dns.promises.Resolver(); - if (config.dns?.external?.servers?.length) { - resolver.setServers(config.dns.external.servers); - } - // Check MX records - try { - const mxRecords = await resolver.resolveMx(baseDomain); - records.mx = mxRecords.map(mx => mx.exchange); - } - catch (error) { - logger.log('debug', `No MX records found for ${baseDomain}`); - } - // Check SPF record - try { - const txtRecords = await resolver.resolveTxt(baseDomain); - const spfRecord = txtRecords.find(records => records.some(record => record.startsWith('v=spf1'))); - if (spfRecord) { - records.spf = spfRecord.join(''); - } - } - catch (error) { - logger.log('debug', `No SPF record found for ${baseDomain}`); - } - // Check DKIM record - try { - const dkimRecords = await resolver.resolveTxt(`${selector}._domainkey.${config.domain}`); - const dkimRecord = dkimRecords.find(records => records.some(record => record.includes('v=DKIM1'))); - if (dkimRecord) { - records.dkim = dkimRecord.join(''); - } - } - catch (error) { - logger.log('debug', `No DKIM record found for ${selector}._domainkey.${config.domain}`); - } - // Check DMARC record - try { - const dmarcRecords = await resolver.resolveTxt(`_dmarc.${baseDomain}`); - const dmarcRecord = dmarcRecords.find(records => records.some(record => record.startsWith('v=DMARC1'))); - if (dmarcRecord) { - records.dmarc = dmarcRecord.join(''); - } - } - catch (error) { - logger.log('debug', `No DMARC record found for _dmarc.${baseDomain}`); - } - return records; - } - /** - * Resolve NS records for a domain - */ - async resolveNs(domain) { - try { - const resolver = new plugins.dns.promises.Resolver(); - const nsRecords = await resolver.resolveNs(domain); - return nsRecords; - } - catch (error) { - logger.log('warn', `Failed to resolve NS records for ${domain}: ${error.message}`); - return []; - } - } - /** - * Get base domain from email domain (e.g., mail.example.com -> example.com) - */ - getBaseDomain(domain) { - const parts = domain.split('.'); - if (parts.length <= 2) { - return domain; - } - // For subdomains like mail.example.com, return example.com - // But preserve domain structure for longer TLDs like .co.uk - if (parts[parts.length - 2].length <= 3 && parts[parts.length - 1].length === 2) { - // Likely a country code TLD like .co.uk - return parts.slice(-3).join('.'); - } - return parts.slice(-2).join('.'); - } - /** - * Ensure all DNS records are created for configured domains - * This is the main entry point for DNS record management - */ - async ensureDnsRecords(domainConfigs, dkimCreator) { - logger.log('info', `Ensuring DNS records for ${domainConfigs.length} domains`); - // First, validate all domains - const validationResults = await this.validateAllDomains(domainConfigs); - // Then create records for internal-dns domains - const internalDnsDomains = domainConfigs.filter(config => config.dnsMode === 'internal-dns'); - if (internalDnsDomains.length > 0) { - await this.createInternalDnsRecords(internalDnsDomains); - // Create DKIM records if DKIMCreator is provided - if (dkimCreator) { - await this.createDkimRecords(domainConfigs, dkimCreator); - } - } - // Log validation results for external-dns domains - for (const [domain, result] of validationResults) { - const config = domainConfigs.find(c => c.domain === domain); - if (config?.dnsMode === 'external-dns' && result.requiredChanges.length > 0) { - logger.log('warn', `External DNS configuration required for ${domain}`); - } - } - } - /** - * Create DNS records for internal-dns mode domains - */ - async createInternalDnsRecords(domainConfigs) { - // Check if DNS server is available - if (!this.dcRouter.dnsServer) { - logger.log('warn', 'DNS server not available, skipping internal DNS record creation'); - return; - } - logger.log('info', `Creating DNS records for ${domainConfigs.length} internal-dns domains`); - for (const domainConfig of domainConfigs) { - const domain = domainConfig.domain; - const ttl = domainConfig.dns?.internal?.ttl || 3600; - const mxPriority = domainConfig.dns?.internal?.mxPriority || 10; - try { - // 1. Register MX record - points to the email domain itself - this.dcRouter.dnsServer.registerHandler(domain, ['MX'], () => ({ - name: domain, - type: 'MX', - class: 'IN', - ttl: ttl, - data: { - priority: mxPriority, - exchange: domain - } - })); - logger.log('info', `MX record registered for ${domain} -> ${domain} (priority ${mxPriority})`); - // Store MX record in StorageManager - await this.storageManager.set(`/email/dns/${domain}/mx`, JSON.stringify({ - type: 'MX', - priority: mxPriority, - exchange: domain, - ttl: ttl - })); - // 2. Register SPF record - allows the domain to send emails - const spfRecord = `v=spf1 a mx ~all`; - this.dcRouter.dnsServer.registerHandler(domain, ['TXT'], () => ({ - name: domain, - type: 'TXT', - class: 'IN', - ttl: ttl, - data: spfRecord - })); - logger.log('info', `SPF record registered for ${domain}: "${spfRecord}"`); - // Store SPF record in StorageManager - await this.storageManager.set(`/email/dns/${domain}/spf`, JSON.stringify({ - type: 'TXT', - data: spfRecord, - ttl: ttl - })); - // 3. Register DMARC record - policy for handling email authentication - const dmarcRecord = `v=DMARC1; p=none; rua=mailto:dmarc@${domain}`; - this.dcRouter.dnsServer.registerHandler(`_dmarc.${domain}`, ['TXT'], () => ({ - name: `_dmarc.${domain}`, - type: 'TXT', - class: 'IN', - ttl: ttl, - data: dmarcRecord - })); - logger.log('info', `DMARC record registered for _dmarc.${domain}: "${dmarcRecord}"`); - // Store DMARC record in StorageManager - await this.storageManager.set(`/email/dns/${domain}/dmarc`, JSON.stringify({ - type: 'TXT', - name: `_dmarc.${domain}`, - data: dmarcRecord, - ttl: ttl - })); - // Log summary of DNS records created - logger.log('info', `✅ DNS records created for ${domain}: - - MX: ${domain} (priority ${mxPriority}) - - SPF: ${spfRecord} - - DMARC: ${dmarcRecord} - - DKIM: Will be created when keys are generated`); - } - catch (error) { - logger.log('error', `Failed to create DNS records for ${domain}: ${error.message}`); - } - } - } - /** - * Create DKIM DNS records for all domains - */ - async createDkimRecords(domainConfigs, dkimCreator) { - for (const domainConfig of domainConfigs) { - const domain = domainConfig.domain; - const selector = domainConfig.dkim?.selector || 'default'; - try { - // Get DKIM DNS record from DKIMCreator - const dnsRecord = await dkimCreator.getDNSRecordForDomain(domain); - // For internal-dns domains, register the DNS handler - if (domainConfig.dnsMode === 'internal-dns' && this.dcRouter.dnsServer) { - const ttl = domainConfig.dns?.internal?.ttl || 3600; - this.dcRouter.dnsServer.registerHandler(`${selector}._domainkey.${domain}`, ['TXT'], () => ({ - name: `${selector}._domainkey.${domain}`, - type: 'TXT', - class: 'IN', - ttl: ttl, - data: dnsRecord.value - })); - logger.log('info', `DKIM DNS record registered for ${selector}._domainkey.${domain}`); - // Store DKIM record in StorageManager - await this.storageManager.set(`/email/dns/${domain}/dkim`, JSON.stringify({ - type: 'TXT', - name: `${selector}._domainkey.${domain}`, - data: dnsRecord.value, - ttl: ttl - })); - } - // For external-dns domains, just log what should be configured - if (domainConfig.dnsMode === 'external-dns') { - logger.log('info', `DKIM record for external DNS: ${dnsRecord.name} -> "${dnsRecord.value}"`); - } - } - catch (error) { - logger.log('warn', `Could not create DKIM DNS record for ${domain}: ${error.message}`); - } - } - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5kbnMubWFuYWdlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3RzL21haWwvcm91dGluZy9jbGFzc2VzLmRucy5tYW5hZ2VyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sS0FBSyxPQUFPLE1BQU0sa0JBQWtCLENBQUM7QUFFNUMsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBbUN6Qzs7O0dBR0c7QUFDSCxNQUFNLE9BQU8sVUFBVTtJQUNiLFFBQVEsQ0FBZ0I7SUFDeEIsY0FBYyxDQUFzQjtJQUU1QyxZQUFZLFFBQXVCO1FBQ2pDLElBQUksQ0FBQyxRQUFRLEdBQUcsUUFBUSxDQUFDO1FBQ3pCLElBQUksQ0FBQyxjQUFjLEdBQUcsUUFBUSxDQUFDLGNBQWMsQ0FBQztJQUNoRCxDQUFDO0lBRUQ7O09BRUc7SUFDSCxLQUFLLENBQUMsa0JBQWtCLENBQUMsYUFBbUM7UUFDMUQsTUFBTSxPQUFPLEdBQUcsSUFBSSxHQUFHLEVBQWdDLENBQUM7UUFFeEQsS0FBSyxNQUFNLE1BQU0sSUFBSSxhQUFhLEVBQUUsQ0FBQztZQUNuQyxNQUFNLE1BQU0sR0FBRyxNQUFNLElBQUksQ0FBQyxjQUFjLENBQUMsTUFBTSxDQUFDLENBQUM7WUFDakQsT0FBTyxDQUFDLEdBQUcsQ0FBQyxNQUFNLENBQUMsTUFBTSxFQUFFLE1BQU0sQ0FBQyxDQUFDO1FBQ3JDLENBQUM7UUFFRCxPQUFPLE9BQU8sQ0FBQztJQUNqQixDQUFDO0lBRUQ7O09BRUc7SUFDSCxLQUFLLENBQUMsY0FBYyxDQUFDLE1BQTBCO1FBQzdDLFFBQVEsTUFBTSxDQUFDLE9BQU8sRUFBRSxDQUFDO1lBQ3ZCLEtBQUssU0FBUztnQkFDWixPQUFPLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxNQUFNLENBQUMsQ0FBQztZQUMxQyxLQUFLLGNBQWM7Z0JBQ2pCLE9BQU8sSUFBSSxDQUFDLHVCQUF1QixDQUFDLE1BQU0sQ0FBQyxDQUFD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 \ No newline at end of file diff --git a/dist_ts/mail/routing/classes.dnsmanager.d.ts b/dist_ts/mail/routing/classes.dnsmanager.d.ts deleted file mode 100644 index 854d285..0000000 --- a/dist_ts/mail/routing/classes.dnsmanager.d.ts +++ /dev/null @@ -1,165 +0,0 @@ -import * as plugins from '../../plugins.js'; -import { DKIMCreator } from '../security/classes.dkimcreator.js'; -/** - * Interface for DNS record information - */ -export interface IDnsRecord { - name: string; - type: string; - value: string; - ttl?: number; - dnsSecEnabled?: boolean; -} -/** - * Interface for DNS lookup options - */ -export interface IDnsLookupOptions { - /** Cache time to live in milliseconds, 0 to disable caching */ - cacheTtl?: number; - /** Timeout for DNS queries in milliseconds */ - timeout?: number; -} -/** - * Interface for DNS verification result - */ -export interface IDnsVerificationResult { - record: string; - found: boolean; - valid: boolean; - value?: string; - expectedValue?: string; - error?: string; -} -/** - * Manager for DNS-related operations, including record lookups, verification, and generation - */ -export declare class DNSManager { - dkimCreator: DKIMCreator; - private cache; - private defaultOptions; - constructor(dkimCreatorArg: DKIMCreator, options?: IDnsLookupOptions); - /** - * Lookup MX records for a domain - * @param domain Domain to look up - * @param options Lookup options - * @returns Array of MX records sorted by priority - */ - lookupMx(domain: string, options?: IDnsLookupOptions): Promise; - /** - * Lookup TXT records for a domain - * @param domain Domain to look up - * @param options Lookup options - * @returns Array of TXT records - */ - lookupTxt(domain: string, options?: IDnsLookupOptions): Promise; - /** - * Find specific TXT record by subdomain and prefix - * @param domain Base domain - * @param subdomain Subdomain prefix (e.g., "dkim._domainkey") - * @param prefix Record prefix to match (e.g., "v=DKIM1") - * @param options Lookup options - * @returns Matching TXT record or null if not found - */ - findTxtRecord(domain: string, subdomain?: string, prefix?: string, options?: IDnsLookupOptions): Promise; - /** - * Verify if a domain has a valid SPF record - * @param domain Domain to verify - * @returns Verification result - */ - verifySpfRecord(domain: string): Promise; - /** - * Verify if a domain has a valid DKIM record - * @param domain Domain to verify - * @param selector DKIM selector (usually "mta" in our case) - * @returns Verification result - */ - verifyDkimRecord(domain: string, selector?: string): Promise; - /** - * Verify if a domain has a valid DMARC record - * @param domain Domain to verify - * @returns Verification result - */ - verifyDmarcRecord(domain: string): Promise; - /** - * Check all email authentication records (SPF, DKIM, DMARC) for a domain - * @param domain Domain to check - * @param dkimSelector DKIM selector - * @returns Object with verification results for each record type - */ - verifyEmailAuthRecords(domain: string, dkimSelector?: string): Promise<{ - spf: IDnsVerificationResult; - dkim: IDnsVerificationResult; - dmarc: IDnsVerificationResult; - }>; - /** - * Generate a recommended SPF record for a domain - * @param domain Domain name - * @param options Configuration options for the SPF record - * @returns Generated SPF record - */ - generateSpfRecord(domain: string, options?: { - includeMx?: boolean; - includeA?: boolean; - includeIps?: string[]; - includeSpf?: string[]; - policy?: 'none' | 'neutral' | 'softfail' | 'fail' | 'reject'; - }): IDnsRecord; - /** - * Generate a recommended DMARC record for a domain - * @param domain Domain name - * @param options Configuration options for the DMARC record - * @returns Generated DMARC record - */ - generateDmarcRecord(domain: string, options?: { - policy?: 'none' | 'quarantine' | 'reject'; - subdomainPolicy?: 'none' | 'quarantine' | 'reject'; - pct?: number; - rua?: string; - ruf?: string; - daysInterval?: number; - }): IDnsRecord; - /** - * Save DNS record recommendations to a file - * @param domain Domain name - * @param records DNS records to save - */ - saveDnsRecommendations(domain: string, records: IDnsRecord[]): Promise; - /** - * Get cache key value - * @param key Cache key - * @returns Cached value or undefined if not found or expired - */ - private getFromCache; - /** - * Set cache key value - * @param key Cache key - * @param data Data to cache - * @param ttl TTL in milliseconds - */ - private setInCache; - /** - * Clear the DNS cache - * @param key Optional specific key to clear, or all cache if not provided - */ - clearCache(key?: string): void; - /** - * Promise-based wrapper for dns.resolveMx - * @param domain Domain to resolve - * @param timeout Timeout in milliseconds - * @returns Promise resolving to MX records - */ - private dnsResolveMx; - /** - * Promise-based wrapper for dns.resolveTxt - * @param domain Domain to resolve - * @param timeout Timeout in milliseconds - * @returns Promise resolving to TXT records - */ - private dnsResolveTxt; - /** - * Generate all recommended DNS records for proper email authentication - * @param domain Domain to generate records for - * @returns Array of recommended DNS records - */ - generateAllRecommendedRecords(domain: string): Promise; -} diff --git a/dist_ts/mail/routing/classes.dnsmanager.js b/dist_ts/mail/routing/classes.dnsmanager.js deleted file mode 100644 index 16ea2af..0000000 --- a/dist_ts/mail/routing/classes.dnsmanager.js +++ /dev/null @@ -1,431 +0,0 @@ -import * as plugins from '../../plugins.js'; -import * as paths from '../../paths.js'; -import { DKIMCreator } from '../security/classes.dkimcreator.js'; -/** - * Manager for DNS-related operations, including record lookups, verification, and generation - */ -export class DNSManager { - dkimCreator; - cache = new Map(); - defaultOptions = { - cacheTtl: 300000, // 5 minutes - timeout: 5000 // 5 seconds - }; - constructor(dkimCreatorArg, options) { - this.dkimCreator = dkimCreatorArg; - if (options) { - this.defaultOptions = { - ...this.defaultOptions, - ...options - }; - } - // Ensure the DNS records directory exists - plugins.fs.mkdirSync(paths.dnsRecordsDir, { recursive: true }); - } - /** - * Lookup MX records for a domain - * @param domain Domain to look up - * @param options Lookup options - * @returns Array of MX records sorted by priority - */ - async lookupMx(domain, options) { - const lookupOptions = { ...this.defaultOptions, ...options }; - const cacheKey = `mx:${domain}`; - // Check cache first - const cached = this.getFromCache(cacheKey); - if (cached) { - return cached; - } - try { - const records = await this.dnsResolveMx(domain, lookupOptions.timeout); - // Sort by priority - records.sort((a, b) => a.priority - b.priority); - // Cache the result - this.setInCache(cacheKey, records, lookupOptions.cacheTtl); - return records; - } - catch (error) { - console.error(`Error looking up MX records for ${domain}:`, error); - throw new Error(`Failed to lookup MX records for ${domain}: ${error.message}`); - } - } - /** - * Lookup TXT records for a domain - * @param domain Domain to look up - * @param options Lookup options - * @returns Array of TXT records - */ - async lookupTxt(domain, options) { - const lookupOptions = { ...this.defaultOptions, ...options }; - const cacheKey = `txt:${domain}`; - // Check cache first - const cached = this.getFromCache(cacheKey); - if (cached) { - return cached; - } - try { - const records = await this.dnsResolveTxt(domain, lookupOptions.timeout); - // Cache the result - this.setInCache(cacheKey, records, lookupOptions.cacheTtl); - return records; - } - catch (error) { - console.error(`Error looking up TXT records for ${domain}:`, error); - throw new Error(`Failed to lookup TXT records for ${domain}: ${error.message}`); - } - } - /** - * Find specific TXT record by subdomain and prefix - * @param domain Base domain - * @param subdomain Subdomain prefix (e.g., "dkim._domainkey") - * @param prefix Record prefix to match (e.g., "v=DKIM1") - * @param options Lookup options - * @returns Matching TXT record or null if not found - */ - async findTxtRecord(domain, subdomain = '', prefix = '', options) { - const fullDomain = subdomain ? `${subdomain}.${domain}` : domain; - try { - const records = await this.lookupTxt(fullDomain, options); - for (const recordArray of records) { - // TXT records can be split into chunks, join them - const record = recordArray.join(''); - if (!prefix || record.startsWith(prefix)) { - return record; - } - } - return null; - } - catch (error) { - // Domain might not exist or no TXT records - console.log(`No matching TXT record found for ${fullDomain} with prefix ${prefix}`); - return null; - } - } - /** - * Verify if a domain has a valid SPF record - * @param domain Domain to verify - * @returns Verification result - */ - async verifySpfRecord(domain) { - const result = { - record: 'SPF', - found: false, - valid: false - }; - try { - const spfRecord = await this.findTxtRecord(domain, '', 'v=spf1'); - if (spfRecord) { - result.found = true; - result.value = spfRecord; - // Basic validation - check if it contains all, include, ip4, ip6, or mx mechanisms - const isValid = /v=spf1\s+([-~?+]?(all|include:|ip4:|ip6:|mx|a|exists:))/.test(spfRecord); - result.valid = isValid; - if (!isValid) { - result.error = 'SPF record format is invalid'; - } - } - else { - result.error = 'No SPF record found'; - } - } - catch (error) { - result.error = `Error verifying SPF: ${error.message}`; - } - return result; - } - /** - * Verify if a domain has a valid DKIM record - * @param domain Domain to verify - * @param selector DKIM selector (usually "mta" in our case) - * @returns Verification result - */ - async verifyDkimRecord(domain, selector = 'mta') { - const result = { - record: 'DKIM', - found: false, - valid: false - }; - try { - const dkimSelector = `${selector}._domainkey`; - const dkimRecord = await this.findTxtRecord(domain, dkimSelector, 'v=DKIM1'); - if (dkimRecord) { - result.found = true; - result.value = dkimRecord; - // Basic validation - check for required fields - const hasP = dkimRecord.includes('p='); - result.valid = dkimRecord.includes('v=DKIM1') && hasP; - if (!result.valid) { - result.error = 'DKIM record is missing required fields'; - } - else if (dkimRecord.includes('p=') && !dkimRecord.match(/p=[a-zA-Z0-9+/]+/)) { - result.valid = false; - result.error = 'DKIM record has invalid public key format'; - } - } - else { - result.error = `No DKIM record found for selector ${selector}`; - } - } - catch (error) { - result.error = `Error verifying DKIM: ${error.message}`; - } - return result; - } - /** - * Verify if a domain has a valid DMARC record - * @param domain Domain to verify - * @returns Verification result - */ - async verifyDmarcRecord(domain) { - const result = { - record: 'DMARC', - found: false, - valid: false - }; - try { - const dmarcDomain = `_dmarc.${domain}`; - const dmarcRecord = await this.findTxtRecord(dmarcDomain, '', 'v=DMARC1'); - if (dmarcRecord) { - result.found = true; - result.value = dmarcRecord; - // Basic validation - check for required fields - const hasPolicy = dmarcRecord.includes('p='); - result.valid = dmarcRecord.includes('v=DMARC1') && hasPolicy; - if (!result.valid) { - result.error = 'DMARC record is missing required fields'; - } - } - else { - result.error = 'No DMARC record found'; - } - } - catch (error) { - result.error = `Error verifying DMARC: ${error.message}`; - } - return result; - } - /** - * Check all email authentication records (SPF, DKIM, DMARC) for a domain - * @param domain Domain to check - * @param dkimSelector DKIM selector - * @returns Object with verification results for each record type - */ - async verifyEmailAuthRecords(domain, dkimSelector = 'mta') { - const [spf, dkim, dmarc] = await Promise.all([ - this.verifySpfRecord(domain), - this.verifyDkimRecord(domain, dkimSelector), - this.verifyDmarcRecord(domain) - ]); - return { spf, dkim, dmarc }; - } - /** - * Generate a recommended SPF record for a domain - * @param domain Domain name - * @param options Configuration options for the SPF record - * @returns Generated SPF record - */ - generateSpfRecord(domain, options = {}) { - const { includeMx = true, includeA = true, includeIps = [], includeSpf = [], policy = 'softfail' } = options; - let value = 'v=spf1'; - if (includeMx) { - value += ' mx'; - } - if (includeA) { - value += ' a'; - } - // Add IP addresses - for (const ip of includeIps) { - if (ip.includes(':')) { - value += ` ip6:${ip}`; - } - else { - value += ` ip4:${ip}`; - } - } - // Add includes - for (const include of includeSpf) { - value += ` include:${include}`; - } - // Add policy - const policyMap = { - 'none': '?all', - 'neutral': '~all', - 'softfail': '~all', - 'fail': '-all', - 'reject': '-all' - }; - value += ` ${policyMap[policy]}`; - return { - name: domain, - type: 'TXT', - value: value - }; - } - /** - * Generate a recommended DMARC record for a domain - * @param domain Domain name - * @param options Configuration options for the DMARC record - * @returns Generated DMARC record - */ - generateDmarcRecord(domain, options = {}) { - const { policy = 'none', subdomainPolicy, pct = 100, rua, ruf, daysInterval = 1 } = options; - let value = 'v=DMARC1; p=' + policy; - if (subdomainPolicy) { - value += `; sp=${subdomainPolicy}`; - } - if (pct !== 100) { - value += `; pct=${pct}`; - } - if (rua) { - value += `; rua=mailto:${rua}`; - } - if (ruf) { - value += `; ruf=mailto:${ruf}`; - } - if (daysInterval !== 1) { - value += `; ri=${daysInterval * 86400}`; - } - // Add reporting format and ADKIM/ASPF alignment - value += '; fo=1; adkim=r; aspf=r'; - return { - name: `_dmarc.${domain}`, - type: 'TXT', - value: value - }; - } - /** - * Save DNS record recommendations to a file - * @param domain Domain name - * @param records DNS records to save - */ - async saveDnsRecommendations(domain, records) { - try { - const filePath = plugins.path.join(paths.dnsRecordsDir, `${domain}.recommendations.json`); - await plugins.smartfs.file(filePath).write(JSON.stringify(records, null, 2)); - console.log(`DNS recommendations for ${domain} saved to ${filePath}`); - } - catch (error) { - console.error(`Error saving DNS recommendations for ${domain}:`, error); - } - } - /** - * Get cache key value - * @param key Cache key - * @returns Cached value or undefined if not found or expired - */ - getFromCache(key) { - const cached = this.cache.get(key); - if (cached && cached.expires > Date.now()) { - return cached.data; - } - // Remove expired entry - if (cached) { - this.cache.delete(key); - } - return undefined; - } - /** - * Set cache key value - * @param key Cache key - * @param data Data to cache - * @param ttl TTL in milliseconds - */ - setInCache(key, data, ttl = this.defaultOptions.cacheTtl) { - if (ttl <= 0) - return; // Don't cache if TTL is disabled - this.cache.set(key, { - data, - expires: Date.now() + ttl - }); - } - /** - * Clear the DNS cache - * @param key Optional specific key to clear, or all cache if not provided - */ - clearCache(key) { - if (key) { - this.cache.delete(key); - } - else { - this.cache.clear(); - } - } - /** - * Promise-based wrapper for dns.resolveMx - * @param domain Domain to resolve - * @param timeout Timeout in milliseconds - * @returns Promise resolving to MX records - */ - dnsResolveMx(domain, timeout = 5000) { - return new Promise((resolve, reject) => { - const timeoutId = setTimeout(() => { - reject(new Error(`DNS MX lookup timeout for ${domain}`)); - }, timeout); - plugins.dns.resolveMx(domain, (err, addresses) => { - clearTimeout(timeoutId); - if (err) { - reject(err); - } - else { - resolve(addresses); - } - }); - }); - } - /** - * Promise-based wrapper for dns.resolveTxt - * @param domain Domain to resolve - * @param timeout Timeout in milliseconds - * @returns Promise resolving to TXT records - */ - dnsResolveTxt(domain, timeout = 5000) { - return new Promise((resolve, reject) => { - const timeoutId = setTimeout(() => { - reject(new Error(`DNS TXT lookup timeout for ${domain}`)); - }, timeout); - plugins.dns.resolveTxt(domain, (err, records) => { - clearTimeout(timeoutId); - if (err) { - reject(err); - } - else { - resolve(records); - } - }); - }); - } - /** - * Generate all recommended DNS records for proper email authentication - * @param domain Domain to generate records for - * @returns Array of recommended DNS records - */ - async generateAllRecommendedRecords(domain) { - const records = []; - // Get DKIM record (already created by DKIMCreator) - try { - // Call the DKIM creator directly - const dkimRecord = await this.dkimCreator.getDNSRecordForDomain(domain); - records.push(dkimRecord); - } - catch (error) { - console.error(`Error getting DKIM record for ${domain}:`, error); - } - // Generate SPF record - const spfRecord = this.generateSpfRecord(domain, { - includeMx: true, - includeA: true, - policy: 'softfail' - }); - records.push(spfRecord); - // Generate DMARC record - const dmarcRecord = this.generateDmarcRecord(domain, { - policy: 'none', // Start with monitoring mode - rua: `dmarc@${domain}` // Replace with appropriate report address - }); - records.push(dmarcRecord); - // Save recommendations - await this.saveDnsRecommendations(domain, records); - return records; - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5kbnNtYW5hZ2VyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vdHMvbWFpbC9yb3V0aW5nL2NsYXNzZXMuZG5zbWFuYWdlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssT0FBTyxNQUFNLGtCQUFrQixDQUFDO0FBQzVDLE9BQU8sS0FBSyxLQUFLLE1BQU0sZ0JBQWdCLENBQUM7QUFDeEMsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLG9DQUFvQyxDQUFDO0FBbUNqRTs7R0FFRztBQUNILE1BQU0sT0FBTyxVQUFVO0lBQ2QsV0FBVyxDQUFjO0lBQ3hCLEtBQUssR0FBZ0QsSUFBSSxHQUFHLEVBQUUsQ0FBQztJQUMvRCxjQUFjLEdBQXNCO1FBQzFDLFFBQVEsRUFBRSxNQUFNLEVBQUUsWUFBWTtRQUM5QixPQUFPLEVBQUUsSUFBSSxDQUFDLFlBQVk7S0FDM0IsQ0FBQztJQUVGLFlBQVksY0FBMkIsRUFBRSxPQUEyQjtRQUNsRSxJQUFJLENBQUMsV0FBVyxHQUFHLGNBQWMsQ0FBQztRQUVsQyxJQUFJLE9BQU8sRUFBRSxDQUFDO1lBQ1osSUFBSSxDQUFDLGNBQWMsR0FBRztnQkFDcEIsR0FBRyxJQUFJLENBQUMsY0FBYztnQkFDdEIsR0FBRyxPQUFPO2FBQ1gsQ0FBQztRQUNKLENBQUM7UUFFRCwwQ0FBMEM7UUFDMUMsT0FBTyxDQUFDLEVBQUUsQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLGFBQWEsRUFBRSxFQUFFLFNBQVMsRUFBRSxJQUFJLEVBQUUsQ0FBQyxDQUFDO0lBQ2pFLENBQUM7SUFFRDs7Ozs7T0FLRztJQUNJLEtBQUssQ0FBQyxRQUFRLENBQUMsTUFBYyxFQUFFLE9BQTJCO1FBQy9ELE1BQU0sYUFBYSxHQUFHLEVBQUUsR0FBRyxJQUFJLENBQUMsY0FBYyxFQUFFLEdBQUcsT0FBTyxFQUFFLENBQUM7UUFDN0QsTUFBTSxRQUFRLEdBQUcsTUFBTSxNQUFNLEVBQUUsQ0FBQztRQUVo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 \ No newline at end of file diff --git a/dist_ts/mail/routing/classes.domain.registry.d.ts b/dist_ts/mail/routing/classes.domain.registry.d.ts deleted file mode 100644 index 1e09c30..0000000 --- a/dist_ts/mail/routing/classes.domain.registry.d.ts +++ /dev/null @@ -1,54 +0,0 @@ -import type { IEmailDomainConfig } from './interfaces.js'; -/** - * Registry for email domain configurations - * Provides fast lookups and validation for domains - */ -export declare class DomainRegistry { - private domains; - private defaults; - constructor(domainConfigs: IEmailDomainConfig[], defaults?: { - dnsMode?: 'forward' | 'internal-dns' | 'external-dns'; - dkim?: IEmailDomainConfig['dkim']; - rateLimits?: IEmailDomainConfig['rateLimits']; - }); - /** - * Get default DKIM configuration - */ - private getDefaultDkimConfig; - /** - * Apply defaults to a domain configuration - */ - private applyDefaults; - /** - * Check if a domain is registered - */ - isDomainRegistered(domain: string): boolean; - /** - * Check if an email address belongs to a registered domain - */ - isEmailRegistered(email: string): boolean; - /** - * Get domain configuration - */ - getDomainConfig(domain: string): IEmailDomainConfig | undefined; - /** - * Get domain configuration for an email address - */ - getEmailDomainConfig(email: string): IEmailDomainConfig | undefined; - /** - * Extract domain from email address - */ - private extractDomain; - /** - * Get all registered domains - */ - getAllDomains(): string[]; - /** - * Get all domain configurations - */ - getAllConfigs(): IEmailDomainConfig[]; - /** - * Get domains by DNS mode - */ - getDomainsByMode(mode: 'forward' | 'internal-dns' | 'external-dns'): IEmailDomainConfig[]; -} diff --git a/dist_ts/mail/routing/classes.domain.registry.js b/dist_ts/mail/routing/classes.domain.registry.js deleted file mode 100644 index e1c1ef3..0000000 --- a/dist_ts/mail/routing/classes.domain.registry.js +++ /dev/null @@ -1,119 +0,0 @@ -import { logger } from '../../logger.js'; -/** - * Registry for email domain configurations - * Provides fast lookups and validation for domains - */ -export class DomainRegistry { - domains = new Map(); - defaults; - constructor(domainConfigs, defaults) { - // Set defaults - this.defaults = { - dnsMode: defaults?.dnsMode || 'external-dns', - ...this.getDefaultDkimConfig(), - ...defaults?.dkim, - rateLimits: defaults?.rateLimits - }; - // Process and store domain configurations - for (const config of domainConfigs) { - const processedConfig = this.applyDefaults(config); - this.domains.set(config.domain.toLowerCase(), processedConfig); - logger.log('info', `Registered domain: ${config.domain} with DNS mode: ${processedConfig.dnsMode}`); - } - } - /** - * Get default DKIM configuration - */ - getDefaultDkimConfig() { - return { - selector: 'default', - keySize: 2048, - rotateKeys: false, - rotationInterval: 90 - }; - } - /** - * Apply defaults to a domain configuration - */ - applyDefaults(config) { - return { - ...config, - dnsMode: config.dnsMode || this.defaults.dnsMode, - dkim: { - ...this.getDefaultDkimConfig(), - ...this.defaults, - ...config.dkim - }, - rateLimits: { - ...this.defaults.rateLimits, - ...config.rateLimits, - outbound: { - ...this.defaults.rateLimits?.outbound, - ...config.rateLimits?.outbound - }, - inbound: { - ...this.defaults.rateLimits?.inbound, - ...config.rateLimits?.inbound - } - } - }; - } - /** - * Check if a domain is registered - */ - isDomainRegistered(domain) { - return this.domains.has(domain.toLowerCase()); - } - /** - * Check if an email address belongs to a registered domain - */ - isEmailRegistered(email) { - const domain = this.extractDomain(email); - if (!domain) - return false; - return this.isDomainRegistered(domain); - } - /** - * Get domain configuration - */ - getDomainConfig(domain) { - return this.domains.get(domain.toLowerCase()); - } - /** - * Get domain configuration for an email address - */ - getEmailDomainConfig(email) { - const domain = this.extractDomain(email); - if (!domain) - return undefined; - return this.getDomainConfig(domain); - } - /** - * Extract domain from email address - */ - extractDomain(email) { - const parts = email.toLowerCase().split('@'); - if (parts.length !== 2) - return null; - return parts[1]; - } - /** - * Get all registered domains - */ - getAllDomains() { - return Array.from(this.domains.keys()); - } - /** - * Get all domain configurations - */ - getAllConfigs() { - return Array.from(this.domains.values()); - } - /** - * Get domains by DNS mode - */ - getDomainsByMode(mode) { - return Array.from(this.domains.values()).filter(config => config.dnsMode === mode); - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5kb21haW4ucmVnaXN0cnkuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi90cy9tYWlsL3JvdXRpbmcvY2xhc3Nlcy5kb21haW4ucmVnaXN0cnkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBRXpDOzs7R0FHRztBQUNILE1BQU0sT0FBTyxjQUFjO0lBQ2pCLE9BQU8sR0FBb0MsSUFBSSxHQUFHLEVBQUUsQ0FBQztJQUNyRCxRQUFRLENBR2Q7SUFFRixZQUNFLGFBQW1DLEVBQ25DLFFBSUM7UUFFRCxlQUFlO1FBQ2YsSUFBSSxDQUFDLFFBQVEsR0FBRztZQUNkLE9BQU8sRUFBRSxRQUFRLEVBQUUsT0FBTyxJQUFJLGNBQWM7WUFDNUMsR0FBRyxJQUFJLENBQUMsb0JBQW9CLEVBQUU7WUFDOUIsR0FBRyxRQUFRLEVBQUUsSUFBSTtZQUNqQixVQUFVLEVBQUUsUUFBUSxFQUFFLFVBQVU7U0FDakMsQ0FBQztRQUVGLDBDQUEwQztRQUMxQyxLQUFLLE1BQU0sTUFBTSxJQUFJLGFBQWEsRUFBRSxDQUFDO1lBQ25DLE1BQU0sZUFBZSxHQUFHLElBQUksQ0FBQyxhQUFhLENBQUMsTUFBTSxDQUFDLENBQUM7WUFDbkQsSUFBSSxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxXQUFXLEVBQUUsRUFBRSxlQUFlLENBQUMsQ0FBQztZQUMvRCxNQUFNLENBQUMsR0FBRyxDQUFDLE1BQU0sRUFBRSxzQkFBc0IsTUFBTSxDQUFDLE1BQU0sbUJBQW1CLGVBQWUsQ0FBQyxPQUFPLEVBQUUsQ0FBQyxDQUFDO1FBQ3RHLENBQUM7SUFDSCxDQUFDO0lBRUQ7O09BRUc7SUFDSyxvQkFBb0I7UUFDMUIsT0FBTztZQUNMLFFBQVEsRUFBRSxTQUFTO1lBQ25CLE9BQU8sRUFBRSxJQUFJO1lBQ2IsVUFBVSxFQUFFLEtBQUs7WUFDakIsZ0JB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 \ No newline at end of file diff --git a/dist_ts/mail/routing/classes.email.config.d.ts b/dist_ts/mail/routing/classes.email.config.d.ts deleted file mode 100644 index b8e8d76..0000000 --- a/dist_ts/mail/routing/classes.email.config.d.ts +++ /dev/null @@ -1,64 +0,0 @@ -import type { EmailProcessingMode } from '../delivery/interfaces.js'; -export type { EmailProcessingMode }; -/** - * Domain rule interface for pattern-based routing - */ -export interface IDomainRule { - pattern: string; - mode: EmailProcessingMode; - target?: { - server: string; - port?: number; - useTls?: boolean; - authentication?: { - user?: string; - pass?: string; - }; - }; - mtaOptions?: IMtaOptions; - contentScanning?: boolean; - scanners?: IContentScanner[]; - transformations?: ITransformation[]; - rateLimits?: { - maxMessagesPerMinute?: number; - maxRecipientsPerMessage?: number; - }; -} -/** - * MTA options interface - */ -export interface IMtaOptions { - domain?: string; - allowLocalDelivery?: boolean; - localDeliveryPath?: string; - dkimSign?: boolean; - dkimOptions?: { - domainName: string; - keySelector: string; - privateKey?: string; - }; - smtpBanner?: string; - maxConnections?: number; - connTimeout?: number; - spoolDir?: string; -} -/** - * Content scanner interface - */ -export interface IContentScanner { - type: 'spam' | 'virus' | 'attachment'; - threshold?: number; - action: 'tag' | 'reject'; - blockedExtensions?: string[]; -} -/** - * Transformation interface - */ -export interface ITransformation { - type: string; - header?: string; - value?: string; - domains?: string[]; - append?: boolean; - [key: string]: any; -} diff --git a/dist_ts/mail/routing/classes.email.config.js b/dist_ts/mail/routing/classes.email.config.js deleted file mode 100644 index dd8d9dc..0000000 --- a/dist_ts/mail/routing/classes.email.config.js +++ /dev/null @@ -1,2 +0,0 @@ -export {}; -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5lbWFpbC5jb25maWcuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi90cy9tYWlsL3JvdXRpbmcvY2xhc3Nlcy5lbWFpbC5jb25maWcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiJ9 \ No newline at end of file diff --git a/dist_ts/mail/routing/classes.email.router.d.ts b/dist_ts/mail/routing/classes.email.router.d.ts deleted file mode 100644 index 03a51ca..0000000 --- a/dist_ts/mail/routing/classes.email.router.d.ts +++ /dev/null @@ -1,171 +0,0 @@ -import { EventEmitter } from 'node:events'; -import type { IEmailRoute, IEmailContext } from './interfaces.js'; -/** - * Email router that evaluates routes and determines actions - */ -export declare class EmailRouter extends EventEmitter { - private routes; - private patternCache; - private storageManager?; - private persistChanges; - /** - * Create a new email router - * @param routes Array of email routes - * @param options Router options - */ - constructor(routes: IEmailRoute[], options?: { - storageManager?: any; - persistChanges?: boolean; - }); - /** - * Sort routes by priority (higher priority first) - * @param routes Routes to sort - * @returns Sorted routes - */ - private sortRoutesByPriority; - /** - * Get all configured routes - * @returns Array of routes - */ - getRoutes(): IEmailRoute[]; - /** - * Update routes - * @param routes New routes - * @param persist Whether to persist changes (defaults to persistChanges setting) - */ - updateRoutes(routes: IEmailRoute[], persist?: boolean): Promise; - /** - * Set routes (alias for updateRoutes) - * @param routes New routes - * @param persist Whether to persist changes - */ - setRoutes(routes: IEmailRoute[], persist?: boolean): Promise; - /** - * Clear the pattern cache - */ - clearCache(): void; - /** - * Evaluate routes and find the first match - * @param context Email context - * @returns Matched route or null - */ - evaluateRoutes(context: IEmailContext): Promise; - /** - * Check if a route matches the context - * @param route Route to check - * @param context Email context - * @returns True if route matches - */ - private matchesRoute; - /** - * Check if email recipients match patterns - * @param email Email to check - * @param patterns Patterns to match - * @returns True if any recipient matches - */ - private matchesRecipients; - /** - * Check if email sender matches patterns - * @param email Email to check - * @param patterns Patterns to match - * @returns True if sender matches - */ - private matchesSenders; - /** - * Check if client IP matches patterns - * @param context Email context - * @param patterns IP patterns to match - * @returns True if IP matches - */ - private matchesClientIp; - /** - * Check if email headers match patterns - * @param email Email to check - * @param headerPatterns Header patterns to match - * @returns True if headers match - */ - private matchesHeaders; - /** - * Check if email size matches range - * @param email Email to check - * @param sizeRange Size range to match - * @returns True if size is in range - */ - private matchesSize; - /** - * Check if email subject matches pattern - * @param email Email to check - * @param pattern Pattern to match - * @returns True if subject matches - */ - private matchesSubject; - /** - * Check if a string matches a glob pattern - * @param str String to check - * @param pattern Glob pattern - * @returns True if matches - */ - private matchesPattern; - /** - * Convert glob pattern to RegExp - * @param pattern Glob pattern - * @returns Regular expression - */ - private globToRegExp; - /** - * Check if IP is in CIDR range - * @param ip IP address to check - * @param cidr CIDR notation (e.g., '192.168.0.0/16') - * @returns True if IP is in range - */ - private ipInCidr; - /** - * Convert IP address to number - * @param ip IP address - * @returns Number representation - */ - private ipToNumber; - /** - * Calculate approximate email size in bytes - * @param email Email to measure - * @returns Size in bytes - */ - private calculateEmailSize; - /** - * Save current routes to storage - */ - saveRoutes(): Promise; - /** - * Load routes from storage - * @param options Load options - */ - loadRoutes(options?: { - merge?: boolean; - replace?: boolean; - }): Promise; - /** - * Add a route - * @param route Route to add - * @param persist Whether to persist changes - */ - addRoute(route: IEmailRoute, persist?: boolean): Promise; - /** - * Remove a route by name - * @param name Route name - * @param persist Whether to persist changes - */ - removeRoute(name: string, persist?: boolean): Promise; - /** - * Update a route - * @param name Route name - * @param route Updated route data - * @param persist Whether to persist changes - */ - updateRoute(name: string, route: IEmailRoute, persist?: boolean): Promise; - /** - * Get a route by name - * @param name Route name - * @returns Route or undefined - */ - getRoute(name: string): IEmailRoute | undefined; -} diff --git a/dist_ts/mail/routing/classes.email.router.js b/dist_ts/mail/routing/classes.email.router.js deleted file mode 100644 index 142c2c5..0000000 --- a/dist_ts/mail/routing/classes.email.router.js +++ /dev/null @@ -1,494 +0,0 @@ -import * as plugins from '../../plugins.js'; -import { EventEmitter } from 'node:events'; -/** - * Email router that evaluates routes and determines actions - */ -export class EmailRouter extends EventEmitter { - routes; - patternCache = new Map(); - storageManager; // StorageManager instance - persistChanges; - /** - * Create a new email router - * @param routes Array of email routes - * @param options Router options - */ - constructor(routes, options) { - super(); - this.routes = this.sortRoutesByPriority(routes); - this.storageManager = options?.storageManager; - this.persistChanges = options?.persistChanges ?? !!this.storageManager; - // If storage manager is provided, try to load persisted routes - if (this.storageManager) { - this.loadRoutes({ merge: true }).catch(error => { - console.error(`Failed to load persisted routes: ${error.message}`); - }); - } - } - /** - * Sort routes by priority (higher priority first) - * @param routes Routes to sort - * @returns Sorted routes - */ - sortRoutesByPriority(routes) { - return [...routes].sort((a, b) => { - const priorityA = a.priority ?? 0; - const priorityB = b.priority ?? 0; - return priorityB - priorityA; // Higher priority first - }); - } - /** - * Get all configured routes - * @returns Array of routes - */ - getRoutes() { - return [...this.routes]; - } - /** - * Update routes - * @param routes New routes - * @param persist Whether to persist changes (defaults to persistChanges setting) - */ - async updateRoutes(routes, persist) { - this.routes = this.sortRoutesByPriority(routes); - this.clearCache(); - this.emit('routesUpdated', this.routes); - // Persist if requested or if persistChanges is enabled - if (persist ?? this.persistChanges) { - await this.saveRoutes(); - } - } - /** - * Set routes (alias for updateRoutes) - * @param routes New routes - * @param persist Whether to persist changes - */ - async setRoutes(routes, persist) { - await this.updateRoutes(routes, persist); - } - /** - * Clear the pattern cache - */ - clearCache() { - this.patternCache.clear(); - this.emit('cacheCleared'); - } - /** - * Evaluate routes and find the first match - * @param context Email context - * @returns Matched route or null - */ - async evaluateRoutes(context) { - for (const route of this.routes) { - if (await this.matchesRoute(route, context)) { - this.emit('routeMatched', route, context); - return route; - } - } - return null; - } - /** - * Check if a route matches the context - * @param route Route to check - * @param context Email context - * @returns True if route matches - */ - async matchesRoute(route, context) { - const match = route.match; - // Check recipients - if (match.recipients && !this.matchesRecipients(context.email, match.recipients)) { - return false; - } - // Check senders - if (match.senders && !this.matchesSenders(context.email, match.senders)) { - return false; - } - // Check client IP - if (match.clientIp && !this.matchesClientIp(context, match.clientIp)) { - return false; - } - // Check authentication - if (match.authenticated !== undefined && - context.session.authenticated !== match.authenticated) { - return false; - } - // Check headers - if (match.headers && !this.matchesHeaders(context.email, match.headers)) { - return false; - } - // Check size - if (match.sizeRange && !this.matchesSize(context.email, match.sizeRange)) { - return false; - } - // Check subject - if (match.subject && !this.matchesSubject(context.email, match.subject)) { - return false; - } - // Check attachments - if (match.hasAttachments !== undefined && - (context.email.attachments.length > 0) !== match.hasAttachments) { - return false; - } - // All checks passed - return true; - } - /** - * Check if email recipients match patterns - * @param email Email to check - * @param patterns Patterns to match - * @returns True if any recipient matches - */ - matchesRecipients(email, patterns) { - const patternArray = Array.isArray(patterns) ? patterns : [patterns]; - const recipients = email.getAllRecipients(); - for (const recipient of recipients) { - for (const pattern of patternArray) { - if (this.matchesPattern(recipient, pattern)) { - return true; - } - } - } - return false; - } - /** - * Check if email sender matches patterns - * @param email Email to check - * @param patterns Patterns to match - * @returns True if sender matches - */ - matchesSenders(email, patterns) { - const patternArray = Array.isArray(patterns) ? patterns : [patterns]; - const sender = email.from; - for (const pattern of patternArray) { - if (this.matchesPattern(sender, pattern)) { - return true; - } - } - return false; - } - /** - * Check if client IP matches patterns - * @param context Email context - * @param patterns IP patterns to match - * @returns True if IP matches - */ - matchesClientIp(context, patterns) { - const patternArray = Array.isArray(patterns) ? patterns : [patterns]; - const clientIp = context.session.remoteAddress; - if (!clientIp) { - return false; - } - for (const pattern of patternArray) { - // Check for CIDR notation - if (pattern.includes('/')) { - if (this.ipInCidr(clientIp, pattern)) { - return true; - } - } - else { - // Exact match - if (clientIp === pattern) { - return true; - } - } - } - return false; - } - /** - * Check if email headers match patterns - * @param email Email to check - * @param headerPatterns Header patterns to match - * @returns True if headers match - */ - matchesHeaders(email, headerPatterns) { - for (const [header, pattern] of Object.entries(headerPatterns)) { - const value = email.headers[header]; - if (!value) { - return false; - } - if (pattern instanceof RegExp) { - if (!pattern.test(value)) { - return false; - } - } - else { - if (value !== pattern) { - return false; - } - } - } - return true; - } - /** - * Check if email size matches range - * @param email Email to check - * @param sizeRange Size range to match - * @returns True if size is in range - */ - matchesSize(email, sizeRange) { - // Calculate approximate email size - const size = this.calculateEmailSize(email); - if (sizeRange.min !== undefined && size < sizeRange.min) { - return false; - } - if (sizeRange.max !== undefined && size > sizeRange.max) { - return false; - } - return true; - } - /** - * Check if email subject matches pattern - * @param email Email to check - * @param pattern Pattern to match - * @returns True if subject matches - */ - matchesSubject(email, pattern) { - const subject = email.subject || ''; - if (pattern instanceof RegExp) { - return pattern.test(subject); - } - else { - return this.matchesPattern(subject, pattern); - } - } - /** - * Check if a string matches a glob pattern - * @param str String to check - * @param pattern Glob pattern - * @returns True if matches - */ - matchesPattern(str, pattern) { - // Check cache - const cacheKey = `${str}:${pattern}`; - const cached = this.patternCache.get(cacheKey); - if (cached !== undefined) { - return cached; - } - // Convert glob to regex - const regexPattern = this.globToRegExp(pattern); - const matches = regexPattern.test(str); - // Cache result - this.patternCache.set(cacheKey, matches); - return matches; - } - /** - * Convert glob pattern to RegExp - * @param pattern Glob pattern - * @returns Regular expression - */ - globToRegExp(pattern) { - // Escape special regex characters except * and ? - let regexString = pattern - .replace(/[.+^${}()|[\]\\]/g, '\\$&') - .replace(/\*/g, '.*') - .replace(/\?/g, '.'); - return new RegExp(`^${regexString}$`, 'i'); - } - /** - * Check if IP is in CIDR range - * @param ip IP address to check - * @param cidr CIDR notation (e.g., '192.168.0.0/16') - * @returns True if IP is in range - */ - ipInCidr(ip, cidr) { - try { - const [range, bits] = cidr.split('/'); - const mask = parseInt(bits, 10); - // Convert IPs to numbers - const ipNum = this.ipToNumber(ip); - const rangeNum = this.ipToNumber(range); - // Calculate mask - const maskBits = 0xffffffff << (32 - mask); - // Check if in range - return (ipNum & maskBits) === (rangeNum & maskBits); - } - catch { - return false; - } - } - /** - * Convert IP address to number - * @param ip IP address - * @returns Number representation - */ - ipToNumber(ip) { - const parts = ip.split('.'); - return parts.reduce((acc, part, index) => { - return acc + (parseInt(part, 10) << (8 * (3 - index))); - }, 0); - } - /** - * Calculate approximate email size in bytes - * @param email Email to measure - * @returns Size in bytes - */ - calculateEmailSize(email) { - let size = 0; - // Headers - for (const [key, value] of Object.entries(email.headers)) { - size += key.length + value.length + 4; // ": " + "\r\n" - } - // Body - size += (email.text || '').length; - size += (email.html || '').length; - // Attachments - for (const attachment of email.attachments) { - if (attachment.content) { - size += attachment.content.length; - } - } - return size; - } - /** - * Save current routes to storage - */ - async saveRoutes() { - if (!this.storageManager) { - this.emit('persistenceWarning', 'Cannot save routes: StorageManager not configured'); - return; - } - try { - // Validate all routes before saving - for (const route of this.routes) { - if (!route.name || !route.match || !route.action) { - throw new Error(`Invalid route: ${JSON.stringify(route)}`); - } - } - const routesData = JSON.stringify(this.routes, null, 2); - await this.storageManager.set('/email/routes/config.json', routesData); - this.emit('routesPersisted', this.routes.length); - } - catch (error) { - console.error(`Failed to save routes: ${error.message}`); - throw error; - } - } - /** - * Load routes from storage - * @param options Load options - */ - async loadRoutes(options) { - if (!this.storageManager) { - this.emit('persistenceWarning', 'Cannot load routes: StorageManager not configured'); - return []; - } - try { - const routesData = await this.storageManager.get('/email/routes/config.json'); - if (!routesData) { - return []; - } - const loadedRoutes = JSON.parse(routesData); - // Validate loaded routes - for (const route of loadedRoutes) { - if (!route.name || !route.match || !route.action) { - console.warn(`Skipping invalid route: ${JSON.stringify(route)}`); - continue; - } - } - if (options?.replace) { - // Replace all routes - this.routes = this.sortRoutesByPriority(loadedRoutes); - } - else if (options?.merge) { - // Merge with existing routes (loaded routes take precedence) - const routeMap = new Map(); - // Add existing routes - for (const route of this.routes) { - routeMap.set(route.name, route); - } - // Override with loaded routes - for (const route of loadedRoutes) { - routeMap.set(route.name, route); - } - this.routes = this.sortRoutesByPriority(Array.from(routeMap.values())); - } - this.clearCache(); - this.emit('routesLoaded', loadedRoutes.length); - return loadedRoutes; - } - catch (error) { - console.error(`Failed to load routes: ${error.message}`); - throw error; - } - } - /** - * Add a route - * @param route Route to add - * @param persist Whether to persist changes - */ - async addRoute(route, persist) { - // Validate route - if (!route.name || !route.match || !route.action) { - throw new Error('Invalid route: missing required fields'); - } - // Check if route already exists - const existingIndex = this.routes.findIndex(r => r.name === route.name); - if (existingIndex >= 0) { - throw new Error(`Route '${route.name}' already exists`); - } - // Add route - this.routes.push(route); - this.routes = this.sortRoutesByPriority(this.routes); - this.clearCache(); - this.emit('routeAdded', route); - this.emit('routesUpdated', this.routes); - // Persist if requested - if (persist ?? this.persistChanges) { - await this.saveRoutes(); - } - } - /** - * Remove a route by name - * @param name Route name - * @param persist Whether to persist changes - */ - async removeRoute(name, persist) { - const index = this.routes.findIndex(r => r.name === name); - if (index < 0) { - throw new Error(`Route '${name}' not found`); - } - const removedRoute = this.routes.splice(index, 1)[0]; - this.clearCache(); - this.emit('routeRemoved', removedRoute); - this.emit('routesUpdated', this.routes); - // Persist if requested - if (persist ?? this.persistChanges) { - await this.saveRoutes(); - } - } - /** - * Update a route - * @param name Route name - * @param route Updated route data - * @param persist Whether to persist changes - */ - async updateRoute(name, route, persist) { - // Validate route - if (!route.name || !route.match || !route.action) { - throw new Error('Invalid route: missing required fields'); - } - const index = this.routes.findIndex(r => r.name === name); - if (index < 0) { - throw new Error(`Route '${name}' not found`); - } - // Update route - this.routes[index] = route; - this.routes = this.sortRoutesByPriority(this.routes); - this.clearCache(); - this.emit('routeUpdated', route); - this.emit('routesUpdated', this.routes); - // Persist if requested - if (persist ?? this.persistChanges) { - await this.saveRoutes(); - } - } - /** - * Get a route by name - * @param name Route name - * @returns Route or undefined - */ - getRoute(name) { - return this.routes.find(r => r.name === name); - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5lbWFpbC5yb3V0ZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi90cy9tYWlsL3JvdXRpbmcvY2xhc3Nlcy5lbWFpbC5yb3V0ZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxLQUFLLE9BQU8sTUFBTSxrQkFBa0IsQ0FBQztBQUM1QyxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sYUFBYSxDQUFDO0FBSTNDOztHQUVHO0FBQ0gsTUFBTSxPQUFPLFdBQVksU0FBUSxZQUFZO0lBQ25DLE1BQU0sQ0FBZ0I7SUFDdEIsWUFBWSxHQUF5QixJQUFJLEdBQUcsRUFBRSxDQUFDO0lBQy9DLGNBQWMsQ0FBTyxDQUFDLDBCQUEwQjtJQUNoRCxjQUFjLENBQVU7SUFFaEM7Ozs7T0FJRztJQUNILFlBQVksTUFBcUIsRUFBRSxPQUdsQztRQUNDLEtBQUssRUFBRSxDQUFDO1FBQ1IsSUFBSSxDQUFDLE1BQU0sR0FBRyxJQUFJLENBQUMsb0JBQW9CLENBQUMsTUFBTSxDQUFDLENBQUM7UUFDaEQsSUFBSSxDQUFDLGNBQWMsR0FBRyxPQUFPLEVBQUUsY0FBYyxDQUFDO1FBQzlDLElBQUksQ0FBQyxjQUFjLEdBQUcsT0FBTyxFQUFFLGNBQWMsSUFBSSxDQUFDLENBQUMsSUFBSSxDQUFDLGNBQWMsQ0FBQztRQUV2RSwrREFBK0Q7UUFDL0QsSUFBSSxJQUFJLENBQUMsY0FBYyxFQUFFLENBQUM7WUFDeEIsSUFBSSxDQUFDLFVBQVUsQ0FBQyxFQUFFLEtBQUssRUFBRSxJQUFJLEVBQUUsQ0FBQyxDQUFDLEtBQUssQ0FBQyxLQUFLLENBQUMsRUFBRTtnQkFDN0MsT0FBTyxDQUFDLEtBQUssQ0FBQyxvQ0FBb0MsS0FBSyxDQUFDLE9BQU8sRUFBRSxDQUFDLENBQUM7WUFDckUsQ0FBQyxDQUFDLENBQUM7UUFDTCxDQUFDO0lBQ0gsQ0FBQztJQUVEOzs7O09BSUc7SUFDSyxvQkFBb0IsQ0FBQyxNQUFxQjtRQUNoRCxPQUFPLENB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 \ No newline at end of file diff --git a/dist_ts/mail/routing/classes.unified.email.server.d.ts b/dist_ts/mail/routing/classes.unified.email.server.d.ts deleted file mode 100644 index 2db93bb..0000000 --- a/dist_ts/mail/routing/classes.unified.email.server.d.ts +++ /dev/null @@ -1,451 +0,0 @@ -import { EventEmitter } from 'events'; -import { DKIMCreator } from '../security/classes.dkimcreator.js'; -interface IIPWarmupConfig { - enabled?: boolean; - ips?: string[]; - [key: string]: any; -} -interface IReputationMonitorConfig { - enabled?: boolean; - domains?: string[]; - [key: string]: any; -} -import type { IEmailRoute, IEmailDomainConfig } from './interfaces.js'; -import { Email } from '../core/classes.email.js'; -import { DomainRegistry } from './classes.domain.registry.js'; -import { BounceType, BounceCategory } from '../core/classes.bouncemanager.js'; -import type { SmtpClient } from '../delivery/smtpclient/smtp-client.js'; -import { MultiModeDeliverySystem } from '../delivery/classes.delivery.system.js'; -import { UnifiedDeliveryQueue } from '../delivery/classes.delivery.queue.js'; -import { UnifiedRateLimiter, type IHierarchicalRateLimits } from '../delivery/classes.unified.rate.limiter.js'; -import type { EmailProcessingMode, ISmtpSession as IBaseSmtpSession } from '../delivery/interfaces.js'; -/** External DcRouter interface shape used by UnifiedEmailServer */ -interface DcRouter { - storageManager: any; - dnsServer?: any; - options?: any; -} -/** - * Extended SMTP session interface with route information - */ -export interface IExtendedSmtpSession extends ISmtpSession { - /** - * Matched route for this session - */ - matchedRoute?: IEmailRoute; -} -/** - * Options for the unified email server - */ -export interface IUnifiedEmailServerOptions { - ports: number[]; - hostname: string; - domains: IEmailDomainConfig[]; - banner?: string; - debug?: boolean; - useSocketHandler?: boolean; - auth?: { - required?: boolean; - methods?: ('PLAIN' | 'LOGIN' | 'OAUTH2')[]; - users?: Array<{ - username: string; - password: string; - }>; - }; - tls?: { - certPath?: string; - keyPath?: string; - caPath?: string; - minVersion?: string; - ciphers?: string; - }; - maxMessageSize?: number; - maxClients?: number; - maxConnections?: number; - connectionTimeout?: number; - socketTimeout?: number; - routes: IEmailRoute[]; - defaults?: { - dnsMode?: 'forward' | 'internal-dns' | 'external-dns'; - dkim?: IEmailDomainConfig['dkim']; - rateLimits?: IEmailDomainConfig['rateLimits']; - }; - outbound?: { - maxConnections?: number; - connectionTimeout?: number; - socketTimeout?: number; - retryAttempts?: number; - defaultFrom?: string; - }; - rateLimits?: IHierarchicalRateLimits; - ipWarmupConfig?: IIPWarmupConfig; - reputationMonitorConfig?: IReputationMonitorConfig; -} -/** - * Extended SMTP session interface for UnifiedEmailServer - */ -export interface ISmtpSession extends IBaseSmtpSession { - /** - * User information if authenticated - */ - user?: { - username: string; - [key: string]: any; - }; - /** - * Matched route for this session - */ - matchedRoute?: IEmailRoute; -} -/** - * Authentication data for SMTP - */ -import type { ISmtpAuth } from '../delivery/interfaces.js'; -export type IAuthData = ISmtpAuth; -/** - * Server statistics - */ -export interface IServerStats { - startTime: Date; - connections: { - current: number; - total: number; - }; - messages: { - processed: number; - delivered: number; - failed: number; - }; - processingTime: { - avg: number; - max: number; - min: number; - }; -} -/** - * Unified email server that handles all email traffic with pattern-based routing - */ -export declare class UnifiedEmailServer extends EventEmitter { - private dcRouter; - private options; - private emailRouter; - domainRegistry: DomainRegistry; - private servers; - private stats; - dkimCreator: DKIMCreator; - private rustBridge; - private ipReputationChecker; - private bounceManager; - private ipWarmupManager; - private senderReputationMonitor; - deliveryQueue: UnifiedDeliveryQueue; - deliverySystem: MultiModeDeliverySystem; - private rateLimiter; - private dkimKeys; - private smtpClients; - constructor(dcRouter: DcRouter, options: IUnifiedEmailServerOptions); - /** - * Get or create an SMTP client for the given host and port - * Uses connection pooling for efficiency - */ - getSmtpClient(host: string, port?: number): SmtpClient; - /** - * Start the unified email server - */ - start(): Promise; - /** - * Stop the unified email server - */ - stop(): Promise; - /** - * Handle an emailReceived event from the Rust SMTP server. - * Decodes the email data, processes it through the routing system, - * and sends back the result via the correlation-ID callback. - */ - private handleRustEmailReceived; - /** - * Handle an authRequest event from the Rust SMTP server. - * Validates credentials and sends back the result. - */ - private handleRustAuthRequest; - /** - * Verify inbound email security (DKIM/SPF/DMARC) using pre-computed Rust results - * or falling back to IPC call if no pre-computed results are available. - */ - private verifyInboundSecurity; - /** - * Process email based on routing rules - */ - processEmailByMode(emailData: Email | Buffer, session: IExtendedSmtpSession): Promise; - /** - * Execute action based on route configuration - */ - private executeAction; - /** - * Handle forward action - */ - private handleForwardAction; - /** - * Handle process action - */ - private handleProcessAction; - /** - * Handle deliver action - */ - private handleDeliverAction; - /** - * Handle reject action - */ - private handleRejectAction; - /** - * Handle email in MTA mode (programmatic processing) - */ - private _handleMtaMode; - /** - * Handle email in process mode (store-and-forward with scanning) - */ - private _handleProcessMode; - /** - * Get file extension from filename - */ - private getFileExtension; - /** - * Set up DKIM configuration for all domains - */ - private setupDkimForDomains; - /** - * Apply per-domain rate limits from domain configurations - */ - private applyDomainRateLimits; - /** - * Check and rotate DKIM keys if needed - */ - private checkAndRotateDkimKeys; - /** - * Generate SmartProxy routes for email ports - */ - generateProxyRoutes(portMapping?: Record): any[]; - /** - * Update server configuration - */ - updateOptions(options: Partial): void; - /** - * Update email routes - */ - updateEmailRoutes(routes: IEmailRoute[]): void; - /** - * Get server statistics - */ - getStats(): IServerStats; - /** - * Get domain registry - */ - getDomainRegistry(): DomainRegistry; - /** - * Update email routes dynamically - */ - updateRoutes(routes: IEmailRoute[]): void; - /** - * Send an email through the delivery system - * @param email The email to send - * @param mode The processing mode to use - * @param rule Optional rule to apply - * @param options Optional sending options - * @returns The ID of the queued email - */ - sendEmail(email: Email, mode?: EmailProcessingMode, route?: IEmailRoute, options?: { - skipSuppressionCheck?: boolean; - ipAddress?: string; - isTransactional?: boolean; - }): Promise; - /** - * Handle DKIM signing for an email - * @param email The email to sign - * @param domain The domain to sign with - * @param selector The DKIM selector - */ - private handleDkimSigning; - /** - * Process a bounce notification email - * @param bounceEmail The email containing bounce notification information - * @returns Processed bounce record or null if not a bounce - */ - processBounceNotification(bounceEmail: Email): Promise; - /** - * Process an SMTP failure as a bounce - * @param recipient Recipient email that failed - * @param smtpResponse SMTP error response - * @param options Additional options for bounce processing - * @returns Processed bounce record - */ - processSmtpFailure(recipient: string, smtpResponse: string, options?: { - sender?: string; - originalEmailId?: string; - statusCode?: string; - headers?: Record; - }): Promise; - /** - * Check if an email address is suppressed (has bounced previously) - * @param email Email address to check - * @returns Whether the email is suppressed - */ - isEmailSuppressed(email: string): boolean; - /** - * Get suppression information for an email - * @param email Email address to check - * @returns Suppression information or null if not suppressed - */ - getSuppressionInfo(email: string): { - reason: string; - timestamp: number; - expiresAt?: number; - } | null; - /** - * Get bounce history information for an email - * @param email Email address to check - * @returns Bounce history or null if no bounces - */ - getBounceHistory(email: string): { - lastBounce: number; - count: number; - type: BounceType; - category: BounceCategory; - } | null; - /** - * Get all suppressed email addresses - * @returns Array of suppressed email addresses - */ - getSuppressionList(): string[]; - /** - * Get all hard bounced email addresses - * @returns Array of hard bounced email addresses - */ - getHardBouncedAddresses(): string[]; - /** - * Add an email to the suppression list - * @param email Email address to suppress - * @param reason Reason for suppression - * @param expiresAt Optional expiration time (undefined for permanent) - */ - addToSuppressionList(email: string, reason: string, expiresAt?: number): void; - /** - * Remove an email from the suppression list - * @param email Email address to remove from suppression - */ - removeFromSuppressionList(email: string): void; - /** - * Get the status of IP warmup process - * @param ipAddress Optional specific IP to check - * @returns Status of IP warmup - */ - getIPWarmupStatus(ipAddress?: string): any; - /** - * Add a new IP address to the warmup process - * @param ipAddress IP address to add - */ - addIPToWarmup(ipAddress: string): void; - /** - * Remove an IP address from the warmup process - * @param ipAddress IP address to remove - */ - removeIPFromWarmup(ipAddress: string): void; - /** - * Update metrics for an IP in the warmup process - * @param ipAddress IP address - * @param metrics Metrics to update - */ - updateIPWarmupMetrics(ipAddress: string, metrics: { - openRate?: number; - bounceRate?: number; - complaintRate?: number; - }): void; - /** - * Check if an IP can send more emails today - * @param ipAddress IP address to check - * @returns Whether the IP can send more today - */ - canIPSendMoreToday(ipAddress: string): boolean; - /** - * Check if an IP can send more emails in the current hour - * @param ipAddress IP address to check - * @returns Whether the IP can send more this hour - */ - canIPSendMoreThisHour(ipAddress: string): boolean; - /** - * Get the best IP to use for sending an email based on warmup status - * @param emailInfo Information about the email being sent - * @returns Best IP to use or null - */ - getBestIPForSending(emailInfo: { - from: string; - to: string[]; - domain: string; - isTransactional?: boolean; - }): string | null; - /** - * Set the active IP allocation policy for warmup - * @param policyName Name of the policy to set - */ - setIPAllocationPolicy(policyName: string): void; - /** - * Record that an email was sent using a specific IP - * @param ipAddress IP address used for sending - */ - recordIPSend(ipAddress: string): void; - /** - * Get reputation data for a domain - * @param domain Domain to get reputation for - * @returns Domain reputation metrics - */ - getDomainReputationData(domain: string): any; - /** - * Get summary reputation data for all monitored domains - * @returns Summary data for all domains - */ - getReputationSummary(): any; - /** - * Add a domain to the reputation monitoring system - * @param domain Domain to add - */ - addDomainToMonitoring(domain: string): void; - /** - * Remove a domain from the reputation monitoring system - * @param domain Domain to remove - */ - removeDomainFromMonitoring(domain: string): void; - /** - * Record an email event for domain reputation tracking - * @param domain Domain sending the email - * @param event Event details - */ - recordReputationEvent(domain: string, event: { - type: 'sent' | 'delivered' | 'bounce' | 'complaint' | 'open' | 'click'; - count?: number; - hardBounce?: boolean; - receivingDomain?: string; - }): void; - /** - * Check if DKIM key exists for a domain - * @param domain Domain to check - */ - hasDkimKey(domain: string): boolean; - /** - * Record successful email delivery - * @param domain Sending domain - */ - recordDelivery(domain: string): void; - /** - * Record email bounce - * @param domain Sending domain - * @param receivingDomain Receiving domain that bounced - * @param bounceType Type of bounce (hard/soft) - * @param reason Bounce reason - */ - recordBounce(domain: string, receivingDomain: string, bounceType: 'hard' | 'soft', reason: string): void; - /** - * Get the rate limiter instance - * @returns The unified rate limiter - */ - getRateLimiter(): UnifiedRateLimiter; -} -export {}; diff --git a/dist_ts/mail/routing/classes.unified.email.server.js b/dist_ts/mail/routing/classes.unified.email.server.js deleted file mode 100644 index 0da1147..0000000 --- a/dist_ts/mail/routing/classes.unified.email.server.js +++ /dev/null @@ -1,1555 +0,0 @@ -import * as plugins from '../../plugins.js'; -import * as paths from '../../paths.js'; -import { EventEmitter } from 'events'; -import { logger } from '../../logger.js'; -import { SecurityLogger, SecurityLogLevel, SecurityEventType } from '../../security/index.js'; -import { DKIMCreator } from '../security/classes.dkimcreator.js'; -import { IPReputationChecker } from '../../security/classes.ipreputationchecker.js'; -import { RustSecurityBridge } from '../../security/classes.rustsecuritybridge.js'; -import { EmailRouter } from './classes.email.router.js'; -import { Email } from '../core/classes.email.js'; -import { DomainRegistry } from './classes.domain.registry.js'; -import { DnsManager } from './classes.dns.manager.js'; -import { BounceManager, BounceType, BounceCategory } from '../core/classes.bouncemanager.js'; -import { createPooledSmtpClient } from '../delivery/smtpclient/create-client.js'; -import { MultiModeDeliverySystem } from '../delivery/classes.delivery.system.js'; -import { UnifiedDeliveryQueue } from '../delivery/classes.delivery.queue.js'; -import { UnifiedRateLimiter } from '../delivery/classes.unified.rate.limiter.js'; -import { SmtpState } from '../delivery/interfaces.js'; -/** - * Unified email server that handles all email traffic with pattern-based routing - */ -export class UnifiedEmailServer extends EventEmitter { - dcRouter; - options; - emailRouter; - domainRegistry; - servers = []; - stats; - // Add components needed for sending and securing emails - dkimCreator; - rustBridge; - ipReputationChecker; - bounceManager; - ipWarmupManager; - senderReputationMonitor; - deliveryQueue; - deliverySystem; - rateLimiter; // TODO: Implement rate limiting in SMTP server handlers - dkimKeys = new Map(); // domain -> private key - smtpClients = new Map(); // host:port -> client - constructor(dcRouter, options) { - super(); - this.dcRouter = dcRouter; - // Set default options - this.options = { - ...options, - banner: options.banner || `${options.hostname} ESMTP UnifiedEmailServer`, - maxMessageSize: options.maxMessageSize || 10 * 1024 * 1024, // 10MB - maxClients: options.maxClients || 100, - maxConnections: options.maxConnections || 1000, - connectionTimeout: options.connectionTimeout || 60000, // 1 minute - socketTimeout: options.socketTimeout || 60000 // 1 minute - }; - // Initialize Rust security bridge (singleton) - this.rustBridge = RustSecurityBridge.getInstance(); - // Initialize DKIM creator with storage manager - this.dkimCreator = new DKIMCreator(paths.keysDir, dcRouter.storageManager); - // Initialize IP reputation checker with storage manager - this.ipReputationChecker = IPReputationChecker.getInstance({ - enableLocalCache: true, - enableDNSBL: true, - enableIPInfo: true - }, dcRouter.storageManager); - // Initialize bounce manager with storage manager - this.bounceManager = new BounceManager({ - maxCacheSize: 10000, - cacheTTL: 30 * 24 * 60 * 60 * 1000, // 30 days - storageManager: dcRouter.storageManager - }); - // IP warmup manager and sender reputation monitor are optional - // They will be initialized when the deliverability module is available - this.ipWarmupManager = null; - this.senderReputationMonitor = null; - // Initialize domain registry - this.domainRegistry = new DomainRegistry(options.domains, options.defaults); - // Initialize email router with routes and storage manager - this.emailRouter = new EmailRouter(options.routes || [], { - storageManager: dcRouter.storageManager, - persistChanges: true - }); - // Initialize rate limiter - this.rateLimiter = new UnifiedRateLimiter(options.rateLimits || { - global: { - maxConnectionsPerIP: 10, - maxMessagesPerMinute: 100, - maxRecipientsPerMessage: 50, - maxErrorsPerIP: 10, - maxAuthFailuresPerIP: 5, - blockDuration: 300000 // 5 minutes - } - }); - // Initialize delivery components - const queueOptions = { - storageType: 'memory', // Default to memory storage - maxRetries: 3, - baseRetryDelay: 300000, // 5 minutes - maxRetryDelay: 3600000 // 1 hour - }; - this.deliveryQueue = new UnifiedDeliveryQueue(queueOptions); - const deliveryOptions = { - globalRateLimit: 100, // Default to 100 emails per minute - concurrentDeliveries: 10, - processBounces: true, - bounceHandler: { - processSmtpFailure: this.processSmtpFailure.bind(this) - }, - onDeliverySuccess: async (item, _result) => { - // Record delivery success event for reputation monitoring - const email = item.processingResult; - const senderDomain = email.from.split('@')[1]; - if (senderDomain) { - this.recordReputationEvent(senderDomain, { - type: 'delivered', - count: email.to.length - }); - } - } - }; - this.deliverySystem = new MultiModeDeliverySystem(this.deliveryQueue, deliveryOptions, this); - // Initialize statistics - this.stats = { - startTime: new Date(), - connections: { - current: 0, - total: 0 - }, - messages: { - processed: 0, - delivered: 0, - failed: 0 - }, - processingTime: { - avg: 0, - max: 0, - min: 0 - } - }; - // We'll create the SMTP servers during the start() method - } - /** - * Get or create an SMTP client for the given host and port - * Uses connection pooling for efficiency - */ - getSmtpClient(host, port = 25) { - const clientKey = `${host}:${port}`; - // Check if we already have a client for this destination - let client = this.smtpClients.get(clientKey); - if (!client) { - // Create a new pooled SMTP client - client = createPooledSmtpClient({ - host, - port, - secure: port === 465, - connectionTimeout: this.options.outbound?.connectionTimeout || 30000, - socketTimeout: this.options.outbound?.socketTimeout || 120000, - maxConnections: this.options.outbound?.maxConnections || 10, - maxMessages: 1000, // Messages per connection before reconnect - pool: true, - debug: false - }); - this.smtpClients.set(clientKey, client); - logger.log('info', `Created new SMTP client pool for ${clientKey}`); - } - return client; - } - /** - * Start the unified email server - */ - async start() { - logger.log('info', `Starting UnifiedEmailServer on ports: ${this.options.ports.join(', ')}`); - try { - // Initialize the delivery queue - await this.deliveryQueue.initialize(); - logger.log('info', 'Email delivery queue initialized'); - // Start the delivery system - await this.deliverySystem.start(); - logger.log('info', 'Email delivery system started'); - // Start Rust security bridge — required for all security operations - const bridgeOk = await this.rustBridge.start(); - if (!bridgeOk) { - throw new Error('Rust security bridge failed to start. The mailer-bin binary is required. Run "pnpm build" to compile it.'); - } - logger.log('info', 'Rust security bridge started — Rust is the primary security backend'); - // Set up DKIM for all domains - await this.setupDkimForDomains(); - logger.log('info', 'DKIM configuration completed for all domains'); - // Create DNS manager and ensure all DNS records are created - const dnsManager = new DnsManager(this.dcRouter); - await dnsManager.ensureDnsRecords(this.domainRegistry.getAllConfigs(), this.dkimCreator); - logger.log('info', 'DNS records ensured for all configured domains'); - // Apply per-domain rate limits - this.applyDomainRateLimits(); - logger.log('info', 'Per-domain rate limits configured'); - // Check and rotate DKIM keys if needed - await this.checkAndRotateDkimKeys(); - logger.log('info', 'DKIM key rotation check completed'); - // Ensure we have the necessary TLS options - const hasTlsConfig = this.options.tls?.keyPath && this.options.tls?.certPath; - // Prepare the certificate and key if available - let tlsCertPem; - let tlsKeyPem; - if (hasTlsConfig) { - try { - tlsKeyPem = plugins.fs.readFileSync(this.options.tls.keyPath, 'utf8'); - tlsCertPem = plugins.fs.readFileSync(this.options.tls.certPath, 'utf8'); - logger.log('info', 'TLS certificates loaded successfully'); - } - catch (error) { - logger.log('warn', `Failed to load TLS certificates: ${error.message}`); - } - } - // --- Start Rust SMTP server --- - // Register event handlers for email reception and auth - this.rustBridge.onEmailReceived(async (data) => { - try { - await this.handleRustEmailReceived(data); - } - catch (err) { - logger.log('error', `Error handling email from Rust SMTP: ${err.message}`); - // Send rejection back to Rust - await this.rustBridge.sendEmailProcessingResult({ - correlationId: data.correlationId, - accepted: false, - smtpCode: 451, - smtpMessage: 'Internal processing error', - }); - } - }); - this.rustBridge.onAuthRequest(async (data) => { - try { - await this.handleRustAuthRequest(data); - } - catch (err) { - logger.log('error', `Error handling auth from Rust SMTP: ${err.message}`); - await this.rustBridge.sendAuthResult({ - correlationId: data.correlationId, - success: false, - message: 'Internal auth error', - }); - } - }); - // Determine which ports need STARTTLS and which need implicit TLS - const smtpPorts = this.options.ports.filter(p => p !== 465); - const securePort = this.options.ports.find(p => p === 465); - const started = await this.rustBridge.startSmtpServer({ - hostname: this.options.hostname, - ports: smtpPorts, - securePort: securePort, - tlsCertPem, - tlsKeyPem, - maxMessageSize: this.options.maxMessageSize || 10 * 1024 * 1024, - maxConnections: this.options.maxConnections || this.options.maxClients || 100, - maxRecipients: 100, - connectionTimeoutSecs: this.options.connectionTimeout ? Math.floor(this.options.connectionTimeout / 1000) : 30, - dataTimeoutSecs: 60, - authEnabled: !!this.options.auth?.required || !!(this.options.auth?.users?.length), - maxAuthFailures: 3, - socketTimeoutSecs: this.options.socketTimeout ? Math.floor(this.options.socketTimeout / 1000) : 300, - processingTimeoutSecs: 30, - rateLimits: this.options.rateLimits ? { - maxConnectionsPerIp: this.options.rateLimits.global?.maxConnectionsPerIP || 50, - maxMessagesPerSender: this.options.rateLimits.global?.maxMessagesPerMinute || 100, - maxAuthFailuresPerIp: this.options.rateLimits.global?.maxAuthFailuresPerIP || 5, - windowSecs: 60, - } : undefined, - }); - if (!started) { - throw new Error('Failed to start Rust SMTP server'); - } - logger.log('info', `Rust SMTP server listening on ports: ${smtpPorts.join(', ')}${securePort ? ` + ${securePort} (TLS)` : ''}`); - logger.log('info', 'UnifiedEmailServer started successfully'); - this.emit('started'); - } - catch (error) { - logger.log('error', `Failed to start UnifiedEmailServer: ${error.message}`); - throw error; - } - } - /** - * Stop the unified email server - */ - async stop() { - logger.log('info', 'Stopping UnifiedEmailServer'); - try { - // Stop the Rust SMTP server first - try { - await this.rustBridge.stopSmtpServer(); - logger.log('info', 'Rust SMTP server stopped'); - } - catch (err) { - logger.log('warn', `Error stopping Rust SMTP server: ${err.message}`); - } - // Clear the servers array - servers will be garbage collected - this.servers = []; - // Stop Rust security bridge - await this.rustBridge.stop(); - // Stop the delivery system - if (this.deliverySystem) { - await this.deliverySystem.stop(); - logger.log('info', 'Email delivery system stopped'); - } - // Shut down the delivery queue - if (this.deliveryQueue) { - await this.deliveryQueue.shutdown(); - logger.log('info', 'Email delivery queue shut down'); - } - // Close all SMTP client connections - for (const [clientKey, client] of this.smtpClients) { - try { - await client.close(); - logger.log('info', `Closed SMTP client pool for ${clientKey}`); - } - catch (error) { - logger.log('warn', `Error closing SMTP client for ${clientKey}: ${error.message}`); - } - } - this.smtpClients.clear(); - logger.log('info', 'UnifiedEmailServer stopped successfully'); - this.emit('stopped'); - } - catch (error) { - logger.log('error', `Error stopping UnifiedEmailServer: ${error.message}`); - throw error; - } - } - // ----------------------------------------------------------------------- - // Rust SMTP server event handlers - // ----------------------------------------------------------------------- - /** - * Handle an emailReceived event from the Rust SMTP server. - * Decodes the email data, processes it through the routing system, - * and sends back the result via the correlation-ID callback. - */ - async handleRustEmailReceived(data) { - const { correlationId, mailFrom, rcptTo, remoteAddr, clientHostname, secure, authenticatedUser } = data; - logger.log('info', `Rust SMTP received email from=${mailFrom} to=${rcptTo.join(',')} remote=${remoteAddr}`); - try { - // Decode the email data - let rawMessageBuffer; - if (data.data.type === 'inline' && data.data.base64) { - rawMessageBuffer = Buffer.from(data.data.base64, 'base64'); - } - else if (data.data.type === 'file' && data.data.path) { - rawMessageBuffer = plugins.fs.readFileSync(data.data.path); - // Clean up temp file - try { - plugins.fs.unlinkSync(data.data.path); - } - catch { - // Ignore cleanup errors - } - } - else { - throw new Error('Invalid email data transport'); - } - // Build a session-like object for processEmailByMode - const session = { - id: data.sessionId || 'rust-' + Math.random().toString(36).substring(2), - state: SmtpState.FINISHED, - mailFrom: mailFrom, - rcptTo: rcptTo, - emailData: rawMessageBuffer.toString('utf8'), - useTLS: secure, - connectionEnded: false, - remoteAddress: remoteAddr, - clientHostname: clientHostname || '', - secure: secure, - authenticated: !!authenticatedUser, - envelope: { - mailFrom: { address: mailFrom, args: {} }, - rcptTo: rcptTo.map(addr => ({ address: addr, args: {} })), - }, - }; - if (authenticatedUser) { - session.user = { username: authenticatedUser }; - } - // Attach pre-computed security results from Rust in-process pipeline - if (data.securityResults) { - session._precomputedSecurityResults = data.securityResults; - } - // Process the email through the routing system - await this.processEmailByMode(rawMessageBuffer, session); - // Send acceptance back to Rust - await this.rustBridge.sendEmailProcessingResult({ - correlationId, - accepted: true, - smtpCode: 250, - smtpMessage: '2.0.0 Message accepted for delivery', - }); - } - catch (err) { - logger.log('error', `Failed to process email from Rust SMTP: ${err.message}`); - await this.rustBridge.sendEmailProcessingResult({ - correlationId, - accepted: false, - smtpCode: 550, - smtpMessage: `5.0.0 Processing failed: ${err.message}`, - }); - } - } - /** - * Handle an authRequest event from the Rust SMTP server. - * Validates credentials and sends back the result. - */ - async handleRustAuthRequest(data) { - const { correlationId, username, password, remoteAddr } = data; - logger.log('info', `Rust SMTP auth request for user=${username} from=${remoteAddr}`); - // Check against configured users - const users = this.options.auth?.users || []; - const matched = users.find(u => u.username === username && u.password === password); - if (matched) { - await this.rustBridge.sendAuthResult({ - correlationId, - success: true, - }); - } - else { - logger.log('warn', `Auth failed for user=${username} from=${remoteAddr}`); - await this.rustBridge.sendAuthResult({ - correlationId, - success: false, - message: 'Invalid credentials', - }); - } - } - /** - * Verify inbound email security (DKIM/SPF/DMARC) using pre-computed Rust results - * or falling back to IPC call if no pre-computed results are available. - */ - async verifyInboundSecurity(email, session) { - try { - // Check for pre-computed results from Rust in-process security pipeline - const precomputed = session._precomputedSecurityResults; - let result; - if (precomputed) { - logger.log('info', 'Using pre-computed security results from Rust in-process pipeline'); - result = precomputed; - } - else { - // Fallback: IPC round-trip to Rust (for backward compat / handleSocket mode) - const rawMessage = session.emailData || email.toRFC822String(); - result = await this.rustBridge.verifyEmail({ - rawMessage, - ip: session.remoteAddress, - heloDomain: session.clientHostname || '', - hostname: this.options.hostname, - mailFrom: session.envelope?.mailFrom?.address || session.mailFrom || '', - }); - } - // Apply DKIM result headers - if (result.dkim && result.dkim.length > 0) { - const dkimSummary = result.dkim - .map((d) => `${d.status}${d.domain ? ` (${d.domain})` : ''}`) - .join(', '); - email.addHeader('X-DKIM-Result', dkimSummary); - } - // Apply SPF result header - if (result.spf) { - email.addHeader('Received-SPF', `${result.spf.result} (domain: ${result.spf.domain}, ip: ${result.spf.ip})`); - // Mark as spam on SPF hard fail - if (result.spf.result === 'fail') { - email.mightBeSpam = true; - logger.log('warn', `SPF fail for ${session.remoteAddress} — marking as potential spam`); - } - } - // Apply DMARC result header and policy - if (result.dmarc) { - email.addHeader('X-DMARC-Result', `${result.dmarc.action} (policy=${result.dmarc.policy}, dkim=${result.dmarc.dkim_result}, spf=${result.dmarc.spf_result})`); - if (result.dmarc.action === 'reject') { - email.mightBeSpam = true; - logger.log('warn', `DMARC reject for domain ${result.dmarc.domain} — marking as spam`); - } - else if (result.dmarc.action === 'quarantine') { - email.mightBeSpam = true; - logger.log('info', `DMARC quarantine for domain ${result.dmarc.domain} — marking as potential spam`); - } - } - // Apply content scan results (from pre-computed pipeline) - if (result.contentScan) { - const scan = result.contentScan; - if (scan.threatScore > 0) { - email.addHeader('X-Spam-Score', String(scan.threatScore)); - if (scan.threatType) { - email.addHeader('X-Spam-Type', scan.threatType); - } - if (scan.threatScore >= 50) { - email.mightBeSpam = true; - logger.log('warn', `Content scan threat score ${scan.threatScore} (${scan.threatType}) — marking as potential spam`); - } - } - } - // Apply IP reputation results (from pre-computed pipeline) - if (result.ipReputation) { - const rep = result.ipReputation; - email.addHeader('X-IP-Reputation-Score', String(rep.score)); - if (rep.is_spam) { - email.mightBeSpam = true; - logger.log('warn', `IP ${rep.ip} flagged by reputation check (score=${rep.score}) — marking as potential spam`); - } - } - logger.log('info', `Inbound security verified for email from ${session.remoteAddress}: DKIM=${result.dkim?.[0]?.status ?? 'none'}, SPF=${result.spf?.result ?? 'none'}, DMARC=${result.dmarc?.action ?? 'none'}`); - } - catch (err) { - logger.log('warn', `Inbound security verification failed: ${err.message} — accepting email`); - } - } - /** - * Process email based on routing rules - */ - async processEmailByMode(emailData, session) { - // Convert Buffer to Email if needed - let email; - if (Buffer.isBuffer(emailData)) { - // Parse the email data buffer into an Email object - try { - const parsed = await plugins.mailparser.simpleParser(emailData); - email = new Email({ - from: parsed.from?.value[0]?.address || session.envelope.mailFrom.address, - to: session.envelope.rcptTo[0]?.address || '', - subject: parsed.subject || '', - text: parsed.text || '', - html: parsed.html || undefined, - attachments: parsed.attachments?.map(att => ({ - filename: att.filename || '', - content: att.content, - contentType: att.contentType - })) || [] - }); - } - catch (error) { - logger.log('error', `Error parsing email data: ${error.message}`); - throw new Error(`Error parsing email data: ${error.message}`); - } - } - else { - email = emailData; - } - // Run inbound security verification (DKIM/SPF/DMARC) via Rust bridge - if (session.remoteAddress && session.remoteAddress !== '127.0.0.1') { - await this.verifyInboundSecurity(email, session); - } - // First check if this is a bounce notification email - // Look for common bounce notification subject patterns - const subject = email.subject || ''; - const isBounceLike = /mail delivery|delivery (failed|status|notification)|failure notice|returned mail|undeliverable|delivery problem/i.test(subject); - if (isBounceLike) { - logger.log('info', `Email subject matches bounce notification pattern: "${subject}"`); - // Try to process as a bounce - const isBounce = await this.processBounceNotification(email); - if (isBounce) { - logger.log('info', 'Successfully processed as bounce notification, skipping regular processing'); - return email; - } - logger.log('info', 'Not a valid bounce notification, continuing with regular processing'); - } - // Find matching route - const context = { email, session }; - const route = await this.emailRouter.evaluateRoutes(context); - if (!route) { - // No matching route - reject - throw new Error('No matching route for email'); - } - // Store matched route in session - session.matchedRoute = route; - // Execute action based on route - await this.executeAction(route.action, email, context); - // Return the processed email - return email; - } - /** - * Execute action based on route configuration - */ - async executeAction(action, email, context) { - switch (action.type) { - case 'forward': - await this.handleForwardAction(action, email, context); - break; - case 'process': - await this.handleProcessAction(action, email, context); - break; - case 'deliver': - await this.handleDeliverAction(action, email, context); - break; - case 'reject': - await this.handleRejectAction(action, email, context); - break; - default: - throw new Error(`Unknown action type: ${action.type}`); - } - } - /** - * Handle forward action - */ - async handleForwardAction(_action, email, context) { - if (!_action.forward) { - throw new Error('Forward action requires forward configuration'); - } - const { host, port = 25, auth, addHeaders } = _action.forward; - logger.log('info', `Forwarding email to ${host}:${port}`); - // Add forwarding headers - if (addHeaders) { - for (const [key, value] of Object.entries(addHeaders)) { - email.headers[key] = value; - } - } - // Add standard forwarding headers - email.headers['X-Forwarded-For'] = context.session.remoteAddress || 'unknown'; - email.headers['X-Forwarded-To'] = email.to.join(', '); - email.headers['X-Forwarded-Date'] = new Date().toISOString(); - // Get SMTP client - const client = this.getSmtpClient(host, port); - try { - // Send email - await client.sendMail(email); - logger.log('info', `Successfully forwarded email to ${host}:${port}`); - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.INFO, - type: SecurityEventType.EMAIL_FORWARDING, - message: 'Email forwarded successfully', - ipAddress: context.session.remoteAddress, - details: { - sessionId: context.session.id, - routeName: context.session.matchedRoute?.name, - targetHost: host, - targetPort: port, - recipients: email.to - }, - success: true - }); - } - catch (error) { - logger.log('error', `Failed to forward email: ${error.message}`); - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.ERROR, - type: SecurityEventType.EMAIL_FORWARDING, - message: 'Email forwarding failed', - ipAddress: context.session.remoteAddress, - details: { - sessionId: context.session.id, - routeName: context.session.matchedRoute?.name, - targetHost: host, - targetPort: port, - error: error.message - }, - success: false - }); - // Handle as bounce - for (const recipient of email.getAllRecipients()) { - await this.bounceManager.processSmtpFailure(recipient, error.message, { - sender: email.from, - originalEmailId: email.headers['Message-ID'] - }); - } - throw error; - } - } - /** - * Handle process action - */ - async handleProcessAction(action, email, context) { - logger.log('info', `Processing email with action options`); - // Apply scanning if requested - if (action.process?.scan) { - // Use existing content scanner - // Note: ContentScanner integration would go here - logger.log('info', 'Content scanning requested'); - } - // Note: DKIM signing will be applied at delivery time to ensure signature validity - // Queue for delivery - const queue = action.process?.queue || 'normal'; - await this.deliveryQueue.enqueue(email, 'process', context.session.matchedRoute); - logger.log('info', `Email queued for delivery in ${queue} queue`); - } - /** - * Handle deliver action - */ - async handleDeliverAction(_action, email, context) { - logger.log('info', `Delivering email locally`); - // Queue for local delivery - await this.deliveryQueue.enqueue(email, 'mta', context.session.matchedRoute); - logger.log('info', 'Email queued for local delivery'); - } - /** - * Handle reject action - */ - async handleRejectAction(action, email, context) { - const code = action.reject?.code || 550; - const message = action.reject?.message || 'Message rejected'; - logger.log('info', `Rejecting email with code ${code}: ${message}`); - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.WARN, - type: SecurityEventType.EMAIL_PROCESSING, - message: 'Email rejected by routing rule', - ipAddress: context.session.remoteAddress, - details: { - sessionId: context.session.id, - routeName: context.session.matchedRoute?.name, - rejectCode: code, - rejectMessage: message, - from: email.from, - to: email.to - }, - success: false - }); - // Throw error with SMTP code and message - const error = new Error(message); - error.responseCode = code; - throw error; - } - /** - * Handle email in MTA mode (programmatic processing) - */ - async _handleMtaMode(email, session) { - logger.log('info', `Handling email in MTA mode for session ${session.id}`); - try { - // Apply MTA rule options if provided - if (session.matchedRoute?.action.options?.mtaOptions) { - const options = session.matchedRoute.action.options.mtaOptions; - // Apply DKIM signing if enabled - if (options.dkimSign && options.dkimOptions) { - const dkimDomain = options.dkimOptions.domainName; - const dkimSelector = options.dkimOptions.keySelector || 'mta'; - logger.log('info', `Signing email with DKIM for domain ${dkimDomain}`); - await this.handleDkimSigning(email, dkimDomain, dkimSelector); - } - } - // Get email content for logging/processing - const subject = email.subject; - const recipients = email.getAllRecipients().join(', '); - logger.log('info', `Email processed by MTA: ${subject} to ${recipients}`); - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.INFO, - type: SecurityEventType.EMAIL_PROCESSING, - message: 'Email processed by MTA', - ipAddress: session.remoteAddress, - details: { - sessionId: session.id, - ruleName: session.matchedRoute?.name || 'default', - subject, - recipients - }, - success: true - }); - } - catch (error) { - logger.log('error', `Failed to process email in MTA mode: ${error.message}`); - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.ERROR, - type: SecurityEventType.EMAIL_PROCESSING, - message: 'MTA processing failed', - ipAddress: session.remoteAddress, - details: { - sessionId: session.id, - ruleName: session.matchedRoute?.name || 'default', - error: error.message - }, - success: false - }); - throw error; - } - } - /** - * Handle email in process mode (store-and-forward with scanning) - */ - async _handleProcessMode(email, session) { - logger.log('info', `Handling email in process mode for session ${session.id}`); - try { - const route = session.matchedRoute; - // Apply content scanning if enabled - if (route?.action.options?.contentScanning && route.action.options.scanners && route.action.options.scanners.length > 0) { - logger.log('info', 'Performing content scanning'); - // Apply each scanner - for (const scanner of route.action.options.scanners) { - switch (scanner.type) { - case 'spam': - logger.log('info', 'Scanning for spam content'); - // Implement spam scanning - break; - case 'virus': - logger.log('info', 'Scanning for virus content'); - // Implement virus scanning - break; - case 'attachment': - logger.log('info', 'Scanning attachments'); - // Check for blocked extensions - if (scanner.blockedExtensions && scanner.blockedExtensions.length > 0) { - for (const attachment of email.attachments) { - const ext = this.getFileExtension(attachment.filename); - if (scanner.blockedExtensions.includes(ext)) { - if (scanner.action === 'reject') { - throw new Error(`Blocked attachment type: ${ext}`); - } - else { // tag - email.addHeader('X-Attachment-Warning', `Potentially unsafe attachment: ${attachment.filename}`); - } - } - } - } - break; - } - } - } - // Apply transformations if defined - if (route?.action.options?.transformations && route.action.options.transformations.length > 0) { - logger.log('info', 'Applying email transformations'); - for (const transform of route.action.options.transformations) { - switch (transform.type) { - case 'addHeader': - if (transform.header && transform.value) { - email.addHeader(transform.header, transform.value); - } - break; - } - } - } - logger.log('info', `Email successfully processed in store-and-forward mode`); - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.INFO, - type: SecurityEventType.EMAIL_PROCESSING, - message: 'Email processed and queued', - ipAddress: session.remoteAddress, - details: { - sessionId: session.id, - ruleName: route?.name || 'default', - contentScanning: route?.action.options?.contentScanning || false, - subject: email.subject - }, - success: true - }); - } - catch (error) { - logger.log('error', `Failed to process email: ${error.message}`); - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.ERROR, - type: SecurityEventType.EMAIL_PROCESSING, - message: 'Email processing failed', - ipAddress: session.remoteAddress, - details: { - sessionId: session.id, - ruleName: session.matchedRoute?.name || 'default', - error: error.message - }, - success: false - }); - throw error; - } - } - /** - * Get file extension from filename - */ - getFileExtension(filename) { - return filename.substring(filename.lastIndexOf('.')).toLowerCase(); - } - /** - * Set up DKIM configuration for all domains - */ - async setupDkimForDomains() { - const domainConfigs = this.domainRegistry.getAllConfigs(); - if (domainConfigs.length === 0) { - logger.log('warn', 'No domains configured for DKIM'); - return; - } - for (const domainConfig of domainConfigs) { - const domain = domainConfig.domain; - const selector = domainConfig.dkim?.selector || 'default'; - try { - // Check if DKIM keys already exist for this domain - let keyPair; - try { - // Try to read existing keys - keyPair = await this.dkimCreator.readDKIMKeys(domain); - logger.log('info', `Using existing DKIM keys for domain: ${domain}`); - } - catch (error) { - // Generate new keys if they don't exist - keyPair = await this.dkimCreator.createDKIMKeys(); - // Store them for future use - await this.dkimCreator.createAndStoreDKIMKeys(domain); - logger.log('info', `Generated new DKIM keys for domain: ${domain}`); - } - // Store the private key for signing - this.dkimKeys.set(domain, keyPair.privateKey); - // DNS record creation is now handled by DnsManager - logger.log('info', `DKIM keys loaded for domain: ${domain} with selector: ${selector}`); - } - catch (error) { - logger.log('error', `Failed to set up DKIM for domain ${domain}: ${error.message}`); - } - } - } - /** - * Apply per-domain rate limits from domain configurations - */ - applyDomainRateLimits() { - const domainConfigs = this.domainRegistry.getAllConfigs(); - for (const domainConfig of domainConfigs) { - if (domainConfig.rateLimits) { - const domain = domainConfig.domain; - const rateLimitConfig = {}; - // Convert domain-specific rate limits to the format expected by UnifiedRateLimiter - if (domainConfig.rateLimits.outbound) { - if (domainConfig.rateLimits.outbound.messagesPerMinute) { - rateLimitConfig.maxMessagesPerMinute = domainConfig.rateLimits.outbound.messagesPerMinute; - } - // Note: messagesPerHour and messagesPerDay would need additional implementation in rate limiter - } - if (domainConfig.rateLimits.inbound) { - if (domainConfig.rateLimits.inbound.messagesPerMinute) { - rateLimitConfig.maxMessagesPerMinute = domainConfig.rateLimits.inbound.messagesPerMinute; - } - if (domainConfig.rateLimits.inbound.connectionsPerIp) { - rateLimitConfig.maxConnectionsPerIP = domainConfig.rateLimits.inbound.connectionsPerIp; - } - if (domainConfig.rateLimits.inbound.recipientsPerMessage) { - rateLimitConfig.maxRecipientsPerMessage = domainConfig.rateLimits.inbound.recipientsPerMessage; - } - } - // Apply the rate limits if we have any - if (Object.keys(rateLimitConfig).length > 0) { - this.rateLimiter.applyDomainLimits(domain, rateLimitConfig); - logger.log('info', `Applied rate limits for domain ${domain}:`, rateLimitConfig); - } - } - } - } - /** - * Check and rotate DKIM keys if needed - */ - async checkAndRotateDkimKeys() { - const domainConfigs = this.domainRegistry.getAllConfigs(); - for (const domainConfig of domainConfigs) { - const domain = domainConfig.domain; - const selector = domainConfig.dkim?.selector || 'default'; - const rotateKeys = domainConfig.dkim?.rotateKeys || false; - const rotationInterval = domainConfig.dkim?.rotationInterval || 90; - const keySize = domainConfig.dkim?.keySize || 2048; - if (!rotateKeys) { - logger.log('debug', `DKIM key rotation disabled for ${domain}`); - continue; - } - try { - // Check if keys need rotation - const needsRotation = await this.dkimCreator.needsRotation(domain, selector, rotationInterval); - if (needsRotation) { - logger.log('info', `DKIM keys need rotation for ${domain} (selector: ${selector})`); - // Rotate the keys - const newSelector = await this.dkimCreator.rotateDkimKeys(domain, selector, keySize); - // Update the domain config with new selector - domainConfig.dkim = { - ...domainConfig.dkim, - selector: newSelector - }; - // Re-register DNS handler for new selector if internal-dns mode - if (domainConfig.dnsMode === 'internal-dns' && this.dcRouter.dnsServer) { - // Get new public key - const keyPair = await this.dkimCreator.readDKIMKeysForSelector(domain, newSelector); - const publicKeyBase64 = keyPair.publicKey - .replace(/-----BEGIN PUBLIC KEY-----/g, '') - .replace(/-----END PUBLIC KEY-----/g, '') - .replace(/\s/g, ''); - const ttl = domainConfig.dns?.internal?.ttl || 3600; - // Register new selector - this.dcRouter.dnsServer.registerHandler(`${newSelector}._domainkey.${domain}`, ['TXT'], () => ({ - name: `${newSelector}._domainkey.${domain}`, - type: 'TXT', - class: 'IN', - ttl: ttl, - data: `v=DKIM1; k=rsa; p=${publicKeyBase64}` - })); - logger.log('info', `DKIM DNS handler registered for new selector: ${newSelector}._domainkey.${domain}`); - // Store the updated public key in storage - await this.dcRouter.storageManager.set(`/email/dkim/${domain}/public.key`, keyPair.publicKey); - } - // Clean up old keys after grace period (async, don't wait) - this.dkimCreator.cleanupOldKeys(domain, 30).catch(error => { - logger.log('warn', `Failed to cleanup old DKIM keys for ${domain}: ${error.message}`); - }); - } - else { - logger.log('debug', `DKIM keys for ${domain} are up to date`); - } - } - catch (error) { - logger.log('error', `Failed to check/rotate DKIM keys for ${domain}: ${error.message}`); - } - } - } - /** - * Generate SmartProxy routes for email ports - */ - generateProxyRoutes(portMapping) { - const routes = []; - const defaultPortMapping = { - 25: 10025, - 587: 10587, - 465: 10465 - }; - const actualPortMapping = portMapping || defaultPortMapping; - // Generate routes for each configured port - for (const externalPort of this.options.ports) { - const internalPort = actualPortMapping[externalPort] || externalPort + 10000; - let routeName = 'email-route'; - let tlsMode = 'passthrough'; - // Configure based on port - switch (externalPort) { - case 25: - routeName = 'smtp-route'; - tlsMode = 'passthrough'; // STARTTLS - break; - case 587: - routeName = 'submission-route'; - tlsMode = 'passthrough'; // STARTTLS - break; - case 465: - routeName = 'smtps-route'; - tlsMode = 'terminate'; // Implicit TLS - break; - default: - routeName = `email-port-${externalPort}-route`; - } - routes.push({ - name: routeName, - match: { - ports: [externalPort] - }, - action: { - type: 'forward', - target: { - host: 'localhost', - port: internalPort - }, - tls: { - mode: tlsMode - } - } - }); - } - return routes; - } - /** - * Update server configuration - */ - updateOptions(options) { - // Stop the server if changing ports - const portsChanged = options.ports && - (!this.options.ports || - JSON.stringify(options.ports) !== JSON.stringify(this.options.ports)); - if (portsChanged) { - this.stop().then(() => { - this.options = { ...this.options, ...options }; - this.start(); - }); - } - else { - // Update options without restart - this.options = { ...this.options, ...options }; - // Update domain registry if domains changed - if (options.domains) { - this.domainRegistry = new DomainRegistry(options.domains, options.defaults || this.options.defaults); - } - // Update email router if routes changed - if (options.routes) { - this.emailRouter.updateRoutes(options.routes); - } - } - } - /** - * Update email routes - */ - updateEmailRoutes(routes) { - this.options.routes = routes; - this.emailRouter.updateRoutes(routes); - } - /** - * Get server statistics - */ - getStats() { - return { ...this.stats }; - } - /** - * Get domain registry - */ - getDomainRegistry() { - return this.domainRegistry; - } - /** - * Update email routes dynamically - */ - updateRoutes(routes) { - this.emailRouter.setRoutes(routes); - logger.log('info', `Updated email routes with ${routes.length} routes`); - } - /** - * Send an email through the delivery system - * @param email The email to send - * @param mode The processing mode to use - * @param rule Optional rule to apply - * @param options Optional sending options - * @returns The ID of the queued email - */ - async sendEmail(email, mode = 'mta', route, options) { - logger.log('info', `Sending email: ${email.subject} to ${email.to.join(', ')}`); - try { - // Validate the email - if (!email.from) { - throw new Error('Email must have a sender address'); - } - if (!email.to || email.to.length === 0) { - throw new Error('Email must have at least one recipient'); - } - // Check if any recipients are on the suppression list (unless explicitly skipped) - if (!options?.skipSuppressionCheck) { - const suppressedRecipients = email.to.filter(recipient => this.isEmailSuppressed(recipient)); - if (suppressedRecipients.length > 0) { - // Filter out suppressed recipients - const originalCount = email.to.length; - const suppressed = suppressedRecipients.map(recipient => { - const info = this.getSuppressionInfo(recipient); - return { - email: recipient, - reason: info?.reason || 'Unknown', - until: info?.expiresAt ? new Date(info.expiresAt).toISOString() : 'permanent' - }; - }); - logger.log('warn', `Filtering out ${suppressedRecipients.length} suppressed recipient(s)`, { suppressed }); - // If all recipients are suppressed, throw an error - if (suppressedRecipients.length === originalCount) { - throw new Error('All recipients are on the suppression list'); - } - // Filter the recipients list to only include non-suppressed addresses - email.to = email.to.filter(recipient => !this.isEmailSuppressed(recipient)); - } - } - // IP warmup handling - let ipAddress = options?.ipAddress; - // If no specific IP was provided, use IP warmup manager to find the best IP - if (!ipAddress) { - const domain = email.from.split('@')[1]; - ipAddress = this.getBestIPForSending({ - from: email.from, - to: email.to, - domain, - isTransactional: options?.isTransactional - }); - if (ipAddress) { - logger.log('info', `Selected IP ${ipAddress} for sending based on warmup status`); - } - } - // If an IP is provided or selected by warmup manager, check its capacity - if (ipAddress) { - // Check if the IP can send more today - if (!this.canIPSendMoreToday(ipAddress)) { - logger.log('warn', `IP ${ipAddress} has reached its daily sending limit, email will be queued for later delivery`); - } - // Check if the IP can send more this hour - if (!this.canIPSendMoreThisHour(ipAddress)) { - logger.log('warn', `IP ${ipAddress} has reached its hourly sending limit, email will be queued for later delivery`); - } - // Record the send for IP warmup tracking - this.recordIPSend(ipAddress); - // Add IP header to the email - email.addHeader('X-Sending-IP', ipAddress); - } - // Check if the sender domain has DKIM keys and sign the email if needed - if (mode === 'mta' && route?.action.options?.mtaOptions?.dkimSign) { - const domain = email.from.split('@')[1]; - await this.handleDkimSigning(email, domain, route.action.options.mtaOptions.dkimOptions?.keySelector || 'mta'); - } - // Generate a unique ID for this email - const id = plugins.uuid.v4(); - // Queue the email for delivery - await this.deliveryQueue.enqueue(email, mode, route); - // Record 'sent' event for domain reputation monitoring - const senderDomain = email.from.split('@')[1]; - if (senderDomain) { - this.recordReputationEvent(senderDomain, { - type: 'sent', - count: email.to.length - }); - } - logger.log('info', `Email queued with ID: ${id}`); - return id; - } - catch (error) { - logger.log('error', `Failed to send email: ${error.message}`); - throw error; - } - } - /** - * Handle DKIM signing for an email - * @param email The email to sign - * @param domain The domain to sign with - * @param selector The DKIM selector - */ - async handleDkimSigning(email, domain, selector) { - try { - // Ensure we have DKIM keys for this domain - await this.dkimCreator.handleDKIMKeysForDomain(domain); - // Get the private key - const { privateKey } = await this.dkimCreator.readDKIMKeys(domain); - // Convert Email to raw format for signing - const rawEmail = email.toRFC822String(); - // Sign the email via Rust bridge - const signResult = await this.rustBridge.signDkim({ - rawMessage: rawEmail, - domain, - selector, - privateKey, - }); - if (signResult.header) { - email.addHeader('DKIM-Signature', signResult.header); - logger.log('info', `Successfully added DKIM signature for ${domain}`); - } - } - catch (error) { - logger.log('error', `Failed to sign email with DKIM: ${error.message}`); - // Continue without DKIM rather than failing the send - } - } - /** - * Process a bounce notification email - * @param bounceEmail The email containing bounce notification information - * @returns Processed bounce record or null if not a bounce - */ - async processBounceNotification(bounceEmail) { - logger.log('info', 'Processing potential bounce notification email'); - try { - // Process as a bounce notification (no conversion needed anymore) - const bounceRecord = await this.bounceManager.processBounceEmail(bounceEmail); - if (bounceRecord) { - logger.log('info', `Successfully processed bounce notification for ${bounceRecord.recipient}`, { - bounceType: bounceRecord.bounceType, - bounceCategory: bounceRecord.bounceCategory - }); - // Notify any registered listeners about the bounce - this.emit('bounceProcessed', bounceRecord); - // Record bounce event for domain reputation tracking - if (bounceRecord.domain) { - this.recordReputationEvent(bounceRecord.domain, { - type: 'bounce', - hardBounce: bounceRecord.bounceCategory === BounceCategory.HARD, - receivingDomain: bounceRecord.recipient.split('@')[1] - }); - } - // Log security event - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.INFO, - type: SecurityEventType.EMAIL_VALIDATION, - message: `Bounce notification processed for recipient`, - domain: bounceRecord.domain, - details: { - recipient: bounceRecord.recipient, - bounceType: bounceRecord.bounceType, - bounceCategory: bounceRecord.bounceCategory - }, - success: true - }); - return true; - } - else { - logger.log('info', 'Email not recognized as a bounce notification'); - return false; - } - } - catch (error) { - logger.log('error', `Error processing bounce notification: ${error.message}`); - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.ERROR, - type: SecurityEventType.EMAIL_VALIDATION, - message: 'Failed to process bounce notification', - details: { - error: error.message, - subject: bounceEmail.subject - }, - success: false - }); - return false; - } - } - /** - * Process an SMTP failure as a bounce - * @param recipient Recipient email that failed - * @param smtpResponse SMTP error response - * @param options Additional options for bounce processing - * @returns Processed bounce record - */ - async processSmtpFailure(recipient, smtpResponse, options = {}) { - logger.log('info', `Processing SMTP failure for ${recipient}: ${smtpResponse}`); - try { - // Process the SMTP failure through the bounce manager - const bounceRecord = await this.bounceManager.processSmtpFailure(recipient, smtpResponse, options); - logger.log('info', `Successfully processed SMTP failure for ${recipient} as ${bounceRecord.bounceCategory} bounce`, { - bounceType: bounceRecord.bounceType - }); - // Notify any registered listeners about the bounce - this.emit('bounceProcessed', bounceRecord); - // Record bounce event for domain reputation tracking - if (bounceRecord.domain) { - this.recordReputationEvent(bounceRecord.domain, { - type: 'bounce', - hardBounce: bounceRecord.bounceCategory === BounceCategory.HARD, - receivingDomain: bounceRecord.recipient.split('@')[1] - }); - } - // Log security event - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.INFO, - type: SecurityEventType.EMAIL_VALIDATION, - message: `SMTP failure processed for recipient`, - domain: bounceRecord.domain, - details: { - recipient: bounceRecord.recipient, - bounceType: bounceRecord.bounceType, - bounceCategory: bounceRecord.bounceCategory, - smtpResponse - }, - success: true - }); - return true; - } - catch (error) { - logger.log('error', `Error processing SMTP failure: ${error.message}`); - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.ERROR, - type: SecurityEventType.EMAIL_VALIDATION, - message: 'Failed to process SMTP failure', - details: { - recipient, - smtpResponse, - error: error.message - }, - success: false - }); - return false; - } - } - /** - * Check if an email address is suppressed (has bounced previously) - * @param email Email address to check - * @returns Whether the email is suppressed - */ - isEmailSuppressed(email) { - return this.bounceManager.isEmailSuppressed(email); - } - /** - * Get suppression information for an email - * @param email Email address to check - * @returns Suppression information or null if not suppressed - */ - getSuppressionInfo(email) { - return this.bounceManager.getSuppressionInfo(email); - } - /** - * Get bounce history information for an email - * @param email Email address to check - * @returns Bounce history or null if no bounces - */ - getBounceHistory(email) { - return this.bounceManager.getBounceInfo(email); - } - /** - * Get all suppressed email addresses - * @returns Array of suppressed email addresses - */ - getSuppressionList() { - return this.bounceManager.getSuppressionList(); - } - /** - * Get all hard bounced email addresses - * @returns Array of hard bounced email addresses - */ - getHardBouncedAddresses() { - return this.bounceManager.getHardBouncedAddresses(); - } - /** - * Add an email to the suppression list - * @param email Email address to suppress - * @param reason Reason for suppression - * @param expiresAt Optional expiration time (undefined for permanent) - */ - addToSuppressionList(email, reason, expiresAt) { - this.bounceManager.addToSuppressionList(email, reason, expiresAt); - logger.log('info', `Added ${email} to suppression list: ${reason}`); - } - /** - * Remove an email from the suppression list - * @param email Email address to remove from suppression - */ - removeFromSuppressionList(email) { - this.bounceManager.removeFromSuppressionList(email); - logger.log('info', `Removed ${email} from suppression list`); - } - /** - * Get the status of IP warmup process - * @param ipAddress Optional specific IP to check - * @returns Status of IP warmup - */ - getIPWarmupStatus(ipAddress) { - return this.ipWarmupManager.getWarmupStatus(ipAddress); - } - /** - * Add a new IP address to the warmup process - * @param ipAddress IP address to add - */ - addIPToWarmup(ipAddress) { - this.ipWarmupManager.addIPToWarmup(ipAddress); - } - /** - * Remove an IP address from the warmup process - * @param ipAddress IP address to remove - */ - removeIPFromWarmup(ipAddress) { - this.ipWarmupManager.removeIPFromWarmup(ipAddress); - } - /** - * Update metrics for an IP in the warmup process - * @param ipAddress IP address - * @param metrics Metrics to update - */ - updateIPWarmupMetrics(ipAddress, metrics) { - this.ipWarmupManager.updateMetrics(ipAddress, metrics); - } - /** - * Check if an IP can send more emails today - * @param ipAddress IP address to check - * @returns Whether the IP can send more today - */ - canIPSendMoreToday(ipAddress) { - return this.ipWarmupManager.canSendMoreToday(ipAddress); - } - /** - * Check if an IP can send more emails in the current hour - * @param ipAddress IP address to check - * @returns Whether the IP can send more this hour - */ - canIPSendMoreThisHour(ipAddress) { - return this.ipWarmupManager.canSendMoreThisHour(ipAddress); - } - /** - * Get the best IP to use for sending an email based on warmup status - * @param emailInfo Information about the email being sent - * @returns Best IP to use or null - */ - getBestIPForSending(emailInfo) { - return this.ipWarmupManager.getBestIPForSending(emailInfo); - } - /** - * Set the active IP allocation policy for warmup - * @param policyName Name of the policy to set - */ - setIPAllocationPolicy(policyName) { - this.ipWarmupManager.setActiveAllocationPolicy(policyName); - } - /** - * Record that an email was sent using a specific IP - * @param ipAddress IP address used for sending - */ - recordIPSend(ipAddress) { - this.ipWarmupManager.recordSend(ipAddress); - } - /** - * Get reputation data for a domain - * @param domain Domain to get reputation for - * @returns Domain reputation metrics - */ - getDomainReputationData(domain) { - return this.senderReputationMonitor.getReputationData(domain); - } - /** - * Get summary reputation data for all monitored domains - * @returns Summary data for all domains - */ - getReputationSummary() { - return this.senderReputationMonitor.getReputationSummary(); - } - /** - * Add a domain to the reputation monitoring system - * @param domain Domain to add - */ - addDomainToMonitoring(domain) { - this.senderReputationMonitor.addDomain(domain); - } - /** - * Remove a domain from the reputation monitoring system - * @param domain Domain to remove - */ - removeDomainFromMonitoring(domain) { - this.senderReputationMonitor.removeDomain(domain); - } - /** - * Record an email event for domain reputation tracking - * @param domain Domain sending the email - * @param event Event details - */ - recordReputationEvent(domain, event) { - this.senderReputationMonitor.recordSendEvent(domain, event); - } - /** - * Check if DKIM key exists for a domain - * @param domain Domain to check - */ - hasDkimKey(domain) { - return this.dkimKeys.has(domain); - } - /** - * Record successful email delivery - * @param domain Sending domain - */ - recordDelivery(domain) { - this.recordReputationEvent(domain, { - type: 'delivered', - count: 1 - }); - } - /** - * Record email bounce - * @param domain Sending domain - * @param receivingDomain Receiving domain that bounced - * @param bounceType Type of bounce (hard/soft) - * @param reason Bounce reason - */ - recordBounce(domain, receivingDomain, bounceType, reason) { - // Record bounce in bounce manager - const bounceRecord = { - id: `bounce_${Date.now()}_${Math.random().toString(36).substring(2, 9)}`, - recipient: `user@${receivingDomain}`, - sender: `user@${domain}`, - domain: domain, - bounceType: bounceType === 'hard' ? BounceType.INVALID_RECIPIENT : BounceType.TEMPORARY_FAILURE, - bounceCategory: bounceType === 'hard' ? BounceCategory.HARD : BounceCategory.SOFT, - timestamp: Date.now(), - smtpResponse: reason, - diagnosticCode: reason, - statusCode: bounceType === 'hard' ? '550' : '450', - processed: false - }; - // Process the bounce - this.bounceManager.processBounce(bounceRecord); - // Record reputation event - this.recordReputationEvent(domain, { - type: 'bounce', - count: 1, - hardBounce: bounceType === 'hard', - receivingDomain - }); - } - /** - * Get the rate limiter instance - * @returns The unified rate limiter - */ - getRateLimiter() { - return this.rateLimiter; - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy51bmlmaWVkLmVtYWlsLnNlcnZlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3RzL21haWwvcm91dGluZy9jbGFzc2VzLnVuaWZpZWQuZW1haWwuc2VydmVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sS0FBSyxPQUFPLE1BQU0sa0JBQWtCLENBQUM7QUFDNUMsT0FBTyxLQUFLLEtBQUssTUFBTSxnQkFBZ0IsQ0FBQztBQUN4QyxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sUUFBUSxDQUFDO0FBQ3RDLE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUN6QyxPQUFPLEVBQ0wsY0FBYyxFQUNkLGdCQUFnQixFQUNoQixpQkFBaUIsRUFDbEIsTUFBTSx5QkFBeUIsQ0FBQztBQUNqQyxPQUFPLEVBQUUsV0FBVyxFQUFFLE1BQU0sb0NBQW9DLENBQUM7QUFDakUsT0FBTyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sK0NBQStDLENBQUM7QUFDcEYsT0FBTyxFQUFFLGtCQUFrQixFQUFFLE1BQU0sOENBQThDLENBQUM7QUErQmxGLE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUV4RCxPQUFPLEVBQUUsS0FBSyxFQUFFLE1BQU0sMEJBQTBCLENBQUM7QUFDakQsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLDhCQUE4QixDQUFDO0FBQzlELE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQztBQUN0RCxPQUFPLEVBQUUsYUFBYSxFQUFFLFVBQVUsRUFBRSxjQUFjLEVBQUUsTUFBTSxrQ0FBa0MsQ0FBQztBQUM3RixPQUFPLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSx5Q0FBeUMsQ0FBQztBQUVqRixPQUFPLEVBQUUsdUJBQXVCLEVBQWtDLE1BQU0sd0NBQXdDLENBQUM7QUFDakgsT0FBTyxFQUFFLG9CQUFvQixFQUFzQixNQUFNLHVDQUF1QyxDQUFDO0FB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 \ No newline at end of file diff --git a/dist_ts/mail/routing/index.d.ts b/dist_ts/mail/routing/index.d.ts deleted file mode 100644 index e2c5ee7..0000000 --- a/dist_ts/mail/routing/index.d.ts +++ /dev/null @@ -1,5 +0,0 @@ -export * from './classes.email.router.js'; -export * from './classes.unified.email.server.js'; -export * from './classes.dns.manager.js'; -export * from './interfaces.js'; -export * from './classes.domain.registry.js'; diff --git a/dist_ts/mail/routing/index.js b/dist_ts/mail/routing/index.js deleted file mode 100644 index 8d14b03..0000000 --- a/dist_ts/mail/routing/index.js +++ /dev/null @@ -1,7 +0,0 @@ -// Email routing components -export * from './classes.email.router.js'; -export * from './classes.unified.email.server.js'; -export * from './classes.dns.manager.js'; -export * from './interfaces.js'; -export * from './classes.domain.registry.js'; -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi90cy9tYWlsL3JvdXRpbmcvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsMkJBQTJCO0FBQzNCLGNBQWMsMkJBQTJCLENBQUM7QUFDMUMsY0FBYyxtQ0FBbUMsQ0FBQztBQUNsRCxjQUFjLDBCQUEwQixDQUFDO0FBQ3pDLGNBQWMsaUJBQWlCLENBQUM7QUFDaEMsY0FBYyw4QkFBOEIsQ0FBQyJ9 \ No newline at end of file diff --git a/dist_ts/mail/routing/interfaces.d.ts b/dist_ts/mail/routing/interfaces.d.ts deleted file mode 100644 index fa054c2..0000000 --- a/dist_ts/mail/routing/interfaces.d.ts +++ /dev/null @@ -1,187 +0,0 @@ -import type { Email } from '../core/classes.email.js'; -import type { IExtendedSmtpSession } from './classes.unified.email.server.js'; -/** - * Route configuration for email routing - */ -export interface IEmailRoute { - /** Route identifier */ - name: string; - /** Order of evaluation (higher priority evaluated first, default: 0) */ - priority?: number; - /** Conditions to match */ - match: IEmailMatch; - /** Action to take when matched */ - action: IEmailAction; -} -/** - * Match criteria for email routing - */ -export interface IEmailMatch { - /** Email patterns to match recipients: "*@example.com", "admin@*" */ - recipients?: string | string[]; - /** Email patterns to match senders */ - senders?: string | string[]; - /** IP addresses or CIDR ranges to match */ - clientIp?: string | string[]; - /** Require authentication status */ - authenticated?: boolean; - /** Headers to match */ - headers?: Record; - /** Message size range */ - sizeRange?: { - min?: number; - max?: number; - }; - /** Subject line patterns */ - subject?: string | RegExp; - /** Has attachments */ - hasAttachments?: boolean; -} -/** - * Action to take when route matches - */ -export interface IEmailAction { - /** Type of action to perform */ - type: 'forward' | 'deliver' | 'reject' | 'process'; - /** Forward action configuration */ - forward?: { - /** Target host to forward to */ - host: string; - /** Target port (default: 25) */ - port?: number; - /** Authentication credentials */ - auth?: { - user: string; - pass: string; - }; - /** Preserve original headers */ - preserveHeaders?: boolean; - /** Additional headers to add */ - addHeaders?: Record; - }; - /** Reject action configuration */ - reject?: { - /** SMTP response code */ - code: number; - /** SMTP response message */ - message: string; - }; - /** Process action configuration */ - process?: { - /** Enable content scanning */ - scan?: boolean; - /** Enable DKIM signing */ - dkim?: boolean; - /** Delivery queue priority */ - queue?: 'normal' | 'priority' | 'bulk'; - }; - /** Options for various action types */ - options?: { - /** MTA specific options */ - mtaOptions?: { - domain?: string; - allowLocalDelivery?: boolean; - localDeliveryPath?: string; - dkimSign?: boolean; - dkimOptions?: { - domainName: string; - keySelector: string; - privateKey?: string; - }; - smtpBanner?: string; - maxConnections?: number; - connTimeout?: number; - spoolDir?: string; - }; - /** Content scanning configuration */ - contentScanning?: boolean; - scanners?: Array<{ - type: 'spam' | 'virus' | 'attachment'; - threshold?: number; - action: 'tag' | 'reject'; - blockedExtensions?: string[]; - }>; - /** Email transformations */ - transformations?: Array<{ - type: string; - header?: string; - value?: string; - domains?: string[]; - append?: boolean; - [key: string]: any; - }>; - }; - /** Delivery options (applies to forward/process/deliver) */ - delivery?: { - /** Rate limit (messages per minute) */ - rateLimit?: number; - /** Number of retry attempts */ - retries?: number; - }; -} -/** - * Context for route evaluation - */ -export interface IEmailContext { - /** The email being routed */ - email: Email; - /** The SMTP session */ - session: IExtendedSmtpSession; -} -/** - * Email domain configuration - */ -export interface IEmailDomainConfig { - /** Domain name */ - domain: string; - /** DNS handling mode */ - dnsMode: 'forward' | 'internal-dns' | 'external-dns'; - /** DNS configuration based on mode */ - dns?: { - /** For 'forward' mode */ - forward?: { - /** Skip DNS validation (default: false) */ - skipDnsValidation?: boolean; - /** Target server's expected domain */ - targetDomain?: string; - }; - /** For 'internal-dns' mode */ - internal?: { - /** TTL for DNS records in seconds (default: 3600) */ - ttl?: number; - /** MX record priority (default: 10) */ - mxPriority?: number; - }; - /** For 'external-dns' mode */ - external?: { - /** Custom DNS servers (default: system DNS) */ - servers?: string[]; - /** Which records to validate (default: ['MX', 'SPF', 'DKIM', 'DMARC']) */ - requiredRecords?: ('MX' | 'SPF' | 'DKIM' | 'DMARC')[]; - }; - }; - /** Per-domain DKIM settings (DKIM always enabled) */ - dkim?: { - /** DKIM selector (default: 'default') */ - selector?: string; - /** Key size in bits (default: 2048) */ - keySize?: number; - /** Automatically rotate keys (default: false) */ - rotateKeys?: boolean; - /** Days between key rotations (default: 90) */ - rotationInterval?: number; - }; - /** Per-domain rate limits */ - rateLimits?: { - outbound?: { - messagesPerMinute?: number; - messagesPerHour?: number; - messagesPerDay?: number; - }; - inbound?: { - messagesPerMinute?: number; - connectionsPerIp?: number; - recipientsPerMessage?: number; - }; - }; -} diff --git a/dist_ts/mail/routing/interfaces.js b/dist_ts/mail/routing/interfaces.js deleted file mode 100644 index 4f5847f..0000000 --- a/dist_ts/mail/routing/interfaces.js +++ /dev/null @@ -1,2 +0,0 @@ -export {}; -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW50ZXJmYWNlcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3RzL21haWwvcm91dGluZy9pbnRlcmZhY2VzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIifQ== \ No newline at end of file diff --git a/dist_ts/mail/security/classes.dkimcreator.d.ts b/dist_ts/mail/security/classes.dkimcreator.d.ts deleted file mode 100644 index 2516ec8..0000000 --- a/dist_ts/mail/security/classes.dkimcreator.d.ts +++ /dev/null @@ -1,68 +0,0 @@ -import * as plugins from '../../plugins.js'; -import { Email } from '../core/classes.email.js'; -export interface IKeyPaths { - privateKeyPath: string; - publicKeyPath: string; -} -export interface IDkimKeyMetadata { - domain: string; - selector: string; - createdAt: number; - rotatedAt?: number; - previousSelector?: string; - keySize: number; -} -export declare class DKIMCreator { - private keysDir; - private storageManager?; - constructor(keysDir?: string, storageManager?: any); - getKeyPathsForDomain(domainArg: string): Promise; - handleDKIMKeysForDomain(domainArg: string): Promise; - handleDKIMKeysForEmail(email: Email): Promise; - readDKIMKeys(domainArg: string): Promise<{ - privateKey: string; - publicKey: string; - }>; - createDKIMKeys(): Promise<{ - privateKey: string; - publicKey: string; - }>; - storeDKIMKeys(privateKey: string, publicKey: string, privateKeyPath: string, publicKeyPath: string): Promise; - createAndStoreDKIMKeys(domain: string): Promise; - getDNSRecordForDomain(domainArg: string): Promise; - /** - * Get DKIM key metadata for a domain - */ - private getKeyMetadata; - /** - * Save DKIM key metadata - */ - private saveKeyMetadata; - /** - * Check if DKIM keys need rotation - */ - needsRotation(domain: string, selector?: string, rotationIntervalDays?: number): Promise; - /** - * Rotate DKIM keys for a domain - */ - rotateDkimKeys(domain: string, currentSelector?: string, keySize?: number): Promise; - /** - * Get key paths for a specific selector - */ - getKeyPathsForSelector(domain: string, selector: string): Promise; - /** - * Read DKIM keys for a specific selector - */ - readDKIMKeysForSelector(domain: string, selector: string): Promise<{ - privateKey: string; - publicKey: string; - }>; - /** - * Get DNS record for a specific selector - */ - getDNSRecordForSelector(domain: string, selector: string): Promise; - /** - * Clean up old DKIM keys after grace period - */ - cleanupOldKeys(domain: string, gracePeriodDays?: number): Promise; -} diff --git a/dist_ts/mail/security/classes.dkimcreator.js b/dist_ts/mail/security/classes.dkimcreator.js deleted file mode 100644 index afce4b3..0000000 --- a/dist_ts/mail/security/classes.dkimcreator.js +++ /dev/null @@ -1,348 +0,0 @@ -import * as plugins from '../../plugins.js'; -import * as paths from '../../paths.js'; -import { Email } from '../core/classes.email.js'; -// MtaService reference removed -const readFile = plugins.util.promisify(plugins.fs.readFile); -const writeFile = plugins.util.promisify(plugins.fs.writeFile); -const generateKeyPair = plugins.util.promisify(plugins.crypto.generateKeyPair); -export class DKIMCreator { - keysDir; - storageManager; // StorageManager instance - constructor(keysDir = paths.keysDir, storageManager) { - this.keysDir = keysDir; - this.storageManager = storageManager; - } - async getKeyPathsForDomain(domainArg) { - return { - privateKeyPath: plugins.path.join(this.keysDir, `${domainArg}-private.pem`), - publicKeyPath: plugins.path.join(this.keysDir, `${domainArg}-public.pem`), - }; - } - // Check if a DKIM key is present and creates one and stores it to disk otherwise - async handleDKIMKeysForDomain(domainArg) { - try { - await this.readDKIMKeys(domainArg); - } - catch (error) { - console.log(`No DKIM keys found for ${domainArg}. Generating...`); - await this.createAndStoreDKIMKeys(domainArg); - const dnsValue = await this.getDNSRecordForDomain(domainArg); - await plugins.smartfs.directory(paths.dnsRecordsDir).recursive().create(); - await plugins.smartfs.file(plugins.path.join(paths.dnsRecordsDir, `${domainArg}.dkimrecord.json`)).write(JSON.stringify(dnsValue, null, 2)); - } - } - async handleDKIMKeysForEmail(email) { - const domain = email.from.split('@')[1]; - await this.handleDKIMKeysForDomain(domain); - } - // Read DKIM keys - always use storage manager, migrate from filesystem if needed - async readDKIMKeys(domainArg) { - // Try to read from storage manager first - if (this.storageManager) { - try { - const [privateKey, publicKey] = await Promise.all([ - this.storageManager.get(`/email/dkim/${domainArg}/private.key`), - this.storageManager.get(`/email/dkim/${domainArg}/public.key`) - ]); - if (privateKey && publicKey) { - return { privateKey, publicKey }; - } - } - catch (error) { - // Fall through to migration check - } - // Check if keys exist in filesystem and migrate them to storage manager - const keyPaths = await this.getKeyPathsForDomain(domainArg); - try { - const [privateKeyBuffer, publicKeyBuffer] = await Promise.all([ - readFile(keyPaths.privateKeyPath), - readFile(keyPaths.publicKeyPath), - ]); - // Convert the buffers to strings - const privateKey = privateKeyBuffer.toString(); - const publicKey = publicKeyBuffer.toString(); - // Migrate to storage manager - console.log(`Migrating DKIM keys for ${domainArg} from filesystem to StorageManager`); - await Promise.all([ - this.storageManager.set(`/email/dkim/${domainArg}/private.key`, privateKey), - this.storageManager.set(`/email/dkim/${domainArg}/public.key`, publicKey) - ]); - return { privateKey, publicKey }; - } - catch (error) { - if (error.code === 'ENOENT') { - // Keys don't exist anywhere - throw new Error(`DKIM keys not found for domain ${domainArg}`); - } - throw error; - } - } - else { - // No storage manager, use filesystem directly - const keyPaths = await this.getKeyPathsForDomain(domainArg); - const [privateKeyBuffer, publicKeyBuffer] = await Promise.all([ - readFile(keyPaths.privateKeyPath), - readFile(keyPaths.publicKeyPath), - ]); - const privateKey = privateKeyBuffer.toString(); - const publicKey = publicKeyBuffer.toString(); - return { privateKey, publicKey }; - } - } - // Create a DKIM key pair - changed to public for API access - async createDKIMKeys() { - const { privateKey, publicKey } = await generateKeyPair('rsa', { - modulusLength: 2048, - publicKeyEncoding: { type: 'spki', format: 'pem' }, - privateKeyEncoding: { type: 'pkcs1', format: 'pem' }, - }); - return { privateKey, publicKey }; - } - // Store a DKIM key pair - uses storage manager if available, else disk - async storeDKIMKeys(privateKey, publicKey, privateKeyPath, publicKeyPath) { - // Store in storage manager if available - if (this.storageManager) { - // Extract domain from path (e.g., /path/to/keys/example.com-private.pem -> example.com) - const match = privateKeyPath.match(/\/([^\/]+)-private\.pem$/); - if (match) { - const domain = match[1]; - await Promise.all([ - this.storageManager.set(`/email/dkim/${domain}/private.key`, privateKey), - this.storageManager.set(`/email/dkim/${domain}/public.key`, publicKey) - ]); - } - } - // Also store to filesystem for backward compatibility - await Promise.all([writeFile(privateKeyPath, privateKey), writeFile(publicKeyPath, publicKey)]); - } - // Create a DKIM key pair and store it to disk - changed to public for API access - async createAndStoreDKIMKeys(domain) { - const { privateKey, publicKey } = await this.createDKIMKeys(); - const keyPaths = await this.getKeyPathsForDomain(domain); - await this.storeDKIMKeys(privateKey, publicKey, keyPaths.privateKeyPath, keyPaths.publicKeyPath); - console.log(`DKIM keys for ${domain} created and stored.`); - } - // Changed to public for API access - async getDNSRecordForDomain(domainArg) { - await this.handleDKIMKeysForDomain(domainArg); - const keys = await this.readDKIMKeys(domainArg); - // Remove the PEM header and footer and newlines - const pemHeader = '-----BEGIN PUBLIC KEY-----'; - const pemFooter = '-----END PUBLIC KEY-----'; - const keyContents = keys.publicKey - .replace(pemHeader, '') - .replace(pemFooter, '') - .replace(/\n/g, ''); - // Now generate the DKIM DNS TXT record - const dnsRecordValue = `v=DKIM1; h=sha256; k=rsa; p=${keyContents}`; - return { - name: `mta._domainkey.${domainArg}`, - type: 'TXT', - dnsSecEnabled: null, - value: dnsRecordValue, - }; - } - /** - * Get DKIM key metadata for a domain - */ - async getKeyMetadata(domain, selector = 'default') { - if (!this.storageManager) { - return null; - } - const metadataKey = `/email/dkim/${domain}/${selector}/metadata`; - const metadataStr = await this.storageManager.get(metadataKey); - if (!metadataStr) { - return null; - } - return JSON.parse(metadataStr); - } - /** - * Save DKIM key metadata - */ - async saveKeyMetadata(metadata) { - if (!this.storageManager) { - return; - } - const metadataKey = `/email/dkim/${metadata.domain}/${metadata.selector}/metadata`; - await this.storageManager.set(metadataKey, JSON.stringify(metadata)); - } - /** - * Check if DKIM keys need rotation - */ - async needsRotation(domain, selector = 'default', rotationIntervalDays = 90) { - const metadata = await this.getKeyMetadata(domain, selector); - if (!metadata) { - // No metadata means old keys, should rotate - return true; - } - const now = Date.now(); - const keyAgeMs = now - metadata.createdAt; - const keyAgeDays = keyAgeMs / (1000 * 60 * 60 * 24); - return keyAgeDays >= rotationIntervalDays; - } - /** - * Rotate DKIM keys for a domain - */ - async rotateDkimKeys(domain, currentSelector = 'default', keySize = 2048) { - console.log(`Rotating DKIM keys for ${domain}...`); - // Generate new selector based on date - const now = new Date(); - const newSelector = `key${now.getFullYear()}${String(now.getMonth() + 1).padStart(2, '0')}`; - // Create new keys with custom key size - const { privateKey, publicKey } = await generateKeyPair('rsa', { - modulusLength: keySize, - publicKeyEncoding: { type: 'spki', format: 'pem' }, - privateKeyEncoding: { type: 'pkcs1', format: 'pem' }, - }); - // Store new keys with new selector - const newKeyPaths = await this.getKeyPathsForSelector(domain, newSelector); - // Store in storage manager if available - if (this.storageManager) { - await Promise.all([ - this.storageManager.set(`/email/dkim/${domain}/${newSelector}/private.key`, privateKey), - this.storageManager.set(`/email/dkim/${domain}/${newSelector}/public.key`, publicKey) - ]); - } - // Also store to filesystem - await this.storeDKIMKeys(privateKey, publicKey, newKeyPaths.privateKeyPath, newKeyPaths.publicKeyPath); - // Save metadata for new keys - const metadata = { - domain, - selector: newSelector, - createdAt: Date.now(), - previousSelector: currentSelector, - keySize - }; - await this.saveKeyMetadata(metadata); - // Update metadata for old keys - const oldMetadata = await this.getKeyMetadata(domain, currentSelector); - if (oldMetadata) { - oldMetadata.rotatedAt = Date.now(); - await this.saveKeyMetadata(oldMetadata); - } - console.log(`DKIM keys rotated for ${domain}. New selector: ${newSelector}`); - return newSelector; - } - /** - * Get key paths for a specific selector - */ - async getKeyPathsForSelector(domain, selector) { - return { - privateKeyPath: plugins.path.join(this.keysDir, `${domain}-${selector}-private.pem`), - publicKeyPath: plugins.path.join(this.keysDir, `${domain}-${selector}-public.pem`), - }; - } - /** - * Read DKIM keys for a specific selector - */ - async readDKIMKeysForSelector(domain, selector) { - // Try to read from storage manager first - if (this.storageManager) { - try { - const [privateKey, publicKey] = await Promise.all([ - this.storageManager.get(`/email/dkim/${domain}/${selector}/private.key`), - this.storageManager.get(`/email/dkim/${domain}/${selector}/public.key`) - ]); - if (privateKey && publicKey) { - return { privateKey, publicKey }; - } - } - catch (error) { - // Fall through to migration check - } - // Check if keys exist in filesystem and migrate them to storage manager - const keyPaths = await this.getKeyPathsForSelector(domain, selector); - try { - const [privateKeyBuffer, publicKeyBuffer] = await Promise.all([ - readFile(keyPaths.privateKeyPath), - readFile(keyPaths.publicKeyPath), - ]); - const privateKey = privateKeyBuffer.toString(); - const publicKey = publicKeyBuffer.toString(); - // Migrate to storage manager - console.log(`Migrating DKIM keys for ${domain}/${selector} from filesystem to StorageManager`); - await Promise.all([ - this.storageManager.set(`/email/dkim/${domain}/${selector}/private.key`, privateKey), - this.storageManager.set(`/email/dkim/${domain}/${selector}/public.key`, publicKey) - ]); - return { privateKey, publicKey }; - } - catch (error) { - if (error.code === 'ENOENT') { - throw new Error(`DKIM keys not found for domain ${domain} with selector ${selector}`); - } - throw error; - } - } - else { - // No storage manager, use filesystem directly - const keyPaths = await this.getKeyPathsForSelector(domain, selector); - const [privateKeyBuffer, publicKeyBuffer] = await Promise.all([ - readFile(keyPaths.privateKeyPath), - readFile(keyPaths.publicKeyPath), - ]); - const privateKey = privateKeyBuffer.toString(); - const publicKey = publicKeyBuffer.toString(); - return { privateKey, publicKey }; - } - } - /** - * Get DNS record for a specific selector - */ - async getDNSRecordForSelector(domain, selector) { - const keys = await this.readDKIMKeysForSelector(domain, selector); - // Remove the PEM header and footer and newlines - const pemHeader = '-----BEGIN PUBLIC KEY-----'; - const pemFooter = '-----END PUBLIC KEY-----'; - const keyContents = keys.publicKey - .replace(pemHeader, '') - .replace(pemFooter, '') - .replace(/\n/g, ''); - // Generate the DKIM DNS TXT record - const dnsRecordValue = `v=DKIM1; h=sha256; k=rsa; p=${keyContents}`; - return { - name: `${selector}._domainkey.${domain}`, - type: 'TXT', - dnsSecEnabled: null, - value: dnsRecordValue, - }; - } - /** - * Clean up old DKIM keys after grace period - */ - async cleanupOldKeys(domain, gracePeriodDays = 30) { - if (!this.storageManager) { - return; - } - // List all selectors for the domain - const metadataKeys = await this.storageManager.list(`/email/dkim/${domain}/`); - for (const key of metadataKeys) { - if (key.endsWith('/metadata')) { - const metadataStr = await this.storageManager.get(key); - if (metadataStr) { - const metadata = JSON.parse(metadataStr); - // Check if key is rotated and past grace period - if (metadata.rotatedAt) { - const gracePeriodMs = gracePeriodDays * 24 * 60 * 60 * 1000; - const now = Date.now(); - if (now - metadata.rotatedAt > gracePeriodMs) { - console.log(`Cleaning up old DKIM keys for ${domain} selector ${metadata.selector}`); - // Delete key files - const keyPaths = await this.getKeyPathsForSelector(domain, metadata.selector); - try { - await plugins.fs.promises.unlink(keyPaths.privateKeyPath); - await plugins.fs.promises.unlink(keyPaths.publicKeyPath); - } - catch (error) { - console.warn(`Failed to delete old key files: ${error.message}`); - } - // Delete metadata - await this.storageManager.delete(key); - } - } - } - } - } - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5ka2ltY3JlYXRvci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3RzL21haWwvc2VjdXJpdHkvY2xhc3Nlcy5ka2ltY3JlYXRvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssT0FBTyxNQUFNLGtCQUFrQixDQUFDO0FBQzVDLE9BQU8sS0FBSyxLQUFLLE1BQU0sZ0JBQWdCLENBQUM7QUFFeEMsT0FBTyxFQUFFLEtBQUssRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBQ2pELCtCQUErQjtBQUUvQixNQUFNLFFBQVEsR0FBRyxPQUFPLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxPQUFPLENBQUMsRUFBRSxDQUFDLFFBQVEsQ0FBQyxDQUFDO0FBQzdELE1BQU0sU0FBUyxHQUFHLE9BQU8sQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUMsU0FBUyxDQUFDLENBQUM7QUFDL0QsTUFBTSxlQUFlLEdBQUcsT0FBTyxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxlQUFlLENBQUMsQ0FBQztBQWdCL0UsTUFBTSxPQUFPLFdBQVc7SUFDZCxPQUFPLENBQVM7SUFDaEIsY0FBYyxDQUFPLENBQUMsMEJBQTBCO0lBRXhELFlBQVksT0FBTyxHQUFHLEtBQUssQ0FBQyxPQUFPLEVBQUUsY0FBb0I7UUFDdkQsSUFBSSxDQUFDLE9BQU8sR0FBRyxPQUFPLENBQUM7UUFDdkIsSUFBSSxDQUFDLGNBQWMsR0FBRyxjQUFjLENBQUM7SUFDdkMsQ0FBQztJQUVNLEtBQUssQ0FBQyxvQkFBb0IsQ0FBQyxTQUFpQjtRQUNqRCxPQUFPO1lBQ0wsY0FBYyxFQUFFLE9BQU8sQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxPQUFPLEVBQUUsR0FBRyxTQUFTLGNBQWMsQ0FBQztZQUMzRSxhQUFhLEVBQUUsT0FBTyxDQUFDLElBQUksQ0FBQyxJQUFJ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 \ No newline at end of file diff --git a/dist_ts/mail/security/classes.dkimverifier.d.ts b/dist_ts/mail/security/classes.dkimverifier.d.ts deleted file mode 100644 index 743b186..0000000 --- a/dist_ts/mail/security/classes.dkimverifier.d.ts +++ /dev/null @@ -1,29 +0,0 @@ -/** - * Result of a DKIM verification - */ -export interface IDkimVerificationResult { - isValid: boolean; - domain?: string; - selector?: string; - status?: string; - details?: any; - errorMessage?: string; - signatureFields?: Record; -} -/** - * DKIM verifier — delegates to the Rust security bridge. - */ -export declare class DKIMVerifier { - constructor(); - /** - * Verify DKIM signature for an email via Rust bridge - */ - verify(emailData: string, options?: { - useCache?: boolean; - returnDetails?: boolean; - }): Promise; - /** No-op — Rust bridge handles its own caching */ - clearCache(): void; - /** Always 0 — cache is managed by the Rust side */ - getCacheSize(): number; -} diff --git a/dist_ts/mail/security/classes.dkimverifier.js b/dist_ts/mail/security/classes.dkimverifier.js deleted file mode 100644 index d2638db..0000000 --- a/dist_ts/mail/security/classes.dkimverifier.js +++ /dev/null @@ -1,58 +0,0 @@ -import { logger } from '../../logger.js'; -import { SecurityLogger, SecurityLogLevel, SecurityEventType } from '../../security/index.js'; -import { RustSecurityBridge } from '../../security/classes.rustsecuritybridge.js'; -/** - * DKIM verifier — delegates to the Rust security bridge. - */ -export class DKIMVerifier { - constructor() { } - /** - * Verify DKIM signature for an email via Rust bridge - */ - async verify(emailData, options = {}) { - try { - const bridge = RustSecurityBridge.getInstance(); - const results = await bridge.verifyDkim(emailData); - const first = results[0]; - const result = { - isValid: first?.is_valid ?? false, - domain: first?.domain ?? undefined, - selector: first?.selector ?? undefined, - status: first?.status ?? 'none', - details: options.returnDetails ? results : undefined, - }; - SecurityLogger.getInstance().logEvent({ - level: result.isValid ? SecurityLogLevel.INFO : SecurityLogLevel.WARN, - type: SecurityEventType.DKIM, - message: `DKIM verification ${result.isValid ? 'passed' : 'failed'} for domain ${result.domain || 'unknown'}`, - details: { selector: result.selector, status: result.status }, - domain: result.domain || 'unknown', - success: result.isValid - }); - logger.log(result.isValid ? 'info' : 'warn', `DKIM verification: ${result.status} for domain ${result.domain || 'unknown'}`); - return result; - } - catch (error) { - logger.log('error', `DKIM verification failed: ${error.message}`); - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.ERROR, - type: SecurityEventType.DKIM, - message: `DKIM verification error`, - details: { error: error.message }, - success: false - }); - return { - isValid: false, - status: 'temperror', - errorMessage: `Verification error: ${error.message}` - }; - } - } - /** No-op — Rust bridge handles its own caching */ - clearCache() { } - /** Always 0 — cache is managed by the Rust side */ - getCacheSize() { - return 0; - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5ka2ltdmVyaWZpZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi90cy9tYWlsL3NlY3VyaXR5L2NsYXNzZXMuZGtpbXZlcmlmaWVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUN6QyxPQUFPLEVBQUUsY0FBYyxFQUFFLGdCQUFnQixFQUFFLGlCQUFpQixFQUFFLE1BQU0seUJBQXlCLENBQUM7QUFDOUYsT0FBTyxFQUFFLGtCQUFrQixFQUFFLE1BQU0sOENBQThDLENBQUM7QUFlbEY7O0dBRUc7QUFDSCxNQUFNLE9BQU8sWUFBWTtJQUN2QixnQkFBZSxDQUFDO0lBRWhCOztPQUVHO0lBQ0ksS0FBSyxDQUFDLE1BQU0sQ0FDakIsU0FBaUIsRUFDakIsVUFHSSxFQUFFO1FBRU4sSUFBSSxDQUFDO1lBQ0gsTUFBTSxNQUFNLEdBQUcsa0JBQWtCLENBQUMsV0FBVyxFQUFFLENBQUM7WUFDaEQsTUFBTSxPQUFPLEdBQUcsTUFBTSxNQUFNLENBQUMsVUFBVSxDQUFDLFNBQVMsQ0FBQyxDQUFDO1lBQ25ELE1BQU0sS0FBSyxHQUFHLE9BQU8sQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUV6QixNQUFNLE1BQU0sR0FBNEI7Z0JBQ3RDLE9BQU8sRUFBRSxLQUFLLEVBQUUsUUFBUSxJQUFJLEtBQUs7Z0JBQ2pDLE1BQU0sRUFBRSxLQUFLLEVBQUUsTUFBTSxJQUFJLFNBQVM7Z0JBQ2xDLFFBQVEsRUFBRSxLQUFLLEVBQUUsUUFBUSxJQUFJLFNBQVM7Z0JBQ3RDLE1BQU0sRUFBRSxLQUFLLEVBQUUsTUFBTSxJQUFJLE1BQU07Z0JBQy9CLE9BQU8sRUFBRSxPQUFPLENBQUMsYUFBYSxDQUFDLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLFNBQVM7YUFDckQsQ0FBQztZQUVGLGNBQWMsQ0FBQyxXQUFXLEVBQUUsQ0FBQyxRQUFRLENBQUM7Z0JBQ3BDLEtBQUssRUFBRSxNQUFNLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLGdCQUFnQixDQUFDLElBQUk7Z0JBQ3JFLElBQUksRUFBRSxpQkFBaUIsQ0FBQyxJQUFJO2dCQUM1QixPQUFPLEVBQUUscUJBQXFCLE1BQU0sQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLFFBQVEsQ0FBQyxDQUFDLENBQUMsUUFBUSxlQUFlLE1BQU0sQ0FBQyxNQUFNLElBQUksU0FBUyxFQUFFO2dCQUM3RyxPQUFPLEVBQUUsRUFBRSxRQUFRLEVBQUUsTUFBTSxDQUFDLFFBQVEsRUFBRSxNQUFNLEVBQUUsTUFBTSxDQUFDLE1BQU0sRUFBRTtnQkFDN0QsTUFBTSxFQUFFLE1BQU0sQ0FBQyxNQUFNLElBQUksU0FBUztnQkFDbEMsT0FBTyxFQUFFLE1BQU0sQ0FBQyxPQUFPO2FBQ3hCLENBQUMsQ0FBQztZQUVILE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxNQUFNLEVBQ3pDLHNCQUFzQixNQUFNLENBQUMsTUFBTSxlQUFlLE1BQU0sQ0FBQyxNQUFNLElBQUksU0FBUyxFQUFFLENBQUMsQ0FBQztZQUVsRixPQUFPLE1BQU0sQ0FBQztRQUNoQixDQUFDO1FBQUMsT0FBTyxLQUFLLEVBQUUsQ0FBQztZQUNmLE1BQU0sQ0FBQyxHQUFHLENBQUMsT0FBTyxFQUFFLDZCQUE2QixLQUFLLENBQUMsT0FBTyxFQUFFLENBQUMsQ0FBQztZQUVsRSxjQUFjLENBQUMsV0FBVyxFQUFFLENBQUMsUUFBUSxDQUFDO2dCQUNwQyxLQUFLLEVBQUUsZ0JBQWdCLENBQUMsS0FBSztnQkFDN0IsSUFBSSxFQUFFLGlCQUFpQixDQUFDLElBQUk7Z0JBQzVCLE9BQU8sRUFBRSx5QkFBeUI7Z0JBQ2xDLE9BQU8sRUFBRSxFQUFFLEtBQUssRUFBRSxLQUFLLENBQUMsT0FBTyxFQUFFO2dCQUNqQyxPQUFPLEVBQUUsS0FBSzthQUNmLENBQUMsQ0FBQztZQUVILE9BQU87Z0JBQ0wsT0FBTyxFQUFFLEtBQUs7Z0JBQ2QsTUFBTSxFQUFFLFdBQVc7Z0JBQ25CLFlBQVksRUFBRSx1QkFBdUIsS0FBSyxDQUFDLE9BQU8sRUFBRTthQUNyRCxDQUFDO1FBQ0osQ0FBQztJQUNILENBQUM7SUFFRCxrREFBa0Q7SUFDM0MsVUFBVSxLQUFVLENBQUM7SUFFNUIsbURBQW1EO0lBQzVDLFlBQVk7UUFDakIsT0FBTyxDQUFDLENBQUM7SUFDWCxDQUFDO0NBQ0YifQ== \ No newline at end of file diff --git a/dist_ts/mail/security/classes.dmarcverifier.d.ts b/dist_ts/mail/security/classes.dmarcverifier.d.ts deleted file mode 100644 index 43e5fb5..0000000 --- a/dist_ts/mail/security/classes.dmarcverifier.d.ts +++ /dev/null @@ -1,123 +0,0 @@ -import type { Email } from '../core/classes.email.js'; -/** - * DMARC policy types - */ -export declare enum DmarcPolicy { - NONE = "none", - QUARANTINE = "quarantine", - REJECT = "reject" -} -/** - * DMARC alignment modes - */ -export declare enum DmarcAlignment { - RELAXED = "r", - STRICT = "s" -} -/** - * DMARC record fields - */ -export interface DmarcRecord { - version: string; - policy: DmarcPolicy; - subdomainPolicy?: DmarcPolicy; - pct?: number; - adkim?: DmarcAlignment; - aspf?: DmarcAlignment; - reportInterval?: number; - failureOptions?: string; - reportUriAggregate?: string[]; - reportUriForensic?: string[]; -} -/** - * DMARC verification result - */ -export interface DmarcResult { - hasDmarc: boolean; - record?: DmarcRecord; - spfDomainAligned: boolean; - dkimDomainAligned: boolean; - spfPassed: boolean; - dkimPassed: boolean; - policyEvaluated: DmarcPolicy; - actualPolicy: DmarcPolicy; - appliedPercentage: number; - action: 'pass' | 'quarantine' | 'reject'; - details: string; - error?: string; -} -/** - * Class for verifying and enforcing DMARC policies - */ -export declare class DmarcVerifier { - private dnsManager?; - constructor(dnsManager?: any); - /** - * Parse a DMARC record from a TXT record string - * @param record DMARC TXT record string - * @returns Parsed DMARC record or null if invalid - */ - parseDmarcRecord(record: string): DmarcRecord | null; - /** - * Check if domains are aligned according to DMARC policy - * @param headerDomain Domain from header (From) - * @param authDomain Domain from authentication (SPF, DKIM) - * @param alignment Alignment mode - * @returns Whether the domains are aligned - */ - private isDomainAligned; - /** - * Extract domain from an email address - * @param email Email address - * @returns Domain part of the email - */ - private getDomainFromEmail; - /** - * Check if DMARC verification should be applied based on percentage - * @param record DMARC record - * @returns Whether DMARC verification should be applied - */ - private shouldApplyDmarc; - /** - * Determine the action to take based on DMARC policy - * @param policy DMARC policy - * @returns Action to take - */ - private determineAction; - /** - * Verify DMARC for an incoming email - * @param email Email to verify - * @param spfResult SPF verification result - * @param dkimResult DKIM verification result - * @returns DMARC verification result - */ - verify(email: Email, spfResult: { - domain: string; - result: boolean; - }, dkimResult: { - domain: string; - result: boolean; - }): Promise; - /** - * Apply DMARC policy to an email - * @param email Email to apply policy to - * @param dmarcResult DMARC verification result - * @returns Whether the email should be accepted - */ - applyPolicy(email: Email, dmarcResult: DmarcResult): boolean; - /** - * End-to-end DMARC verification and policy application - * This method should be called after SPF and DKIM verification - * @param email Email to verify - * @param spfResult SPF verification result - * @param dkimResult DKIM verification result - * @returns Whether the email should be accepted - */ - verifyAndApply(email: Email, spfResult: { - domain: string; - result: boolean; - }, dkimResult: { - domain: string; - result: boolean; - }): Promise; -} diff --git a/dist_ts/mail/security/classes.dmarcverifier.js b/dist_ts/mail/security/classes.dmarcverifier.js deleted file mode 100644 index 64631d5..0000000 --- a/dist_ts/mail/security/classes.dmarcverifier.js +++ /dev/null @@ -1,366 +0,0 @@ -import { logger } from '../../logger.js'; -import { SecurityLogger, SecurityLogLevel, SecurityEventType } from '../../security/index.js'; -/** - * DMARC policy types - */ -export var DmarcPolicy; -(function (DmarcPolicy) { - DmarcPolicy["NONE"] = "none"; - DmarcPolicy["QUARANTINE"] = "quarantine"; - DmarcPolicy["REJECT"] = "reject"; -})(DmarcPolicy || (DmarcPolicy = {})); -/** - * DMARC alignment modes - */ -export var DmarcAlignment; -(function (DmarcAlignment) { - DmarcAlignment["RELAXED"] = "r"; - DmarcAlignment["STRICT"] = "s"; -})(DmarcAlignment || (DmarcAlignment = {})); -/** - * Class for verifying and enforcing DMARC policies - */ -export class DmarcVerifier { - // DNS Manager reference for verifying records - dnsManager; - constructor(dnsManager) { - this.dnsManager = dnsManager; - } - /** - * Parse a DMARC record from a TXT record string - * @param record DMARC TXT record string - * @returns Parsed DMARC record or null if invalid - */ - parseDmarcRecord(record) { - if (!record.startsWith('v=DMARC1')) { - return null; - } - try { - // Initialize record with default values - const dmarcRecord = { - version: 'DMARC1', - policy: DmarcPolicy.NONE, - pct: 100, - adkim: DmarcAlignment.RELAXED, - aspf: DmarcAlignment.RELAXED - }; - // Split the record into tag/value pairs - const parts = record.split(';').map(part => part.trim()); - for (const part of parts) { - if (!part || !part.includes('=')) - continue; - const [tag, value] = part.split('=').map(p => p.trim()); - // Process based on tag - switch (tag.toLowerCase()) { - case 'v': - dmarcRecord.version = value; - break; - case 'p': - dmarcRecord.policy = value; - break; - case 'sp': - dmarcRecord.subdomainPolicy = value; - break; - case 'pct': - const pctValue = parseInt(value, 10); - if (!isNaN(pctValue) && pctValue >= 0 && pctValue <= 100) { - dmarcRecord.pct = pctValue; - } - break; - case 'adkim': - dmarcRecord.adkim = value; - break; - case 'aspf': - dmarcRecord.aspf = value; - break; - case 'ri': - const interval = parseInt(value, 10); - if (!isNaN(interval) && interval > 0) { - dmarcRecord.reportInterval = interval; - } - break; - case 'fo': - dmarcRecord.failureOptions = value; - break; - case 'rua': - dmarcRecord.reportUriAggregate = value.split(',').map(uri => { - if (uri.startsWith('mailto:')) { - return uri.substring(7).trim(); - } - return uri.trim(); - }); - break; - case 'ruf': - dmarcRecord.reportUriForensic = value.split(',').map(uri => { - if (uri.startsWith('mailto:')) { - return uri.substring(7).trim(); - } - return uri.trim(); - }); - break; - } - } - // Ensure subdomain policy is set if not explicitly provided - if (!dmarcRecord.subdomainPolicy) { - dmarcRecord.subdomainPolicy = dmarcRecord.policy; - } - return dmarcRecord; - } - catch (error) { - logger.log('error', `Error parsing DMARC record: ${error.message}`, { - record, - error: error.message - }); - return null; - } - } - /** - * Check if domains are aligned according to DMARC policy - * @param headerDomain Domain from header (From) - * @param authDomain Domain from authentication (SPF, DKIM) - * @param alignment Alignment mode - * @returns Whether the domains are aligned - */ - isDomainAligned(headerDomain, authDomain, alignment) { - if (!headerDomain || !authDomain) { - return false; - } - // For strict alignment, domains must match exactly - if (alignment === DmarcAlignment.STRICT) { - return headerDomain.toLowerCase() === authDomain.toLowerCase(); - } - // For relaxed alignment, the authenticated domain must be a subdomain of the header domain - // or the same as the header domain - const headerParts = headerDomain.toLowerCase().split('.'); - const authParts = authDomain.toLowerCase().split('.'); - // Ensures we have at least two parts (domain and TLD) - if (headerParts.length < 2 || authParts.length < 2) { - return false; - } - // Get organizational domain (last two parts) - const headerOrgDomain = headerParts.slice(-2).join('.'); - const authOrgDomain = authParts.slice(-2).join('.'); - return headerOrgDomain === authOrgDomain; - } - /** - * Extract domain from an email address - * @param email Email address - * @returns Domain part of the email - */ - getDomainFromEmail(email) { - if (!email) - return ''; - // Handle name + email format: "John Doe " - const matches = email.match(/<([^>]+)>/); - const address = matches ? matches[1] : email; - const parts = address.split('@'); - return parts.length > 1 ? parts[1] : ''; - } - /** - * Check if DMARC verification should be applied based on percentage - * @param record DMARC record - * @returns Whether DMARC verification should be applied - */ - shouldApplyDmarc(record) { - if (record.pct === undefined || record.pct === 100) { - return true; - } - // Apply DMARC randomly based on percentage - const random = Math.floor(Math.random() * 100) + 1; - return random <= record.pct; - } - /** - * Determine the action to take based on DMARC policy - * @param policy DMARC policy - * @returns Action to take - */ - determineAction(policy) { - switch (policy) { - case DmarcPolicy.REJECT: - return 'reject'; - case DmarcPolicy.QUARANTINE: - return 'quarantine'; - case DmarcPolicy.NONE: - default: - return 'pass'; - } - } - /** - * Verify DMARC for an incoming email - * @param email Email to verify - * @param spfResult SPF verification result - * @param dkimResult DKIM verification result - * @returns DMARC verification result - */ - async verify(email, spfResult, dkimResult) { - const securityLogger = SecurityLogger.getInstance(); - // Initialize result - const result = { - hasDmarc: false, - spfDomainAligned: false, - dkimDomainAligned: false, - spfPassed: spfResult.result, - dkimPassed: dkimResult.result, - policyEvaluated: DmarcPolicy.NONE, - actualPolicy: DmarcPolicy.NONE, - appliedPercentage: 100, - action: 'pass', - details: 'DMARC not configured' - }; - try { - // Extract From domain - const fromHeader = email.getFromEmail(); - const fromDomain = this.getDomainFromEmail(fromHeader); - if (!fromDomain) { - result.error = 'Invalid From domain'; - return result; - } - // Check alignment - result.spfDomainAligned = this.isDomainAligned(fromDomain, spfResult.domain, DmarcAlignment.RELAXED); - result.dkimDomainAligned = this.isDomainAligned(fromDomain, dkimResult.domain, DmarcAlignment.RELAXED); - // Lookup DMARC record - const dmarcVerificationResult = this.dnsManager ? - await this.dnsManager.verifyDmarcRecord(fromDomain) : - { found: false, valid: false, error: 'DNS Manager not available' }; - // If DMARC record exists and is valid - if (dmarcVerificationResult.found && dmarcVerificationResult.valid) { - result.hasDmarc = true; - // Parse DMARC record - const parsedRecord = this.parseDmarcRecord(dmarcVerificationResult.value); - if (parsedRecord) { - result.record = parsedRecord; - result.actualPolicy = parsedRecord.policy; - result.appliedPercentage = parsedRecord.pct || 100; - // Override alignment modes if specified in record - if (parsedRecord.adkim) { - result.dkimDomainAligned = this.isDomainAligned(fromDomain, dkimResult.domain, parsedRecord.adkim); - } - if (parsedRecord.aspf) { - result.spfDomainAligned = this.isDomainAligned(fromDomain, spfResult.domain, parsedRecord.aspf); - } - // Determine DMARC compliance - const spfAligned = result.spfPassed && result.spfDomainAligned; - const dkimAligned = result.dkimPassed && result.dkimDomainAligned; - // Email passes DMARC if either SPF or DKIM passes with alignment - const dmarcPass = spfAligned || dkimAligned; - // Use record percentage to determine if policy should be applied - const applyPolicy = this.shouldApplyDmarc(parsedRecord); - if (!dmarcPass) { - // DMARC failed, apply policy - result.policyEvaluated = applyPolicy ? parsedRecord.policy : DmarcPolicy.NONE; - result.action = this.determineAction(result.policyEvaluated); - result.details = `DMARC failed: SPF aligned=${spfAligned}, DKIM aligned=${dkimAligned}, policy=${result.policyEvaluated}`; - } - else { - result.policyEvaluated = DmarcPolicy.NONE; - result.action = 'pass'; - result.details = `DMARC passed: SPF aligned=${spfAligned}, DKIM aligned=${dkimAligned}`; - } - } - else { - result.error = 'Invalid DMARC record format'; - result.details = 'DMARC record invalid'; - } - } - else { - // No DMARC record found or invalid - result.details = dmarcVerificationResult.error || 'No DMARC record found'; - } - // Log the DMARC verification - securityLogger.logEvent({ - level: result.action === 'pass' ? SecurityLogLevel.INFO : SecurityLogLevel.WARN, - type: SecurityEventType.DMARC, - message: result.details, - domain: fromDomain, - details: { - fromDomain, - spfDomain: spfResult.domain, - dkimDomain: dkimResult.domain, - spfPassed: result.spfPassed, - dkimPassed: result.dkimPassed, - spfAligned: result.spfDomainAligned, - dkimAligned: result.dkimDomainAligned, - dmarcPolicy: result.policyEvaluated, - action: result.action - }, - success: result.action === 'pass' - }); - return result; - } - catch (error) { - logger.log('error', `Error verifying DMARC: ${error.message}`, { - error: error.message, - emailId: email.getMessageId() - }); - result.error = `DMARC verification error: ${error.message}`; - // Log error - securityLogger.logEvent({ - level: SecurityLogLevel.ERROR, - type: SecurityEventType.DMARC, - message: `DMARC verification failed with error`, - details: { - error: error.message, - emailId: email.getMessageId() - }, - success: false - }); - return result; - } - } - /** - * Apply DMARC policy to an email - * @param email Email to apply policy to - * @param dmarcResult DMARC verification result - * @returns Whether the email should be accepted - */ - applyPolicy(email, dmarcResult) { - // Apply action based on DMARC verification result - switch (dmarcResult.action) { - case 'reject': - // Reject the email - email.mightBeSpam = true; - logger.log('warn', `Email rejected due to DMARC policy: ${dmarcResult.details}`, { - emailId: email.getMessageId(), - from: email.getFromEmail(), - subject: email.subject - }); - return false; - case 'quarantine': - // Quarantine the email (mark as spam) - email.mightBeSpam = true; - // Add spam header - if (!email.headers['X-Spam-Flag']) { - email.headers['X-Spam-Flag'] = 'YES'; - } - // Add DMARC reason header - email.headers['X-DMARC-Result'] = dmarcResult.details; - logger.log('warn', `Email quarantined due to DMARC policy: ${dmarcResult.details}`, { - emailId: email.getMessageId(), - from: email.getFromEmail(), - subject: email.subject - }); - return true; - case 'pass': - default: - // Accept the email - // Add DMARC result header for information - email.headers['X-DMARC-Result'] = dmarcResult.details; - return true; - } - } - /** - * End-to-end DMARC verification and policy application - * This method should be called after SPF and DKIM verification - * @param email Email to verify - * @param spfResult SPF verification result - * @param dkimResult DKIM verification result - * @returns Whether the email should be accepted - */ - async verifyAndApply(email, spfResult, dkimResult) { - // Verify DMARC - const dmarcResult = await this.verify(email, spfResult, dkimResult); - // Apply DMARC policy - return this.applyPolicy(email, dmarcResult); - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5kbWFyY3ZlcmlmaWVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vdHMvbWFpbC9zZWN1cml0eS9jbGFzc2VzLmRtYXJjdmVyaWZpZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQ3pDLE9BQU8sRUFBRSxjQUFjLEVBQUUsZ0JBQWdCLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUc5Rjs7R0FFRztBQUNILE1BQU0sQ0FBTixJQUFZLFdBSVg7QUFKRCxXQUFZLFdBQVc7SUFDckIsNEJBQWEsQ0FBQTtJQUNiLHdDQUF5QixDQUFBO0lBQ3pCLGdDQUFpQixDQUFBO0FBQ25CLENBQUMsRUFKVyxXQUFXLEtBQVgsV0FBVyxRQUl0QjtBQUVEOztHQUVHO0FBQ0gsTUFBTSxDQUFOLElBQVksY0FHWDtBQUhELFdBQVksY0FBYztJQUN4QiwrQkFBYSxDQUFBO0lBQ2IsOEJBQVksQ0FBQTtBQUNkLENBQUMsRUFIVyxjQUFjLEtBQWQsY0FBYyxRQUd6QjtBQXVDRDs7R0FFRztBQUNILE1BQU0sT0FBTyxhQUFhO0lBQ3hCLDhDQUE4QztJQUN0QyxVQUFVLENBQU87SUFFekIsWUFBWSxVQUFnQjtRQUMxQixJQUFJLENBQUMsVUFBVSxHQUFHLFVBQVUsQ0FBQztJQUMvQixDQUFDO0lBRUQ7Ozs7T0FJRztJQUNJLGdCQUFnQixDQUFDLE1BQWM7UUFDcEMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxVQUFVLENBQUMsVUFBVSxDQUFDLEVBQUUsQ0FBQztZQUNuQyxPQUFPLElBQUksQ0FBQztRQUNkLENBQUM7UUFFRCxJQUFJLENBQUM7WUFDSCx3Q0FBd0M7WUFDeEMsTUFBTSxXQUFXLEdBQWdCO2dCQUMvQixPQUFPLEVBQUUsUUFBUTtnQkFDakIsTUFBTSxFQUFFLFdBQVcsQ0FBQyxJQUFJO2dCQUN4QixHQUFHLEVBQUUsR0FBRztnQkFDUixL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 \ No newline at end of file diff --git a/dist_ts/mail/security/classes.spfverifier.d.ts b/dist_ts/mail/security/classes.spfverifier.d.ts deleted file mode 100644 index c1d6b91..0000000 --- a/dist_ts/mail/security/classes.spfverifier.d.ts +++ /dev/null @@ -1,71 +0,0 @@ -import type { Email } from '../core/classes.email.js'; -/** - * SPF result qualifiers - */ -export declare enum SpfQualifier { - PASS = "+", - NEUTRAL = "?", - SOFTFAIL = "~", - FAIL = "-" -} -/** - * SPF mechanism types - */ -export declare enum SpfMechanismType { - ALL = "all", - INCLUDE = "include", - A = "a", - MX = "mx", - IP4 = "ip4", - IP6 = "ip6", - EXISTS = "exists", - REDIRECT = "redirect", - EXP = "exp" -} -/** - * SPF mechanism definition - */ -export interface SpfMechanism { - qualifier: SpfQualifier; - type: SpfMechanismType; - value?: string; -} -/** - * SPF record parsed data - */ -export interface SpfRecord { - version: string; - mechanisms: SpfMechanism[]; - modifiers: Record; -} -/** - * SPF verification result - */ -export interface SpfResult { - result: 'pass' | 'neutral' | 'softfail' | 'fail' | 'temperror' | 'permerror' | 'none'; - explanation?: string; - domain: string; - ip: string; - record?: string; - error?: string; -} -/** - * Class for verifying SPF records. - * Delegates actual SPF evaluation to the Rust security bridge. - * Retains parseSpfRecord() for lightweight local parsing. - */ -export declare class SpfVerifier { - constructor(_dnsManager?: any); - /** - * Parse SPF record from TXT record (pure string parsing, no DNS) - */ - parseSpfRecord(record: string): SpfRecord | null; - /** - * Verify SPF for a given email — delegates to Rust bridge - */ - verify(email: Email, ip: string, heloDomain: string): Promise; - /** - * Check if email passes SPF verification and apply headers - */ - verifyAndApply(email: Email, ip: string, heloDomain: string): Promise; -} diff --git a/dist_ts/mail/security/classes.spfverifier.js b/dist_ts/mail/security/classes.spfverifier.js deleted file mode 100644 index d2eb854..0000000 --- a/dist_ts/mail/security/classes.spfverifier.js +++ /dev/null @@ -1,171 +0,0 @@ -import * as plugins from '../../plugins.js'; -import { logger } from '../../logger.js'; -import { SecurityLogger, SecurityLogLevel, SecurityEventType } from '../../security/index.js'; -import { RustSecurityBridge } from '../../security/classes.rustsecuritybridge.js'; -/** - * SPF result qualifiers - */ -export var SpfQualifier; -(function (SpfQualifier) { - SpfQualifier["PASS"] = "+"; - SpfQualifier["NEUTRAL"] = "?"; - SpfQualifier["SOFTFAIL"] = "~"; - SpfQualifier["FAIL"] = "-"; -})(SpfQualifier || (SpfQualifier = {})); -/** - * SPF mechanism types - */ -export var SpfMechanismType; -(function (SpfMechanismType) { - SpfMechanismType["ALL"] = "all"; - SpfMechanismType["INCLUDE"] = "include"; - SpfMechanismType["A"] = "a"; - SpfMechanismType["MX"] = "mx"; - SpfMechanismType["IP4"] = "ip4"; - SpfMechanismType["IP6"] = "ip6"; - SpfMechanismType["EXISTS"] = "exists"; - SpfMechanismType["REDIRECT"] = "redirect"; - SpfMechanismType["EXP"] = "exp"; -})(SpfMechanismType || (SpfMechanismType = {})); -/** - * Class for verifying SPF records. - * Delegates actual SPF evaluation to the Rust security bridge. - * Retains parseSpfRecord() for lightweight local parsing. - */ -export class SpfVerifier { - constructor(_dnsManager) { - // dnsManager is no longer needed — Rust handles DNS lookups - } - /** - * Parse SPF record from TXT record (pure string parsing, no DNS) - */ - parseSpfRecord(record) { - if (!record.startsWith('v=spf1')) { - return null; - } - try { - const spfRecord = { - version: 'spf1', - mechanisms: [], - modifiers: {} - }; - const terms = record.split(' ').filter(term => term.length > 0); - for (let i = 1; i < terms.length; i++) { - const term = terms[i]; - if (term.includes('=')) { - const [name, value] = term.split('='); - spfRecord.modifiers[name] = value; - continue; - } - let qualifier = SpfQualifier.PASS; - let mechanismText = term; - if (term.startsWith('+') || term.startsWith('-') || - term.startsWith('~') || term.startsWith('?')) { - qualifier = term[0]; - mechanismText = term.substring(1); - } - const colonIndex = mechanismText.indexOf(':'); - let type; - let value; - if (colonIndex !== -1) { - type = mechanismText.substring(0, colonIndex); - value = mechanismText.substring(colonIndex + 1); - } - else { - type = mechanismText; - } - spfRecord.mechanisms.push({ qualifier, type, value }); - } - return spfRecord; - } - catch (error) { - logger.log('error', `Error parsing SPF record: ${error.message}`, { - record, - error: error.message - }); - return null; - } - } - /** - * Verify SPF for a given email — delegates to Rust bridge - */ - async verify(email, ip, heloDomain) { - const securityLogger = SecurityLogger.getInstance(); - const mailFrom = email.from || ''; - const domain = mailFrom.split('@')[1] || ''; - try { - const bridge = RustSecurityBridge.getInstance(); - const result = await bridge.checkSpf({ - ip, - heloDomain, - hostname: plugins.os.hostname(), - mailFrom, - }); - const spfResult = { - result: result.result, - domain: result.domain, - ip: result.ip, - explanation: result.explanation ?? undefined, - }; - securityLogger.logEvent({ - level: spfResult.result === 'pass' ? SecurityLogLevel.INFO : - (spfResult.result === 'fail' ? SecurityLogLevel.WARN : SecurityLogLevel.INFO), - type: SecurityEventType.SPF, - message: `SPF ${spfResult.result} for ${spfResult.domain} from IP ${ip}`, - domain: spfResult.domain, - details: { ip, heloDomain, result: spfResult.result, explanation: spfResult.explanation }, - success: spfResult.result === 'pass' - }); - return spfResult; - } - catch (error) { - logger.log('error', `SPF verification error: ${error.message}`, { domain, ip, error: error.message }); - securityLogger.logEvent({ - level: SecurityLogLevel.ERROR, - type: SecurityEventType.SPF, - message: `SPF verification error for ${domain}`, - domain, - details: { ip, error: error.message }, - success: false - }); - return { - result: 'temperror', - explanation: `Error verifying SPF: ${error.message}`, - domain, - ip, - error: error.message - }; - } - } - /** - * Check if email passes SPF verification and apply headers - */ - async verifyAndApply(email, ip, heloDomain) { - const result = await this.verify(email, ip, heloDomain); - email.headers['Received-SPF'] = `${result.result} (${result.domain}: ${result.explanation || ''}) client-ip=${ip}; envelope-from=${email.getEnvelopeFrom()}; helo=${heloDomain};`; - switch (result.result) { - case 'fail': - email.mightBeSpam = true; - logger.log('warn', `SPF failed for ${result.domain} from ${ip}: ${result.explanation}`); - return false; - case 'softfail': - email.mightBeSpam = true; - logger.log('info', `SPF softfailed for ${result.domain} from ${ip}: ${result.explanation}`); - return true; - case 'neutral': - case 'none': - logger.log('info', `SPF ${result.result} for ${result.domain} from ${ip}: ${result.explanation}`); - return true; - case 'pass': - logger.log('info', `SPF passed for ${result.domain} from ${ip}: ${result.explanation}`); - return true; - case 'temperror': - case 'permerror': - logger.log('error', `SPF error for ${result.domain} from ${ip}: ${result.explanation}`); - return true; - default: - return true; - } - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5zcGZ2ZXJpZmllci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3RzL21haWwvc2VjdXJpdHkvY2xhc3Nlcy5zcGZ2ZXJpZmllci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssT0FBTyxNQUFNLGtCQUFrQixDQUFDO0FBQzVDLE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUN6QyxPQUFPLEVBQUUsY0FBYyxFQUFFLGdCQUFnQixFQUFFLGlCQUFpQixFQUFFLE1BQU0seUJBQXlCLENBQUM7QUFDOUYsT0FBTyxFQUFFLGtCQUFrQixFQUFFLE1BQU0sOENBQThDLENBQUM7QUFHbEY7O0dBRUc7QUFDSCxNQUFNLENBQU4sSUFBWSxZQUtYO0FBTEQsV0FBWSxZQUFZO0lBQ3RCLDBCQUFVLENBQUE7SUFDViw2QkFBYSxDQUFBO0lBQ2IsOEJBQWMsQ0FBQTtJQUNkLDBCQUFVLENBQUE7QUFDWixDQUFDLEVBTFcsWUFBWSxLQUFaLFlBQVksUUFLdkI7QUFFRDs7R0FFRztBQUNILE1BQU0sQ0FBTixJQUFZLGdCQVVYO0FBVkQsV0FBWSxnQkFBZ0I7SUFDMUIsK0JBQVcsQ0FBQTtJQUNYLHVDQUFtQixDQUFBO0lBQ25CLDJCQUFPLENBQUE7SUFDUCw2QkFBUyxDQUFBO0lBQ1QsK0JBQVcsQ0FBQTtJQUNYLCtCQUFXLENBQUE7SUFDWCxxQ0FBaUIsQ0FBQTtJQUNqQix5Q0FBcUIsQ0FBQTtJQUNyQiwrQkFBVyxDQUFBO0FBQ2IsQ0FBQyxFQVZXLGdCQUFnQixLQUFoQixnQkFBZ0IsUUFVM0I7QUFnQ0Q7Ozs7R0FJRztBQUNILE1BQU0sT0FBTyxXQUFXO0lBQ3RCLFlBQVksV0FBaUI7UUFDM0IsNERBQTREO0lBQzlELENBQUM7SUFFRDs7T0FFRztJQUNJLGNBQWMsQ0FBQyxNQUFjO1FBQ2xDLElBQUksQ0FBQyxNQUFNLENBQUMsVUFBVSxDQUFDLFFBQVEsQ0FBQyxFQUFFLENBQUM7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 \ No newline at end of file diff --git a/dist_ts/mail/security/index.d.ts b/dist_ts/mail/security/index.d.ts deleted file mode 100644 index 9f880de..0000000 --- a/dist_ts/mail/security/index.d.ts +++ /dev/null @@ -1,4 +0,0 @@ -export * from './classes.dkimcreator.js'; -export * from './classes.dkimverifier.js'; -export * from './classes.dmarcverifier.js'; -export * from './classes.spfverifier.js'; diff --git a/dist_ts/mail/security/index.js b/dist_ts/mail/security/index.js deleted file mode 100644 index 5c360c1..0000000 --- a/dist_ts/mail/security/index.js +++ /dev/null @@ -1,6 +0,0 @@ -// Email security components -export * from './classes.dkimcreator.js'; -export * from './classes.dkimverifier.js'; -export * from './classes.dmarcverifier.js'; -export * from './classes.spfverifier.js'; -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi90cy9tYWlsL3NlY3VyaXR5L2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLDRCQUE0QjtBQUM1QixjQUFjLDBCQUEwQixDQUFDO0FBQ3pDLGNBQWMsMkJBQTJCLENBQUM7QUFDMUMsY0FBYyw0QkFBNEIsQ0FBQztBQUMzQyxjQUFjLDBCQUEwQixDQUFDIn0= \ No newline at end of file diff --git a/dist_ts/paths.d.ts b/dist_ts/paths.d.ts deleted file mode 100644 index 22cc3b5..0000000 --- a/dist_ts/paths.d.ts +++ /dev/null @@ -1,14 +0,0 @@ -export declare const baseDir: string; -export declare const packageDir: string; -export declare const distServe: string; -export declare const dataDir: string; -export declare const keysDir: string; -export declare const dnsRecordsDir: string; -export declare const sentEmailsDir: string; -export declare const receivedEmailsDir: string; -export declare const failedEmailsDir: string; -export declare const logsDir: string; -export declare const emailTemplatesDir: string; -export declare const MtaAttachmentsDir: string; -export declare const configPath: string; -export declare function ensureDirectories(): Promise; diff --git a/dist_ts/paths.js b/dist_ts/paths.js deleted file mode 100644 index 5bcad5a..0000000 --- a/dist_ts/paths.js +++ /dev/null @@ -1,39 +0,0 @@ -import * as plugins from './plugins.js'; -// Base directories -export const baseDir = process.cwd(); -export const packageDir = plugins.path.join(plugins.smartpath.get.dirnameFromImportMetaUrl(import.meta.url), '../'); -export const distServe = plugins.path.join(packageDir, './dist_serve'); -// Configure data directory with environment variable or default to .nogit/data -const DEFAULT_DATA_PATH = '.nogit/data'; -export const dataDir = process.env.DATA_DIR - ? process.env.DATA_DIR - : plugins.path.join(baseDir, DEFAULT_DATA_PATH); -// MTA directories -export const keysDir = plugins.path.join(dataDir, 'keys'); -export const dnsRecordsDir = plugins.path.join(dataDir, 'dns'); -export const sentEmailsDir = plugins.path.join(dataDir, 'emails', 'sent'); -export const receivedEmailsDir = plugins.path.join(dataDir, 'emails', 'received'); -export const failedEmailsDir = plugins.path.join(dataDir, 'emails', 'failed'); // For failed emails -export const logsDir = plugins.path.join(dataDir, 'logs'); // For logs -// Email template directories -export const emailTemplatesDir = plugins.path.join(dataDir, 'templates', 'email'); -export const MtaAttachmentsDir = plugins.path.join(dataDir, 'attachments'); // For email attachments -// Configuration path -export const configPath = process.env.CONFIG_PATH - ? process.env.CONFIG_PATH - : plugins.path.join(baseDir, 'config.json'); -// Create directories if they don't exist -export async function ensureDirectories() { - // Ensure data directories - await plugins.smartfs.directory(dataDir).recursive().create(); - await plugins.smartfs.directory(keysDir).recursive().create(); - await plugins.smartfs.directory(dnsRecordsDir).recursive().create(); - await plugins.smartfs.directory(sentEmailsDir).recursive().create(); - await plugins.smartfs.directory(receivedEmailsDir).recursive().create(); - await plugins.smartfs.directory(failedEmailsDir).recursive().create(); - await plugins.smartfs.directory(logsDir).recursive().create(); - // Ensure email template directories - await plugins.smartfs.directory(emailTemplatesDir).recursive().create(); - await plugins.smartfs.directory(MtaAttachmentsDir).recursive().create(); -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGF0aHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi90cy9wYXRocy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssT0FBTyxNQUFNLGNBQWMsQ0FBQztBQUV4QyxtQkFBbUI7QUFDbkIsTUFBTSxDQUFDLE1BQU0sT0FBTyxHQUFHLE9BQU8sQ0FBQyxHQUFHLEVBQUUsQ0FBQztBQUNyQyxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQUcsT0FBTyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQ3pDLE9BQU8sQ0FBQyxTQUFTLENBQUMsR0FBRyxDQUFDLHdCQUF3QixDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLEVBQy9ELEtBQUssQ0FDTixDQUFDO0FBQ0YsTUFBTSxDQUFDLE1BQU0sU0FBUyxHQUFHLE9BQU8sQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLFVBQVUsRUFBRSxjQUFjLENBQUMsQ0FBQztBQUV2RSwrRUFBK0U7QUFDL0UsTUFBTSxpQkFBaUIsR0FBRyxhQUFhLENBQUM7QUFDeEMsTUFBTSxDQUFDLE1BQU0sT0FBTyxHQUFHLE9BQU8sQ0FBQyxHQUFHLENBQUMsUUFBUTtJQUN6QyxDQUFDLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxRQUFRO0lBQ3RCLENBQUMsQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxPQUFPLEVBQUUsaUJBQWlCLENBQUMsQ0FBQztBQUVsRCxtQkFBbUI7QUFDbkIsTUFBTSxDQUFDLE1BQU0sT0FBTyxHQUFHLE9BQU8sQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFBRSxNQUFNLENBQUMsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxhQUFhLEdBQUcsT0FBTyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsT0FBTyxFQUFFLEtBQUssQ0FBQyxDQUFDO0FBQy9ELE1BQU0sQ0FBQyxNQUFNLGFBQWEsR0FBRyxPQUFPLENBQUMsSUFBSSxDQUFDLElBQUks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 \ No newline at end of file diff --git a/dist_ts/plugins.d.ts b/dist_ts/plugins.d.ts deleted file mode 100644 index b3eaee4..0000000 --- a/dist_ts/plugins.d.ts +++ /dev/null @@ -1,47 +0,0 @@ -import * as dns from 'dns'; -import * as fs from 'fs'; -import * as crypto from 'crypto'; -import * as http from 'http'; -import * as net from 'net'; -import * as os from 'os'; -import * as path from 'path'; -import * as tls from 'tls'; -import * as util from 'util'; -export { dns, fs, crypto, http, net, os, path, tls, util, }; -import * as servezoneInterfaces from '@serve.zone/interfaces'; -export { servezoneInterfaces }; -import * as typedrequest from '@api.global/typedrequest'; -import * as typedserver from '@api.global/typedserver'; -import * as typedsocket from '@api.global/typedsocket'; -export { typedrequest, typedserver, typedsocket, }; -import * as projectinfo from '@push.rocks/projectinfo'; -import * as qenv from '@push.rocks/qenv'; -import * as smartacme from '@push.rocks/smartacme'; -import * as smartdata from '@push.rocks/smartdata'; -import * as smartdns from '@push.rocks/smartdns'; -import * as smartfile from '@push.rocks/smartfile'; -import { SmartFs } from '@push.rocks/smartfs'; -import * as smartguard from '@push.rocks/smartguard'; -import * as smartjwt from '@push.rocks/smartjwt'; -import * as smartlog from '@push.rocks/smartlog'; -import * as smartmail from '@push.rocks/smartmail'; -import * as smartmetrics from '@push.rocks/smartmetrics'; -import * as smartnetwork from '@push.rocks/smartnetwork'; -import * as smartpath from '@push.rocks/smartpath'; -import * as smartproxy from '@push.rocks/smartproxy'; -import * as smartpromise from '@push.rocks/smartpromise'; -import * as smartrequest from '@push.rocks/smartrequest'; -import * as smartrule from '@push.rocks/smartrule'; -import * as smartrust from '@push.rocks/smartrust'; -import * as smartrx from '@push.rocks/smartrx'; -import * as smartunique from '@push.rocks/smartunique'; -export declare const smartfs: SmartFs; -export { projectinfo, qenv, smartacme, smartdata, smartdns, smartfile, SmartFs, smartguard, smartjwt, smartlog, smartmail, smartmetrics, smartnetwork, smartpath, smartproxy, smartpromise, smartrequest, smartrule, smartrust, smartrx, smartunique }; -export type TLogLevel = 'error' | 'warn' | 'info' | 'success' | 'debug'; -import * as cloudflare from '@apiclient.xyz/cloudflare'; -export { cloudflare, }; -import * as tsclass from '@tsclass/tsclass'; -export { tsclass, }; -import mailparser from 'mailparser'; -import * as uuid from 'uuid'; -export { mailparser, uuid, }; diff --git a/dist_ts/plugins.js b/dist_ts/plugins.js deleted file mode 100644 index b9dd418..0000000 --- a/dist_ts/plugins.js +++ /dev/null @@ -1,54 +0,0 @@ -// node native -import * as dns from 'dns'; -import * as fs from 'fs'; -import * as crypto from 'crypto'; -import * as http from 'http'; -import * as net from 'net'; -import * as os from 'os'; -import * as path from 'path'; -import * as tls from 'tls'; -import * as util from 'util'; -export { dns, fs, crypto, http, net, os, path, tls, util, }; -// @serve.zone scope -import * as servezoneInterfaces from '@serve.zone/interfaces'; -export { servezoneInterfaces }; -// @api.global scope -import * as typedrequest from '@api.global/typedrequest'; -import * as typedserver from '@api.global/typedserver'; -import * as typedsocket from '@api.global/typedsocket'; -export { typedrequest, typedserver, typedsocket, }; -// @push.rocks scope -import * as projectinfo from '@push.rocks/projectinfo'; -import * as qenv from '@push.rocks/qenv'; -import * as smartacme from '@push.rocks/smartacme'; -import * as smartdata from '@push.rocks/smartdata'; -import * as smartdns from '@push.rocks/smartdns'; -import * as smartfile from '@push.rocks/smartfile'; -import { SmartFs, SmartFsProviderNode } from '@push.rocks/smartfs'; -import * as smartguard from '@push.rocks/smartguard'; -import * as smartjwt from '@push.rocks/smartjwt'; -import * as smartlog from '@push.rocks/smartlog'; -import * as smartmail from '@push.rocks/smartmail'; -import * as smartmetrics from '@push.rocks/smartmetrics'; -import * as smartnetwork from '@push.rocks/smartnetwork'; -import * as smartpath from '@push.rocks/smartpath'; -import * as smartproxy from '@push.rocks/smartproxy'; -import * as smartpromise from '@push.rocks/smartpromise'; -import * as smartrequest from '@push.rocks/smartrequest'; -import * as smartrule from '@push.rocks/smartrule'; -import * as smartrust from '@push.rocks/smartrust'; -import * as smartrx from '@push.rocks/smartrx'; -import * as smartunique from '@push.rocks/smartunique'; -export const smartfs = new SmartFs(new SmartFsProviderNode()); -export { projectinfo, qenv, smartacme, smartdata, smartdns, smartfile, SmartFs, smartguard, smartjwt, smartlog, smartmail, smartmetrics, smartnetwork, smartpath, smartproxy, smartpromise, smartrequest, smartrule, smartrust, smartrx, smartunique }; -// apiclient.xyz scope -import * as cloudflare from '@apiclient.xyz/cloudflare'; -export { cloudflare, }; -// tsclass scope -import * as tsclass from '@tsclass/tsclass'; -export { tsclass, }; -// third party -import mailparser from 'mailparser'; -import * as uuid from 'uuid'; -export { mailparser, uuid, }; -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGx1Z2lucy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3RzL3BsdWdpbnMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYztBQUNkLE9BQU8sS0FBSyxHQUFHLE1BQU0sS0FBSyxDQUFDO0FBQzNCLE9BQU8sS0FBSyxFQUFFLE1BQU0sSUFBSSxDQUFDO0FBQ3pCLE9BQU8sS0FBSyxNQUFNLE1BQU0sUUFBUSxDQUFDO0FBQ2pDLE9BQU8sS0FBSyxJQUFJLE1BQU0sTUFBTSxDQUFDO0FBQzdCLE9BQU8sS0FBSyxHQUFHLE1BQU0sS0FBSyxDQUFDO0FBQzNCLE9BQU8sS0FBSyxFQUFFLE1BQU0sSUFBSSxDQUFDO0FBQ3pCLE9BQU8sS0FBSyxJQUFJLE1BQU0sTUFBTSxDQUFDO0FBQzdCLE9BQU8sS0FBSyxHQUFHLE1BQU0sS0FBSyxDQUFDO0FBQzNCLE9BQU8sS0FBSyxJQUFJLE1BQU0sTUFBTSxDQUFDO0FBRTdCLE9BQU8sRUFDTCxHQUFHLEVBQ0gsRUFBRSxFQUNGLE1BQU0sRUFDTixJQUFJLEVBQ0osR0FBRyxFQUNILEVBQUUsRUFDRixJQUFJLEVBQ0osR0FBRyxFQUNILElBQUksR0FDTCxDQUFBO0FBRUQsb0JBQW9CO0FBQ3BCLE9BQU8sS0FBSyxtQkFBbUIsTUFBTSx3QkFBd0IsQ0FBQztBQUU5RCxPQUFPLEVBQ0wsbUJBQW1CLEVBQ3BCLENBQUE7QUFFRCxvQkFBb0I7QUFDcEIsT0FBTyxLQUFLLFlBQVksTUFBTSwwQkFBMEIsQ0FBQztBQUN6RCxPQUFPLEtBQUssV0FBVyxNQUFNLHlCQUF5QixDQUFDO0FBQ3ZELE9BQU8sS0FBSyxXQUFXLE1BQU0seUJBQXlCLENBQUM7QUFFdkQsT0FBTyxFQUNMLFlBQVksRUFDWixXQUFXLEVBQ1gsV0FBVyxHQUNaLENBQUE7QUFFRCxvQkFBb0I7QUFDcEIsT0FBTyxLQUFLLFdBQVcsTUFBTSx5QkFBeUIsQ0FBQztBQUN2RCxPQUFPLEtBQUssSUFBSSxNQUFNLGtCQUFrQixDQUFDO0FBQ3pDLE9BQU8sS0FBSyxTQUFTLE1BQU0sdUJBQXVCLENBQUM7QUFDbkQsT0FBTyxLQUFLLFNBQVMsTUFBTSx1QkFBdUIsQ0FBQztBQUNuRCxPQUFPLEtBQUssUUFBUSxNQUFNLHNCQUFzQixDQUFDO0FBQ2pELE9BQU8sS0FBSyxTQUFTLE1BQU0sdUJBQXVCLENBQUM7QUFDbkQsT0FBTyxFQUFFLE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBQ25FLE9BQU8sS0FBSyxVQUFVLE1BQU0sd0JBQXdCLENBQUM7QUFDckQsT0FBTyxLQUFLLFFBQVEsTUFBTSxzQkFBc0IsQ0FBQztBQUNqRCxPQUFPLEtBQUssUUFBUSxNQUFNLHNCQUFzQixDQUFDO0FBQ2pELE9BQU8sS0FBSyxTQUFTLE1BQU0sdUJBQXVCLENBQUM7QUFDbkQsT0FBTyxLQUFLLFlBQVksTUFBTSwwQkFBMEIsQ0FBQztBQUN6RCxPQUFPLEtBQUssWUFBWSxNQUFNLDBCQUEwQixDQUFDO0FBQ3pELE9BQU8sS0FBSyxTQUFTLE1BQU0sdUJBQXVCLENBQUM7QUFDbkQsT0FBTyxLQUFLLFVBQVUsTUFBTSx3QkFBd0IsQ0FBQztBQUNyRCxPQUFPLEtBQUssWUFBWSxNQUFNLDBCQUEwQixDQUFDO0FBQ3pELE9BQU8sS0FBSyxZQUFZLE1BQU0sMEJBQTBCLENBQUM7QUFDekQsT0FBTyxLQUFLLFNBQVMsTUFBTSx1QkFBdUIsQ0FBQztBQUNuRCxPQUFPLEtBQUssU0FBUyxNQUFNLHVCQUF1QixDQUFDO0FBQ25ELE9BQU8sS0FBSyxPQUFPLE1BQU0scUJBQXFCLENBQUM7QUFDL0MsT0FBTyxLQUFLLFdBQVcsTUFBTSx5QkFBeUIsQ0FBQztBQUV2RCxNQUFNLENBQUMsTUFBTSxPQUFPLEdBQUcsSUFBSSxPQUFPLENBQUMsSUFBSSxtQkFBbUIsRUFBRSxDQUFDLENBQUM7QUFFOUQsT0FBTyxFQUFFLFdBQVcsRUFBRSxJQUFJLEVBQUUsU0FBUyxFQUFFLFNBQVMsRUFBRSxRQUFRLEVBQUUsU0FBUyxFQUFFLE9BQU8sRUFBRSxVQUFVLEVBQUUsUUFBUSxFQUFFLFFBQVEsRUFBRSxTQUFTLEVBQUUsWUFBWSxFQUFFLFlBQVksRUFBRSxTQUFTLEVBQUUsVUFBVSxFQUFFLFlBQVksRUFBRSxZQUFZLEVBQUUsU0FBUyxFQUFFLFNBQVMsRUFBRSxPQUFPLEVBQUUsV0FBVyxFQUFFLENBQUM7QUFLdlAsc0JBQXNCO0FBQ3RCLE9BQU8sS0FBSyxVQUFVLE1BQU0sMkJBQTJCLENBQUM7QUFFeEQsT0FBTyxFQUNMLFVBQVUsR0FDWCxDQUFBO0FBRUQsZ0JBQWdCO0FBQ2hCLE9BQU8sS0FBSyxPQUFPLE1BQU0sa0JBQWtCLENBQUM7QUFFNUMsT0FBTyxFQUNMLE9BQU8sR0FDUixDQUFBO0FBRUQsY0FBYztBQUNkLE9BQU8sVUFBVSxNQUFNLFlBQVksQ0FBQztBQUNwQyxPQUFPLEtBQUssSUFBSSxNQUFNLE1BQU0sQ0FBQztBQUU3QixPQUFPLEVBQ0wsVUFBVSxFQUNWLElBQUksR0FDTCxDQUFBIn0= \ No newline at end of file diff --git a/dist_ts/security/classes.contentscanner.d.ts b/dist_ts/security/classes.contentscanner.d.ts deleted file mode 100644 index 41c55d7..0000000 --- a/dist_ts/security/classes.contentscanner.d.ts +++ /dev/null @@ -1,130 +0,0 @@ -import { Email } from '../mail/core/classes.email.js'; -/** - * Scan result information - */ -export interface IScanResult { - isClean: boolean; - threatType?: string; - threatDetails?: string; - threatScore: number; - scannedElements: string[]; - timestamp: number; -} -/** - * Options for content scanner configuration - */ -export interface IContentScannerOptions { - maxCacheSize?: number; - cacheTTL?: number; - scanSubject?: boolean; - scanBody?: boolean; - scanAttachments?: boolean; - maxAttachmentSizeToScan?: number; - scanAttachmentNames?: boolean; - blockExecutables?: boolean; - blockMacros?: boolean; - customRules?: Array<{ - pattern: string | RegExp; - type: string; - score: number; - description: string; - }>; - minThreatScore?: number; - highThreatScore?: number; -} -/** - * Threat categories - */ -export declare enum ThreatCategory { - SPAM = "spam", - PHISHING = "phishing", - MALWARE = "malware", - EXECUTABLE = "executable", - SUSPICIOUS_LINK = "suspicious_link", - MALICIOUS_MACRO = "malicious_macro", - XSS = "xss", - SENSITIVE_DATA = "sensitive_data", - BLACKLISTED_CONTENT = "blacklisted_content", - CUSTOM_RULE = "custom_rule" -} -/** - * Content Scanner for detecting malicious email content - */ -export declare class ContentScanner { - private static instance; - private scanCache; - private options; - /** - * Default options for the content scanner - */ - private static readonly DEFAULT_OPTIONS; - /** - * Constructor for the ContentScanner - * @param options Configuration options - */ - constructor(options?: IContentScannerOptions); - /** - * Get the singleton instance of the scanner - * @param options Configuration options - * @returns Singleton scanner instance - */ - static getInstance(options?: IContentScannerOptions): ContentScanner; - /** - * Scan an email for malicious content. - * Delegates text/subject/html/filename pattern scanning to Rust. - * Binary attachment scanning (PE headers, VBA macros) stays in TS. - * @param email The email to scan - * @returns Scan result - */ - scanEmail(email: Email): Promise; - /** - * Generate a cache key from an email - * @param email The email to generate a key for - * @returns Cache key - */ - private generateCacheKey; - /** - * Scan attachment binary content for PE headers and VBA macros. - * This stays in TS because it accesses raw Buffer data (too large for IPC). - * @param attachment The attachment to scan - * @param result The scan result to update - */ - private scanAttachmentBinary; - /** - * Apply custom rules (runtime-configured patterns) to the email. - * These stay in TS because they are configured at runtime. - * @param email The email to check - * @param result The scan result to update - */ - private applyCustomRules; - /** - * Extract text from a binary buffer for scanning - * @param buffer Binary content - * @returns Extracted text (may be partial) - */ - private extractTextFromBuffer; - /** - * Check if an Office document likely contains macros - * @param attachment The attachment to check - * @returns Whether the file likely contains macros - */ - private likelyContainsMacros; - /** - * Log a high threat finding to the security logger - * @param email The email containing the threat - * @param result The scan result - */ - private logHighThreatFound; - /** - * Log a threat finding to the security logger - * @param email The email containing the threat - * @param result The scan result - */ - private logThreatFound; - /** - * Get threat level description based on score - * @param score Threat score - * @returns Threat level description - */ - static getThreatLevel(score: number): 'none' | 'low' | 'medium' | 'high'; -} diff --git a/dist_ts/security/classes.contentscanner.js b/dist_ts/security/classes.contentscanner.js deleted file mode 100644 index 481b1dd..0000000 --- a/dist_ts/security/classes.contentscanner.js +++ /dev/null @@ -1,338 +0,0 @@ -import * as plugins from '../plugins.js'; -import * as paths from '../paths.js'; -import { logger } from '../logger.js'; -import { Email } from '../mail/core/classes.email.js'; -import { SecurityLogger, SecurityLogLevel, SecurityEventType } from './classes.securitylogger.js'; -import { RustSecurityBridge } from './classes.rustsecuritybridge.js'; -import { LRUCache } from 'lru-cache'; -/** - * Threat categories - */ -export var ThreatCategory; -(function (ThreatCategory) { - ThreatCategory["SPAM"] = "spam"; - ThreatCategory["PHISHING"] = "phishing"; - ThreatCategory["MALWARE"] = "malware"; - ThreatCategory["EXECUTABLE"] = "executable"; - ThreatCategory["SUSPICIOUS_LINK"] = "suspicious_link"; - ThreatCategory["MALICIOUS_MACRO"] = "malicious_macro"; - ThreatCategory["XSS"] = "xss"; - ThreatCategory["SENSITIVE_DATA"] = "sensitive_data"; - ThreatCategory["BLACKLISTED_CONTENT"] = "blacklisted_content"; - ThreatCategory["CUSTOM_RULE"] = "custom_rule"; -})(ThreatCategory || (ThreatCategory = {})); -/** - * Content Scanner for detecting malicious email content - */ -export class ContentScanner { - static instance; - scanCache; - options; - /** - * Default options for the content scanner - */ - static DEFAULT_OPTIONS = { - maxCacheSize: 10000, - cacheTTL: 24 * 60 * 60 * 1000, // 24 hours - scanSubject: true, - scanBody: true, - scanAttachments: true, - maxAttachmentSizeToScan: 10 * 1024 * 1024, // 10MB - scanAttachmentNames: true, - blockExecutables: true, - blockMacros: true, - customRules: [], - minThreatScore: 30, // Minimum score to consider content as a threat - highThreatScore: 70 // Score above which content is considered high threat - }; - /** - * Constructor for the ContentScanner - * @param options Configuration options - */ - constructor(options = {}) { - // Merge with default options - this.options = { - ...ContentScanner.DEFAULT_OPTIONS, - ...options - }; - // Initialize cache - this.scanCache = new LRUCache({ - max: this.options.maxCacheSize, - ttl: this.options.cacheTTL, - }); - logger.log('info', 'ContentScanner initialized'); - } - /** - * Get the singleton instance of the scanner - * @param options Configuration options - * @returns Singleton scanner instance - */ - static getInstance(options = {}) { - if (!ContentScanner.instance) { - ContentScanner.instance = new ContentScanner(options); - } - return ContentScanner.instance; - } - /** - * Scan an email for malicious content. - * Delegates text/subject/html/filename pattern scanning to Rust. - * Binary attachment scanning (PE headers, VBA macros) stays in TS. - * @param email The email to scan - * @returns Scan result - */ - async scanEmail(email) { - try { - // Generate a cache key from the email - const cacheKey = this.generateCacheKey(email); - // Check cache first - const cachedResult = this.scanCache.get(cacheKey); - if (cachedResult) { - logger.log('info', `Using cached scan result for email ${email.getMessageId()}`); - return cachedResult; - } - // Delegate text/subject/html/filename scanning to Rust - const bridge = RustSecurityBridge.getInstance(); - const rustResult = await bridge.scanContent({ - subject: this.options.scanSubject ? email.subject : undefined, - textBody: this.options.scanBody ? email.text : undefined, - htmlBody: this.options.scanBody ? email.html : undefined, - attachmentNames: this.options.scanAttachmentNames - ? email.attachments?.map(a => a.filename) ?? [] - : [], - }); - const result = { - isClean: true, - threatScore: rustResult.threatScore, - threatType: rustResult.threatType ?? undefined, - threatDetails: rustResult.threatDetails ?? undefined, - scannedElements: rustResult.scannedElements, - timestamp: Date.now(), - }; - // Attachment binary scanning stays in TS (PE headers, macro detection) - if (this.options.scanAttachments && email.attachments?.length > 0) { - for (const attachment of email.attachments) { - this.scanAttachmentBinary(attachment, result); - } - } - // Apply custom rules (TS-only, runtime-configured) - this.applyCustomRules(email, result); - // Determine if the email is clean based on threat score - result.isClean = result.threatScore < this.options.minThreatScore; - // Save to cache - this.scanCache.set(cacheKey, result); - // Log high threat findings - if (result.threatScore >= this.options.highThreatScore) { - this.logHighThreatFound(email, result); - } - else if (!result.isClean) { - this.logThreatFound(email, result); - } - return result; - } - catch (error) { - logger.log('error', `Error scanning email: ${error.message}`, { - messageId: email.getMessageId(), - error: error.stack - }); - // Return a safe default with error indication - return { - isClean: true, - threatScore: 0, - scannedElements: ['error'], - timestamp: Date.now(), - threatType: 'scan_error', - threatDetails: `Scan error: ${error.message}` - }; - } - } - /** - * Generate a cache key from an email - * @param email The email to generate a key for - * @returns Cache key - */ - generateCacheKey(email) { - // Use message ID if available - if (email.getMessageId()) { - return `email:${email.getMessageId()}`; - } - // Fallback to a hash of key content - const contentToHash = [ - email.from, - email.subject || '', - email.text?.substring(0, 1000) || '', - email.html?.substring(0, 1000) || '', - email.attachments?.length || 0 - ].join(':'); - return `email:${plugins.crypto.createHash('sha256').update(contentToHash).digest('hex')}`; - } - /** - * Scan attachment binary content for PE headers and VBA macros. - * This stays in TS because it accesses raw Buffer data (too large for IPC). - * @param attachment The attachment to scan - * @param result The scan result to update - */ - scanAttachmentBinary(attachment, result) { - if (!attachment.content) { - return; - } - // Skip large attachments - if (attachment.content.length > this.options.maxAttachmentSizeToScan) { - return; - } - const filename = attachment.filename.toLowerCase(); - // Check for PE headers (Windows executables disguised with non-.exe extensions) - if (attachment.content.length > 64 && - attachment.content[0] === 0x4D && - attachment.content[1] === 0x5A) { // 'MZ' header - result.threatScore += 80; - result.threatType = ThreatCategory.EXECUTABLE; - result.threatDetails = `Attachment contains executable code: ${filename}`; - return; - } - // Check for VBA macro indicators in Office documents - if (this.options.blockMacros && this.likelyContainsMacros(attachment)) { - result.threatScore += 60; - result.threatType = ThreatCategory.MALICIOUS_MACRO; - result.threatDetails = `Attachment appears to contain macros: ${filename}`; - } - } - /** - * Apply custom rules (runtime-configured patterns) to the email. - * These stay in TS because they are configured at runtime. - * @param email The email to check - * @param result The scan result to update - */ - applyCustomRules(email, result) { - if (!this.options.customRules.length) { - return; - } - const textsToCheck = []; - if (email.subject) - textsToCheck.push(email.subject); - if (email.text) - textsToCheck.push(email.text); - if (email.html) - textsToCheck.push(email.html); - for (const rule of this.options.customRules) { - const pattern = rule.pattern instanceof RegExp ? rule.pattern : new RegExp(rule.pattern, 'i'); - for (const text of textsToCheck) { - if (pattern.test(text)) { - result.threatScore += rule.score; - result.threatType = rule.type; - result.threatDetails = rule.description; - return; - } - } - } - } - /** - * Extract text from a binary buffer for scanning - * @param buffer Binary content - * @returns Extracted text (may be partial) - */ - extractTextFromBuffer(buffer) { - try { - // Limit the amount we convert to avoid memory issues - const sampleSize = Math.min(buffer.length, 100 * 1024); // 100KB max sample - const sample = buffer.slice(0, sampleSize); - // Try to convert to string, filtering out non-printable chars - return sample.toString('utf8') - .replace(/[\x00-\x09\x0B-\x1F\x7F-\x9F]/g, '') // Remove control chars - .replace(/\uFFFD/g, ''); // Remove replacement char - } - catch (error) { - logger.log('warn', `Error extracting text from buffer: ${error.message}`); - return ''; - } - } - /** - * Check if an Office document likely contains macros - * @param attachment The attachment to check - * @returns Whether the file likely contains macros - */ - likelyContainsMacros(attachment) { - const content = this.extractTextFromBuffer(attachment.content); - const macroIndicators = [ - /vbaProject\.bin/i, - /Microsoft VBA/i, - /\bVBA\b/, - /Auto_Open/i, - /AutoExec/i, - /DocumentOpen/i, - /AutoOpen/i, - /\bExecute\(/i, - /\bShell\(/i, - /\bCreateObject\(/i - ]; - for (const indicator of macroIndicators) { - if (indicator.test(content)) { - return true; - } - } - return false; - } - /** - * Log a high threat finding to the security logger - * @param email The email containing the threat - * @param result The scan result - */ - logHighThreatFound(email, result) { - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.ERROR, - type: SecurityEventType.MALWARE, - message: `High threat content detected in email from ${email.from} to ${email.to.join(', ')}`, - details: { - messageId: email.getMessageId(), - threatType: result.threatType, - threatDetails: result.threatDetails, - threatScore: result.threatScore, - scannedElements: result.scannedElements, - subject: email.subject - }, - success: false, - domain: email.getFromDomain() - }); - } - /** - * Log a threat finding to the security logger - * @param email The email containing the threat - * @param result The scan result - */ - logThreatFound(email, result) { - SecurityLogger.getInstance().logEvent({ - level: SecurityLogLevel.WARN, - type: SecurityEventType.SPAM, - message: `Suspicious content detected in email from ${email.from} to ${email.to.join(', ')}`, - details: { - messageId: email.getMessageId(), - threatType: result.threatType, - threatDetails: result.threatDetails, - threatScore: result.threatScore, - scannedElements: result.scannedElements, - subject: email.subject - }, - success: false, - domain: email.getFromDomain() - }); - } - /** - * Get threat level description based on score - * @param score Threat score - * @returns Threat level description - */ - static getThreatLevel(score) { - if (score < 20) { - return 'none'; - } - else if (score < 40) { - return 'low'; - } - else if (score < 70) { - return 'medium'; - } - else { - return 'high'; - } - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5jb250ZW50c2Nhbm5lci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3RzL3NlY3VyaXR5L2NsYXNzZXMuY29udGVudHNjYW5uZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxLQUFLLE9BQU8sTUFBTSxlQUFlLENBQUM7QUFDekMsT0FBTyxLQUFLLEtBQUssTUFBTSxhQUFhLENBQUM7QUFDckMsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGNBQWMsQ0FBQztBQUN0QyxPQUFPLEVBQUUsS0FBSyxFQUFFLE1BQU0sK0JBQStCLENBQUM7QUFFdEQsT0FBTyxFQUFFLGNBQWMsRUFBRSxnQkFBZ0IsRUFBRSxpQkFBaUIsRUFBRSxNQUFNLDZCQUE2QixDQUFDO0FBQ2xHLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLGlDQUFpQyxDQUFDO0FBQ3JFLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxXQUFXLENBQUM7QUFxQ3JDOztHQUVHO0FBQ0gsTUFBTSxDQUFOLElBQVksY0FXWDtBQVhELFdBQVksY0FBYztJQUN4QiwrQkFBYSxDQUFBO0lBQ2IsdUNBQXFCLENBQUE7SUFDckIscUNBQW1CLENBQUE7SUFDbkIsMkNBQXlCLENBQUE7SUFDekIscURBQW1DLENBQUE7SUFDbkMscURBQW1DLENBQUE7SUFDbkMsNkJBQVcsQ0FBQTtJQUNYLG1EQUFpQyxDQUFBO0lBQ2pDLDZEQUEyQyxDQUFBO0lBQzNDLDZDQUEyQixDQUFBO0FBQzdCLENBQUMsRUFYVyxjQUFjLEtBQWQsY0FBYyxRQVd6QjtBQUVEOztHQUVHO0FBQ0gsTUFBTSxPQUFPLGNBQWM7SUFDakIsTUFBTSxDQUFDLFFBQVEsQ0FBaUI7SUFDaEMsU0FBUyxDQUFnQztJQUN6QyxPQUFPLENBQW1DO0lBRWxEOztPQUVHO0lBQ0ssTUFBTSxDQUFVLGVBQWUsR0FBcUM7UUFDMUUsWUFBWSxFQUFFLEtBQUs7UUFDbkIsUUFBUSxFQUFFLEVBQUUsR0FBRyxFQUFFLEdBQUcsRUFBRSxHQUFHLElBQUksRUFBRSxXQUFXO1FBQzFDLFdBQVcsRUFBRSxJQUFJO1FBQ2pCLFFBQVEsRUFBRSxJQUFJO1FBQ2QsZUFBZSxFQUFFLElBQUk7UUFDckIsdUJBQXVCLEVBQUUsRUFBRSxHQUFHLElBQUksR0FBRyxJQUFJLEVBQUUsT0FBTztRQUNsRCxtQkFBbUIsRUFBRSxJQUFJO1FBQ3pCLGdCQUFnQixFQUFFLElBQUk7UUFDdEIsV0FBVyxFQUFFLElBQUk7UUFDakIsV0FBVyxFQUFFLEVBQUU7UUFDZixjQUFjLEVBQUUsRUFBRSxFQUFFLGdEQUFnRDtRQUNwRSxlQUFlLEVBQUUsRUFBRSxDQUFFLHNEQUFzRDtLQUM1RSxDQUFDO0lBRUY7OztPQUdHO0lBQ0gsWUFBWSxVQUFrQyxFQUFFO1FBQzlDLDZCQUE2QjtRQUM3QixJQUFJLENBQUMsT0FBTyxHQUFHO1lBQ2IsR0FBRyxjQUFjLENBQUMsZUFBZTtZQUNqQyxHQUFHLE9BQU87U0FDWCxDQUFDO1FBRUYsbUJBQW1CO1FBQ25CLElBQUksQ0FBQyxTQUFTLEdBQUcsSUFBSSxRQUFRLENBQXNCO1lBQ2pELEdBQUcsRUFBRSxJQUFJLENBQUMsT0FBTyxDQUFDLFlBQVk7WUFDOUIsR0FBRyxFQUFFLElBQUksQ0FBQyxPQUFPLENBQUMsUUFBUTtTQUMzQixDQUFDLENBQUM7UUFFSCxNQUFNLENBQUMsR0FBRyxDQUFDLE1BQU0sRUFBRSw0QkFBNEIsQ0FBQyxDQUFDO0lBQ25ELENBQUM7SUFFRDs7OztPQUlHO0lBQ0ksTUFBTSxDQUFDLFdBQVcsQ0FBQyxVQUFrQyxFQUFFO1FBQzVELElBQUksQ0FBQyxjQUFjLENBQUMsUUFBUSxFQUFFLENBQUM7WUFDN0IsY0FBYyxDQUFDLFFBQVEsR0FBRyxJQUFJLGNBQWMsQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUN4RCxDQUFDO1FBQ0QsT0FBTyxjQUFjLENBQUMsUUFBUSxDQUFDO0lBQ2pDLENBQUM7SUFFRDs7Ozs7O09BTUc7SUFDSSxLQUFLLENBQUMsU0FBUyxDQUFDLEtBQVk7UUFDakMsSUFBSSxDQUFDO1lBQ0gsc0NBQXNDO1lBQ3RDLE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxLQUFLLENBQUMsQ0FBQztZQUU5QyxvQkFBb0I7WUFDcEIsTUFBTSxZQUFZLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxHQUFHLENBQUMsUUFBUSxDQUFDLENBQUM7WUFDbEQsSUFBSSxZQUFZLEVBQUUsQ0FBQztnQkFDakIsTUFBTSxDQUFDLEdBQUcsQ0FBQyxNQUFNLEVBQUUsc0NBQXNDLEtBQUssQ0FBQyxZQUFZLEVBQUUsRUFBRSxDQUFDLENBQUM7Z0JBQ2pGLE9BQU8sWUFBWSxDQUFDO1lBQ3RCLENBQUM7WUFFRCx1REFBdUQ7WUFDdkQsTUFBTSxNQUFNLEdBQUcsa0JBQWtCLENBQUMsV0FBVyxFQUFFLENBQUM7WUFDaEQsTUFBTSxVQUFVLEdBQUcsTUFBTSxNQUFNLENBQUMsV0FBVyxDQUFDO2dCQUMxQyxPQUFPLEVBQUUsSUFBSSxDQUFDLE9BQU8sQ0FBQyxXQUFXLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLFNBQVM7Z0JBQzdELFFBQVEsRUFBRSxJQUFJLENBQUMsT0FBTyxDQUFDLFFBQVEsQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsU0FBUztnQkFDeEQsUUFBUSxFQUFFLElBQUksQ0FBQyxPQUFPLENBQUMsUUFBUSxDQUFDLENBQUMsQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxTQUFTO2dCQUN4RCxlQUFlLEVBQUUsSUFBSSxDQUFDLE9BQU8sQ0FBQyxtQkFBbUI7b0JBQy9DLENBQUMsQ0FBQyxLQUFLLENBQUMsV0FBVyxFQUFFLEdBQUcsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxRQUFRLENBQUMsSUFBSSxFQUFFO29CQUMvQyxDQUFDLENBQUMsRUFBRTthQUNQLENBQUMsQ0FBQztZQUVILE1BQU0sTUFBTSxHQUFnQjtnQkFDMUIsT0FBTyxFQUFFLElBQUk7Z0JBQ2IsV0FBVyxFQUFFLFVBQVUsQ0FBQyxXQUFXO2dCQUNuQyxVQUFVLEVBQUUsVUFBVSxDQUFDLFVBQVUsSUFBSSxTQUFTO2dCQUM5QyxhQUFhLEVBQUUsVUFBVSxDQUFDLGFBQWEsSUFBSSxTQUFTO2dCQUNwRCxlQUFlLEVBQUUsVUFBVSxDQUFDLGVBQWU7Z0JBQzNDLFNBQVMsRUFBRSxJQUFJLENBQUMsR0FBRyxFQUFFO2FBQ3RCLENBQUM7WUFFRix1RUFBdUU7WUFDdkUsSUFBSSxJQUFJLENBQUMsT0FBTyxDQUFDLGVBQWUsSUFBSSxLQUFLLENBQUMsV0FBVyxFQUFFLE1BQU0sR0FBRyxDQUFDLEVBQUUsQ0FBQztnQkFDbEUsS0FBSyxNQUFNLFVBQVUsSUFBSSxLQUFLLENBQUMsV0FBVyxFQUFFLENBQUM7b0JBQzNDLElBQUksQ0FBQyxvQkFBb0IsQ0FBQyxVQUFVLEVBQUUsTUFBTSxDQUFDLENBQUM7Z0JBQ2hELENBQUM7WUFDSCxDQUFDO1lBRUQsbURBQW1EO1lBQ25ELElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxLQUFLLEVBQUUsTUFBTSxDQUFDLENBQUM7WUFFckMsd0RBQXdEO1lBQ3hELE1BQU0sQ0FBQyxPQUFPLEdBQUcsTUFBTSxDQUFDLFdBQVcsR0FBRyxJQUFJLENBQUMsT0FBTyxDQUFDLGNBQWMsQ0FBQztZQUVsRSxnQkFBZ0I7WUFDaEIsSUFBSSxDQUFDLFNBQVMsQ0FBQyxHQUFHLENBQUMsUUFBUSxFQUFFLE1BQU0sQ0FBQyxDQUFDO1lBRXJDLDJCQUEyQjtZQUMzQixJQUFJLE1BQU0sQ0FBQyxXQUFXLElBQUksSUFBSSxDQUFDLE9BQU8sQ0FBQyxlQUFlLEVBQUUsQ0FBQztnQkFDdkQsSUFBSSxDQUFDLGtCQUFrQixDQUFDLEtBQUssRUFBRSxNQUFNLENBQUMsQ0FBQztZQUN6QyxDQUFDO2lCQUFNLElBQUksQ0FBQyxNQUFNLENBQUMsT0FBTyxFQUFFLENBQUM7Z0JBQzNCLElBQUksQ0FBQyxjQUFjLENBQUMsS0FBSyxFQUFFLE1BQU0sQ0FBQyxDQUFDO1lBQ3JDLENBQUM7WUFFRCxPQUFPLE1BQU0sQ0FBQztRQUNoQixDQUFDO1FBQUMsT0FBTyxLQUFLLEVBQUUsQ0FBQztZQUNmLE1BQU0sQ0FBQyxHQUFHLENBQUMsT0FBTyxFQUFFLHlCQUF5QixLQUFLLENBQUMsT0FBTyxFQUFFLEVBQUU7Z0JBQzVELFNBQVMsRUFBRSxLQUFLLENBQUMsWUFBWSxFQUFFO2dCQUMvQixLQUFLLEVBQUUsS0FBSyxDQUFDLEtBQUs7YUFDbkIsQ0FBQyxDQUFDO1lBRUgsOENBQThDO1lBQzlDLE9BQU87Z0JBQ0wsT0FBTyxFQUFFLElBQUk7Z0JBQ2IsV0FBVyxFQUFFLENBQUM7Z0JBQ2QsZUFBZSxFQUFFLENBQUMsT0FBTyxDQUFDO2dCQUMxQixTQUFTLEVBQUUsSUFBSSxDQUFDLEdBQUcsRUFBRTtnQkFDckIsVUFBVSxFQUFFLFlBQVk7Z0JBQ3hCLGFBQWEsRUFBRSxlQUFlLEtBQUssQ0FBQyxPQUFPLEVBQUU7YUFDOUMsQ0FBQztRQUNKLENBQUM7SUFDSCxDQUFDO0lBRUQ7Ozs7T0FJRztJQUNLLGdCQUFnQixDQUFDLEtBQVk7UUFDbkMsOEJBQThCO1FBQzlCLElBQUksS0FBSyxDQUFDLFlBQVksRUFBRSxFQUFFLENBQUM7WUFDekIsT0FBTyxTQUFTLEtBQUssQ0FBQyxZQUFZLEVBQUUsRUFBRSxDQUFDO1FBQ3pDLENBQUM7UUFFRCxvQ0FBb0M7UUFDcEMsTUFBTSxhQUFhLEdBQUc7WUFDcEIsS0FBSyxDQUFDLElBQUk7WUFDVixLQUFLLENBQUMsT0FBTyxJQUFJLEVBQUU7WUFDbkIsS0FBSyxDQUFDLElBQUksRUFBRSxTQUFTLENBQUMsQ0FBQyxFQUFFLElBQUksQ0FBQyxJQUFJLEVBQUU7WUFDcEMsS0FBSyxDQUFDLElBQUksRUFBRSxTQUFTLENBQUMsQ0FBQyxFQUFFLElBQUksQ0FBQyxJQUFJLEVBQUU7WUFDcEMsS0FBSyxDQUFDLFdBQVcsRUFBRSxNQUFNLElBQUksQ0FBQztTQUMvQixDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUVaLE9BQU8sU0FBUyxPQUFPLENBQUMsTUFBTSxDQUFDLFVBQVUsQ0FBQyxRQUFRLENBQUMsQ0FBQyxNQUFNLENBQUMsYUFBYSxDQUFDLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxFQUFFLENBQUM7SUFDNUYsQ0FBQztJQUVEOzs7OztPQUtHO0lBQ0ssb0JBQW9CLENBQUMsVUFBdUIsRUFBRSxNQUFtQjtRQUN2RSxJQUFJLENBQUMsVUFBVSxDQUFDLE9BQU8sRUFBRSxDQUFDO1lBQ3hCLE9BQU87UUFDVCxDQUFDO1FBRUQseUJBQXlCO1FBQ3pCLElBQUksVUFBVSxDQUFDLE9BQU8sQ0FBQyxNQUFNLEdBQUcsSUFBSSxDQUFDLE9BQU8sQ0FBQyx1QkFBdUIsRUFBRSxDQUFDO1lBQ3JFLE9BQU87UUFDVCxDQUFDO1FBRUQsTUFBTSxRQUFRLEdBQUcsVUFBVSxDQUFDLFFBQVEsQ0FBQyxXQUFXLEVBQUUsQ0FBQztRQUVuRCxnRkFBZ0Y7UUFDaEYsSUFBSSxVQUFVLENBQUMsT0FBTyxDQUFDLE1BQU0sR0FBRyxFQUFFO1lBQzlCLFVBQVUsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLEtBQUssSUFBSTtZQUM5QixVQUFVLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxLQUFLLElBQUksRUFBRSxDQUFDLENBQUMsY0FBYztZQUNsRCxNQUFNLENBQUMsV0FBVyxJQUFJLEVBQUUsQ0FBQztZQUN6QixNQUFNLENBQUMsVUFBVSxHQUFHLGNBQWMsQ0FBQyxVQUFVLENBQUM7WUFDOUMsTUFBTSxDQUFDLGFBQWEsR0FBRyx3Q0FBd0MsUUFBUSxFQUFFLENBQUM7WUFDMUUsT0FBTztRQUNULENBQUM7UUFFRCxxREFBcUQ7UUFDckQsSUFBSSxJQUFJLENBQUMsT0FBTyxDQUFDLFdBQVcsSUFBSSxJQUFJLENBQUMsb0JBQW9CLENBQUMsVUFBVSxDQUFDLEVBQUUsQ0FBQztZQUN0RSxNQUFNLENBQUMsV0FBVyxJQUFJLEVBQUUsQ0FBQztZQUN6QixNQUFNLENBQUMsVUFBVSxHQUFHLGNBQWMsQ0FBQyxlQUFlLENBQUM7WUFDbkQsTUFBTSxDQUFDLGFBQWEsR0FBRyx5Q0FBeUMsUUFBUSxFQUFFLENBQUM7UUFDN0UsQ0FBQztJQUNILENBQUM7SUFFRDs7Ozs7T0FLRztJQUNLLGdCQUFnQixDQUFDLEtBQVksRUFBRSxNQUFtQjtRQUN4RCxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxXQUFXLENBQUMsTUFBTSxFQUFFLENBQUM7WUFDckMsT0FBTztRQUNULENBQUM7UUFFRCxNQUFNLFlBQVksR0FBYSxFQUFFLENBQUM7UUFDbEMsSUFBSSxLQUFLLENBQUMsT0FBTztZQUFFLFlBQVksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBQ3BELElBQUksS0FBSyxDQUFDLElBQUk7WUFBRSxZQUFZLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUM5QyxJQUFJLEtBQUssQ0FBQyxJQUFJO1lBQUUsWUFBWSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUM7UUFFOUMsS0FBSyxNQUFNLElBQUksSUFBSSxJQUFJLENBQUMsT0FBTyxDQUFDLFdBQVcsRUFBRSxDQUFDO1lBQzVDLE1BQU0sT0FBTyxHQUFHLElBQUksQ0FBQyxPQUFPLFlBQVksTUFBTSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxJQUFJLE1BQU0sQ0FBQyxJQUFJLENBQUMsT0FBTyxFQUFFLEdBQUcsQ0FBQyxDQUFDO1lBQzlGLEtBQUssTUFBTSxJQUFJLElBQUksWUFBWSxFQUFFLENBQUM7Z0JBQ2hDLElBQUksT0FBTyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDO29CQUN2QixNQUFNLENBQUMsV0FBVyxJQUFJLElBQUksQ0FBQyxLQUFLLENBQUM7b0JBQ2pDLE1BQU0sQ0FBQyxVQUFVLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQztvQkFDOUIsTUFBTSxDQUFDLGFBQWEsR0FBRyxJQUFJLENBQUMsV0FBVyxDQUFDO29CQUN4QyxPQUFPO2dCQUNULENBQUM7WUFDSCxDQUFDO1FBQ0gsQ0FBQztJQUNILENBQUM7SUFFRDs7OztPQUlHO0lBQ0sscUJBQXFCLENBQUMsTUFBYztRQUMxQyxJQUFJLENBQUM7WUFDSCxxREFBcUQ7WUFDckQsTUFBTSxVQUFVLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQyxNQUFNLENBQUMsTUFBTSxFQUFFLEdBQUcsR0FBRyxJQUFJLENBQUMsQ0FBQyxDQUFDLG1CQUFtQjtZQUMzRSxNQUFNLE1BQU0sR0FBRyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsRUFBRSxVQUFVLENBQUMsQ0FBQztZQUUzQyw4REFBOEQ7WUFDOUQsT0FBTyxNQUFNLENBQUMsUUFBUSxDQUFDLE1BQU0sQ0FBQztpQkFDM0IsT0FBTyxDQUFDLGdDQUFnQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLHVCQUF1QjtpQkFDckUsT0FBTyxDQUFDLFNBQVMsRUFBRSxFQUFFLENBQUMsQ0FBQyxDQUFDLDBCQUEwQjtRQUN2RCxDQUFDO1FBQUMsT0FBTyxLQUFLLEVBQUUsQ0FBQztZQUNmLE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLHNDQUFzQyxLQUFLLENBQUMsT0FBTyxFQUFFLENBQUMsQ0FBQztZQUMxRSxPQUFPLEVBQUUsQ0FBQztRQUNaLENBQUM7SUFDSCxDQUFDO0lBRUQ7Ozs7T0FJRztJQUNLLG9CQUFvQixDQUFDLFVBQXVCO1FBQ2xELE1BQU0sT0FBTyxHQUFHLElBQUksQ0FBQyxxQkFBcUIsQ0FBQyxVQUFVLENBQUMsT0FBTyxDQUFDLENBQUM7UUFDL0QsTUFBTSxlQUFlLEdBQUc7WUFDdEIsa0JBQWtCO1lBQ2xCLGdCQUFnQjtZQUNoQixTQUFTO1lBQ1QsWUFBWTtZQUNaLFdBQVc7WUFDWCxlQUFlO1lBQ2YsV0FBVztZQUNYLGNBQWM7WUFDZCxZQUFZO1lBQ1osbUJBQW1CO1NBQ3BCLENBQUM7UUFFRixLQUFLLE1BQU0sU0FBUyxJQUFJLGVBQWUsRUFBRSxDQUFDO1lBQ3hDLElBQUksU0FBUyxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsRUFBRSxDQUFDO2dCQUM1QixPQUFPLElBQUksQ0FBQztZQUNkLENBQUM7UUFDSCxDQUFDO1FBRUQsT0FBTyxLQUFLLENBQUM7SUFDZixDQUFDO0lBRUQ7Ozs7T0FJRztJQUNLLGtCQUFrQixDQUFDLEtBQVksRUFBRSxNQUFtQjtRQUMxRCxjQUFjLENBQUMsV0FBVyxFQUFFLENBQUMsUUFBUSxDQUFDO1lBQ3BDLEtBQUssRUFBRSxnQkFBZ0IsQ0FBQyxLQUFLO1lBQzdCLElBQUksRUFBRSxpQkFBaUIsQ0FBQyxPQUFPO1lBQy9CLE9BQU8sRUFBRSw4Q0FBOEMsS0FBSyxDQUFDLElBQUksT0FBTyxLQUFLLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsRUFBRTtZQUM3RixPQUFPLEVBQUU7Z0JBQ1AsU0FBUyxFQUFFLEtBQUssQ0FBQyxZQUFZLEVBQUU7Z0JBQy9CLFVBQVUsRUFBRSxNQUFNLENBQUMsVUFBVTtnQkFDN0IsYUFBYSxFQUFFLE1BQU0sQ0FBQyxhQUFhO2dCQUNuQyxXQUFXLEVBQUUsTUFBTSxDQUFDLFdBQVc7Z0JBQy9CLGVBQWUsRUFBRSxNQUFNLENBQUMsZUFBZTtnQkFDdkMsT0FBTyxFQUFFLEtBQUssQ0FBQyxPQUFPO2FBQ3ZCO1lBQ0QsT0FBTyxFQUFFLEtBQUs7WUFDZCxNQUFNLEVBQUUsS0FBSyxDQUFDLGFBQWEsRUFBRTtTQUM5QixDQUFDLENBQUM7SUFDTCxDQUFDO0lBRUQ7Ozs7T0FJRztJQUNLLGNBQWMsQ0FBQyxLQUFZLEVBQUUsTUFBbUI7UUFDdEQsY0FBYyxDQUFDLFdBQVcsRUFBRSxDQUFDLFFBQVEsQ0FBQztZQUNwQyxLQUFLLEVBQUUsZ0JBQWdCLENBQUMsSUFBSTtZQUM1QixJQUFJLEVBQUUsaUJBQWlCLENBQUMsSUFBSTtZQUM1QixPQUFPLEVBQUUsNkNBQTZDLEtBQUssQ0FBQyxJQUFJLE9BQU8sS0FBSyxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLEVBQUU7WUFDNUYsT0FBTyxFQUFFO2dCQUNQLFNBQVMsRUFBRSxLQUFLLENBQUMsWUFBWSxFQUFFO2dCQUMvQixVQUFVLEVBQUUsTUFBTSxDQUFDLFVBQVU7Z0JBQzdCLGFBQWEsRUFBRSxNQUFNLENBQUMsYUFBYTtnQkFDbkMsV0FBVyxFQUFFLE1BQU0sQ0FBQyxXQUFXO2dCQUMvQixlQUFlLEVBQUUsTUFBTSxDQUFDLGVBQWU7Z0JBQ3ZDLE9BQU8sRUFBRSxLQUFLLENBQUMsT0FBTzthQUN2QjtZQUNELE9BQU8sRUFBRSxLQUFLO1lBQ2QsTUFBTSxFQUFFLEtBQUssQ0FBQyxhQUFhLEVBQUU7U0FDOUIsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztJQUVEOzs7O09BSUc7SUFDSSxNQUFNLENBQUMsY0FBYyxDQUFDLEtBQWE7UUFDeEMsSUFBSSxLQUFLLEdBQUcsRUFBRSxFQUFFLENBQUM7WUFDZixPQUFPLE1BQU0sQ0FBQztRQUNoQixDQUFDO2FBQU0sSUFBSSxLQUFLLEdBQUcsRUFBRSxFQUFFLENBQUM7WUFDdEIsT0FBTyxLQUFLLENBQUM7UUFDZixDQUFDO2FBQU0sSUFBSSxLQUFLLEdBQUcsRUFBRSxFQUFFLENBQUM7WUFDdEIsT0FBTyxRQUFRLENBQUM7UUFDbEIsQ0FBQzthQUFNLENBQUM7WUFDTixPQUFPLE1BQU0sQ0FBQztRQUNoQixDQUFDO0lBQ0gsQ0FBQyJ9 \ No newline at end of file diff --git a/dist_ts/security/classes.ipreputationchecker.d.ts b/dist_ts/security/classes.ipreputationchecker.d.ts deleted file mode 100644 index 73a7c6d..0000000 --- a/dist_ts/security/classes.ipreputationchecker.d.ts +++ /dev/null @@ -1,73 +0,0 @@ -/** - * Reputation check result information - */ -export interface IReputationResult { - score: number; - isSpam: boolean; - isProxy: boolean; - isTor: boolean; - isVPN: boolean; - country?: string; - asn?: string; - org?: string; - blacklists?: string[]; - timestamp: number; - error?: string; -} -/** - * Reputation threshold scores - */ -export declare enum ReputationThreshold { - HIGH_RISK = 20,// Score below this is considered high risk - MEDIUM_RISK = 50,// Score below this is considered medium risk - LOW_RISK = 80 -} -/** - * IP type classifications - */ -export declare enum IPType { - RESIDENTIAL = "residential", - DATACENTER = "datacenter", - PROXY = "proxy", - TOR = "tor", - VPN = "vpn", - UNKNOWN = "unknown" -} -/** - * Options for the IP Reputation Checker - */ -export interface IIPReputationOptions { - maxCacheSize?: number; - cacheTTL?: number; - dnsblServers?: string[]; - highRiskThreshold?: number; - mediumRiskThreshold?: number; - lowRiskThreshold?: number; - enableLocalCache?: boolean; - enableDNSBL?: boolean; - enableIPInfo?: boolean; -} -/** - * IP reputation checker — delegates DNSBL lookups to the Rust security bridge. - * Retains LRU caching and disk persistence in TypeScript. - */ -export declare class IPReputationChecker { - private static instance; - private reputationCache; - private options; - private storageManager?; - private static readonly DEFAULT_OPTIONS; - constructor(options?: IIPReputationOptions, storageManager?: any); - static getInstance(options?: IIPReputationOptions, storageManager?: any): IPReputationChecker; - /** - * Check an IP address's reputation via the Rust bridge - */ - checkReputation(ip: string): Promise; - private createErrorResult; - private isValidIPAddress; - private logReputationCheck; - private saveCache; - private loadCache; - static getRiskLevel(score: number): 'high' | 'medium' | 'low' | 'trusted'; - updateStorageManager(storageManager: any): void; -} diff --git a/dist_ts/security/classes.ipreputationchecker.js b/dist_ts/security/classes.ipreputationchecker.js deleted file mode 100644 index dd1ce60..0000000 --- a/dist_ts/security/classes.ipreputationchecker.js +++ /dev/null @@ -1,263 +0,0 @@ -import * as plugins from '../plugins.js'; -import * as paths from '../paths.js'; -import { logger } from '../logger.js'; -import { SecurityLogger, SecurityLogLevel, SecurityEventType } from './classes.securitylogger.js'; -import { RustSecurityBridge } from './classes.rustsecuritybridge.js'; -import { LRUCache } from 'lru-cache'; -/** - * Reputation threshold scores - */ -export var ReputationThreshold; -(function (ReputationThreshold) { - ReputationThreshold[ReputationThreshold["HIGH_RISK"] = 20] = "HIGH_RISK"; - ReputationThreshold[ReputationThreshold["MEDIUM_RISK"] = 50] = "MEDIUM_RISK"; - ReputationThreshold[ReputationThreshold["LOW_RISK"] = 80] = "LOW_RISK"; // Score below this is considered low risk (but not trusted) -})(ReputationThreshold || (ReputationThreshold = {})); -/** - * IP type classifications - */ -export var IPType; -(function (IPType) { - IPType["RESIDENTIAL"] = "residential"; - IPType["DATACENTER"] = "datacenter"; - IPType["PROXY"] = "proxy"; - IPType["TOR"] = "tor"; - IPType["VPN"] = "vpn"; - IPType["UNKNOWN"] = "unknown"; -})(IPType || (IPType = {})); -/** - * IP reputation checker — delegates DNSBL lookups to the Rust security bridge. - * Retains LRU caching and disk persistence in TypeScript. - */ -export class IPReputationChecker { - static instance; - reputationCache; - options; - storageManager; - static DEFAULT_OPTIONS = { - maxCacheSize: 10000, - cacheTTL: 24 * 60 * 60 * 1000, - dnsblServers: [], - highRiskThreshold: ReputationThreshold.HIGH_RISK, - mediumRiskThreshold: ReputationThreshold.MEDIUM_RISK, - lowRiskThreshold: ReputationThreshold.LOW_RISK, - enableLocalCache: true, - enableDNSBL: true, - enableIPInfo: true - }; - constructor(options = {}, storageManager) { - this.options = { - ...IPReputationChecker.DEFAULT_OPTIONS, - ...options - }; - this.storageManager = storageManager; - this.reputationCache = new LRUCache({ - max: this.options.maxCacheSize, - ttl: this.options.cacheTTL, - }); - if (this.options.enableLocalCache) { - this.loadCache().catch(error => { - logger.log('error', `Failed to load IP reputation cache during initialization: ${error.message}`); - }); - } - } - static getInstance(options = {}, storageManager) { - if (!IPReputationChecker.instance) { - IPReputationChecker.instance = new IPReputationChecker(options, storageManager); - } - return IPReputationChecker.instance; - } - /** - * Check an IP address's reputation via the Rust bridge - */ - async checkReputation(ip) { - try { - if (!this.isValidIPAddress(ip)) { - logger.log('warn', `Invalid IP address format: ${ip}`); - return this.createErrorResult(ip, 'Invalid IP address format'); - } - // Check cache first - const cachedResult = this.reputationCache.get(ip); - if (cachedResult) { - logger.log('info', `Using cached reputation data for IP ${ip}`, { - score: cachedResult.score, - isSpam: cachedResult.isSpam - }); - return cachedResult; - } - // Delegate to Rust bridge - const bridge = RustSecurityBridge.getInstance(); - const rustResult = await bridge.checkIpReputation(ip); - const result = { - score: rustResult.score, - isSpam: rustResult.listed_count > 0, - isProxy: rustResult.ip_type === 'proxy', - isTor: rustResult.ip_type === 'tor', - isVPN: rustResult.ip_type === 'vpn', - blacklists: rustResult.dnsbl_results - .filter(d => d.listed) - .map(d => d.server), - timestamp: Date.now(), - }; - this.reputationCache.set(ip, result); - if (this.options.enableLocalCache) { - this.saveCache().catch(error => { - logger.log('error', `Failed to save IP reputation cache: ${error.message}`); - }); - } - this.logReputationCheck(ip, result); - return result; - } - catch (error) { - logger.log('error', `Error checking IP reputation for ${ip}: ${error.message}`, { - ip, - stack: error.stack - }); - const errorResult = this.createErrorResult(ip, error.message); - // Cache error results to avoid repeated failing lookups - this.reputationCache.set(ip, errorResult); - return errorResult; - } - } - createErrorResult(ip, errorMessage) { - return { - score: 50, - isSpam: false, - isProxy: false, - isTor: false, - isVPN: false, - timestamp: Date.now(), - error: errorMessage - }; - } - isValidIPAddress(ip) { - const ipv4Pattern = /^((25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)$/; - return ipv4Pattern.test(ip); - } - logReputationCheck(ip, result) { - let logLevel = SecurityLogLevel.INFO; - if (result.score < this.options.highRiskThreshold) { - logLevel = SecurityLogLevel.WARN; - } - SecurityLogger.getInstance().logEvent({ - level: logLevel, - type: SecurityEventType.IP_REPUTATION, - message: `IP reputation check ${result.isSpam ? 'flagged spam' : 'completed'} for ${ip}`, - ipAddress: ip, - details: { - score: result.score, - isSpam: result.isSpam, - isProxy: result.isProxy, - isTor: result.isTor, - isVPN: result.isVPN, - country: result.country, - blacklists: result.blacklists - }, - success: !result.isSpam - }); - } - async saveCache() { - try { - const entries = Array.from(this.reputationCache.entries()).map(([ip, data]) => ({ - ip, - data - })); - if (entries.length === 0) { - return; - } - const cacheData = JSON.stringify(entries); - if (this.storageManager) { - await this.storageManager.set('/security/ip-reputation-cache.json', cacheData); - logger.log('info', `Saved ${entries.length} IP reputation cache entries to StorageManager`); - } - else { - const cacheDir = plugins.path.join(paths.dataDir, 'security'); - await plugins.smartfs.directory(cacheDir).recursive().create(); - const cacheFile = plugins.path.join(cacheDir, 'ip_reputation_cache.json'); - await plugins.smartfs.file(cacheFile).write(cacheData); - logger.log('info', `Saved ${entries.length} IP reputation cache entries to disk`); - } - } - catch (error) { - logger.log('error', `Failed to save IP reputation cache: ${error.message}`); - } - } - async loadCache() { - try { - let cacheData = null; - let fromFilesystem = false; - if (this.storageManager) { - try { - cacheData = await this.storageManager.get('/security/ip-reputation-cache.json'); - if (!cacheData) { - const cacheFile = plugins.path.join(paths.dataDir, 'security', 'ip_reputation_cache.json'); - if (plugins.fs.existsSync(cacheFile)) { - logger.log('info', 'Migrating IP reputation cache from filesystem to StorageManager'); - cacheData = plugins.fs.readFileSync(cacheFile, 'utf8'); - fromFilesystem = true; - await this.storageManager.set('/security/ip-reputation-cache.json', cacheData); - logger.log('info', 'IP reputation cache migrated to StorageManager successfully'); - try { - plugins.fs.unlinkSync(cacheFile); - logger.log('info', 'Old cache file removed after migration'); - } - catch (deleteError) { - logger.log('warn', `Could not delete old cache file: ${deleteError.message}`); - } - } - } - } - catch (error) { - logger.log('error', `Error loading from StorageManager: ${error.message}`); - } - } - else { - const cacheFile = plugins.path.join(paths.dataDir, 'security', 'ip_reputation_cache.json'); - if (plugins.fs.existsSync(cacheFile)) { - cacheData = plugins.fs.readFileSync(cacheFile, 'utf8'); - fromFilesystem = true; - } - } - if (cacheData) { - const entries = JSON.parse(cacheData); - const now = Date.now(); - const validEntries = entries.filter(entry => { - const age = now - entry.data.timestamp; - return age < this.options.cacheTTL; - }); - for (const entry of validEntries) { - this.reputationCache.set(entry.ip, entry.data); - } - const source = fromFilesystem ? 'disk' : 'StorageManager'; - logger.log('info', `Loaded ${validEntries.length} IP reputation cache entries from ${source}`); - } - } - catch (error) { - logger.log('error', `Failed to load IP reputation cache: ${error.message}`); - } - } - static getRiskLevel(score) { - if (score < ReputationThreshold.HIGH_RISK) { - return 'high'; - } - else if (score < ReputationThreshold.MEDIUM_RISK) { - return 'medium'; - } - else if (score < ReputationThreshold.LOW_RISK) { - return 'low'; - } - else { - return 'trusted'; - } - } - updateStorageManager(storageManager) { - this.storageManager = storageManager; - logger.log('info', 'IPReputationChecker storage manager updated'); - if (this.options.enableLocalCache && this.reputationCache.size > 0) { - this.saveCache().catch(error => { - logger.log('error', `Failed to save cache to new storage manager: ${error.message}`); - }); - } - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5pcHJlcHV0YXRpb25jaGVja2VyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vdHMvc2VjdXJpdHkvY2xhc3Nlcy5pcHJlcHV0YXRpb25jaGVja2VyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sS0FBSyxPQUFPLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sS0FBSyxLQUFLLE1BQU0sYUFBYSxDQUFDO0FBQ3JDLE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSxjQUFjLENBQUM7QUFDdEMsT0FBTyxFQUFFLGNBQWMsRUFBRSxnQkFBZ0IsRUFBRSxpQkFBaUIsRUFBRSxNQUFNLDZCQUE2QixDQUFDO0FBQ2xHLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLGlDQUFpQyxDQUFDO0FBQ3JFLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxXQUFXLENBQUM7QUFtQnJDOztHQUVHO0FBQ0gsTUFBTSxDQUFOLElBQVksbUJBSVg7QUFKRCxXQUFZLG1CQUFtQjtJQUM3Qix3RUFBYyxDQUFBO0lBQ2QsNEVBQWdCLENBQUE7SUFDaEIsc0VBQWEsQ0FBQSxDQUFRLDREQUE0RDtBQUNuRixDQUFDLEVBSlcsbUJBQW1CLEtBQW5CLG1CQUFtQixRQUk5QjtBQUVEOztHQUVHO0FBQ0gsTUFBTSxDQUFOLElBQVksTUFPWDtBQVBELFdBQVksTUFBTTtJQUNoQixxQ0FBMkIsQ0FBQTtJQUMzQixtQ0FBeUIsQ0FBQTtJQUN6Qix5QkFBZSxDQUFBO0lBQ2YscUJBQVcsQ0FBQTtJQUNYLHFCQUFXLENBQUE7SUFDWCw2QkFBbUIsQ0FBQTtBQUNyQixDQUFDLEVBUFcsTUFBTSxLQUFOLE1BQU0sUUFPakI7QUFpQkQ7OztHQUdHO0FBQ0gsTUFBTSxPQUFPLG1CQUFtQjtJQUN0QixNQUFNLENBQUMsUUFBUSxDQUFzQjtJQUNyQyxlQUFlLENBQXNDO0lBQ3JELE9BQU8sQ0FBaUM7SUFDeEMsY0FBYyxDQUFPO0lBRXJCLE1BQU0sQ0FBVSxl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 \ No newline at end of file diff --git a/dist_ts/security/classes.rustsecuritybridge.d.ts b/dist_ts/security/classes.rustsecuritybridge.d.ts deleted file mode 100644 index e008e4d..0000000 --- a/dist_ts/security/classes.rustsecuritybridge.d.ts +++ /dev/null @@ -1,229 +0,0 @@ -interface IDkimVerificationResult { - is_valid: boolean; - domain: string | null; - selector: string | null; - status: string; - details: string | null; -} -interface ISpfResult { - result: string; - domain: string; - ip: string; - explanation: string | null; -} -interface IDmarcResult { - passed: boolean; - policy: string; - domain: string; - dkim_result: string; - spf_result: string; - action: string; - details: string | null; -} -interface IEmailSecurityResult { - dkim: IDkimVerificationResult[]; - spf: ISpfResult | null; - dmarc: IDmarcResult | null; -} -interface IValidationResult { - valid: boolean; - formatValid: boolean; - score: number; - error: string | null; -} -interface IBounceDetection { - bounce_type: string; - category: string; -} -interface IReputationResult { - ip: string; - score: number; - risk_level: string; - ip_type: string; - dnsbl_results: Array<{ - server: string; - listed: boolean; - response: string | null; - }>; - listed_count: number; - total_checked: number; -} -interface IContentScanResult { - threatScore: number; - threatType: string | null; - threatDetails: string | null; - scannedElements: string[]; -} -interface IVersionInfo { - bin: string; - core: string; - security: string; - smtp: string; -} -interface ISmtpServerConfig { - hostname: string; - ports: number[]; - securePort?: number; - tlsCertPem?: string; - tlsKeyPem?: string; - maxMessageSize?: number; - maxConnections?: number; - maxRecipients?: number; - connectionTimeoutSecs?: number; - dataTimeoutSecs?: number; - authEnabled?: boolean; - maxAuthFailures?: number; - socketTimeoutSecs?: number; - processingTimeoutSecs?: number; - rateLimits?: IRateLimitConfig; -} -interface IRateLimitConfig { - maxConnectionsPerIp?: number; - maxMessagesPerSender?: number; - maxAuthFailuresPerIp?: number; - windowSecs?: number; -} -interface IEmailData { - type: 'inline' | 'file'; - base64?: string; - path?: string; -} -interface IEmailReceivedEvent { - correlationId: string; - sessionId: string; - mailFrom: string; - rcptTo: string[]; - data: IEmailData; - remoteAddr: string; - clientHostname: string | null; - secure: boolean; - authenticatedUser: string | null; - securityResults: any | null; -} -interface IAuthRequestEvent { - correlationId: string; - sessionId: string; - username: string; - password: string; - remoteAddr: string; -} -/** - * Bridge between TypeScript and the Rust `mailer-bin` binary. - * - * Uses `@push.rocks/smartrust` for JSON-over-stdin/stdout IPC. - * Singleton — access via `RustSecurityBridge.getInstance()`. - */ -export declare class RustSecurityBridge { - private static instance; - private bridge; - private _running; - private constructor(); - /** Get or create the singleton instance. */ - static getInstance(): RustSecurityBridge; - /** Whether the Rust process is currently running and accepting commands. */ - get running(): boolean; - /** - * Spawn the Rust binary and wait for the ready signal. - * @returns `true` if the binary started successfully, `false` otherwise. - */ - start(): Promise; - /** Kill the Rust process. */ - stop(): Promise; - /** Ping the Rust process. */ - ping(): Promise; - /** Get version information for all Rust crates. */ - getVersion(): Promise; - /** Validate an email address. */ - validateEmail(email: string): Promise; - /** Detect bounce type from SMTP response / diagnostic code. */ - detectBounce(opts: { - smtpResponse?: string; - diagnosticCode?: string; - statusCode?: string; - }): Promise; - /** Scan email content for threats (phishing, spam, malware, etc.). */ - scanContent(opts: { - subject?: string; - textBody?: string; - htmlBody?: string; - attachmentNames?: string[]; - }): Promise; - /** Check IP reputation via DNSBL. */ - checkIpReputation(ip: string): Promise; - /** Verify DKIM signatures on a raw email message. */ - verifyDkim(rawMessage: string): Promise; - /** Sign an email with DKIM. */ - signDkim(opts: { - rawMessage: string; - domain: string; - selector?: string; - privateKey: string; - }): Promise<{ - header: string; - signedMessage: string; - }>; - /** Check SPF for a sender. */ - checkSpf(opts: { - ip: string; - heloDomain: string; - hostname?: string; - mailFrom: string; - }): Promise; - /** - * Compound email security verification: DKIM + SPF + DMARC in one IPC call. - * - * This is the preferred method for inbound email verification — it avoids - * 3 sequential round-trips and correctly passes raw mail-auth types internally. - */ - verifyEmail(opts: { - rawMessage: string; - ip: string; - heloDomain: string; - hostname?: string; - mailFrom: string; - }): Promise; - /** - * Start the Rust SMTP server. - * The server will listen on the configured ports and emit events for - * emailReceived and authRequest that must be handled by the caller. - */ - startSmtpServer(config: ISmtpServerConfig): Promise; - /** Stop the Rust SMTP server. */ - stopSmtpServer(): Promise; - /** - * Send the result of email processing back to the Rust SMTP server. - * This resolves a pending correlation-ID callback, allowing the Rust - * server to send the SMTP response to the client. - */ - sendEmailProcessingResult(opts: { - correlationId: string; - accepted: boolean; - smtpCode?: number; - smtpMessage?: string; - }): Promise; - /** - * Send the result of authentication validation back to the Rust SMTP server. - */ - sendAuthResult(opts: { - correlationId: string; - success: boolean; - message?: string; - }): Promise; - /** Update rate limit configuration at runtime. */ - configureRateLimits(config: IRateLimitConfig): Promise; - /** - * Register a handler for emailReceived events from the Rust SMTP server. - * These events fire when a complete email has been received and needs processing. - */ - onEmailReceived(handler: (data: IEmailReceivedEvent) => void): void; - /** - * Register a handler for authRequest events from the Rust SMTP server. - * The handler must call sendAuthResult() with the correlationId. - */ - onAuthRequest(handler: (data: IAuthRequestEvent) => void): void; - /** Remove an emailReceived event handler. */ - offEmailReceived(handler: (data: IEmailReceivedEvent) => void): void; - /** Remove an authRequest event handler. */ - offAuthRequest(handler: (data: IAuthRequestEvent) => void): void; -} -export type { IDkimVerificationResult, ISpfResult, IDmarcResult, IEmailSecurityResult, IValidationResult, IBounceDetection, IContentScanResult, IReputationResult as IRustReputationResult, IVersionInfo, ISmtpServerConfig, IRateLimitConfig, IEmailData, IEmailReceivedEvent, IAuthRequestEvent, }; diff --git a/dist_ts/security/classes.rustsecuritybridge.js b/dist_ts/security/classes.rustsecuritybridge.js deleted file mode 100644 index 349a922..0000000 --- a/dist_ts/security/classes.rustsecuritybridge.js +++ /dev/null @@ -1,204 +0,0 @@ -import * as plugins from '../plugins.js'; -import * as paths from '../paths.js'; -import { logger } from '../logger.js'; -// --------------------------------------------------------------------------- -// RustSecurityBridge — singleton wrapper around smartrust.RustBridge -// --------------------------------------------------------------------------- -/** - * Bridge between TypeScript and the Rust `mailer-bin` binary. - * - * Uses `@push.rocks/smartrust` for JSON-over-stdin/stdout IPC. - * Singleton — access via `RustSecurityBridge.getInstance()`. - */ -export class RustSecurityBridge { - static instance = null; - bridge; - _running = false; - constructor() { - this.bridge = new plugins.smartrust.RustBridge({ - binaryName: 'mailer-bin', - cliArgs: ['--management'], - requestTimeoutMs: 30_000, - readyTimeoutMs: 10_000, - localPaths: [ - plugins.path.join(paths.packageDir, 'dist_rust', 'mailer-bin'), - plugins.path.join(paths.packageDir, 'rust', 'target', 'release', 'mailer-bin'), - plugins.path.join(paths.packageDir, 'rust', 'target', 'debug', 'mailer-bin'), - ], - searchSystemPath: false, - }); - // Forward lifecycle events - this.bridge.on('ready', () => { - this._running = true; - logger.log('info', 'Rust security bridge is ready'); - }); - this.bridge.on('exit', (code, signal) => { - this._running = false; - logger.log('warn', `Rust security bridge exited (code=${code}, signal=${signal})`); - }); - this.bridge.on('stderr', (line) => { - logger.log('debug', `[rust-bridge] ${line}`); - }); - } - /** Get or create the singleton instance. */ - static getInstance() { - if (!RustSecurityBridge.instance) { - RustSecurityBridge.instance = new RustSecurityBridge(); - } - return RustSecurityBridge.instance; - } - /** Whether the Rust process is currently running and accepting commands. */ - get running() { - return this._running; - } - // ----------------------------------------------------------------------- - // Lifecycle - // ----------------------------------------------------------------------- - /** - * Spawn the Rust binary and wait for the ready signal. - * @returns `true` if the binary started successfully, `false` otherwise. - */ - async start() { - if (this._running) { - return true; - } - try { - const ok = await this.bridge.spawn(); - this._running = ok; - if (ok) { - logger.log('info', 'Rust security bridge started'); - } - else { - logger.log('warn', 'Rust security bridge failed to start (binary not found or timeout)'); - } - return ok; - } - catch (err) { - logger.log('error', `Failed to start Rust security bridge: ${err.message}`); - return false; - } - } - /** Kill the Rust process. */ - async stop() { - if (!this._running) { - return; - } - try { - this.bridge.kill(); - this._running = false; - logger.log('info', 'Rust security bridge stopped'); - } - catch (err) { - logger.log('error', `Error stopping Rust security bridge: ${err.message}`); - } - } - // ----------------------------------------------------------------------- - // Commands — thin typed wrappers over sendCommand - // ----------------------------------------------------------------------- - /** Ping the Rust process. */ - async ping() { - const res = await this.bridge.sendCommand('ping', {}); - return res?.pong === true; - } - /** Get version information for all Rust crates. */ - async getVersion() { - return this.bridge.sendCommand('version', {}); - } - /** Validate an email address. */ - async validateEmail(email) { - return this.bridge.sendCommand('validateEmail', { email }); - } - /** Detect bounce type from SMTP response / diagnostic code. */ - async detectBounce(opts) { - return this.bridge.sendCommand('detectBounce', opts); - } - /** Scan email content for threats (phishing, spam, malware, etc.). */ - async scanContent(opts) { - return this.bridge.sendCommand('scanContent', opts); - } - /** Check IP reputation via DNSBL. */ - async checkIpReputation(ip) { - return this.bridge.sendCommand('checkIpReputation', { ip }); - } - /** Verify DKIM signatures on a raw email message. */ - async verifyDkim(rawMessage) { - return this.bridge.sendCommand('verifyDkim', { rawMessage }); - } - /** Sign an email with DKIM. */ - async signDkim(opts) { - return this.bridge.sendCommand('signDkim', opts); - } - /** Check SPF for a sender. */ - async checkSpf(opts) { - return this.bridge.sendCommand('checkSpf', opts); - } - /** - * Compound email security verification: DKIM + SPF + DMARC in one IPC call. - * - * This is the preferred method for inbound email verification — it avoids - * 3 sequential round-trips and correctly passes raw mail-auth types internally. - */ - async verifyEmail(opts) { - return this.bridge.sendCommand('verifyEmail', opts); - } - // ----------------------------------------------------------------------- - // SMTP Server lifecycle - // ----------------------------------------------------------------------- - /** - * Start the Rust SMTP server. - * The server will listen on the configured ports and emit events for - * emailReceived and authRequest that must be handled by the caller. - */ - async startSmtpServer(config) { - const result = await this.bridge.sendCommand('startSmtpServer', config); - return result?.started === true; - } - /** Stop the Rust SMTP server. */ - async stopSmtpServer() { - await this.bridge.sendCommand('stopSmtpServer', {}); - } - /** - * Send the result of email processing back to the Rust SMTP server. - * This resolves a pending correlation-ID callback, allowing the Rust - * server to send the SMTP response to the client. - */ - async sendEmailProcessingResult(opts) { - await this.bridge.sendCommand('emailProcessingResult', opts); - } - /** - * Send the result of authentication validation back to the Rust SMTP server. - */ - async sendAuthResult(opts) { - await this.bridge.sendCommand('authResult', opts); - } - /** Update rate limit configuration at runtime. */ - async configureRateLimits(config) { - await this.bridge.sendCommand('configureRateLimits', config); - } - // ----------------------------------------------------------------------- - // Event registration — delegates to the underlying bridge EventEmitter - // ----------------------------------------------------------------------- - /** - * Register a handler for emailReceived events from the Rust SMTP server. - * These events fire when a complete email has been received and needs processing. - */ - onEmailReceived(handler) { - this.bridge.on('management:emailReceived', handler); - } - /** - * Register a handler for authRequest events from the Rust SMTP server. - * The handler must call sendAuthResult() with the correlationId. - */ - onAuthRequest(handler) { - this.bridge.on('management:authRequest', handler); - } - /** Remove an emailReceived event handler. */ - offEmailReceived(handler) { - this.bridge.off('management:emailReceived', handler); - } - /** Remove an authRequest event handler. */ - offAuthRequest(handler) { - this.bridge.off('management:authRequest', handler); - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5ydXN0c2VjdXJpdHlicmlkZ2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi90cy9zZWN1cml0eS9jbGFzc2VzLnJ1c3RzZWN1cml0eWJyaWRnZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssT0FBTyxNQUFNLGVBQWUsQ0FBQztBQUN6QyxPQUFPLEtBQUssS0FBSyxNQUFNLGFBQWEsQ0FBQztBQUNyQyxPQUFPLEVBQUUsTUFBTSxFQUFFLE1BQU0sY0FBYyxDQUFDO0FBcU50Qyw4RUFBOEU7QUFDOUUscUVBQXFFO0FBQ3JFLDhFQUE4RTtBQUU5RTs7Ozs7R0FLRztBQUNILE1BQU0sT0FBTyxrQkFBa0I7SUFDckIsTUFBTSxDQUFDLFFBQVEsR0FBOEIsSUFBSSxDQUFDO0lBRWxELE1BQU0sQ0FBcUU7SUFDM0UsUUFBUSxHQUFHLEtBQUssQ0FBQztJQUV6QjtRQUNFLElBQUksQ0FBQyxNQUFNLEdBQUcsSUFBSSxPQUFPLENBQUMsU0FBUyxDQUFDLFVBQVUsQ0FBa0I7WUFDOUQsVUFBVSxFQUFFLFlBQVk7WUFDeEIsT0FBTyxFQUFFLENBQUMsY0FBYyxDQUFDO1lBQ3pCLGdCQUFnQixFQUFFLE1BQU07WUFDeEIsY0FBYyxFQUFFLE1BQU07WUFDdEIsVUFBVSxFQUFFO2dCQUNWLE9BQU8sQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxVQUFVLEVBQUUsV0FBVyxFQUFFLFlBQVksQ0FBQztnQkFDOUQsT0FBTyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLFVBQVUsRUFBRSxNQUFNLEVBQUUsUUFBUSxFQUFFLFNBQVMsRUFBRSxZQUFZLENBQUM7Z0JBQzlFLE9BQU8sQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxVQUFVLEVBQUUsTUFBTSxFQUFFLFFBQVEsRUFBRSxPQUFPLEVBQUUsWUFBWSxDQUFDO2FBQzdFO1lBQ0QsZ0JBQWdCLEVBQUUsS0FBSztT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 \ No newline at end of file diff --git a/dist_ts/security/classes.securitylogger.d.ts b/dist_ts/security/classes.securitylogger.d.ts deleted file mode 100644 index 352c71b..0000000 --- a/dist_ts/security/classes.securitylogger.d.ts +++ /dev/null @@ -1,140 +0,0 @@ -/** - * Log level for security events - */ -export declare enum SecurityLogLevel { - INFO = "info", - WARN = "warn", - ERROR = "error", - CRITICAL = "critical" -} -/** - * Security event types for categorization - */ -export declare enum SecurityEventType { - AUTHENTICATION = "authentication", - ACCESS_CONTROL = "access_control", - EMAIL_VALIDATION = "email_validation", - EMAIL_PROCESSING = "email_processing", - EMAIL_FORWARDING = "email_forwarding", - EMAIL_DELIVERY = "email_delivery", - DKIM = "dkim", - SPF = "spf", - DMARC = "dmarc", - RATE_LIMIT = "rate_limit", - RATE_LIMITING = "rate_limiting", - SPAM = "spam", - MALWARE = "malware", - CONNECTION = "connection", - DATA_EXPOSURE = "data_exposure", - CONFIGURATION = "configuration", - IP_REPUTATION = "ip_reputation", - REJECTED_CONNECTION = "rejected_connection" -} -/** - * Security event interface - */ -export interface ISecurityEvent { - timestamp: number; - level: SecurityLogLevel; - type: SecurityEventType; - message: string; - details?: any; - ipAddress?: string; - userId?: string; - sessionId?: string; - emailId?: string; - domain?: string; - action?: string; - result?: string; - success?: boolean; -} -/** - * Security logger for enhanced security monitoring - */ -export declare class SecurityLogger { - private static instance; - private securityEvents; - private maxEventHistory; - private enableNotifications; - private constructor(); - /** - * Get singleton instance - */ - static getInstance(options?: { - maxEventHistory?: number; - enableNotifications?: boolean; - }): SecurityLogger; - /** - * Log a security event - * @param event The security event to log - */ - logEvent(event: Omit): void; - /** - * Get recent security events - * @param limit Maximum number of events to return - * @param filter Filter for specific event types - * @returns Recent security events - */ - getRecentEvents(limit?: number, filter?: { - level?: SecurityLogLevel; - type?: SecurityEventType; - fromTimestamp?: number; - toTimestamp?: number; - }): ISecurityEvent[]; - /** - * Get events by security level - * @param level The security level to filter by - * @param limit Maximum number of events to return - * @returns Security events matching the level - */ - getEventsByLevel(level: SecurityLogLevel, limit?: number): ISecurityEvent[]; - /** - * Get events by security type - * @param type The event type to filter by - * @param limit Maximum number of events to return - * @returns Security events matching the type - */ - getEventsByType(type: SecurityEventType, limit?: number): ISecurityEvent[]; - /** - * Get security events for a specific IP address - * @param ipAddress The IP address to filter by - * @param limit Maximum number of events to return - * @returns Security events for the IP address - */ - getEventsByIP(ipAddress: string, limit?: number): ISecurityEvent[]; - /** - * Get security events for a specific domain - * @param domain The domain to filter by - * @param limit Maximum number of events to return - * @returns Security events for the domain - */ - getEventsByDomain(domain: string, limit?: number): ISecurityEvent[]; - /** - * Send a notification for critical security events - * @param event The security event to notify about - * @private - */ - private sendNotification; - /** - * Clear event history - */ - clearEvents(): void; - /** - * Get statistical summary of security events - * @param timeWindow Optional time window in milliseconds - * @returns Summary of security events - */ - getEventsSummary(timeWindow?: number): { - total: number; - byLevel: Record; - byType: Record; - topIPs: Array<{ - ip: string; - count: number; - }>; - topDomains: Array<{ - domain: string; - count: number; - }>; - }; -} diff --git a/dist_ts/security/classes.securitylogger.js b/dist_ts/security/classes.securitylogger.js deleted file mode 100644 index ce24678..0000000 --- a/dist_ts/security/classes.securitylogger.js +++ /dev/null @@ -1,235 +0,0 @@ -import * as plugins from '../plugins.js'; -import { logger } from '../logger.js'; -/** - * Log level for security events - */ -export var SecurityLogLevel; -(function (SecurityLogLevel) { - SecurityLogLevel["INFO"] = "info"; - SecurityLogLevel["WARN"] = "warn"; - SecurityLogLevel["ERROR"] = "error"; - SecurityLogLevel["CRITICAL"] = "critical"; -})(SecurityLogLevel || (SecurityLogLevel = {})); -/** - * Security event types for categorization - */ -export var SecurityEventType; -(function (SecurityEventType) { - SecurityEventType["AUTHENTICATION"] = "authentication"; - SecurityEventType["ACCESS_CONTROL"] = "access_control"; - SecurityEventType["EMAIL_VALIDATION"] = "email_validation"; - SecurityEventType["EMAIL_PROCESSING"] = "email_processing"; - SecurityEventType["EMAIL_FORWARDING"] = "email_forwarding"; - SecurityEventType["EMAIL_DELIVERY"] = "email_delivery"; - SecurityEventType["DKIM"] = "dkim"; - SecurityEventType["SPF"] = "spf"; - SecurityEventType["DMARC"] = "dmarc"; - SecurityEventType["RATE_LIMIT"] = "rate_limit"; - SecurityEventType["RATE_LIMITING"] = "rate_limiting"; - SecurityEventType["SPAM"] = "spam"; - SecurityEventType["MALWARE"] = "malware"; - SecurityEventType["CONNECTION"] = "connection"; - SecurityEventType["DATA_EXPOSURE"] = "data_exposure"; - SecurityEventType["CONFIGURATION"] = "configuration"; - SecurityEventType["IP_REPUTATION"] = "ip_reputation"; - SecurityEventType["REJECTED_CONNECTION"] = "rejected_connection"; -})(SecurityEventType || (SecurityEventType = {})); -/** - * Security logger for enhanced security monitoring - */ -export class SecurityLogger { - static instance; - securityEvents = []; - maxEventHistory; - enableNotifications; - constructor(options) { - this.maxEventHistory = options?.maxEventHistory || 1000; - this.enableNotifications = options?.enableNotifications || false; - } - /** - * Get singleton instance - */ - static getInstance(options) { - if (!SecurityLogger.instance) { - SecurityLogger.instance = new SecurityLogger(options); - } - return SecurityLogger.instance; - } - /** - * Log a security event - * @param event The security event to log - */ - logEvent(event) { - const fullEvent = { - ...event, - timestamp: Date.now() - }; - // Store in memory buffer - this.securityEvents.push(fullEvent); - // Trim history if needed - if (this.securityEvents.length > this.maxEventHistory) { - this.securityEvents.shift(); - } - // Log to regular logger with appropriate level - switch (event.level) { - case SecurityLogLevel.INFO: - logger.log('info', `[SECURITY:${event.type}] ${event.message}`, event.details); - break; - case SecurityLogLevel.WARN: - logger.log('warn', `[SECURITY:${event.type}] ${event.message}`, event.details); - break; - case SecurityLogLevel.ERROR: - case SecurityLogLevel.CRITICAL: - logger.log('error', `[SECURITY:${event.type}] ${event.message}`, event.details); - // Send notification for critical events if enabled - if (event.level === SecurityLogLevel.CRITICAL && this.enableNotifications) { - this.sendNotification(fullEvent); - } - break; - } - } - /** - * Get recent security events - * @param limit Maximum number of events to return - * @param filter Filter for specific event types - * @returns Recent security events - */ - getRecentEvents(limit = 100, filter) { - let filteredEvents = this.securityEvents; - // Apply filters - if (filter) { - if (filter.level) { - filteredEvents = filteredEvents.filter(event => event.level === filter.level); - } - if (filter.type) { - filteredEvents = filteredEvents.filter(event => event.type === filter.type); - } - if (filter.fromTimestamp) { - filteredEvents = filteredEvents.filter(event => event.timestamp >= filter.fromTimestamp); - } - if (filter.toTimestamp) { - filteredEvents = filteredEvents.filter(event => event.timestamp <= filter.toTimestamp); - } - } - // Return most recent events up to limit - return filteredEvents - .sort((a, b) => b.timestamp - a.timestamp) - .slice(0, limit); - } - /** - * Get events by security level - * @param level The security level to filter by - * @param limit Maximum number of events to return - * @returns Security events matching the level - */ - getEventsByLevel(level, limit = 100) { - return this.getRecentEvents(limit, { level }); - } - /** - * Get events by security type - * @param type The event type to filter by - * @param limit Maximum number of events to return - * @returns Security events matching the type - */ - getEventsByType(type, limit = 100) { - return this.getRecentEvents(limit, { type }); - } - /** - * Get security events for a specific IP address - * @param ipAddress The IP address to filter by - * @param limit Maximum number of events to return - * @returns Security events for the IP address - */ - getEventsByIP(ipAddress, limit = 100) { - return this.securityEvents - .filter(event => event.ipAddress === ipAddress) - .sort((a, b) => b.timestamp - a.timestamp) - .slice(0, limit); - } - /** - * Get security events for a specific domain - * @param domain The domain to filter by - * @param limit Maximum number of events to return - * @returns Security events for the domain - */ - getEventsByDomain(domain, limit = 100) { - return this.securityEvents - .filter(event => event.domain === domain) - .sort((a, b) => b.timestamp - a.timestamp) - .slice(0, limit); - } - /** - * Send a notification for critical security events - * @param event The security event to notify about - * @private - */ - sendNotification(event) { - // In a production environment, this would integrate with a notification service - // For now, we'll just log that we would send a notification - logger.log('error', `[SECURITY NOTIFICATION] ${event.message}`, { - ...event, - notificationSent: true - }); - // Future integration with alerting systems would go here - } - /** - * Clear event history - */ - clearEvents() { - this.securityEvents = []; - } - /** - * Get statistical summary of security events - * @param timeWindow Optional time window in milliseconds - * @returns Summary of security events - */ - getEventsSummary(timeWindow) { - // Filter by time window if provided - let events = this.securityEvents; - if (timeWindow) { - const cutoff = Date.now() - timeWindow; - events = events.filter(e => e.timestamp >= cutoff); - } - // Count by level - const byLevel = Object.values(SecurityLogLevel).reduce((acc, level) => { - acc[level] = events.filter(e => e.level === level).length; - return acc; - }, {}); - // Count by type - const byType = Object.values(SecurityEventType).reduce((acc, type) => { - acc[type] = events.filter(e => e.type === type).length; - return acc; - }, {}); - // Count by IP - const ipCounts = new Map(); - events.forEach(e => { - if (e.ipAddress) { - ipCounts.set(e.ipAddress, (ipCounts.get(e.ipAddress) || 0) + 1); - } - }); - // Count by domain - const domainCounts = new Map(); - events.forEach(e => { - if (e.domain) { - domainCounts.set(e.domain, (domainCounts.get(e.domain) || 0) + 1); - } - }); - // Sort and limit top entries - const topIPs = Array.from(ipCounts.entries()) - .map(([ip, count]) => ({ ip, count })) - .sort((a, b) => b.count - a.count) - .slice(0, 10); - const topDomains = Array.from(domainCounts.entries()) - .map(([domain, count]) => ({ domain, count })) - .sort((a, b) => b.count - a.count) - .slice(0, 10); - return { - total: events.length, - byLevel, - byType, - topIPs, - topDomains - }; - } -} -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xhc3Nlcy5zZWN1cml0eWxvZ2dlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3RzL3NlY3VyaXR5L2NsYXNzZXMuc2VjdXJpdHlsb2dnZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxLQUFLLE9BQU8sTUFBTSxlQUFlLENBQUM7QUFDekMsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGNBQWMsQ0FBQztBQUV0Qzs7R0FFRztBQUNILE1BQU0sQ0FBTixJQUFZLGdCQUtYO0FBTEQsV0FBWSxnQkFBZ0I7SUFDMUIsaUNBQWEsQ0FBQTtJQUNiLGlDQUFhLENBQUE7SUFDYixtQ0FBZSxDQUFBO0lBQ2YseUNBQXFCLENBQUE7QUFDdkIsQ0FBQyxFQUxXLGdCQUFnQixLQUFoQixnQkFBZ0IsUUFLM0I7QUFFRDs7R0FFRztBQUNILE1BQU0sQ0FBTixJQUFZLGlCQW1CWDtBQW5CRCxXQUFZLGlCQUFpQjtJQUMzQixzREFBaUMsQ0FBQTtJQUNqQyxzREFBaUMsQ0FBQTtJQUNqQywwREFBcUMsQ0FBQTtJQUNyQywwREFBcUMsQ0FBQTtJQUNyQywwREFBcUMsQ0FBQTtJQUNyQyxzREFBaUMsQ0FBQTtJQUNqQyxrQ0FBYSxDQUFBO0lBQ2IsZ0NBQVcsQ0FBQTtJQUNYLG9DQUFlLENBQUE7SUFDZiw4Q0FBeUIsQ0FBQTtJQUN6QixvREFBK0IsQ0FBQTtJQUMvQixrQ0FBYSxDQUFBO0lBQ2Isd0NBQW1CLENBQUE7SUFDbkIsOENBQXlCLENBQUE7SUFDekIsb0RBQStCLENBQUE7SUFDL0Isb0RBQStCLENBQUE7SUFDL0Isb0RBQStCLENBQUE7SUFDL0IsZ0VBQTJDLENBQUE7QUFDN0MsQ0FBQyxFQW5CVyxpQkFBaUIsS0FBakIsaUJBQWlCLFFBbUI1QjtBQXFCRDs7R0FFRztBQUNILE1BQU0sT0FBTyxjQUFjO0lBQ2pCLE1BQU0sQ0FBQyxRQUFRLENBQWlCO0lBQ2hDLGNBQWMsR0FBcUIsRUFBRSxDQUFDO0lBQ3RDLGVBQWUsQ0FB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 \ No newline at end of file diff --git a/dist_ts/security/index.d.ts b/dist_ts/security/index.d.ts deleted file mode 100644 index 88c9ad1..0000000 --- a/dist_ts/security/index.d.ts +++ /dev/null @@ -1,4 +0,0 @@ -export { SecurityLogger, SecurityLogLevel, SecurityEventType, type ISecurityEvent } from './classes.securitylogger.js'; -export { IPReputationChecker, ReputationThreshold, IPType, type IReputationResult, type IIPReputationOptions } from './classes.ipreputationchecker.js'; -export { ContentScanner, ThreatCategory, type IScanResult, type IContentScannerOptions } from './classes.contentscanner.js'; -export { RustSecurityBridge, type IDkimVerificationResult, type ISpfResult, type IDmarcResult, type IEmailSecurityResult, type IValidationResult, type IBounceDetection, type IRustReputationResult, type IVersionInfo, } from './classes.rustsecuritybridge.js'; diff --git a/dist_ts/security/index.js b/dist_ts/security/index.js deleted file mode 100644 index 6d1ca4a..0000000 --- a/dist_ts/security/index.js +++ /dev/null @@ -1,5 +0,0 @@ -export { SecurityLogger, SecurityLogLevel, SecurityEventType } from './classes.securitylogger.js'; -export { IPReputationChecker, ReputationThreshold, IPType } from './classes.ipreputationchecker.js'; -export { ContentScanner, ThreatCategory } from './classes.contentscanner.js'; -export { RustSecurityBridge, } from './classes.rustsecuritybridge.js'; -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi90cy9zZWN1cml0eS9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQ0wsY0FBYyxFQUNkLGdCQUFnQixFQUNoQixpQkFBaUIsRUFFbEIsTUFBTSw2QkFBNkIsQ0FBQztBQUVyQyxPQUFPLEVBQ0wsbUJBQW1CLEVBQ25CLG1CQUFtQixFQUNuQixNQUFNLEVBR1AsTUFBTSxrQ0FBa0MsQ0FBQztBQUUxQyxPQUFPLEVBQ0wsY0FBYyxFQUNkLGNBQWMsRUFHZixNQUFNLDZCQUE2QixDQUFDO0FBRXJDLE9BQU8sRUFDTCxrQkFBa0IsR0FTbkIsTUFBTSxpQ0FBaUMsQ0FBQyJ9 \ No newline at end of file diff --git a/npmextra.json b/npmextra.json index 6fc225e..7df3b82 100644 --- a/npmextra.json +++ b/npmextra.json @@ -8,9 +8,20 @@ "@git.zone/cli": { "release": { "registries": [ - "https://verdaccio.lossless.digital" + "https://verdaccio.lossless.digital", + "https://registry.npmjs.org" ], "accessLevel": "public" + }, + "projectType": "npm", + "module": { + "githost": "code.foss.global", + "gitscope": "push.rocks", + "gitrepo": "smartmta", + "description": "an mta implementation as TypeScript package, with network side implemented in rust", + "npmPackagename": "@push.rocks/smartmta", + "license": "MIT" } - } + }, + "@ship.zone/szci": {} } \ No newline at end of file diff --git a/ts/00_commitinfo_data.ts b/ts/00_commitinfo_data.ts index 7e87997..64312dc 100644 --- a/ts/00_commitinfo_data.ts +++ b/ts/00_commitinfo_data.ts @@ -3,6 +3,6 @@ */ export const commitinfo = { name: '@push.rocks/smartmta', - version: '2.3.0', + version: '2.3.1', description: 'A high-performance, enterprise-grade Mail Transfer Agent (MTA) built from scratch in TypeScript with Rust acceleration.' }