2023-12-17 08:46:29 +13:00
|
|
|
package cmd
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"io"
|
|
|
|
"net/mail"
|
|
|
|
"net/smtp"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/axllent/mailpit/internal/logger"
|
|
|
|
sendmail "github.com/axllent/mailpit/sendmail/cmd"
|
|
|
|
"github.com/spf13/cobra"
|
|
|
|
"golang.org/x/text/language"
|
|
|
|
"golang.org/x/text/message"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
ingestRecent int
|
|
|
|
)
|
|
|
|
|
|
|
|
// ingestCmd represents the ingest command
|
|
|
|
var ingestCmd = &cobra.Command{
|
|
|
|
Use: "ingest <file|folder> ...[file|folder]",
|
|
|
|
Short: "Ingest a file or folder of emails for testing",
|
|
|
|
Long: `Ingest a file or folder of emails for testing.
|
|
|
|
|
|
|
|
This command will scan the folder for emails and deliver them via SMTP to a running
|
|
|
|
Mailpit server. Each email must be a separate file (eg: Maildir format, not mbox).
|
|
|
|
The --recent flag will only consider files with a modification date within the last X days.`,
|
|
|
|
// Hidden: true,
|
|
|
|
Args: cobra.MinimumNArgs(1),
|
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
|
|
var count int
|
|
|
|
var total int
|
|
|
|
var per100start = time.Now()
|
|
|
|
p := message.NewPrinter(language.English)
|
|
|
|
|
|
|
|
for _, a := range args {
|
|
|
|
err := filepath.Walk(a,
|
|
|
|
func(path string, info os.FileInfo, err error) error {
|
|
|
|
if err != nil {
|
|
|
|
logger.Log().Error(err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if !isFile(path) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-03-24 21:37:37 +13:00
|
|
|
if ingestRecent > 0 && time.Since(info.ModTime()) > time.Duration(ingestRecent)*24*time.Hour {
|
2023-12-17 08:46:29 +13:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
f, err := os.Open(filepath.Clean(path))
|
|
|
|
if err != nil {
|
|
|
|
logger.Log().Errorf("%s: %s", path, err.Error())
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
defer f.Close() // #nosec
|
|
|
|
|
|
|
|
body, err := io.ReadAll(f)
|
|
|
|
if err != nil {
|
|
|
|
logger.Log().Errorf("%s: %s", path, err.Error())
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
msg, err := mail.ReadMessage(bytes.NewReader(body))
|
|
|
|
if err != nil {
|
|
|
|
logger.Log().Errorf("error parsing message body: %s", err.Error())
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
recipients := []string{}
|
|
|
|
// get all recipients in To, Cc and Bcc
|
|
|
|
if to, err := msg.Header.AddressList("To"); err == nil {
|
|
|
|
for _, a := range to {
|
|
|
|
recipients = append(recipients, a.Address)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if cc, err := msg.Header.AddressList("Cc"); err == nil {
|
|
|
|
for _, a := range cc {
|
|
|
|
recipients = append(recipients, a.Address)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if bcc, err := msg.Header.AddressList("Bcc"); err == nil {
|
|
|
|
for _, a := range bcc {
|
|
|
|
recipients = append(recipients, a.Address)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if sendmail.FromAddr == "" {
|
|
|
|
if fromAddresses, err := msg.Header.AddressList("From"); err == nil {
|
|
|
|
sendmail.FromAddr = fromAddresses[0].Address
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(recipients) == 0 {
|
|
|
|
// Bcc
|
|
|
|
recipients = []string{sendmail.FromAddr}
|
|
|
|
}
|
|
|
|
|
|
|
|
returnPath := strings.Trim(msg.Header.Get("Return-Path"), "<>")
|
|
|
|
if returnPath == "" {
|
|
|
|
if fromAddresses, err := msg.Header.AddressList("From"); err == nil {
|
|
|
|
returnPath = fromAddresses[0].Address
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
err = smtp.SendMail(sendmail.SMTPAddr, nil, returnPath, recipients, body)
|
|
|
|
if err != nil {
|
2024-01-01 15:25:38 +13:00
|
|
|
logger.Log().Errorf("error sending mail: %s (%s)", err.Error(), path)
|
2023-12-17 08:46:29 +13:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
count++
|
|
|
|
total++
|
|
|
|
if count%100 == 0 {
|
|
|
|
formatted := p.Sprintf("%d", total)
|
|
|
|
logger.Log().Infof("[%s] 100 messages in %s", formatted, time.Since(per100start))
|
|
|
|
|
|
|
|
per100start = time.Now()
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
logger.Log().Error(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
rootCmd.AddCommand(ingestCmd)
|
|
|
|
|
|
|
|
ingestCmd.Flags().StringVarP(&sendmail.SMTPAddr, "smtp-addr", "S", sendmail.SMTPAddr, "SMTP server address")
|
|
|
|
ingestCmd.Flags().IntVarP(&ingestRecent, "recent", "r", 0, "Only ingest messages from the last X days (default all)")
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsFile returns if a path is a file
|
|
|
|
func isFile(path string) bool {
|
|
|
|
info, err := os.Stat(path)
|
|
|
|
if os.IsNotExist(err) || !info.Mode().IsRegular() {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|