email.go 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104
  1. package common
  2. import (
  3. "crypto/tls"
  4. "encoding/base64"
  5. "fmt"
  6. "net/smtp"
  7. "slices"
  8. "strings"
  9. "time"
  10. )
  11. func generateMessageID() (string, error) {
  12. split := strings.Split(SMTPFrom, "@")
  13. if len(split) < 2 {
  14. return "", fmt.Errorf("invalid SMTP account")
  15. }
  16. domain := strings.Split(SMTPFrom, "@")[1]
  17. return fmt.Sprintf("<%d.%s@%s>", time.Now().UnixNano(), GetRandomString(12), domain), nil
  18. }
  19. func shouldUseSMTPLoginAuth() bool {
  20. if SMTPForceAuthLogin {
  21. return true
  22. }
  23. return isOutlookServer(SMTPAccount) || slices.Contains(EmailLoginAuthServerList, SMTPServer)
  24. }
  25. func getSMTPAuth() smtp.Auth {
  26. if shouldUseSMTPLoginAuth() {
  27. return LoginAuth(SMTPAccount, SMTPToken)
  28. }
  29. return smtp.PlainAuth("", SMTPAccount, SMTPToken, SMTPServer)
  30. }
  31. func SendEmail(subject string, receiver string, content string) error {
  32. if SMTPFrom == "" { // for compatibility
  33. SMTPFrom = SMTPAccount
  34. }
  35. id, err2 := generateMessageID()
  36. if err2 != nil {
  37. return err2
  38. }
  39. if SMTPServer == "" && SMTPAccount == "" {
  40. return fmt.Errorf("SMTP 服务器未配置")
  41. }
  42. encodedSubject := fmt.Sprintf("=?UTF-8?B?%s?=", base64.StdEncoding.EncodeToString([]byte(subject)))
  43. mail := []byte(fmt.Sprintf("To: %s\r\n"+
  44. "From: %s <%s>\r\n"+
  45. "Subject: %s\r\n"+
  46. "Date: %s\r\n"+
  47. "Message-ID: %s\r\n"+ // 添加 Message-ID 头
  48. "Content-Type: text/html; charset=UTF-8\r\n\r\n%s\r\n",
  49. receiver, SystemName, SMTPFrom, encodedSubject, time.Now().Format(time.RFC1123Z), id, content))
  50. auth := getSMTPAuth()
  51. addr := fmt.Sprintf("%s:%d", SMTPServer, SMTPPort)
  52. to := strings.Split(receiver, ";")
  53. var err error
  54. if SMTPPort == 465 || SMTPSSLEnabled {
  55. tlsConfig := &tls.Config{
  56. InsecureSkipVerify: true,
  57. ServerName: SMTPServer,
  58. }
  59. conn, err := tls.Dial("tcp", fmt.Sprintf("%s:%d", SMTPServer, SMTPPort), tlsConfig)
  60. if err != nil {
  61. return err
  62. }
  63. client, err := smtp.NewClient(conn, SMTPServer)
  64. if err != nil {
  65. return err
  66. }
  67. defer client.Close()
  68. if err = client.Auth(auth); err != nil {
  69. return err
  70. }
  71. if err = client.Mail(SMTPFrom); err != nil {
  72. return err
  73. }
  74. receiverEmails := strings.Split(receiver, ";")
  75. for _, receiver := range receiverEmails {
  76. if err = client.Rcpt(receiver); err != nil {
  77. return err
  78. }
  79. }
  80. w, err := client.Data()
  81. if err != nil {
  82. return err
  83. }
  84. _, err = w.Write(mail)
  85. if err != nil {
  86. return err
  87. }
  88. err = w.Close()
  89. if err != nil {
  90. return err
  91. }
  92. } else {
  93. err = smtp.SendMail(addr, auth, SMTPFrom, to, mail)
  94. }
  95. if err != nil {
  96. SysError(fmt.Sprintf("failed to send email to %s: %v", receiver, err))
  97. }
  98. return err
  99. }