暂无图片
暂无图片
暂无图片
暂无图片
暂无图片

一文搞定javaMail邮件发送后退信获取那些事!

小罗技术笔记 2020-02-12
1738

点击上方“小罗技术笔记”,关注公众号

第一时间送达实用干货

本文来自自己实际项目需求。

前言

邮件发送很简单网上一找一大把例子,我也是这样照着网上copy判断是否发送成功也是一个try/catch 完事。越简单的事越容易出篓子,有一天运营那边突然过来找我说用户反映没有收到资料邮件让我怎么在规定日期内提交材料理赔!涉及到赔偿了....,一顿叼说我为啥显示发送成功(程序员就是命苦),去邮件服务器一查好家伙被退信了(运营那帮家伙是不会自己去邮箱看的),邮件发送是异步的只要邮箱、文案正确不会抛异常所以try/catch 没用,然后就开始琢磨有了本文的内容。

正文

首先找到未发生成功的邮件就得读取邮箱里面的邮件信息,如何避免每次读邮箱全部读这个问题呢,下面将一一讲述。

javaMail收邮件主要有两种协议,一种是pop3,一种是imap。这两种协议都可以用来收邮件,但是在其中的处理上是有区别的。pop3是不支持判断邮件是否为已读的,也就是说你不能直接从收件箱里面取到未读邮件,这需要自己进行判断,然而imap就提供了这样的功能,使用imap时可以很轻松的判断该邮件是否为已读或未读或其他。

此外收件箱中的每一封邮件都对应着一个MessageNumber,所以可以通过一个MessageNumber拿到对应的那封邮件。如:Session.getInstanceMessage message = folder.getMessage(messageNumber),这样定时任务每次读邮箱邮件不用全量读了。

一、imap协议模式下读取

  1.     public static void readMail() {    

  2.           String protocol = "imap";

  3.           Properties props = new Properties();

  4.           props.setProperty("mail.transport.protocol",  protocol);

  5.            props.setProperty("mail.smtp.host", host);

  6.           // 获取连接  getInstance避免和发件发送session冲突

  7.            Session session = Session.getInstance(props);

  8.           session.setDebug(false);

  9.           // 获取Store对象

  10.            Store store = session.getStore(protocol);

  11.            store.connect(host, userName, password);

  12.            IMAPFolder inbox = (IMAPFolder)store.getFolder("INBOX");    

  13.             //如果需要在取得邮件数后将邮件置为已读则这里需要使用READ_WRITE,否则READ_ONLY就可以

  14.            inbox.open(Folder.READ_WRITE);    

  15.            // 全部邮件数    

  16.            int messageCount = inbox.getMessageCount();    

  17.            System.out.println(messageCount);  

  18.            //folder.getUnreadMessageCount() 未读取邮件总数

  19.            Message[] messages = inbox.getMessages(folder.getMessageCount()-inbox.getUnreadMessageCount()+1,inbox.getMessageCount());

  20.            List<Message> mesList=new ArrayList<>();

  21.            for (int i = 0; i < messages.length; i++) {    

  22.                Message message = messages[i];    

  23.                System.out.println(message.getSubject());    

  24.                //保存我将要设置为已读的message列表    

  25.                mesList.add(message);

  26.                 //解析邮件内容(网上很多我就不一一贴代码了)

  27.               .......

  28.             }

  29.              //将刚才我获取的邮件设置为已读

  30.             if (mesList.size() > 0) {

  31.                 Message[] savedMailMessage = new Message[mesList.size()];

  32.                 mesList.toArray(savedMailMessage);

  33.                 inbox.setFlags(savedMailMessage, new Flags(Flags.Flag.SEEN),true);

  34.             }

  35.            folder.close(true);    

  36.            store.close();    

  37.        }    

二、pop3协议模式下读取

由于pop3是不支持判断邮件是否为已读的,不能直接从收件箱里面取到未读邮件所以无法获取未读邮箱数量,但是支持inbox.getMessages(start,end)方式获取邮件区间,因此我采用redis存储已读邮件位置和每批次读取邮件数量来实现!

  1. public  Integer readMail(Integer readNum,Integer size) {

  2.        String protocol = "pop3";

  3.        Properties props = new Properties();

  4.        props.setProperty("mail.transport.protocol",  protocol);

  5.        props.setProperty("mail.smtp.host", host);

  6.        // 获取连接

  7.        Session session = Session.getInstance(props);

  8.        session.setDebug(false);

  9.        // 获取Store对象

  10.        Store store = session.getStore(protocol);

  11.        store.connect(host, userName, password);

  12.        Folder folder = store.getFolder("INBOX");

  13.        folder.open(Folder.READ_WRITE);

  14.        int mailTotalNum=folder.getMessageCount();

  15.        log.info("定时器读取邮件总数: {}", mailTotalNum);

  16.        Integer endNum;

  17.        if(readNum<=mailTotalNum) {

  18.        //我在读取redis中的readNum+1了所以这里要-1

  19.             endNum = readNum + size - 1 > mailTotalNum ? mailTotalNum : readNum + size - 1;

  20.            log.info("读取邮件区间:{}-{}", readNum, endNum);

  21.            Message[] messages = folder.getMessages(readNum, endNum);

  22.           //解析邮件内容(网上很多我就不一一贴代码了)

  23.          ...........

  24.        }else {

  25.            endNum=readNum-1;

  26.        }

  27.        folder.close(false);

  28.        store.close();

  29.        //返回本次读取位置  redis 存储当前读取到多少条

  30.        return endNum;

  31. }

三、如何判断发送失败的邮件

由于查询了好久没有方法可以直接判断邮件发送状态所以我只能用我所遇到的退信格式来判断了,如下图(公司生产邮箱内容所以打码了,不同收件邮箱退信格式不同,渣浪的邮箱投递不成功没有退信)

根据上图可以发现退信的邮件主题和一般邮件不一样所以我采用如下代码判断(很挫)

  1. String subject = getSubject(msg);

  2. if (subject.startsWith("未送达")|| subject.startsWith("系统退 信")) {

  3. }

找到发送失败的邮箱然后再提取邮件内容中的收件邮箱、发送时间等去更新数据库中邮件的发送状态。

到此成功获取邮件发送后状态了!

前俩天功能上线,能用,再也不用挨运营的叼了!

本文只是以我自己解决的方法来写的,如果有更好的方法可以联系指出来,感激不尽!



长按二维码关注

点个在看再走呗!

文章转载自小罗技术笔记,如果涉嫌侵权,请发送邮件至:contact@modb.pro进行举报,并提供相关证据,一经查实,墨天轮将立刻删除相关内容。

评论