2022-10-07 19:48:50 +13:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2023-09-14 22:30:20 +12:00
|
|
|
"io"
|
2022-10-07 19:48:50 +13:00
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
|
|
|
"net/url"
|
2024-04-05 15:48:32 +13:00
|
|
|
"os"
|
2022-10-07 19:48:50 +13:00
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/axllent/mailpit/config"
|
2023-09-25 18:08:04 +13:00
|
|
|
"github.com/axllent/mailpit/internal/logger"
|
2023-09-25 19:25:45 +13:00
|
|
|
"github.com/axllent/mailpit/internal/storage"
|
2022-10-07 19:48:50 +13:00
|
|
|
"github.com/axllent/mailpit/server/apiv1"
|
|
|
|
"github.com/jhillyerd/enmime"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
putDataStruct struct {
|
|
|
|
Read bool `json:"read"`
|
|
|
|
IDs []string `json:"ids"`
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2023-09-25 22:14:19 +13:00
|
|
|
func TestAPIv1Messages(t *testing.T) {
|
2022-10-07 19:48:50 +13:00
|
|
|
setup()
|
|
|
|
defer storage.Close()
|
|
|
|
|
2023-09-14 22:30:20 +12:00
|
|
|
r := apiRoutes()
|
2022-10-07 19:48:50 +13:00
|
|
|
|
|
|
|
ts := httptest.NewServer(r)
|
|
|
|
defer ts.Close()
|
|
|
|
|
|
|
|
m, err := fetchMessages(ts.URL + "/api/v1/messages")
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf(err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
// check count of empty database
|
|
|
|
assertStatsEqual(t, ts.URL+"/api/v1/messages", 0, 0)
|
|
|
|
|
|
|
|
// insert 100
|
|
|
|
t.Log("Insert 100 messages")
|
|
|
|
insertEmailData(t)
|
|
|
|
assertStatsEqual(t, ts.URL+"/api/v1/messages", 100, 100)
|
|
|
|
|
|
|
|
m, err = fetchMessages(ts.URL + "/api/v1/messages")
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf(err.Error())
|
|
|
|
}
|
|
|
|
|
2023-09-25 22:14:19 +13:00
|
|
|
// read first 10 messages
|
2022-10-13 02:48:23 +13:00
|
|
|
t.Log("Read first 10 messages including raw & headers")
|
2023-09-14 22:30:20 +12:00
|
|
|
for idx, msg := range m.Messages {
|
|
|
|
if idx == 10 {
|
2022-10-07 19:48:50 +13:00
|
|
|
break
|
|
|
|
}
|
|
|
|
|
2022-10-13 02:48:23 +13:00
|
|
|
if _, err := clientGet(ts.URL + "/api/v1/message/" + msg.ID); err != nil {
|
|
|
|
t.Errorf(err.Error())
|
|
|
|
}
|
|
|
|
|
2023-09-25 22:14:19 +13:00
|
|
|
// get RAW
|
2022-10-13 02:48:23 +13:00
|
|
|
if _, err := clientGet(ts.URL + "/api/v1/message/" + msg.ID + "/raw"); err != nil {
|
|
|
|
t.Errorf(err.Error())
|
|
|
|
}
|
|
|
|
|
2024-03-24 21:37:37 +13:00
|
|
|
// get headers
|
2022-10-13 02:48:23 +13:00
|
|
|
if _, err := clientGet(ts.URL + "/api/v1/message/" + msg.ID + "/headers"); err != nil {
|
2022-10-07 19:48:50 +13:00
|
|
|
t.Errorf(err.Error())
|
|
|
|
}
|
|
|
|
}
|
2023-09-25 22:14:19 +13:00
|
|
|
|
|
|
|
// 10 should be marked as read
|
2022-10-07 19:48:50 +13:00
|
|
|
assertStatsEqual(t, ts.URL+"/api/v1/messages", 90, 100)
|
|
|
|
|
2023-09-25 22:14:19 +13:00
|
|
|
// delete all
|
|
|
|
t.Log("Delete all messages")
|
|
|
|
_, err = clientDelete(ts.URL+"/api/v1/messages", "{}")
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Expected nil, received %s", err.Error())
|
|
|
|
}
|
|
|
|
assertStatsEqual(t, ts.URL+"/api/v1/messages", 0, 0)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAPIv1ToggleReadStatus(t *testing.T) {
|
|
|
|
setup()
|
|
|
|
defer storage.Close()
|
|
|
|
|
|
|
|
r := apiRoutes()
|
|
|
|
|
|
|
|
ts := httptest.NewServer(r)
|
|
|
|
defer ts.Close()
|
|
|
|
|
|
|
|
m, err := fetchMessages(ts.URL + "/api/v1/messages")
|
2022-10-07 19:48:50 +13:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf(err.Error())
|
|
|
|
}
|
2023-09-25 22:14:19 +13:00
|
|
|
|
|
|
|
// check count of empty database
|
|
|
|
assertStatsEqual(t, ts.URL+"/api/v1/messages", 0, 0)
|
|
|
|
|
|
|
|
// insert 100
|
|
|
|
t.Log("Insert 100 messages")
|
|
|
|
insertEmailData(t)
|
|
|
|
assertStatsEqual(t, ts.URL+"/api/v1/messages", 100, 100)
|
|
|
|
|
|
|
|
m, err = fetchMessages(ts.URL + "/api/v1/messages")
|
2022-10-07 19:48:50 +13:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf(err.Error())
|
|
|
|
}
|
2023-09-25 22:14:19 +13:00
|
|
|
|
|
|
|
// read first 10 IDs
|
|
|
|
t.Log("Get first 10 IDs")
|
|
|
|
putIDS := []string{}
|
|
|
|
for idx, msg := range m.Messages {
|
|
|
|
if idx == 10 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
// store for later
|
|
|
|
putIDS = append(putIDS, msg.ID)
|
|
|
|
}
|
2022-10-07 19:48:50 +13:00
|
|
|
assertStatsEqual(t, ts.URL+"/api/v1/messages", 100, 100)
|
|
|
|
|
2023-09-25 22:14:19 +13:00
|
|
|
// mark first 10 as unread
|
2022-10-07 19:48:50 +13:00
|
|
|
t.Log("Mark first 10 as read")
|
2023-09-25 22:14:19 +13:00
|
|
|
putData := putDataStruct
|
2022-10-07 19:48:50 +13:00
|
|
|
putData.Read = true
|
2023-09-25 22:14:19 +13:00
|
|
|
putData.IDs = putIDS
|
|
|
|
j, err := json.Marshal(putData)
|
2022-10-07 19:48:50 +13:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf(err.Error())
|
|
|
|
}
|
|
|
|
_, err = clientPut(ts.URL+"/api/v1/messages", string(j))
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf(err.Error())
|
|
|
|
}
|
|
|
|
assertStatsEqual(t, ts.URL+"/api/v1/messages", 90, 100)
|
|
|
|
|
2023-09-25 22:14:19 +13:00
|
|
|
// mark first 10 as read
|
|
|
|
t.Log("Mark first 10 as unread")
|
|
|
|
putData.Read = false
|
|
|
|
j, err = json.Marshal(putData)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf(err.Error())
|
|
|
|
}
|
|
|
|
_, err = clientPut(ts.URL+"/api/v1/messages", string(j))
|
2022-10-07 19:48:50 +13:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf(err.Error())
|
|
|
|
}
|
2023-09-25 22:14:19 +13:00
|
|
|
assertStatsEqual(t, ts.URL+"/api/v1/messages", 100, 100)
|
2022-10-07 19:48:50 +13:00
|
|
|
|
|
|
|
// mark all as read
|
|
|
|
putData.Read = true
|
|
|
|
putData.IDs = []string{}
|
|
|
|
j, err = json.Marshal(putData)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf(err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
t.Log("Mark all read")
|
|
|
|
_, err = clientPut(ts.URL+"/api/v1/messages", string(j))
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf(err.Error())
|
|
|
|
}
|
2023-09-25 22:14:19 +13:00
|
|
|
assertStatsEqual(t, ts.URL+"/api/v1/messages", 0, 100)
|
|
|
|
}
|
2022-10-07 19:48:50 +13:00
|
|
|
|
2023-09-25 22:14:19 +13:00
|
|
|
func TestAPIv1Search(t *testing.T) {
|
|
|
|
setup()
|
|
|
|
defer storage.Close()
|
|
|
|
|
|
|
|
r := apiRoutes()
|
|
|
|
|
|
|
|
ts := httptest.NewServer(r)
|
|
|
|
defer ts.Close()
|
|
|
|
|
|
|
|
// insert 100
|
2024-01-01 23:46:34 +13:00
|
|
|
t.Log("Insert 100 messages & tag")
|
2023-09-25 22:14:19 +13:00
|
|
|
insertEmailData(t)
|
|
|
|
assertStatsEqual(t, ts.URL+"/api/v1/messages", 100, 100)
|
|
|
|
|
|
|
|
// search
|
|
|
|
assertSearchEqual(t, ts.URL+"/api/v1/search", "from-1@example.com", 1)
|
|
|
|
assertSearchEqual(t, ts.URL+"/api/v1/search", "from:from-1@example.com", 1)
|
|
|
|
assertSearchEqual(t, ts.URL+"/api/v1/search", "-from:from-1@example.com", 99)
|
2024-01-25 22:19:32 +13:00
|
|
|
assertSearchEqual(t, ts.URL+"/api/v1/search", "-FROM:FROM-1@EXAMPLE.COM", 99)
|
2023-09-25 22:14:19 +13:00
|
|
|
assertSearchEqual(t, ts.URL+"/api/v1/search", "to:from-1@example.com", 0)
|
|
|
|
assertSearchEqual(t, ts.URL+"/api/v1/search", "from:@example.com", 100)
|
|
|
|
assertSearchEqual(t, ts.URL+"/api/v1/search", "subject:\"Subject line\"", 100)
|
2024-01-25 22:19:32 +13:00
|
|
|
assertSearchEqual(t, ts.URL+"/api/v1/search", "subject:\"SUBJECT LINE 17 END\"", 1)
|
2023-09-25 22:14:19 +13:00
|
|
|
assertSearchEqual(t, ts.URL+"/api/v1/search", "!thisdoesnotexist", 100)
|
2024-01-25 22:19:32 +13:00
|
|
|
assertSearchEqual(t, ts.URL+"/api/v1/search", "-ThisDoesNotExist", 100)
|
2023-09-25 22:14:19 +13:00
|
|
|
assertSearchEqual(t, ts.URL+"/api/v1/search", "thisdoesnotexist", 0)
|
2024-01-01 23:46:34 +13:00
|
|
|
assertSearchEqual(t, ts.URL+"/api/v1/search", "tag:\"Test tag 065\"", 1)
|
2024-01-25 22:19:32 +13:00
|
|
|
assertSearchEqual(t, ts.URL+"/api/v1/search", "tag:\"TEST TAG 065\"", 1)
|
2024-01-01 23:46:34 +13:00
|
|
|
assertSearchEqual(t, ts.URL+"/api/v1/search", "!tag:\"Test tag 023\"", 99)
|
2022-10-07 19:48:50 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
func setup() {
|
2023-04-21 11:59:26 +12:00
|
|
|
logger.NoLogging = true
|
2022-10-07 19:48:50 +13:00
|
|
|
config.MaxMessages = 0
|
2024-04-12 14:47:47 +12:00
|
|
|
config.Database = os.Getenv("MP_DATABASE")
|
2022-10-07 19:48:50 +13:00
|
|
|
|
|
|
|
if err := storage.InitDB(); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2024-04-05 15:48:32 +13:00
|
|
|
|
|
|
|
if err := storage.DeleteAllMessages(); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2022-10-07 19:48:50 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
func assertStatsEqual(t *testing.T, uri string, unread, total int) {
|
2022-10-21 22:55:15 +13:00
|
|
|
m := apiv1.MessagesSummary{}
|
2022-10-07 19:48:50 +13:00
|
|
|
|
|
|
|
data, err := clientGet(uri)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := json.Unmarshal(data, &m); err != nil {
|
|
|
|
t.Errorf(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-04-05 15:48:32 +13:00
|
|
|
assertEqual(t, float64(unread), m.Unread, "wrong unread count")
|
|
|
|
assertEqual(t, float64(total), m.Total, "wrong total count")
|
2022-10-07 19:48:50 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
func assertSearchEqual(t *testing.T, uri, query string, count int) {
|
|
|
|
t.Logf("Test search: %s", query)
|
2022-10-21 22:55:15 +13:00
|
|
|
m := apiv1.MessagesSummary{}
|
2022-10-07 19:48:50 +13:00
|
|
|
|
2022-10-14 17:25:07 +13:00
|
|
|
limit := fmt.Sprintf("%d", count)
|
|
|
|
|
|
|
|
data, err := clientGet(uri + "?query=" + url.QueryEscape(query) + "&limit=" + limit)
|
2022-10-07 19:48:50 +13:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := json.Unmarshal(data, &m); err != nil {
|
|
|
|
t.Errorf(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-04-05 15:48:32 +13:00
|
|
|
assertEqual(t, float64(count), m.MessagesCount, "wrong search results count")
|
2022-10-07 19:48:50 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
func insertEmailData(t *testing.T) {
|
|
|
|
for i := 0; i < 100; i++ {
|
|
|
|
msg := enmime.Builder().
|
|
|
|
From(fmt.Sprintf("From %d", i), fmt.Sprintf("from-%d@example.com", i)).
|
|
|
|
Subject(fmt.Sprintf("Subject line %d end", i)).
|
|
|
|
Text([]byte(fmt.Sprintf("This is the email body %d <jdsauk;dwqmdqw;>.", i))).
|
|
|
|
To(fmt.Sprintf("To %d", i), fmt.Sprintf("to-%d@example.com", i))
|
|
|
|
|
|
|
|
env, err := msg.Build()
|
|
|
|
if err != nil {
|
|
|
|
t.Log("error ", err)
|
|
|
|
t.Fail()
|
|
|
|
}
|
|
|
|
|
|
|
|
buf := new(bytes.Buffer)
|
|
|
|
|
|
|
|
if err := env.Encode(buf); err != nil {
|
|
|
|
t.Log("error ", err)
|
|
|
|
t.Fail()
|
|
|
|
}
|
|
|
|
|
2024-01-02 13:14:21 +13:00
|
|
|
bufBytes := buf.Bytes()
|
|
|
|
|
|
|
|
id, err := storage.Store(&bufBytes)
|
2024-01-01 23:46:34 +13:00
|
|
|
if err != nil {
|
|
|
|
t.Log("error ", err)
|
|
|
|
t.Fail()
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := storage.SetMessageTags(id, []string{fmt.Sprintf("Test tag %03d", i)}); err != nil {
|
2022-10-07 19:48:50 +13:00
|
|
|
t.Log("error ", err)
|
|
|
|
t.Fail()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2022-10-21 22:55:15 +13:00
|
|
|
func fetchMessages(url string) (apiv1.MessagesSummary, error) {
|
|
|
|
m := apiv1.MessagesSummary{}
|
2022-10-07 19:48:50 +13:00
|
|
|
|
|
|
|
data, err := clientGet(url)
|
|
|
|
if err != nil {
|
|
|
|
return m, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := json.Unmarshal(data, &m); err != nil {
|
|
|
|
return m, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return m, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func clientGet(url string) ([]byte, error) {
|
|
|
|
resp, err := http.Get(url)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
return nil, fmt.Errorf("%s returned status %d", url, resp.StatusCode)
|
|
|
|
}
|
|
|
|
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
2023-09-14 22:30:20 +12:00
|
|
|
data, err := io.ReadAll(resp.Body)
|
2022-10-07 19:48:50 +13:00
|
|
|
|
|
|
|
return data, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func clientDelete(url, body string) ([]byte, error) {
|
|
|
|
client := new(http.Client)
|
|
|
|
|
|
|
|
b := strings.NewReader(body)
|
|
|
|
req, err := http.NewRequest("DELETE", url, b)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, err := client.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
return nil, fmt.Errorf("%s returned status %d", url, resp.StatusCode)
|
|
|
|
}
|
|
|
|
|
2023-09-14 22:30:20 +12:00
|
|
|
data, err := io.ReadAll(resp.Body)
|
2022-10-07 19:48:50 +13:00
|
|
|
|
|
|
|
return data, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func clientPut(url, body string) ([]byte, error) {
|
|
|
|
client := new(http.Client)
|
|
|
|
|
|
|
|
b := strings.NewReader(body)
|
|
|
|
req, err := http.NewRequest("PUT", url, b)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, err := client.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
return nil, fmt.Errorf("%s returned status %d", url, resp.StatusCode)
|
|
|
|
}
|
|
|
|
|
2023-09-14 22:30:20 +12:00
|
|
|
data, err := io.ReadAll(resp.Body)
|
2022-10-07 19:48:50 +13:00
|
|
|
|
|
|
|
return data, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func assertEqual(t *testing.T, a interface{}, b interface{}, message string) {
|
|
|
|
if a == b {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
message = fmt.Sprintf("%s: \"%v\" != \"%v\"", message, a, b)
|
|
|
|
t.Fatal(message)
|
|
|
|
}
|