go-common/vendor/github.com/scorredoira/email/email.go

228 lines
5.4 KiB
Go

// Package email allows to send emails with attachments.
package email
import (
"bytes"
"encoding/base64"
"fmt"
"io/ioutil"
"mime"
"net/mail"
"net/smtp"
"path/filepath"
"strings"
"time"
)
// Attachment represents an email attachment.
type Attachment struct {
Filename string
Data []byte
Inline bool
}
// Header represents an additional email header.
type Header struct {
Key string
Value string
}
// Message represents a smtp message.
type Message struct {
From mail.Address
To []string
Cc []string
Bcc []string
ReplyTo string
Subject string
Body string
BodyContentType string
Headers []Header
Attachments map[string]*Attachment
}
func (m *Message) attach(file string, inline bool) error {
data, err := ioutil.ReadFile(file)
if err != nil {
return err
}
_, filename := filepath.Split(file)
m.Attachments[filename] = &Attachment{
Filename: filename,
Data: data,
Inline: inline,
}
return nil
}
func (m *Message) AddTo(address mail.Address) []string {
m.To = append(m.To, address.String())
return m.To
}
func (m *Message) AddCc(address mail.Address) []string {
m.Cc = append(m.Cc, address.String())
return m.Cc
}
func (m *Message) AddBcc(address mail.Address) []string {
m.Bcc = append(m.Bcc, address.String())
return m.Bcc
}
// AttachBuffer attaches a binary attachment.
func (m *Message) AttachBuffer(filename string, buf []byte, inline bool) error {
m.Attachments[filename] = &Attachment{
Filename: filename,
Data: buf,
Inline: inline,
}
return nil
}
// Attach attaches a file.
func (m *Message) Attach(file string) error {
return m.attach(file, false)
}
// Inline includes a file as an inline attachment.
func (m *Message) Inline(file string) error {
return m.attach(file, true)
}
// Ads a Header to message
func (m *Message) AddHeader(key string, value string) Header {
newHeader := Header{Key: key, Value: value}
m.Headers = append(m.Headers, newHeader)
return newHeader
}
func newMessage(subject string, body string, bodyContentType string) *Message {
m := &Message{Subject: subject, Body: body, BodyContentType: bodyContentType}
m.Attachments = make(map[string]*Attachment)
return m
}
// NewMessage returns a new Message that can compose an email with attachments
func NewMessage(subject string, body string) *Message {
return newMessage(subject, body, "text/plain")
}
// NewHTMLMessage returns a new Message that can compose an HTML email with attachments
func NewHTMLMessage(subject string, body string) *Message {
return newMessage(subject, body, "text/html")
}
// Tolist returns all the recipients of the email
func (m *Message) Tolist() []string {
tolist := m.To
for _, cc := range m.Cc {
tolist = append(tolist, cc)
}
for _, bcc := range m.Bcc {
tolist = append(tolist, bcc)
}
return tolist
}
// Bytes returns the mail data
func (m *Message) Bytes() []byte {
buf := bytes.NewBuffer(nil)
buf.WriteString("From: " + m.From.String() + "\r\n")
t := time.Now()
buf.WriteString("Date: " + t.Format(time.RFC1123Z) + "\r\n")
buf.WriteString("To: " + strings.Join(m.To, ",") + "\r\n")
if len(m.Cc) > 0 {
buf.WriteString("Cc: " + strings.Join(m.Cc, ",") + "\r\n")
}
//fix Encode
var coder = base64.StdEncoding
var subject = "=?UTF-8?B?" + coder.EncodeToString([]byte(m.Subject)) + "?="
buf.WriteString("Subject: " + subject + "\r\n")
if len(m.ReplyTo) > 0 {
buf.WriteString("Reply-To: " + m.ReplyTo + "\r\n")
}
buf.WriteString("MIME-Version: 1.0\r\n")
// Add custom headers
if len(m.Headers) > 0 {
for _, header := range m.Headers {
buf.WriteString(fmt.Sprintf("%s: %s\r\n", header.Key, header.Value))
}
}
boundary := "f46d043c813270fc6b04c2d223da"
if len(m.Attachments) > 0 {
buf.WriteString("Content-Type: multipart/mixed; boundary=" + boundary + "\r\n")
buf.WriteString("\r\n--" + boundary + "\r\n")
}
buf.WriteString(fmt.Sprintf("Content-Type: %s; charset=utf-8\r\n\r\n", m.BodyContentType))
buf.WriteString(m.Body)
buf.WriteString("\r\n")
if len(m.Attachments) > 0 {
for _, attachment := range m.Attachments {
buf.WriteString("\r\n\r\n--" + boundary + "\r\n")
if attachment.Inline {
buf.WriteString("Content-Type: message/rfc822\r\n")
buf.WriteString("Content-Disposition: inline; filename=\"" + attachment.Filename + "\"\r\n\r\n")
buf.Write(attachment.Data)
} else {
ext := filepath.Ext(attachment.Filename)
mimetype := mime.TypeByExtension(ext)
if mimetype != "" {
mime := fmt.Sprintf("Content-Type: %s\r\n", mimetype)
buf.WriteString(mime)
} else {
buf.WriteString("Content-Type: application/octet-stream\r\n")
}
buf.WriteString("Content-Transfer-Encoding: base64\r\n")
buf.WriteString("Content-Disposition: attachment; filename=\"=?UTF-8?B?")
buf.WriteString(coder.EncodeToString([]byte(attachment.Filename)))
buf.WriteString("?=\"\r\n\r\n")
b := make([]byte, base64.StdEncoding.EncodedLen(len(attachment.Data)))
base64.StdEncoding.Encode(b, attachment.Data)
// write base64 content in lines of up to 76 chars
for i, l := 0, len(b); i < l; i++ {
buf.WriteByte(b[i])
if (i+1)%76 == 0 {
buf.WriteString("\r\n")
}
}
}
buf.WriteString("\r\n--" + boundary)
}
buf.WriteString("--")
}
return buf.Bytes()
}
// Send sends the message.
func Send(addr string, auth smtp.Auth, m *Message) error {
return smtp.SendMail(addr, auth, m.From.Address, m.Tolist(), m.Bytes())
}