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 (
2014-10-10 02:08:07 +04:00
"crypto/tls"
2014-03-19 16:27:27 +04:00
"fmt"
2014-10-10 02:08:07 +04:00
"net"
2014-03-19 16:27:27 +04:00
"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-10-08 11:37:54 +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-10-10 02:08:07 +04: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 04:52:03 +03:00
if ok , _ := client . Extension ( "AUTH" ) ; ok && auth != nil {
2014-10-10 02:08:07 +04: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 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-10-10 02:08:07 +04:00
err := 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-10-10 02:08:07 +04:00
err := 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" ,
}
}