All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.eclipse.angus.mail.imap.IMAPBodyPart Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) 1997, 2023 Oracle and/or its affiliates. All rights reserved.
 *
 * This program and the accompanying materials are made available under the
 * terms of the Eclipse Public License v. 2.0, which is available at
 * http://www.eclipse.org/legal/epl-2.0.
 *
 * This Source Code may also be made available under the following Secondary
 * Licenses when the conditions for such availability set forth in the
 * Eclipse Public License v. 2.0 are satisfied: GNU General Public License,
 * version 2 with the GNU Classpath Exception, which is available at
 * https://www.gnu.org/software/classpath/license.html.
 *
 * SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
 */

package org.eclipse.angus.mail.imap;

import jakarta.activation.DataHandler;
import jakarta.mail.FolderClosedException;
import jakarta.mail.Header;
import jakarta.mail.IllegalWriteException;
import jakarta.mail.MessagingException;
import jakarta.mail.Multipart;
import jakarta.mail.internet.ContentType;
import jakarta.mail.internet.InternetHeaders;
import jakarta.mail.internet.MimeBodyPart;
import jakarta.mail.internet.MimeUtility;
import org.eclipse.angus.mail.iap.ConnectionException;
import org.eclipse.angus.mail.iap.ProtocolException;
import org.eclipse.angus.mail.imap.protocol.BODY;
import org.eclipse.angus.mail.imap.protocol.BODYSTRUCTURE;
import org.eclipse.angus.mail.imap.protocol.IMAPProtocol;
import org.eclipse.angus.mail.util.LineOutputStream;
import org.eclipse.angus.mail.util.PropUtil;
import org.eclipse.angus.mail.util.ReadableMime;
import org.eclipse.angus.mail.util.SharedByteArrayOutputStream;

import java.io.ByteArrayInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.SequenceInputStream;
import java.io.UnsupportedEncodingException;
import java.util.Enumeration;

/**
 * An IMAP body part.
 *
 * @author John Mani
 * @author Bill Shannon
 */

public class IMAPBodyPart extends MimeBodyPart implements ReadableMime {
    private IMAPMessage message;
    private BODYSTRUCTURE bs;
    private String sectionId;

    // processed values ..
    private String type;
    private String description;

    private boolean headersLoaded = false;

    private static final boolean decodeFileName =
            PropUtil.getBooleanSystemProperty("mail.mime.decodefilename", false);

    protected IMAPBodyPart(BODYSTRUCTURE bs, String sid, IMAPMessage message) {
        super();
        this.bs = bs;
        this.sectionId = sid;
        this.message = message;
        // generate content-type
        ContentType ct = new ContentType(bs.type, bs.subtype, bs.cParams);
        type = ct.toString();
    }

    /* Override this method to make it a no-op, rather than throw
     * an IllegalWriteException. This will permit IMAPBodyParts to
     * be inserted in newly crafted MimeMessages, especially when
     * forwarding or replying to messages.
     */
    @Override
    protected void updateHeaders() {
        return;
    }

    @Override
    public int getSize() throws MessagingException {
        return bs.size;
    }

    @Override
    public int getLineCount() throws MessagingException {
        return bs.lines;
    }

    @Override
    public String getContentType() throws MessagingException {
        return type;
    }

    @Override
    public String getDisposition() throws MessagingException {
        return bs.disposition;
    }

    @Override
    public void setDisposition(String disposition) throws MessagingException {
        throw new IllegalWriteException("IMAPBodyPart is read-only");
    }

    @Override
    public String getEncoding() throws MessagingException {
        return bs.encoding;
    }

    @Override
    public String getContentID() throws MessagingException {
        return bs.id;
    }

    @Override
    public String getContentMD5() throws MessagingException {
        return bs.md5;
    }

    @Override
    public void setContentMD5(String md5) throws MessagingException {
        throw new IllegalWriteException("IMAPBodyPart is read-only");
    }

    @Override
    public String getDescription() throws MessagingException {
        if (description != null) // cached value ?
            return description;

        if (bs.description == null)
            return null;

        try {
            description = MimeUtility.decodeText(bs.description);
        } catch (UnsupportedEncodingException ex) {
            description = bs.description;
        }

        return description;
    }

    @Override
    public void setDescription(String description, String charset)
            throws MessagingException {
        throw new IllegalWriteException("IMAPBodyPart is read-only");
    }

