Java Mailを使用して添付ファイルをダウンロードする


96

すべてのメッセージをダウンロードして保存しました

Message[] temp;

これらの各メッセージの添付ファイルのリストを取得する方法

List<File> attachments;

注:サードパーティのライブラリはありません。JavaMailのみです。

回答:


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){}であるかどうかを確認して、メールの本文が保存されないようにする必要がありますか?
フォロン2009年

8
StringUtils.isBlank()は、!StringUtils.isNotBlankを使用するよりも自然に読み取れるでしょうか?
クチ

この回答では、ネストされたマルチパートアタッチメントは考慮されていません(Thunderbirdなどで一般的に使用されています)。ネストされたマルチパート添付ファイルを見つけることができるようにするには、@ mefi回答を参照してください。
Ruslan Stelmachenko

3
スニペットは、発生するのを待っているセキュリティ違反です。これを強調するためにスニペットを編集しました。
rzwitserloot

1
もちろん、ありがとうございます。このスニペットは、当然、それは生産コードではありません、それを行うことができる方法を実証するだけだった
デビッドRabinowitz

33

質問は非常に古いですが、多分それは誰かを助けるでしょう。デビッド・ラビノウィッツの答えを拡大したいと思います。

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 2015年

kumunda:こんにちは。リストを繰り返し、org.apache.commons.io.FileUtils.copyInputStreamToFile(InputStream source、File destination)を使用して、それぞれを別のコレクションにファイルを追加する必要があります。あなたはグアバを使用する場合は、繰り返しのinstadを使用することができますLists.transform(...)を、確認してください(各ファイルのインスタンスを初期化する必要があるかによって異なります)
mefi

9

添付ファイルを保存するコードの時間節約:

javaxメールバージョン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
    }
}
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.