- Implement CMD-03 tests for RCPT TO command, validating recipient addresses, handling multiple recipients, and enforcing command sequence. - Implement CMD-04 tests for DATA command, ensuring proper email content transmission, handling of dot-stuffing, large messages, and correct command sequence. - Implement CMD-13 tests for QUIT command, verifying graceful connection termination and idempotency. - Implement CM-01 tests for TLS connections, including STARTTLS capability and direct TLS connections. - Implement EP-01 tests for basic email sending, covering complete SMTP transaction flow, MIME attachments, HTML emails, custom headers, and minimal emails.
		
			
				
	
	
		
			246 lines
		
	
	
		
			9.0 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			246 lines
		
	
	
		
			9.0 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| /**
 | |
|  * EP-01: Basic Email Sending Tests
 | |
|  * Tests complete email sending lifecycle through SMTP server
 | |
|  */
 | |
| 
 | |
| import { assert, assertEquals } from '@std/assert';
 | |
| import { startTestServer, stopTestServer, type ITestServer } from '../../helpers/server.loader.ts';
 | |
| import {
 | |
|   connectToSmtp,
 | |
|   waitForGreeting,
 | |
|   sendSmtpCommand,
 | |
|   readSmtpResponse,
 | |
|   closeSmtpConnection,
 | |
| } from '../../helpers/utils.ts';
 | |
| 
 | |
| const TEST_PORT = 25258;
 | |
| let testServer: ITestServer;
 | |
| 
 | |
| Deno.test({
 | |
|   name: 'EP-01: Setup - Start SMTP server',
 | |
|   async fn() {
 | |
|     testServer = await startTestServer({ port: TEST_PORT });
 | |
|     assert(testServer, 'Test server should be created');
 | |
|   },
 | |
|   sanitizeResources: false,
 | |
|   sanitizeOps: false,
 | |
| });
 | |
| 
 | |
| Deno.test({
 | |
|   name: 'EP-01: Basic Email - complete SMTP transaction flow',
 | |
|   async fn() {
 | |
|     const conn = await connectToSmtp('localhost', TEST_PORT);
 | |
|     const fromAddress = 'sender@example.com';
 | |
|     const toAddress = 'recipient@example.com';
 | |
|     const emailContent = `Subject: Production Test Email\r\nFrom: ${fromAddress}\r\nTo: ${toAddress}\r\nDate: ${new Date().toUTCString()}\r\n\r\nThis is a test email sent during production testing.\r\nTest ID: EP-01\r\nTimestamp: ${Date.now()}\r\n`;
 | |
| 
 | |
|     try {
 | |
|       // Step 1: CONNECT - Wait for greeting
 | |
|       const greeting = await waitForGreeting(conn);
 | |
|       assert(greeting.includes('220'), 'Should receive 220 greeting');
 | |
| 
 | |
|       // Step 2: EHLO
 | |
|       const ehloResponse = await sendSmtpCommand(conn, 'EHLO test.example.com', '250');
 | |
|       assert(ehloResponse.includes('250'), 'Should accept EHLO');
 | |
| 
 | |
|       // Step 3: MAIL FROM
 | |
|       const mailFromResponse = await sendSmtpCommand(conn, `MAIL FROM:<${fromAddress}>`, '250');
 | |
|       assert(mailFromResponse.includes('250'), 'Should accept MAIL FROM');
 | |
| 
 | |
|       // Step 4: RCPT TO
 | |
|       const rcptToResponse = await sendSmtpCommand(conn, `RCPT TO:<${toAddress}>`, '250');
 | |
|       assert(rcptToResponse.includes('250'), 'Should accept RCPT TO');
 | |
| 
 | |
|       // Step 5: DATA
 | |
|       const dataResponse = await sendSmtpCommand(conn, 'DATA', '354');
 | |
|       assert(dataResponse.includes('354'), 'Should accept DATA command');
 | |
| 
 | |
|       // Step 6: EMAIL CONTENT
 | |
|       const encoder = new TextEncoder();
 | |
|       await conn.write(encoder.encode(emailContent));
 | |
|       await conn.write(encoder.encode('.\r\n')); // End of data marker
 | |
| 
 | |
|       const contentResponse = await readSmtpResponse(conn, '250');
 | |
|       assert(contentResponse.includes('250'), 'Should accept email content');
 | |
| 
 | |
|       // Step 7: QUIT
 | |
|       const quitResponse = await sendSmtpCommand(conn, 'QUIT', '221');
 | |
|       assert(quitResponse.includes('221'), 'Should respond to QUIT');
 | |
| 
 | |
|       console.log('✓ Complete email sending flow: CONNECT → EHLO → MAIL FROM → RCPT TO → DATA → CONTENT → QUIT');
 | |
|     } finally {
 | |
|       try {
 | |
|         conn.close();
 | |
|       } catch {
 | |
|         // Connection may already be closed
 | |
|       }
 | |
|     }
 | |
|   },
 | |
|   sanitizeResources: false,
 | |
|   sanitizeOps: false,
 | |
| });
 | |