    @Override
    public String getFileName() throws MessagingException {
        String filename = null;
        if (bs.dParams != null)
            filename = bs.dParams.get("filename");
        if ((filename == null || filename.isEmpty()) && bs.cParams != null)
            filename = bs.cParams.get("name");
        if (decodeFileName && filename != null) {
            try {
                filename = MimeUtility.decodeText(filename);
            } catch (UnsupportedEncodingException ex) {
                throw new MessagingException("Can't decode filename", ex);
            }
        }
        return filename;
    }

    @Override
    public void setFileName(String filename) throws MessagingException {
        throw new IllegalWriteException("IMAPBodyPart is read-only");
    }

    @Override
    protected InputStream getContentStream() throws MessagingException {
        InputStream is = null;
        boolean pk = message.getPeek();    // acquire outside of message cache lock

        // Acquire MessageCacheLock, to freeze seqnum.
        synchronized (message.getMessageCacheLock()) {
            try {
                IMAPProtocol p = message.getProtocol();

                // Check whether this message is expunged
                message.checkExpunged();

                if (p.isREV1() && (message.getFetchBlockSize() != -1))
                    return new IMAPInputStream(message, sectionId,
                            message.ignoreBodyStructureSize() ? -1 : bs.size, pk);

                // Else, vanila IMAP4, no partial fetch

                int seqnum = message.getSequenceNumber();
                BODY b;
                if (pk)
                    b = p.peekBody(seqnum, sectionId);
                else
                    b = p.fetchBody(seqnum, sectionId);
                if (b != null)
                    is = b.getByteArrayInputStream();
            } catch (ConnectionException cex) {
                throw new FolderClosedException(
                        message.getFolder(), cex.getMessage());
            } catch (ProtocolException pex) {
                throw new MessagingException(pex.getMessage(), pex);
            }
        }

        if (is == null) {
            message.forceCheckExpunged(); // may throw MessageRemovedException
            // nope, the server doesn't think it's expunged.
            // can't tell the difference between the server returning NIL
            // and some other error that caused null to be returned above,
            // so we'll just assume it was empty content.
            is = new ByteArrayInputStream(new byte[0]);
        }
        return is;
    }

    /**
     * Return the MIME format stream of headers for this body part.
     */
    private InputStream getHeaderStream() throws MessagingException {
        if (!message.isREV1())
            loadHeaders();    // will be needed below

        // Acquire MessageCacheLock, to freeze seqnum.
        synchronized (message.getMessageCacheLock()) {
            try {
                IMAPProtocol p = message.getProtocol();

                // Check whether this message got expunged
                message.checkExpunged();

                if (p.isREV1()) {
                    int seqnum = message.getSequenceNumber();
                    BODY b = p.peekBody(seqnum, sectionId + ".MIME");

                    if (b == null)
                        throw new MessagingException("Failed to fetch headers");

                    ByteArrayInputStream bis = b.getByteArrayInputStream();
                    if (bis == null)
                        throw new MessagingException("Failed to fetch headers");
                    return bis;

                } else {
                    // Can't read it from server, have to fake it
                    SharedByteArrayOutputStream bos =
                            new SharedByteArrayOutputStream(0);
                    LineOutputStream los = new LineOutputStream(bos);

                    try {
                        // Write out the header
                        Enumeration hdrLines
                                = super.getAllHeaderLines();
                        while (hdrLines.hasMoreElements())
                            los.writeln(hdrLines.nextElement());

                        // The CRLF separator between header and content
                        los.writeln();
                    } catch (IOException ioex) {
                        // should never happen
                    } finally {
                        try {
                            los.close();
                        } catch (IOException cex) {
                        }
                    }
                    return bos.toStream();
                }
            } catch (ConnectionException cex) {
                throw new FolderClosedException(
                        message.getFolder(), cex.getMessage());
            } catch (ProtocolException pex) {
                throw new MessagingException(pex.getMessage(), pex);
            }
        }
    }

    /**
     * Return the MIME format stream corresponding to this message part.
     *
     * @return the MIME format stream
     * @since JavaMail 1.4.5
     */
    @Override
    public InputStream getMimeStream() throws MessagingException {
        /*
         * The IMAP protocol doesn't support returning the entire
         * part content in one operation so we have to fake it by
         * concatenating the header stream and the content stream.
         */
        return new SequenceInputStream(getHeaderStream(), getContentStream());
    }

