mirror of
https://github.com/foxcpp/maddy.git
synced 2025-04-05 22:17:39 +03:00
Move most code from the repo root into subdirectories
The intention is to keep to repo root clean while the list of packages is slowly growing. Additionally, a bunch of small (~30 LoC) files in the repo root is merged into a single maddy.go file, for the same reason. Most of the internal code is moved into the internal/ directory. Go toolchain will make it impossible to import these packages from external applications. Some packages are renamed and moved into the pkg/ directory in the root. According to https://github.com/golang-standards/project-layout this is the de-facto standard to place "library code that's ok to use by external applications" in. To clearly define the purpose of top-level directories, README.md files are added to each.
This commit is contained in:
parent
c4df3af4af
commit
bf188e454f
180 changed files with 722 additions and 684 deletions
430
internal/msgpipeline/check_test.go
Normal file
430
internal/msgpipeline/check_test.go
Normal file
|
@ -0,0 +1,430 @@
|
|||
package msgpipeline
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"testing"
|
||||
|
||||
"github.com/emersion/go-message/textproto"
|
||||
"github.com/emersion/go-msgauth/authres"
|
||||
"github.com/foxcpp/maddy/internal/module"
|
||||
"github.com/foxcpp/maddy/internal/testutils"
|
||||
)
|
||||
|
||||
func TestMsgPipeline_Checks(t *testing.T) {
|
||||
target := testutils.Target{}
|
||||
check1, check2 := testutils.Check{}, testutils.Check{}
|
||||
d := MsgPipeline{
|
||||
msgpipelineCfg: msgpipelineCfg{
|
||||
globalChecks: []module.Check{&check1, &check2},
|
||||
perSource: map[string]sourceBlock{},
|
||||
defaultSource: sourceBlock{
|
||||
perRcpt: map[string]*rcptBlock{},
|
||||
defaultRcpt: &rcptBlock{
|
||||
targets: []module.DeliveryTarget{&target},
|
||||
},
|
||||
},
|
||||
},
|
||||
Log: testutils.Logger(t, "msgpipeline"),
|
||||
}
|
||||
|
||||
testutils.DoTestDelivery(t, &d, "whatever@whatever", []string{"whatever@whatever"})
|
||||
|
||||
if len(target.Messages) != 1 {
|
||||
t.Fatalf("wrong amount of messages received, want %d, got %d", 1, len(target.Messages))
|
||||
}
|
||||
if target.Messages[0].MsgMeta.Quarantine {
|
||||
t.Fatalf("message is quarantined when it shouldn't")
|
||||
}
|
||||
|
||||
if check1.UnclosedStates != 0 || check2.UnclosedStates != 0 {
|
||||
t.Fatalf("checks state objects leak or double-closed, alive counters: %v, %v", check1.UnclosedStates, check2.UnclosedStates)
|
||||
}
|
||||
}
|
||||
|
||||
func TestMsgPipeline_AuthResults(t *testing.T) {
|
||||
target := testutils.Target{}
|
||||
check1, check2 := testutils.Check{
|
||||
BodyRes: module.CheckResult{
|
||||
AuthResult: []authres.Result{
|
||||
&authres.SPFResult{
|
||||
Value: authres.ResultFail,
|
||||
From: "FROM",
|
||||
Helo: "HELO",
|
||||
},
|
||||
},
|
||||
},
|
||||
}, testutils.Check{
|
||||
BodyRes: module.CheckResult{
|
||||
AuthResult: []authres.Result{
|
||||
&authres.SPFResult{
|
||||
Value: authres.ResultFail,
|
||||
From: "FROM2",
|
||||
Helo: "HELO2",
|
||||
},
|
||||
},
|
||||
},
|
||||
}
|
||||
d := MsgPipeline{
|
||||
msgpipelineCfg: msgpipelineCfg{
|
||||
globalChecks: []module.Check{&check1, &check2},
|
||||
perSource: map[string]sourceBlock{},
|
||||
defaultSource: sourceBlock{
|
||||
perRcpt: map[string]*rcptBlock{},
|
||||
defaultRcpt: &rcptBlock{
|
||||
targets: []module.DeliveryTarget{&target},
|
||||
},
|
||||
},
|
||||
},
|
||||
Hostname: "TEST-HOST",
|
||||
Log: testutils.Logger(t, "msgpipeline"),
|
||||
}
|
||||
|
||||
testutils.DoTestDelivery(t, &d, "whatever@whatever", []string{"whatever@whatever"})
|
||||
|
||||
if len(target.Messages) != 1 {
|
||||
t.Fatalf("wrong amount of messages received, want %d, got %d", 1, len(target.Messages))
|
||||
}
|
||||
|
||||
authRes := target.Messages[0].Header.Get("Authentication-Results")
|
||||
id, parsed, err := authres.Parse(authRes)
|
||||
if err != nil {
|
||||
t.Fatalf("failed to parse results")
|
||||
}
|
||||
if id != "TEST-HOST" {
|
||||
t.Fatalf("wrong authres identifier")
|
||||
}
|
||||
if len(parsed) != 2 {
|
||||
t.Fatalf("wrong amount of parts, want %d, got %d", 2, len(parsed))
|
||||
}
|
||||
|
||||
var seen1, seen2 bool
|
||||
for _, parts := range parsed {
|
||||
spfPart, ok := parts.(*authres.SPFResult)
|
||||
if !ok {
|
||||
t.Fatalf("Not SPFResult")
|
||||
}
|
||||
|
||||
if spfPart.From == "FROM" {
|
||||
seen1 = true
|
||||
}
|
||||
if spfPart.From == "FROM2" {
|
||||
seen2 = true
|
||||
}
|
||||
}
|
||||
|
||||
if !seen1 {
|
||||
t.Fatalf("First authRes is missing")
|
||||
}
|
||||
if !seen2 {
|
||||
t.Fatalf("Second authRes is missing")
|
||||
}
|
||||
|
||||
if check1.UnclosedStates != 0 || check2.UnclosedStates != 0 {
|
||||
t.Fatalf("checks state objects leak or double-closed, alive counters: %v, %v", check1.UnclosedStates, check2.UnclosedStates)
|
||||
}
|
||||
}
|
||||
|
||||
func TestMsgPipeline_Headers(t *testing.T) {
|
||||
hdr1 := textproto.Header{}
|
||||
hdr1.Add("HDR1", "1")
|
||||
hdr2 := textproto.Header{}
|
||||
hdr2.Add("HDR2", "2")
|
||||
|
||||
target := testutils.Target{}
|
||||
check1, check2 := testutils.Check{
|
||||
BodyRes: module.CheckResult{
|
||||
Header: hdr1,
|
||||
},
|
||||
}, testutils.Check{
|
||||
BodyRes: module.CheckResult{
|
||||
Header: hdr2,
|
||||
},
|
||||
}
|
||||
d := MsgPipeline{
|
||||
msgpipelineCfg: msgpipelineCfg{
|
||||
globalChecks: []module.Check{&check1, &check2},
|
||||
perSource: map[string]sourceBlock{},
|
||||
defaultSource: sourceBlock{
|
||||
perRcpt: map[string]*rcptBlock{},
|
||||
defaultRcpt: &rcptBlock{
|
||||
targets: []module.DeliveryTarget{&target},
|
||||
},
|
||||
},
|
||||
},
|
||||
Hostname: "TEST-HOST",
|
||||
Log: testutils.Logger(t, "msgpipeline"),
|
||||
}
|
||||
|
||||
testutils.DoTestDelivery(t, &d, "whatever@whatever", []string{"whatever@whatever"})
|
||||
|
||||
if len(target.Messages) != 1 {
|
||||
t.Fatalf("wrong amount of messages received, want %d, got %d", 1, len(target.Messages))
|
||||
}
|
||||
|
||||
if target.Messages[0].Header.Get("HDR1") != "1" {
|
||||
t.Fatalf("wrong HDR1 value, want %s, got %s", "1", target.Messages[0].Header.Get("HDR1"))
|
||||
}
|
||||
if target.Messages[0].Header.Get("HDR2") != "2" {
|
||||
t.Fatalf("wrong HDR2 value, want %s, got %s", "1", target.Messages[0].Header.Get("HDR2"))
|
||||
}
|
||||
|
||||
if check1.UnclosedStates != 0 || check2.UnclosedStates != 0 {
|
||||
t.Fatalf("checks state objects leak or double-closed, alive counters: %v, %v", check1.UnclosedStates, check2.UnclosedStates)
|
||||
}
|
||||
}
|
||||
|
||||
func TestMsgPipeline_Globalcheck_Errors(t *testing.T) {
|
||||
target := testutils.Target{}
|
||||
check_ := testutils.Check{
|
||||
InitErr: errors.New("1"),
|
||||
ConnRes: module.CheckResult{Reject: true, Reason: errors.New("2")},
|
||||
SenderRes: module.CheckResult{Reject: true, Reason: errors.New("3")},
|
||||
RcptRes: module.CheckResult{Reject: true, Reason: errors.New("4")},
|
||||
BodyRes: module.CheckResult{Reject: true, Reason: errors.New("5")},
|
||||
}
|
||||
d := MsgPipeline{
|
||||
msgpipelineCfg: msgpipelineCfg{
|
||||
globalChecks: []module.Check{&check_},
|
||||
perSource: map[string]sourceBlock{},
|
||||
defaultSource: sourceBlock{
|
||||
perRcpt: map[string]*rcptBlock{},
|
||||
defaultRcpt: &rcptBlock{
|
||||
targets: []module.DeliveryTarget{&target},
|
||||
},
|
||||
},
|
||||
},
|
||||
Hostname: "TEST-HOST",
|
||||
Log: testutils.Logger(t, "msgpipeline"),
|
||||
}
|
||||
|
||||
t.Run("init err", func(t *testing.T) {
|
||||
_, err := testutils.DoTestDeliveryErr(t, &d, "sender@example.com", []string{"rcpt1@example.com", "rcpt2@example.com"})
|
||||
if err == nil {
|
||||
t.Fatal("expected error")
|
||||
}
|
||||
})
|
||||
|
||||
check_.InitErr = nil
|
||||
|
||||
t.Run("conn err", func(t *testing.T) {
|
||||
_, err := testutils.DoTestDeliveryErr(t, &d, "sender@example.com", []string{"rcpt1@example.com", "rcpt2@example.com"})
|
||||
if err == nil {
|
||||
t.Fatal("expected error")
|
||||
}
|
||||
})
|
||||
|
||||
check_.ConnRes.Reject = false
|
||||
|
||||
t.Run("mail from err", func(t *testing.T) {
|
||||
_, err := testutils.DoTestDeliveryErr(t, &d, "sender@example.com", []string{"rcpt1@example.com", "rcpt2@example.com"})
|
||||
if err == nil {
|
||||
t.Fatal("expected error")
|
||||
}
|
||||
})
|
||||
|
||||
check_.SenderRes.Reject = false
|
||||
|
||||
t.Run("rcpt to err", func(t *testing.T) {
|
||||
_, err := testutils.DoTestDeliveryErr(t, &d, "sender@example.com", []string{"rcpt1@example.com", "rcpt2@example.com"})
|
||||
if err == nil {
|
||||
t.Fatal("expected error")
|
||||
}
|
||||
})
|
||||
|
||||
check_.RcptRes.Reject = false
|
||||
|
||||
t.Run("body err", func(t *testing.T) {
|
||||
_, err := testutils.DoTestDeliveryErr(t, &d, "sender@example.com", []string{"rcpt1@example.com", "rcpt2@example.com"})
|
||||
if err == nil {
|
||||
t.Fatal("expected error")
|
||||
}
|
||||
})
|
||||
|
||||
check_.BodyRes.Reject = false
|
||||
|
||||
t.Run("no err", func(t *testing.T) {
|
||||
testutils.DoTestDelivery(t, &d, "sender@example.com", []string{"rcpt1@example.com", "rcpt2@example.com"})
|
||||
})
|
||||
|
||||
if check_.UnclosedStates != 0 {
|
||||
t.Fatalf("check state objects leak or double-closed, counters: %d", check_.UnclosedStates)
|
||||
}
|
||||
}
|
||||
|
||||
func TestMsgPipeline_SourceCheck_Errors(t *testing.T) {
|
||||
target := testutils.Target{}
|
||||
check_ := testutils.Check{
|
||||
InitErr: errors.New("1"),
|
||||
ConnRes: module.CheckResult{Reject: true, Reason: errors.New("2")},
|
||||
SenderRes: module.CheckResult{Reject: true, Reason: errors.New("3")},
|
||||
RcptRes: module.CheckResult{Reject: true, Reason: errors.New("4")},
|
||||
BodyRes: module.CheckResult{Reject: true, Reason: errors.New("5")},
|
||||
}
|
||||
globalCheck := testutils.Check{}
|
||||
d := MsgPipeline{
|
||||
msgpipelineCfg: msgpipelineCfg{
|
||||
globalChecks: []module.Check{&globalCheck},
|
||||
perSource: map[string]sourceBlock{},
|
||||
defaultSource: sourceBlock{
|
||||
perRcpt: map[string]*rcptBlock{},
|
||||
checks: []module.Check{&check_},
|
||||
defaultRcpt: &rcptBlock{
|
||||
targets: []module.DeliveryTarget{&target},
|
||||
},
|
||||
},
|
||||
},
|
||||
Hostname: "TEST-HOST",
|
||||
Log: testutils.Logger(t, "msgpipeline"),
|
||||
}
|
||||
|
||||
t.Run("init err", func(t *testing.T) {
|
||||
_, err := testutils.DoTestDeliveryErr(t, &d, "sender@example.com", []string{"rcpt1@example.com", "rcpt2@example.com"})
|
||||
if err == nil {
|
||||
t.Fatal("expected error")
|
||||
}
|
||||
})
|
||||
|
||||
check_.InitErr = nil
|
||||
|
||||
t.Run("conn err", func(t *testing.T) {
|
||||
_, err := testutils.DoTestDeliveryErr(t, &d, "sender@example.com", []string{"rcpt1@example.com", "rcpt2@example.com"})
|
||||
if err == nil {
|
||||
t.Fatal("expected error")
|
||||
}
|
||||
})
|
||||
|
||||
check_.ConnRes.Reject = false
|
||||
|
||||
t.Run("mail from err", func(t *testing.T) {
|
||||
_, err := testutils.DoTestDeliveryErr(t, &d, "sender@example.com", []string{"rcpt1@example.com", "rcpt2@example.com"})
|
||||
if err == nil {
|
||||
t.Fatal("expected error")
|
||||
}
|
||||
})
|
||||
|
||||
check_.SenderRes.Reject = false
|
||||
|
||||
t.Run("rcpt to err", func(t *testing.T) {
|
||||
_, err := testutils.DoTestDeliveryErr(t, &d, "sender@example.com", []string{"rcpt1@example.com", "rcpt2@example.com"})
|
||||
if err == nil {
|
||||
t.Fatal("expected error")
|
||||
}
|
||||
})
|
||||
|
||||
check_.RcptRes.Reject = false
|
||||
|
||||
t.Run("body err", func(t *testing.T) {
|
||||
_, err := testutils.DoTestDeliveryErr(t, &d, "sender@example.com", []string{"rcpt1@example.com", "rcpt2@example.com"})
|
||||
if err == nil {
|
||||
t.Fatal("expected error")
|
||||
}
|
||||
})
|
||||
|
||||
check_.BodyRes.Reject = false
|
||||
|
||||
t.Run("no err", func(t *testing.T) {
|
||||
testutils.DoTestDelivery(t, &d, "sender@example.com", []string{"rcpt1@example.com", "rcpt2@example.com"})
|
||||
})
|
||||
|
||||
if check_.UnclosedStates != 0 || globalCheck.UnclosedStates != 0 {
|
||||
t.Fatalf("check state objects leak or double-closed, counters: %d, %d",
|
||||
check_.UnclosedStates, globalCheck.UnclosedStates)
|
||||
}
|
||||
}
|
||||
|
||||
func TestMsgPipeline_RcptCheck_Errors(t *testing.T) {
|
||||
target := testutils.Target{}
|
||||
check_ := testutils.Check{
|
||||
InitErr: errors.New("1"),
|
||||
ConnRes: module.CheckResult{Reject: true, Reason: errors.New("2")},
|
||||
SenderRes: module.CheckResult{Reject: true, Reason: errors.New("3")},
|
||||
RcptRes: module.CheckResult{Reject: true, Reason: errors.New("4")},
|
||||
BodyRes: module.CheckResult{Reject: true, Reason: errors.New("5")},
|
||||
|
||||
InstName: "err_check",
|
||||
}
|
||||
// Added to check whether it leaks.
|
||||
globalCheck := testutils.Check{InstName: "global_check"}
|
||||
sourceCheck := testutils.Check{InstName: "source_check"}
|
||||
d := MsgPipeline{
|
||||
msgpipelineCfg: msgpipelineCfg{
|
||||
globalChecks: []module.Check{&globalCheck},
|
||||
perSource: map[string]sourceBlock{},
|
||||
defaultSource: sourceBlock{
|
||||
perRcpt: map[string]*rcptBlock{},
|
||||
checks: []module.Check{&check_},
|
||||
defaultRcpt: &rcptBlock{
|
||||
targets: []module.DeliveryTarget{&target},
|
||||
},
|
||||
},
|
||||
},
|
||||
Hostname: "TEST-HOST",
|
||||
Log: testutils.Logger(t, "msgpipeline"),
|
||||
}
|
||||
|
||||
t.Run("init err", func(t *testing.T) {
|
||||
d.Log = testutils.Logger(t, "msgpipeline")
|
||||
_, err := testutils.DoTestDeliveryErr(t, &d, "sender@example.com", []string{"rcpt1@example.com", "rcpt2@example.com"})
|
||||
if err == nil {
|
||||
t.Fatal("expected error")
|
||||
}
|
||||
|
||||
t.Log("!!!", check_.UnclosedStates)
|
||||
})
|
||||
|
||||
check_.InitErr = nil
|
||||
|
||||
t.Run("conn err", func(t *testing.T) {
|
||||
d.Log = testutils.Logger(t, "msgpipeline")
|
||||
_, err := testutils.DoTestDeliveryErr(t, &d, "sender@example.com", []string{"rcpt1@example.com", "rcpt2@example.com"})
|
||||
if err == nil {
|
||||
t.Fatal("expected error")
|
||||
}
|
||||
|
||||
t.Log("!!!", check_.UnclosedStates)
|
||||
})
|
||||
|
||||
check_.ConnRes.Reject = false
|
||||
|
||||
t.Run("mail from err", func(t *testing.T) {
|
||||
d.Log = testutils.Logger(t, "msgpipeline")
|
||||
_, err := testutils.DoTestDeliveryErr(t, &d, "sender@example.com", []string{"rcpt1@example.com", "rcpt2@example.com"})
|
||||
if err == nil {
|
||||
t.Fatal("expected error")
|
||||
}
|
||||
|
||||
t.Log("!!!", check_.UnclosedStates)
|
||||
})
|
||||
|
||||
check_.SenderRes.Reject = false
|
||||
|
||||
t.Run("rcpt to err", func(t *testing.T) {
|
||||
d.Log = testutils.Logger(t, "msgpipeline")
|
||||
_, err := testutils.DoTestDeliveryErr(t, &d, "sender@example.com", []string{"rcpt1@example.com", "rcpt2@example.com"})
|
||||
if err == nil {
|
||||
t.Fatal("expected error")
|
||||
}
|
||||
})
|
||||
|
||||
check_.RcptRes.Reject = false
|
||||
|
||||
t.Run("body err", func(t *testing.T) {
|
||||
d.Log = testutils.Logger(t, "msgpipeline")
|
||||
_, err := testutils.DoTestDeliveryErr(t, &d, "sender@example.com", []string{"rcpt1@example.com", "rcpt2@example.com"})
|
||||
if err == nil {
|
||||
t.Fatal("expected error")
|
||||
}
|
||||
})
|
||||
|
||||
check_.BodyRes.Reject = false
|
||||
|
||||
t.Run("no err", func(t *testing.T) {
|
||||
d.Log = testutils.Logger(t, "msgpipeline")
|
||||
testutils.DoTestDelivery(t, &d, "sender@example.com", []string{"rcpt1@example.com", "rcpt2@example.com"})
|
||||
})
|
||||
|
||||
if check_.UnclosedStates != 0 || sourceCheck.UnclosedStates != 0 || globalCheck.UnclosedStates != 0 {
|
||||
t.Fatalf("check state objects leak or double-closed, counters: %d, %d, %d",
|
||||
check_.UnclosedStates, sourceCheck.UnclosedStates, globalCheck.UnclosedStates)
|
||||
}
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue