BufferedInputFilter.java
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.coyote.http11.filters;
import java.io.IOException;
import java.nio.BufferOverflowException;
import java.nio.ByteBuffer;
import java.nio.charset.StandardCharsets;
import org.apache.coyote.InputBuffer;
import org.apache.coyote.Request;
import org.apache.coyote.http11.InputFilter;
import org.apache.tomcat.util.buf.ByteChunk;
import org.apache.tomcat.util.net.ApplicationBufferHandler;
import org.apache.tomcat.util.res.StringManager;
/**
* Input filter responsible for reading and buffering the request body, so that
* it does not interfere with client SSL handshake messages.
*/
public class BufferedInputFilter implements InputFilter, ApplicationBufferHandler {
private static final StringManager sm = StringManager.getManager(BufferedInputFilter.class);
private static final String ENCODING_NAME = "buffered";
private static final ByteChunk ENCODING = new ByteChunk();
static {
ENCODING.setBytes(ENCODING_NAME.getBytes(StandardCharsets.ISO_8859_1), 0, ENCODING_NAME.length());
}
// Use ByteChunk since it correctly handles the special buffer size of -1
// for maxSavePostSize.
private ByteChunk buffered;
private ByteBuffer tempRead;
private InputBuffer buffer;
private boolean hasRead = false;
private final int maxSwallowSize;
public BufferedInputFilter(int maxSwallowSize) {
this.maxSwallowSize = maxSwallowSize;
}
// --------------------------------------------------------- Public Methods
/**
* Set the buffering limit. This should be reset every time the buffer is
* used.
*
* @param limit The maximum number of bytes that will be buffered
*/
public void setLimit(int limit) {
if (buffered == null) {
buffered = new ByteChunk();
buffered.setLimit(limit);
}
}
// ---------------------------------------------------- InputBuffer Methods
/**
* Reads the request body and buffers it.
*/
@Override
public void setRequest(Request request) {
// save off the Request body
try {
if (buffered.getLimit() == 0) {
// Special case - ignore (swallow) body. Do so within a limit.
long swallowed = 0;
int read = 0;
while ((read = buffer.doRead(this)) >= 0) {
swallowed += read;
if (maxSwallowSize > -1 && swallowed > maxSwallowSize) {
// No need for i18n - this isn't going to get logged
throw new IOException(sm.getString("bufferedInputFilter.maxSwallowSize"));
}
}
} else {
while (buffer.doRead(this) >= 0) {
buffered.append(tempRead);
tempRead = null;
}
}
} catch(IOException | BufferOverflowException ioe) {
// No need for i18n - this isn't going to get logged anywhere
throw new IllegalStateException(sm.getString("bufferedInputFilter.bodySize", ioe.getMessage()));
}
}
/**
* Fills the given ByteChunk with the buffered request body.
*
* @deprecated Unused. Will be removed in Tomcat 9. Use
* {@link #doRead(ApplicationBufferHandler)}
*/
@Deprecated
@Override
public int doRead(ByteChunk chunk) throws IOException {
if (isFinished()) {
return -1;
}
chunk.append(buffered);
hasRead = true;
return chunk.getLength();
}
/**
* Fills the given ByteBuffer with the buffered request body.
*/
@Override
public int doRead(ApplicationBufferHandler handler) throws IOException {
if (isFinished()) {
return -1;
}
handler.setByteBuffer(ByteBuffer.wrap(buffered.getBuffer(), buffered.getStart(), buffered.getLength()));
hasRead = true;
return buffered.getLength();
}
@Override
public void setBuffer(InputBuffer buffer) {
this.buffer = buffer;
}
@Override
public void recycle() {
if (buffered != null) {
if (buffered.getBuffer() != null && buffered.getBuffer().length > 65536) {
buffered = null;
} else {
buffered.recycle();
}
}
hasRead = false;
buffer = null;
}
@Override
public ByteChunk getEncodingName() {
return ENCODING;
}
@Override
public long end() throws IOException {
return 0;
}
@Override
public int available() {
int available = buffered.getLength();
if (available == 0) {
// No data buffered here. Try the next filter in the chain.
return buffer.available();
} else {
return available;
}
}
@Override
public boolean isFinished() {
return hasRead || buffered.getLength() <= 0;
}
@Override
public void setByteBuffer(ByteBuffer buffer) {
tempRead = buffer;
}
@Override
public ByteBuffer getByteBuffer() {
return tempRead;
}
@Override
public void expand(int size) {
// no-op
}
}