org.glassfish.admingui.common.servlet.DownloadServlet Maven / Gradle / Ivy
/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright (c) 2009-2012 Oracle and/or its affiliates. All rights reserved.
*
* The contents of this file are subject to the terms of either the GNU
* General Public License Version 2 only ("GPL") or the Common Development
* and Distribution License("CDDL") (collectively, the "License"). You
* may not use this file except in compliance with the License. You can
* obtain a copy of the License at
* https://glassfish.dev.java.net/public/CDDL+GPL_1_1.html
* or packager/legal/LICENSE.txt. See the License for the specific
* language governing permissions and limitations under the License.
*
* When distributing the software, include this License Header Notice in each
* file and include the License file at packager/legal/LICENSE.txt.
*
* GPL Classpath Exception:
* Oracle designates this particular file as subject to the "Classpath"
* exception as provided by Oracle in the GPL Version 2 section of the License
* file that accompanied this code.
*
* Modifications:
* If applicable, add the following below the License Header, with the fields
* enclosed by brackets [] replaced by your own identifying information:
* "Portions Copyright [year] [name of copyright owner]"
*
* Contributor(s):
* If you wish your version of this file to be governed by only the CDDL or
* only the GPL Version 2, indicate your decision by adding "[Contributor]
* elects to include this software in this distribution under the [CDDL or GPL
* Version 2] license." If you don't indicate a single choice of license, a
* recipient has the option to distribute your version of this file under
* either the CDDL, the GPL Version 2 or to extend the choice of license to
* its licensees as provided above. However, if you add GPL Version 2 code
* and therefore, elected the GPL Version 2 license, then the option applies
* only if the new code is made subject to such option by the copyright
* holder.
*/
package org.glassfish.admingui.common.servlet;
import java.io.IOException;
import java.io.BufferedInputStream;
import java.io.InputStream;
import java.util.HashMap;
import java.util.Map;
import java.util.StringTokenizer;
import javax.servlet.Servlet;
import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
/**
* This Servlet provides the ability to download information from the
* Server to the client. It provides the ability to set the content type
* of the downloaded file, if not specified, it will attempt to guess
* based on the extension (if possible). It requires the
* {@link DownloadServlet#ContentSource} of the data to download to be
* specified by passing in a ServletRequest
parameter named
* {@link DownloadServlet#CONTENT_SOURCE_ID}. The
* {@link DownloadServlet.ContentSource} provides a plugable means
* of obtaining data from an arbitrary source (i.e. the filesystem,
* generated on the fly, from some network location, etc.). The available
* {@link DownloadServlet.ContentSource} implemenatations must be
* specified via a Servlet
init parameter named
* {@link DownloadServlet#CONTENT_SOURCES}.
*/
public class DownloadServlet extends HttpServlet {
/**
* Default Constructor.
*/
public DownloadServlet() {
super();
}
/**
* Servlet initialization method.
*/
public void init(ServletConfig config) throws ServletException {
super.init(config);
// Register ContentSources
String sources = config.getInitParameter(CONTENT_SOURCES);
if ((sources == null) || (sources.trim().length() == 0)) {
throw new ServletException("No ContentSources specified! Ensure "
+ "at least 1 DownloadServlet.ContentSource is provided as"
+ " a Servlet init parameter (key: " + CONTENT_SOURCES
+ ").");
}
StringTokenizer tokens = new StringTokenizer(sources, " \t\n\r\f,;:");
while (tokens.hasMoreTokens()) {
registerContentSource(tokens.nextToken());
}
}
/**
* This method registers the given class name as a
* {@link DownloadServlet#ContentSource}. This method will attempt
* to resolve and instantiate the class using the current
* classloader.
*/
public void registerContentSource(String className) {
// Sanity Check
if ((className == null) || className.trim().equals("")) {
return;
}
Class cls = null;
try {
cls = Class.forName(className);
} catch (Exception ex) {
throw new RuntimeException(ex);
}
registerContentSource(cls);
}
/**
* This method registers the given class name as a
* {@link DownloadServlet#ContentSource}. This method will attempt
* to instantiate the class via the default constructor.
*/
public void registerContentSource(Class cls) {
// Create a new instance
DownloadServlet.ContentSource source = null;
try {
source = (DownloadServlet.ContentSource) cls.newInstance();
} catch (Exception ex) {
throw new RuntimeException(ex);
}
// Add the new instance to the registered ContentSources
_contentSources.put(source.getId(), source);
}
/**
* This method looks up a DownloadServlet.ContentSource given its id.
* The {@link DownloadServlet#ContentSource} must be previously
* registered.
*/
public DownloadServlet.ContentSource getContentSource(String id) {
return _contentSources.get(id);
}
/**
* This method delegates to the {@link #doPost()} method.
*/
public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doPost(request, response);
}
/**
* This method is the main method for this class when used in an
* HttpServlet
environment. It drives the process, which
* includes creating a {@link DownloadServet#Context}, choosing the
* appropriate {@link DownloadServlet#ContentSource}, and copying the
* output of the {@link DownloadServlet#ContentSource} to the
* ServletResponse
's OutputStream
.
*/
public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// Get the Download Context
DownloadServlet.Context context = getDownloadContext(request, response);
// Get the ContentSource
ContentSource source = getContentSource(request);
// Write Content
writeContent(source, context);
// Clean Up
source.cleanUp(context);
}
/**
* This method instantiates a {@link DownloadServlet.Context} and
* initializes it with the Servlet, ServletConfig, ServletRequest,
* and ServletResponse.
*/
protected DownloadServlet.Context getDownloadContext(HttpServletRequest request, HttpServletResponse response) {
DownloadServlet.Context ctx =
(DownloadServlet.Context) request.getAttribute(DOWNLOAD_CONTEXT);
if (ctx == null) {
ctx = new DownloadServlet.Context();
ctx.setServlet(this);
ctx.setServletConfig(getServletConfig());
ctx.setServletRequest(request);
request.setAttribute(DOWNLOAD_CONTEXT, ctx);
}
// This is done differently b/c the response may initially be null,
// subsequent calls may provide this information
ctx.setServletResponse(response);
return ctx;
}
/**
* This method locates the appropriate
* {@link DownloadServlet#ContentSource} for this request. It uses
* the given ServletRequest
to look for a
* ServletRequest Parameter named {@link #CONTENT_SOURCE_ID}.
* This value is used as the key when looking up registered
* {@link DownloadServlet#ContentSource} implementations.
*/
protected DownloadServlet.ContentSource getContentSource(ServletRequest request) {
// Get the ContentSource id
String id = request.getParameter(CONTENT_SOURCE_ID);
if (id == null) {
id = getServletConfig().getInitParameter(CONTENT_SOURCE_ID);
if(id == null) {
throw new RuntimeException("You must provide the '"
+ CONTENT_SOURCE_ID + "' request parameter!");
}
}
// Get the ContentSource
DownloadServlet.ContentSource src = getContentSource(id);
if (src == null) {
throw new RuntimeException("The ContentSource with id '" + id
+ "' is not registered!");
}
// Return the ContentSource
return src;
}
/**
*
This method is responsible for setting the response header
* information.
*/
protected void writeHeader(DownloadServlet.ContentSource source, DownloadServlet.Context context) {
ServletResponse resp = context.getServletResponse();
if (!(resp instanceof HttpServletResponse)) {
// This implementation is only valid for HttpServletResponse
return;
}
// Set the "Last-Modified" Header
// First check context
long longTime = source.getLastModified(context);
if (longTime != -1) {
((HttpServletResponse) resp).
setDateHeader("Last-Modified", longTime);
}
// First check CONTENT_TYPE
String contentType = (String) context.getAttribute(CONTENT_TYPE);
if (contentType == null) {
// Not found yet, check EXTENSION
String ext = (String) context.getAttribute(EXTENSION);
if (ext != null) {
contentType = mimeTypes.get(ext);
}
if (contentType == null) {
// Default Content-type is: application/octet-stream
contentType = DEFAULT_CONTENT_TYPE;
}
}
((HttpServletResponse) resp).setHeader("Content-type", contentType);
// Write additional headers
Object o = context.getAttribute(HEADERS);
if (o instanceof Map) {
@SuppressWarnings("unchecked")
Map headers = (Map) o;
for (Map.Entry h: headers.entrySet()) {
((HttpServletResponse) resp).setHeader(h.getKey(), h.getValue());
}
}
// TODO: log warning
}
/**
* This method is responsible for copying the data from the given
* InputStream
to the ServletResponse
's
* OutputStream
. The InputStream
should be
* the from the {@link DownloadServlet#ContentSource}.
*/
protected void writeContent(DownloadServlet.ContentSource source, DownloadServlet.Context context) {
// Get the InputStream
InputStream in = source.getInputStream(context);
// Get the OutputStream
ServletResponse resp = context.getServletResponse();
if (in == null) {
//nothing to write
String jspPage = (String) context.getAttribute("JSP_PAGE_SERVED");
if (jspPage != null && (jspPage.equals("false"))) {
try {
//Mainly to take care of javahelp2, bcz javahelp2 code needs an Exception to be thrown for FileNotFound.
//We may have to localize this message.
((HttpServletResponse) resp).sendError(404, "File Not Found");
} catch (IOException ex) {
//squelch it, just return.
}
}
return;
}
try (InputStream stream = new BufferedInputStream(in)) {
javax.servlet.ServletOutputStream out = resp.getOutputStream();
// Write the header
writeHeader(source, context);
// Copy the data to the ServletOutputStream
byte[] buf = new byte[512]; // Set our buffer at 512 bytes
int read = stream.read(buf, 0, 512);
while (read != -1) {
// Write data from the OutputStream to the InputStream
out.write(buf, 0, read);
// Read more...
read = stream.read(buf, 0, 512);
}
} catch (IOException ex) {
throw new RuntimeException(ex);
}
}
//////////////////////////////////////////////////////////////////////////
// Inner Classes
//////////////////////////////////////////////////////////////////////////
/**
* Implement this interface to provide an Object that is capable of
* providing data to DownloadServlet
.
* ContentSource
implementations must be thread safe.
* The DownloadServlet
will reuse the same instance when
* 2 requests are made to the same ContentSource type. Instance
* variables, therefore, should not be used; you may use the context
* to store local information.
*/
public static interface ContentSource {
/**
* This method should return a unique string used to identify this
* ContentSource
. This string must be specified in
* order to select the appropriate ContentSource
when
* using the DownloadServlet
.
*/
public String getId();
/**
* This method is responsible for generating the content and
* returning an InputStream to that content. It is also
* responsible for setting any attribute values in the
* {@link DownloadServlet#Context}, such as {@link EXTENSION} or
* {@link CONTENT_TYPE}.
*/
public InputStream getInputStream(DownloadServlet.Context ctx);
/**
* This method may be used to clean up any temporary resources.
* It will be invoked after the InputStream
has
* been completely read.
*/
public void cleanUp(DownloadServlet.Context ctx);
/**
* This method is responsible for returning the last modified date
* of the content, or -1 if not applicable. This information will
* be used for caching.
*/
public long getLastModified(DownloadServlet.Context context);
}
/**
* This class provides information about the request that may be
* necessary for the DownloadServlet.ContentSource
to
* provide content. The DownloadServlet
is responsible
* for supplying this object to the
* DownloadServlet.ContentSource
.
*/
public static class Context {
/**
* The default constructor.
*/
public Context() {
}
/**
* This method may be used to manage arbitrary information between
* the DownloadServlet
and the
* DownloadServlet.ContentSource
. This method
* retrieves an attribute.
*/
public Object getAttribute(String name) {
if (name == null) {
return null;
}
// First check the local attribute Map
Object value = _att.get(name);
if (value == null) {
// Not found, check the Request attributes...
value = getServletRequest().getParameter(name);
}
// Return the value (if any)
return value;
}
/**
* This method may be used to manage arbitrary information between
* the DownloadServlet
and the
* DownloadServlet.ContentSource
. This method sets
* an attribute.
*/
public void setAttribute(String name, Object value) {
if (name != null) {
_att.put(name, value);
}
}
/**
* This method may be used to manage arbitrary information between
* the DownloadServlet
and the
* DownloadServlet.ContentSource
. This method
* removes an attribute.
*/
public void removeAttribute(String name) {
_att.remove(name);
}
/**
* This returns the Servlet
associated with the
* request. This may be cast to the specific Servlet
* instance, such as HttpServlet
.
*/
public Servlet getServlet() {
return _servlet;
}
/**
* This sets the Servlet
associated with the
* request.
*/
protected void setServlet(Servlet servlet) {
_servlet = servlet;
}
/**
* This returns the ServletConfig
.
*/
public ServletConfig getServletConfig() {
return _servletConfig;
}
/**
* This sets the ServletConfig
.
*/
protected void setServletConfig(ServletConfig config) {
_servletConfig = config;
}
/**
* This returns the ServletRequest
associated with
* the request. This may be cast to the specific type, such as
* HttpServletRequest
.
*/
public ServletRequest getServletRequest() {
return _request;
}
/**
* This sets the ServletRequest
associated with the
* request.
*/
protected void setServletRequest(ServletRequest request) {
_request = request;
}
/**
* This returns the ServletResponse
associated with
* the request. This may be cast to the specific type, such as
* HttpServletResponse
.
*/
public ServletResponse getServletResponse() {
return _response;
}
/**
* This sets the ServletResponse
associated with the
* request.
*/
protected void setServletResponse(ServletResponse response) {
_response = response;
}
private Servlet _servlet = null;
private ServletConfig _servletConfig = null;
private ServletRequest _request = null;
private ServletResponse _response = null;
private Map _att = new HashMap();
}
/**
* This method gets called before the doGet/doPost method. The
* requires us to create the {@link DownloadServlet#Context} here.
* However, we do not have the HttpServletResponse
yet,
* so it will be null.
*/
protected long getLastModified(HttpServletRequest request) {
// Get the DownloadServlet Context
DownloadServlet.Context context = getDownloadContext(request, null);
// Get the ContentSource
ContentSource source = getContentSource(request);
// Calculate the last modified date
return source.getLastModified(context);
}
/**
* HashMap to hold mimetypes by extension.
*/
private static Map mimeTypes =
new HashMap(120);
static {
mimeTypes.put("aif", "audio/x-aiff");
mimeTypes.put("aifc", "audio/x-aiff");
mimeTypes.put("aiff", "audio/x-aiff");
mimeTypes.put("asc", "text/plain");
mimeTypes.put("asf", "application/x-ms-asf");
mimeTypes.put("asx", "application/x-ms-asf");
mimeTypes.put("au", "audio/basic");
mimeTypes.put("avi", "video/x-msvideo");
mimeTypes.put("bin", "application/octet-stream");
mimeTypes.put("bmp", "image/bmp");
mimeTypes.put("bwf", "audio/wav");
mimeTypes.put("bz2", "application/x-bzip2");
mimeTypes.put("c", "text/plain");
mimeTypes.put("cc", "text/plain");
mimeTypes.put("cdda", "audio/x-aiff");
mimeTypes.put("class", "application/octet-stream");
mimeTypes.put("com", "application/octet-stream");
mimeTypes.put("cpp", "text/plain");
mimeTypes.put("cpr", "image/cpr");
mimeTypes.put("css", "text/css");
mimeTypes.put("doc", "application/msword");
mimeTypes.put("dot", "application/msword");
mimeTypes.put("dtd", "text/xml");
mimeTypes.put("ear", "application/zip");
mimeTypes.put("exe", "application/octet-stream");
mimeTypes.put("flc", "video/flc");
mimeTypes.put("fm", "application/x-maker");
mimeTypes.put("frame", "application/x-maker");
mimeTypes.put("frm", "application/x-maker");
mimeTypes.put("h", "text/plain");
mimeTypes.put("hh", "text/plain");
mimeTypes.put("hpp", "text/plain");
mimeTypes.put("hqx", "application/mac-binhex40");
mimeTypes.put("htm", "text/html");
mimeTypes.put("html", "text/html");
mimeTypes.put("gif", "image/gif");
mimeTypes.put("gz", "application/x-gunzip");
mimeTypes.put("ico", "image/x-icon");
mimeTypes.put("iso", "application/octet-stream");
mimeTypes.put("jar", "application/zip");
mimeTypes.put("java", "text/plain");
mimeTypes.put("jnlp", "application/x-java-jnlp-file");
mimeTypes.put("jpeg", "image/jpeg");
mimeTypes.put("jpe", "image/jpeg");
mimeTypes.put("jpg", "image/jpeg");
mimeTypes.put("js", "text/x-javascript");
mimeTypes.put("m3u", "audio/x-mpegurl");
mimeTypes.put("maker", "application/x-maker");
mimeTypes.put("mid", "audio/midi");
mimeTypes.put("midi", "audio/midi");
mimeTypes.put("mim", "application/mime");
mimeTypes.put("mime", "application/mime");
mimeTypes.put("mov", "video/quicktime");
mimeTypes.put("mp2", "audio/mpeg");
mimeTypes.put("mp3", "audio/mpeg");
mimeTypes.put("mp4", "video/mpeg4");
mimeTypes.put("mpa", "video/mpeg");
mimeTypes.put("mpe", "video/mpeg");
mimeTypes.put("mpeg", "video/mpeg");
mimeTypes.put("mpg", "video/mpeg");
mimeTypes.put("mpga", "audio/mpeg");
mimeTypes.put("mpm", "video/mpeg");
mimeTypes.put("mpv", "video/mpeg");
mimeTypes.put("pdf", "application/pdf");
mimeTypes.put("pic", "image/x-pict");
mimeTypes.put("pict", "image/x-pict");
mimeTypes.put("pct", "image/x-pict");
mimeTypes.put("pl", "application/x-perl");
mimeTypes.put("png", "image/png");
mimeTypes.put("pnm", "image/x-portable-anymap");
mimeTypes.put("pbm", "image/x-portable-bitmap");
mimeTypes.put("ppm", "image/x-portable-pixmap");
mimeTypes.put("ps", "application/postscript");
mimeTypes.put("ppt", "application/vnd.ms-powerpoint");
mimeTypes.put("qt", "video/quicktime");
mimeTypes.put("ra", "application/vnd.rn-realaudio");
mimeTypes.put("rar", "application/zip");
mimeTypes.put("rf", "application/vnd.rn-realflash");
mimeTypes.put("ra", "audio/vnd.rn-realaudio");
mimeTypes.put("ram", "audio/x-pn-realaudio");
mimeTypes.put("rm", "application/vnd.rn-realmedia");
mimeTypes.put("rmm", "audio/x-pn-realaudio");
mimeTypes.put("rsml", "application/vnd.rn-rsml");
mimeTypes.put("rtf", "text/rtf");
mimeTypes.put("rv", "video/vnd.rn-realvideo");
mimeTypes.put("spl", "application/futuresplash");
mimeTypes.put("snd", "audio/basic");
mimeTypes.put("ssm", "application/smil");
mimeTypes.put("swf", "application/x-shockwave-flash");
mimeTypes.put("tar", "application/x-tar");
mimeTypes.put("tgz", "application/x-gtar");
mimeTypes.put("tif", "image/tiff");
mimeTypes.put("tiff", "image/tiff");
mimeTypes.put("txt", "text/plain");
mimeTypes.put("ulw", "audio/basic");
mimeTypes.put("war", "application/zip");
mimeTypes.put("wav", "audio/x-wav");
mimeTypes.put("wax", "application/x-ms-wax");
mimeTypes.put("wm", "application/x-ms-wm");
mimeTypes.put("wma", "application/x-ms-wma");
mimeTypes.put("wml", "text/wml");
mimeTypes.put("wmw", "application/x-ms-wmw");
mimeTypes.put("wrd", "application/msword");
mimeTypes.put("wvx", "application/x-ms-wvx");
mimeTypes.put("xbm", "image/x-xbitmap");
mimeTypes.put("xpm", "image/image/x-xpixmap");
mimeTypes.put("xml", "text/xml");
mimeTypes.put("xsl", "text/xml");
mimeTypes.put("xls", "application/vnd.ms-excel");
mimeTypes.put("zip", "application/zip");
mimeTypes.put("z", "application/x-compress");
mimeTypes.put("Z", "application/x-compress");
}
private static Map _contentSources =
new HashMap();
/**
* This String ("downloadContext") is the name if the
* ServletRequest Attribute used to store the
* {@link DownloadServlet#Context} object for this request.
*/
public static final String DOWNLOAD_CONTEXT = "downloadContext";
/**
* This String ("ContentSources") is the name if the Servlet Init
* Parameter that should be used to register all available
* {@link DownloadServlet#ContentSource} implementations.
*/
public static final String CONTENT_SOURCES = "ContentSources";
/**
* This is the ServletRequest Parameter that should be provided
* to identify the DownloadServlet.ContentSource
* implementation that should be used. This value must match the
* value returned by the DownloadServlet.ContentSource
* implementation's getId()
method.
*/
public static final String CONTENT_SOURCE_ID = "contentSourceId";
/**
* The Content-type ("ContentType"). This is the
* {@link DownloadServlet#Context} attribute used to specify an
* explicit "Content-type". It may be set by the
* {@link DownloadServlet#ContentSource}, or may be passed in via a
* request parameter. If not specified, the {@link #EXTENSION} will
* be used. If that fails, the {@link #DEFAULT_CONTENT_TYPE} will
* apply.
*/
public static final String CONTENT_TYPE = "ContentType";
/**
* The Default Content-type ("application/octet-stream").
*/
public static final String DEFAULT_CONTENT_TYPE =
"application/octet-stream";
/**
* This is the {@link DownloadServlet#Context} attribute name used to
* specify the filename extension of the content. It is the
* responsibility of the {@link DownloadServlet#ContentSource} to set
* this value. The value should represent the filename extension of
* the content if it were saved to a filesystem.
*/
public static final String EXTENSION = "extension";
/**
* This is the {@link DownloadServlet#Context} attribute name used to
* specify optional additional headers. It must be set to
* Map
object when needed.
*/
public static final String HEADERS = "Headers";
}