mirror of
https://github.com/ghostersk/gowebmail.git
synced 2026-04-17 16:46:01 +01:00
first commit
This commit is contained in:
759
internal/email/imap.go
Normal file
759
internal/email/imap.go
Normal file
@@ -0,0 +1,759 @@
|
||||
// Package email provides IMAP fetch/sync and SMTP send.
|
||||
package email
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"context"
|
||||
"crypto/tls"
|
||||
"encoding/base64"
|
||||
"fmt"
|
||||
"io"
|
||||
"log"
|
||||
"mime"
|
||||
"mime/multipart"
|
||||
"mime/quotedprintable"
|
||||
netmail "net/mail"
|
||||
"net/smtp"
|
||||
"path/filepath"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"github.com/emersion/go-imap"
|
||||
"github.com/emersion/go-imap/client"
|
||||
|
||||
gomailModels "github.com/yourusername/gomail/internal/models"
|
||||
)
|
||||
|
||||
func imapHostFor(provider gomailModels.AccountProvider) (string, int) {
|
||||
switch provider {
|
||||
case gomailModels.ProviderGmail:
|
||||
return "imap.gmail.com", 993
|
||||
case gomailModels.ProviderOutlook:
|
||||
return "outlook.office365.com", 993
|
||||
default:
|
||||
return "", 993
|
||||
}
|
||||
}
|
||||
|
||||
func smtpHostFor(provider gomailModels.AccountProvider) (string, int) {
|
||||
switch provider {
|
||||
case gomailModels.ProviderGmail:
|
||||
return "smtp.gmail.com", 587
|
||||
case gomailModels.ProviderOutlook:
|
||||
return "smtp.office365.com", 587
|
||||
default:
|
||||
return "", 587
|
||||
}
|
||||
}
|
||||
|
||||
// ---- SASL / OAuth2 Auth ----
|
||||
|
||||
type xoauth2Client struct{ user, token string }
|
||||
|
||||
func (x *xoauth2Client) Start() (string, []byte, error) {
|
||||
payload := fmt.Sprintf("user=%s\x01auth=Bearer %s\x01\x01", x.user, x.token)
|
||||
return "XOAUTH2", []byte(payload), nil
|
||||
}
|
||||
func (x *xoauth2Client) Next([]byte) ([]byte, error) { return []byte{}, nil }
|
||||
|
||||
type xoauth2SMTP struct{ user, token string }
|
||||
|
||||
func (x *xoauth2SMTP) Start(_ *smtp.ServerInfo) (string, []byte, error) {
|
||||
payload := fmt.Sprintf("user=%s\x01auth=Bearer %s\x01\x01", x.user, x.token)
|
||||
return "XOAUTH2", []byte(payload), nil
|
||||
}
|
||||
func (x *xoauth2SMTP) Next(fromServer []byte, more bool) ([]byte, error) {
|
||||
if more {
|
||||
if dec, err := base64.StdEncoding.DecodeString(string(fromServer)); err == nil {
|
||||
return nil, fmt.Errorf("XOAUTH2 error: %s", dec)
|
||||
}
|
||||
return nil, fmt.Errorf("XOAUTH2 error: %s", fromServer)
|
||||
}
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
// ---- IMAP Client ----
|
||||
|
||||
type Client struct {
|
||||
imap *client.Client
|
||||
account *gomailModels.EmailAccount
|
||||
}
|
||||
|
||||
func Connect(ctx context.Context, account *gomailModels.EmailAccount) (*Client, error) {
|
||||
host, port := imapHostFor(account.Provider)
|
||||
if account.IMAPHost != "" {
|
||||
host = account.IMAPHost
|
||||
port = account.IMAPPort
|
||||
}
|
||||
if host == "" {
|
||||
return nil, fmt.Errorf("IMAP host not configured for account %s", account.EmailAddress)
|
||||
}
|
||||
|
||||
addr := fmt.Sprintf("%s:%d", host, port)
|
||||
var c *client.Client
|
||||
var err error
|
||||
|
||||
if port == 993 {
|
||||
c, err = client.DialTLS(addr, &tls.Config{ServerName: host})
|
||||
} else {
|
||||
c, err = client.Dial(addr)
|
||||
if err == nil {
|
||||
// Attempt STARTTLS; ignore error if server doesn't support it
|
||||
_ = c.StartTLS(&tls.Config{ServerName: host})
|
||||
}
|
||||
}
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("IMAP connect %s: %w", addr, err)
|
||||
}
|
||||
|
||||
switch account.Provider {
|
||||
case gomailModels.ProviderGmail, gomailModels.ProviderOutlook:
|
||||
sasl := &xoauth2Client{user: account.EmailAddress, token: account.AccessToken}
|
||||
if err := c.Authenticate(sasl); err != nil {
|
||||
c.Logout()
|
||||
return nil, fmt.Errorf("IMAP OAuth auth failed: %w", err)
|
||||
}
|
||||
default:
|
||||
if err := c.Login(account.EmailAddress, account.AccessToken); err != nil {
|
||||
c.Logout()
|
||||
return nil, fmt.Errorf("IMAP login failed for %s: %w", account.EmailAddress, err)
|
||||
}
|
||||
}
|
||||
|
||||
return &Client{imap: c, account: account}, nil
|
||||
}
|
||||
|
||||
func TestConnection(account *gomailModels.EmailAccount) error {
|
||||
c, err := Connect(context.Background(), account)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
c.Close()
|
||||
return nil
|
||||
}
|
||||
|
||||
func (c *Client) Close() { c.imap.Logout() }
|
||||
|
||||
func (c *Client) ListMailboxes() ([]*imap.MailboxInfo, error) {
|
||||
ch := make(chan *imap.MailboxInfo, 64)
|
||||
done := make(chan error, 1)
|
||||
go func() { done <- c.imap.List("", "*", ch) }()
|
||||
var result []*imap.MailboxInfo
|
||||
for mb := range ch {
|
||||
result = append(result, mb)
|
||||
}
|
||||
return result, <-done
|
||||
}
|
||||
|
||||
// FetchMessages fetches messages received within the last `days` days.
|
||||
// Falls back to the most recent 200 if the server does not support SEARCH.
|
||||
func (c *Client) FetchMessages(mailboxName string, days int) ([]*gomailModels.Message, error) {
|
||||
mbox, err := c.imap.Select(mailboxName, true)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("select %s: %w", mailboxName, err)
|
||||
}
|
||||
if mbox.Messages == 0 {
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
since := time.Now().AddDate(0, 0, -days)
|
||||
criteria := imap.NewSearchCriteria()
|
||||
criteria.Since = since
|
||||
|
||||
uids, err := c.imap.Search(criteria)
|
||||
if err != nil || len(uids) == 0 {
|
||||
from := uint32(1)
|
||||
if mbox.Messages > 200 {
|
||||
from = mbox.Messages - 199
|
||||
}
|
||||
seqSet := new(imap.SeqSet)
|
||||
seqSet.AddRange(from, mbox.Messages)
|
||||
return c.fetchBySeqSet(seqSet)
|
||||
}
|
||||
|
||||
seqSet := new(imap.SeqSet)
|
||||
for _, uid := range uids {
|
||||
seqSet.AddNum(uid)
|
||||
}
|
||||
return c.fetchBySeqSet(seqSet)
|
||||
}
|
||||
|
||||
func (c *Client) fetchBySeqSet(seqSet *imap.SeqSet) ([]*gomailModels.Message, error) {
|
||||
// Fetch FetchRFC822 (full raw message) so we can properly parse MIME
|
||||
items := []imap.FetchItem{
|
||||
imap.FetchUid, imap.FetchEnvelope,
|
||||
imap.FetchFlags, imap.FetchBodyStructure,
|
||||
imap.FetchRFC822, // full message including headers – needed for proper MIME parsing
|
||||
}
|
||||
|
||||
ch := make(chan *imap.Message, 64)
|
||||
done := make(chan error, 1)
|
||||
go func() { done <- c.imap.Fetch(seqSet, items, ch) }()
|
||||
|
||||
var results []*gomailModels.Message
|
||||
for msg := range ch {
|
||||
m, err := parseIMAPMessage(msg, c.account)
|
||||
if err != nil {
|
||||
log.Printf("parse message uid=%d: %v", msg.Uid, err)
|
||||
continue
|
||||
}
|
||||
results = append(results, m)
|
||||
}
|
||||
if err := <-done; err != nil {
|
||||
return results, fmt.Errorf("fetch: %w", err)
|
||||
}
|
||||
return results, nil
|
||||
}
|
||||
|
||||
func parseIMAPMessage(msg *imap.Message, account *gomailModels.EmailAccount) (*gomailModels.Message, error) {
|
||||
m := &gomailModels.Message{
|
||||
AccountID: account.ID,
|
||||
RemoteUID: fmt.Sprintf("%d", msg.Uid),
|
||||
}
|
||||
|
||||
if env := msg.Envelope; env != nil {
|
||||
m.Subject = env.Subject
|
||||
m.Date = env.Date
|
||||
m.MessageID = env.MessageId
|
||||
if len(env.From) > 0 {
|
||||
m.FromEmail = env.From[0].Address()
|
||||
m.FromName = env.From[0].PersonalName
|
||||
}
|
||||
m.ToList = formatAddressList(env.To)
|
||||
m.CCList = formatAddressList(env.Cc)
|
||||
m.BCCList = formatAddressList(env.Bcc)
|
||||
if len(env.ReplyTo) > 0 {
|
||||
m.ReplyTo = env.ReplyTo[0].Address()
|
||||
}
|
||||
}
|
||||
|
||||
for _, flag := range msg.Flags {
|
||||
switch flag {
|
||||
case imap.SeenFlag:
|
||||
m.IsRead = true
|
||||
case imap.FlaggedFlag:
|
||||
m.IsStarred = true
|
||||
case imap.DraftFlag:
|
||||
m.IsDraft = true
|
||||
}
|
||||
}
|
||||
|
||||
// Parse MIME body from the full raw RFC822 message
|
||||
for _, literal := range msg.Body {
|
||||
raw, err := io.ReadAll(literal)
|
||||
if err != nil {
|
||||
continue
|
||||
}
|
||||
text, html, attachments := parseMIME(raw)
|
||||
if m.BodyText == "" {
|
||||
m.BodyText = text
|
||||
}
|
||||
if m.BodyHTML == "" {
|
||||
m.BodyHTML = html
|
||||
}
|
||||
if len(attachments) > 0 {
|
||||
m.Attachments = append(m.Attachments, attachments...)
|
||||
m.HasAttachment = true
|
||||
}
|
||||
break // only need first body literal
|
||||
}
|
||||
|
||||
if msg.BodyStructure != nil && !m.HasAttachment {
|
||||
m.HasAttachment = hasAttachment(msg.BodyStructure)
|
||||
}
|
||||
if m.Date.IsZero() {
|
||||
m.Date = time.Now()
|
||||
}
|
||||
return m, nil
|
||||
}
|
||||
|
||||
// parseMIME takes a full RFC822 raw message (with headers) and extracts
|
||||
// text/plain, text/html and attachment metadata.
|
||||
func parseMIME(raw []byte) (text, html string, attachments []gomailModels.Attachment) {
|
||||
msg, err := netmail.ReadMessage(bytes.NewReader(raw))
|
||||
if err != nil {
|
||||
// Last-resort fallback: treat whole thing as plain text, strip obvious headers
|
||||
return stripMIMEHeaders(string(raw)), "", nil
|
||||
}
|
||||
ct := msg.Header.Get("Content-Type")
|
||||
if ct == "" {
|
||||
ct = "text/plain"
|
||||
}
|
||||
body, _ := io.ReadAll(msg.Body)
|
||||
text, html, attachments = parsePart(ct, msg.Header.Get("Content-Transfer-Encoding"), body)
|
||||
return
|
||||
}
|
||||
|
||||
// parsePart recursively handles a MIME part.
|
||||
func parsePart(contentType, transferEncoding string, body []byte) (text, html string, attachments []gomailModels.Attachment) {
|
||||
mediaType, params, err := mime.ParseMediaType(contentType)
|
||||
if err != nil {
|
||||
return string(body), "", nil
|
||||
}
|
||||
mediaType = strings.ToLower(mediaType)
|
||||
|
||||
decoded := decodeTransfer(transferEncoding, body)
|
||||
|
||||
switch {
|
||||
case mediaType == "text/plain":
|
||||
text = decodeCharset(params["charset"], decoded)
|
||||
case mediaType == "text/html":
|
||||
html = decodeCharset(params["charset"], decoded)
|
||||
case strings.HasPrefix(mediaType, "multipart/"):
|
||||
boundary := params["boundary"]
|
||||
if boundary == "" {
|
||||
return string(decoded), "", nil
|
||||
}
|
||||
mr := multipart.NewReader(bytes.NewReader(decoded), boundary)
|
||||
for {
|
||||
part, err := mr.NextPart()
|
||||
if err != nil {
|
||||
break
|
||||
}
|
||||
partBody, _ := io.ReadAll(part)
|
||||
partCT := part.Header.Get("Content-Type")
|
||||
if partCT == "" {
|
||||
partCT = "text/plain"
|
||||
}
|
||||
partTE := part.Header.Get("Content-Transfer-Encoding")
|
||||
disposition := part.Header.Get("Content-Disposition")
|
||||
dispType, dispParams, _ := mime.ParseMediaType(disposition)
|
||||
|
||||
if strings.EqualFold(dispType, "attachment") {
|
||||
filename := dispParams["filename"]
|
||||
if filename == "" {
|
||||
filename = part.FileName()
|
||||
}
|
||||
if filename == "" {
|
||||
filename = "attachment"
|
||||
}
|
||||
partMedia, _, _ := mime.ParseMediaType(partCT)
|
||||
attachments = append(attachments, gomailModels.Attachment{
|
||||
Filename: filename,
|
||||
ContentType: partMedia,
|
||||
Size: int64(len(partBody)),
|
||||
})
|
||||
continue
|
||||
}
|
||||
|
||||
t, h, atts := parsePart(partCT, partTE, partBody)
|
||||
if text == "" && t != "" {
|
||||
text = t
|
||||
}
|
||||
if html == "" && h != "" {
|
||||
html = h
|
||||
}
|
||||
attachments = append(attachments, atts...)
|
||||
}
|
||||
default:
|
||||
// Any other type – treat as attachment if it has a filename
|
||||
mt, _, _ := mime.ParseMediaType(contentType)
|
||||
_ = mt
|
||||
}
|
||||
return
|
||||
}
|
||||
|
||||
func decodeTransfer(encoding string, data []byte) []byte {
|
||||
switch strings.ToLower(strings.TrimSpace(encoding)) {
|
||||
case "base64":
|
||||
// Strip whitespace before decoding
|
||||
cleaned := bytes.ReplaceAll(data, []byte("\r\n"), []byte(""))
|
||||
cleaned = bytes.ReplaceAll(cleaned, []byte("\n"), []byte(""))
|
||||
dst := make([]byte, base64.StdEncoding.DecodedLen(len(cleaned)))
|
||||
n, err := base64.StdEncoding.Decode(dst, cleaned)
|
||||
if err != nil {
|
||||
return data
|
||||
}
|
||||
return dst[:n]
|
||||
case "quoted-printable":
|
||||
decoded, err := io.ReadAll(quotedprintable.NewReader(bytes.NewReader(data)))
|
||||
if err != nil {
|
||||
return data
|
||||
}
|
||||
return decoded
|
||||
default:
|
||||
return data
|
||||
}
|
||||
}
|
||||
|
||||
func decodeCharset(charset string, data []byte) string {
|
||||
// We only handle UTF-8 and ASCII natively; for others return as-is
|
||||
// (a proper charset library would be needed for full support)
|
||||
return string(data)
|
||||
}
|
||||
|
||||
// stripMIMEHeaders removes MIME boundary/header lines from a raw body string
|
||||
// when proper parsing fails completely.
|
||||
func stripMIMEHeaders(raw string) string {
|
||||
lines := strings.Split(raw, "\n")
|
||||
var out []string
|
||||
inHeader := false
|
||||
for _, line := range lines {
|
||||
trimmed := strings.TrimSpace(line)
|
||||
if strings.HasPrefix(trimmed, "--") && len(trimmed) > 2 {
|
||||
inHeader = true
|
||||
continue
|
||||
}
|
||||
if inHeader {
|
||||
if trimmed == "" {
|
||||
inHeader = false
|
||||
}
|
||||
continue
|
||||
}
|
||||
out = append(out, line)
|
||||
}
|
||||
return strings.Join(out, "\n")
|
||||
}
|
||||
|
||||
func formatAddressList(addrs []*imap.Address) string {
|
||||
parts := make([]string, 0, len(addrs))
|
||||
for _, a := range addrs {
|
||||
if a.PersonalName != "" {
|
||||
parts = append(parts, fmt.Sprintf("%s <%s>", a.PersonalName, a.Address()))
|
||||
} else {
|
||||
parts = append(parts, a.Address())
|
||||
}
|
||||
}
|
||||
return strings.Join(parts, ", ")
|
||||
}
|
||||
|
||||
func hasAttachment(bs *imap.BodyStructure) bool {
|
||||
if bs == nil {
|
||||
return false
|
||||
}
|
||||
if strings.EqualFold(bs.Disposition, "attachment") {
|
||||
return true
|
||||
}
|
||||
for _, part := range bs.Parts {
|
||||
if hasAttachment(part) {
|
||||
return true
|
||||
}
|
||||
}
|
||||
return false
|
||||
}
|
||||
|
||||
// InferFolderType returns a canonical folder type from mailbox name/attributes.
|
||||
func InferFolderType(name string, attributes []string) string {
|
||||
for _, attr := range attributes {
|
||||
switch strings.ToLower(attr) {
|
||||
case `\inbox`:
|
||||
return "inbox"
|
||||
case `\sent`:
|
||||
return "sent"
|
||||
case `\drafts`:
|
||||
return "drafts"
|
||||
case `\trash`, `\deleted`:
|
||||
return "trash"
|
||||
case `\junk`, `\spam`:
|
||||
return "spam"
|
||||
case `\archive`:
|
||||
return "archive"
|
||||
}
|
||||
}
|
||||
lower := strings.ToLower(name)
|
||||
switch {
|
||||
case lower == "inbox":
|
||||
return "inbox"
|
||||
case strings.Contains(lower, "sent"):
|
||||
return "sent"
|
||||
case strings.Contains(lower, "draft"):
|
||||
return "drafts"
|
||||
case strings.Contains(lower, "trash") || strings.Contains(lower, "deleted"):
|
||||
return "trash"
|
||||
case strings.Contains(lower, "spam") || strings.Contains(lower, "junk"):
|
||||
return "spam"
|
||||
case strings.Contains(lower, "archive"):
|
||||
return "archive"
|
||||
default:
|
||||
return "custom"
|
||||
}
|
||||
}
|
||||
|
||||
// ---- SMTP Send ----
|
||||
|
||||
func authSMTP(c *smtp.Client, account *gomailModels.EmailAccount, host string) error {
|
||||
switch account.Provider {
|
||||
case gomailModels.ProviderGmail, gomailModels.ProviderOutlook:
|
||||
return c.Auth(&xoauth2SMTP{user: account.EmailAddress, token: account.AccessToken})
|
||||
default:
|
||||
ok, _ := c.Extension("AUTH")
|
||||
if !ok {
|
||||
return nil
|
||||
}
|
||||
// PlainAuth requires the correct hostname for TLS verification
|
||||
return c.Auth(smtp.PlainAuth("", account.EmailAddress, account.AccessToken, host))
|
||||
}
|
||||
}
|
||||
|
||||
// SendMessageFull sends an email via SMTP using the account's configured server.
|
||||
// It also appends the sent message to the IMAP Sent folder.
|
||||
func SendMessageFull(ctx context.Context, account *gomailModels.EmailAccount, req *gomailModels.ComposeRequest) error {
|
||||
host, port := smtpHostFor(account.Provider)
|
||||
if account.SMTPHost != "" {
|
||||
host = account.SMTPHost
|
||||
port = account.SMTPPort
|
||||
}
|
||||
if host == "" {
|
||||
return fmt.Errorf("SMTP host not configured")
|
||||
}
|
||||
|
||||
var buf bytes.Buffer
|
||||
buildMIMEMessage(&buf, account, req)
|
||||
rawMsg := buf.Bytes()
|
||||
|
||||
addr := fmt.Sprintf("%s:%d", host, port)
|
||||
|
||||
var c *smtp.Client
|
||||
var err error
|
||||
|
||||
if port == 465 {
|
||||
// Implicit TLS (SMTPS)
|
||||
conn, err2 := tls.Dial("tcp", addr, &tls.Config{ServerName: host})
|
||||
if err2 != nil {
|
||||
return fmt.Errorf("SMTPS dial %s: %w", addr, err2)
|
||||
}
|
||||
c, err = smtp.NewClient(conn, host)
|
||||
} else {
|
||||
// Plain SMTP then upgrade with STARTTLS (port 587 / 25)
|
||||
c, err = smtp.Dial(addr)
|
||||
if err == nil {
|
||||
if err2 := c.Hello("localhost"); err2 != nil {
|
||||
c.Close()
|
||||
return fmt.Errorf("SMTP EHLO: %w", err2)
|
||||
}
|
||||
if ok, _ := c.Extension("STARTTLS"); ok {
|
||||
if err2 := c.StartTLS(&tls.Config{ServerName: host}); err2 != nil {
|
||||
c.Close()
|
||||
return fmt.Errorf("STARTTLS %s: %w", host, err2)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if err != nil {
|
||||
return fmt.Errorf("SMTP dial %s: %w", addr, err)
|
||||
}
|
||||
defer c.Close()
|
||||
|
||||
if err := authSMTP(c, account, host); err != nil {
|
||||
return fmt.Errorf("SMTP auth: %w", err)
|
||||
}
|
||||
|
||||
if err := c.Mail(account.EmailAddress); err != nil {
|
||||
return fmt.Errorf("MAIL FROM: %w", err)
|
||||
}
|
||||
|
||||
allRcpt := append(append([]string{}, req.To...), req.CC...)
|
||||
allRcpt = append(allRcpt, req.BCC...)
|
||||
for _, rcpt := range allRcpt {
|
||||
rcpt = strings.TrimSpace(rcpt)
|
||||
if rcpt == "" {
|
||||
continue
|
||||
}
|
||||
if err := c.Rcpt(rcpt); err != nil {
|
||||
return fmt.Errorf("RCPT %s: %w", rcpt, err)
|
||||
}
|
||||
}
|
||||
|
||||
wc, err := c.Data()
|
||||
if err != nil {
|
||||
return fmt.Errorf("SMTP DATA: %w", err)
|
||||
}
|
||||
if _, err := wc.Write(rawMsg); err != nil {
|
||||
wc.Close()
|
||||
return fmt.Errorf("SMTP write: %w", err)
|
||||
}
|
||||
if err := wc.Close(); err != nil {
|
||||
return fmt.Errorf("SMTP DATA close: %w", err)
|
||||
}
|
||||
if err := c.Quit(); err != nil {
|
||||
log.Printf("SMTP QUIT: %v (ignored)", err)
|
||||
}
|
||||
|
||||
// Append to Sent folder via IMAP (best-effort, don't fail the send)
|
||||
go func() {
|
||||
imapClient, err := Connect(context.Background(), account)
|
||||
if err != nil {
|
||||
log.Printf("AppendToSent: IMAP connect: %v", err)
|
||||
return
|
||||
}
|
||||
defer imapClient.Close()
|
||||
if err := imapClient.AppendToSent(rawMsg); err != nil {
|
||||
log.Printf("AppendToSent: %v", err)
|
||||
}
|
||||
}()
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func buildMIMEMessage(buf *bytes.Buffer, account *gomailModels.EmailAccount, req *gomailModels.ComposeRequest) string {
|
||||
from := netmail.Address{Name: account.DisplayName, Address: account.EmailAddress}
|
||||
boundary := fmt.Sprintf("gomail_%x", time.Now().UnixNano())
|
||||
msgID := fmt.Sprintf("<%d.%s@gomail>", time.Now().UnixNano(), strings.ReplaceAll(account.EmailAddress, "@", "."))
|
||||
|
||||
buf.WriteString("Message-ID: " + msgID + "\r\n")
|
||||
buf.WriteString("From: " + from.String() + "\r\n")
|
||||
buf.WriteString("To: " + strings.Join(req.To, ", ") + "\r\n")
|
||||
if len(req.CC) > 0 {
|
||||
buf.WriteString("Cc: " + strings.Join(req.CC, ", ") + "\r\n")
|
||||
}
|
||||
// Never write BCC to headers — only used for RCPT TO commands
|
||||
buf.WriteString("Subject: " + encodeMIMEHeader(req.Subject) + "\r\n")
|
||||
buf.WriteString("Date: " + time.Now().Format("Mon, 02 Jan 2006 15:04:05 -0700") + "\r\n")
|
||||
buf.WriteString("MIME-Version: 1.0\r\n")
|
||||
buf.WriteString("Content-Type: multipart/alternative; boundary=\"" + boundary + "\"\r\n")
|
||||
buf.WriteString("\r\n")
|
||||
|
||||
// Plain text part
|
||||
buf.WriteString("--" + boundary + "\r\n")
|
||||
buf.WriteString("Content-Type: text/plain; charset=utf-8\r\n")
|
||||
buf.WriteString("Content-Transfer-Encoding: quoted-printable\r\n\r\n")
|
||||
qpw := quotedprintable.NewWriter(buf)
|
||||
plainText := req.BodyText
|
||||
if plainText == "" && req.BodyHTML != "" {
|
||||
plainText = htmlToPlainText(req.BodyHTML)
|
||||
}
|
||||
qpw.Write([]byte(plainText))
|
||||
qpw.Close()
|
||||
buf.WriteString("\r\n")
|
||||
|
||||
// HTML part
|
||||
buf.WriteString("--" + boundary + "\r\n")
|
||||
buf.WriteString("Content-Type: text/html; charset=utf-8\r\n")
|
||||
buf.WriteString("Content-Transfer-Encoding: quoted-printable\r\n\r\n")
|
||||
qpw2 := quotedprintable.NewWriter(buf)
|
||||
if req.BodyHTML != "" {
|
||||
qpw2.Write([]byte(req.BodyHTML))
|
||||
} else {
|
||||
qpw2.Write([]byte("<pre>" + htmlEscape(plainText) + "</pre>"))
|
||||
}
|
||||
qpw2.Close()
|
||||
buf.WriteString("\r\n")
|
||||
|
||||
buf.WriteString("--" + boundary + "--\r\n")
|
||||
return msgID
|
||||
}
|
||||
|
||||
// encodeMIMEHeader encodes a header value with UTF-8 if it contains non-ASCII chars.
|
||||
func encodeMIMEHeader(s string) string {
|
||||
for _, r := range s {
|
||||
if r > 127 {
|
||||
return mime.QEncoding.Encode("utf-8", s)
|
||||
}
|
||||
}
|
||||
return s
|
||||
}
|
||||
|
||||
// htmlToPlainText does a very basic HTML→plain-text strip for the text/plain fallback.
|
||||
func htmlToPlainText(html string) string {
|
||||
// Strip tags
|
||||
var out strings.Builder
|
||||
inTag := false
|
||||
for _, r := range html {
|
||||
switch {
|
||||
case r == '<':
|
||||
inTag = true
|
||||
case r == '>':
|
||||
inTag = false
|
||||
out.WriteRune(' ')
|
||||
case !inTag:
|
||||
out.WriteRune(r)
|
||||
}
|
||||
}
|
||||
// Collapse excessive whitespace
|
||||
lines := strings.Split(out.String(), "\n")
|
||||
var result []string
|
||||
for _, l := range lines {
|
||||
l = strings.TrimSpace(l)
|
||||
if l != "" {
|
||||
result = append(result, l)
|
||||
}
|
||||
}
|
||||
return strings.Join(result, "\n")
|
||||
}
|
||||
|
||||
// AppendToSent saves the sent message to the IMAP Sent folder via APPEND command.
|
||||
func (c *Client) AppendToSent(rawMsg []byte) error {
|
||||
mailboxes, err := c.ListMailboxes()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
// Find the Sent folder
|
||||
var sentName string
|
||||
for _, mb := range mailboxes {
|
||||
ft := InferFolderType(mb.Name, mb.Attributes)
|
||||
if ft == "sent" {
|
||||
sentName = mb.Name
|
||||
break
|
||||
}
|
||||
}
|
||||
if sentName == "" {
|
||||
return nil // no Sent folder found, skip silently
|
||||
}
|
||||
flags := []string{imap.SeenFlag}
|
||||
now := time.Now()
|
||||
return c.imap.Append(sentName, flags, now, bytes.NewReader(rawMsg))
|
||||
}
|
||||
|
||||
func htmlEscape(s string) string {
|
||||
s = strings.ReplaceAll(s, "&", "&")
|
||||
s = strings.ReplaceAll(s, "<", "<")
|
||||
s = strings.ReplaceAll(s, ">", ">")
|
||||
return s
|
||||
}
|
||||
|
||||
// FetchAttachmentData fetches the raw bytes for a specific attachment part from IMAP.
|
||||
// partPath is a dot-separated MIME section path e.g. "2" or "1.2"
|
||||
func (c *Client) FetchAttachmentData(mailboxName, uid, partPath string) ([]byte, string, error) {
|
||||
mbox, err := c.imap.Select(mailboxName, true)
|
||||
if err != nil || mbox == nil {
|
||||
return nil, "", fmt.Errorf("select %s: %w", mailboxName, err)
|
||||
}
|
||||
|
||||
seqSet := new(imap.SeqSet)
|
||||
// Use UID fetch
|
||||
bodySection := &imap.BodySectionName{
|
||||
BodyPartName: imap.BodyPartName{
|
||||
Specifier: imap.MIMESpecifier,
|
||||
Path: partPathToInts(partPath),
|
||||
},
|
||||
}
|
||||
items := []imap.FetchItem{bodySection.FetchItem()}
|
||||
|
||||
ch := make(chan *imap.Message, 1)
|
||||
done := make(chan error, 1)
|
||||
uidNum := uint32(0)
|
||||
fmt.Sscanf(uid, "%d", &uidNum)
|
||||
seqSet.AddNum(uidNum)
|
||||
go func() { done <- c.imap.Fetch(seqSet, items, ch) }()
|
||||
|
||||
var data []byte
|
||||
for msg := range ch {
|
||||
for _, literal := range msg.Body {
|
||||
data, _ = io.ReadAll(literal)
|
||||
break
|
||||
}
|
||||
}
|
||||
if err := <-done; err != nil {
|
||||
return nil, "", err
|
||||
}
|
||||
ct := "application/octet-stream"
|
||||
ext := filepath.Ext(partPath)
|
||||
if ext != "" {
|
||||
ct = mime.TypeByExtension(ext)
|
||||
}
|
||||
return data, ct, nil
|
||||
}
|
||||
|
||||
func partPathToInts(path string) []int {
|
||||
if path == "" {
|
||||
return nil
|
||||
}
|
||||
parts := strings.Split(path, ".")
|
||||
result := make([]int, 0, len(parts))
|
||||
for _, p := range parts {
|
||||
var n int
|
||||
fmt.Sscanf(p, "%d", &n)
|
||||
result = append(result, n)
|
||||
}
|
||||
return result
|
||||
}
|
||||
Reference in New Issue
Block a user