Bugzilla 47001 - Fixed WriteAccessRecord and LinkTable to handle unusual format written by Google Docs

git-svn-id: https://svn.apache.org/repos/asf/poi/trunk@763391 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Josh Micich 2009-04-08 21:01:43 +00:00
parent 0c1095db76
commit 9fa24eea2c
12 changed files with 185 additions and 25 deletions

View File

@ -37,6 +37,7 @@
<!-- Don't forget to update status.xml too! --> <!-- Don't forget to update status.xml too! -->
<release version="3.5-beta6" date="2009-??-??"> <release version="3.5-beta6" date="2009-??-??">
<action dev="POI-DEVELOPERS" type="fix">47001 - Fixed WriteAccessRecord and LinkTable to handle unusual format written by Google Docs</action>
<action dev="POI-DEVELOPERS" type="fix">46973 - Fixed defined names to behave better when refersToFormula is unset</action> <action dev="POI-DEVELOPERS" type="fix">46973 - Fixed defined names to behave better when refersToFormula is unset</action>
<action dev="POI-DEVELOPERS" type="fix">46832 - Allow merged regions with columns greater than 255 or rows bigger than 65536 in XSSF</action> <action dev="POI-DEVELOPERS" type="fix">46832 - Allow merged regions with columns greater than 255 or rows bigger than 65536 in XSSF</action>
<action dev="POI-DEVELOPERS" type="fix">46951 - Fixed formula parser to better handle range operators and whole row/column refs.</action> <action dev="POI-DEVELOPERS" type="fix">46951 - Fixed formula parser to better handle range operators and whole row/column refs.</action>

View File

@ -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-beta6" date="2009-??-??"> <release version="3.5-beta6" date="2009-??-??">
<action dev="POI-DEVELOPERS" type="fix">47001 - Fixed WriteAccessRecord and LinkTable to handle unusual format written by Google Docs</action>
<action dev="POI-DEVELOPERS" type="fix">46973 - Fixed defined names to behave better when refersToFormula is unset</action> <action dev="POI-DEVELOPERS" type="fix">46973 - Fixed defined names to behave better when refersToFormula is unset</action>
<action dev="POI-DEVELOPERS" type="fix">46832 - Allow merged regions with columns greater than 255 or rows bigger than 65536 in XSSF</action> <action dev="POI-DEVELOPERS" type="fix">46832 - Allow merged regions with columns greater than 255 or rows bigger than 65536 in XSSF</action>
<action dev="POI-DEVELOPERS" type="fix">46951 - Fixed formula parser to better handle range operators and whole row/column refs.</action> <action dev="POI-DEVELOPERS" type="fix">46951 - Fixed formula parser to better handle range operators and whole row/column refs.</action>

View File

