ma.vi.base.io.BoundedInputStream Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of com.vikmad.base Show documentation
Show all versions of com.vikmad.base Show documentation
Base algos, data structures and utilities
The newest version!
/*
* Copyright (c) 2018 Vikash Madhow
*/
package ma.vi.base.io;
import java.io.IOException;
import java.io.InputStream;
/**
* This input stream places a bound on the length of the input that can be read from it. It responds with an
* IOException if an attempt is made to read beyond this bound.
*
* @author [email protected]
*/
public class BoundedInputStream extends InputStream {
/**
* Underlying input stream.
*/
protected InputStream in = null;
/**
* Content length (in bytes) of the message. Set on construction.
*/
protected int bound = -1;
/**
* Number of bytes that have been read from HTTP body by the various read() methods. This is used to prevent reading
* beyond the HTTP body.
*/
protected int bytesRead = 0;
/**
* Construct a BoundedInputStream wrapping the provided java.io.InputStream and with the provided bound.
*
* @param in java.io.InputStream to wrap
* @param bound Bounded length
* @throws IOException If the provided java.io.InputStream is null or if the bound is negative.
*/
public BoundedInputStream(InputStream in, int bound) throws IOException {
if (in == null) {
throw new IOException("InputStream must not be null");
}
if (bound < 0) {
throw new IOException("Bound must not be negative");
}
this.in = in;
this.bound = bound;
}
/**
* Returns the number of bytes that can be read without blocking
*/
@Override
public int available() throws IOException {
if (bytesRead >= bound) {
return 0;
} else {
return bound - bytesRead;
}
}
/**
* Reads a byte.
*
* @return The byte read or -1 if end-of-input reached
* @throws IOException If an error occurs during input
*/
@Override
public int read() throws IOException {
if (bytesRead == bound) {
bytesRead++;
return -1;
} else if (bytesRead > bound) {
throw new IOException("Read beyond end-of-stream");
} else {
bytesRead++;
return in.read();
}
}
/**
* Reads some bytes into the byte array.
*
* @param b byte array where read bytes will be copied
* @return The number of bytes read
* @throws IOException If an error occurs during input
*/
@Override
public int read(byte[] b) throws IOException {
return read(b, 0, b.length);
}
/**
* Reads some bytes into the byte array.
*
* @param b byte array where the read bytes will be copied
* @param offset Read bytes will be copied into the byte array starting at this position
* @param length Maximum number of bytes to read
* @return The number of bytes actually read. May be less than length.
* @throws IOException If an error occurs during input
*/
@Override
public int read(byte[] b, int offset, int length) throws IOException {
if (bytesRead == bound) {
bytesRead++;
return -1;
} else if (bytesRead > bound) {
throw new IOException("Read beyond bounds");
} else {
// don't let caller try to read more bytes than avaiable
int available = bound - bytesRead;
if (length > available) {
length = available;
}
// read as much as possible
int count = 0, read = 0;
while ((length > 0) &&
((count = in.read(b, offset, length)) != -1)) {
length -= count;
offset += count;
read += count;
}
bytesRead += read;
return read == 0 ? -1 : read;
}
}
/**
* Closes the underlying stream.
*
* @throws IOException If an I/O error occurs
*/
@Override
public void close() throws IOException {
in.close();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy