2023-09-27 17:29:03 +13:00
|
|
|
package handlers
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
2023-11-02 16:15:45 +13:00
|
|
|
"net/url"
|
2023-09-27 17:29:03 +13:00
|
|
|
"regexp"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/axllent/mailpit/config"
|
|
|
|
"github.com/axllent/mailpit/internal/storage"
|
|
|
|
"github.com/gorilla/mux"
|
|
|
|
)
|
|
|
|
|
2023-11-02 16:15:45 +13:00
|
|
|
// RedirectToLatestMessage (method: GET) redirects the web UI to the latest message
|
|
|
|
func RedirectToLatestMessage(w http.ResponseWriter, r *http.Request) {
|
2023-11-11 23:47:20 +13:00
|
|
|
messages := []storage.MessageSummary{}
|
|
|
|
var err error
|
|
|
|
|
|
|
|
search := strings.TrimSpace(r.URL.Query().Get("query"))
|
|
|
|
if search != "" {
|
|
|
|
messages, _, err = storage.Search(search, 0, 1)
|
|
|
|
if err != nil {
|
|
|
|
httpError(w, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
messages, err = storage.List(0, 1)
|
|
|
|
if err != nil {
|
|
|
|
httpError(w, err.Error())
|
|
|
|
return
|
|
|
|
}
|
2023-11-02 16:15:45 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
uri := config.Webroot
|
|
|
|
|
|
|
|
if len(messages) == 1 {
|
|
|
|
uri, err = url.JoinPath(uri, "/view/"+messages[0].ID)
|
|
|
|
if err != nil {
|
|
|
|
httpError(w, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
http.Redirect(w, r, uri, 302)
|
|
|
|
}
|
|
|
|
|
2023-09-27 17:29:03 +13:00
|
|
|
// GetMessageHTML (method: GET) returns a rendered version of a message's HTML part
|
|
|
|
func GetMessageHTML(w http.ResponseWriter, r *http.Request) {
|
|
|
|
// swagger:route GET /view/{ID}.html testing GetMessageHTML
|
|
|
|
//
|
|
|
|
// # Render message HTML part
|
|
|
|
//
|
|
|
|
// Renders just the message's HTML part which can be used for UI integration testing.
|
|
|
|
// Attached inline images are modified to link to the API provided they exist.
|
|
|
|
// Note that is the message does not contain a HTML part then an 404 error is returned.
|
|
|
|
//
|
|
|
|
// The ID can be set to `latest` to return the latest message.
|
|
|
|
//
|
|
|
|
// Produces:
|
|
|
|
// - text/html
|
|
|
|
//
|
|
|
|
// Schemes: http, https
|
|
|
|
//
|
|
|
|
// Parameters:
|
|
|
|
// + name: ID
|
|
|
|
// in: path
|
|
|
|
// description: Database ID or latest
|
|
|
|
// required: true
|
|
|
|
// type: string
|
|
|
|
//
|
|
|
|
// Responses:
|
|
|
|
// 200: HTMLResponse
|
|
|
|
// default: ErrorResponse
|
|
|
|
|
|
|
|
vars := mux.Vars(r)
|
|
|
|
|
|
|
|
id := vars["id"]
|
|
|
|
|
|
|
|
if id == "latest" {
|
2023-11-11 23:47:20 +13:00
|
|
|
var err error
|
2023-12-17 08:36:58 +13:00
|
|
|
id, err = storage.LatestID(r)
|
2023-12-09 23:24:42 +13:00
|
|
|
if err != nil {
|
2023-09-27 17:29:03 +13:00
|
|
|
w.WriteHeader(404)
|
2023-12-09 23:24:42 +13:00
|
|
|
fmt.Fprint(w, err.Error())
|
2023-09-27 17:29:03 +13:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
msg, err := storage.GetMessage(id)
|
|
|
|
if err != nil {
|
|
|
|
w.WriteHeader(404)
|
|
|
|
fmt.Fprint(w, "Message not found")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if msg.HTML == "" {
|
|
|
|
w.WriteHeader(404)
|
|
|
|
fmt.Fprint(w, "This message does not contain a HTML part")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-11-11 23:47:20 +13:00
|
|
|
html := linkInlineImages(msg)
|
2023-09-27 17:29:03 +13:00
|
|
|
w.Header().Add("Content-Type", "text/html; charset=utf-8")
|
|
|
|
_, _ = w.Write([]byte(html))
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetMessageText (method: GET) returns a message's text part
|
|
|
|
func GetMessageText(w http.ResponseWriter, r *http.Request) {
|
|
|
|
// swagger:route GET /view/{ID}.txt testing GetMessageText
|
|
|
|
//
|
|
|
|
// # Render message text part
|
|
|
|
//
|
|
|
|
// Renders just the message's text part which can be used for UI integration testing.
|
|
|
|
//
|
|
|
|
// The ID can be set to `latest` to return the latest message.
|
|
|
|
//
|
|
|
|
// Produces:
|
|
|
|
// - text/plain
|
|
|
|
//
|
|
|
|
// Schemes: http, https
|
|
|
|
//
|
|
|
|
// Parameters:
|
|
|
|
// + name: ID
|
|
|
|
// in: path
|
|
|
|
// description: Database ID or latest
|
|
|
|
// required: true
|
|
|
|
// type: string
|
|
|
|
//
|
|
|
|
// Responses:
|
|
|
|
// 200: TextResponse
|
|
|
|
// default: ErrorResponse
|
|
|
|
|
|
|
|
vars := mux.Vars(r)
|
|
|
|
|
|
|
|
id := vars["id"]
|
|
|
|
|
|
|
|
if id == "latest" {
|
2023-11-11 23:47:20 +13:00
|
|
|
var err error
|
2023-12-17 08:36:58 +13:00
|
|
|
id, err = storage.LatestID(r)
|
2023-12-09 23:24:42 +13:00
|
|
|
if err != nil {
|
2023-09-27 17:29:03 +13:00
|
|
|
w.WriteHeader(404)
|
2023-12-09 23:24:42 +13:00
|
|
|
fmt.Fprint(w, err.Error())
|
2023-09-27 17:29:03 +13:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
msg, err := storage.GetMessage(id)
|
|
|
|
if err != nil {
|
|
|
|
w.WriteHeader(404)
|
|
|
|
fmt.Fprint(w, "Message not found")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
w.Header().Add("Content-Type", "text/plain; charset=utf-8")
|
|
|
|
_, _ = w.Write([]byte(msg.Text))
|
|
|
|
}
|
|
|
|
|
2023-11-11 23:47:20 +13:00
|
|
|
// This will rewrite all inline image paths to API URLs
|
|
|
|
func linkInlineImages(msg *storage.Message) string {
|
2023-09-27 17:29:03 +13:00
|
|
|
html := msg.HTML
|
|
|
|
|
|
|
|
for _, a := range msg.Inline {
|
|
|
|
if a.ContentID != "" {
|
|
|
|
re := regexp.MustCompile(`(?i)(=["\']?)(cid:` + regexp.QuoteMeta(a.ContentID) + `)(["|\'|\\s|\\/|>|;])`)
|
|
|
|
u := config.Webroot + "api/v1/message/" + msg.ID + "/part/" + a.PartID
|
|
|
|
matches := re.FindAllStringSubmatch(html, -1)
|
|
|
|
for _, m := range matches {
|
|
|
|
html = strings.ReplaceAll(html, m[0], m[1]+u+m[3])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, a := range msg.Attachments {
|
|
|
|
if a.ContentID != "" {
|
|
|
|
re := regexp.MustCompile(`(?i)(=["\']?)(cid:` + regexp.QuoteMeta(a.ContentID) + `)(["|\'|\\s|\\/|>|;])`)
|
|
|
|
u := config.Webroot + "api/v1/message/" + msg.ID + "/part/" + a.PartID
|
|
|
|
matches := re.FindAllStringSubmatch(html, -1)
|
|
|
|
for _, m := range matches {
|
|
|
|
html = strings.ReplaceAll(html, m[0], m[1]+u+m[3])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return html
|
|
|
|
}
|