From 9d7c124057212989de04fce634d7343e4e7b2c0b Mon Sep 17 00:00:00 2001 From: Dominik Stadler Date: Sat, 2 Nov 2013 10:01:01 +0000 Subject: [PATCH] Fix Bug 54400 by updating the index in the LinkTable whenever sheets are removed. git-svn-id: https://svn.apache.org/repos/asf/poi/trunk@1538163 13f79535-47bb-0310-9956-ffa450edef68 --- .../poi/hssf/model/InternalWorkbook.java | 3 ++ .../org/apache/poi/hssf/model/LinkTable.java | 21 +++++--- .../poi/hssf/record/ExternSheetRecord.java | 13 +++++ .../poi/hssf/usermodel/TestHSSFWorkbook.java | 50 ++++++++++++++++++ test-data/spreadsheet/54500.xls | Bin 0 -> 11776 bytes 5 files changed, 79 insertions(+), 8 deletions(-) create mode 100644 test-data/spreadsheet/54500.xls diff --git a/src/java/org/apache/poi/hssf/model/InternalWorkbook.java b/src/java/org/apache/poi/hssf/model/InternalWorkbook.java index 24b30180d8..0e51637e91 100644 --- a/src/java/org/apache/poi/hssf/model/InternalWorkbook.java +++ b/src/java/org/apache/poi/hssf/model/InternalWorkbook.java @@ -716,6 +716,9 @@ public final class InternalWorkbook { // Bump down by one, so still points // at the same sheet nr.setSheetNumber(nr.getSheetNumber()-1); + + // also update the link-table as otherwise references might point at invalid sheets + linkTable.updateIndexToInternalSheet(i, -1); } } } diff --git a/src/java/org/apache/poi/hssf/model/LinkTable.java b/src/java/org/apache/poi/hssf/model/LinkTable.java index 1cec62de3c..3304ae2351 100644 --- a/src/java/org/apache/poi/hssf/model/LinkTable.java +++ b/src/java/org/apache/poi/hssf/model/LinkTable.java @@ -31,13 +31,17 @@ import org.apache.poi.hssf.record.NameCommentRecord; import org.apache.poi.hssf.record.NameRecord; import org.apache.poi.hssf.record.Record; import org.apache.poi.hssf.record.SupBookRecord; -import org.apache.poi.ss.formula.ptg.*; +import org.apache.poi.ss.formula.ptg.Area3DPtg; +import org.apache.poi.ss.formula.ptg.ErrPtg; +import org.apache.poi.ss.formula.ptg.NameXPtg; +import org.apache.poi.ss.formula.ptg.Ptg; +import org.apache.poi.ss.formula.ptg.Ref3DPtg; /** * Link Table (OOO pdf reference: 4.10.3 )

* * The main data of all types of references is stored in the Link Table inside the Workbook Globals - * Substream (4.2.5). The Link Table itself is optional and occurs only, if there are any + * Substream (4.2.5). The Link Table itself is optional and occurs only if there are any * references in the document. *

