mailer.go 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. // Copyright 2014 The Gogs Authors. All rights reserved.
  2. // Use of this source code is governed by a MIT-style
  3. // license that can be found in the LICENSE file.
  4. package notifications
  5. import (
  6. "bytes"
  7. "crypto/tls"
  8. "fmt"
  9. "html/template"
  10. "net"
  11. "strconv"
  12. m "github.com/grafana/grafana/pkg/models"
  13. "github.com/grafana/grafana/pkg/setting"
  14. gomail "gopkg.in/mail.v2"
  15. )
  16. func (ns *NotificationService) send(msg *Message) (int, error) {
  17. dialer, err := ns.createDialer()
  18. if err != nil {
  19. return 0, err
  20. }
  21. for _, address := range msg.To {
  22. m := gomail.NewMessage()
  23. m.SetHeader("From", msg.From)
  24. m.SetHeader("To", address)
  25. m.SetHeader("Subject", msg.Subject)
  26. for _, file := range msg.EmbededFiles {
  27. m.Embed(file)
  28. }
  29. m.SetBody("text/html", msg.Body)
  30. if err := dialer.DialAndSend(m); err != nil {
  31. return 0, err
  32. }
  33. }
  34. return len(msg.To), nil
  35. }
  36. func (ns *NotificationService) createDialer() (*gomail.Dialer, error) {
  37. host, port, err := net.SplitHostPort(ns.Cfg.Smtp.Host)
  38. if err != nil {
  39. return nil, err
  40. }
  41. iPort, err := strconv.Atoi(port)
  42. if err != nil {
  43. return nil, err
  44. }
  45. tlsconfig := &tls.Config{
  46. InsecureSkipVerify: ns.Cfg.Smtp.SkipVerify,
  47. ServerName: host,
  48. }
  49. if ns.Cfg.Smtp.CertFile != "" {
  50. cert, err := tls.LoadX509KeyPair(ns.Cfg.Smtp.CertFile, ns.Cfg.Smtp.KeyFile)
  51. if err != nil {
  52. return nil, fmt.Errorf("Could not load cert or key file. error: %v", err)
  53. }
  54. tlsconfig.Certificates = []tls.Certificate{cert}
  55. }
  56. d := gomail.NewDialer(host, iPort, ns.Cfg.Smtp.User, ns.Cfg.Smtp.Password)
  57. d.TLSConfig = tlsconfig
  58. if ns.Cfg.Smtp.EhloIdentity != "" {
  59. d.LocalName = ns.Cfg.Smtp.EhloIdentity
  60. } else {
  61. d.LocalName = setting.InstanceName
  62. }
  63. return d, nil
  64. }
  65. func (ns *NotificationService) buildEmailMessage(cmd *m.SendEmailCommand) (*Message, error) {
  66. if !ns.Cfg.Smtp.Enabled {
  67. return nil, m.ErrSmtpNotEnabled
  68. }
  69. var buffer bytes.Buffer
  70. var err error
  71. data := cmd.Data
  72. if data == nil {
  73. data = make(map[string]interface{}, 10)
  74. }
  75. setDefaultTemplateData(data, nil)
  76. err = mailTemplates.ExecuteTemplate(&buffer, cmd.Template, data)
  77. if err != nil {
  78. return nil, err
  79. }
  80. subject := cmd.Subject
  81. if cmd.Subject == "" {
  82. var subjectText interface{}
  83. subjectData := data["Subject"].(map[string]interface{})
  84. subjectText, hasSubject := subjectData["value"]
  85. if !hasSubject {
  86. return nil, fmt.Errorf("Missing subject in Template %s", cmd.Template)
  87. }
  88. subjectTmpl, err := template.New("subject").Parse(subjectText.(string))
  89. if err != nil {
  90. return nil, err
  91. }
  92. var subjectBuffer bytes.Buffer
  93. err = subjectTmpl.ExecuteTemplate(&subjectBuffer, "subject", data)
  94. if err != nil {
  95. return nil, err
  96. }
  97. subject = subjectBuffer.String()
  98. }
  99. return &Message{
  100. To: cmd.To,
  101. From: fmt.Sprintf("%s <%s>", ns.Cfg.Smtp.FromName, ns.Cfg.Smtp.FromAddress),
  102. Subject: subject,
  103. Body: buffer.String(),
  104. EmbededFiles: cmd.EmbededFiles,
  105. }, nil
  106. }