mirror of
https://github.com/axllent/mailpit.git
synced 2025-01-22 03:39:59 +02:00
139 lines
2.9 KiB
Go
139 lines
2.9 KiB
Go
package storage
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"database/sql"
|
|
"encoding/json"
|
|
"net/mail"
|
|
"os"
|
|
|
|
"github.com/axllent/mailpit/internal/logger"
|
|
"github.com/axllent/mailpit/internal/tools"
|
|
"github.com/jhillyerd/enmime"
|
|
"github.com/leporo/sqlf"
|
|
)
|
|
|
|
// ReindexAll will regenerate the search text and snippet for a message
|
|
// and update the database.
|
|
func ReindexAll() {
|
|
ids := []string{}
|
|
var i string
|
|
chunkSize := 1000
|
|
|
|
finished := 0
|
|
|
|
err := sqlf.Select("ID").To(&i).
|
|
From("mailbox").
|
|
OrderBy("Created DESC").
|
|
QueryAndClose(nil, db, func(row *sql.Rows) {
|
|
ids = append(ids, i)
|
|
})
|
|
|
|
if err != nil {
|
|
logger.Log().Errorf("[db] %s", err.Error())
|
|
os.Exit(1)
|
|
}
|
|
|
|
total := len(ids)
|
|
|
|
chunks := chunkBy(ids, chunkSize)
|
|
|
|
logger.Log().Infof("reindexing %d messages", total)
|
|
|
|
type updateStruct struct {
|
|
ID string
|
|
SearchText string
|
|
Snippet string
|
|
Metadata string
|
|
}
|
|
|
|
for _, ids := range chunks {
|
|
updates := []updateStruct{}
|
|
|
|
for _, id := range ids {
|
|
raw, err := GetMessageRaw(id)
|
|
if err != nil {
|
|
logger.Log().Error(err)
|
|
continue
|
|
}
|
|
|
|
r := bytes.NewReader(raw)
|
|
|
|
env, err := enmime.ReadEnvelope(r)
|
|
if err != nil {
|
|
logger.Log().Errorf("[message] %s", err.Error())
|
|
continue
|
|
}
|
|
|
|
from := &mail.Address{}
|
|
fromJSON := addressToSlice(env, "From")
|
|
if len(fromJSON) > 0 {
|
|
from = fromJSON[0]
|
|
} else if env.GetHeader("From") != "" {
|
|
from = &mail.Address{Name: env.GetHeader("From")}
|
|
}
|
|
|
|
obj := DBMailSummary{
|
|
From: from,
|
|
To: addressToSlice(env, "To"),
|
|
Cc: addressToSlice(env, "Cc"),
|
|
Bcc: addressToSlice(env, "Bcc"),
|
|
ReplyTo: addressToSlice(env, "Reply-To"),
|
|
}
|
|
|
|
MetadataJSON, err := json.Marshal(obj)
|
|
if err != nil {
|
|
logger.Log().Errorf("[message] %s", err.Error())
|
|
continue
|
|
}
|
|
|
|
searchText := createSearchText(env)
|
|
snippet := tools.CreateSnippet(env.Text, env.HTML)
|
|
|
|
u := updateStruct{}
|
|
u.ID = id
|
|
u.SearchText = searchText
|
|
u.Snippet = snippet
|
|
u.Metadata = string(MetadataJSON)
|
|
|
|
updates = append(updates, u)
|
|
}
|
|
|
|
ctx := context.Background()
|
|
tx, err := db.BeginTx(ctx, nil)
|
|
if err != nil {
|
|
logger.Log().Errorf("[db] %s", err.Error())
|
|
continue
|
|
}
|
|
|
|
// roll back if it fails
|
|
defer tx.Rollback()
|
|
|
|
// insert mail summary data
|
|
for _, u := range updates {
|
|
_, err = tx.Exec("UPDATE mailbox SET SearchText = ?, Snippet = ?, Metadata = ? WHERE ID = ?", u.SearchText, u.Snippet, u.Metadata, u.ID)
|
|
if err != nil {
|
|
logger.Log().Errorf("[db] %s", err.Error())
|
|
continue
|
|
}
|
|
}
|
|
|
|
if err := tx.Commit(); err != nil {
|
|
logger.Log().Errorf("[db] %s", err.Error())
|
|
continue
|
|
}
|
|
|
|
finished += len(updates)
|
|
|
|
logger.Log().Printf("reindexed: %d / %d (%d%%)", finished, total, finished*100/total)
|
|
}
|
|
}
|
|
|
|
func chunkBy[T any](items []T, chunkSize int) (chunks [][]T) {
|
|
for chunkSize < len(items) {
|
|
items, chunks = items[chunkSize:], append(chunks, items[0:chunkSize:chunkSize])
|
|
}
|
|
return append(chunks, items)
|
|
}
|