* @@ -63,9 +67,7 @@ import org.apache.poi.ss.formula.ptg.*; */ final class LinkTable { - // TODO make this class into a record aggregate - private static final class CRNBlock { private final CRNCountRecord _countRecord; @@ -174,7 +176,7 @@ final class LinkTable { private final int _recordCount; private final WorkbookRecordList _workbookRecordList; // TODO - would be nice to remove this - public LinkTable(List inputList, int startIndex, WorkbookRecordList workbookRecordList, Map commentRecords) { + public LinkTable(List inputList, int startIndex, WorkbookRecordList workbookRecordList, Map commentRecords) { _workbookRecordList = workbookRecordList; RecordStream rs = new RecordStream(inputList, startIndex); @@ -412,6 +414,10 @@ final class LinkTable { public int getIndexToInternalSheet(int extRefIndex) { return _externSheetRecord.getFirstSheetIndexFromRefIndex(extRefIndex); } + + public void updateIndexToInternalSheet(int extRefIndex, int offset) { + _externSheetRecord.adjustIndex(extRefIndex, offset); + } public int getSheetIndexFromExternSheetIndex(int extRefIndex) { if (extRefIndex >= _externSheetRecord.getNumOfRefs() || extRefIndex < 0) { @@ -442,7 +448,6 @@ final class LinkTable { return _externSheetRecord.addRef(thisWbIndex, sheetIndex, sheetIndex); } - /** * copied from Workbook */ @@ -533,8 +538,8 @@ final class LinkTable { int supLinkIndex = 0; // find the posistion of the Add-In SupBookRecord in the workbook stream, // the created ExternalNameRecord will be appended to it - for (Iterator iterator = _workbookRecordList.iterator(); iterator.hasNext(); supLinkIndex++) { - Record record = (Record) iterator.next(); + for (Iterator iterator = _workbookRecordList.iterator(); iterator.hasNext(); supLinkIndex++) { + Record record = iterator.next(); if (record instanceof SupBookRecord) { if (((SupBookRecord) record).isAddInFunctions()) break; } diff --git a/src/java/org/apache/poi/hssf/record/ExternSheetRecord.java b/src/java/org/apache/poi/hssf/record/ExternSheetRecord.java index 963e325ab9..4de0f60b02 100644 --- a/src/java/org/apache/poi/hssf/record/ExternSheetRecord.java +++ b/src/java/org/apache/poi/hssf/record/ExternSheetRecord.java @@ -41,6 +41,10 @@ public class ExternSheetRecord extends StandardRecord { private int _firstSheetIndex; // may be -1 (0xFFFF) private int _lastSheetIndex; // may be -1 (0xFFFF) + public void adjustIndex(int offset) { + _firstSheetIndex += offset; + _lastSheetIndex += offset; + } /** a Constructor for making new sub record */ @@ -66,6 +70,7 @@ public class ExternSheetRecord extends StandardRecord { return _lastSheetIndex; } + @Override public String toString() { StringBuffer buffer = new StringBuffer(); buffer.append("extBook=").append(_extBookIndex); @@ -122,6 +127,7 @@ public class ExternSheetRecord extends StandardRecord { } + @Override public String toString() { StringBuffer sb = new StringBuffer(); int nItems = _list.size(); @@ -138,10 +144,12 @@ public class ExternSheetRecord extends StandardRecord { return sb.toString(); } + @Override protected int getDataSize() { return 2 + _list.size() * RefSubRecord.ENCODED_SIZE; } + @Override public void serialize(LittleEndianOutput out) { int nItems = _list.size(); @@ -156,9 +164,14 @@ public class ExternSheetRecord extends StandardRecord { return _list.get(i); } + public void adjustIndex(int extRefIndex, int offset) { + getRef(extRefIndex).adjustIndex(offset); + } + /** * return the non static version of the id for this record. */ + @Override public short getSid() { return sid; } diff --git a/src/testcases/org/apache/poi/hssf/usermodel/TestHSSFWorkbook.java b/src/testcases/org/apache/poi/hssf/usermodel/TestHSSFWorkbook.java index 1927727e42..88fa84ac21 100644 --- a/src/testcases/org/apache/poi/hssf/usermodel/TestHSSFWorkbook.java +++ b/src/testcases/org/apache/poi/hssf/usermodel/TestHSSFWorkbook.java @@ -48,6 +48,7 @@ import org.apache.poi.poifs.filesystem.NPOIFSFileSystem; import org.apache.poi.poifs.filesystem.POIFSFileSystem; import org.apache.poi.ss.formula.ptg.Area3DPtg; import org.apache.poi.ss.usermodel.BaseTestWorkbook; +import org.apache.poi.ss.usermodel.Name; import org.apache.poi.ss.util.CellRangeAddress; import org.apache.poi.util.LittleEndian; import org.apache.poi.util.TempFile; @@ -965,4 +966,53 @@ public final class TestHSSFWorkbook extends BaseTestWorkbook { HSSFWorkbook read = HSSFTestDataSamples.writeOutAndReadBack(wb); assertSheetOrder(read, "Invoice", "Deferred", "Received", "Digest"); } + + public void testBug54500() throws Exception { + String nameName = "AName"; + String sheetName = "ASheet"; + HSSFWorkbook wb = HSSFTestDataSamples.openSampleWorkbook("54500.xls"); + + assertSheetOrder(wb, "Sheet1", "Sheet2", "Sheet3"); + + wb.createSheet(sheetName); + + assertSheetOrder(wb, "Sheet1", "Sheet2", "Sheet3", "ASheet"); + + Name n = wb.createName(); + n.setNameName(nameName); + n.setSheetIndex(3); + n.setRefersToFormula(sheetName + "!A1"); + + assertSheetOrder(wb, "Sheet1", "Sheet2", "Sheet3", "ASheet"); + assertEquals("ASheet!A1", wb.getName(nameName).getRefersToFormula()); + + ByteArrayOutputStream stream = new ByteArrayOutputStream(); + wb.write(stream); + + assertSheetOrder(wb, "Sheet1", "Sheet2", "Sheet3", "ASheet"); + assertEquals("ASheet!A1", wb.getName(nameName).getRefersToFormula()); + + wb.removeSheetAt(1); + + assertSheetOrder(wb, "Sheet1", "Sheet3", "ASheet"); + assertEquals("ASheet!A1", wb.getName(nameName).getRefersToFormula()); + + ByteArrayOutputStream stream2 = new ByteArrayOutputStream(); + wb.write(stream2); + + assertSheetOrder(wb, "Sheet1", "Sheet3", "ASheet"); + assertEquals("ASheet!A1", wb.getName(nameName).getRefersToFormula()); + + expectName( + new HSSFWorkbook(new ByteArrayInputStream(stream.toByteArray())), + nameName, "ASheet!A1"); + expectName( + new HSSFWorkbook( + new ByteArrayInputStream(stream2.toByteArray())), + nameName, "ASheet!A1"); + } + + private void expectName(HSSFWorkbook wb, String name, String expect) { + assertEquals(expect, wb.getName(name).getRefersToFormula()); + } } diff --git a/test-data/spreadsheet/54500.xls b/test-data/spreadsheet/54500.xls new file mode 100644 index 0000000000000000000000000000000000000000..0cd18b51d0c06e9b397be795b2cf5e8e0642f525 GIT binary patch literal 11776 zcmeHN-)me&6h3!TTLqb2&4! z-<_t7g z_2~Fnt4DvOb=GqKE4lu(19rsuLd-fOKknyxwsifa`Gw0?7+}HukQ~D9%gLl1)qObW zSFbDI2J3UP{I{@We7_jNG9WL=^LR#d?>~vR;FykyNM0t8KjEk@8Fxv*q7C)FeM>4; zft>$0E*6%(=wVdHFs^Ul&5dW*GUm^&2VKL@caeeVbvI=>u^WXup+gE=A#D)a{*92k zAYPj{s&C3}4Fu7fxQRe z`^EBI5KL3&&hcg-w53NN9M24dT#+L`C**Da#Aq%%ojsPf_hfS?v$>J{M3&zv&V@Xe z%lFNUj!$XZ^3K)GU0;4>dGn#}S|2}oWXLe8njaw(3GHFkl!k3>XFs1BL;^ zfMLKeU>JbkCIW9kLs$lzi;bsTLgwLZ_&9V)eGWeZ8|sZ_ErRq1lDS1CNE zV;HgM_jtr=fWIUFKg@Z=EwWihM$bR}BaAPq-q(&cd?+(`?(HA!q=$vRmcxOJ>b9w~Uw?Z*&J&Id_3ETus$?BE1hqmG#0^YJ%XRIU9(UBbP z?jP#%ehhWDhwg*VXlq6Jvj@k=o~u-fVUf0#m3otv?cg_?+78!PtXiw8HCOc^i3?n2 z!yHQ7_D1IZ9O_bFsJq_!uNSxdmEv|qjkF}>;mXdO8C3SC zL-Tc+PM16FlOC*6xl^zH-~D=(wd-=tIlZ|M`J-fWySQQ2CeDqZ`0w@M+>J}=*TUKA zQu?HQrqVPt?HUHSnf?;O7D%i57w-rmckrP1_0~O*a%8HI;&B zy0H*09f7T||aj0l|D zCKv_`1OGDyXfT}9SG3~vwJyN>n{w!bVzZuA)*~Ln?Ha96tDab*uqF#1{QCX=qX*K% zipLf2(fD;m3OaWZjW6(U01etI3;#{vqBvujOM7Xt_zZM$ zV-?VUQaG^2y8~_Vnf_%k<3qP$%0y