org.rssowl.core.connection.HttpConnectionInputStream.java Source code

Java tutorial

Introduction

Here is the source code for org.rssowl.core.connection.HttpConnectionInputStream.java

Source

/*   **********************************************************************  **
 **   Copyright notice                                                       **
 **                                                                          **
 **   (c) 2005-2009 RSSOwl Development Team                                  **
 **   http://www.rssowl.org/                                                 **
 **                                                                          **
 **   All rights reserved                                                    **
 **                                                                          **
 **   This program and the accompanying materials are made available under   **
 **   the terms of the Eclipse Public License v1.0 which accompanies this    **
 **   distribution, and is available at:                                     **
 **   http://www.rssowl.org/legal/epl-v10.html                               **
 **                                                                          **
 **   A copy is found in the file epl-v10.html and important notices to the  **
 **   license from the team is found in the textfile LICENSE.txt distributed **
 **   in this package.                                                       **
 **                                                                          **
 **   This copyright notice MUST APPEAR in all copies of the file!           **
 **                                                                          **
 **   Contributors:                                                          **
 **     RSSOwl Development Team - initial API and implementation             **
 **                                                                          **
 **  **********************************************************************  */

package org.rssowl.core.connection;

import org.apache.commons.httpclient.Header;
import org.apache.commons.httpclient.HttpMethodBase;
import org.apache.commons.httpclient.URIException;
import org.eclipse.core.runtime.IProgressMonitor;

import java.io.FilterInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.URI;
import java.net.URISyntaxException;

/**
 * <p>
 * This kind of FilterInputStream makes sure that the HTTP Method responsible
 * for the given InputStream is releasing its connection after the stream has
 * been closed. This class is also keeping two important headers to be used for
 * the Conditional GET Mechanism of HTTP.
 * </p>
 * <p>
 * Passing an instance of <code>IProgressMonitor</code> into the class allows
 * for early cancelation by throwing an Exception from the various
 * stream-methods as soon as the monitor is canceled.
 * </p>
 *
 * @author bpasero
 */
public class HttpConnectionInputStream extends FilterInputStream implements IConditionalGetCompatible, IAbortable {

    /* Response Header */
    private static final String HEADER_RESPONSE_ETAG = "ETag"; //$NON-NLS-1$
    private static final String HEADER_RESPONSE_LAST_MODIFIED = "Last-Modified"; //$NON-NLS-1$
    private static final String HEADER_RESPONSE_CONTENT_LENGTH = "Content-Length"; //$NON-NLS-1$
    private static final String HEADER_RESPONSE_CONTENT_TYPE = "Content-Type"; //$NON-NLS-1$
    private static final String HEADER_RESPONSE_CONTENT_DISPOSITION = "Content-Disposition"; //$NON-NLS-1$
    private static final String HEADER_RESPONSE_CONTENT_ENCODING = "Content-Encoding"; //$NON-NLS-1$

    private final HttpMethodBase fMethod;
    private final IProgressMonitor fMonitor;
    private String fIfModifiedSince;
    private String fIfNoneMatch;
    private final URI fLink;

    /**
     * Creates a <code>HttpConnectionInputStream</code> by assigning the argument
     * <code>inS</code> to the field <code>this.in</code> so as to remember it for
     * later use.
     *
     * @param link the {@link URI} that was used to create the Stream.
     * @param method The Method holding the connection of the given Stream.
     * @param monitor A ProgressMonitor to support early cancelation, or
     * <code>NULL</code> if no monitor is being used.
     * @param inS the underlying input Stream.
     */
    public HttpConnectionInputStream(URI link, HttpMethodBase method, IProgressMonitor monitor, InputStream inS) {
        super(inS);
        fLink = link;
        fMethod = method;
        fMonitor = monitor;

        /* Keep some important Headers */
        Header headerLastModified = method.getResponseHeader(HEADER_RESPONSE_LAST_MODIFIED);
        if (headerLastModified != null)
            setIfModifiedSince(headerLastModified.getValue());

        Header headerETag = method.getResponseHeader(HEADER_RESPONSE_ETAG);
        if (headerETag != null)
            setIfNoneMatch(headerETag.getValue());
    }

