mirror of
https://github.com/foxcpp/maddy.git
synced 2025-04-06 14:37:37 +03:00
It is useful to define background tasks lifetimes more precisely, especially involving timeouts and other cancellation methods. On top of that, several tracing facilities are context-based (e.g. runtime/trace), so it is possible to use them now.
310 lines
7.9 KiB
Go
310 lines
7.9 KiB
Go
package modify
|
|
|
|
import (
|
|
"context"
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
"reflect"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/foxcpp/maddy/internal/config"
|
|
"github.com/foxcpp/maddy/internal/testutils"
|
|
)
|
|
|
|
func TestReadFile(t *testing.T) {
|
|
test := func(file string, expected map[string]string) {
|
|
t.Helper()
|
|
|
|
f, err := ioutil.TempFile("", "maddy-tests-")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer os.Remove(f.Name())
|
|
defer f.Close()
|
|
if _, err := f.WriteString(file); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
actual := map[string]string{}
|
|
err = readFile(f.Name(), actual)
|
|
if expected == nil {
|
|
if err == nil {
|
|
t.Errorf("expected failure, got %+v", actual)
|
|
}
|
|
return
|
|
}
|
|
if err != nil {
|
|
t.Errorf("unexpected failure: %v", err)
|
|
return
|
|
}
|
|
|
|
if !reflect.DeepEqual(actual, expected) {
|
|
t.Errorf("wrong results\n want %+v\n got %+v", expected, actual)
|
|
}
|
|
}
|
|
|
|
test("a: b", map[string]string{"a": "b"})
|
|
test("a@example.org: b@example.com", map[string]string{"a@example.org": "b@example.com"})
|
|
test(`"a @ a"@example.org: b@example.com`, map[string]string{`"a @ a"@example.org`: "b@example.com"})
|
|
test(`a@example.org: "b @ b"@example.com`, map[string]string{`a@example.org`: `"b @ b"@example.com`})
|
|
test(`"a @ a": "b @ b"`, map[string]string{`"a @ a"`: `"b @ b"`})
|
|
test("a: b, c", nil)
|
|
test(": b", nil)
|
|
test(":", nil)
|
|
test("aaa", nil)
|
|
test(": b", nil)
|
|
test(" testing@example.com : arbitrary-whitespace@example.org ",
|
|
map[string]string{"testing@example.com": "arbitrary-whitespace@example.org"})
|
|
test(`# skip comments
|
|
a: b`, map[string]string{"a": "b"})
|
|
test(`# and empty lines
|
|
|
|
a: b`, map[string]string{"a": "b"})
|
|
test("# with whitespace too\n \na: b", map[string]string{"a": "b"})
|
|
test("A: b", map[string]string{"a": "b"})
|
|
test("CaSe: FoLdInG@ExAmPle.OrG", map[string]string{"case": "folding@example.org"})
|
|
|
|
// In case of postmaster handling we are not sure whether this is a <postmaster>
|
|
// or <postmaster@domain> since we try both.
|
|
// Given 'postmaster: b', <postmaster> will be rewritten to <b>
|
|
// but <postmaster@domain> will be rewritten to <b@domain>.
|
|
// As <b> is not a valid address, this can cause problems. Hence we reject
|
|
// this case due to ambiguity and require fully qualified address as a
|
|
// replacement for 'postmaster'.
|
|
test("postmaster: b", nil)
|
|
test("postmaster: b@example.org", map[string]string{"postmaster": "b@example.org"})
|
|
}
|
|
|
|
func TestRewriteRcpt(t *testing.T) {
|
|
test := func(addr, expected string, aliases map[string]string) {
|
|
t.Helper()
|
|
|
|
s := state{m: &Modifier{
|
|
aliases: aliases,
|
|
}}
|
|
|
|
actual, err := s.RewriteRcpt(context.Background(), addr)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if actual != expected {
|
|
t.Errorf("want %s, got %s", expected, actual)
|
|
}
|
|
}
|
|
|
|
test("test@example.org", "test@example.org", nil)
|
|
test("postmaster", "postmaster", nil)
|
|
test("test@example.com", "test@example.org",
|
|
map[string]string{"test@example.com": "test@example.org"})
|
|
test(`"\"test @ test\""@example.com`, "test@example.org",
|
|
map[string]string{`"\"test @ test\""@example.com`: "test@example.org"})
|
|
test(`test@example.com`, `"\"test @ test\""@example.org`,
|
|
map[string]string{`test@example.com`: `"\"test @ test\""@example.org`})
|
|
test(`"\"test @ test\""@example.com`, `"\"b @ b\""@example.com`,
|
|
map[string]string{`"\"test @ test\""`: `"\"b @ b\""`})
|
|
test("TeSt@eXAMple.com", "test@example.org",
|
|
map[string]string{"test@example.com": "test@example.org"})
|
|
test("test@example.com", "test2@example.com",
|
|
map[string]string{"test": "test2"})
|
|
test("test@example.com", "test2@example.org",
|
|
map[string]string{"test": "test2@example.org"})
|
|
test("postmaster", "test2@example.org",
|
|
map[string]string{"postmaster": "test2@example.org"})
|
|
test("TeSt@examPLE.com", "test2@example.com",
|
|
map[string]string{"test": "test2"})
|
|
test("test@example.com", "test3@example.com",
|
|
map[string]string{
|
|
"test@example.com": "test3@example.com",
|
|
"test": "test2",
|
|
})
|
|
test("rcpt@E\u0301.example.com", "rcpt@foo.example.com",
|
|
map[string]string{
|
|
"rcpt@\u00E9.example.com": "rcpt@foo.example.com",
|
|
})
|
|
test("E\u0301@foo.example.com", "rcpt@foo.example.com",
|
|
map[string]string{
|
|
"\u00E9@foo.example.com": "rcpt@foo.example.com",
|
|
})
|
|
|
|
// This case merely documents current behavior, I am not sure
|
|
// whether this is the right behavior or it should hard-fail
|
|
// in this case.
|
|
test("postmaster", "test2",
|
|
map[string]string{"postmaster": "test2"})
|
|
}
|
|
|
|
func TestAliasesReload(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
const file = `cat: dog`
|
|
|
|
f, err := ioutil.TempFile("", "maddy-tests-")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer os.Remove(f.Name())
|
|
if _, err := f.WriteString(file); err != nil {
|
|
f.Close()
|
|
t.Fatal(err)
|
|
}
|
|
f.Close()
|
|
|
|
mod, err := New("", "", nil, []string{f.Name()})
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
aliases := mod.(*Modifier)
|
|
aliases.log = testutils.Logger(t, "alias_file")
|
|
defer aliases.Close()
|
|
|
|
if err := mod.Init(&config.Map{Block: &config.Node{}}); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
// This delay is somehow important. Not sure why.
|
|
time.Sleep(250 * time.Millisecond)
|
|
|
|
if err := ioutil.WriteFile(f.Name(), []byte("dog: cat"), os.ModePerm); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
for i := 0; i < 10; i++ {
|
|
time.Sleep(reloadInterval + 50*time.Millisecond)
|
|
aliases.aliasesLck.RLock()
|
|
if aliases.aliases["dog"] != "" {
|
|
aliases.aliasesLck.RUnlock()
|
|
break
|
|
}
|
|
aliases.aliasesLck.RUnlock()
|
|
}
|
|
|
|
aliases.aliasesLck.RLock()
|
|
defer aliases.aliasesLck.RUnlock()
|
|
if aliases.aliases["dog"] == "" {
|
|
t.Fatal("New aliases were not loaded")
|
|
}
|
|
}
|
|
|
|
func TestAliasesReload_Broken(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
const file = `cat: dog`
|
|
|
|
f, err := ioutil.TempFile("", "maddy-tests-")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer os.Remove(f.Name())
|
|
if _, err := f.WriteString(file); err != nil {
|
|
f.Close()
|
|
t.Fatal(err)
|
|
}
|
|
f.Close()
|
|
|
|
mod, err := New("", "", nil, []string{f.Name()})
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
aliases := mod.(*Modifier)
|
|
aliases.log = testutils.Logger(t, "alias_file")
|
|
defer aliases.Close()
|
|
|
|
if err := mod.Init(&config.Map{Block: &config.Node{}}); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
// This delay is somehow important. Not sure why.
|
|
time.Sleep(250 * time.Millisecond)
|
|
|
|
if err := ioutil.WriteFile(f.Name(), []byte("dog"), os.ModePerm); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
time.Sleep(3 * reloadInterval)
|
|
|
|
aliases.aliasesLck.RLock()
|
|
defer aliases.aliasesLck.RUnlock()
|
|
if aliases.aliases["cat"] == "" {
|
|
t.Fatal("New aliases were loaded or map changed", aliases.aliases)
|
|
}
|
|
}
|
|
|
|
func TestAliasesReload_Removed(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
const file = `cat: dog`
|
|
|
|
f, err := ioutil.TempFile("", "maddy-tests-")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if _, err := f.WriteString(file); err != nil {
|
|
f.Close()
|
|
t.Fatal(err)
|
|
}
|
|
f.Close()
|
|
|
|
mod, err := New("", "", nil, []string{f.Name()})
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
aliases := mod.(*Modifier)
|
|
aliases.log = testutils.Logger(t, "alias_file")
|
|
defer aliases.Close()
|
|
|
|
if err := mod.Init(&config.Map{Block: &config.Node{}}); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
// This delay is somehow important. Not sure why.
|
|
time.Sleep(250 * time.Millisecond)
|
|
|
|
os.Remove(f.Name())
|
|
|
|
time.Sleep(3 * reloadInterval)
|
|
|
|
aliases.aliasesLck.RLock()
|
|
defer aliases.aliasesLck.RUnlock()
|
|
if aliases.aliases["cat"] != "" {
|
|
t.Fatal("Old aliases are still loaded")
|
|
}
|
|
}
|
|
|
|
func TestAliasesInit_NonExistent(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
const file = `cat: dog`
|
|
|
|
f, err := ioutil.TempFile("", "maddy-tests-")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if _, err := f.WriteString(file); err != nil {
|
|
f.Close()
|
|
t.Fatal(err)
|
|
}
|
|
f.Close()
|
|
|
|
mod, err := New("", "", nil, []string{f.Name(), filepath.Join(os.TempDir(), "totally-non-existent-path-i-really-hope")})
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
aliases := mod.(*Modifier)
|
|
aliases.log = testutils.Logger(t, "alias_file")
|
|
defer aliases.Close()
|
|
|
|
if err := mod.Init(&config.Map{Block: &config.Node{}}); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if aliases.aliases["cat"] != "dog" {
|
|
t.Fatal("Aliases are not loaded")
|
|
}
|
|
}
|
|
|
|
func init() {
|
|
reloadInterval = 250 * time.Millisecond
|
|
}
|