Java Mail을 사용하여 첨부 파일 다운로드


96

이제 모든 메시지를 다운로드하여

Message[] temp;

각 메시지에 대한 첨부 파일 목록을 가져 오는 방법

List<File> attachments;

참고 : 타사 라이브러리는 없습니다. JavaMail 만 사용하십시오.

답변:


110

예외 처리가 없지만 여기에 있습니다.

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 2010

8
StringUtils.isBlank ()가! StringUtils.isNotBlank를 사용하는 것보다 읽기가 더 자연스럽지 않습니까?
Kuchi

이 답변은 중첩 된 다중 부분 첨부 파일 (예 : Thunderbird에서 일반적으로 사용됨)을 고려하지 않습니다. 중첩 된 멀티 파트 첨부 파일을 찾으려면 @mefi 답변을 참조하십시오.
Ruslan Stelmachenko

3
스 니펫은 발생하기를 기다리는 보안 위반입니다. 이것을 강조하기 위해 스 니펫을 편집했습니다.
rzwitserloot

1
물론 감사합니다. 이 조각은 자연스럽게 프로덕션 코드가 아닙니다, 그것을 할 수있는 방법을 보여줍니다 만했다
데이비드 라 비노 위츠

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

List <InputStream>을 List <File>로 변환하는 방법을 알 수 있습니까?
kumuda

kumunda : 안녕하세요, 당신은 그 목록을 반복하고 org.apache.commons.io.FileUtils.copyInputStreamToFile (InputStream source, File destination)을 사용하고 각각 다른 컬렉션에 파일을 추가해야합니다. Guava를 사용하는 경우 Lists.transform (...)을 확인하십시오. 반복을 사용할 수 있습니다 (각 File 인스턴스를 초기화하는 방법에 따라 다름)
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.