@ -161,7 +161,12 @@ final class LinkTable {
if (_externalBookBlocks.length > 0) { if (_externalBookBlocks.length > 0) {
// If any ExternalBookBlock present, there is always 1 of ExternSheetRecord // If any ExternalBookBlock present, there is always 1 of ExternSheetRecord
_externSheetRecord = readExtSheetRecord(rs); if (rs.peekNextClass() != ExternSheetRecord.class) {
// not quite - if written by google docs
_externSheetRecord = null;
} else {
_externSheetRecord = readExtSheetRecord(rs);
}
} else { } else {
_externSheetRecord = null; _externSheetRecord = null;
} }

View File

@ -19,6 +19,7 @@ package org.apache.poi.hssf.record;
import java.util.Arrays; import java.util.Arrays;
import org.apache.poi.util.LittleEndian;
import org.apache.poi.util.LittleEndianOutput; import org.apache.poi.util.LittleEndianOutput;
import org.apache.poi.util.StringUtil; import org.apache.poi.util.StringUtil;
@ -40,11 +41,13 @@ public final class WriteAccessRecord extends StandardRecord {
private static final int DATA_SIZE = 112; private static final int DATA_SIZE = 112;
private String field_1_username; private String field_1_username;
/** this record is always padded to a constant length */ /** this record is always padded to a constant length */
private byte[] padding; private static final byte[] PADDING = new byte[DATA_SIZE];
static {
Arrays.fill(PADDING, PAD_CHAR);
}
public WriteAccessRecord() { public WriteAccessRecord() {
setUsername(""); setUsername("");
padding = new byte[DATA_SIZE - 3];
} }
public WriteAccessRecord(RecordInputStream in) { public WriteAccessRecord(RecordInputStream in) {
@ -57,21 +60,33 @@ public final class WriteAccessRecord extends StandardRecord {
int nChars = in.readUShort(); int nChars = in.readUShort();
int is16BitFlag = in.readUByte(); int is16BitFlag = in.readUByte();
int expectedPadSize = DATA_SIZE - 3; if (nChars > DATA_SIZE || (is16BitFlag & 0xFE) != 0) {
if ((is16BitFlag & 0x01) == 0x00) { // String header looks wrong (probably missing)
field_1_username = StringUtil.readCompressedUnicode(in, nChars); // OOO doc says this is optional anyway.
expectedPadSize -= nChars; // reconstruct data
} else { byte[] data = new byte[3 + in.remaining()];
field_1_username = StringUtil.readUnicodeLE(in, nChars); LittleEndian.putUShort(data, 0, nChars);
expectedPadSize -= nChars * 2; LittleEndian.putByte(data, 2, is16BitFlag);
in.readFully(data, 3, data.length-3);
String rawValue = new String(data);
setUsername(rawValue.trim());
return;
} }
padding = new byte[expectedPadSize];
String rawText;
if ((is16BitFlag & 0x01) == 0x00) {
rawText = StringUtil.readCompressedUnicode(in, nChars);
} else {
rawText = StringUtil.readUnicodeLE(in, nChars);
}
field_1_username = rawText.trim();
// consume padding
int padSize = in.remaining(); int padSize = in.remaining();
in.readFully(padding, 0, padSize); while (padSize > 0) {
if (padSize < expectedPadSize) { // in some cases this seems to be garbage (non spaces)
// this occurs in a couple of test examples: "42564.xls", in.readUByte();
// "bug_42794.xls" padSize--;
Arrays.fill(padding, padSize, expectedPadSize, PAD_CHAR);
} }
} }
@ -88,8 +103,6 @@ public final class WriteAccessRecord extends StandardRecord {
if (paddingSize < 0) { if (paddingSize < 0) {
throw new IllegalArgumentException("Name is too long: " + username); throw new IllegalArgumentException("Name is too long: " + username);
} }
padding = new byte[paddingSize];
Arrays.fill(padding, PAD_CHAR);
field_1_username = username; field_1_username = username;
} }
@ -109,7 +122,7 @@ public final class WriteAccessRecord extends StandardRecord {
StringBuffer buffer = new StringBuffer(); StringBuffer buffer = new StringBuffer();
buffer.append("[WRITEACCESS]\n"); buffer.append("[WRITEACCESS]\n");
buffer.append(" .name = ").append(field_1_username.toString()).append("\n"); buffer.append(" .name = ").append(field_1_username.toString()).append("\n");
buffer.append("[/WRITEACCESS]\n"); buffer.append("[/WRITEACCESS]\n");
return buffer.toString(); return buffer.toString();
} }
@ -125,7 +138,9 @@ public final class WriteAccessRecord extends StandardRecord {
} else { } else {
StringUtil.putCompressedUnicode(username, out); StringUtil.putCompressedUnicode(username, out);
} }
out.write(padding); int encodedByteCount = 3 + username.length() * (is16bit ? 2 : 1);
int paddingSize = DATA_SIZE - encodedByteCount;
out.write(PADDING, 0, paddingSize);
} }
protected int getDataSize() { protected int getDataSize() {

View File

@ -34,6 +34,7 @@ public final class AllModelTests {
result.addTestSuite(TestFormulaParser.class); result.addTestSuite(TestFormulaParser.class);
result.addTestSuite(TestFormulaParserEval.class); result.addTestSuite(TestFormulaParserEval.class);
result.addTestSuite(TestFormulaParserIf.class); result.addTestSuite(TestFormulaParserIf.class);
result.addTestSuite(TestLinkTable.class);
result.addTestSuite(TestOperandClassTransformer.class); result.addTestSuite(TestOperandClassTransformer.class);
result.addTestSuite(TestRowBlocksReader.class); result.addTestSuite(TestRowBlocksReader.class);
result.addTestSuite(TestRVA.class); result.addTestSuite(TestRVA.class);

View File

@ -15,12 +15,20 @@
limitations under the License. limitations under the License.
==================================================================== */ ==================================================================== */
package org.apache.poi.hssf.usermodel; package org.apache.poi.hssf.model;
import java.util.Arrays;
import java.util.List;
import junit.framework.AssertionFailedError; import junit.framework.AssertionFailedError;
import junit.framework.TestCase; import junit.framework.TestCase;
import org.apache.poi.hssf.HSSFTestDataSamples; import org.apache.poi.hssf.HSSFTestDataSamples;
import org.apache.poi.hssf.record.Record;
import org.apache.poi.hssf.record.SSTRecord;
import org.apache.poi.hssf.record.SupBookRecord;
import org.apache.poi.hssf.usermodel.HSSFCell;
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
/** /**
* Tests for {@link LinkTable} * Tests for {@link LinkTable}
* *
@ -81,7 +89,7 @@ public final class TestLinkTable extends TestCase {
The original file produces the same error. The original file produces the same error.
This bug was caused by a combination of invalid sheet indexes in the EXTERNSHEET This bug was caused by a combination of invalid sheet indexes in the EXTERNSHEET
record, and eager initialisation of the extern sheet references. Note - the worbook record, and eager initialisation of the extern sheet references. Note - the workbook
has 2 sheets, but the EXTERNSHEET record refers to sheet indexes 0, 1 and 2. has 2 sheets, but the EXTERNSHEET record refers to sheet indexes 0, 1 and 2.
Offset 0x3954 (14676) Offset 0x3954 (14676)
@ -114,4 +122,30 @@ public final class TestLinkTable extends TestCase {
} }
assertEquals("Data!$A2", cellFormula); assertEquals("Data!$A2", cellFormula);
} }
/**
* This problem was visible in POI svn r763332
* when reading the workbook of attachment 23468 from bugzilla 47001
*/
public void testMissingExternSheetRecord_bug47001b() {
Record[] recs = {
SupBookRecord.createAddInFunctions(),
new SSTRecord(),
};
List<Record> recList = Arrays.asList(recs);
WorkbookRecordList wrl = new WorkbookRecordList();
LinkTable lt;
try {
lt = new LinkTable(recList, 0, wrl);
} catch (RuntimeException e) {
if (e.getMessage().equals("Expected an EXTERNSHEET record but got (org.apache.poi.hssf.record.SSTRecord)")) {
throw new AssertionFailedError("Identified bug 47001b");
}
throw e;
}
assertNotNull(lt);
}
} }

View File

@ -80,6 +80,7 @@ public final class AllRecordTests {
result.addTestSuite(TestTextObjectRecord.class); result.addTestSuite(TestTextObjectRecord.class);
result.addTestSuite(TestUnicodeNameRecord.class); result.addTestSuite(TestUnicodeNameRecord.class);
result.addTestSuite(TestUnicodeString.class); result.addTestSuite(TestUnicodeString.class);
result.addTestSuite(TestWriteAccessRecord.class);
result.addTestSuite(TestCellRange.class); result.addTestSuite(TestCellRange.class);
result.addTestSuite(TestConstantValueParser.class); result.addTestSuite(TestConstantValueParser.class);
return result; return result;

View File

@ -0,0 +1,103 @@
/* ====================================================================
Licensed to the Apache Software Foundation (ASF) under one or more
contributor license agreements. See the NOTICE file distributed with
this work for additional information regarding copyright ownership.
The ASF licenses this file to You under the Apache License, Version 2.0
(the "License"); you may not use this file except in compliance with
the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
==================================================================== */
package org.apache.poi.hssf.record;
import junit.framework.AssertionFailedError;
import junit.framework.TestCase;
import org.apache.poi.util.HexRead;
/**
* Tests for {@link WriteAccessRecord}
*
* @author Josh Micich
*/
public final class TestWriteAccessRecord extends TestCase {
private static final String HEX_SIXTYFOUR_SPACES = ""
+ "20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 "
+ "20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 "
+ "20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 "
+ "20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20";
public void testMissingStringHeader_bug47001a() {
/*
* Data taken from offset 0x0224 in
* attachment 23468 from bugzilla 47001
*/
byte[] data = HexRead.readFromString(""
+ "5C 00 70 00 "
+ "4A 61 76 61 20 45 78 63 65 6C 20 41 50 49 20 76 "
+ "32 2E 36 2E 34"
+ "20 20 20 20 20 20 20 20 20 20 20 "
+ "20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 "
+ HEX_SIXTYFOUR_SPACES);
RecordInputStream in = TestcaseRecordInputStream.create(data);
WriteAccessRecord rec;
try {
rec = new WriteAccessRecord(in);
} catch (RecordFormatException e) {
if (e.getMessage().equals("Not enough data (0) to read requested (1) bytes")) {
throw new AssertionFailedError("Identified bug 47001a");
}
throw e;
}
assertEquals("Java Excel API v2.6.4", rec.getUsername());
byte[] expectedEncoding = HexRead.readFromString(""
+ "15 00 00 4A 61 76 61 20 45 78 63 65 6C 20 41 50 "
+ "49 20 76 32 2E 36 2E 34"
+ "20 20 20 20 20 20 20 20 "
+ "20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 "
+ HEX_SIXTYFOUR_SPACES);
TestcaseRecordInputStream.confirmRecordEncoding(WriteAccessRecord.sid, expectedEncoding, rec.serialize());
}
public void testShortRecordWrittenByMSAccess() {
/*
* Data taken from two example files
* ex42564-21435.xls
* bug_42794.xls (from bug 42794 attachment 20429)
* In both cases, this data is found at offset 0x0C1C.
*/
byte[] data = HexRead.readFromString(""
+ "5C 00 39 00 "
+ "36 00 00 41 20 73 61 74 69 73 66 69 65 64 20 4D "
+ "69 63 72 6F 73 6F 66 74 20 4F 66 66 69 63 65 39 "
+ "20 55 73 65 72"
+ "20 20 20 20 20 20 20 20 20 20 20 "
+ "20 20 20 20 20 20 20 20 20");
RecordInputStream in = TestcaseRecordInputStream.create(data);
WriteAccessRecord rec = new WriteAccessRecord(in);
assertEquals("A satisfied Microsoft Office9 User", rec.getUsername());
byte[] expectedEncoding = HexRead.readFromString(""
+ "22 00 00 41 20 73 61 74 69 73 66 69 65 64 20 4D "
+ "69 63 72 6F 73 6F 66 74 20 4F 66 66 69 63 65 39 "
+ "20 55 73 65 72"
+ "20 20 20 20 20 20 20 20 20 20 20 "
+ HEX_SIXTYFOUR_SPACES);
TestcaseRecordInputStream.confirmRecordEncoding(WriteAccessRecord.sid, expectedEncoding, rec.serialize());
}
}

View File

@ -59,7 +59,6 @@ public class AllUserModelTests {
result.addTestSuite(TestHSSFSheet.class); result.addTestSuite(TestHSSFSheet.class);
result.addTestSuite(TestHSSFTextbox.class); result.addTestSuite(TestHSSFTextbox.class);
result.addTestSuite(TestHSSFWorkbook.class); result.addTestSuite(TestHSSFWorkbook.class);
result.addTestSuite(TestLinkTable.class);
result.addTestSuite(TestHSSFName.class); result.addTestSuite(TestHSSFName.class);
result.addTestSuite(TestOLE2Embeding.class); result.addTestSuite(TestOLE2Embeding.class);
result.addTestSuite(TestPOIFSProperties.class); result.addTestSuite(TestPOIFSProperties.class);

View File

@ -586,7 +586,7 @@ public final class TestBugs extends BaseTestBugzillaIssues {
* when reading the BOFRecord * when reading the BOFRecord
*/ */
public void test42564() { public void test42564() {
HSSFWorkbook wb = openSample("42564.xls"); HSSFWorkbook wb = openSample("ex42564-21435.xls");
writeOutAndReadBack(wb); writeOutAndReadBack(wb);
} }
@ -596,7 +596,7 @@ public final class TestBugs extends BaseTestBugzillaIssues {
* issue. * issue.
*/ */
public void test42564Alt() { public void test42564Alt() {
HSSFWorkbook wb = openSample("42564-2.xls"); HSSFWorkbook wb = openSample("ex42564-21503.xls");
writeOutAndReadBack(wb); writeOutAndReadBack(wb);
} }