mirror of https://github.com/apache/poi.git
Fix bug #45290 - Support odd files where the POIFS header block comes after the data blocks, and is on the data blocks list
git-svn-id: https://svn.apache.org/repos/asf/poi/trunk@713447 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
84d10b7ecd
commit
1b24850c7c
|
@ -37,6 +37,7 @@
|
||||||
|
|
||||||
<!-- Don't forget to update status.xml too! -->
|
<!-- Don't forget to update status.xml too! -->
|
||||||
<release version="3.5-beta4" date="2008-??-??">
|
<release version="3.5-beta4" date="2008-??-??">
|
||||||
|
<action dev="POI-DEVELOPERS" type="fix">45290 - Support odd files where the POIFS header block comes after the data blocks, and is on the data blocks list</header>
|
||||||
<action dev="POI-DEVELOPERS" type="fix">46184 - More odd escaped date formats</action>
|
<action dev="POI-DEVELOPERS" type="fix">46184 - More odd escaped date formats</action>
|
||||||
<action dev="POI-DEVELOPERS" type="add">Include the sheet number in the output of XLS2CSVmra</action>
|
<action dev="POI-DEVELOPERS" type="add">Include the sheet number in the output of XLS2CSVmra</action>
|
||||||
<action dev="POI-DEVELOPERS" type="fix">46043 - correctly write out HPSF properties with HWPF</action>
|
<action dev="POI-DEVELOPERS" type="fix">46043 - correctly write out HPSF properties with HWPF</action>
|
||||||
|
|
|
@ -34,6 +34,7 @@
|
||||||
<!-- Don't forget to update changes.xml too! -->
|
<!-- Don't forget to update changes.xml too! -->
|
||||||
<changes>
|
<changes>
|
||||||
<release version="3.5-beta4" date="2008-??-??">
|
<release version="3.5-beta4" date="2008-??-??">
|
||||||
|
<action dev="POI-DEVELOPERS" type="fix">45290 - Support odd files where the POIFS header block comes after the data blocks, and is on the data blocks list</header>
|
||||||
<action dev="POI-DEVELOPERS" type="fix">46184 - More odd escaped date formats</action>
|
<action dev="POI-DEVELOPERS" type="fix">46184 - More odd escaped date formats</action>
|
||||||
<action dev="POI-DEVELOPERS" type="add">Include the sheet number in the output of XLS2CSVmra</action>
|
<action dev="POI-DEVELOPERS" type="add">Include the sheet number in the output of XLS2CSVmra</action>
|
||||||
<action dev="POI-DEVELOPERS" type="fix">46043 - correctly write out HPSF properties with HWPF</action>
|
<action dev="POI-DEVELOPERS" type="fix">46043 - correctly write out HPSF properties with HWPF</action>
|
||||||
|
|
|
@ -245,13 +245,13 @@ public class POIFSReader
|
||||||
{
|
{
|
||||||
document =
|
document =
|
||||||
new POIFSDocument(name, small_blocks
|
new POIFSDocument(name, small_blocks
|
||||||
.fetchBlocks(startBlock), size);
|
.fetchBlocks(startBlock, -1), size);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
document =
|
document =
|
||||||
new POIFSDocument(name, big_blocks
|
new POIFSDocument(name, big_blocks
|
||||||
.fetchBlocks(startBlock), size);
|
.fetchBlocks(startBlock, -1), size);
|
||||||
}
|
}
|
||||||
while (listeners.hasNext())
|
while (listeners.hasNext())
|
||||||
{
|
{
|
||||||
|
@ -270,11 +270,11 @@ public class POIFSReader
|
||||||
// consume the document's data and discard it
|
// consume the document's data and discard it
|
||||||
if (property.shouldUseSmallBlocks())
|
if (property.shouldUseSmallBlocks())
|
||||||
{
|
{
|
||||||
small_blocks.fetchBlocks(startBlock);
|
small_blocks.fetchBlocks(startBlock, -1);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
big_blocks.fetchBlocks(startBlock);
|
big_blocks.fetchBlocks(startBlock, -1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -173,11 +173,16 @@ public class POIFSFileSystem
|
||||||
data_blocks);
|
data_blocks);
|
||||||
|
|
||||||
// init documents
|
// init documents
|
||||||
processProperties(SmallBlockTableReader
|
processProperties(
|
||||||
.getSmallDocumentBlocks(data_blocks, properties
|
SmallBlockTableReader.getSmallDocumentBlocks(
|
||||||
.getRoot(), header_block_reader
|
data_blocks, properties.getRoot(),
|
||||||
.getSBATStart()), data_blocks, properties.getRoot()
|
header_block_reader.getSBATStart()
|
||||||
.getChildren(), null);
|
),
|
||||||
|
data_blocks,
|
||||||
|
properties.getRoot().getChildren(),
|
||||||
|
null,
|
||||||
|
header_block_reader.getPropertyStart()
|
||||||
|
);
|
||||||
}
|
}
|
||||||
/**
|
/**
|
||||||
* @param stream the stream to be closed
|
* @param stream the stream to be closed
|
||||||
|
@ -491,7 +496,8 @@ public class POIFSFileSystem
|
||||||
private void processProperties(final BlockList small_blocks,
|
private void processProperties(final BlockList small_blocks,
|
||||||
final BlockList big_blocks,
|
final BlockList big_blocks,
|
||||||
final Iterator properties,
|
final Iterator properties,
|
||||||
final DirectoryNode dir)
|
final DirectoryNode dir,
|
||||||
|
final int headerPropertiesStartAt)
|
||||||
throws IOException
|
throws IOException
|
||||||
{
|
{
|
||||||
while (properties.hasNext())
|
while (properties.hasNext())
|
||||||
|
@ -511,7 +517,8 @@ public class POIFSFileSystem
|
||||||
|
|
||||||
processProperties(
|
processProperties(
|
||||||
small_blocks, big_blocks,
|
small_blocks, big_blocks,
|
||||||
(( DirectoryProperty ) property).getChildren(), new_dir);
|
(( DirectoryProperty ) property).getChildren(),
|
||||||
|
new_dir, headerPropertiesStartAt);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -522,14 +529,15 @@ public class POIFSFileSystem
|
||||||
if (property.shouldUseSmallBlocks())
|
if (property.shouldUseSmallBlocks())
|
||||||
{
|
{
|
||||||
document =
|
document =
|
||||||
new POIFSDocument(name, small_blocks
|
new POIFSDocument(name,
|
||||||
.fetchBlocks(startBlock), size);
|
small_blocks.fetchBlocks(startBlock, headerPropertiesStartAt),
|
||||||
|
size);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
document =
|
document =
|
||||||
new POIFSDocument(name,
|
new POIFSDocument(name,
|
||||||
big_blocks.fetchBlocks(startBlock),
|
big_blocks.fetchBlocks(startBlock, headerPropertiesStartAt),
|
||||||
size);
|
size);
|
||||||
}
|
}
|
||||||
parent.createDocument(document);
|
parent.createDocument(document);
|
||||||
|
|
|
@ -78,7 +78,7 @@ public class PropertyTable
|
||||||
_blocks = null;
|
_blocks = null;
|
||||||
_properties =
|
_properties =
|
||||||
PropertyFactory
|
PropertyFactory
|
||||||
.convertToProperties(blockList.fetchBlocks(startBlock));
|
.convertToProperties(blockList.fetchBlocks(startBlock, -1));
|
||||||
populatePropertyTree(( DirectoryProperty ) _properties.get(0));
|
populatePropertyTree(( DirectoryProperty ) _properties.get(0));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -179,17 +179,34 @@ public class BlockAllocationTableReader
|
||||||
*/
|
*/
|
||||||
|
|
||||||
ListManagedBlock [] fetchBlocks(final int startBlock,
|
ListManagedBlock [] fetchBlocks(final int startBlock,
|
||||||
|
final int headerPropertiesStartBlock,
|
||||||
final BlockList blockList)
|
final BlockList blockList)
|
||||||
throws IOException
|
throws IOException
|
||||||
{
|
{
|
||||||
List blocks = new ArrayList();
|
List blocks = new ArrayList();
|
||||||
int currentBlock = startBlock;
|
int currentBlock = startBlock;
|
||||||
|
boolean firstPass = true;
|
||||||
|
|
||||||
while (currentBlock != POIFSConstants.END_OF_CHAIN)
|
// Process the chain from the start to the end
|
||||||
{
|
// Normally we have header, data, end
|
||||||
blocks.add(blockList.remove(currentBlock));
|
// Sometimes we have data, header, end
|
||||||
currentBlock = _entries.get(currentBlock);
|
// For those cases, stop at the header, not the end
|
||||||
|
while (currentBlock != POIFSConstants.END_OF_CHAIN) {
|
||||||
|
try {
|
||||||
|
blocks.add(blockList.remove(currentBlock));
|
||||||
|
currentBlock = _entries.get(currentBlock);
|
||||||
|
} catch(IOException e) {
|
||||||
|
if(currentBlock == headerPropertiesStartBlock) {
|
||||||
|
// Special case where things are in the wrong order
|
||||||
|
System.err.println("Warning, header block comes after data blocks in POIFS block listing");
|
||||||
|
currentBlock = POIFSConstants.END_OF_CHAIN;
|
||||||
|
} else {
|
||||||
|
// Ripple up
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return ( ListManagedBlock [] ) blocks
|
return ( ListManagedBlock [] ) blocks
|
||||||
.toArray(new ListManagedBlock[ 0 ]);
|
.toArray(new ListManagedBlock[ 0 ]);
|
||||||
}
|
}
|
||||||
|
|
|
@ -59,13 +59,14 @@ public interface BlockList
|
||||||
* blocks are removed from the list.
|
* blocks are removed from the list.
|
||||||
*
|
*
|
||||||
* @param startBlock the index of the first block in the stream
|
* @param startBlock the index of the first block in the stream
|
||||||
|
* @param headerPropertiesStartBlock the index of the first header block in the stream
|
||||||
*
|
*
|
||||||
* @return the stream as an array of correctly ordered blocks
|
* @return the stream as an array of correctly ordered blocks
|
||||||
*
|
*
|
||||||
* @exception IOException if blocks are missing
|
* @exception IOException if blocks are missing
|
||||||
*/
|
*/
|
||||||
|
|
||||||
public ListManagedBlock [] fetchBlocks(final int startBlock)
|
public ListManagedBlock [] fetchBlocks(final int startBlock, final int headerPropertiesStartBlock)
|
||||||
throws IOException;
|
throws IOException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -94,8 +94,10 @@ class BlockListImpl
|
||||||
result = _blocks[ index ];
|
result = _blocks[ index ];
|
||||||
if (result == null)
|
if (result == null)
|
||||||
{
|
{
|
||||||
throw new IOException("block[ " + index
|
throw new IOException(
|
||||||
+ " ] already removed");
|
"block[ " + index + " ] already removed - " +
|
||||||
|
"does your POIFS have circular or duplicate block references?"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
_blocks[ index ] = null;
|
_blocks[ index ] = null;
|
||||||
}
|
}
|
||||||
|
@ -119,7 +121,7 @@ class BlockListImpl
|
||||||
* @exception IOException if blocks are missing
|
* @exception IOException if blocks are missing
|
||||||
*/
|
*/
|
||||||
|
|
||||||
public ListManagedBlock [] fetchBlocks(final int startBlock)
|
public ListManagedBlock [] fetchBlocks(final int startBlock, final int headerPropertiesStartBlock)
|
||||||
throws IOException
|
throws IOException
|
||||||
{
|
{
|
||||||
if (_bat == null)
|
if (_bat == null)
|
||||||
|
@ -127,7 +129,7 @@ class BlockListImpl
|
||||||
throw new IOException(
|
throw new IOException(
|
||||||
"Improperly initialized list: no block allocation table provided");
|
"Improperly initialized list: no block allocation table provided");
|
||||||
}
|
}
|
||||||
return _bat.fetchBlocks(startBlock, this);
|
return _bat.fetchBlocks(startBlock, headerPropertiesStartBlock, this);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -56,9 +56,9 @@ public class SmallBlockTableReader
|
||||||
{
|
{
|
||||||
BlockList list =
|
BlockList list =
|
||||||
new SmallDocumentBlockList(SmallDocumentBlock
|
new SmallDocumentBlockList(SmallDocumentBlock
|
||||||
.extract(blockList.fetchBlocks(root.getStartBlock())));
|
.extract(blockList.fetchBlocks(root.getStartBlock(), -1)));
|
||||||
|
|
||||||
new BlockAllocationTableReader(blockList.fetchBlocks(sbatStart),
|
new BlockAllocationTableReader(blockList.fetchBlocks(sbatStart, -1),
|
||||||
list);
|
list);
|
||||||
return list;
|
return list;
|
||||||
}
|
}
|
||||||
|
|
Binary file not shown.
|
@ -27,9 +27,6 @@ import java.util.List;
|
||||||
import junit.framework.AssertionFailedError;
|
import junit.framework.AssertionFailedError;
|
||||||
import junit.framework.TestCase;
|
import junit.framework.TestCase;
|
||||||
|
|
||||||
import org.apache.poi.ss.util.Region;
|
|
||||||
import org.apache.poi.ss.util.CellRangeAddress;
|
|
||||||
|
|
||||||
import org.apache.poi.hssf.HSSFTestDataSamples;
|
import org.apache.poi.hssf.HSSFTestDataSamples;
|
||||||
import org.apache.poi.hssf.model.Workbook;
|
import org.apache.poi.hssf.model.Workbook;
|
||||||
import org.apache.poi.hssf.record.CellValueRecordInterface;
|
import org.apache.poi.hssf.record.CellValueRecordInterface;
|
||||||
|
@ -38,6 +35,7 @@ import org.apache.poi.hssf.record.NameRecord;
|
||||||
import org.apache.poi.hssf.record.aggregates.FormulaRecordAggregate;
|
import org.apache.poi.hssf.record.aggregates.FormulaRecordAggregate;
|
||||||
import org.apache.poi.hssf.record.formula.DeletedArea3DPtg;
|
import org.apache.poi.hssf.record.formula.DeletedArea3DPtg;
|
||||||
import org.apache.poi.hssf.record.formula.Ptg;
|
import org.apache.poi.hssf.record.formula.Ptg;
|
||||||
|
import org.apache.poi.ss.util.CellRangeAddress;
|
||||||
import org.apache.poi.util.TempFile;
|
import org.apache.poi.util.TempFile;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1533,4 +1531,13 @@ public final class TestBugs extends TestCase {
|
||||||
assertEquals(7, wb.getNumberOfSheets());
|
assertEquals(7, wb.getNumberOfSheets());
|
||||||
wb = HSSFTestDataSamples.writeOutAndReadBack(wb);
|
wb = HSSFTestDataSamples.writeOutAndReadBack(wb);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Odd POIFS blocks issue:
|
||||||
|
* block[ 44 ] already removed from org.apache.poi.poifs.storage.BlockListImpl.remove
|
||||||
|
*/
|
||||||
|
public void test45290() {
|
||||||
|
HSSFWorkbook wb = openSample("45290.xls");
|
||||||
|
assertEquals(1, wb.getNumberOfSheets());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1325,7 +1325,7 @@ public class TestBlockAllocationTableReader
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
ListManagedBlock[] dataBlocks =
|
ListManagedBlock[] dataBlocks =
|
||||||
table.fetchBlocks(start_blocks[ j ], list);
|
table.fetchBlocks(start_blocks[ j ], -1, list);
|
||||||
|
|
||||||
if (expected_length[ j ] == -1)
|
if (expected_length[ j ] == -1)
|
||||||
{
|
{
|
||||||
|
|
|
@ -285,7 +285,7 @@ public class TestBlockListImpl
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
ListManagedBlock[] dataBlocks =
|
ListManagedBlock[] dataBlocks =
|
||||||
list.fetchBlocks(start_blocks[ j ]);
|
list.fetchBlocks(start_blocks[ j ], -1);
|
||||||
|
|
||||||
if (expected_length[ j ] == -1)
|
if (expected_length[ j ] == -1)
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in New Issue