    @Override
    public synchronized DataHandler getDataHandler()
            throws MessagingException {
        if (dh == null) {
            if (bs.isMulti())
                dh = new DataHandler(
                        new IMAPMultipartDataSource(
                                this, bs.bodies, sectionId, message)
                );
            else if (bs.isNested() && message.isREV1() && bs.envelope != null)
                dh = new DataHandler(
                        new IMAPNestedMessage(message,
                                bs.bodies[0],
                                bs.envelope,
                                sectionId),
                        type
                );
        }

        return super.getDataHandler();
    }

    @Override
    public void setDataHandler(DataHandler content) throws MessagingException {
        throw new IllegalWriteException("IMAPBodyPart is read-only");
    }

    @Override
    public void setContent(Object o, String type) throws MessagingException {
        throw new IllegalWriteException("IMAPBodyPart is read-only");
    }

    @Override
    public void setContent(Multipart mp) throws MessagingException {
        throw new IllegalWriteException("IMAPBodyPart is read-only");
    }

    @Override
    public String[] getHeader(String name) throws MessagingException {
        loadHeaders();
        return super.getHeader(name);
    }

    @Override
    public void setHeader(String name, String value)
            throws MessagingException {
        throw new IllegalWriteException("IMAPBodyPart is read-only");
    }

    @Override
    public void addHeader(String name, String value)
            throws MessagingException {
        throw new IllegalWriteException("IMAPBodyPart is read-only");
    }

    @Override
    public void removeHeader(String name) throws MessagingException {
        throw new IllegalWriteException("IMAPBodyPart is read-only");
    }

    @Override
    public Enumeration
getAllHeaders() throws MessagingException { loadHeaders(); return super.getAllHeaders(); } @Override public Enumeration
getMatchingHeaders(String[] names) throws MessagingException { loadHeaders(); return super.getMatchingHeaders(names); } @Override public Enumeration
getNonMatchingHeaders(String[] names) throws MessagingException { loadHeaders(); return super.getNonMatchingHeaders(names); } @Override public void addHeaderLine(String line) throws MessagingException { throw new IllegalWriteException("IMAPBodyPart is read-only"); } @Override public Enumeration getAllHeaderLines() throws MessagingException { loadHeaders(); return super.getAllHeaderLines(); } @Override public Enumeration getMatchingHeaderLines(String[] names) throws MessagingException { loadHeaders(); return super.getMatchingHeaderLines(names); } @Override public Enumeration getNonMatchingHeaderLines(String[] names) throws MessagingException { loadHeaders(); return super.getNonMatchingHeaderLines(names); } private synchronized void loadHeaders() throws MessagingException { if (headersLoaded) return; // "headers" should never be null since it's set in the constructor. // If something did go wrong this will fix it, but is an unsynchronized // assignment of "headers". if (headers == null) headers = new InternetHeaders(); // load headers // Acquire MessageCacheLock, to freeze seqnum. synchronized (message.getMessageCacheLock()) { try { IMAPProtocol p = message.getProtocol(); // Check whether this message got expunged message.checkExpunged(); if (p.isREV1()) { int seqnum = message.getSequenceNumber(); BODY b = p.peekBody(seqnum, sectionId + ".MIME"); if (b == null) throw new MessagingException("Failed to fetch headers"); ByteArrayInputStream bis = b.getByteArrayInputStream(); if (bis == null) throw new MessagingException("Failed to fetch headers"); headers.load(bis); } else { // RFC 1730 does not provide for fetching BodyPart headers // So, just dump the RFC1730 BODYSTRUCTURE into the // headerStore // Content-Type headers.addHeader("Content-Type", type); // Content-Transfer-Encoding headers.addHeader("Content-Transfer-Encoding", bs.encoding); // Content-Description if (bs.description != null) headers.addHeader("Content-Description", bs.description); // Content-ID if (bs.id != null) headers.addHeader("Content-ID", bs.id); // Content-MD5 if (bs.md5 != null) headers.addHeader("Content-MD5", bs.md5); } } catch (ConnectionException cex) { throw new FolderClosedException( message.getFolder(), cex.getMessage()); } catch (ProtocolException pex) { throw new MessagingException(pex.getMessage(), pex); } } headersLoaded = true; } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy