使用Java Mail下载附件


96

现在,我已经下载了所有消息,并将它们存储到

Message[] temp;

我如何获取每个邮件的附件列表

List<File> attachments;

注意:请没有第三方库,只有JavaMail。

Answers:


109

没有异常处理,但这里有:

List<File> attachments = new ArrayList<File>();
for (Message message : temp) {
    Multipart multipart = (Multipart) message.getContent();

    for (int i = 0; i < multipart.getCount(); i++) {
        BodyPart bodyPart = multipart.getBodyPart(i);
        if(!Part.ATTACHMENT.equalsIgnoreCase(bodyPart.getDisposition()) &&
               StringUtils.isBlank(bodyPart.getFileName())) {
            continue; // dealing with attachments only
        } 
        InputStream is = bodyPart.getInputStream();
        // -- EDIT -- SECURITY ISSUE --
        // do not do this in production code -- a malicious email can easily contain this filename: "../etc/passwd", or any other path: They can overwrite _ANY_ file on the system that this code has write access to!
//      File f = new File("/tmp/" + bodyPart.getFileName());
        FileOutputStream fos = new FileOutputStream(f);
        byte[] buf = new byte[4096];
        int bytesRead;
        while((bytesRead = is.read(buf))!=-1) {
            fos.write(buf, 0, bytesRead);
        }
        fos.close();
        attachments.add(f);
    }
}

2
但是,请稍等一下,我们是否应该在保存文件之前检查(bodyPart.getDisposition()== Part.ATTACHMENT){},以便不保存电子邮件的正文?
folone

8
与使用!StringUtils.isNotBlank相比,StringUtils.isBlank()会更自然地读取吗?
Kuchi 2014年

此答案不考虑嵌套的多部分附件(例如,通常由Thunderbird使用)。为了能够找到嵌套的多部分附件,请参阅@mefi答案。
Ruslan Stelmachenko '18

3
该代码段是一个安全漏洞,正在等待发生。我已经编辑了片段以突出显示此内容。
rzwitserloot

1
当然,感谢您。此代码片段只是为了演示如何完成,自然不是生产代码
David Rabinowitz

33

问题很老,但也许会帮助别人。我想扩大David Rabinowitz的回答。

if(!Part.ATTACHMENT.equalsIgnoreCase(bodyPart.getDisposition()))

不应按您期望的那样返回所有附件,因为您可以在混合部分没有定义的处置的情况下收到邮件。

   ----boundary_328630_1e15ac03-e817-4763-af99-d4b23cfdb600
Content-Type: application/octet-stream;
    name="00000000009661222736_236225959_20130731-7.txt"
Content-Transfer-Encoding: base64

因此,在这种情况下,您还可以检查文件名。像这样:

if (!Part.ATTACHMENT.equalsIgnoreCase(part.getDisposition()) && StringUtils.isBlank(part.getFileName())) {...}

编辑

因为上面的条件描述了整个工作代码。由于每个部分都可以封装另一个部分,并且应该嵌套附件,因此使用递归遍历所有部分

public List<InputStream> getAttachments(Message message) throws Exception {
    Object content = message.getContent();
    if (content instanceof String)
        return null;        

    if (content instanceof Multipart) {
        Multipart multipart = (Multipart) content;
        List<InputStream> result = new ArrayList<InputStream>();

        for (int i = 0; i < multipart.getCount(); i++) {
            result.addAll(getAttachments(multipart.getBodyPart(i)));
        }
        return result;

    }
    return null;
}

private List<InputStream> getAttachments(BodyPart part) throws Exception {
    List<InputStream> result = new ArrayList<InputStream>();
    Object content = part.getContent();
    if (content instanceof InputStream || content instanceof String) {
        if (Part.ATTACHMENT.equalsIgnoreCase(part.getDisposition()) || StringUtils.isNotBlank(part.getFileName())) {
            result.add(part.getInputStream());
            return result;
        } else {
            return new ArrayList<InputStream>();
        }
    }

    if (content instanceof Multipart) {
            Multipart multipart = (Multipart) content;
            for (int i = 0; i < multipart.getCount(); i++) {
                BodyPart bodyPart = multipart.getBodyPart(i);
                result.addAll(getAttachments(bodyPart));
            }
    }
    return result;
}

表达式检查文件名是否为空或null。那是对的吗?
Keerthivasan 2014年

章鱼:是的。检查CharSequence是否不为空(“”),不为null和仅空白。
mefi 2014年

嘿,您能告诉我们如何将List <InputStream>转换为List <File>吗?
kumuda

kumunda:嗨,您应该迭代该列表并使用org.apache.commons.io.FileUtils.copyInputStreamToFile(InputStream源,File目标),并将每个添加文件添加到另一个集合中。如果使用Guava,请检查Lists.transform(...),您可以使用instad进行迭代(取决于初始化每个File实例的方式)
mefi 2015年

9

保存附件文件的代码节省了一些时间:

使用Javax Mail 1.4及更高版本,您可以说

// SECURITY LEAK - do not do this! Do not trust the 'getFileName' input. Imagine it is: "../etc/passwd", for example.
// bodyPart.saveFile("/tmp/" + bodyPart.getFileName());

代替

    InputStream is = bodyPart.getInputStream();
    File f = new File("/tmp/" + bodyPart.getFileName());
    FileOutputStream fos = new FileOutputStream(f);
    byte[] buf = new byte[4096];
    int bytesRead;
    while((bytesRead = is.read(buf))!=-1) {
        fos.write(buf, 0, bytesRead);
    }
    fos.close();

3
显然 bodyPart应该先转换为MimeBodyPart,例如:((MimeBodyPart) bodyPart).saveFile("/tmp/" + bodyPart.getFileName());
yair

5

您可以将Commons IO和Commons Lang一起使用Apache Commons Mail API MimeMessageParser-getAttachmentList()

MimeMessageParser parser = ....
parser.parse();
for(DataSource dataSource : parser.getAttachmentList()) {

    if (StringUtils.isNotBlank(dataSource.getName())) {}

        //use apache commons IOUtils to save attachments
        IOUtils.copy(dataSource.getInputStream(), ..dataSource.getName()...)
    } else {
        //handle how you would want attachments without file names
        //ex. mails within emails have no file name
    }
}
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.