2022-06-17 15:05:54 +02:00
|
|
|
// SPDX-FileCopyrightText: 2022 Winni Neessen <winni@neessen.dev>
|
|
|
|
//
|
|
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
|
2022-03-06 15:15:42 +01:00
|
|
|
package mail
|
|
|
|
|
|
|
|
import (
|
2022-03-16 21:02:31 +01:00
|
|
|
"context"
|
2022-03-15 21:10:29 +01:00
|
|
|
"crypto/tls"
|
2022-12-31 12:40:42 +01:00
|
|
|
"errors"
|
2022-03-15 22:37:55 +01:00
|
|
|
"fmt"
|
2022-03-15 21:10:29 +01:00
|
|
|
"net/smtp"
|
2022-03-16 21:02:31 +01:00
|
|
|
"os"
|
2023-01-07 11:31:46 +01:00
|
|
|
"strconv"
|
2022-12-10 13:41:00 +01:00
|
|
|
"strings"
|
2022-03-06 15:15:42 +01:00
|
|
|
"testing"
|
2022-03-07 16:24:49 +01:00
|
|
|
"time"
|
2022-10-17 18:13:00 +02:00
|
|
|
|
|
|
|
"github.com/wneessen/go-mail/auth"
|
2022-03-06 15:15:42 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
// DefaultHost is used as default hostname for the Client
|
|
|
|
const DefaultHost = "localhost"
|
|
|
|
|
2022-03-21 10:17:48 +01:00
|
|
|
// TestRcpt
|
|
|
|
const TestRcpt = "go-mail@mytrashmailer.com"
|
|
|
|
|
2022-03-15 21:10:29 +01:00
|
|
|
// TestNewClient tests the NewClient() method with its default options
|
|
|
|
func TestNewClient(t *testing.T) {
|
|
|
|
host := "mail.example.com"
|
2022-03-15 21:48:36 +01:00
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
host string
|
|
|
|
shouldfail bool
|
|
|
|
}{
|
|
|
|
{"Default", "mail.example.com", false},
|
|
|
|
{"Empty host should fail", "", true},
|
2022-03-15 21:10:29 +01:00
|
|
|
}
|
2022-03-15 21:48:36 +01:00
|
|
|
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
c, err := NewClient(tt.host)
|
|
|
|
if err != nil && !tt.shouldfail {
|
|
|
|
t.Errorf("failed to create new client: %s", err)
|
|
|
|
return
|
|
|
|
}
|
2022-03-15 21:50:25 +01:00
|
|
|
if c.host != tt.host {
|
2022-03-15 21:48:36 +01:00
|
|
|
t.Errorf("failed to create new client. Host expected: %s, got: %s", host, c.host)
|
|
|
|
}
|
|
|
|
if c.cto != DefaultTimeout {
|
|
|
|
t.Errorf("failed to create new client. Timeout expected: %s, got: %s", DefaultTimeout.String(),
|
|
|
|
c.cto.String())
|
|
|
|
}
|
|
|
|
if c.port != DefaultPort {
|
|
|
|
t.Errorf("failed to create new client. Port expected: %d, got: %d", DefaultPort, c.port)
|
|
|
|
}
|
|
|
|
if c.tlspolicy != TLSMandatory {
|
|
|
|
t.Errorf("failed to create new client. TLS policy expected: %d, got: %d", TLSMandatory, c.tlspolicy)
|
|
|
|
}
|
2022-03-15 21:50:25 +01:00
|
|
|
if c.tlsconfig.ServerName != tt.host {
|
2022-03-15 21:48:36 +01:00
|
|
|
t.Errorf("failed to create new client. TLS config host expected: %s, got: %s",
|
|
|
|
host, c.tlsconfig.ServerName)
|
|
|
|
}
|
|
|
|
if c.tlsconfig.MinVersion != DefaultTLSMinVersion {
|
|
|
|
t.Errorf("failed to create new client. TLS config min versino expected: %d, got: %d",
|
|
|
|
DefaultTLSMinVersion, c.tlsconfig.MinVersion)
|
|
|
|
}
|
2022-03-15 22:37:55 +01:00
|
|
|
if c.ServerAddr() != fmt.Sprintf("%s:%d", tt.host, c.port) {
|
|
|
|
t.Errorf("failed to create new client. c.ServerAddr() expected: %s, got: %s",
|
|
|
|
fmt.Sprintf("%s:%d", tt.host, c.port), c.ServerAddr())
|
|
|
|
}
|
2022-03-15 21:48:36 +01:00
|
|
|
})
|
2022-03-15 21:10:29 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestNewClient tests the NewClient() method with its custom options
|
|
|
|
func TestNewClientWithOptions(t *testing.T) {
|
|
|
|
host := "mail.example.com"
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
option Option
|
|
|
|
shouldfail bool
|
|
|
|
}{
|
|
|
|
{"nil option", nil, true},
|
|
|
|
{"WithPort()", WithPort(465), false},
|
|
|
|
{"WithPort(); port is too high", WithPort(100000), true},
|
|
|
|
{"WithTimeout()", WithTimeout(time.Second * 5), false},
|
|
|
|
{"WithTimeout()", WithTimeout(-10), true},
|
|
|
|
{"WithSSL()", WithSSL(), false},
|
|
|
|
{"WithHELO()", WithHELO(host), false},
|
2022-03-15 22:37:55 +01:00
|
|
|
{"WithHELO(); helo is empty", WithHELO(""), true},
|
2022-03-15 21:10:29 +01:00
|
|
|
{"WithTLSPolicy()", WithTLSPolicy(TLSOpportunistic), false},
|
|
|
|
{"WithTLSConfig()", WithTLSConfig(&tls.Config{}), false},
|
|
|
|
{"WithTLSConfig(); config is nil", WithTLSConfig(nil), true},
|
|
|
|
{"WithSMTPAuth()", WithSMTPAuth(SMTPAuthLogin), false},
|
2022-10-17 18:13:00 +02:00
|
|
|
{
|
|
|
|
"WithSMTPAuthCustom()",
|
2022-03-15 21:10:29 +01:00
|
|
|
WithSMTPAuthCustom(smtp.PlainAuth("", "", "", "")),
|
2022-10-17 18:13:00 +02:00
|
|
|
false,
|
|
|
|
},
|
2022-03-15 21:10:29 +01:00
|
|
|
{"WithUsername()", WithUsername("test"), false},
|
|
|
|
{"WithPassword()", WithPassword("test"), false},
|
2022-09-11 21:07:15 +02:00
|
|
|
{"WithDSN()", WithDSN(), false},
|
|
|
|
{"WithDSNMailReturnType()", WithDSNMailReturnType(DSNMailReturnFull), false},
|
|
|
|
{"WithDSNMailReturnType() wrong option", WithDSNMailReturnType("FAIL"), true},
|
|
|
|
{"WithDSNRcptNotifyType()", WithDSNRcptNotifyType(DSNRcptNotifySuccess), false},
|
|
|
|
{"WithDSNRcptNotifyType() wrong option", WithDSNRcptNotifyType("FAIL"), true},
|
2022-12-26 13:59:42 +01:00
|
|
|
{"WithoutNoop()", WithoutNoop(), false},
|
|
|
|
|
2022-10-17 18:13:00 +02:00
|
|
|
{
|
|
|
|
"WithDSNRcptNotifyType() NEVER combination",
|
|
|
|
WithDSNRcptNotifyType(DSNRcptNotifySuccess, DSNRcptNotifyNever), true,
|
|
|
|
},
|
2022-03-15 21:10:29 +01:00
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
2022-03-18 15:05:33 +01:00
|
|
|
c, err := NewClient(DefaultHost, tt.option, nil)
|
2022-03-15 21:10:29 +01:00
|
|
|
if err != nil && !tt.shouldfail {
|
|
|
|
t.Errorf("failed to create new client: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
_ = c
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-07 16:24:49 +01:00
|
|
|
// TestWithHELO tests the WithHELO() option for the NewClient() method
|
2022-03-06 15:15:42 +01:00
|
|
|
func TestWithHELO(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
value string
|
|
|
|
want string
|
|
|
|
}{
|
|
|
|
{"HELO test.de", "test.de", "test.de"},
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
c, err := NewClient(DefaultHost, WithHELO(tt.value))
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("failed to create new client: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if c.helo != tt.want {
|
|
|
|
t.Errorf("failed to set custom HELO. Want: %s, got: %s", tt.want, c.helo)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2022-03-07 16:24:49 +01:00
|
|
|
|
|
|
|
// TestWithPort tests the WithPort() option for the NewClient() method
|
|
|
|
func TestWithPort(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
value int
|
|
|
|
want int
|
2022-03-15 22:37:55 +01:00
|
|
|
sf bool
|
2022-03-07 16:24:49 +01:00
|
|
|
}{
|
2022-03-15 22:37:55 +01:00
|
|
|
{"set port to 25", 25, 25, false},
|
|
|
|
{"set port to 465", 465, 465, false},
|
|
|
|
{"set port to 100000", 100000, 25, true},
|
|
|
|
{"set port to -10", -10, 25, true},
|
2022-03-07 16:24:49 +01:00
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
c, err := NewClient(DefaultHost, WithPort(tt.value))
|
2022-03-15 22:37:55 +01:00
|
|
|
if err != nil && !tt.sf {
|
2022-03-07 16:24:49 +01:00
|
|
|
t.Errorf("failed to create new client: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if c.port != tt.want {
|
|
|
|
t.Errorf("failed to set custom port. Want: %d, got: %d", tt.want, c.port)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestWithTimeout tests the WithTimeout() option for the NewClient() method
|
|
|
|
func TestWithTimeout(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
value time.Duration
|
|
|
|
want time.Duration
|
2022-03-15 22:37:55 +01:00
|
|
|
sf bool
|
2022-03-07 16:24:49 +01:00
|
|
|
}{
|
2022-03-15 22:37:55 +01:00
|
|
|
{"set timeout to 5s", time.Second * 5, time.Second * 5, false},
|
|
|
|
{"set timeout to 30s", time.Second * 30, time.Second * 30, false},
|
|
|
|
{"set timeout to 1m", time.Minute, time.Minute, false},
|
|
|
|
{"set timeout to 0", 0, DefaultTimeout, true},
|
2022-03-07 16:24:49 +01:00
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
c, err := NewClient(DefaultHost, WithTimeout(tt.value))
|
2022-03-15 22:37:55 +01:00
|
|
|
if err != nil && !tt.sf {
|
2022-03-07 16:24:49 +01:00
|
|
|
t.Errorf("failed to create new client: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if c.cto != tt.want {
|
|
|
|
t.Errorf("failed to set custom timeout. Want: %d, got: %d", tt.want, c.cto)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-09 16:52:23 +01:00
|
|
|
// TestWithTLSPolicy tests the WithTLSPolicy() option for the NewClient() method
|
|
|
|
func TestWithTLSPolicy(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
value TLSPolicy
|
2022-03-15 22:37:55 +01:00
|
|
|
want string
|
|
|
|
sf bool
|
2022-03-09 16:52:23 +01:00
|
|
|
}{
|
2022-03-15 22:37:55 +01:00
|
|
|
{"Policy: TLSMandatory", TLSMandatory, TLSMandatory.String(), false},
|
|
|
|
{"Policy: TLSOpportunistic", TLSOpportunistic, TLSOpportunistic.String(), false},
|
|
|
|
{"Policy: NoTLS", NoTLS, NoTLS.String(), false},
|
|
|
|
{"Policy: Invalid", -1, "UnknownPolicy", true},
|
2022-03-09 16:52:23 +01:00
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
c, err := NewClient(DefaultHost, WithTLSPolicy(tt.value))
|
2022-03-15 22:37:55 +01:00
|
|
|
if err != nil && !tt.sf {
|
2022-03-09 16:52:23 +01:00
|
|
|
t.Errorf("failed to create new client: %s", err)
|
|
|
|
return
|
|
|
|
}
|
2022-03-15 22:37:55 +01:00
|
|
|
if c.tlspolicy.String() != tt.want {
|
2022-03-09 16:52:23 +01:00
|
|
|
t.Errorf("failed to set TLSPolicy. Want: %s, got: %s", tt.want, c.tlspolicy)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestSetTLSPolicy tests the SetTLSPolicy() method for the Client object
|
|
|
|
func TestSetTLSPolicy(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
value TLSPolicy
|
2022-03-15 22:37:55 +01:00
|
|
|
want string
|
|
|
|
sf bool
|
2022-03-09 16:52:23 +01:00
|
|
|
}{
|
2022-03-15 22:37:55 +01:00
|
|
|
{"Policy: TLSMandatory", TLSMandatory, TLSMandatory.String(), false},
|
|
|
|
{"Policy: TLSOpportunistic", TLSOpportunistic, TLSOpportunistic.String(), false},
|
|
|
|
{"Policy: NoTLS", NoTLS, NoTLS.String(), false},
|
|
|
|
{"Policy: Invalid", -1, "UnknownPolicy", true},
|
2022-03-09 16:52:23 +01:00
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
c, err := NewClient(DefaultHost, WithTLSPolicy(NoTLS))
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("failed to create new client: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
c.SetTLSPolicy(tt.value)
|
2022-03-15 22:37:55 +01:00
|
|
|
if c.tlspolicy.String() != tt.want {
|
2022-03-09 16:52:23 +01:00
|
|
|
t.Errorf("failed to set TLSPolicy. Want: %s, got: %s", tt.want, c.tlspolicy)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2022-03-16 14:09:50 +01:00
|
|
|
|
|
|
|
// TestSetTLSConfig tests the SetTLSConfig() method for the Client object
|
|
|
|
func TestSetTLSConfig(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
value *tls.Config
|
|
|
|
sf bool
|
|
|
|
}{
|
|
|
|
{"default config", &tls.Config{}, false},
|
|
|
|
{"nil config", nil, true},
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
c, err := NewClient(DefaultHost)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("failed to create new client: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if err := c.SetTLSConfig(tt.value); err != nil && !tt.sf {
|
|
|
|
t.Errorf("failed to set TLSConfig: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-16 21:02:31 +01:00
|
|
|
// TestSetSSL tests the SetSSL() method for the Client object
|
|
|
|
func TestSetSSL(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
value bool
|
|
|
|
}{
|
|
|
|
{"SSL: on", true},
|
|
|
|
{"SSL: off", false},
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
c, err := NewClient(DefaultHost)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("failed to create new client: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
c.SetSSL(tt.value)
|
|
|
|
if c.ssl != tt.value {
|
|
|
|
t.Errorf("failed to set SSL setting. Got: %t, want: %t", c.ssl, tt.value)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-16 14:09:50 +01:00
|
|
|
// TestSetUsername tests the SetUsername method for the Client object
|
|
|
|
func TestSetUsername(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
value string
|
|
|
|
want string
|
|
|
|
sf bool
|
|
|
|
}{
|
|
|
|
{"normal username", "testuser", "testuser", false},
|
|
|
|
{"empty username", "", "", false},
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
c, err := NewClient(DefaultHost)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("failed to create new client: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
c.SetUsername(tt.value)
|
|
|
|
if c.user != tt.want {
|
|
|
|
t.Errorf("failed to set username. Expected %s, got: %s", tt.want, c.user)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestSetPassword tests the SetPassword method for the Client object
|
|
|
|
func TestSetPassword(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
value string
|
|
|
|
want string
|
|
|
|
sf bool
|
|
|
|
}{
|
|
|
|
{"normal password", "testpass", "testpass", false},
|
|
|
|
{"empty password", "", "", false},
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
c, err := NewClient(DefaultHost)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("failed to create new client: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
c.SetPassword(tt.value)
|
|
|
|
if c.pass != tt.want {
|
|
|
|
t.Errorf("failed to set password. Expected %s, got: %s", tt.want, c.pass)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestSetSMTPAuth tests the SetSMTPAuth method for the Client object
|
|
|
|
func TestSetSMTPAuth(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
value SMTPAuthType
|
|
|
|
want string
|
|
|
|
sf bool
|
|
|
|
}{
|
|
|
|
{"SMTPAuth: LOGIN", SMTPAuthLogin, "LOGIN", false},
|
|
|
|
{"SMTPAuth: PLAIN", SMTPAuthPlain, "PLAIN", false},
|
|
|
|
{"SMTPAuth: CRAM-MD5", SMTPAuthCramMD5, "CRAM-MD5", false},
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
c, err := NewClient(DefaultHost)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("failed to create new client: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
c.SetSMTPAuth(tt.value)
|
|
|
|
if string(c.satype) != tt.want {
|
|
|
|
t.Errorf("failed to set SMTP auth type. Expected %s, got: %s", tt.want, string(c.satype))
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-11 21:07:15 +02:00
|
|
|
// TestWithDSN tests the WithDSN method for the Client object
|
|
|
|
func TestWithDSN(t *testing.T) {
|
|
|
|
c, err := NewClient(DefaultHost, WithDSN())
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("failed to create new client: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if !c.dsn {
|
|
|
|
t.Errorf("WithDSN failed. c.dsn expected to be: %t, got: %t", true, c.dsn)
|
|
|
|
}
|
|
|
|
if c.dsnmrtype != DSNMailReturnFull {
|
|
|
|
t.Errorf("WithDSN failed. c.dsnmrtype expected to be: %s, got: %s", DSNMailReturnFull,
|
|
|
|
c.dsnmrtype)
|
|
|
|
}
|
|
|
|
if c.dsnrntype[0] != string(DSNRcptNotifyFailure) {
|
|
|
|
t.Errorf("WithDSN failed. c.dsnrntype[0] expected to be: %s, got: %s", DSNRcptNotifyFailure,
|
|
|
|
c.dsnrntype[0])
|
|
|
|
}
|
|
|
|
if c.dsnrntype[1] != string(DSNRcptNotifySuccess) {
|
|
|
|
t.Errorf("WithDSN failed. c.dsnrntype[1] expected to be: %s, got: %s", DSNRcptNotifySuccess,
|
|
|
|
c.dsnrntype[1])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestWithDSNMailReturnType tests the WithDSNMailReturnType method for the Client object
|
|
|
|
func TestWithDSNMailReturnType(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
value DSNMailReturnOption
|
|
|
|
want string
|
|
|
|
sf bool
|
|
|
|
}{
|
|
|
|
{"WithDSNMailReturnType: FULL", DSNMailReturnFull, "FULL", false},
|
|
|
|
{"WithDSNMailReturnType: HDRS", DSNMailReturnHeadersOnly, "HDRS", false},
|
|
|
|
{"WithDSNMailReturnType: INVALID", "INVALID", "", true},
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
c, err := NewClient(DefaultHost, WithDSNMailReturnType(tt.value))
|
|
|
|
if err != nil && !tt.sf {
|
|
|
|
t.Errorf("failed to create new client: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if string(c.dsnmrtype) != tt.want {
|
|
|
|
t.Errorf("WithDSNMailReturnType failed. Expected %s, got: %s", tt.want, string(c.dsnmrtype))
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestWithDSNRcptNotifyType tests the WithDSNRcptNotifyType method for the Client object
|
|
|
|
func TestWithDSNRcptNotifyType(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
value DSNRcptNotifyOption
|
|
|
|
want string
|
|
|
|
sf bool
|
|
|
|
}{
|
|
|
|
{"WithDSNRcptNotifyType: NEVER", DSNRcptNotifyNever, "NEVER", false},
|
|
|
|
{"WithDSNRcptNotifyType: SUCCESS", DSNRcptNotifySuccess, "SUCCESS", false},
|
|
|
|
{"WithDSNRcptNotifyType: FAILURE", DSNRcptNotifyFailure, "FAILURE", false},
|
|
|
|
{"WithDSNRcptNotifyType: DELAY", DSNRcptNotifyDelay, "DELAY", false},
|
|
|
|
{"WithDSNRcptNotifyType: INVALID", "INVALID", "", true},
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
c, err := NewClient(DefaultHost, WithDSNRcptNotifyType(tt.value))
|
|
|
|
if err != nil && !tt.sf {
|
|
|
|
t.Errorf("failed to create new client: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if len(c.dsnrntype) <= 0 && !tt.sf {
|
|
|
|
t.Errorf("WithDSNRcptNotifyType failed. Expected at least one DSNRNType but got none")
|
|
|
|
}
|
|
|
|
if !tt.sf && c.dsnrntype[0] != tt.want {
|
|
|
|
t.Errorf("WithDSNRcptNotifyType failed. Expected %s, got: %s", tt.want, c.dsnrntype[0])
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-26 13:59:42 +01:00
|
|
|
// TestWithoutNoop tests the WithoutNoop method for the Client object
|
|
|
|
func TestWithoutNoop(t *testing.T) {
|
|
|
|
c, err := NewClient(DefaultHost, WithoutNoop())
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("failed to create new client: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if !c.noNoop {
|
|
|
|
t.Errorf("WithoutNoop failed. c.noNoop expected to be: %t, got: %t", true, c.noNoop)
|
|
|
|
}
|
|
|
|
|
|
|
|
c, err = NewClient(DefaultHost)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("failed to create new client: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if c.noNoop {
|
|
|
|
t.Errorf("WithoutNoop failed. c.noNoop expected to be: %t, got: %t", false, c.noNoop)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-16 14:09:50 +01:00
|
|
|
// TestSetSMTPAuthCustom tests the SetSMTPAuthCustom method for the Client object
|
|
|
|
func TestSetSMTPAuthCustom(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
value smtp.Auth
|
|
|
|
want string
|
|
|
|
sf bool
|
|
|
|
}{
|
|
|
|
{"SMTPAuth: PLAIN", smtp.PlainAuth("", "", "", ""), "PLAIN", false},
|
|
|
|
{"SMTPAuth: CRAM-MD5", smtp.CRAMMD5Auth("", ""), "CRAM-MD5", false},
|
|
|
|
{"SMTPAuth: LOGIN", auth.LoginAuth("", "", ""), "LOGIN", false},
|
|
|
|
}
|
|
|
|
si := smtp.ServerInfo{TLS: true}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
c, err := NewClient(DefaultHost)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("failed to create new client: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
c.SetSMTPAuthCustom(tt.value)
|
|
|
|
if c.sa == nil {
|
|
|
|
t.Errorf("failed to set custom SMTP auth method. SMTP Auth method is empty")
|
|
|
|
}
|
|
|
|
p, _, err := c.sa.Start(&si)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("SMTP Auth Start() method returned error: %s", err)
|
|
|
|
}
|
|
|
|
if p != tt.want {
|
|
|
|
t.Errorf("SMTP Auth Start() method is returned proto: %s, expected: %s", p, tt.want)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2022-03-16 21:02:31 +01:00
|
|
|
|
2022-03-16 21:32:51 +01:00
|
|
|
// TestClient_DialWithContext tests the DialWithContext method for the Client object
|
|
|
|
func TestClient_DialWithContext(t *testing.T) {
|
|
|
|
c, err := getTestConnection(true)
|
|
|
|
if err != nil {
|
|
|
|
t.Skipf("failed to create test client: %s. Skipping tests", err)
|
|
|
|
}
|
|
|
|
ctx := context.Background()
|
|
|
|
if err := c.DialWithContext(ctx); err != nil {
|
|
|
|
t.Errorf("failed to dial with context: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if c.co == nil {
|
|
|
|
t.Errorf("DialWithContext didn't fail but no connection found.")
|
|
|
|
}
|
|
|
|
if c.sc == nil {
|
|
|
|
t.Errorf("DialWithContext didn't fail but no SMTP client found.")
|
|
|
|
}
|
|
|
|
if err := c.Close(); err != nil {
|
|
|
|
t.Errorf("failed to close connection: %s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-18 10:15:02 +01:00
|
|
|
// TestClient_DialWithContextInvalidHost tests the DialWithContext method with intentional breaking
|
|
|
|
// for the Client object
|
|
|
|
func TestClient_DialWithContextInvalidHost(t *testing.T) {
|
|
|
|
c, err := getTestConnection(true)
|
|
|
|
if err != nil {
|
|
|
|
t.Skipf("failed to create test client: %s. Skipping tests", err)
|
|
|
|
}
|
|
|
|
c.co = nil
|
|
|
|
c.host = "invalid.addr"
|
|
|
|
ctx := context.Background()
|
|
|
|
if err := c.DialWithContext(ctx); err == nil {
|
|
|
|
t.Errorf("dial succeeded but was supposed to fail")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestClient_DialWithContextInvalidHELO tests the DialWithContext method with intentional breaking
|
|
|
|
// for the Client object
|
|
|
|
func TestClient_DialWithContextInvalidHELO(t *testing.T) {
|
|
|
|
c, err := getTestConnection(true)
|
|
|
|
if err != nil {
|
|
|
|
t.Skipf("failed to create test client: %s. Skipping tests", err)
|
|
|
|
}
|
|
|
|
c.co = nil
|
|
|
|
c.helo = ""
|
|
|
|
ctx := context.Background()
|
|
|
|
if err := c.DialWithContext(ctx); err == nil {
|
|
|
|
t.Errorf("dial succeeded but was supposed to fail")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestClient_DialWithContextInvalidAuth tests the DialWithContext method with intentional breaking
|
|
|
|
// for the Client object
|
|
|
|
func TestClient_DialWithContextInvalidAuth(t *testing.T) {
|
|
|
|
c, err := getTestConnection(true)
|
|
|
|
if err != nil {
|
|
|
|
t.Skipf("failed to create test client: %s. Skipping tests", err)
|
|
|
|
}
|
|
|
|
c.user = "invalid"
|
|
|
|
c.pass = "invalid"
|
|
|
|
c.SetSMTPAuthCustom(auth.LoginAuth("invalid", "invalid", "invalid"))
|
|
|
|
ctx := context.Background()
|
|
|
|
if err := c.DialWithContext(ctx); err == nil {
|
|
|
|
t.Errorf("dial succeeded but was supposed to fail")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestClient_checkConn tests the checkConn method with intentional breaking for the Client object
|
|
|
|
func TestClient_checkConn(t *testing.T) {
|
|
|
|
c, err := getTestConnection(true)
|
|
|
|
if err != nil {
|
|
|
|
t.Skipf("failed to create test client: %s. Skipping tests", err)
|
|
|
|
}
|
|
|
|
c.co = nil
|
|
|
|
if err := c.checkConn(); err == nil {
|
|
|
|
t.Errorf("connCheck() should fail but succeeded")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-16 21:32:51 +01:00
|
|
|
// TestClient_DiealWithContextOptions tests the DialWithContext method plus different options
|
|
|
|
// for the Client object
|
|
|
|
func TestClient_DialWithContextOptions(t *testing.T) {
|
2022-03-16 21:02:31 +01:00
|
|
|
tests := []struct {
|
2022-03-16 21:32:51 +01:00
|
|
|
name string
|
|
|
|
wantssl bool
|
|
|
|
wanttls TLSPolicy
|
|
|
|
sf bool
|
2022-03-16 21:02:31 +01:00
|
|
|
}{
|
2022-03-16 21:32:51 +01:00
|
|
|
{"Want SSL (should fail)", true, NoTLS, true},
|
|
|
|
{"Want Mandatory TLS", false, TLSMandatory, false},
|
2022-03-16 21:02:31 +01:00
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
c, err := getTestConnection(true)
|
|
|
|
if err != nil {
|
|
|
|
t.Skipf("failed to create test client: %s. Skipping tests", err)
|
|
|
|
}
|
2022-03-16 21:32:51 +01:00
|
|
|
if tt.wantssl {
|
|
|
|
c.SetSSL(true)
|
|
|
|
}
|
|
|
|
if tt.wanttls != NoTLS {
|
|
|
|
c.SetTLSPolicy(tt.wanttls)
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx := context.Background()
|
|
|
|
if err := c.DialWithContext(ctx); err != nil && !tt.sf {
|
|
|
|
t.Errorf("failed to dial with context: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if !tt.sf {
|
|
|
|
if c.co == nil && !tt.sf {
|
|
|
|
t.Errorf("DialWithContext didn't fail but no connection found.")
|
|
|
|
}
|
|
|
|
if c.sc == nil && !tt.sf {
|
|
|
|
t.Errorf("DialWithContext didn't fail but no SMTP client found.")
|
|
|
|
}
|
|
|
|
if err := c.Reset(); err != nil {
|
|
|
|
t.Errorf("failed to reset connection: %s", err)
|
|
|
|
}
|
|
|
|
if err := c.Close(); err != nil {
|
|
|
|
t.Errorf("failed to close connection: %s", err)
|
|
|
|
}
|
2022-03-16 21:02:31 +01:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2022-03-21 10:17:48 +01:00
|
|
|
}
|
|
|
|
|
2022-03-21 10:38:22 +01:00
|
|
|
// TestClient_DialSendClose tests the Dial(), Send() and Close() method of Client
|
2022-03-21 10:17:48 +01:00
|
|
|
func TestClient_DialSendClose(t *testing.T) {
|
|
|
|
if os.Getenv("TEST_ALLOW_SEND") == "" {
|
|
|
|
t.Skipf("TEST_ALLOW_SEND is not set. Skipping mail sending test")
|
|
|
|
}
|
|
|
|
m := NewMsg()
|
|
|
|
_ = m.FromFormat("go-mail Test Mailer", os.Getenv("TEST_FROM"))
|
|
|
|
_ = m.To(TestRcpt)
|
|
|
|
m.Subject(fmt.Sprintf("This is a test mail from go-mail/v%s", VERSION))
|
|
|
|
m.SetBulk()
|
|
|
|
m.SetDate()
|
|
|
|
m.SetMessageID()
|
|
|
|
m.SetBodyString(TypeTextPlain, "This is a test mail from the go-mail library")
|
|
|
|
|
|
|
|
c, err := getTestConnection(true)
|
|
|
|
if err != nil {
|
2022-03-21 12:18:08 +01:00
|
|
|
t.Skipf("failed to create test client: %s. Skipping tests", err)
|
2022-03-21 10:17:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
ctx, cfn := context.WithTimeout(context.Background(), time.Second*10)
|
|
|
|
defer cfn()
|
|
|
|
if err := c.DialWithContext(ctx); err != nil {
|
2022-03-21 10:38:22 +01:00
|
|
|
t.Errorf("Dial() failed: %s", err)
|
2022-03-21 10:17:48 +01:00
|
|
|
}
|
|
|
|
if err := c.Send(m); err != nil {
|
|
|
|
t.Errorf("Send() failed: %s", err)
|
|
|
|
}
|
|
|
|
if err := c.Close(); err != nil {
|
|
|
|
t.Errorf("Close() failed: %s", err)
|
|
|
|
}
|
2022-03-16 21:02:31 +01:00
|
|
|
}
|
|
|
|
|
2022-09-26 10:40:57 +02:00
|
|
|
// TestClient_DialAndSendWithContext tests the DialAndSendWithContext() method of Client
|
|
|
|
func TestClient_DialAndSendWithContext(t *testing.T) {
|
|
|
|
if os.Getenv("TEST_ALLOW_SEND") == "" {
|
|
|
|
t.Skipf("TEST_ALLOW_SEND is not set. Skipping mail sending test")
|
|
|
|
}
|
|
|
|
m := NewMsg()
|
|
|
|
_ = m.FromFormat("go-mail Test Mailer", os.Getenv("TEST_FROM"))
|
|
|
|
_ = m.To(TestRcpt)
|
|
|
|
m.Subject(fmt.Sprintf("This is a test mail from go-mail/v%s", VERSION))
|
|
|
|
m.SetBulk()
|
|
|
|
m.SetDate()
|
|
|
|
m.SetMessageID()
|
|
|
|
m.SetBodyString(TypeTextPlain, "This is a test mail from the go-mail library")
|
|
|
|
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
to time.Duration
|
|
|
|
sf bool
|
|
|
|
}{
|
|
|
|
{"Timeout: 100s", time.Second * 100, false},
|
|
|
|
{"Timeout: 100ms", time.Millisecond * 100, true},
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
c, err := getTestConnection(true)
|
|
|
|
if err != nil {
|
|
|
|
t.Skipf("failed to create test client: %s. Skipping tests", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx, cfn := context.WithTimeout(context.Background(), tt.to)
|
|
|
|
defer cfn()
|
|
|
|
if err := c.DialAndSendWithContext(ctx, m); err != nil && !tt.sf {
|
|
|
|
t.Errorf("DialAndSendWithContext() failed: %s", err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-21 10:38:22 +01:00
|
|
|
// TestClient_DialAndSend tests the DialAndSend() method of Client
|
|
|
|
func TestClient_DialAndSend(t *testing.T) {
|
|
|
|
if os.Getenv("TEST_ALLOW_SEND") == "" {
|
|
|
|
t.Skipf("TEST_ALLOW_SEND is not set. Skipping mail sending test")
|
|
|
|
}
|
|
|
|
m := NewMsg()
|
|
|
|
_ = m.FromFormat("go-mail Test Mailer", os.Getenv("TEST_FROM"))
|
|
|
|
_ = m.To(TestRcpt)
|
|
|
|
m.Subject(fmt.Sprintf("This is a test mail from go-mail/v%s", VERSION))
|
|
|
|
m.SetBulk()
|
|
|
|
m.SetDate()
|
|
|
|
m.SetMessageID()
|
|
|
|
m.SetBodyString(TypeTextPlain, "This is a test mail from the go-mail library")
|
|
|
|
|
|
|
|
c, err := getTestConnection(true)
|
|
|
|
if err != nil {
|
2022-03-21 12:18:08 +01:00
|
|
|
t.Skipf("failed to create test client: %s. Skipping tests", err)
|
2022-03-21 10:38:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if err := c.DialAndSend(m); err != nil {
|
|
|
|
t.Errorf("DialAndSend() failed: %s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-12 09:31:42 +02:00
|
|
|
// TestClient_DialAndSendWithDSN tests the DialAndSend() method of Client with DSN enabled
|
|
|
|
func TestClient_DialAndSendWithDSN(t *testing.T) {
|
|
|
|
if os.Getenv("TEST_ALLOW_SEND") == "" {
|
|
|
|
t.Skipf("TEST_ALLOW_SEND is not set. Skipping mail sending test")
|
|
|
|
}
|
|
|
|
m := NewMsg()
|
|
|
|
_ = m.FromFormat("go-mail Test Mailer", os.Getenv("TEST_FROM"))
|
|
|
|
_ = m.To(TestRcpt)
|
|
|
|
m.Subject(fmt.Sprintf("This is a test mail from go-mail/v%s", VERSION))
|
|
|
|
m.SetBulk()
|
|
|
|
m.SetDate()
|
|
|
|
m.SetMessageID()
|
|
|
|
m.SetBodyString(TypeTextPlain, "This is a test mail from the go-mail library")
|
|
|
|
|
|
|
|
c, err := getTestConnectionWithDSN(true)
|
|
|
|
if err != nil {
|
|
|
|
t.Skipf("failed to create test client: %s. Skipping tests", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := c.DialAndSend(m); err != nil {
|
|
|
|
t.Errorf("DialAndSend() failed: %s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-21 12:18:08 +01:00
|
|
|
// TestClient_DialSendCloseBroken tests the Dial(), Send() and Close() method of Client with broken settings
|
2022-03-21 10:38:22 +01:00
|
|
|
func TestClient_DialSendCloseBroken(t *testing.T) {
|
|
|
|
if os.Getenv("TEST_ALLOW_SEND") == "" {
|
|
|
|
t.Skipf("TEST_ALLOW_SEND is not set. Skipping mail sending test")
|
|
|
|
}
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
from string
|
|
|
|
to string
|
|
|
|
closestart bool
|
|
|
|
closeearly bool
|
|
|
|
sf bool
|
|
|
|
}{
|
|
|
|
{"Invalid FROM", "foo@foo", TestRcpt, false, false, true},
|
|
|
|
{"Invalid TO", os.Getenv("TEST_FROM"), "foo@foo", false, false, true},
|
|
|
|
{"No FROM", "", TestRcpt, false, false, true},
|
|
|
|
{"No TO", os.Getenv("TEST_FROM"), "", false, false, true},
|
|
|
|
{"Close early", os.Getenv("TEST_FROM"), TestRcpt, false, true, true},
|
|
|
|
{"Close start", os.Getenv("TEST_FROM"), TestRcpt, true, false, true},
|
|
|
|
{"Close start/early", os.Getenv("TEST_FROM"), TestRcpt, true, true, true},
|
|
|
|
}
|
|
|
|
|
|
|
|
m := NewMsg(WithEncoding(NoEncoding))
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
m.SetAddrHeaderIgnoreInvalid(HeaderFrom, tt.from)
|
|
|
|
m.SetAddrHeaderIgnoreInvalid(HeaderTo, tt.to)
|
|
|
|
|
|
|
|
c, err := getTestConnection(true)
|
|
|
|
if err != nil {
|
2022-03-21 12:18:08 +01:00
|
|
|
t.Skipf("failed to create test client: %s. Skipping tests", err)
|
2022-03-21 10:38:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
ctx, cfn := context.WithTimeout(context.Background(), time.Second*10)
|
|
|
|
defer cfn()
|
|
|
|
if err := c.DialWithContext(ctx); err != nil && !tt.sf {
|
|
|
|
t.Errorf("Dail() failed: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if tt.closestart {
|
|
|
|
_ = c.sc.Close()
|
|
|
|
_ = c.co.Close()
|
|
|
|
}
|
|
|
|
if err := c.Send(m); err != nil && !tt.sf {
|
|
|
|
t.Errorf("Send() failed: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if tt.closeearly {
|
|
|
|
_ = c.sc.Close()
|
|
|
|
_ = c.co.Close()
|
|
|
|
}
|
|
|
|
if err := c.Close(); err != nil && !tt.sf {
|
|
|
|
t.Errorf("Close() failed: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-12 09:31:42 +02:00
|
|
|
// TestClient_DialSendCloseBrokenWithDSN tests the Dial(), Send() and Close() method of Client with
|
|
|
|
// broken settings and DSN enabled
|
|
|
|
func TestClient_DialSendCloseBrokenWithDSN(t *testing.T) {
|
|
|
|
if os.Getenv("TEST_ALLOW_SEND") == "" {
|
|
|
|
t.Skipf("TEST_ALLOW_SEND is not set. Skipping mail sending test")
|
|
|
|
}
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
from string
|
|
|
|
to string
|
|
|
|
closestart bool
|
|
|
|
closeearly bool
|
|
|
|
sf bool
|
|
|
|
}{
|
|
|
|
{"Invalid FROM", "foo@foo", TestRcpt, false, false, true},
|
|
|
|
{"Invalid TO", os.Getenv("TEST_FROM"), "foo@foo", false, false, true},
|
|
|
|
{"No FROM", "", TestRcpt, false, false, true},
|
|
|
|
{"No TO", os.Getenv("TEST_FROM"), "", false, false, true},
|
|
|
|
{"Close early", os.Getenv("TEST_FROM"), TestRcpt, false, true, true},
|
|
|
|
{"Close start", os.Getenv("TEST_FROM"), TestRcpt, true, false, true},
|
|
|
|
{"Close start/early", os.Getenv("TEST_FROM"), TestRcpt, true, true, true},
|
|
|
|
}
|
|
|
|
|
|
|
|
m := NewMsg(WithEncoding(NoEncoding))
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
m.SetAddrHeaderIgnoreInvalid(HeaderFrom, tt.from)
|
|
|
|
m.SetAddrHeaderIgnoreInvalid(HeaderTo, tt.to)
|
|
|
|
|
|
|
|
c, err := getTestConnectionWithDSN(true)
|
|
|
|
if err != nil {
|
|
|
|
t.Skipf("failed to create test client: %s. Skipping tests", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx, cfn := context.WithTimeout(context.Background(), time.Second*10)
|
|
|
|
defer cfn()
|
|
|
|
if err := c.DialWithContext(ctx); err != nil && !tt.sf {
|
|
|
|
t.Errorf("Dail() failed: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if tt.closestart {
|
|
|
|
_ = c.sc.Close()
|
|
|
|
_ = c.co.Close()
|
|
|
|
}
|
|
|
|
if err := c.Send(m); err != nil && !tt.sf {
|
|
|
|
t.Errorf("Send() failed: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if tt.closeearly {
|
|
|
|
_ = c.sc.Close()
|
|
|
|
_ = c.co.Close()
|
|
|
|
}
|
|
|
|
if err := c.Close(); err != nil && !tt.sf {
|
|
|
|
t.Errorf("Close() failed: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-10 13:41:00 +01:00
|
|
|
// TestClient_Send_withBrokenRecipient tests the Send() method of Client with a broken and a working recipient
|
|
|
|
func TestClient_Send_withBrokenRecipient(t *testing.T) {
|
|
|
|
if os.Getenv("TEST_ALLOW_SEND") == "" {
|
|
|
|
t.Skipf("TEST_ALLOW_SEND is not set. Skipping mail sending test")
|
|
|
|
}
|
|
|
|
var msgs []*Msg
|
|
|
|
rcpts := []string{"invalid@domain.tld", TestRcpt, "invalid@address.invalid"}
|
|
|
|
for _, rcpt := range rcpts {
|
|
|
|
m := NewMsg()
|
|
|
|
_ = m.FromFormat("go-mail Test Mailer", os.Getenv("TEST_FROM"))
|
|
|
|
_ = m.To(rcpt)
|
|
|
|
m.Subject(fmt.Sprintf("This is a test mail from go-mail/v%s", VERSION))
|
|
|
|
m.SetBulk()
|
|
|
|
m.SetDate()
|
|
|
|
m.SetMessageID()
|
|
|
|
m.SetBodyString(TypeTextPlain, "This is a test mail from the go-mail library")
|
|
|
|
msgs = append(msgs, m)
|
|
|
|
}
|
|
|
|
|
|
|
|
c, err := getTestConnection(true)
|
|
|
|
if err != nil {
|
|
|
|
t.Skipf("failed to create test client: %s. Skipping tests", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx, cfn := context.WithTimeout(context.Background(), DefaultTimeout)
|
|
|
|
defer cfn()
|
|
|
|
if err := c.DialWithContext(ctx); err != nil {
|
|
|
|
t.Errorf("failed to dial to sending server: %s", err)
|
|
|
|
}
|
|
|
|
if err := c.Send(msgs...); err != nil {
|
|
|
|
if !strings.Contains(err.Error(), "invalid@domain.tld") ||
|
|
|
|
!strings.Contains(err.Error(), "invalid@address.invalid") {
|
|
|
|
t.Errorf("sending mails to invalid addresses was supposed to fail but didn't")
|
|
|
|
}
|
|
|
|
if strings.Contains(err.Error(), TestRcpt) {
|
|
|
|
t.Errorf("sending mail to valid addresses failed: %s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if err := c.Close(); err != nil {
|
|
|
|
t.Errorf("failed to close client connection: %s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-21 12:18:08 +01:00
|
|
|
// TestClient_auth tests the Dial(), Send() and Close() method of Client with broken settings
|
|
|
|
func TestClient_auth(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
auth SMTPAuthType
|
|
|
|
sf bool
|
|
|
|
}{
|
|
|
|
{"SMTP AUTH: PLAIN", SMTPAuthPlain, false},
|
|
|
|
{"SMTP AUTH: LOGIN", SMTPAuthLogin, false},
|
|
|
|
{"SMTP AUTH: CRAM-MD5", SMTPAuthCramMD5, true},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
c, err := getTestConnection(false)
|
|
|
|
if err != nil {
|
|
|
|
t.Skipf("failed to create test client: %s. Skipping tests", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx, cfn := context.WithTimeout(context.Background(), time.Second*5)
|
|
|
|
defer cfn()
|
|
|
|
if err := c.DialWithContext(ctx); err != nil {
|
|
|
|
t.Errorf("auth() failed: could not Dial() => %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
c.SetSMTPAuth(tt.auth)
|
|
|
|
c.SetUsername(os.Getenv("TEST_SMTPAUTH_USER"))
|
|
|
|
c.SetPassword(os.Getenv("TEST_SMTPAUTH_PASS"))
|
|
|
|
if err := c.auth(); err != nil && !tt.sf {
|
|
|
|
t.Errorf("auth() failed: %s", err)
|
|
|
|
}
|
|
|
|
if err := c.Close(); err != nil {
|
|
|
|
t.Errorf("auth() failed: could not Close() => %s", err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2022-09-11 22:05:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// TestValidateLine tests the validateLine() method
|
|
|
|
func TestValidateLine(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
value string
|
|
|
|
sf bool
|
|
|
|
}{
|
|
|
|
{"valid line", "valid line", false},
|
|
|
|
{`invalid line: \n`, "invalid line\n", true},
|
|
|
|
{`invalid line: \r`, "invalid line\r", true},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
if err := validateLine(tt.value); err != nil && !tt.sf {
|
|
|
|
t.Errorf("validateLine failed: %s", err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2022-03-21 12:18:08 +01:00
|
|
|
}
|
|
|
|
|
2023-01-02 22:22:31 +01:00
|
|
|
// TestClient_Send_MsgSendError tests the Client.Send method with a broken recipient and verifies
|
2022-12-31 12:40:42 +01:00
|
|
|
// that the SendError type works properly
|
|
|
|
func TestClient_Send_MsgSendError(t *testing.T) {
|
|
|
|
if os.Getenv("TEST_ALLOW_SEND") == "" {
|
|
|
|
t.Skipf("TEST_ALLOW_SEND is not set. Skipping mail sending test")
|
|
|
|
}
|
|
|
|
var msgs []*Msg
|
|
|
|
rcpts := []string{"invalid@domain.tld", "invalid@address.invalid"}
|
|
|
|
for _, rcpt := range rcpts {
|
|
|
|
m := NewMsg()
|
|
|
|
_ = m.FromFormat("go-mail Test Mailer", os.Getenv("TEST_FROM"))
|
|
|
|
_ = m.To(rcpt)
|
|
|
|
m.Subject(fmt.Sprintf("This is a test mail from go-mail/v%s", VERSION))
|
|
|
|
m.SetBulk()
|
|
|
|
m.SetDate()
|
|
|
|
m.SetMessageID()
|
|
|
|
m.SetBodyString(TypeTextPlain, "This is a test mail from the go-mail library")
|
|
|
|
msgs = append(msgs, m)
|
|
|
|
}
|
|
|
|
|
|
|
|
c, err := getTestConnection(true)
|
|
|
|
if err != nil {
|
|
|
|
t.Skipf("failed to create test client: %s. Skipping tests", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx, cfn := context.WithTimeout(context.Background(), DefaultTimeout)
|
|
|
|
defer cfn()
|
|
|
|
if err := c.DialWithContext(ctx); err != nil {
|
|
|
|
t.Errorf("failed to dial to sending server: %s", err)
|
|
|
|
}
|
|
|
|
if err := c.Send(msgs...); err == nil {
|
|
|
|
t.Errorf("sending messages with broken recipients was supposed to fail but didn't")
|
|
|
|
}
|
|
|
|
if err := c.Close(); err != nil {
|
|
|
|
t.Errorf("failed to close client connection: %s", err)
|
|
|
|
}
|
|
|
|
for _, m := range msgs {
|
|
|
|
if !m.HasSendError() {
|
|
|
|
t.Errorf("message was expected to have a send error, but didn't")
|
|
|
|
}
|
2023-01-01 14:20:13 +01:00
|
|
|
se := &SendError{Reason: ErrSMTPRcptTo}
|
|
|
|
if !errors.Is(m.SendError(), se) {
|
|
|
|
t.Errorf("error mismatch, expected: %s, got: %s", se, m.SendError())
|
|
|
|
}
|
|
|
|
if m.SendErrorIsTemp() {
|
|
|
|
t.Errorf("message was not expected to be a temporary error, but reported as such")
|
2022-12-31 12:40:42 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-02 12:14:14 +01:00
|
|
|
// TestClient_DialAndSendWithContext_withSendError tests the Client.DialAndSendWithContext method
|
|
|
|
// with a broken recipient to make sure that the returned error satisfies the Msg.SendError type
|
|
|
|
func TestClient_DialAndSendWithContext_withSendError(t *testing.T) {
|
|
|
|
if os.Getenv("TEST_ALLOW_SEND") == "" {
|
|
|
|
t.Skipf("TEST_ALLOW_SEND is not set. Skipping mail sending test")
|
|
|
|
}
|
|
|
|
m := NewMsg()
|
|
|
|
_ = m.FromFormat("go-mail Test Mailer", os.Getenv("TEST_FROM"))
|
|
|
|
_ = m.To("invalid@domain.tld")
|
|
|
|
m.Subject(fmt.Sprintf("This is a test mail from go-mail/v%s", VERSION))
|
|
|
|
m.SetBulk()
|
|
|
|
m.SetDate()
|
|
|
|
m.SetMessageID()
|
|
|
|
m.SetBodyString(TypeTextPlain, "This is a test mail from the go-mail library")
|
|
|
|
|
|
|
|
c, err := getTestConnection(true)
|
|
|
|
if err != nil {
|
|
|
|
t.Skipf("failed to create test client: %s. Skipping tests", err)
|
|
|
|
}
|
|
|
|
ctx, cfn := context.WithTimeout(context.Background(), DefaultTimeout)
|
|
|
|
defer cfn()
|
|
|
|
err = c.DialAndSendWithContext(ctx, m)
|
|
|
|
if err == nil {
|
|
|
|
t.Errorf("expected DialAndSendWithContext with broken mail recipient to fail, but didn't")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
var se *SendError
|
|
|
|
if !errors.As(err, &se) {
|
|
|
|
t.Errorf("expected *SendError type as returned error, but didn't")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if se.IsTemp() {
|
|
|
|
t.Errorf("expected permanent error but IsTemp() returned true")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-16 21:02:31 +01:00
|
|
|
// getTestConnection takes environment variables to establish a connection to a real
|
|
|
|
// SMTP server to test all functionality that requires a connection
|
|
|
|
func getTestConnection(auth bool) (*Client, error) {
|
2022-03-18 10:15:02 +01:00
|
|
|
if os.Getenv("TEST_SKIP_ONLINE") != "" {
|
|
|
|
return nil, fmt.Errorf("env variable TEST_SKIP_ONLINE is set. Skipping online tests")
|
|
|
|
}
|
2022-03-16 21:02:31 +01:00
|
|
|
th := os.Getenv("TEST_HOST")
|
|
|
|
if th == "" {
|
|
|
|
return nil, fmt.Errorf("no TEST_HOST set")
|
|
|
|
}
|
2023-01-12 01:01:33 +01:00
|
|
|
fmt.Printf("XXX: TEST_HOST: %s\n", th)
|
2023-01-07 11:31:46 +01:00
|
|
|
tp := 25
|
|
|
|
if tps := os.Getenv("TEST_PORT"); tps != "" {
|
|
|
|
tpi, err := strconv.Atoi(tps)
|
|
|
|
if err == nil {
|
|
|
|
tp = tpi
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sv := false
|
|
|
|
if sve := os.Getenv("TEST_TLS_SKIP_VERIFY"); sve != "" {
|
|
|
|
sv = true
|
|
|
|
}
|
|
|
|
c, err := NewClient(th, WithPort(tp))
|
2022-03-16 21:02:31 +01:00
|
|
|
if err != nil {
|
|
|
|
return c, err
|
|
|
|
}
|
2023-01-07 11:31:46 +01:00
|
|
|
c.tlsconfig.InsecureSkipVerify = sv
|
2022-03-16 21:02:31 +01:00
|
|
|
if auth {
|
|
|
|
st := os.Getenv("TEST_SMTPAUTH_TYPE")
|
|
|
|
if st != "" {
|
|
|
|
c.SetSMTPAuth(SMTPAuthType(st))
|
|
|
|
}
|
|
|
|
u := os.Getenv("TEST_SMTPAUTH_USER")
|
|
|
|
if u != "" {
|
|
|
|
c.SetUsername(u)
|
|
|
|
}
|
|
|
|
p := os.Getenv("TEST_SMTPAUTH_PASS")
|
|
|
|
if p != "" {
|
|
|
|
c.SetPassword(p)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if err := c.DialWithContext(context.Background()); err != nil {
|
2022-10-17 18:16:00 +02:00
|
|
|
return c, fmt.Errorf("connection to test server failed: %w", err)
|
2022-03-16 21:02:31 +01:00
|
|
|
}
|
|
|
|
if err := c.Close(); err != nil {
|
2022-10-17 18:16:00 +02:00
|
|
|
return c, fmt.Errorf("disconnect from test server failed: %w", err)
|
2022-03-16 21:02:31 +01:00
|
|
|
}
|
|
|
|
return c, nil
|
|
|
|
}
|
2022-09-12 09:31:42 +02:00
|
|
|
|
|
|
|
// getTestConnectionWithDSN takes environment variables to establish a connection to a real
|
|
|
|
// SMTP server to test all functionality that requires a connection. It also enables DSN
|
|
|
|
func getTestConnectionWithDSN(auth bool) (*Client, error) {
|
|
|
|
if os.Getenv("TEST_SKIP_ONLINE") != "" {
|
|
|
|
return nil, fmt.Errorf("env variable TEST_SKIP_ONLINE is set. Skipping online tests")
|
|
|
|
}
|
|
|
|
th := os.Getenv("TEST_HOST")
|
|
|
|
if th == "" {
|
|
|
|
return nil, fmt.Errorf("no TEST_HOST set")
|
|
|
|
}
|
|
|
|
c, err := NewClient(th, WithDSN())
|
|
|
|
if err != nil {
|
|
|
|
return c, err
|
|
|
|
}
|
|
|
|
if auth {
|
|
|
|
st := os.Getenv("TEST_SMTPAUTH_TYPE")
|
|
|
|
if st != "" {
|
|
|
|
c.SetSMTPAuth(SMTPAuthType(st))
|
|
|
|
}
|
|
|
|
u := os.Getenv("TEST_SMTPAUTH_USER")
|
|
|
|
if u != "" {
|
|
|
|
c.SetUsername(u)
|
|
|
|
}
|
|
|
|
p := os.Getenv("TEST_SMTPAUTH_PASS")
|
|
|
|
if p != "" {
|
|
|
|
c.SetPassword(p)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if err := c.DialWithContext(context.Background()); err != nil {
|
2022-10-17 18:16:00 +02:00
|
|
|
return c, fmt.Errorf("connection to test server failed: %w", err)
|
2022-09-12 09:31:42 +02:00
|
|
|
}
|
|
|
|
if err := c.Close(); err != nil {
|
2022-10-17 18:16:00 +02:00
|
|
|
return c, fmt.Errorf("disconnect from test server failed: %w", err)
|
2022-09-12 09:31:42 +02:00
|
|
|
}
|
|
|
|
return c, nil
|
|
|
|
}
|