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

org.eclipse.jetty.servlets.MultiPartFilter Maven / Gradle / Ivy

There is a newer version: 11.0.0.beta1
Show newest version
//
//  ========================================================================
//  Copyright (c) 1995-2018 Mort Bay Consulting Pty. Ltd.
//  ------------------------------------------------------------------------
//  All rights reserved. This program and the accompanying materials
//  are made available under the terms of the Eclipse Public License v1.0
//  and Apache License v2.0 which accompanies this distribution.
//
//      The Eclipse Public License is available at
//      http://www.eclipse.org/legal/epl-v10.html
//
//      The Apache License v2.0 is available at
//      http://www.opensource.org/licenses/apache2.0.php
//
//  You may elect to redistribute this code under either of these licenses.
//  ========================================================================
//

package org.eclipse.jetty.servlets;

import java.io.BufferedInputStream;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.UnsupportedEncodingException;
import java.nio.charset.Charset;
import java.nio.charset.StandardCharsets;
import java.nio.charset.UnsupportedCharsetException;
import java.util.Collection;
import java.util.Collections;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;

import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.MultipartConfigElement;
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;
import javax.servlet.http.Part;

import org.eclipse.jetty.http.MimeTypes;
import org.eclipse.jetty.util.IO;
import org.eclipse.jetty.util.LazyList;
import org.eclipse.jetty.util.MultiMap;
import org.eclipse.jetty.util.MultiPartInputStreamParser;
import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger;


/* ------------------------------------------------------------ */
/**
 * Multipart Form Data Filter.
 * 

* This class is ONLY needed if you cannot use the Servlet 3.0 APIs for * configuring and handling multipart requests. See javax.servlet.http.HttpServletRequest.getParts(). If * you use the new servlet apis then you should REMOVE this filter from your webapp. *

* This class decodes the multipart/form-data stream sent by a HTML form that uses a file input * item. Any files sent are stored to a temporary file and a File object added to the request * as an attribute. All other values are made available via the normal getParameter API and * the setCharacterEncoding mechanism is respected when converting bytes to Strings. *

* Init Parameters: *

*
delete
*
(boolean) * If set to "true", any files created will be deleted when the current request returns.
*
maxFormKeys
*
(number) * Sets the maximum number of keys that may be present in a * form (default set by system property org.eclipse.jetty.server.Request.maxFormKeys or 1000) to protect * against DOS attacks by bad hash keys. *
*
deleteFiles
*
(boolean) * Controls if uploaded files are automatically deleted after the request completes. *
*
maxFileSize
*
(size in bytes) * Set the max size file that can be uploaded. *
*
maxRequestSize
*
(size in bytes) * To limit the size of the multipart request. *
*
* @deprecated See servlet 3.0 apis like javax.servlet.http.HttpServletRequest.getParts() */ public class MultiPartFilter implements Filter { private static final Logger LOG = Log.getLogger(MultiPartFilter.class); public final static String CONTENT_TYPE_SUFFIX=".org.eclipse.jetty.servlet.contentType"; private final static String MULTIPART = "org.eclipse.jetty.servlet.MultiPartFile.multiPartInputStream"; private File tempdir; private boolean _deleteFiles; private ServletContext _context; private int _fileOutputBuffer = 0; private long _maxFileSize = -1L; private long _maxRequestSize = -1L; private int _maxFormKeys = Integer.getInteger("org.eclipse.jetty.server.Request.maxFormKeys", 1000); /* ------------------------------------------------------------------------------- */ /** * @see javax.servlet.Filter#init(javax.servlet.FilterConfig) */ public void init(FilterConfig filterConfig) throws ServletException { tempdir=(File)filterConfig.getServletContext().getAttribute("javax.servlet.context.tempdir"); _deleteFiles="true".equals(filterConfig.getInitParameter("deleteFiles")); String fileOutputBuffer = filterConfig.getInitParameter("fileOutputBuffer"); if(fileOutputBuffer!=null) _fileOutputBuffer = Integer.parseInt(fileOutputBuffer); String maxFileSize = filterConfig.getInitParameter("maxFileSize"); if (maxFileSize != null) _maxFileSize = Long.parseLong(maxFileSize.trim()); String maxRequestSize = filterConfig.getInitParameter("maxRequestSize"); if (maxRequestSize != null) _maxRequestSize = Long.parseLong(maxRequestSize.trim()); _context=filterConfig.getServletContext(); String mfks = filterConfig.getInitParameter("maxFormKeys"); if (mfks!=null) _maxFormKeys=Integer.parseInt(mfks); } /* ------------------------------------------------------------------------------- */ /** * @see javax.servlet.Filter#doFilter(javax.servlet.ServletRequest, * javax.servlet.ServletResponse, javax.servlet.FilterChain) */ public void doFilter(ServletRequest request,ServletResponse response,FilterChain chain) throws IOException, ServletException { HttpServletRequest srequest=(HttpServletRequest)request; if(srequest.getContentType()==null||!srequest.getContentType().startsWith("multipart/form-data")) { chain.doFilter(request,response); return; } InputStream in = new BufferedInputStream(request.getInputStream()); String content_type=srequest.getContentType(); //Get current parameters so we can merge into them MultiMap params = new MultiMap(); for (Map.Entry entry : request.getParameterMap().entrySet()) { Object value = entry.getValue(); if (value instanceof String[]) params.addValues(entry.getKey(), (String[])value); else params.add(entry.getKey(), value); } MultipartConfigElement config = new MultipartConfigElement(tempdir.getCanonicalPath(), _maxFileSize, _maxRequestSize, _fileOutputBuffer); MultiPartInputStreamParser mpis = new MultiPartInputStreamParser(in, content_type, config, tempdir); mpis.setDeleteOnExit(_deleteFiles); request.setAttribute(MULTIPART, mpis); try { Collection parts = mpis.getParts(); if (parts != null) { Iterator itor = parts.iterator(); while (itor.hasNext() && params.size() < _maxFormKeys) { Part p = itor.next(); MultiPartInputStreamParser.MultiPart mp = (MultiPartInputStreamParser.MultiPart)p; if (mp.getFile() != null) { request.setAttribute(mp.getName(),mp.getFile()); if (mp.getContentDispositionFilename() != null) { params.add(mp.getName(), mp.getContentDispositionFilename()); if (mp.getContentType() != null) params.add(mp.getName()+CONTENT_TYPE_SUFFIX, mp.getContentType()); } } else { ByteArrayOutputStream bytes = new ByteArrayOutputStream(); IO.copy(p.getInputStream(), bytes); params.add(p.getName(), bytes.toByteArray()); if (p.getContentType() != null) params.add(p.getName()+CONTENT_TYPE_SUFFIX, p.getContentType()); } } } // handle request chain.doFilter(new Wrapper(srequest,params),response); } finally { deleteFiles(request); } } /* ------------------------------------------------------------ */ private void deleteFiles(ServletRequest request) { if (!_deleteFiles) return; MultiPartInputStreamParser mpis = (MultiPartInputStreamParser)request.getAttribute(MULTIPART); if (mpis != null) { try { mpis.deleteParts(); } catch (Exception e) { _context.log("Error deleting multipart tmp files", e); } } request.removeAttribute(MULTIPART); } /* ------------------------------------------------------------------------------- */ /** * @see javax.servlet.Filter#destroy() */ public void destroy() { } /* ------------------------------------------------------------------------------- */ /* ------------------------------------------------------------------------------- */ private static class Wrapper extends HttpServletRequestWrapper { Charset _encoding=StandardCharsets.UTF_8; MultiMap _params; /* ------------------------------------------------------------------------------- */ /** Constructor. * @param request */ public Wrapper(HttpServletRequest request, MultiMap map) { super(request); this._params=map; } /* ------------------------------------------------------------------------------- */ /** * @see javax.servlet.ServletRequest#getContentLength() */ @Override public int getContentLength() { return 0; } /* ------------------------------------------------------------------------------- */ /** * @see javax.servlet.ServletRequest#getParameter(java.lang.String) */ @Override public String getParameter(String name) { Object o=_params.get(name); if (!(o instanceof byte[]) && LazyList.size(o)>0) o=LazyList.get(o,0); if (o instanceof byte[]) { try { return getParameterBytesAsString(name, (byte[])o); } catch(Exception e) { LOG.warn(e); } } else if (o!=null) return String.valueOf(o); return null; } /* ------------------------------------------------------------------------------- */ /** * @see javax.servlet.ServletRequest#getParameterMap() */ @Override public Map getParameterMap() { Map cmap = new HashMap(); for ( Object key : _params.keySet() ) { cmap.put((String)key,getParameterValues((String)key)); } return Collections.unmodifiableMap(cmap); } /* ------------------------------------------------------------------------------- */ /** * @see javax.servlet.ServletRequest#getParameterNames() */ @Override public Enumeration getParameterNames() { return Collections.enumeration(_params.keySet()); } /* ------------------------------------------------------------------------------- */ /** * @see javax.servlet.ServletRequest#getParameterValues(java.lang.String) */ @Override public String[] getParameterValues(String name) { List l=_params.getValues(name); if (l==null || l.size()==0) return new String[0]; String[] v = new String[l.size()]; for (int i=0;i