    /**
     * @return the actual {@link URI} used to open this stream.
     */
    public URI getLink() {
        try {
            return new URI(fMethod.getURI().toString());
        } catch (URIException e) {
            return fLink;
        } catch (URISyntaxException e) {
            return fLink;
        }
    }

    /*
     * @seeorg.rssowl.core.connection.internal.http.IConditionalGetCompatible#
     * getIfModifiedSince()
     */
    public String getIfModifiedSince() {
        return fIfModifiedSince;
    }

    /*
     * @seeorg.rssowl.core.connection.internal.http.IConditionalGetCompatible#
     * getIfNoneMatch()
     */
    public String getIfNoneMatch() {
        return fIfNoneMatch;
    }

    /*
     * @seeorg.rssowl.core.connection.internal.http.IConditionalGetCompatible#
     * setIfModifiedSince(java.lang.String)
     */
    public void setIfModifiedSince(String ifModifiedSince) {
        fIfModifiedSince = ifModifiedSince;
    }

    /*
     * @seeorg.rssowl.core.connection.internal.http.IConditionalGetCompatible#
     * setIfNoneMatch(java.lang.String)
     */
    public void setIfNoneMatch(String ifNoneMatch) {
        fIfNoneMatch = ifNoneMatch;
    }

    /*
     * @see org.rssowl.core.connection.IAbortable#abort()
     */
    public void abort() {
        fMethod.abort();
        fMethod.releaseConnection();
    }

    /*
     * @see java.io.FilterInputStream#close()
     */
    @Override
    public void close() throws IOException {
        super.close();
        fMethod.releaseConnection();
    }

    /*
     * @see java.io.FilterInputStream#read()
     */
    @Override
    public int read() throws IOException {

        /* Support early cancelation */
        if (fMonitor != null && fMonitor.isCanceled())
            throw new MonitorCanceledException(Messages.HttpConnectionInputStream_ERROR_CONNECTION_CANCELED);

        return super.read();
    }

    /*
     * @see java.io.FilterInputStream#read(byte[], int, int)
     */
    @Override
    public int read(byte[] b, int off, int len) throws IOException {

        /* Support early cancelation */
        if (fMonitor != null && fMonitor.isCanceled())
            throw new MonitorCanceledException(Messages.HttpConnectionInputStream_ERROR_CONNECTION_CANCELED);

        return super.read(b, off, len);
    }

    /*
     * @see java.io.FilterInputStream#available()
     */
    @Override
    public int available() throws IOException {

        /* Support early cancelation */
        if (fMonitor != null && fMonitor.isCanceled())
            throw new MonitorCanceledException(Messages.HttpConnectionInputStream_ERROR_CONNECTION_CANCELED);

        return super.available();
    }

    /**
     * @return the content length of the content served by this stream or -1 if
     * not available.
     */
    public int getContentLength() {
        Header header = fMethod.getResponseHeader(HEADER_RESPONSE_CONTENT_LENGTH);
        if (header != null) {
            String value = header.getValue();
            try {
                return Integer.parseInt(value);
            } catch (NumberFormatException e) {
                return -1;
            }
        }

        return -1;
    }

    /**
     * @return the content type of the content served by this stream or null if
     * not available.
     */
    public String getContentType() {
        Header header = fMethod.getResponseHeader(HEADER_RESPONSE_CONTENT_TYPE);
        if (header != null)
            return header.getValue();

        return null;
    }

    /**
     * @return the content disposition of the content served by this stream or
     * null if not available.
     */
    public String getContentDisposition() {
        Header header = fMethod.getResponseHeader(HEADER_RESPONSE_CONTENT_DISPOSITION);
        if (header != null)
            return header.getValue();

        return null;
    }

    /**
     * @return the content encoding of the content served by this stream or
     * null if not available.
     */
    public String getContentEncoding() {
        Header header = fMethod.getResponseHeader(HEADER_RESPONSE_CONTENT_ENCODING);
        if (header != null)
            return header.getValue();

        return null;
    }
}