1 问题描述
生产服务器发送通知邮件,之前一直都是正常的。可突然有一天业务同事反馈收不到通知邮件了。经过查看生产运行日志,发现是由于出现无效邮件地址导致的,而核心异常日志信息如下:
javax.mail.SendFailedException: Invalid Addresses
at com.sun.mail.smtp.SMTPTransport.rcptTo(SMTPTransport.java:1862) ~[mail-1.4.7.jar:1.4.7]
at com.sun.mail.smtp.SMTPTransport.sendMessage(SMTPTransport.java:1118) ~[mail-1.4.7.jar:1.4.7]
at com.crtrust.trustonline.utils.MailUtils.sendHtmlEmail(MailUtils.java:167) [classes/:?]
at com.crtrust.trustonline.utils.MailUtilsTest2.main(MailUtilsTest2.java:47) [classes/:?]
Caused by: com.sun.mail.smtp.SMTPAddressFailedException: 550 User suspended: [email protected]
at com.sun.mail.smtp.SMTPTransport.rcptTo(SMTPTransport.java:1715) ~[mail-1.4.7.jar:1.4.7]
... 3 more
Caused by: com.sun.mail.smtp.SMTPAddressFailedException: 550 User not found: [email protected]
at com.sun.mail.smtp.SMTPTransport.rcptTo(SMTPTransport.java:1715) ~[mail-1.4.7.jar:1.4.7]
... 3 more
2 问题解析与方案分析
最初我还纳闷,之前邮件通知程序一直都是在正常运行的,咋会突然莫名其妙地出现bug呢,还以为是发版上线引起了。经过分析排查定位,最终确认是由于员工离职邮箱账号被停用导致的邮箱地址无效。
我就在想不能因为账号被禁用就导致群发邮件失败啊,要不然出现一个员工离职就需要调整邮件列表配置或是重启服务,那样也太不灵活了吧,所以肯定还是需要完善一下的。
经过上网搜索及分析排查,发现在抛出的异常类SMTPAddressFailedException中包含有无效的邮箱地址,而通过方法getInvalidAddresses()即可获取到无效的邮箱地址列表,而在获取到无效的邮箱地址之后从原来的邮箱地址列表中剔除掉再重新发送一次,即可把问题完美解决掉。
题外话:其实事后再回想一下,既然异常日志中已明确指明无效的邮件地址,那就说明可以通过某种方式获取到无效的邮件地址列表,而这就应该是我分析定位的方向。可惜最初自己并没有看到这个方向,而是一心想着去网上搜解决方案,也幸亏最后找到了解决方案。在这个问题解决的过程中,我也在反思自己解决问题的技术思维和能力还非常有待进一步提升。
3 实例代码
废话不啰嗦,直接上代码。注意,代码中的邮件服务器地址和账号密码已经隐去,大家换成自己家的配置即可。
package com.demo.utils;
import com.alibaba.fastjson.JSON;
import com.sun.mail.smtp.SMTPAddressFailedException;
import org.apache.commons.lang3.StringUtils;
import javax.activation.DataHandler;
import javax.activation.DataSource;
import javax.activation.FileDataSource;
import javax.mail.*;
import javax.mail.internet.*;
import