Files
smartmta/test/suite/smtpserver_error-handling/test.err-01.syntax-errors.test.ts
Juergen Kunz 6523c55516 feat: Implement Deno-native STARTTLS handler and connection wrapper
- Refactored STARTTLS implementation to use Deno's native TLS via Deno.startTls().
- Introduced ConnectionWrapper to provide a Node.js net.Socket-compatible interface for Deno.Conn and Deno.TlsConn.
- Updated TlsHandler to utilize the new STARTTLS implementation.
- Added comprehensive SMTP authentication tests for PLAIN and LOGIN mechanisms.
- Implemented rate limiting tests for SMTP server connections and commands.
- Enhanced error handling and logging throughout the STARTTLS and connection upgrade processes.
2025-10-28 18:51:33 +00:00

292 lines
7.9 KiB
TypeScript

/**
* ERR-01: Syntax Error Handling Tests
* Tests SMTP server handling of syntax errors and malformed commands
*/
import { assert, assertMatch } 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 = 25261;
let testServer: ITestServer;
Deno.test({
name: 'ERR-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: 'ERR-01: Syntax Errors - rejects invalid command',
async fn() {
const conn = await connectToSmtp('localhost', TEST_PORT);
try {
await waitForGreeting(conn);
// Send invalid command
const encoder = new TextEncoder();
await conn.write(encoder.encode('INVALID_COMMAND\r\n'));
const response = await readSmtpResponse(conn);
// RFC 5321: Should return 500 (syntax error) or 502 (command not implemented)
assertMatch(response, /^(500|502)/, 'Should reject invalid command with 500 or 502');
await sendSmtpCommand(conn, 'QUIT', '221');
console.log('✓ Invalid command rejected with appropriate error code');
} finally {
try {
conn.close();
} catch {
// Ignore
}
}
},
sanitizeResources: false,
sanitizeOps: false,
});
Deno.test({
name: 'ERR-01: Syntax Errors - rejects MAIL FROM without brackets',
async fn() {
const conn = await connectToSmtp('localhost', TEST_PORT);
try {
await waitForGreeting(conn);
await sendSmtpCommand(conn, 'EHLO test.example.com', '250');
// Send MAIL FROM without angle brackets
const encoder = new TextEncoder();
await conn.write(encoder.encode('MAIL FROM:test@example.com\r\n'));
const response = await readSmtpResponse(conn);
// Should return 501 (syntax error in parameters)
assertMatch(response, /^501/, 'Should reject MAIL FROM without brackets with 501');
await sendSmtpCommand(conn, 'QUIT', '221');
console.log('✓ MAIL FROM without brackets rejected');
} finally {
try {
conn.close();
} catch {
// Ignore
}
}
},
sanitizeResources: false,
sanitizeOps: false,
});
Deno.test({
name: 'ERR-01: Syntax Errors - rejects RCPT TO without brackets',
async fn() {
const conn = await connectToSmtp('localhost', TEST_PORT);
try {
await waitForGreeting(conn);
await sendSmtpCommand(conn, 'EHLO test.example.com', '250');
await sendSmtpCommand(conn, 'MAIL FROM:<sender@example.com>', '250');
// Send RCPT TO without angle brackets
const encoder = new TextEncoder();
await conn.write(encoder.encode('RCPT TO:recipient@example.com\r\n'));
const response = await readSmtpResponse(conn);
// Should return 501 (syntax error in parameters)
assertMatch(response, /^501/, 'Should reject RCPT TO without brackets with 501');
await sendSmtpCommand(conn, 'QUIT', '221');
console.log('✓ RCPT TO without brackets rejected');
} finally {
try {
conn.close();
} catch {
// Ignore
}
}
},
sanitizeResources: false,
sanitizeOps: false,
});
Deno.test({
name: 'ERR-01: Syntax Errors - rejects EHLO without hostname',
async fn() {
const conn = await connectToSmtp('localhost', TEST_PORT);
try {
await waitForGreeting(conn);
// Send EHLO without hostname
const encoder = new TextEncoder();
await conn.write(encoder.encode('EHLO\r\n'));
const response = await readSmtpResponse(conn);
// Should return 501 (syntax error in parameters - missing domain)
assertMatch(response, /^501/, 'Should reject EHLO without hostname with 501');
await sendSmtpCommand(conn, 'QUIT', '221');
console.log('✓ EHLO without hostname rejected');
} finally {
try {
conn.close();
} catch {
// Ignore
}
}
},
sanitizeResources: false,
sanitizeOps: false,
});
Deno.test({
name: 'ERR-01: Syntax Errors - handles commands with extra parameters',
async fn() {
const conn = await connectToSmtp('localhost', TEST_PORT);
try {
await waitForGreeting(conn);
await sendSmtpCommand(conn, 'EHLO test.example.com', '250');
// Send QUIT with extra parameters (QUIT doesn't take parameters)
const encoder = new TextEncoder();
await conn.write(encoder.encode('QUIT extra parameters\r\n'));
const response = await readSmtpResponse(conn);
// RFC 5321 Section 4.1.1.10: QUIT syntax is "QUIT <CRLF>" (no parameters)
// Should return 501 (syntax error in parameters)
assertMatch(response, /^501/, 'Should reject QUIT with extra params with 501');
console.log('✓ QUIT with extra parameters correctly rejected with 501');
} finally {
try {
conn.close();
} catch {
// Ignore
}
}
},
sanitizeResources: false,
sanitizeOps: false,
});
Deno.test({
name: 'ERR-01: Syntax Errors - rejects malformed email addresses',
async fn() {
const conn = await connectToSmtp('localhost', TEST_PORT);
try {
await waitForGreeting(conn);
await sendSmtpCommand(conn, 'EHLO test.example.com', '250');
// Send malformed email address
const encoder = new TextEncoder();
await conn.write(encoder.encode('MAIL FROM:<not an email>\r\n'));
const response = await readSmtpResponse(conn);
// RFC 5321: "<not an email>" is a syntax/format error, should return 501
assertMatch(response, /^501/, 'Should reject malformed email with 501');
await sendSmtpCommand(conn, 'QUIT', '221');
console.log('✓ Malformed email address correctly rejected with 501');
} finally {
try {
conn.close();
} catch {
// Ignore
}
}
},
sanitizeResources: false,
sanitizeOps: false,
});
Deno.test({
name: 'ERR-01: Syntax Errors - rejects commands in wrong sequence',
async fn() {
const conn = await connectToSmtp('localhost', TEST_PORT);
try {
await waitForGreeting(conn);
// Send DATA without MAIL FROM/RCPT TO
const encoder = new TextEncoder();
await conn.write(encoder.encode('DATA\r\n'));
const response = await readSmtpResponse(conn);
// Should return 503 (bad sequence of commands)
assertMatch(response, /^503/, 'Should reject DATA without setup with 503');
await sendSmtpCommand(conn, 'QUIT', '221');
console.log('✓ Commands in wrong sequence rejected');
} finally {
try {
conn.close();
} catch {
// Ignore
}
}
},
sanitizeResources: false,
sanitizeOps: false,
});
Deno.test({
name: 'ERR-01: Syntax Errors - handles excessively long commands',
async fn() {
const conn = await connectToSmtp('localhost', TEST_PORT);
try {
await waitForGreeting(conn);
// Send EHLO with excessively long hostname (>512 octets)
const longString = 'A'.repeat(1000);
const encoder = new TextEncoder();
await conn.write(encoder.encode(`EHLO ${longString}\r\n`));
const response = await readSmtpResponse(conn);
// RFC 5321 Section 4.5.3.1.4: Max command line is 512 octets
// Should reject with 500 (syntax error) or 501 (parameter error)
assertMatch(response, /^(500|501)/, 'Should reject command >512 octets with 500 or 501');
await sendSmtpCommand(conn, 'QUIT', '221');
console.log('✓ Excessively long command correctly rejected');
} finally {
try {
conn.close();
} catch {
// Ignore
}
}
},
sanitizeResources: false,
sanitizeOps: false,
});
Deno.test({
name: 'ERR-01: Cleanup - Stop SMTP server',
async fn() {
await stopTestServer(testServer);
},
sanitizeResources: false,
sanitizeOps: false,
});