2017-04-12 08:27:57 -04:00
|
|
|
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
2015-06-14 23:53:32 -08:00
|
|
|
// See License.txt for license information.
|
|
|
|
|
|
|
|
|
|
package utils
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"crypto/tls"
|
2016-11-14 09:11:54 -03:00
|
|
|
"mime"
|
2015-06-14 23:53:32 -08:00
|
|
|
"net"
|
|
|
|
|
"net/mail"
|
|
|
|
|
"net/smtp"
|
2015-08-26 20:17:52 +02:00
|
|
|
"time"
|
2017-07-31 08:15:01 -07:00
|
|
|
|
2017-08-21 15:31:13 -03:00
|
|
|
"gopkg.in/gomail.v2"
|
|
|
|
|
|
2017-07-31 08:15:01 -07:00
|
|
|
l4g "github.com/alecthomas/log4go"
|
2017-08-21 15:31:13 -03:00
|
|
|
"github.com/mattermost/html2text"
|
2017-07-31 08:15:01 -07:00
|
|
|
"github.com/mattermost/platform/model"
|
2017-08-25 11:04:37 +01:00
|
|
|
"net/http"
|
2015-06-14 23:53:32 -08:00
|
|
|
)
|
|
|
|
|
|
2015-10-15 21:17:52 +08:00
|
|
|
func encodeRFC2047Word(s string) string {
|
2016-11-14 09:11:54 -03:00
|
|
|
return mime.BEncoding.Encode("utf-8", s)
|
2015-10-15 21:17:52 +08:00
|
|
|
}
|
|
|
|
|
|
2015-09-21 15:11:56 -07:00
|
|
|
func connectToSMTPServer(config *model.Config) (net.Conn, *model.AppError) {
|
2015-06-16 23:05:37 -08:00
|
|
|
var conn net.Conn
|
|
|
|
|
var err error
|
|
|
|
|
|
2015-09-21 17:34:13 -07:00
|
|
|
if config.EmailSettings.ConnectionSecurity == model.CONN_SECURITY_TLS {
|
2015-06-15 11:14:26 -04:00
|
|
|
tlsconfig := &tls.Config{
|
2017-03-27 12:43:27 +01:00
|
|
|
InsecureSkipVerify: *config.EmailSettings.SkipServerCertificateVerification,
|
2015-09-21 15:11:56 -07:00
|
|
|
ServerName: config.EmailSettings.SMTPServer,
|
2015-06-15 11:14:26 -04:00
|
|
|
}
|
|
|
|
|
|
2015-09-21 15:11:56 -07:00
|
|
|
conn, err = tls.Dial("tcp", config.EmailSettings.SMTPServer+":"+config.EmailSettings.SMTPPort, tlsconfig)
|
2015-06-15 11:14:26 -04:00
|
|
|
if err != nil {
|
2017-08-25 11:04:37 +01:00
|
|
|
return nil, model.NewAppError("SendMail", "utils.mail.connect_smtp.open_tls.app_error", nil, err.Error(), http.StatusInternalServerError)
|
2015-06-15 11:14:26 -04:00
|
|
|
}
|
2015-06-16 23:05:37 -08:00
|
|
|
} else {
|
2015-09-21 15:11:56 -07:00
|
|
|
conn, err = net.Dial("tcp", config.EmailSettings.SMTPServer+":"+config.EmailSettings.SMTPPort)
|
2015-06-16 23:05:37 -08:00
|
|
|
if err != nil {
|
2017-08-25 11:04:37 +01:00
|
|
|
return nil, model.NewAppError("SendMail", "utils.mail.connect_smtp.open.app_error", nil, err.Error(), http.StatusInternalServerError)
|
2015-06-16 23:05:37 -08:00
|
|
|
}
|
2015-06-14 23:53:32 -08:00
|
|
|
}
|
|
|
|
|
|
2015-06-17 11:50:51 -04:00
|
|
|
return conn, nil
|
|
|
|
|
}
|
2015-06-14 23:53:32 -08:00
|
|
|
|
2015-09-21 15:11:56 -07:00
|
|
|
func newSMTPClient(conn net.Conn, config *model.Config) (*smtp.Client, *model.AppError) {
|
|
|
|
|
c, err := smtp.NewClient(conn, config.EmailSettings.SMTPServer+":"+config.EmailSettings.SMTPPort)
|
2015-06-14 23:53:32 -08:00
|
|
|
if err != nil {
|
2016-01-25 00:49:19 -03:00
|
|
|
l4g.Error(T("utils.mail.new_client.open.error"), err)
|
2017-08-25 11:04:37 +01:00
|
|
|
return nil, model.NewAppError("SendMail", "utils.mail.connect_smtp.open_tls.app_error", nil, err.Error(), http.StatusInternalServerError)
|
2015-06-14 23:53:32 -08:00
|
|
|
}
|
2017-05-09 15:34:30 +02:00
|
|
|
|
2017-05-19 16:57:36 -04:00
|
|
|
hostname := GetHostnameFromSiteURL(*config.ServiceSettings.SiteURL)
|
|
|
|
|
if hostname != "" {
|
|
|
|
|
err := c.Hello(hostname)
|
|
|
|
|
if err != nil {
|
|
|
|
|
l4g.Error(T("utils.mail.new_client.helo.error"), err)
|
2017-08-25 11:04:37 +01:00
|
|
|
return nil, model.NewAppError("SendMail", "utils.mail.connect_smtp.helo.app_error", nil, err.Error(), http.StatusInternalServerError)
|
2017-05-19 16:57:36 -04:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-31 08:15:01 -07:00
|
|
|
if config.EmailSettings.ConnectionSecurity == model.CONN_SECURITY_STARTTLS {
|
2015-08-19 09:03:55 +02:00
|
|
|
tlsconfig := &tls.Config{
|
2017-03-27 12:43:27 +01:00
|
|
|
InsecureSkipVerify: *config.EmailSettings.SkipServerCertificateVerification,
|
2015-09-21 15:11:56 -07:00
|
|
|
ServerName: config.EmailSettings.SMTPServer,
|
2015-08-19 09:03:55 +02:00
|
|
|
}
|
|
|
|
|
c.StartTLS(tlsconfig)
|
2017-07-31 08:15:01 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if *config.EmailSettings.EnableSMTPAuth {
|
|
|
|
|
auth := smtp.PlainAuth("", config.EmailSettings.SMTPUsername, config.EmailSettings.SMTPPassword, config.EmailSettings.SMTPServer+":"+config.EmailSettings.SMTPPort)
|
|
|
|
|
|
2016-09-02 11:33:26 -07:00
|
|
|
if err = c.Auth(auth); err != nil {
|
2017-08-25 11:04:37 +01:00
|
|
|
return nil, model.NewAppError("SendMail", "utils.mail.new_client.auth.app_error", nil, err.Error(), http.StatusInternalServerError)
|
2016-09-02 11:33:26 -07:00
|
|
|
}
|
2015-06-14 23:53:32 -08:00
|
|
|
}
|
2015-06-17 11:50:51 -04:00
|
|
|
return c, nil
|
|
|
|
|
}
|
|
|
|
|
|
2015-09-29 11:59:26 -07:00
|
|
|
func TestConnection(config *model.Config) {
|
|
|
|
|
if !config.EmailSettings.SendEmailNotifications {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
conn, err1 := connectToSMTPServer(config)
|
|
|
|
|
if err1 != nil {
|
2016-07-04 09:30:57 -04:00
|
|
|
l4g.Error(T("utils.mail.test.configured.error"), T(err1.Message), err1.DetailedError)
|
2015-09-29 11:59:26 -07:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
defer conn.Close()
|
|
|
|
|
|
|
|
|
|
c, err2 := newSMTPClient(conn, config)
|
|
|
|
|
if err2 != nil {
|
2016-07-04 09:30:57 -04:00
|
|
|
l4g.Error(T("utils.mail.test.configured.error"), T(err2.Message), err2.DetailedError)
|
2015-09-29 11:59:26 -07:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
defer c.Quit()
|
|
|
|
|
defer c.Close()
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-21 15:31:13 -03:00
|
|
|
func SendMail(to, subject, htmlBody string) *model.AppError {
|
|
|
|
|
return SendMailUsingConfig(to, subject, htmlBody, Cfg)
|
2015-09-21 15:11:56 -07:00
|
|
|
}
|
|
|
|
|
|
2017-08-21 15:31:13 -03:00
|
|
|
func SendMailUsingConfig(to, subject, htmlBody string, config *model.Config) *model.AppError {
|
2015-09-29 11:59:26 -07:00
|
|
|
if !config.EmailSettings.SendEmailNotifications || len(config.EmailSettings.SMTPServer) == 0 {
|
2015-07-12 23:36:52 -08:00
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-25 00:49:19 -03:00
|
|
|
l4g.Debug(T("utils.mail.send_mail.sending.debug"), to, subject)
|
2016-01-05 20:35:36 -06:00
|
|
|
|
2017-08-21 15:31:13 -03:00
|
|
|
htmlMessage := "\r\n<html><body>" + htmlBody + "</body></html>"
|
2015-06-17 11:50:51 -04:00
|
|
|
|
2017-08-21 15:31:13 -03:00
|
|
|
fromMail := mail.Address{Name: config.EmailSettings.FeedbackName, Address: config.EmailSettings.FeedbackEmail}
|
2015-06-17 11:50:51 -04:00
|
|
|
|
2017-08-21 15:31:13 -03:00
|
|
|
txtBody, err := html2text.FromString(htmlBody)
|
|
|
|
|
if err != nil {
|
|
|
|
|
l4g.Warn(err)
|
|
|
|
|
txtBody = ""
|
2015-06-17 11:50:51 -04:00
|
|
|
}
|
2017-08-21 15:31:13 -03:00
|
|
|
|
|
|
|
|
m := gomail.NewMessage(gomail.SetCharset("UTF-8"))
|
|
|
|
|
m.SetHeaders(map[string][]string{
|
|
|
|
|
"From": {fromMail.String()},
|
|
|
|
|
"To": {to},
|
|
|
|
|
"Subject": {encodeRFC2047Word(subject)},
|
|
|
|
|
"Content-Transfer-Encoding": {"8bit"},
|
|
|
|
|
})
|
|
|
|
|
m.SetDateHeader("Date", time.Now())
|
|
|
|
|
|
|
|
|
|
m.SetBody("text/plain", txtBody)
|
|
|
|
|
m.AddAlternative("text/html", htmlMessage)
|
2015-06-17 11:50:51 -04:00
|
|
|
|
2015-09-21 15:11:56 -07:00
|
|
|
conn, err1 := connectToSMTPServer(config)
|
2015-06-17 11:50:51 -04:00
|
|
|
if err1 != nil {
|
|
|
|
|
return err1
|
|
|
|
|
}
|
|
|
|
|
defer conn.Close()
|
|
|
|
|
|
2015-09-21 15:11:56 -07:00
|
|
|
c, err2 := newSMTPClient(conn, config)
|
2015-06-17 11:50:51 -04:00
|
|
|
if err2 != nil {
|
|
|
|
|
return err2
|
|
|
|
|
}
|
|
|
|
|
defer c.Quit()
|
|
|
|
|
defer c.Close()
|
2015-06-14 23:53:32 -08:00
|
|
|
|
2015-06-17 11:50:51 -04:00
|
|
|
if err := c.Mail(fromMail.Address); err != nil {
|
2017-08-25 11:04:37 +01:00
|
|
|
return model.NewAppError("SendMail", "utils.mail.send_mail.from_address.app_error", nil, err.Error(), http.StatusInternalServerError)
|
2015-06-14 23:53:32 -08:00
|
|
|
}
|
|
|
|
|
|
2017-08-21 15:31:13 -03:00
|
|
|
if err := c.Rcpt(to); err != nil {
|
2017-08-25 11:04:37 +01:00
|
|
|
return model.NewAppError("SendMail", "utils.mail.send_mail.to_address.app_error", nil, err.Error(), http.StatusInternalServerError)
|
2015-06-14 23:53:32 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
w, err := c.Data()
|
|
|
|
|
if err != nil {
|
2017-08-25 11:04:37 +01:00
|
|
|
return model.NewAppError("SendMail", "utils.mail.send_mail.msg_data.app_error", nil, err.Error(), http.StatusInternalServerError)
|
2015-06-14 23:53:32 -08:00
|
|
|
}
|
|
|
|
|
|
2017-08-21 15:31:13 -03:00
|
|
|
_, err = m.WriteTo(w)
|
2015-06-14 23:53:32 -08:00
|
|
|
if err != nil {
|
2017-08-25 11:04:37 +01:00
|
|
|
return model.NewAppError("SendMail", "utils.mail.send_mail.msg.app_error", nil, err.Error(), http.StatusInternalServerError)
|
2015-06-14 23:53:32 -08:00
|
|
|
}
|
|
|
|
|
err = w.Close()
|
|
|
|
|
if err != nil {
|
2017-08-25 11:04:37 +01:00
|
|
|
return model.NewAppError("SendMail", "utils.mail.send_mail.close.app_error", nil, err.Error(), http.StatusInternalServerError)
|
2015-06-14 23:53:32 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
}
|