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

org.eclipse.jetty.ee8.nested.MultiPartFormInputStream Maven / Gradle / Ivy

There is a newer version: 2.0.31
Show newest version
//
// ========================================================================
// Copyright (c) 1995 Mort Bay Consulting Pty Ltd and others.
//
// This program and the accompanying materials are made available under the
// terms of the Eclipse Public License v. 2.0 which is available at
// https://www.eclipse.org/legal/epl-2.0, or the Apache License, Version 2.0
// which is available at https://www.apache.org/licenses/LICENSE-2.0.
//
// SPDX-License-Identifier: EPL-2.0 OR Apache-2.0
// ========================================================================
//
package org.eclipse.jetty.ee8.nested;

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.nio.ByteBuffer;
import java.nio.file.FileSystems;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.StandardCopyOption;
import java.nio.file.StandardOpenOption;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.stream.Collectors;
import javax.servlet.MultipartConfigElement;
import javax.servlet.ServletInputStream;
import javax.servlet.http.Part;
import org.eclipse.jetty.http.ComplianceViolation;
import org.eclipse.jetty.http.HttpField;
import org.eclipse.jetty.http.MultiPartCompliance;
import org.eclipse.jetty.util.BufferUtil;
import org.eclipse.jetty.util.ByteArrayOutputStream2;
import org.eclipse.jetty.util.ExceptionUtil;
import org.eclipse.jetty.util.MultiMap;
import org.eclipse.jetty.util.QuotedStringTokenizer;
import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.util.thread.AutoLock;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import static org.eclipse.jetty.ee8.nested.ContextHandler.DEFAULT_MAX_FORM_KEYS;

