304 lines
8.6 KiB
TypeScript
304 lines
8.6 KiB
TypeScript
|
|
/**
|
||
|
|
* ERR-02: Invalid Sequence Tests
|
||
|
|
* Tests SMTP server handling of commands in incorrect sequence
|
||
|
|
*/
|
||
|
|
|
||
|
|
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 = 25262;
|
||
|
|
let testServer: ITestServer;
|
||
|
|
|
||
|
|
Deno.test({
|
||
|
|
name: 'ERR-02: 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-02: Invalid Sequence - rejects MAIL FROM before EHLO',
|
||
|
|
async fn() {
|
||
|
|
const conn = await connectToSmtp('localhost', TEST_PORT);
|
||
|
|
|
||
|
|
try {
|
||
|
|
await waitForGreeting(conn);
|
||
|
|
|
||
|
|
// Send MAIL FROM without EHLO
|
||
|
|
const encoder = new TextEncoder();
|
||
|
|
await conn.write(encoder.encode('MAIL FROM:<test@example.com>\r\n'));
|
||
|
|
|
||
|
|
const response = await readSmtpResponse(conn);
|
||
|
|
|
||
|
|
// Should return 503 (bad sequence of commands)
|
||
|
|
assertMatch(response, /^503/, 'Should reject MAIL FROM before EHLO with 503');
|
||
|
|
|
||
|
|
await sendSmtpCommand(conn, 'QUIT', '221');
|
||
|
|
console.log('✓ MAIL FROM before EHLO rejected');
|
||
|
|
} finally {
|
||
|
|
try {
|
||
|
|
conn.close();
|
||
|
|
} catch {
|
||
|
|
// Ignore
|
||
|
|
}
|
||
|
|
}
|
||
|
|
},
|
||
|
|
sanitizeResources: false,
|
||
|
|
sanitizeOps: false,
|
||
|
|
});
|
||
|
|
|
||
|
|
Deno.test({
|
||
|
|
name: 'ERR-02: Invalid Sequence - rejects RCPT TO before MAIL FROM',
|
||
|
|
async fn() {
|
||
|
|
const conn = await connectToSmtp('localhost', TEST_PORT);
|
||
|
|
|
||
|
|
try {
|
||
|
|
await waitForGreeting(conn);
|
||
|
|
await sendSmtpCommand(conn, 'EHLO test.example.com', '250');
|
||
|
|
|
||
|
|
// Send RCPT TO without MAIL FROM
|
||
|
|
const encoder = new TextEncoder();
|
||
|
|
await conn.write(encoder.encode('RCPT TO:<test@example.com>\r\n'));
|
||
|
|
|
||
|
|
const response = await readSmtpResponse(conn);
|
||
|
|
|
||
|
|
// Should return 503 (bad sequence of commands)
|
||
|
|
assertMatch(response, /^503/, 'Should reject RCPT TO before MAIL FROM with 503');
|
||
|
|
|
||
|
|
await sendSmtpCommand(conn, 'QUIT', '221');
|
||
|
|
console.log('✓ RCPT TO before MAIL FROM rejected');
|
||
|
|
} finally {
|
||
|
|
try {
|
||
|
|
conn.close();
|
||
|
|
} catch {
|
||
|
|
// Ignore
|
||
|
|
}
|
||
|
|
}
|
||
|
|
},
|
||
|
|
sanitizeResources: false,
|
||
|
|
sanitizeOps: false,
|
||
|
|
});
|
||
|
|
|
||
|
|
Deno.test({
|
||
|
|
name: 'ERR-02: Invalid Sequence - rejects DATA before RCPT TO',
|
||
|
|
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:<test@example.com>', '250');
|
||
|
|
|
||
|
|
// Send DATA without RCPT TO
|
||
|
|
const encoder = new TextEncoder();
|
||
|
|
await conn.write(encoder.encode('DATA\r\n'));
|
||
|
|
|
||
|
|
const response = await readSmtpResponse(conn);
|
||
|
|
|
||
|
|
// RFC 5321: Should return 503 (bad sequence of commands)
|
||
|
|
assertMatch(response, /^503/, 'Should reject DATA before RCPT TO with 503');
|
||
|
|
|
||
|
|
await sendSmtpCommand(conn, 'QUIT', '221');
|
||
|
|
console.log('✓ DATA before RCPT TO rejected');
|
||
|
|
} finally {
|
||
|
|
try {
|
||
|
|
conn.close();
|
||
|
|
} catch {
|
||
|
|
// Ignore
|
||
|
|
}
|
||
|
|
}
|
||
|
|
},
|
||
|
|
sanitizeResources: false,
|
||
|
|
sanitizeOps: false,
|
||
|
|
});
|
||
|
|
|
||
|
|
Deno.test({
|
||
|
|
name: 'ERR-02: Invalid Sequence - allows multiple EHLO commands',
|
||
|
|
async fn() {
|
||
|
|
const conn = await connectToSmtp('localhost', TEST_PORT);
|
||
|
|
|
||
|
|
try {
|
||
|
|
await waitForGreeting(conn);
|
||
|
|
|
||
|
|
// Send multiple EHLO commands
|
||
|
|
const response1 = await sendSmtpCommand(conn, 'EHLO test1.example.com', '250');
|
||
|
|
assert(response1.includes('250'), 'First EHLO should succeed');
|
||
|
|
|
||
|
|
const response2 = await sendSmtpCommand(conn, 'EHLO test2.example.com', '250');
|
||
|
|
assert(response2.includes('250'), 'Second EHLO should succeed');
|
||
|
|
|
||
|
|
const response3 = await sendSmtpCommand(conn, 'EHLO test3.example.com', '250');
|
||
|
|
assert(response3.includes('250'), 'Third EHLO should succeed');
|
||
|
|
|
||
|
|
await sendSmtpCommand(conn, 'QUIT', '221');
|
||
|
|
console.log('✓ Multiple EHLO commands allowed');
|
||
|
|
} finally {
|
||
|
|
try {
|
||
|
|
conn.close();
|
||
|
|
} catch {
|
||
|
|
// Ignore
|
||
|
|
}
|
||
|
|
}
|
||
|
|
},
|
||
|
|
sanitizeResources: false,
|
||
|
|
sanitizeOps: false,
|
||
|
|
});
|
||
|
|
|
||
|
|
Deno.test({
|
||
|
|
name: 'ERR-02: Invalid Sequence - rejects second MAIL FROM without RSET',
|
||
|
|
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:<sender1@example.com>', '250');
|
||
|
|
|
||
|
|
// Send second MAIL FROM without RSET
|
||
|
|
const encoder = new TextEncoder();
|
||
|
|
await conn.write(encoder.encode('MAIL FROM:<sender2@example.com>\r\n'));
|
||
|
|
|
||
|
|
const response = await readSmtpResponse(conn);
|
||
|
|
|
||
|
|
// Should return 503 (bad sequence) or 250 (some implementations allow overwrite)
|
||
|
|
assertMatch(response, /^(503|250)/, 'Should handle second MAIL FROM');
|
||
|
|
|
||
|
|
await sendSmtpCommand(conn, 'QUIT', '221');
|
||
|
|
console.log(`✓ Second MAIL FROM handled: ${response.substring(0, 3)}`);
|
||
|
|
} finally {
|
||
|
|
try {
|
||
|
|
conn.close();
|
||
|
|
} catch {
|
||
|
|
// Ignore
|
||
|
|
}
|
||
|
|
}
|
||
|
|
},
|
||
|
|
sanitizeResources: false,
|
||
|
|
sanitizeOps: false,
|
||
|
|
});
|
||
|
|
|
||
|
|
Deno.test({
|
||
|
|
name: 'ERR-02: Invalid Sequence - rejects DATA without MAIL FROM',
|
||
|
|
async fn() {
|
||
|
|
const conn = await connectToSmtp('localhost', TEST_PORT);
|
||
|
|
|
||
|
|
try {
|
||
|
|
await waitForGreeting(conn);
|
||
|
|
await sendSmtpCommand(conn, 'EHLO test.example.com', '250');
|
||
|
|
|
||
|
|
// Send DATA without MAIL FROM
|
||
|
|
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 MAIL FROM with 503');
|
||
|
|
|
||
|
|
await sendSmtpCommand(conn, 'QUIT', '221');
|
||
|
|
console.log('✓ DATA without MAIL FROM rejected');
|
||
|
|
} finally {
|
||
|
|
try {
|
||
|
|
conn.close();
|
||
|
|
} catch {
|
||
|
|
// Ignore
|
||
|
|
}
|
||
|
|
}
|
||
|
|
},
|
||
|
|
sanitizeResources: false,
|
||
|
|
sanitizeOps: false,
|
||
|
|
});
|
||
|
|
|
||
|
|
Deno.test({
|
||
|
|
name: 'ERR-02: Invalid Sequence - handles commands after QUIT',
|
||
|
|
async fn() {
|
||
|
|
const conn = await connectToSmtp('localhost', TEST_PORT);
|
||
|
|
|
||
|
|
try {
|
||
|
|
await waitForGreeting(conn);
|
||
|
|
await sendSmtpCommand(conn, 'EHLO test.example.com', '250');
|
||
|
|
await sendSmtpCommand(conn, 'QUIT', '221');
|
||
|
|
|
||
|
|
// Try to send command after QUIT
|
||
|
|
const encoder = new TextEncoder();
|
||
|
|
let writeSucceeded = false;
|
||
|
|
|
||
|
|
try {
|
||
|
|
await conn.write(encoder.encode('EHLO test.example.com\r\n'));
|
||
|
|
writeSucceeded = true;
|
||
|
|
|
||
|
|
// If write succeeded, wait to see if we get a response (we shouldn't)
|
||
|
|
await new Promise((resolve) => setTimeout(resolve, 500));
|
||
|
|
} catch {
|
||
|
|
// Write failed - connection already closed (expected)
|
||
|
|
}
|
||
|
|
|
||
|
|
// Either write failed or no response received after QUIT (both acceptable)
|
||
|
|
assert(true, 'Commands after QUIT handled correctly');
|
||
|
|
console.log(`✓ Commands after QUIT handled (write ${writeSucceeded ? 'succeeded but ignored' : 'failed'})`);
|
||
|
|
} finally {
|
||
|
|
try {
|
||
|
|
conn.close();
|
||
|
|
} catch {
|
||
|
|
// Already closed
|
||
|
|
}
|
||
|
|
}
|
||
|
|
},
|
||
|
|
sanitizeResources: false,
|
||
|
|
sanitizeOps: false,
|
||
|
|
});
|
||
|
|
|
||
|
|
Deno.test({
|
||
|
|
name: 'ERR-02: Invalid Sequence - recovers from syntax error in sequence',
|
||
|
|
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 with wrong syntax (missing brackets)
|
||
|
|
const encoder = new TextEncoder();
|
||
|
|
await conn.write(encoder.encode('RCPT TO:recipient@example.com\r\n'));
|
||
|
|
|
||
|
|
const badResponse = await readSmtpResponse(conn);
|
||
|
|
assertMatch(badResponse, /^501/, 'Should reject RCPT TO without brackets with 501');
|
||
|
|
|
||
|
|
// Now send valid RCPT TO (session should still be valid)
|
||
|
|
const goodResponse = await sendSmtpCommand(conn, 'RCPT TO:<recipient@example.com>', '250');
|
||
|
|
assert(goodResponse.includes('250'), 'Should accept valid RCPT TO after syntax error');
|
||
|
|
|
||
|
|
await sendSmtpCommand(conn, 'QUIT', '221');
|
||
|
|
console.log('✓ Session recovered from syntax error');
|
||
|
|
} finally {
|
||
|
|
try {
|
||
|
|
conn.close();
|
||
|
|
} catch {
|
||
|
|
// Ignore
|
||
|
|
}
|
||
|
|
}
|
||
|
|
},
|
||
|
|
sanitizeResources: false,
|
||
|
|
sanitizeOps: false,
|
||
|
|
});
|
||
|
|
|
||
|
|
Deno.test({
|
||
|
|
name: 'ERR-02: Cleanup - Stop SMTP server',
|
||
|
|
async fn() {
|
||
|
|
await stopTestServer(testServer);
|
||
|
|
},
|
||
|
|
sanitizeResources: false,
|
||
|
|
sanitizeOps: false,
|
||
|
|
});
|