mirror of
https://github.com/mjl-/mox.git
synced 2025-01-14 01:06:27 +03:00
faa08583c0
we could make more types of delays configurable. the current approach isn't great, as it results in an a default value of "0s" in the config file, while the actual default is 15s (which is documented just above, but still).
155 lines
4.4 KiB
Go
155 lines
4.4 KiB
Go
//go:build integration
|
|
|
|
// Run this using docker-compose.yml, see Makefile.
|
|
|
|
package main
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"net"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/mjl-/mox/dns"
|
|
"github.com/mjl-/mox/imapclient"
|
|
"github.com/mjl-/mox/mlog"
|
|
"github.com/mjl-/mox/mox-"
|
|
"github.com/mjl-/mox/sasl"
|
|
"github.com/mjl-/mox/smtpclient"
|
|
"github.com/mjl-/mox/store"
|
|
)
|
|
|
|
var ctxbg = context.Background()
|
|
|
|
func tcheck(t *testing.T, err error, msg string) {
|
|
t.Helper()
|
|
if err != nil {
|
|
t.Fatalf("%s: %s", msg, err)
|
|
}
|
|
}
|
|
|
|
// Submit a message to mox, which sends it to postfix, which forwards back to mox.
|
|
// We check if we receive the message.
|
|
func TestDeliver(t *testing.T) {
|
|
mlog.Logfmt = true
|
|
|
|
// Remove state.
|
|
os.RemoveAll("testdata/integration/data")
|
|
os.MkdirAll("testdata/integration/data", 0750)
|
|
|
|
// Cleanup afterwards, these are owned by root, annoying to have around due to
|
|
// permission errors.
|
|
defer os.RemoveAll("testdata/integration/data")
|
|
|
|
// Load mox config.
|
|
mox.ConfigStaticPath = "testdata/integration/config/mox.conf"
|
|
filepath.Join(filepath.Dir(mox.ConfigStaticPath), "domains.conf")
|
|
if errs := mox.LoadConfig(ctxbg, true, false); len(errs) > 0 {
|
|
t.Fatalf("loading mox config: %v", errs)
|
|
}
|
|
|
|
// Create new accounts
|
|
createAccount := func(email, password string) {
|
|
t.Helper()
|
|
acc, _, err := store.OpenEmail(email)
|
|
tcheck(t, err, "open account")
|
|
err = acc.SetPassword(password)
|
|
tcheck(t, err, "setting password")
|
|
err = acc.Close()
|
|
tcheck(t, err, "closing account")
|
|
}
|
|
|
|
createAccount("moxtest1@mox1.example", "pass1234")
|
|
createAccount("moxtest2@mox2.example", "pass1234")
|
|
createAccount("moxtest3@mox3.example", "pass1234")
|
|
|
|
// Start mox.
|
|
const mtastsdbRefresher = false
|
|
const skipForkExec = true
|
|
err := start(mtastsdbRefresher, skipForkExec)
|
|
tcheck(t, err, "starting mox")
|
|
|
|
// Single update from IMAP IDLE.
|
|
type idleResponse struct {
|
|
untagged imapclient.Untagged
|
|
err error
|
|
}
|
|
|
|
deliver := func(username, desthost, mailfrom, password, rcptto, imapuser string) {
|
|
t.Helper()
|
|
|
|
// Make IMAP connection, we'll wait for a delivery notification with IDLE.
|
|
imapconn, err := net.Dial("tcp", "moxmail1.mox1.example:143")
|
|
tcheck(t, err, "dial imap server")
|
|
defer imapconn.Close()
|
|
client, err := imapclient.New(imapconn, false)
|
|
tcheck(t, err, "new imapclient")
|
|
_, _, err = client.Login(imapuser, "pass1234")
|
|
tcheck(t, err, "imap client login")
|
|
_, _, err = client.Select("inbox")
|
|
tcheck(t, err, "imap select inbox")
|
|
|
|
err = client.Commandf("", "idle")
|
|
tcheck(t, err, "imap idle command")
|
|
|
|
_, _, _, err = client.ReadContinuation()
|
|
tcheck(t, err, "read imap continuation")
|
|
|
|
idle := make(chan idleResponse)
|
|
go func() {
|
|
for {
|
|
untagged, err := client.ReadUntagged()
|
|
idle <- idleResponse{untagged, err}
|
|
if err != nil {
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
defer func() {
|
|
err := client.Writelinef("done")
|
|
tcheck(t, err, "aborting idle")
|
|
}()
|
|
|
|
conn, err := net.Dial("tcp", desthost+":587")
|
|
tcheck(t, err, "dial submission")
|
|
defer conn.Close()
|
|
|
|
msg := fmt.Sprintf(`From: <%s>
|
|
To: <%s>
|
|
Subject: test message
|
|
|
|
This is the message.
|
|
`, mailfrom, rcptto)
|
|
msg = strings.ReplaceAll(msg, "\n", "\r\n")
|
|
auth := []sasl.Client{sasl.NewClientPlain(mailfrom, password)}
|
|
c, err := smtpclient.New(mox.Context, mlog.New("test"), conn, smtpclient.TLSOpportunistic, mox.Conf.Static.HostnameDomain, dns.Domain{ASCII: desthost}, auth)
|
|
tcheck(t, err, "smtp hello")
|
|
t0 := time.Now()
|
|
err = c.Deliver(mox.Context, mailfrom, rcptto, int64(len(msg)), strings.NewReader(msg), false, false)
|
|
tcheck(t, err, "deliver with smtp")
|
|
err = c.Close()
|
|
tcheck(t, err, "close smtpclient")
|
|
|
|
// Wait for notification of delivery.
|
|
select {
|
|
case resp := <-idle:
|
|
tcheck(t, resp.err, "idle notification")
|
|
_, ok := resp.untagged.(imapclient.UntaggedExists)
|
|
if !ok {
|
|
t.Fatalf("got idle %#v, expected untagged exists", resp.untagged)
|
|
}
|
|
if d := time.Since(t0); d < 1*time.Second {
|
|
t.Fatalf("delivery took %v, bt should have taken at least 1 second, the first-time sender delay", d)
|
|
}
|
|
case <-time.After(5 * time.Second):
|
|
t.Fatalf("timeout after 5s waiting for IMAP IDLE notification of new message, should take about 1 second")
|
|
}
|
|
}
|
|
|
|
deliver("moxtest1", "moxmail1.mox1.example", "moxtest1@mox1.example", "pass1234", "root@postfix.example", "moxtest1@mox1.example")
|
|
deliver("moxtest3", "moxmail2.mox2.example", "moxtest2@mox2.example", "pass1234", "moxtest3@mox3.example", "moxtest3@mox3.example")
|
|
}
|