Spring无法发送smtp邮件:mail com.sun.mail.smtp.SMTPSendFailedException: 530 5.7.0 必须先发出STARTTLS命令。

5
我正在尝试从我的使用jhipster 1.1.0生成的spring应用程序中发送邮件,我的spring-boot版本是1.5.2。
我有以下代码:
Application-dev.yml:
spring:
  mail:
    host: smtp.gmail.com
    port: 587
    username: myuser@gmail.com
    password: mypass
    protocol: smtp
    tls: true
  properties.mail.smtp:
    auth: true
    starttls.enable: true
    #ssl.trust: smtp.gmail.com
    starttls.required: true
jhipster:
  mail: # specific JHipster mail property, for standard properties see MailProperties
    from: myuser@gmail.com  
    base-url: http://127.0.0.1:8080

MailService.java

package com.pfg.easyschedule.service;

import com.pfg.easyschedule.domain.User;

import io.github.jhipster.config.JHipsterProperties;

import org.apache.commons.lang3.CharEncoding;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.boot.autoconfigure.mail.MailProperties;
import org.springframework.context.MessageSource;
import org.springframework.mail.javamail.JavaMailSender;
import org.springframework.mail.javamail.MimeMessageHelper;
import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Service;
import org.thymeleaf.context.Context;
import org.thymeleaf.spring4.SpringTemplateEngine;

import javax.inject.Inject;
import javax.mail.internet.MimeMessage;
import java.util.Locale;
import java.util.Map;

/**
 * Service for sending e-mails.
 * <p>
 * We use the @Async annotation to send e-mails asynchronously.
 * </p>
 */
@Service
public class MailService {

    private final Logger log = LoggerFactory.getLogger(MailService.class);
    private static final String USER = "user";
    private static final String BASE_URL = "baseUrl";
    private final JHipsterProperties jHipsterProperties;
    private final JavaMailSender javaMailSender;
    private final MessageSource messageSource;
    private final SpringTemplateEngine templateEngine;


    public MailService(JHipsterProperties jHipsterProperties, JavaMailSender javaMailSender,
            MessageSource messageSource, SpringTemplateEngine templateEngine) {

        this.jHipsterProperties = jHipsterProperties;
        this.javaMailSender = javaMailSender;
        this.messageSource = messageSource;
        this.templateEngine = templateEngine;
    }

    @Async
    public void sendEmail(String to, String subject, String content, boolean isMultipart, boolean isHtml) {
        log.debug("Send e-mail[multipart '{}' and html '{}'] to '{}' with subject '{}' and content={}",
            isMultipart, isHtml, to, subject, content);

        // Prepare message using a Spring helper
        MimeMessage mimeMessage = javaMailSender.createMimeMessage();
        try {
            MimeMessageHelper message = new MimeMessageHelper(mimeMessage, isMultipart, CharEncoding.UTF_8);
            message.setTo(to);
            message.setFrom(jHipsterProperties.getMail().getFrom());
            message.setSubject(subject);
            message.setText(content, isHtml);
            javaMailSender.send(mimeMessage);
            log.debug("Sent e-mail to User '{}'", to);
        } catch (Exception e) {
            log.warn("E-mail could not be sent to user '{}'", to, e);
        }
    }

    @Async
    public void sendActivationEmail(User user) {
        log.debug("Sending activation e-mail to '{}'", user.getEmail());
        Locale locale = Locale.forLanguageTag(user.getLangKey());
        Context context = new Context(locale);
        context.setVariable(USER, user);
        context.setVariable(BASE_URL, jHipsterProperties.getMail().getBaseUrl());
        String content = templateEngine.process("activationEmail", context);
        String subject = messageSource.getMessage("email.activation.title", null, locale);
        sendEmail(user.getEmail(), subject, content, false, true);
    }

    @Async
    public void sendCreationEmail(User user) {
        log.debug("Sending creation e-mail to '{}'", user.getEmail());
        Locale locale = Locale.forLanguageTag(user.getLangKey());
        Context context = new Context(locale);
        context.setVariable(USER, user);
        context.setVariable(BASE_URL, jHipsterProperties.getMail().getBaseUrl());
        String content = templateEngine.process("creationEmail", context);
        String subject = messageSource.getMessage("email.activation.title", null, locale);
        sendEmail(user.getEmail(), subject, content, false, true);
    }

    @Async
    public void sendPasswordResetMail(User user) {
        log.debug("Sending password reset e-mail to '{}'", user.getEmail());
        Locale locale = Locale.forLanguageTag(user.getLangKey());
        Context context = new Context(locale);
        context.setVariable(USER, user);
        context.setVariable(BASE_URL, jHipsterProperties.getMail().getBaseUrl());
        String content = templateEngine.process("passwordResetEmail", context);
        String subject = messageSource.getMessage("email.reset.title", null, locale);
        sendEmail(user.getEmail(), subject, content, false, true);
    }
}

最终我使用这个控制器来发送邮件。

Controller.java

@RestController

@RequestMapping("/api")表示这个类是一个REST API的控制器,它会处理来自"/api"路径的HTTP请求。这个类名为MyClassResource。
private final Logger log = 
LoggerFactory.getLogger(MyClassResource.class);

@Autowired
EntityManager entityManager;
@Autowired
MailService mailService;
@PostMapping ("/myurl")
@Timed
public ResponseEntity<Boolean> mymethod(@RequestBody myObject data) {
mailService.sendEmail(
            "emailTo@gmail.com",
            "message from spring",
            "hi world. this message is from spring",
            true,
            true
        );
}

最后我得到了这个错误:
c....MailService   : E-mail could not be sent to user 'emailTo@gmail.com'

org.springframework.mail.MailSendException: Failed messages: com.sun.mail.smtp.SMTPSendFailedException: 530 5.7.0 Must issue a STARTTLS command first. w125sm9354452wmb.45 - gsmtp

at org.springframework.mail.javamail.JavaMailSenderImpl.doSend(JavaMailSenderImpl.java:474)
at org.springframework.mail.javamail.JavaMailSenderImpl.send(JavaMailSenderImpl.java:345)
at org.springframework.mail.javamail.JavaMailSenderImpl.send(JavaMailSenderImpl.java:340)
.........

我一直在阅读关于这个错误的内容,并像application-dev.yml中一样添加了一些配置,但这个方法不起作用。我试着从Gmail获取一个应用程序密码,因为我读到说这是必要的,但由于我没有进行双步验证,所以我无法获取该密码。最后,我改变了接受来自较不安全应用程序的连接,但仍然不起作用。我不知道为什么它不起作用。

你的 application.yml 文件中有很多内容,但如果你查看文档,键应该是 spring.mail.* - chrylis -cautiouslyoptimistic-
我忘记在邮件标签之前写Spring标签了。我已经更新了application-dev.yml文件。 - Kara
properties.mail.smtp 没有正确缩进,必须嵌套在邮件下面,就像官方文档中所示:https://www.jhipster.tech/tips/011_tip_configuring_email_in_jhipster.html。而且 JHipster 版本很不可能是 1.1.0,因为那已经是 5 年前的版本,也不使用 Spring Boot 1.5。 - Gaël Marziou
尝试设置以下属性:EnableSSL.enable: truesocketFactory.class: "javax.net.ssl.SSLSocketFactory"socketFactory.fallback: falsesocketFactory.port: 587 - user10098815
1个回答

6
感谢大家的帮助。
最终我采用了 @Gaël 的建议,现在它可以工作了。这是我的新配置:
Application-dev.yml:
mail:
    host: smtp.gmail.com
    port: 587
    username: myuser@gmail.com
    password: mypass        
    protocol: smtp
    tls: true
    properties.mail.smtp:
        auth: true
        starttls.enable: true
        ssl.trust: smtp.gmail.com
jhipster:
    mail: # specific JHipster mail property, for standard properties see MailProperties
    from: myuser@gmail.com
    base-url: http://127.0.0.1:8080

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接