/**
 * MultiPartInputStream
 * 

* Handle a MultiPart Mime input stream, breaking it up on the boundary into files and strings. *

*

* Deleting the parts can be done from a different thread if the parts are parsed asynchronously. * Because of this we use the state to fail the parsing and coordinate which thread will delete any remaining parts. * The deletion of parts is done by the cleanup thread in all cases except the transition from DELETING->DELETED which * is done by the parsing thread. *

*
{@code
 * UNPARSED - Parsing has not started, there are no parts which need to be cleaned up.
 * PARSING  - The parsing thread is reading from the InputStream and generating parts.
 * PARSED   - Parsing has complete and no more parts will be generated.
 * DELETING - deleteParts() has been called while we were in PARSING state, parsing thread will do the delete.
 * DELETED  - The parts have been deleted, this is the terminal state.
 *
 *                              deleteParts()
 *     +--------------------------------------------------------------+
 *     |                                                              |
 *     |                                          deleteParts()       v
 *  UNPARSED -------> PARSING --------> PARSED  ------------------>DELETED
 *                      |                                             ^
 *                      |                                             |
 *                      +---------------> DELETING -------------------+
 *                        deleteParts()               parsing thread
 * }
* @see https://tools.ietf.org/html/rfc7578 */ public class MultiPartFormInputStream implements MultiPart.Parser { private enum State { UNPARSED, PARSING, PARSED, DELETING, DELETED } private static final Logger LOG = LoggerFactory.getLogger(MultiPartFormInputStream.class); private static final QuotedStringTokenizer QUOTED_STRING_TOKENIZER = QuotedStringTokenizer.builder().delimiters(";").ignoreOptionalWhiteSpace().allowEscapeOnlyForQuotes().allowEmbeddedQuotes().build(); private final AutoLock _lock = new AutoLock(); private final MultiMap _parts = new MultiMap<>(); private final List _nonComplianceWarnings = new ArrayList<>(); private final InputStream _in; private final MultipartConfigElement _config; private final File _contextTmpDir; private final String _contentType; private final int _maxParts; private int _numParts = 0; private volatile Throwable _err; private volatile Path _tmpDir; private volatile boolean _writeFilesWithFilenames; private volatile int _bufferSize = 16 * 1024; private State state = State.UNPARSED; /** * @return an EnumSet of non compliances with the RFC that were accepted by this parser */ @Override public List getNonComplianceWarnings() { return _nonComplianceWarnings; } public class MultiPart implements Part { protected String _name; protected String _filename; protected File _file; protected OutputStream _out; protected ByteArrayOutputStream2 _bout; protected String _contentType; protected MultiMap _headers; protected long _size = 0; protected boolean _temporary = true; public MultiPart(String name, String filename) { _name = name; _filename = filename; } @Override public String toString() { return String.format("Part{n=%s,fn=%s,ct=%s,s=%d,tmp=%b,file=%s}", _name, _filename, _contentType, _size, _temporary, _file); } protected void setContentType(String contentType) { _contentType = contentType; } protected void open() throws IOException { // We will either be writing to a file, if it has a filename on the content-disposition // and otherwise a byte-array-input-stream, OR if we exceed the getFileSizeThreshold, we // will need to change to write to a file. if (isWriteFilesWithFilenames() && _filename != null && !_filename.trim().isEmpty()) { createFile(); } else { // Write to a buffer in memory until we discover we've exceed the // MultipartConfig fileSizeThreshold _out = _bout = new ByteArrayOutputStream2(); } } protected void close() throws IOException { _out.close(); } protected void write(int b) throws IOException { if (MultiPartFormInputStream.this._config.getMaxFileSize() > 0 && _size + 1 > MultiPartFormInputStream.this._config.getMaxFileSize()) throw new IllegalStateException("Multipart Mime part " + _name + " exceeds max filesize"); if (MultiPartFormInputStream.this._config.getFileSizeThreshold() > 0 && _size + 1 > MultiPartFormInputStream.this._config.getFileSizeThreshold() && _file == null) createFile(); _out.write(b); _size++; } protected void write(byte[] bytes, int offset, int length) throws IOException { if (MultiPartFormInputStream.this._config.getMaxFileSize() > 0 && _size + length > MultiPartFormInputStream.this._config.getMaxFileSize()) throw new IllegalStateException("Multipart Mime part " + _name + " exceeds max filesize"); if (MultiPartFormInputStream.this._config.getFileSizeThreshold() > 0 && _size + length > MultiPartFormInputStream.this._config.getFileSizeThreshold() && _file == null) createFile(); _out.write(bytes, offset, length); _size += length; } @Override public void write(String fileName) throws IOException { Path p = Path.of(fileName); if (!p.isAbsolute()) p = _tmpDir.resolve(p); if (_file == null) { _temporary = false; // part data is only in the ByteArrayOutputStream and never been written to disk _file = Files.createFile(p).toFile(); try (BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream(_file))) { _bout.writeTo(bos); bos.flush(); } finally { _bout = null; } } else { // the part data is already written to a temporary file, just rename it _temporary = false; Path src = _file.toPath(); Files.move(src, p, StandardCopyOption.REPLACE_EXISTING); _file = p.toFile(); } } protected void createFile() throws IOException { Path parent = MultiPartFormInputStream.this._tmpDir; Path tempFile = Files.createTempFile(parent, "MultiPart", ""); _file = tempFile.toFile(); OutputStream fos = Files.newOutputStream(tempFile, StandardOpenOption.WRITE); BufferedOutputStream bos = new BufferedOutputStream(fos); if (_size > 0 && _out != null) { // already written some bytes, so need to copy them into the file _out.flush(); _bout.writeTo(bos); _out.close(); } _bout = null; _out = bos; } protected void setHeaders(MultiMap headers) { _headers = headers; } @Override public String getContentType() { return _contentType; } @Override public String getHeader(String name) { if (name == null) return null; return _headers.getValue(StringUtil.asciiToLowerCase(name), 0); } @Override public Collection getHeaderNames() { return _headers.keySet(); } @Override public Collection getHeaders(String name) { Collection headers = _headers.getValues(name); return headers == null ? Collections.emptyList() : headers; } @Override public InputStream getInputStream() throws IOException { if (_file != null) { // written to a file, whether temporary or not return new BufferedInputStream(new FileInputStream(_file)); } else { // part content is in memory return new ByteArrayInputStream(_bout.getBuf(), 0, _bout.size()); } } @Override public String getSubmittedFileName() { return getContentDispositionFilename(); } public byte[] getBytes() { if (_bout != null) return _bout.toByteArray(); return null; } @Override public String getName() { return _name; } @Override public long getSize() { return _size; } /** * Remove the file, whether or not Part.write() was called on it (ie no longer temporary) */ @Override public void delete() throws IOException { if (_file != null && _file.exists()) if (!_file.delete()) throw new IOException("Could Not Delete File"); } /** * Only remove tmp files. * * @throws IOException if unable to delete the file */ public void cleanUp() throws IOException { if (_temporary) delete(); } /** * Get the file * * @return the file, if any, the data has been written to. */ public File getFile() { return _file; } /** * Get the filename from the content-disposition. * * @return null or the filename */ public String getContentDispositionFilename() { return _filename; } } /** * @param in Request input stream * @param contentType Content-Type header * @param config MultipartConfigElement * @param contextTmpDir {@value javax.servlet.ServletContext#TEMPDIR} */ public MultiPartFormInputStream(InputStream in, String contentType, MultipartConfigElement config, File contextTmpDir) { this(in, contentType, config, contextTmpDir, DEFAULT_MAX_FORM_KEYS); } /** * @param in Request input stream * @param contentType Content-Type header * @param config MultipartConfigElement * @param contextTmpDir {@value javax.servlet.ServletContext#TEMPDIR} * @param maxParts the maximum number of parts that can be parsed from the multipart content (0 for no parts allowed, -1 for unlimited parts). */ public MultiPartFormInputStream(InputStream in, String contentType, MultipartConfigElement config, File contextTmpDir, int maxParts) { // Must be a multipart request. _contentType = contentType; if (_contentType == null || !_contentType.startsWith("multipart/form-data")) throw new IllegalArgumentException("content type is not multipart/form-data"); _contextTmpDir = (contextTmpDir != null) ? contextTmpDir : new File(System.getProperty("java.io.tmpdir")); _config = (config != null) ? config : new MultipartConfigElement(_contextTmpDir.getAbsolutePath()); _maxParts = maxParts; if (in instanceof ServletInputStream) { if (((ServletInputStream) in).isFinished()) { _in = null; state = State.PARSED; return; } } _in = new BufferedInputStream(in); } /** * @return whether the list of parsed parts is empty * @deprecated use getParts().isEmpty() */ @Deprecated public boolean isEmpty() { if (_parts.isEmpty()) return true; for (List partList : _parts.values()) { if (!partList.isEmpty()) return false; } return true; } /** * Delete any tmp storage for parts, and clear out the parts list. */ public void deleteParts() { try (AutoLock l = _lock.lock()) { switch(state) { case DELETED: case DELETING: return; case PARSING: state = State.DELETING; return; case UNPARSED: state = State.DELETED; return; case PARSED: state = State.DELETED; break; } } delete(); } private void delete() { Throwable err = null; for (List parts : _parts.values()) { for (Part p : parts) { try { ((MultiPart) p).cleanUp(); } catch (Exception e) { err = ExceptionUtil.combine(err, e); } } } _parts.clear(); ExceptionUtil.ifExceptionThrowUnchecked(err); } /** * Parse, if necessary, the multipart data and return the list of Parts. * * @return the parts * @throws IOException if unable to get the parts */ @Override public Collection getParts() throws IOException { parse(); throwIfError(); return _parts.values().stream().flatMap(List::stream).collect(Collectors.toList()); } /** * Get the named Part. * * @param name the part name * @return the parts * @throws IOException if unable to get the part */ @Override public Part getPart(String name) throws IOException { parse(); throwIfError(); return _parts.getValue(name, 0); } /** * Throws an exception if one has been latched. * * @throws IOException the exception (if present) */ protected void throwIfError() throws IOException { if (_err != null) { if (LOG.isDebugEnabled()) LOG.debug("MultiPart parsing failure ", _err); _err.addSuppressed(new Throwable()); if (_err instanceof IOException) throw (IOException) _err; if (_err instanceof IllegalStateException) throw (IllegalStateException) _err; throw new IllegalStateException(_err); } } /** * Parse, if necessary, the multipart stream. */ protected void parse() { try (AutoLock l = _lock.lock()) { switch(state) { case UNPARSED: state = State.PARSING; break; case PARSED: return; default: _err = new IOException(state.name()); return; } } MultiPartParser parser = null; try { // Sort out the location to which to write files: // If there is a MultiPartConfigElement.location, use it // otherwise default to the context tmp dir if (StringUtil.isBlank(_config.getLocation())) _tmpDir = _contextTmpDir.toPath(); else { // If the MultiPartConfigElement.location is // relative, make it relative to the context tmp dir Path location = FileSystems.getDefault().getPath(_config.getLocation()); _tmpDir = (location.isAbsolute() ? location : _contextTmpDir.toPath().resolve(location)); } if (!Files.exists(_tmpDir)) Files.createDirectories(_tmpDir); String contentTypeBoundary = ""; int bstart = _contentType.indexOf("boundary="); if (bstart >= 0) { int bend = _contentType.indexOf(";", bstart); bend = (bend < 0 ? _contentType.length() : bend); contentTypeBoundary = HttpField.PARAMETER_TOKENIZER.unquote(value(_contentType.substring(bstart, bend)).trim()); } parser = new MultiPartParser(new Handler(), contentTypeBoundary); byte[] data = new byte[_bufferSize]; int len; long total = 0; while (true) { try (AutoLock l = _lock.lock()) { if (state != State.PARSING) { _err = new IOException(state.name()); return; } } len = _in.read(data); if (len > 0) { // keep running total of size of bytes read from input and throw an exception if exceeds MultipartConfigElement._maxRequestSize total += len; if (_config.getMaxRequestSize() > 0 && total > _config.getMaxRequestSize()) { _err = new IllegalStateException("Request exceeds maxRequestSize (" + _config.getMaxRequestSize() + ")"); return; } ByteBuffer buffer = BufferUtil.toBuffer(data); buffer.limit(len); if (parser.parse(buffer, false)) break; if (buffer.hasRemaining()) throw new IllegalStateException("Buffer did not fully consume"); } else if (len == -1) { parser.parse(BufferUtil.EMPTY_BUFFER, true); break; } } // check for exceptions if (_err != null) { return; } // check we read to the end of the message if (parser.getState() != MultiPartParser.State.END) { if (parser.getState() == MultiPartParser.State.PREAMBLE) _err = new IOException("Missing content for multipart request"); else _err = new IOException("Incomplete Multipart"); } if (LOG.isDebugEnabled()) { LOG.debug("Parsing Complete {} err={}", parser, _err); } } catch (Throwable e) { _err = e; // Notify parser if failure occurs if (parser != null) parser.parse(BufferUtil.EMPTY_BUFFER, true); } finally { boolean cleanup = false; try (AutoLock l = _lock.lock()) { switch(state) { case PARSING: state = State.PARSED; break; case DELETING: state = State.DELETED; cleanup = true; break; default: _err = new IllegalStateException(state.name()); } } if (cleanup) delete(); } } class Handler implements MultiPartParser.Handler { private MultiPart _part = null; private String contentDisposition = null; private String contentType = null; private MultiMap headers = new MultiMap<>(); @Override public boolean messageComplete() { return true; } @Override public void parsedField(String key, String value) { // Add to headers and mark if one of these fields. // headers.put(StringUtil.asciiToLowerCase(key), value); if (key.equalsIgnoreCase("content-disposition")) contentDisposition = value; else if (key.equalsIgnoreCase("content-type")) contentType = value; // Content Transfer encoding is no longer considered as it is deprecated as per // https://tools.ietf.org/html/rfc7578#section-4.7 if (key.equalsIgnoreCase("content-transfer-encoding")) { if (!"8bit".equalsIgnoreCase(value) && !"binary".equalsIgnoreCase(value)) _nonComplianceWarnings.add(new ComplianceViolation.Event(MultiPartCompliance.RFC7578, MultiPartCompliance.Violation.CONTENT_TRANSFER_ENCODING, value)); } } @Override public boolean headerComplete() { if (LOG.isDebugEnabled()) { LOG.debug("headerComplete {}", this); } try { // Extract content-disposition boolean formData = false; if (contentDisposition == null) { throw new IOException("Missing content-disposition"); } String name = null; String filename = null; for (Iterator i = QUOTED_STRING_TOKENIZER.tokenize(contentDisposition); i.hasNext(); ) { String t = i.next(); String tl = StringUtil.asciiToLowerCase(t); if (tl.startsWith("form-data")) formData = true; else if (tl.startsWith("name=")) name = value(t); else if (tl.startsWith("filename=")) filename = filenameValue(t); } // Check disposition if (!formData) throw new IOException("Part not form-data"); // It is valid for reset and submit buttons to have an empty name. // If no name is supplied, the browser skips sending the info for that field. // However, if you supply the empty string as the name, the browser sends the // field, with name as the empty string. So, only continue this loop if we // have not yet seen a name field. if (name == null) throw new IOException("No name in part"); // create the new part _part = new MultiPart(name, filename); _part.setHeaders(headers); _part.setContentType(contentType); _parts.add(name, _part); try { _part.open(); } catch (IOException e) { _err = e; return true; } } catch (Exception e) { _err = e; return true; } return false; } @Override public boolean content(ByteBuffer buffer, boolean last) { if (_part == null) return false; if (BufferUtil.hasContent(buffer)) { try { _part.write(buffer.array(), buffer.arrayOffset() + buffer.position(), buffer.remaining()); } catch (IOException e) { _err = e; return true; } } if (last) { try { _part.close(); } catch (IOException e) { _err = e; return true; } } return false; } @Override public void startPart() { reset(); _numParts++; if (_maxParts >= 0 && _numParts > _maxParts) throw new IllegalStateException(String.format("Form with too many keys [%d > %d]", _numParts, _maxParts)); } @Override public void earlyEOF() { if (LOG.isDebugEnabled()) LOG.debug("Early EOF {}", MultiPartFormInputStream.this); try { if (_part != null) _part.close(); } catch (IOException e) { LOG.warn("part could not be closed", e); } } public void reset() { _part = null; contentDisposition = null; contentType = null; headers = new MultiMap<>(); } } /** * @deprecated no replacement provided. */ @Deprecated public void setDeleteOnExit(boolean deleteOnExit) { // does nothing. } public void setWriteFilesWithFilenames(boolean writeFilesWithFilenames) { _writeFilesWithFilenames = writeFilesWithFilenames; } public boolean isWriteFilesWithFilenames() { return _writeFilesWithFilenames; } /** * @deprecated no replacement provided */ @Deprecated public boolean isDeleteOnExit() { return false; } private static String value(String nameEqualsValue) { int idx = nameEqualsValue.indexOf('='); String value = nameEqualsValue.substring(idx + 1).trim(); return HttpField.PARAMETER_TOKENIZER.unquote(value); } private static String filenameValue(String nameEqualsValue) { int idx = nameEqualsValue.indexOf('='); String value = nameEqualsValue.substring(idx + 1).trim(); if (value.matches(".??[a-z,A-Z]\\:\\\\[^\\\\].*")) { // incorrectly escaped IE filenames that have the whole path // we just strip any leading & trailing quotes and leave it as is char first = value.charAt(0); if (first == '"' || first == '\'') value = value.substring(1); char last = value.charAt(value.length() - 1); if (last == '"' || last == '\'') value = value.substring(0, value.length() - 1); return value; } else return HttpField.PARAMETER_TOKENIZER.unquote(value); } /** * Get the size of buffer used to read data from the input stream. * @return the size of buffer used to read data from the input stream */ public int getBufferSize() { return _bufferSize; } /** * Set the size of buffer used to read data from the input stream. * @param bufferSize the size of buffer used to read data from the input stream */ public void setBufferSize(int bufferSize) { _bufferSize = bufferSize; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy