2014-03-19 08:27:27 -04:00
// Copyright 2014 The Gogs Authors. All rights reserved.
2017-06-07 03:14:31 +02:00
// Copyright 2017 The Gitea Authors. All rights reserved.
2014-03-19 08:27:27 -04:00
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package mailer
import (
2019-02-03 02:06:52 +00:00
"bytes"
2014-10-09 18:08:07 -04:00
"crypto/tls"
2014-03-19 08:27:27 -04:00
"fmt"
2015-09-17 01:54:12 -04:00
"io"
2014-10-09 18:08:07 -04:00
"net"
2014-03-19 08:27:27 -04:00
"net/smtp"
2015-02-19 10:47:05 +03:00
"os"
2016-12-25 13:55:22 +00:00
"os/exec"
2014-03-19 08:27:27 -04:00
"strings"
2015-09-17 01:54:12 -04:00
"time"
2017-09-11 08:33:28 +02:00
"code.gitea.io/gitea/modules/base"
2020-01-16 17:55:36 +00:00
"code.gitea.io/gitea/modules/graceful"
2016-11-10 17:24:48 +01:00
"code.gitea.io/gitea/modules/log"
2020-01-16 17:55:36 +00:00
"code.gitea.io/gitea/modules/queue"
2016-11-10 17:24:48 +01:00
"code.gitea.io/gitea/modules/setting"
2017-06-07 03:14:31 +02:00
"github.com/jaytaylor/html2text"
"gopkg.in/gomail.v2"
2014-03-19 08:27:27 -04:00
)
2016-11-25 09:44:04 +08:00
// Message mail body and log info
2015-09-17 01:54:12 -04:00
type Message struct {
2020-01-16 17:55:36 +00:00
Info string // Message information for log purpose.
FromAddress string
FromDisplayName string
To [ ] string
Subject string
Date time . Time
Body string
Headers map [ string ] [ ] string
2015-09-17 01:54:12 -04:00
}
2020-01-16 17:55:36 +00:00
// ToMessage converts a Message to gomail.Message
func ( m * Message ) ToMessage ( ) * gomail . Message {
2015-09-17 01:54:12 -04:00
msg := gomail . NewMessage ( )
2020-01-16 17:55:36 +00:00
msg . SetAddressHeader ( "From" , m . FromAddress , m . FromDisplayName )
msg . SetHeader ( "To" , m . To ... )
for header := range m . Headers {
msg . SetHeader ( header , m . Headers [ header ] ... )
}
2019-04-17 05:56:40 +01:00
if len ( setting . MailService . SubjectPrefix ) > 0 {
2020-01-16 17:55:36 +00:00
msg . SetHeader ( "Subject" , setting . MailService . SubjectPrefix + " " + m . Subject )
2019-04-17 05:56:40 +01:00
} else {
2020-01-16 17:55:36 +00:00
msg . SetHeader ( "Subject" , m . Subject )
2019-04-17 05:56:40 +01:00
}
2020-01-16 17:55:36 +00:00
msg . SetDateHeader ( "Date" , m . Date )
2019-04-02 11:45:54 -04:00
msg . SetHeader ( "X-Auto-Response-Suppress" , "All" )
2016-05-30 01:32:01 -07:00
2020-01-16 17:55:36 +00:00
plainBody , err := html2text . FromString ( m . Body )
2017-06-07 03:14:31 +02:00
if err != nil || setting . MailService . SendAsPlainText {
2020-01-16 17:55:36 +00:00
if strings . Contains ( base . TruncateString ( m . Body , 100 ) , "<html>" ) {
2017-06-07 03:14:31 +02:00
log . Warn ( "Mail contains HTML but configured to send as plain text." )
2016-05-30 01:50:20 -07:00
}
2017-06-07 03:14:31 +02:00
msg . SetBody ( "text/plain" , plainBody )
} else {
msg . SetBody ( "text/plain" , plainBody )
2020-01-16 17:55:36 +00:00
msg . AddAlternative ( "text/html" , m . Body )
2016-05-30 10:18:49 +02:00
}
2020-01-16 17:55:36 +00:00
return msg
}
// SetHeader adds additional headers to a message
func ( m * Message ) SetHeader ( field string , value ... string ) {
m . Headers [ field ] = value
}
// NewMessageFrom creates new mail message object with custom From header.
func NewMessageFrom ( to [ ] string , fromDisplayName , fromAddress , subject , body string ) * Message {
log . Trace ( "NewMessageFrom (body):\n%s" , body )
2015-09-17 01:54:12 -04:00
return & Message {
2020-01-16 17:55:36 +00:00
FromAddress : fromAddress ,
FromDisplayName : fromDisplayName ,
To : to ,
Subject : subject ,
Date : time . Now ( ) ,
Body : body ,
Headers : map [ string ] [ ] string { } ,
2015-09-17 01:54:12 -04:00
}
}
// NewMessage creates new mail message object with default From header.
func NewMessage ( to [ ] string , subject , body string ) * Message {
2017-09-21 06:29:45 +02:00
return NewMessageFrom ( to , setting . MailService . FromName , setting . MailService . FromEmail , subject , body )
2015-09-17 01:54:12 -04:00
}
2015-08-20 13:56:25 +08:00
type loginAuth struct {
2015-08-20 19:12:55 +08:00
username , password string
2015-08-20 13:56:25 +08:00
}
2016-11-25 09:44:04 +08:00
// LoginAuth SMTP AUTH LOGIN Auth Handler
2015-08-20 13:56:25 +08:00
func LoginAuth ( username , password string ) smtp . Auth {
return & loginAuth { username , password }
}
2016-11-25 09:44:04 +08:00
// Start start SMTP login auth
2015-08-20 13:56:25 +08:00
func ( a * loginAuth ) Start ( server * smtp . ServerInfo ) ( string , [ ] byte , error ) {
return "LOGIN" , [ ] byte { } , nil
}
2016-11-25 09:44:04 +08:00
// Next next step of SMTP login auth
2015-08-20 13:56:25 +08:00
func ( a * loginAuth ) Next ( fromServer [ ] byte , more bool ) ( [ ] byte , error ) {
if more {
switch string ( fromServer ) {
case "Username:" :
return [ ] byte ( a . username ) , nil
case "Password:" :
return [ ] byte ( a . password ) , nil
default :
2017-06-17 19:30:04 -05:00
return nil , fmt . Errorf ( "unknown fromServer: %s" , string ( fromServer ) )
2015-08-20 13:56:25 +08:00
}
}
return nil , nil
}
2016-12-25 13:55:22 +00:00
// Sender SMTP mail sender
type smtpSender struct {
2014-03-19 08:27:27 -04:00
}
2016-11-25 09:44:04 +08:00
// Send send email
2016-12-25 13:55:22 +00:00
func ( s * smtpSender ) Send ( from string , to [ ] string , msg io . WriterTo ) error {
2015-09-17 01:54:12 -04:00
opts := setting . MailService
2014-03-19 21:05:48 -04:00
2015-09-17 01:54:12 -04:00
host , port , err := net . SplitHostPort ( opts . Host )
2014-10-09 18:08:07 -04:00
if err != nil {
return err
}
2014-12-18 13:34:30 +02:00
tlsconfig := & tls . Config {
2015-09-17 01:54:12 -04:00
InsecureSkipVerify : opts . SkipVerify ,
2014-10-09 18:08:07 -04:00
ServerName : host ,
}
2014-12-18 13:34:30 +02:00
2015-09-17 01:54:12 -04:00
if opts . UseCertificate {
cert , err := tls . LoadX509KeyPair ( opts . CertFile , opts . KeyFile )
2015-02-13 10:33:55 +03:00
if err != nil {
return err
}
tlsconfig . Certificates = [ ] tls . Certificate { cert }
2014-10-09 18:08:07 -04:00
}
2014-12-18 13:34:30 +02:00
2014-12-19 00:24:17 -05:00
conn , err := net . Dial ( "tcp" , net . JoinHostPort ( host , port ) )
if err != nil {
2014-12-18 13:34:30 +02:00
return err
}
defer conn . Close ( )
2018-11-26 20:21:42 +01:00
isSecureConn := opts . IsTLSEnabled || ( strings . HasSuffix ( port , "465" ) )
2014-12-18 13:34:30 +02:00
// Start TLS directly if the port ends with 465 (SMTPS protocol)
2018-11-26 20:21:42 +01:00
if isSecureConn {
2014-12-18 13:34:30 +02:00
conn = tls . Client ( conn , tlsconfig )
}
2014-12-19 00:24:17 -05:00
client , err := smtp . NewClient ( conn , host )
if err != nil {
2015-12-19 02:44:34 -05:00
return fmt . Errorf ( "NewClient: %v" , err )
2014-10-09 18:08:07 -04:00
}
2016-08-29 00:10:21 -07:00
if ! opts . DisableHelo {
hostname := opts . HeloHostname
2015-07-03 14:08:18 +08:00
if len ( hostname ) == 0 {
hostname , err = os . Hostname ( )
if err != nil {
return err
}
}
2015-02-20 10:12:27 +03:00
2015-07-03 14:08:18 +08:00
if err = client . Hello ( hostname ) ; err != nil {
2015-12-19 02:44:34 -05:00
return fmt . Errorf ( "Hello: %v" , err )
2015-07-03 14:08:18 +08:00
}
2015-02-20 10:12:27 +03:00
}
2015-02-19 10:47:05 +03:00
2017-03-14 20:52:01 -04:00
// If not using SMTPS, always use STARTTLS if available
2014-12-19 00:24:17 -05:00
hasStartTLS , _ := client . Extension ( "STARTTLS" )
if ! isSecureConn && hasStartTLS {
2014-12-18 13:34:30 +02:00
if err = client . StartTLS ( tlsconfig ) ; err != nil {
2015-12-19 02:44:34 -05:00
return fmt . Errorf ( "StartTLS: %v" , err )
2014-12-18 13:34:30 +02:00
}
}
2014-12-19 00:24:17 -05:00
canAuth , options := client . Extension ( "AUTH" )
2015-09-17 01:54:12 -04:00
if canAuth && len ( opts . User ) > 0 {
2014-12-18 13:58:18 +02:00
var auth smtp . Auth
2014-12-18 14:15:13 +02:00
if strings . Contains ( options , "CRAM-MD5" ) {
2015-09-17 01:54:12 -04:00
auth = smtp . CRAMMD5Auth ( opts . User , opts . Passwd )
2014-12-18 14:15:13 +02:00
} else if strings . Contains ( options , "PLAIN" ) {
2015-09-17 01:54:12 -04:00
auth = smtp . PlainAuth ( "" , opts . User , opts . Passwd , host )
2015-08-20 13:56:25 +08:00
} else if strings . Contains ( options , "LOGIN" ) {
2015-08-20 19:12:55 +08:00
// Patch for AUTH LOGIN
2015-09-17 01:54:12 -04:00
auth = LoginAuth ( opts . User , opts . Passwd )
2014-12-18 13:58:18 +02:00
}
if auth != nil {
if err = client . Auth ( auth ) ; err != nil {
2015-12-19 02:44:34 -05:00
return fmt . Errorf ( "Auth: %v" , err )
2014-12-18 13:58:18 +02:00
}
2014-10-09 18:08:07 -04:00
}
}
2015-09-17 01:54:12 -04:00
if err = client . Mail ( from ) ; err != nil {
2015-12-19 02:44:34 -05:00
return fmt . Errorf ( "Mail: %v" , err )
2014-10-09 18:08:07 -04:00
}
2015-09-17 01:54:12 -04:00
for _ , rec := range to {
2014-10-09 18:08:07 -04:00
if err = client . Rcpt ( rec ) ; err != nil {
2015-12-19 02:44:34 -05:00
return fmt . Errorf ( "Rcpt: %v" , err )
2014-10-09 18:08:07 -04:00
}
}
w , err := client . Data ( )
if err != nil {
2015-12-19 02:44:34 -05:00
return fmt . Errorf ( "Data: %v" , err )
2015-09-17 01:54:12 -04:00
} else if _ , err = msg . WriteTo ( w ) ; err != nil {
2015-12-19 02:44:34 -05:00
return fmt . Errorf ( "WriteTo: %v" , err )
2015-09-17 01:54:12 -04:00
} else if err = w . Close ( ) ; err != nil {
2015-12-19 02:44:34 -05:00
return fmt . Errorf ( "Close: %v" , err )
2014-10-09 18:08:07 -04:00
}
return client . Quit ( )
}
2016-12-25 13:55:22 +00:00
// Sender sendmail mail sender
type sendmailSender struct {
}
// Send send email
func ( s * sendmailSender ) Send ( from string , to [ ] string , msg io . WriterTo ) error {
var err error
var closeError error
var waitError error
2020-01-24 22:59:39 +00:00
args := [ ] string { "-f" , from , "-i" }
2017-10-25 21:27:25 +02:00
args = append ( args , setting . MailService . SendmailArgs ... )
2016-12-25 13:55:22 +00:00
args = append ( args , to ... )
log . Trace ( "Sending with: %s %v" , setting . MailService . SendmailPath , args )
cmd := exec . Command ( setting . MailService . SendmailPath , args ... )
pipe , err := cmd . StdinPipe ( )
if err != nil {
return err
}
if err = cmd . Start ( ) ; err != nil {
return err
}
2016-12-30 15:26:05 +08:00
_ , err = msg . WriteTo ( pipe )
2014-03-19 08:27:27 -04:00
2016-12-25 13:55:22 +00:00
// we MUST close the pipe or sendmail will hang waiting for more of the message
// Also we should wait on our sendmail command even if something fails
closeError = pipe . Close ( )
2016-12-30 15:26:05 +08:00
waitError = cmd . Wait ( )
2016-12-25 13:55:22 +00:00
if err != nil {
return err
} else if closeError != nil {
return closeError
} else {
return waitError
}
}
2019-02-03 02:06:52 +00:00
// Sender sendmail mail sender
type dummySender struct {
}
// Send send email
func ( s * dummySender ) Send ( from string , to [ ] string , msg io . WriterTo ) error {
buf := bytes . Buffer { }
if _ , err := msg . WriteTo ( & buf ) ; err != nil {
return err
}
log . Info ( "Mail From: %s To: %v Body: %s" , from , to , buf . String ( ) )
return nil
}
2020-01-16 17:55:36 +00:00
var mailQueue queue . Queue
2015-09-17 01:54:12 -04:00
2016-12-25 13:55:22 +00:00
// Sender sender for sending mail synchronously
var Sender gomail . Sender
2016-11-25 09:44:04 +08:00
// NewContext start mail queue service
2015-09-17 01:54:12 -04:00
func NewContext ( ) {
2016-02-20 17:32:34 -05:00
// Need to check if mailQueue is nil because in during reinstall (user had installed
// before but swithed install lock off), this function will be called again
// while mail queue is already processing tasks, and produces a race condition.
if setting . MailService == nil || mailQueue != nil {
2015-09-17 01:54:12 -04:00
return
2014-03-19 08:27:27 -04:00
}
2015-09-17 01:54:12 -04:00
2019-02-03 02:06:52 +00:00
switch setting . MailService . MailerType {
case "smtp" :
2016-12-25 13:55:22 +00:00
Sender = & smtpSender { }
2019-02-03 02:06:52 +00:00
case "sendmail" :
Sender = & sendmailSender { }
case "dummy" :
Sender = & dummySender { }
2016-12-25 13:55:22 +00:00
}
2020-01-16 17:55:36 +00:00
mailQueue = queue . CreateQueue ( "mail" , func ( data ... queue . Data ) {
for _ , datum := range data {
msg := datum . ( * Message )
gomailMsg := msg . ToMessage ( )
log . Trace ( "New e-mail sending request %s: %s" , gomailMsg . GetHeader ( "To" ) , msg . Info )
if err := gomail . Send ( Sender , gomailMsg ) ; err != nil {
log . Error ( "Failed to send emails %s: %s - %v" , gomailMsg . GetHeader ( "To" ) , msg . Info , err )
} else {
log . Trace ( "E-mails sent %s: %s" , gomailMsg . GetHeader ( "To" ) , msg . Info )
}
}
} , & Message { } )
go graceful . GetManager ( ) . RunWithShutdownFns ( mailQueue . Run )
2014-03-19 08:27:27 -04:00
}
2019-11-18 05:08:20 -03:00
// SendAsync send mail asynchronously
2014-03-19 21:05:48 -04:00
func SendAsync ( msg * Message ) {
2014-03-19 08:27:27 -04:00
go func ( ) {
2020-01-16 17:55:36 +00:00
_ = mailQueue . Push ( msg )
2014-03-19 08:27:27 -04:00
} ( )
}
2019-11-18 05:08:20 -03:00
// SendAsyncs send mails asynchronously
func SendAsyncs ( msgs [ ] * Message ) {
go func ( ) {
for _ , msg := range msgs {
2020-01-16 17:55:36 +00:00
_ = mailQueue . Push ( msg )
2019-11-18 05:08:20 -03:00
}
} ( )
}