Fixed InputStreamContentProvider.hasNext() to be idempotent until next() is called.
This commit is contained in:
parent
aa6226e1fa
commit
55c204b3ba
|
@ -25,6 +25,8 @@ import java.util.NoSuchElementException;
|
|||
|
||||
import org.eclipse.jetty.client.api.ContentProvider;
|
||||
import org.eclipse.jetty.util.BufferUtil;
|
||||
import org.eclipse.jetty.util.log.Log;
|
||||
import org.eclipse.jetty.util.log.Logger;
|
||||
|
||||
/**
|
||||
* A {@link ContentProvider} for an {@link InputStream}.
|
||||
|
@ -44,6 +46,8 @@ import org.eclipse.jetty.util.BufferUtil;
|
|||
*/
|
||||
public class InputStreamContentProvider implements ContentProvider
|
||||
{
|
||||
private static final Logger LOG = Log.getLogger(InputStreamContentProvider.class);
|
||||
|
||||
private final InputStream stream;
|
||||
private final int bufferSize;
|
||||
|
||||
|
@ -88,55 +92,84 @@ public class InputStreamContentProvider implements ContentProvider
|
|||
@Override
|
||||
public Iterator<ByteBuffer> iterator()
|
||||
{
|
||||
return new Iterator<ByteBuffer>()
|
||||
return new InputStreamIterator();
|
||||
}
|
||||
|
||||
/**
|
||||
* Iterating over an {@link InputStream} is tricky, because {@link #hasNext()} must return false
|
||||
* if the stream reads -1. However, we don't know what to return until we read the stream, which
|
||||
* means that stream reading must be performed by {@link #hasNext()}, which introduces a side-effect
|
||||
* on what is supposed to be a simple query method (with respect to the Query Command Separation
|
||||
* Principle).
|
||||
* <p />
|
||||
* Alternatively, we could return {@code true} from {@link #hasNext()} even if we don't know that
|
||||
* we will read -1, but then when {@link #next()} reads -1 it must return an empty buffer.
|
||||
* However this is problematic, since GETs with no content indication would become GET with chunked
|
||||
* content, and not understood by servers.
|
||||
* <p />
|
||||
* Therefore we need to make sure that {@link #hasNext()} does not perform any side effect (so that
|
||||
* it can be called multiple times) until {@link #next()} is called.
|
||||
*/
|
||||
private class InputStreamIterator implements Iterator<ByteBuffer>
|
||||
{
|
||||
private final byte[] bytes = new byte[bufferSize];
|
||||
private Exception failure;
|
||||
private ByteBuffer buffer;
|
||||
private Boolean hasNext;
|
||||
|
||||
@Override
|
||||
public boolean hasNext()
|
||||
{
|
||||
try
|
||||
{
|
||||
if (hasNext != null)
|
||||
return hasNext;
|
||||
|
||||
int read = stream.read(bytes);
|
||||
LOG.debug("Read {} bytes from {}", read, stream);
|
||||
if (read > 0)
|
||||
{
|
||||
buffer = onRead(bytes, 0, read);
|
||||
hasNext = Boolean.TRUE;
|
||||
return true;
|
||||
}
|
||||
else if (read < 0)
|
||||
{
|
||||
hasNext = Boolean.FALSE;
|
||||
return false;
|
||||
}
|
||||
else
|
||||
{
|
||||
buffer = BufferUtil.EMPTY_BUFFER;
|
||||
hasNext = Boolean.TRUE;
|
||||
return true;
|
||||
}
|
||||
}
|
||||
catch (Exception x)
|
||||
{
|
||||
LOG.debug(x);
|
||||
if (failure == null)
|
||||
{
|
||||
failure = x;
|
||||
// Signal we have more content to cause a call to
|
||||
// next() which will throw NoSuchElementException.
|
||||
hasNext = Boolean.TRUE;
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
throw new IllegalStateException();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public ByteBuffer next()
|
||||
{
|
||||
ByteBuffer result = buffer;
|
||||
buffer = null;
|
||||
if (failure != null)
|
||||
throw (NoSuchElementException)new NoSuchElementException().initCause(failure);
|
||||
ByteBuffer result = buffer;
|
||||
if (result == null)
|
||||
throw new NoSuchElementException();
|
||||
buffer = null;
|
||||
hasNext = null;
|
||||
return result;
|
||||
}
|
||||
|
||||
|
@ -145,6 +178,5 @@ public class InputStreamContentProvider implements ContentProvider
|
|||
{
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
};
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue