使用JavaMail读取退信
在工作中遇到了一种场景,当我们向用户发送邮件后,偶尔会有一些用户投诉,未收到邮件,对于这种情况,我们需要一种方式来监控是否我们真的没有发送成功。我们使用的发件箱是outlook,当我们发送一个邮件到不存在的邮箱时,outlook会返回一个退信邮件,告知我们邮件并未送达,因此我们的解决方案就是解析该类邮件,并且对应到我们的发送记录中,话不多说直接上代码。
public class ReadMailTest {
private static final String multipart = "multipart/*";
private static final String userName = ""; //待读取的邮箱账户
private static final String password = ""; //待读取的邮箱密码
private static Properties buildInboxProperties() {
Properties props = System.getProperties();
props.setProperty("mail.imap.host", "imap.partner.outlook.cn");
props.setProperty("mail.imap.socketFactory.class", "javax.net.ssl.SSLSocketFactory");
props.setProperty("mail.imap.socketFactory.fallback", "false");
props.setProperty("mail.imap.port", "993");
props.setProperty("mail.imap.socketFactory.port", "993");
props.setProperty("mail.imap.auth", "true");
return props;
}
public static void searchInboxEmail() {
Session session = Session.getInstance(buildInboxProperties());
URLName url = new URLName("imap", "imap.partner.outlook.cn", 993, null, userName, password);
try(Store store = session.getStore(url)) {
store.connect();
try (Folder receiveFolder = store.getFolder("inbox")) {
receiveFolder.open(Folder.READ_ONLY);
int messageCount = receiveFolder.getMessageCount();
if (messageCount > 0) {
//添加日期条件
Date beforeDate = Date.from(LocalDateTime.now().minusHours(3).atZone(ZoneId.systemDefault()).toInstant());
SearchTerm comparisonTermGe = new SentDateTerm(ComparisonTerm.GE, beforeDate);
//添加内容条件,内容中存在Original message headers的邮件为退信邮件
BodyTerm bodyTerm = new BodyTerm("Original message headers");
SearchTerm search = new AndTerm(comparisonTermGe, bodyTerm);
// 调用搜索api
Message[] messages = receiveFolder.search(search);
if (messages.length == 0) {
return;
}
/* 由于Javamail不支持按照时间进行过滤,只支持到日期,因此如果需要精确到时间过滤
需要在获取到搜索结果后再过滤一次*/
List<Message> messageList = Stream.of(messages)
.filter(message -> {
try {
return !message.getSentDate().before(beforeDate);
} catch (MessagingException e) {
e.printStackTrace();
}
return false;
})
.collect(Collectors.toList());
messageHandler(messageList);
System.out.println(messageList.size());
}
}
} catch (MessagingException e) {
e.printStackTrace();
}
}
public static void messageHandler(List<Message> messages) {
messages.stream()
.filter(ReadMailTest::isContainAttachment)
.forEach(message -> {
String messageId = null;
try {
messageId = getMessageId(message);
} catch (Exception e) {
e.printStackTrace();
}
try {
System.out.println("subject=" + message.getSubject() + ",messageId=" + messageId + ",sendDate=" + message.getSentDate());
} catch (MessagingException e) {
e.printStackTrace();
}
});
}
public static String getMessageId(Part part) throws Exception {
if (!part.isMimeType(multipart)) {
return "";
}
Multipart multipart = (Multipart) part.getContent();
for (int i = 0; i < multipart.getCount(); i++) {
BodyPart bodyPart = multipart.getBodyPart(i);
if (part.isMimeType("message/rfc822")) {
return getMessageId((Part) part.getContent());
}
InputStream inputStream = bodyPart.getInputStream();
try (BufferedReader br = new BufferedReader(new InputStreamReader(inputStream))) {
String strLine;
while ((strLine = br.readLine()) != null) {
if (strLine.startsWith("Message-ID:")) {
String[] split = strLine.split("Message-ID:");
return split.length > 1 ? split[1].trim() : null;
}
}
}
}
return "";
}
public static boolean isContainAttachment(Part part) {
boolean attachFlag = false;
try {
if (part.isMimeType(multipart)) {
Multipart mp = (Multipart) part.getContent();
for (int i = 0; i < mp.getCount(); i++) {
BodyPart mpart = mp.getBodyPart(i);
String disposition = mpart.getDisposition();
if ((disposition != null) && ((disposition.equals(Part.ATTACHMENT)) || (disposition.equals(Part.INLINE))))
attachFlag = true;
else if (mpart.isMimeType(multipart)) {
attachFlag = isContainAttachment(mpart);
} else {
String contype = mpart.getContentType();
if (contype.toLowerCase().contains("application"))
attachFlag = true;
if (contype.toLowerCase().contains("name"))
attachFlag = true;
}
}
} else if (part.isMimeType("message/rfc822")) {
attachFlag = isContainAttachment((Part) part.getContent());
}
} catch (MessagingException | IOException e) {
e.printStackTrace();
}
return attachFlag;
}
}
上面的代码是读取退信邮件,根据过滤条件,也可以读取其他自己想要读取的邮件