maddy/internal/msgpipeline/config_test.go
fox.cpp bf188e454f
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.
2019-12-06 01:35:12 +03:00

410 lines
7.5 KiB
Go

package msgpipeline
import (
"reflect"
"strings"
"testing"
"github.com/foxcpp/maddy/internal/exterrors"
parser "github.com/foxcpp/maddy/pkg/cfgparser"
)
func policyError(code int) error {
return &exterrors.SMTPError{
Message: "Message rejected due to a local policy",
Code: code,
EnhancedCode: exterrors.EnhancedCode{5, 7, 0},
Reason: "reject directive used",
}
}
func TestMsgPipelineCfg(t *testing.T) {
cases := []struct {
name string
str string
value msgpipelineCfg
fail bool
}{
{
name: "basic",
str: `
source example.com {
destination example.org {
reject 410
}
default_destination {
reject 420
}
}
default_source {
destination example.org {
reject 430
}
default_destination {
reject 440
}
}`,
value: msgpipelineCfg{
perSource: map[string]sourceBlock{
"example.com": {
perRcpt: map[string]*rcptBlock{
"example.org": {
rejectErr: policyError(410),
},
},
defaultRcpt: &rcptBlock{
rejectErr: policyError(420),
},
},
},
defaultSource: sourceBlock{
perRcpt: map[string]*rcptBlock{
"example.org": {
rejectErr: policyError(430),
},
},
defaultRcpt: &rcptBlock{
rejectErr: policyError(440),
},
},
},
},
{
name: "implied default destination",
str: `
source example.com {
reject 410
}
default_source {
reject 420
}`,
value: msgpipelineCfg{
perSource: map[string]sourceBlock{
"example.com": {
perRcpt: map[string]*rcptBlock{},
defaultRcpt: &rcptBlock{
rejectErr: policyError(410),
},
},
},
defaultSource: sourceBlock{
perRcpt: map[string]*rcptBlock{},
defaultRcpt: &rcptBlock{
rejectErr: policyError(420),
},
},
},
},
{
name: "implied default sender",
str: `
destination example.com {
reject 410
}
default_destination {
reject 420
}`,
value: msgpipelineCfg{
perSource: map[string]sourceBlock{},
defaultSource: sourceBlock{
perRcpt: map[string]*rcptBlock{
"example.com": {
rejectErr: policyError(410),
},
},
defaultRcpt: &rcptBlock{
rejectErr: policyError(420),
},
},
},
},
{
name: "duplicate source",
str: `
source example.org {
reject 410
}
source eXample.org {
reject 410
}
default_source {
reject 420
}
`,
fail: true,
},
{
name: "missing default source handler",
str: `
source example.org {
reject 410
}`,
fail: true,
},
{
name: "missing default destination handler",
str: `
destination example.org {
reject 410
}`,
fail: true,
},
{
name: "invalid domain",
str: `
destination .. {
reject 410
}
default_destination {
reject 500
}`,
fail: true,
},
{
name: "duplicate destination",
str: `
destination xxx.xxx{
reject 410
}
destination xXx.xxx {
reject 410
}
default_destination {
reject 500
}`,
fail: true,
},
{
name: "invalid address",
str: `
destination @example. {
reject 410
}
default_destination {
reject 500
}`,
fail: true,
},
{
name: "invalid address",
str: `
destination @example. {
reject 421
}
default_destination {
reject 500
}`,
fail: true,
},
{
name: "invalid reject code",
str: `
destination example.com {
reject 200
}
default_destination {
reject 500
}`,
fail: true,
},
{
name: "destination together with source",
str: `
destination example.com {
reject 410
}
source example.org {
reject 420
}
default_source {
reject 430
}`,
fail: true,
},
{
name: "empty destination rule",
str: `
destination {
reject 410
}
default_destination {
reject 420
}`,
fail: true,
},
}
for _, case_ := range cases {
case_ := case_
t.Run(case_.name, func(t *testing.T) {
cfg, _ := parser.Read(strings.NewReader(case_.str), "literal")
parsed, err := parseMsgPipelineRootCfg(nil, cfg)
if err != nil && !case_.fail {
t.Fatalf("unexpected parse error: %v", err)
}
if err == nil && case_.fail {
t.Fatalf("unexpected parse success")
}
if case_.fail {
t.Log(err)
return
}
if !reflect.DeepEqual(parsed, case_.value) {
t.Errorf("Wrong parsed configuration")
t.Errorf("Wanted: %+v", case_.value)
t.Errorf("Got: %+v", parsed)
}
})
}
}
func TestMsgPipelineCfg_GlobalChecks(t *testing.T) {
str := `
check {
test_check
}
default_destination {
reject 500
}
`
cfg, _ := parser.Read(strings.NewReader(str), "literal")
parsed, err := parseMsgPipelineRootCfg(nil, cfg)
if err != nil {
t.Fatalf("unexpected parse error: %v", err)
}
if len(parsed.globalChecks) == 0 {
t.Fatalf("missing test_check in globalChecks")
}
}
func TestMsgPipelineCfg_GlobalChecksMultiple(t *testing.T) {
str := `
check {
test_check
}
check {
test_check
}
default_destination {
reject 500
}
`
cfg, _ := parser.Read(strings.NewReader(str), "literal")
parsed, err := parseMsgPipelineRootCfg(nil, cfg)
if err != nil {
t.Fatalf("unexpected parse error: %v", err)
}
if len(parsed.globalChecks) != 2 {
t.Fatalf("wrong amount of test_check's in globalChecks: %d", len(parsed.globalChecks))
}
}
func TestMsgPipelineCfg_SourceChecks(t *testing.T) {
str := `
source example.org {
check {
test_check
}
reject 500
}
default_source {
reject 500
}
`
cfg, _ := parser.Read(strings.NewReader(str), "literal")
parsed, err := parseMsgPipelineRootCfg(nil, cfg)
if err != nil {
t.Fatalf("unexpected parse error: %v", err)
}
if len(parsed.perSource["example.org"].checks) == 0 {
t.Fatalf("missing test_check in source checks")
}
}
func TestMsgPipelineCfg_SourceChecks_Multiple(t *testing.T) {
str := `
source example.org {
check {
test_check
}
check {
test_check
}
reject 500
}
default_source {
reject 500
}
`
cfg, _ := parser.Read(strings.NewReader(str), "literal")
parsed, err := parseMsgPipelineRootCfg(nil, cfg)
if err != nil {
t.Fatalf("unexpected parse error: %v", err)
}
if len(parsed.perSource["example.org"].checks) != 2 {
t.Fatalf("wrong amount of test_check's in source checks: %d", len(parsed.perSource["example.org"].checks))
}
}
func TestMsgPipelineCfg_RcptChecks(t *testing.T) {
str := `
destination example.org {
check {
test_check
}
reject 500
}
default_destination {
reject 500
}
`
cfg, _ := parser.Read(strings.NewReader(str), "literal")
parsed, err := parseMsgPipelineRootCfg(nil, cfg)
if err != nil {
t.Fatalf("unexpected parse error: %v", err)
}
if len(parsed.defaultSource.perRcpt["example.org"].checks) == 0 {
t.Fatalf("missing test_check in rcpt checks")
}
}
func TestMsgPipelineCfg_RcptChecks_Multiple(t *testing.T) {
str := `
destination example.org {
check {
test_check
}
check {
test_check
}
reject 500
}
default_destination {
reject 500
}
`
cfg, _ := parser.Read(strings.NewReader(str), "literal")
parsed, err := parseMsgPipelineRootCfg(nil, cfg)
if err != nil {
t.Fatalf("unexpected parse error: %v", err)
}
if len(parsed.defaultSource.perRcpt["example.org"].checks) != 2 {
t.Fatalf("wrong amount of test_check's in rcpt checks: %d", len(parsed.defaultSource.perRcpt["example.org"].checks))
}
}