| 
 | |
| Deno.test({
 | |
|   name: 'EP-01: Basic Email - send email with MIME attachment',
 | |
|   async fn() {
 | |
|     const conn = await connectToSmtp('localhost', TEST_PORT);
 | |
|     const fromAddress = 'sender@example.com';
 | |
|     const toAddress = 'recipient@example.com';
 | |
|     const boundary = '----=_Part_0_1234567890';
 | |
| 
 | |
|     const emailContent = `Subject: Email with Attachment\r\nFrom: ${fromAddress}\r\nTo: ${toAddress}\r\nMIME-Version: 1.0\r\nContent-Type: multipart/mixed; boundary="${boundary}"\r\n\r\n--${boundary}\r\nContent-Type: text/plain; charset=UTF-8\r\n\r\nThis email contains an attachment.\r\n\r\n--${boundary}\r\nContent-Type: text/plain; name="test.txt"\r\nContent-Disposition: attachment; filename="test.txt"\r\nContent-Transfer-Encoding: base64\r\n\r\nVGhpcyBpcyBhIHRlc3QgZmlsZS4=\r\n\r\n--${boundary}--\r\n`;
 | |
| 
 | |
|     try {
 | |
|       await waitForGreeting(conn);
 | |
|       await sendSmtpCommand(conn, 'EHLO test.example.com', '250');
 | |
|       await sendSmtpCommand(conn, `MAIL FROM:<${fromAddress}>`, '250');
 | |
|       await sendSmtpCommand(conn, `RCPT TO:<${toAddress}>`, '250');
 | |
|       await sendSmtpCommand(conn, 'DATA', '354');
 | |
| 
 | |
|       // Send MIME email content
 | |
|       const encoder = new TextEncoder();
 | |
|       await conn.write(encoder.encode(emailContent));
 | |
|       await conn.write(encoder.encode('.\r\n'));
 | |
| 
 | |
|       const response = await readSmtpResponse(conn, '250');
 | |
|       assert(response.includes('250'), 'Should accept MIME email with attachment');
 | |
| 
 | |
|       await sendSmtpCommand(conn, 'QUIT', '221');
 | |
|       console.log('✓ Successfully sent email with MIME attachment');
 | |
|     } finally {
 | |
|       try {
 | |
|         conn.close();
 | |
|       } catch {
 | |
|         // Ignore
 | |
|       }
 | |
|     }
 | |
|   },
 | |
|   sanitizeResources: false,
 | |
|   sanitizeOps: false,
 | |
| });
 | |
| 
 | |
| Deno.test({
 | |
|   name: 'EP-01: Basic Email - send HTML email',
 | |
|   async fn() {
 | |
|     const conn = await connectToSmtp('localhost', TEST_PORT);
 | |
|     const fromAddress = 'sender@example.com';
 | |
|     const toAddress = 'recipient@example.com';
 | |
|     const boundary = '----=_Part_0_987654321';
 | |
| 
 | |
|     const emailContent = `Subject: HTML Email Test\r\nFrom: ${fromAddress}\r\nTo: ${toAddress}\r\nMIME-Version: 1.0\r\nContent-Type: multipart/alternative; boundary="${boundary}"\r\n\r\n--${boundary}\r\nContent-Type: text/plain; charset=UTF-8\r\n\r\nThis is the plain text version.\r\n\r\n--${boundary}\r\nContent-Type: text/html; charset=UTF-8\r\n\r\n<html><body><h1>HTML Email</h1><p>This is the <strong>HTML</strong> version.</p></body></html>\r\n\r\n--${boundary}--\r\n`;
 | |
| 
 | |
|     try {
 | |
|       await waitForGreeting(conn);
 | |
|       await sendSmtpCommand(conn, 'EHLO test.example.com', '250');
 | |
|       await sendSmtpCommand(conn, `MAIL FROM:<${fromAddress}>`, '250');
 | |
|       await sendSmtpCommand(conn, `RCPT TO:<${toAddress}>`, '250');
 | |
|       await sendSmtpCommand(conn, 'DATA', '354');
 | |
| 
 | |
|       // Send HTML email content
 | |
|       const encoder = new TextEncoder();
 | |
|       await conn.write(encoder.encode(emailContent));
 | |
|       await conn.write(encoder.encode('.\r\n'));
 | |
| 
 | |
|       const response = await readSmtpResponse(conn, '250');
 | |
|       assert(response.includes('250'), 'Should accept HTML email');
 | |
| 
 | |
|       await sendSmtpCommand(conn, 'QUIT', '221');
 | |
|       console.log('✓ Successfully sent HTML email (multipart/alternative)');
 | |
|     } finally {
 | |
|       try {
 | |
|         conn.close();
 | |
|       } catch {
 | |
|         // Ignore
 | |
|       }
 | |
|     }
 | |
|   },
 | |
|   sanitizeResources: false,
 | |
|   sanitizeOps: false,
 | |
| });
 | |
| 
 | |
| Deno.test({
 | |
|   name: 'EP-01: Basic Email - send email with custom headers',
 | |
|   async fn() {
 | |
|     const conn = await connectToSmtp('localhost', TEST_PORT);
 | |
|     const fromAddress = 'sender@example.com';
 | |
|     const toAddress = 'recipient@example.com';
 | |
| 
 | |
|     const emailContent = `Subject: Custom Headers Test\r\nFrom: ${fromAddress}\r\nTo: ${toAddress}\r\nX-Custom-Header: CustomValue\r\nX-Priority: 1\r\nX-Mailer: SMTP Test Suite\r\nReply-To: noreply@example.com\r\nOrganization: Test Organization\r\n\r\nThis email contains custom headers.\r\n`;
 | |
| 
 | |
|     try {
 | |
|       await waitForGreeting(conn);
 | |
|       await sendSmtpCommand(conn, 'EHLO test.example.com', '250');
 | |
|       await sendSmtpCommand(conn, `MAIL FROM:<${fromAddress}>`, '250');
 | |
|       await sendSmtpCommand(conn, `RCPT TO:<${toAddress}>`, '250');
 | |
|       await sendSmtpCommand(conn, 'DATA', '354');
 | |
| 
 | |
|       // Send email with custom headers
 | |
|       const encoder = new TextEncoder();
 | |
|       await conn.write(encoder.encode(emailContent));
 | |
|       await conn.write(encoder.encode('.\r\n'));
 | |
| 
 | |
|       const response = await readSmtpResponse(conn, '250');
 | |
|       assert(response.includes('250'), 'Should accept email with custom headers');
 | |
| 
 | |
|       await sendSmtpCommand(conn, 'QUIT', '221');
 | |
|       console.log('✓ Successfully sent email with custom headers (X-Custom-Header, X-Priority, etc.)');
 | |
|     } finally {
 | |
|       try {
 | |
|         conn.close();
 | |
|       } catch {
 | |
|         // Ignore
 | |
|       }
 | |
|     }
 | |
|   },
 | |
|   sanitizeResources: false,
 | |
|   sanitizeOps: false,
 | |
| });
 | |
| 
 | |
| Deno.test({
 | |
|   name: 'EP-01: Basic Email - send minimal email (no headers)',
 | |
|   async fn() {
 | |
|     const conn = await connectToSmtp('localhost', TEST_PORT);
 | |
|     const fromAddress = 'sender@example.com';
 | |
|     const toAddress = 'recipient@example.com';
 | |
| 
 | |
|     // Minimal email - just a body, no headers
 | |
|     const emailContent = 'This is a minimal email with no headers.\r\n';
 | |
| 
 | |
|     try {
 | |
|       await waitForGreeting(conn);
 | |
|       await sendSmtpCommand(conn, 'EHLO test.example.com', '250');
 | |
|       await sendSmtpCommand(conn, `MAIL FROM:<${fromAddress}>`, '250');
 | |
|       await sendSmtpCommand(conn, `RCPT TO:<${toAddress}>`, '250');
 | |
|       await sendSmtpCommand(conn, 'DATA', '354');
 | |
| 
 | |
|       // Send minimal email
 | |
|       const encoder = new TextEncoder();
 | |
|       await conn.write(encoder.encode(emailContent));
 | |
|       await conn.write(encoder.encode('.\r\n'));
 | |
| 
 | |
|       const response = await readSmtpResponse(conn, '250');
 | |
|       assert(response.includes('250'), 'Should accept minimal email');
 | |
| 
 | |
|       await sendSmtpCommand(conn, 'QUIT', '221');
 | |
|       console.log('✓ Successfully sent minimal email (body only, no headers)');
 | |
|     } finally {
 | |
|       try {
 | |
|         conn.close();
 | |
|       } catch {
 | |
|         // Ignore
 | |
|       }
 | |
|     }
 | |
|   },
 | |
|   sanitizeResources: false,
 | |
|   sanitizeOps: false,
 | |
| });
 | |
| 
 | |
| Deno.test({
 | |
|   name: 'EP-01: Cleanup - Stop SMTP server',
 | |
|   async fn() {
 | |
|     await stopTestServer(testServer);
 | |
|   },
 | |
|   sanitizeResources: false,
 | |
|   sanitizeOps: false,
 | |
| });
 |