2014-03-19 16:27:27 +04: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 (
"fmt"
"net/smtp"
"strings"
"github.com/gogits/gogs/modules/log"
2014-05-26 04:11:25 +04:00
"github.com/gogits/gogs/modules/setting"
2014-03-19 16:27:27 +04: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-03-26 05:37:18 +04:00
content := "From: " + m . From + "<" + m . User +
2014-03-19 16:27:27 +04:00
">\r\nSubject: " + m . Subject + "\r\nContent-Type: " + contentType + "\r\n\r\n" + m . Body
return content
}
2014-03-20 05:05:48 +04:00
var mailQueue chan * Message
2014-03-21 09:48:10 +04:00
func NewMailerContext ( ) {
2014-05-26 04:11:25 +04:00
mailQueue = make ( chan * Message , setting . Cfg . MustInt ( "mailer" , "SEND_BUFFER_LEN" , 10 ) )
2014-03-20 05:05:48 +04: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 08:24:27 +04: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 10:25:21 +04:00
} else {
log . Trace ( fmt . Sprintf ( "Async sent email %d succeed, sent emails: %s%s" , num , tos , info ) )
2014-03-20 05:05:48 +04:00
}
}
}
}
2014-03-19 16:27:27 +04:00
// Direct Send mail message
2014-03-20 05:05:48 +04:00
func Send ( msg * Message ) ( int , error ) {
2014-03-19 16:27:27 +04:00
log . Trace ( "Sending mails to: %s" , strings . Join ( msg . To , "; " ) )
2014-05-26 04:11:25 +04:00
host := strings . Split ( setting . MailService . Host , ":" )
2014-03-19 16:27:27 +04:00
// get message body
content := msg . Content ( )
if len ( msg . To ) == 0 {
return 0 , fmt . Errorf ( "empty receive emails" )
2014-05-15 22:46:04 +04:00
} else if len ( msg . Body ) == 0 {
2014-03-19 16:27:27 +04:00
return 0 , fmt . Errorf ( "empty email body" )
}
2014-05-26 04:11:25 +04:00
auth := smtp . PlainAuth ( "" , setting . MailService . User , setting . MailService . Passwd , host [ 0 ] )
2014-05-15 22:46:04 +04:00
2014-03-19 16:27:27 +04: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-05-26 04:11:25 +04:00
err := smtp . SendMail ( setting . MailService . Host , auth , msg . From , [ ] string { to } , body )
2014-03-19 16:27:27 +04: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-05-26 04:11:25 +04:00
err := smtp . SendMail ( setting . MailService . Host , auth , msg . From , msg . To , body )
2014-03-19 16:27:27 +04:00
if err != nil {
return 0 , err
} else {
return 1 , nil
}
}
}
// Async Send mail message
2014-03-20 05:05:48 +04:00
func SendAsync ( msg * Message ) {
2014-03-19 16:27:27 +04:00
go func ( ) {
2014-03-20 05:05:48 +04:00
mailQueue <- msg
2014-03-19 16:27:27 +04: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" ,
}
}