-
Bug
-
Resolution: Not an Issue
-
P4
-
None
-
1.2.1, 1.4.1
-
x86, sparc
-
solaris_7, windows_2000
Name: nt126004 Date: 12/17/2002
FULL PRODUCT VERSION :
java version "1.4.0"
Java(TM) 2 Runtime Environment, Standard Edition (build 1.4.0-b92)
Java HotSpot(TM) Client VM (build 1.4.0-b92, mixed mode)
(Also happens in 1.4.1_01)
FULL OPERATING SYSTEM VERSION :
Microsoft Windows 2000 [Version 5.00.2195]
ADDITIONAL OPERATING SYSTEMS :
All
A DESCRIPTION OF THE PROBLEM :
InflaterInputStream.available method can return 1 even
though there is nothing more to be read.
If the reader of InflaterInputStream reaches the exact end
of the stream, without trying to read past the end, the
InflaterInputStream will not realize that the stream is
empty and will still report that byte(s) are available.
The "available" contract is:
1. Don't return more bytes than are available.
2. Don't return more bytes than can be read without blocking.
The InflaterInputStream implementation fails on count #1.
STEPS TO FOLLOW TO REPRODUCE THE PROBLEM :
1. Run the test program below.
EXPECTED VERSUS ACTUAL BEHAVIOR :
Test program deflates some bytes and reads them back into
the same buffer. At this point, the input stream is empty
but it still reports that at least one byte is available.
The program below reproduces the bug and reports:
"java.io.IOException: available = 1, read = -1"
Note that available() returns 1 but the next read() fails.
The expected result is that available will return 0 when
there are no more bytes to be read.
ERROR MESSAGES/STACK TRACES THAT OCCUR :
When debugging this, I observed that the InflaterInputStream had not detected
readEOF even though the Inflater itself knew it was finished.
Beware: ZipFile's Inflater's "finished" flag never appears to get set.
REPRODUCIBILITY :
This bug can be reproduced always.
---------- BEGIN SOURCE ----------
/*
* InflaterAvailableTest.java
*/
import java.io.*;
import java.util.zip.*;
/**
* Demonstrates bug in InflaterInputStream.available
* method, where available returns 1 even though there
* is nothing more to be read.
*
* @author joe.bowbeer
*/
public class InflaterAvailableTest {
/**
* Deflate some bytes and read them back into the
* same buffer. At this point, the input stream is
* empty but it still reports that at least one byte
* is available.
*/
public static void main(String[] args)
throws Exception {
byte[] data = new byte[42];
ByteArrayOutputStream baos = new ByteArrayOutputStream();
DeflaterOutputStream dos = new DeflaterOutputStream(baos);
dos.write(data);
dos.close();
ByteArrayInputStream bais = new ByteArrayInputStream(baos.toByteArray());
InflaterInputStream iis = new InflaterInputStream(bais);
DataInputStream dais = new DataInputStream(iis);
dais.readFully(data);
int available = dais.available();
if (available > 0) {
throw new IOException(
"available = " + available + ", " +
"read = " + dais.read());
}
}
}
---------- END SOURCE ----------
(Review ID: 178892)
======================================================================
FULL PRODUCT VERSION :
java version "1.4.0"
Java(TM) 2 Runtime Environment, Standard Edition (build 1.4.0-b92)
Java HotSpot(TM) Client VM (build 1.4.0-b92, mixed mode)
(Also happens in 1.4.1_01)
FULL OPERATING SYSTEM VERSION :
Microsoft Windows 2000 [Version 5.00.2195]
ADDITIONAL OPERATING SYSTEMS :
All
A DESCRIPTION OF THE PROBLEM :
InflaterInputStream.available method can return 1 even
though there is nothing more to be read.
If the reader of InflaterInputStream reaches the exact end
of the stream, without trying to read past the end, the
InflaterInputStream will not realize that the stream is
empty and will still report that byte(s) are available.
The "available" contract is:
1. Don't return more bytes than are available.
2. Don't return more bytes than can be read without blocking.
The InflaterInputStream implementation fails on count #1.
STEPS TO FOLLOW TO REPRODUCE THE PROBLEM :
1. Run the test program below.
EXPECTED VERSUS ACTUAL BEHAVIOR :
Test program deflates some bytes and reads them back into
the same buffer. At this point, the input stream is empty
but it still reports that at least one byte is available.
The program below reproduces the bug and reports:
"java.io.IOException: available = 1, read = -1"
Note that available() returns 1 but the next read() fails.
The expected result is that available will return 0 when
there are no more bytes to be read.
ERROR MESSAGES/STACK TRACES THAT OCCUR :
When debugging this, I observed that the InflaterInputStream had not detected
readEOF even though the Inflater itself knew it was finished.
Beware: ZipFile's Inflater's "finished" flag never appears to get set.
REPRODUCIBILITY :
This bug can be reproduced always.
---------- BEGIN SOURCE ----------
/*
* InflaterAvailableTest.java
*/
import java.io.*;
import java.util.zip.*;
/**
* Demonstrates bug in InflaterInputStream.available
* method, where available returns 1 even though there
* is nothing more to be read.
*
* @author joe.bowbeer
*/
public class InflaterAvailableTest {
/**
* Deflate some bytes and read them back into the
* same buffer. At this point, the input stream is
* empty but it still reports that at least one byte
* is available.
*/
public static void main(String[] args)
throws Exception {
byte[] data = new byte[42];
ByteArrayOutputStream baos = new ByteArrayOutputStream();
DeflaterOutputStream dos = new DeflaterOutputStream(baos);
dos.write(data);
dos.close();
ByteArrayInputStream bais = new ByteArrayInputStream(baos.toByteArray());
InflaterInputStream iis = new InflaterInputStream(bais);
DataInputStream dais = new DataInputStream(iis);
dais.readFully(data);
int available = dais.available();
if (available > 0) {
throw new IOException(
"available = " + available + ", " +
"read = " + dais.read());
}
}
}
---------- END SOURCE ----------
(Review ID: 178892)
======================================================================
- relates to
-
JDK-4868205 InflaterInputStream.available() returns wrong value
-
- Closed
-