2014-03-19 13:27:27 +01:00
|
|
|
// Copyright 2014 The Gogs Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a MIT-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package mailer
|
|
|
|
|
|
|
|
import (
|
2014-10-10 00:08:07 +02:00
|
|
|
"crypto/tls"
|
2014-03-19 13:27:27 +01:00
|
|
|
"fmt"
|
2014-10-10 00:08:07 +02:00
|
|
|
"net"
|
2014-03-19 13:27:27 +01:00
|
|
|
"net/smtp"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/gogits/gogs/modules/log"
|
2014-05-26 02:11:25 +02:00
|
|
|
"github.com/gogits/gogs/modules/setting"
|
2014-03-19 13:27:27 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
type Message struct {
|
|
|
|
To []string
|
|
|
|
From string
|
|
|
|
Subject string
|
|
|
|
Body string
|
|
|
|
User string
|
|
|
|
Type string
|
|
|
|
Massive bool
|
|
|
|
Info string
|
|
|
|
}
|
|
|
|
|
|
|
|
// create mail content
|
|
|
|
func (m Message) Content() string {
|
|
|
|
// set mail type
|
|
|
|
contentType := "text/plain; charset=UTF-8"
|
|
|
|
if m.Type == "html" {
|
|
|
|
contentType = "text/html; charset=UTF-8"
|
|
|
|
}
|
|
|
|
|
|
|
|
// create mail content
|
2014-10-08 09:37:54 +02:00
|
|
|
content := "From: \"" + m.From + "\" <" + m.User +
|
2014-03-19 13:27:27 +01:00
|
|
|
">\r\nSubject: " + m.Subject + "\r\nContent-Type: " + contentType + "\r\n\r\n" + m.Body
|
|
|
|
return content
|
|
|
|
}
|
|
|
|
|
2014-03-20 02:05:48 +01:00
|
|
|
var mailQueue chan *Message
|
|
|
|
|
2014-03-21 06:48:10 +01:00
|
|
|
func NewMailerContext() {
|
2014-05-26 02:11:25 +02:00
|
|
|
mailQueue = make(chan *Message, setting.Cfg.MustInt("mailer", "SEND_BUFFER_LEN", 10))
|
2014-03-20 02:05:48 +01:00
|
|
|
go processMailQueue()
|
|
|
|
}
|
|
|
|
|
|
|
|
func processMailQueue() {
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case msg := <-mailQueue:
|
|
|
|
num, err := Send(msg)
|
|
|
|
tos := strings.Join(msg.To, "; ")
|
|
|
|
info := ""
|
|
|
|
if err != nil {
|
|
|
|
if len(msg.Info) > 0 {
|
|
|
|
info = ", info: " + msg.Info
|
|
|
|
}
|
2014-07-26 06:24:27 +02:00
|
|
|
log.Error(4, fmt.Sprintf("Async sent email %d succeed, not send emails: %s%s err: %s", num, tos, info, err))
|
2014-03-20 07:25:21 +01:00
|
|
|
} else {
|
|
|
|
log.Trace(fmt.Sprintf("Async sent email %d succeed, sent emails: %s%s", num, tos, info))
|
2014-03-20 02:05:48 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-10 00:08:07 +02:00
|
|
|
// sendMail allows mail with self-signed certificates.
|
|
|
|
func sendMail(hostAddressWithPort string, auth smtp.Auth, from string, recipients []string, msgContent []byte) error {
|
|
|
|
client, err := smtp.Dial(hostAddressWithPort)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
host, _, _ := net.SplitHostPort(hostAddressWithPort)
|
|
|
|
tlsConn := &tls.Config{
|
|
|
|
InsecureSkipVerify: true,
|
|
|
|
ServerName: host,
|
|
|
|
}
|
|
|
|
if err = client.StartTLS(tlsConn); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2014-11-01 02:52:03 +01:00
|
|
|
if ok, _ := client.Extension("AUTH"); ok && auth != nil {
|
2014-10-10 00:08:07 +02:00
|
|
|
if err = client.Auth(auth); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if err = client.Mail(from); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, rec := range recipients {
|
|
|
|
if err = client.Rcpt(rec); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
w, err := client.Data()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if _, err = w.Write([]byte(msgContent)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err = w.Close(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return client.Quit()
|
|
|
|
}
|
|
|
|
|
2014-03-19 13:27:27 +01:00
|
|
|
// Direct Send mail message
|
2014-03-20 02:05:48 +01:00
|
|
|
func Send(msg *Message) (int, error) {
|
2014-03-19 13:27:27 +01:00
|
|
|
log.Trace("Sending mails to: %s", strings.Join(msg.To, "; "))
|
2014-05-26 02:11:25 +02:00
|
|
|
host := strings.Split(setting.MailService.Host, ":")
|
2014-03-19 13:27:27 +01:00
|
|
|
|
|
|
|
// get message body
|
|
|
|
content := msg.Content()
|
|
|
|
|
|
|
|
if len(msg.To) == 0 {
|
|
|
|
return 0, fmt.Errorf("empty receive emails")
|
2014-05-15 20:46:04 +02:00
|
|
|
} else if len(msg.Body) == 0 {
|
2014-03-19 13:27:27 +01:00
|
|
|
return 0, fmt.Errorf("empty email body")
|
|
|
|
}
|
|
|
|
|
2014-12-07 14:07:48 +01:00
|
|
|
var auth smtp.Auth
|
|
|
|
if len(setting.MailService.Passwd) > 0 {
|
|
|
|
auth = smtp.PlainAuth("", setting.MailService.User, setting.MailService.Passwd, host[0])
|
|
|
|
}
|
2014-05-15 20:46:04 +02:00
|
|
|
|
2014-03-19 13:27:27 +01:00
|
|
|
if msg.Massive {
|
|
|
|
// send mail to multiple emails one by one
|
|
|
|
num := 0
|
|
|
|
for _, to := range msg.To {
|
|
|
|
body := []byte("To: " + to + "\r\n" + content)
|
2014-10-10 00:08:07 +02:00
|
|
|
err := sendMail(setting.MailService.Host, auth, msg.From, []string{to}, body)
|
2014-03-19 13:27:27 +01:00
|
|
|
if err != nil {
|
|
|
|
return num, err
|
|
|
|
}
|
|
|
|
num++
|
|
|
|
}
|
|
|
|
return num, nil
|
|
|
|
} else {
|
|
|
|
body := []byte("To: " + strings.Join(msg.To, ";") + "\r\n" + content)
|
|
|
|
|
|
|
|
// send to multiple emails in one message
|
2014-10-10 00:08:07 +02:00
|
|
|
err := sendMail(setting.MailService.Host, auth, msg.From, msg.To, body)
|
2014-03-19 13:27:27 +01:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
} else {
|
|
|
|
return 1, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Async Send mail message
|
2014-03-20 02:05:48 +01:00
|
|
|
func SendAsync(msg *Message) {
|
2014-03-19 13:27:27 +01:00
|
|
|
go func() {
|
2014-03-20 02:05:48 +01:00
|
|
|
mailQueue <- msg
|
2014-03-19 13:27:27 +01:00
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create html mail message
|
|
|
|
func NewHtmlMessage(To []string, From, Subject, Body string) Message {
|
|
|
|
return Message{
|
|
|
|
To: To,
|
|
|
|
From: From,
|
|
|
|
Subject: Subject,
|
|
|
|
Body: Body,
|
|
|
|
Type: "html",
|
|
|
|
}
|
|
|
|
}
|