mirror of
https://github.com/jhillyerd/inbucket.git
synced 2025-12-17 17:47:03 +00:00
Test DATA on handler Add a 2 second wait if the test fails so that all the logging data can be collected.
340 lines
7.8 KiB
Go
340 lines
7.8 KiB
Go
package smtpd
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"github.com/jhillyerd/inbucket/config"
|
|
"io"
|
|
"io/ioutil"
|
|
"log"
|
|
"net"
|
|
"net/textproto"
|
|
"os"
|
|
"testing"
|
|
"time"
|
|
)
|
|
|
|
type scriptStep struct {
|
|
send string
|
|
expect int
|
|
}
|
|
|
|
// Test commands in GREET state
|
|
func TestGreetState(t *testing.T) {
|
|
server, logbuf := setupSmtpServer()
|
|
defer teardownSmtpServer(server)
|
|
var script []scriptStep
|
|
|
|
// Test out some mangled HELOs
|
|
script = []scriptStep{
|
|
{"HELLO", 500},
|
|
{"HELL", 500},
|
|
{"hello", 500},
|
|
{"Outlook", 500},
|
|
}
|
|
if err := playSession(t, server, script); err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
// Valid HELOs
|
|
if err := playSession(t, server, []scriptStep{{"HELO", 250}}); err != nil {
|
|
t.Error(err)
|
|
}
|
|
if err := playSession(t, server, []scriptStep{{"HELO mydomain", 250}}); err != nil {
|
|
t.Error(err)
|
|
}
|
|
if err := playSession(t, server, []scriptStep{{"HELO mydom.com", 250}}); err != nil {
|
|
t.Error(err)
|
|
}
|
|
if err := playSession(t, server, []scriptStep{{"HelO mydom.com", 250}}); err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
if t.Failed() {
|
|
// Wait for handler to finish logging
|
|
time.Sleep(2 * time.Second)
|
|
// Dump buffered log data if there was a failure
|
|
io.Copy(os.Stderr, logbuf)
|
|
}
|
|
}
|
|
|
|
// Test commands in READY state
|
|
func TestReadyState(t *testing.T) {
|
|
server, logbuf := setupSmtpServer()
|
|
defer teardownSmtpServer(server)
|
|
var script []scriptStep
|
|
|
|
// Test out some mangled READY commands
|
|
script = []scriptStep{
|
|
{"HELO localhost", 250},
|
|
{"FOOB", 500},
|
|
{"HELO", 503},
|
|
{"DATA", 503},
|
|
{"MAIL", 501},
|
|
{"MAIL FROM john@gmail.com", 501},
|
|
{"MAIL FROM:john@gmail.com", 501},
|
|
{"MAIL FROM:<john@gmail.com> SIZE=147KB", 501},
|
|
{"MAIL FROM: <john@gmail.com> SIZE147", 501},
|
|
}
|
|
if err := playSession(t, server, script); err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
// Test out some valid MAIL commands
|
|
script = []scriptStep{
|
|
{"HELO localhost", 250},
|
|
{"MAIL FROM:<john@gmail.com>", 250},
|
|
{"RSET", 250},
|
|
{"MAIL FROM: <john@gmail.com>", 250},
|
|
{"RSET", 250},
|
|
{"MAIL FROM: <john@gmail.com> BODY=8BITMIME", 250},
|
|
{"RSET", 250},
|
|
{"MAIL FROM:<john@gmail.com> SIZE=1024", 250},
|
|
}
|
|
if err := playSession(t, server, script); err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
if t.Failed() {
|
|
// Wait for handler to finish logging
|
|
time.Sleep(2 * time.Second)
|
|
// Dump buffered log data if there was a failure
|
|
io.Copy(os.Stderr, logbuf)
|
|
}
|
|
}
|
|
|
|
// Test commands in MAIL state
|
|
func TestMailState(t *testing.T) {
|
|
server, logbuf := setupSmtpServer()
|
|
defer teardownSmtpServer(server)
|
|
var script []scriptStep
|
|
|
|
// Test out some mangled READY commands
|
|
script = []scriptStep{
|
|
{"HELO localhost", 250},
|
|
{"MAIL FROM:<john@gmail.com>", 250},
|
|
{"FOOB", 500},
|
|
{"HELO", 503},
|
|
{"DATA", 503},
|
|
{"MAIL", 503},
|
|
{"RCPT", 501},
|
|
{"RCPT TO", 501},
|
|
{"RCPT TO james@gmail.com", 501},
|
|
}
|
|
if err := playSession(t, server, script); err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
// Test out some good RCPT commands
|
|
script = []scriptStep{
|
|
{"HELO localhost", 250},
|
|
{"MAIL FROM:<john@gmail.com>", 250},
|
|
{"RCPT TO:<u1@gmail.com>", 250},
|
|
{"RCPT TO: <u2@gmail.com>", 250},
|
|
{"RCPT TO:u3@gmail.com", 250},
|
|
{"RCPT TO: u4@gmail.com", 250},
|
|
}
|
|
if err := playSession(t, server, script); err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
// Test out recipient limit
|
|
script = []scriptStep{
|
|
{"HELO localhost", 250},
|
|
{"MAIL FROM:<john@gmail.com>", 250},
|
|
{"RCPT TO:<u1@gmail.com>", 250},
|
|
{"RCPT TO:<u2@gmail.com>", 250},
|
|
{"RCPT TO:<u3@gmail.com>", 250},
|
|
{"RCPT TO:<u4@gmail.com>", 250},
|
|
{"RCPT TO:<u5@gmail.com>", 250},
|
|
{"RCPT TO:<u6@gmail.com>", 552},
|
|
}
|
|
if err := playSession(t, server, script); err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
// Test DATA
|
|
script = []scriptStep{
|
|
{"HELO localhost", 250},
|
|
{"MAIL FROM:<john@gmail.com>", 250},
|
|
{"RCPT TO:<u1@gmail.com>", 250},
|
|
{"DATA", 354},
|
|
{".", 250},
|
|
}
|
|
if err := playSession(t, server, script); err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
// Test RSET
|
|
script = []scriptStep{
|
|
{"HELO localhost", 250},
|
|
{"MAIL FROM:<john@gmail.com>", 250},
|
|
{"RCPT TO:<u1@gmail.com>", 250},
|
|
{"RSET", 250},
|
|
{"MAIL FROM:<john@gmail.com>", 250},
|
|
}
|
|
if err := playSession(t, server, script); err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
// Test QUIT
|
|
script = []scriptStep{
|
|
{"HELO localhost", 250},
|
|
{"MAIL FROM:<john@gmail.com>", 250},
|
|
{"RCPT TO:<u1@gmail.com>", 250},
|
|
{"QUIT", 221},
|
|
}
|
|
if err := playSession(t, server, script); err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
if t.Failed() {
|
|
// Wait for handler to finish logging
|
|
time.Sleep(2 * time.Second)
|
|
// Dump buffered log data if there was a failure
|
|
io.Copy(os.Stderr, logbuf)
|
|
}
|
|
}
|
|
|
|
// Test commands in DATA state
|
|
func TestDataState(t *testing.T) {
|
|
server, logbuf := setupSmtpServer()
|
|
defer teardownSmtpServer(server)
|
|
var script []scriptStep
|
|
pipe := setupSmtpSession(server)
|
|
c := textproto.NewConn(pipe)
|
|
|
|
// Get us into DATA state
|
|
if code, _, err := c.ReadCodeLine(220); err != nil {
|
|
t.Errorf("Expected a 220 greeting, got %v", code)
|
|
}
|
|
script = []scriptStep{
|
|
{"HELO localhost", 250},
|
|
{"MAIL FROM:<john@gmail.com>", 250},
|
|
{"RCPT TO:<u1@gmail.com>", 250},
|
|
{"DATA", 354},
|
|
}
|
|
if err := playScriptAgainst(t, c, script); err != nil {
|
|
t.Error(err)
|
|
}
|
|
// Send a message
|
|
body := `To: u1@gmail.com
|
|
From: john@gmail.com
|
|
Subject: test
|
|
|
|
Hi!
|
|
`
|
|
dw := c.DotWriter()
|
|
io.WriteString(dw, body)
|
|
dw.Close()
|
|
if code, _, err := c.ReadCodeLine(250); err != nil {
|
|
t.Errorf("Expected a 250 greeting, got %v", code)
|
|
}
|
|
|
|
if t.Failed() {
|
|
// Wait for handler to finish logging
|
|
time.Sleep(2 * time.Second)
|
|
// Dump buffered log data if there was a failure
|
|
io.Copy(os.Stderr, logbuf)
|
|
}
|
|
}
|
|
|
|
// playSession creates a new session, reads the greeting and then plays the script
|
|
func playSession(t *testing.T, server *Server, script []scriptStep) error {
|
|
pipe := setupSmtpSession(server)
|
|
c := textproto.NewConn(pipe)
|
|
|
|
if code, _, err := c.ReadCodeLine(220); err != nil {
|
|
return fmt.Errorf("Expected a 220 greeting, got %v", code)
|
|
}
|
|
|
|
err := playScriptAgainst(t, c, script)
|
|
|
|
c.Cmd("QUIT")
|
|
c.ReadCodeLine(221)
|
|
|
|
return err
|
|
}
|
|
|
|
// playScriptAgainst an existing connection, does not handle server greeting
|
|
func playScriptAgainst(t *testing.T, c *textproto.Conn, script []scriptStep) error {
|
|
for i, step := range script {
|
|
id, err := c.Cmd(step.send)
|
|
if err != nil {
|
|
return fmt.Errorf("Step %d, failed to send %q: %v", i, step.send, err)
|
|
}
|
|
|
|
c.StartResponse(id)
|
|
code, msg, err := c.ReadCodeLine(step.expect)
|
|
if err != nil {
|
|
err = fmt.Errorf("Step %d, sent %q, expected %v, got %v: %q",
|
|
i, step.send, step.expect, code, msg)
|
|
}
|
|
c.EndResponse(id)
|
|
|
|
if err != nil {
|
|
// Return after c.EndResponse so we don't hang the connection
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// net.Pipe does not implement deadlines
|
|
type mockConn struct {
|
|
net.Conn
|
|
}
|
|
|
|
func (m *mockConn) SetDeadline(t time.Time) error { return nil }
|
|
func (m *mockConn) SetReadDeadline(t time.Time) error { return nil }
|
|
func (m *mockConn) SetWriteDeadline(t time.Time) error { return nil }
|
|
|
|
func setupSmtpServer() (*Server, *bytes.Buffer) {
|
|
// Setup datastore
|
|
path, err := ioutil.TempDir("", "inbucket")
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
ds := NewFileDataStore(path)
|
|
|
|
// Test Server Config
|
|
cfg := config.SmtpConfig{
|
|
Ip4address: net.IPv4(127, 0, 0, 1),
|
|
Ip4port: 2500,
|
|
Domain: "inbucket.local",
|
|
DomainNoStore: "bitbucket.local",
|
|
MaxRecipients: 5,
|
|
MaxIdleSeconds: 5,
|
|
MaxMessageBytes: 5000,
|
|
StoreMessages: true,
|
|
}
|
|
|
|
// Capture log output
|
|
buf := new(bytes.Buffer)
|
|
log.SetOutput(buf)
|
|
|
|
// Create a server, don't start it
|
|
return NewSmtpServer(cfg, ds), buf
|
|
}
|
|
|
|
var sessionNum int
|
|
|
|
func setupSmtpSession(server *Server) net.Conn {
|
|
// Pair of pipes to communicate
|
|
serverConn, clientConn := net.Pipe()
|
|
// Start the session
|
|
server.waitgroup.Add(1)
|
|
sessionNum++
|
|
go server.startSession(sessionNum, &mockConn{serverConn})
|
|
|
|
return clientConn
|
|
}
|
|
|
|
func teardownSmtpServer(server *Server) {
|
|
ds := server.dataStore.(*FileDataStore)
|
|
if err := os.RemoveAll(ds.path); err != nil {
|
|
panic(err)
|
|
}
|
|
//log.SetOutput(os.Stderr)
|
|
}
|