mirror of
https://github.com/mjl-/mox.git
synced 2025-01-28 07:15:55 +03:00
03c3f56a59
this doesn't really test the output of the ctl commands, just that they succeed without error. better than nothing... testing found two small bugs, that are not an issue in practice: 1. we were ack'ing streamed data from the other side of the ctl connection before having read it. when there is no buffer space on the connection (always the case for net.Pipe) that would cause a deadlock. only actually happened during the new tests. 2. the generated dkim keys are relatively to the directory of the dynamic config file. mox looked it up relative to the directory of the _static_ config file at startup. this directory is typicaly the same. users would have noticed if they had triggered this.
145 lines
4 KiB
Go
145 lines
4 KiB
Go
//go:build quickstart
|
|
|
|
// Run this using docker-compose.yml, see Makefile.
|
|
|
|
package main
|
|
|
|
import (
|
|
"context"
|
|
"crypto/tls"
|
|
"fmt"
|
|
"os"
|
|
"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"
|
|
)
|
|
|
|
var ctxbg = context.Background()
|
|
|
|
func tcheck(t *testing.T, err error, errmsg string) {
|
|
if err != nil {
|
|
t.Fatalf("%s: %s", errmsg, err)
|
|
}
|
|
}
|
|
|
|
func TestDeliver(t *testing.T) {
|
|
xlog := mlog.New("quickstart")
|
|
mlog.Logfmt = true
|
|
|
|
hostname, err := os.Hostname()
|
|
tcheck(t, err, "hostname")
|
|
ourHostname, err := dns.ParseDomain(hostname)
|
|
tcheck(t, err, "parse hostname")
|
|
|
|
// Deliver submits a message over submissions, and checks with imap idle if the
|
|
// message is received by the destination mail server.
|
|
deliver := func(desthost, mailfrom, password, rcptto, imaphost, imapuser, imappassword string) {
|
|
t.Helper()
|
|
|
|
// Connect to IMAP, execute IDLE command, which will return on deliver message.
|
|
// TLS certificates work because the container has the CA certificates configured.
|
|
imapconn, err := tls.Dial("tcp", imaphost+":993", nil)
|
|
tcheck(t, err, "dial imap")
|
|
defer imapconn.Close()
|
|
|
|
imaperr := make(chan error, 1)
|
|
go func() {
|
|
go func() {
|
|
x := recover()
|
|
if x == nil {
|
|
return
|
|
}
|
|
imaperr <- x.(error)
|
|
}()
|
|
xcheck := func(err error, format string) {
|
|
if err != nil {
|
|
panic(fmt.Errorf("%s: %w", format, err))
|
|
}
|
|
}
|
|
|
|
imapc, err := imapclient.New(imapconn, false)
|
|
xcheck(err, "new imapclient")
|
|
|
|
_, _, err = imapc.Login(imapuser, imappassword)
|
|
xcheck(err, "imap login")
|
|
|
|
_, _, err = imapc.Select("Inbox")
|
|
xcheck(err, "imap select inbox")
|
|
|
|
err = imapc.Commandf("", "idle")
|
|
xcheck(err, "write imap idle command")
|
|
_, _, _, err = imapc.ReadContinuation()
|
|
xcheck(err, "read imap continuation")
|
|
|
|
done := make(chan error)
|
|
go func() {
|
|
defer func() {
|
|
x := recover()
|
|
if x != nil {
|
|
done <- fmt.Errorf("%v", x)
|
|
}
|
|
}()
|
|
untagged, err := imapc.ReadUntagged()
|
|
if err != nil {
|
|
done <- err
|
|
return
|
|
}
|
|
if _, ok := untagged.(imapclient.UntaggedExists); !ok {
|
|
done <- fmt.Errorf("expected imapclient.UntaggedExists, got %#v", untagged)
|
|
return
|
|
}
|
|
done <- nil
|
|
}()
|
|
|
|
period := 30 * time.Second
|
|
timer := time.NewTimer(period)
|
|
defer timer.Stop()
|
|
select {
|
|
case err = <-done:
|
|
case <-timer.C:
|
|
err = fmt.Errorf("nothing within %v", period)
|
|
}
|
|
xcheck(err, "waiting for imap untagged response to idle")
|
|
imaperr <- nil
|
|
}()
|
|
|
|
conn, err := tls.Dial("tcp", desthost+":465", nil)
|
|
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, xlog, conn, smtpclient.TLSSkip, ourHostname, dns.Domain{ASCII: desthost}, auth)
|
|
tcheck(t, err, "smtp hello")
|
|
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")
|
|
|
|
err = <-imaperr
|
|
tcheck(t, err, "imap idle")
|
|
}
|
|
|
|
xlog.Print("submitting email to moxacmepebble, waiting for imap notification at moxmail2")
|
|
t0 := time.Now()
|
|
deliver("moxacmepebble.mox1.example", "moxtest1@mox1.example", "accountpass1234", "moxtest2@mox2.example", "moxmail2.mox2.example", "moxtest2@mox2.example", "accountpass4321")
|
|
xlog.Print("success", mlog.Field("duration", time.Since(t0)))
|
|
|
|
xlog.Print("submitting email to moxmail2, waiting for imap notification at moxacmepebble")
|
|
t0 = time.Now()
|
|
deliver("moxmail2.mox2.example", "moxtest2@mox2.example", "accountpass4321", "moxtest1@mox1.example", "moxacmepebble.mox1.example", "moxtest1@mox1.example", "accountpass1234")
|
|
xlog.Print("success", mlog.Field("duration", time.Since(t0)))
|
|
}
|