2022-07-29 23:23:08 +12:00
|
|
|
package storage
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/mail"
|
|
|
|
"regexp"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/axllent/mailpit/config"
|
|
|
|
"github.com/axllent/mailpit/logger"
|
|
|
|
"github.com/axllent/mailpit/server/websockets"
|
|
|
|
"github.com/jhillyerd/enmime"
|
|
|
|
"github.com/k3a/html2text"
|
2022-07-30 08:39:58 +12:00
|
|
|
"github.com/ostafen/clover/v2"
|
2022-07-29 23:23:08 +12:00
|
|
|
)
|
|
|
|
|
|
|
|
// Return a header field as a []*mail.Address, or "null" is not found/empty
|
|
|
|
func addressToSlice(env *enmime.Envelope, key string) []*mail.Address {
|
|
|
|
data, _ := env.AddressList(key)
|
|
|
|
|
|
|
|
return data
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generate the search text based on some header fields (to, from, subject etc)
|
|
|
|
// and either the stripped HTML body (if exists) or text body
|
|
|
|
func createSearchText(env *enmime.Envelope) string {
|
|
|
|
var b strings.Builder
|
|
|
|
|
|
|
|
b.WriteString(env.GetHeader("From") + " ")
|
|
|
|
b.WriteString(env.GetHeader("Subject") + " ")
|
|
|
|
b.WriteString(env.GetHeader("To") + " ")
|
|
|
|
b.WriteString(env.GetHeader("Cc") + " ")
|
|
|
|
b.WriteString(env.GetHeader("Bcc") + " ")
|
|
|
|
h := strings.TrimSpace(html2text.HTML2Text(env.HTML))
|
|
|
|
if h != "" {
|
|
|
|
b.WriteString(h + " ")
|
|
|
|
} else {
|
|
|
|
b.WriteString(env.Text + " ")
|
|
|
|
}
|
|
|
|
// add attachment filenames
|
|
|
|
for _, a := range env.Attachments {
|
|
|
|
b.WriteString(a.FileName + " ")
|
|
|
|
}
|
|
|
|
|
2022-07-30 23:00:34 +12:00
|
|
|
d := cleanString(b.String())
|
2022-07-29 23:23:08 +12:00
|
|
|
|
2022-07-30 23:00:34 +12:00
|
|
|
return d
|
|
|
|
}
|
|
|
|
|
|
|
|
// cleanString removed unwanted characters from stored search text and search queries
|
|
|
|
func cleanString(str string) string {
|
2022-07-29 23:23:08 +12:00
|
|
|
// remove/replace new lines
|
|
|
|
re := regexp.MustCompile(`(\r?\n|\t|>|<|"|:|\,|;)`)
|
2022-07-30 23:00:34 +12:00
|
|
|
str = re.ReplaceAllString(str, " ")
|
2022-07-29 23:23:08 +12:00
|
|
|
// remove duplicate whitespace and trim
|
2022-07-30 23:00:34 +12:00
|
|
|
return strings.ToLower(strings.Join(strings.Fields(strings.TrimSpace(str)), " "))
|
2022-07-29 23:23:08 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
// Auto-prune runs every 5 minutes to automatically delete oldest messages
|
|
|
|
// if total is greater than the threshold
|
|
|
|
func pruneCron() {
|
|
|
|
for {
|
|
|
|
// time.Sleep(5 * 60 * time.Second)
|
|
|
|
time.Sleep(60 * time.Second)
|
|
|
|
mailboxes, err := db.ListCollections()
|
|
|
|
if err != nil {
|
|
|
|
logger.Log().Errorf("[db] %s", err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, m := range mailboxes {
|
|
|
|
total, _ := db.Count(clover.NewQuery(m))
|
|
|
|
if total > config.MaxMessages {
|
|
|
|
limit := total - config.MaxMessages
|
|
|
|
if limit > 5000 {
|
|
|
|
limit = 5000
|
|
|
|
}
|
|
|
|
start := time.Now()
|
|
|
|
if err := db.Delete(clover.NewQuery(m).
|
|
|
|
Sort(clover.SortOption{Field: "Created", Direction: 1}).
|
|
|
|
Limit(limit)); err != nil {
|
2022-07-30 10:10:46 +12:00
|
|
|
logger.Log().Warnf("Error pruning %s: %s", m, err.Error())
|
2022-07-29 23:23:08 +12:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
elapsed := time.Since(start)
|
|
|
|
logger.Log().Infof("Pruned %d messages from %s in %s", limit, m, elapsed)
|
2022-08-07 00:09:32 +12:00
|
|
|
_ = statsRefresh(m)
|
2022-07-29 23:23:08 +12:00
|
|
|
if !strings.HasSuffix(m, "_data") {
|
|
|
|
websockets.Broadcast("prune", nil)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-08-06 23:53:15 +12:00
|
|
|
|
|
|
|
// SanitizeMailboxName returns a clean mailbox name
|
|
|
|
// allowing only `alphanumeric` characters and `-``
|
|
|
|
func sanitizeMailboxName(mailbox string) string {
|
|
|
|
re := regexp.MustCompile(`[^a-zA-Z0-9\-]`)
|
|
|
|
|
|
|
|
return re.ReplaceAllString(mailbox, "")
|
|
|
|
}
|