parent
f8f1959ebb
commit
452a17a8d5
@ -0,0 +1,47 @@
|
|||||||
|
// StreamLimitException.java
|
||||||
|
// ---------------------------
|
||||||
|
// Copyright 2017 by luccioman; https://github.com/luccioman
|
||||||
|
//
|
||||||
|
// This is a part of YaCy, a peer-to-peer based web search engine
|
||||||
|
//
|
||||||
|
// LICENSE
|
||||||
|
//
|
||||||
|
// This program is free software; you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU General Public License as published by
|
||||||
|
// the Free Software Foundation; either version 2 of the License, or
|
||||||
|
// (at your option) any later version.
|
||||||
|
//
|
||||||
|
// This program is distributed in the hope that it will be useful,
|
||||||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
// GNU General Public License for more details.
|
||||||
|
//
|
||||||
|
// You should have received a copy of the GNU General Public License
|
||||||
|
// along with this program; if not, write to the Free Software
|
||||||
|
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
|
||||||
|
package net.yacy.cora.util;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Used to indicate a limit on a stream has been reached or exceeded
|
||||||
|
* @author luccioman
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class StreamLimitException extends IOException {
|
||||||
|
|
||||||
|
/** Generated serialization ID */
|
||||||
|
private static final long serialVersionUID = -804446385126524902L;
|
||||||
|
|
||||||
|
public StreamLimitException() {
|
||||||
|
super();
|
||||||
|
}
|
||||||
|
|
||||||
|
public StreamLimitException(String message) {
|
||||||
|
super(message);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,193 @@
|
|||||||
|
// StrictLimitInputStream.java
|
||||||
|
// ---------------------------
|
||||||
|
// Copyright 2017 by luccioman; https://github.com/luccioman
|
||||||
|
//
|
||||||
|
// This is a part of YaCy, a peer-to-peer based web search engine
|
||||||
|
//
|
||||||
|
// LICENSE
|
||||||
|
//
|
||||||
|
// This program is free software; you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU General Public License as published by
|
||||||
|
// the Free Software Foundation; either version 2 of the License, or
|
||||||
|
// (at your option) any later version.
|
||||||
|
//
|
||||||
|
// This program is distributed in the hope that it will be useful,
|
||||||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
// GNU General Public License for more details.
|
||||||
|
//
|
||||||
|
// You should have received a copy of the GNU General Public License
|
||||||
|
// along with this program; if not, write to the Free Software
|
||||||
|
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
|
||||||
|
package net.yacy.cora.util;
|
||||||
|
|
||||||
|
import java.io.FilterInputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
|
||||||
|
import net.yacy.kelondro.util.Formatter;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Strictly limit the number of bytes consumed on a wrapped input stream :
|
||||||
|
* doesn't allow exceeding the limit and throw an exception when it is reached.
|
||||||
|
* See also some alternatives to consider :
|
||||||
|
* <ul>
|
||||||
|
* <li>org.apache.commons.fileupload.util.LimitedInputStream : check the limit
|
||||||
|
* only after reading, thus eventually allowing more bytes than the limit to be
|
||||||
|
* read. Doesn't properly implement mark() and reset() (when resetting, total
|
||||||
|
* count of consumed bytes is not reset)</li>
|
||||||
|
* <li>com.google.common.io.ByteStreams.LimitedInputStream : doesn't throw an
|
||||||
|
* exception on read() when the limit has been reached</li>
|
||||||
|
* <li>org.apache.commons.io.input.BoundedInputStream : doesn't throw an
|
||||||
|
* exception on read() when the limit has been reached</li>
|
||||||
|
* </ul>
|
||||||
|
*
|
||||||
|
* @author luccioman
|
||||||
|
*/
|
||||||
|
public class StrictLimitInputStream extends FilterInputStream {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Strict maximum bytes amount to consume on the wrapped stream. An
|
||||||
|
* exception is raised once consumed bytes is exactly equals to this value.
|
||||||
|
*/
|
||||||
|
private final long maxBytes;
|
||||||
|
|
||||||
|
/** The current position in the wrapped stream */
|
||||||
|
private long position = 0;
|
||||||
|
|
||||||
|
/** The marked position */
|
||||||
|
private long mark = -1;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The error message to use when a StreamLimitException is eventually raised
|
||||||
|
*/
|
||||||
|
private final String limitErrorMessage;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Wrap the given input stream and limit read bytes to maxBytes.
|
||||||
|
*
|
||||||
|
* @param inStream
|
||||||
|
* the input stream to wrap. Must not be null.
|
||||||
|
* @param maxBytes
|
||||||
|
* the maximum number of bytes to consume on the inStream. Must
|
||||||
|
* be greater or equals than zero.
|
||||||
|
* @throws IllegalArgumentException
|
||||||
|
* when inStream is null, or maxBytes is lower than zero
|
||||||
|
*/
|
||||||
|
public StrictLimitInputStream(final InputStream inStream, final long maxBytes) {
|
||||||
|
this(inStream, maxBytes, Formatter.bytesToString(maxBytes) + " limit has been reached");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Wrap the given input stream and limit read bytes to maxBytes.
|
||||||
|
*
|
||||||
|
* @param inStream
|
||||||
|
* the input stream to wrap. Must not be null.
|
||||||
|
* @param maxBytes
|
||||||
|
* the maximum number of bytes to consume on the inStream. Must
|
||||||
|
* be greater or equals than zero.
|
||||||
|
* @param limitErrorMessage
|
||||||
|
* the custom error message to use when a StreamLimitException is
|
||||||
|
* eventually raised. May be null.
|
||||||
|
* @throws IllegalArgumentException
|
||||||
|
* when inStream is null, or maxBytes is lower than zero
|
||||||
|
*/
|
||||||
|
public StrictLimitInputStream(final InputStream inStream, final long maxBytes, final String limitErrorMessage) {
|
||||||
|
super(inStream);
|
||||||
|
if (inStream == null) {
|
||||||
|
throw new IllegalArgumentException("inStream parameter must not be null");
|
||||||
|
}
|
||||||
|
if (maxBytes < 0) {
|
||||||
|
throw new IllegalArgumentException("maxBytes parameter must be greater or equals to zero");
|
||||||
|
}
|
||||||
|
this.maxBytes = maxBytes;
|
||||||
|
this.limitErrorMessage = limitErrorMessage;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*
|
||||||
|
* @throws StreamLimitException
|
||||||
|
* when the maxBytes limit has been reached
|
||||||
|
* @throws IOException
|
||||||
|
* when an I/O error occurs
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public int read() throws IOException {
|
||||||
|
if (this.position >= this.maxBytes) {
|
||||||
|
throw new StreamLimitException(this.limitErrorMessage);
|
||||||
|
}
|
||||||
|
final int result = this.in.read();
|
||||||
|
this.position++;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*
|
||||||
|
* @throws StreamLimitException
|
||||||
|
* when the maxBytes limit has been reached
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public int read(final byte[] b) throws IOException {
|
||||||
|
return this.read(b, 0, b.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*
|
||||||
|
* @throws StreamLimitException
|
||||||
|
* when the maxBytes limit has been reached
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public int read(final byte[] b, final int off, final int len) throws IOException, StreamLimitException {
|
||||||
|
if (this.position >= this.maxBytes) {
|
||||||
|
throw new StreamLimitException(this.limitErrorMessage);
|
||||||
|
}
|
||||||
|
final long maxToRead = Math.min(len, this.maxBytes - this.position);
|
||||||
|
final int nbRead = this.in.read(b, off, (int) maxToRead);
|
||||||
|
|
||||||
|
if (nbRead > 0) {
|
||||||
|
this.position += nbRead;
|
||||||
|
}
|
||||||
|
return nbRead;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*
|
||||||
|
* @throws StreamLimitException
|
||||||
|
* when the maxBytes limit has been reached
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public long skip(final long n) throws IOException {
|
||||||
|
if (this.position >= this.maxBytes) {
|
||||||
|
throw new StreamLimitException(this.limitErrorMessage);
|
||||||
|
}
|
||||||
|
final long toSkip = Math.min(n, this.maxBytes - this.position);
|
||||||
|
final long nbSkipped = this.in.skip(toSkip);
|
||||||
|
this.position += nbSkipped;
|
||||||
|
return nbSkipped;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* We do not override available() even when position has reached maxBytes : limit
|
||||||
|
reached must be signaled to the caller trough a StreamLimitException
|
||||||
|
when reading */
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void reset() throws IOException {
|
||||||
|
this.in.reset();
|
||||||
|
/*
|
||||||
|
* Rely on the wrapped input stream to check and throw an exception if
|
||||||
|
* the mark is invalid
|
||||||
|
*/
|
||||||
|
this.position = this.mark;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void mark(final int readlimit) {
|
||||||
|
this.in.mark(readlimit);
|
||||||
|
this.mark = this.position;
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in new issue