From 98a378dc5f4f7d48c0b4cb85db2e16b2626daf59 Mon Sep 17 00:00:00 2001
From: Josh Micich <josh@apache.org>
Date: Sat, 15 Nov 2008 17:25:32 +0000
Subject: [PATCH] Fix for bug 46213 - FormulaRecordAggregate should gracefully
 ignore extra StringRecords

git-svn-id: https://svn.apache.org/repos/asf/poi/trunk@717883 13f79535-47bb-0310-9956-ffa450edef68
---
 src/documentation/content/xdocs/changes.xml   |  1 +
 src/documentation/content/xdocs/status.xml    |  1 +
 .../aggregates/FormulaRecordAggregate.java    | 20 +++----
 .../TestFormulaRecordAggregate.java           | 57 +++++++++++++++----
 .../poi/hssf/usermodel/RecordInspector.java   |  8 +--
 5 files changed, 62 insertions(+), 25 deletions(-)

diff --git a/src/documentation/content/xdocs/changes.xml b/src/documentation/content/xdocs/changes.xml
index 8257ad5ef1..bb45c63692 100644
--- a/src/documentation/content/xdocs/changes.xml
+++ b/src/documentation/content/xdocs/changes.xml
@@ -37,6 +37,7 @@
 
 		<!-- Don't forget to update status.xml too! -->
         <release version="3.5-beta4" date="2008-??-??">
+           <action dev="POI-DEVELOPERS" type="fix">46213 - Fixed FormulaRecordAggregate to gracefully ignore extra StringRecords</action>
            <action dev="POI-DEVELOPERS" type="fix">46174 - Fixed HSSFName to handle general formulas (not just area references)</action>
            <action dev="POI-DEVELOPERS" type="add">46189 - added chart records: CHARTFRTINFO, STARTBLOCK, ENDBLOCK, STARTOBJECT, ENDOBJECT, and CATLAB</action>
            <action dev="POI-DEVELOPERS" type="fix">46199 - More tweaks to EmbeddedObjectRefSubRecord</action>
diff --git a/src/documentation/content/xdocs/status.xml b/src/documentation/content/xdocs/status.xml
index d3f1ed911b..57a0616d78 100644
--- a/src/documentation/content/xdocs/status.xml
+++ b/src/documentation/content/xdocs/status.xml
@@ -34,6 +34,7 @@
 	<!-- Don't forget to update changes.xml too! -->
     <changes>
         <release version="3.5-beta4" date="2008-??-??">
+           <action dev="POI-DEVELOPERS" type="fix">46213 - Fixed FormulaRecordAggregate to gracefully ignore extra StringRecords</action>
            <action dev="POI-DEVELOPERS" type="fix">46174 - Fixed HSSFName to handle general formulas (not just area references)</action>
            <action dev="POI-DEVELOPERS" type="add">46189 - added chart records: CHARTFRTINFO, STARTBLOCK, ENDBLOCK, STARTOBJECT, ENDOBJECT, and CATLAB</action>
            <action dev="POI-DEVELOPERS" type="fix">46199 - More tweaks to EmbeddedObjectRefSubRecord</action>
diff --git a/src/java/org/apache/poi/hssf/record/aggregates/FormulaRecordAggregate.java b/src/java/org/apache/poi/hssf/record/aggregates/FormulaRecordAggregate.java
index 18653e952e..83a4ce7e0e 100644
--- a/src/java/org/apache/poi/hssf/record/aggregates/FormulaRecordAggregate.java
+++ b/src/java/org/apache/poi/hssf/record/aggregates/FormulaRecordAggregate.java
@@ -26,7 +26,6 @@ import org.apache.poi.hssf.record.StringRecord;
 import org.apache.poi.hssf.record.formula.ExpPtg;
 import org.apache.poi.hssf.record.formula.Ptg;
 import org.apache.poi.hssf.util.CellReference;
-import org.apache.poi.ss.formula.Formula;
 
 /**
  * The formula record aggregate is used to join together the formula record and it's
@@ -51,17 +50,19 @@ public final class FormulaRecordAggregate extends RecordAggregate implements Cel
 		if (svm == null) {
 			throw new IllegalArgumentException("sfm must not be null");
 		}
-		boolean hasStringRec = stringRec != null;
-		boolean hasCachedStringFlag = formulaRec.hasCachedResultString();
-		if (hasStringRec != hasCachedStringFlag) {
-			throw new RecordFormatException("String record was "
-					+ (hasStringRec ? "": "not ") + " supplied but formula record flag is "
-					+ (hasCachedStringFlag ? "" : "not ") + " set");
+		if (formulaRec.hasCachedResultString()) {
+			if (stringRec == null) {
+				throw new RecordFormatException("Formula record flag is set but String record was not found");
+			}
+			_stringRecord = stringRec;
+		} else {
+			// Usually stringRec is null here (in agreement with what the formula rec says).
+			// In the case where an extra StringRecord is erroneously present, Excel (2007)
+			// ignores it (see bug 46213). 
+			_stringRecord = null;
 		}
 
 		if (formulaRec.isSharedFormula()) {
-			CellReference cr = new CellReference(formulaRec.getRow(), formulaRec.getColumn());
-			
 			CellReference firstCell = formulaRec.getFormula().getExpReference();
 			if (firstCell == null) {
 				handleMissingSharedFormulaRecord(formulaRec);
@@ -71,7 +72,6 @@ public final class FormulaRecordAggregate extends RecordAggregate implements Cel
 		}
 		_formulaRecord = formulaRec;
 		_sharedValueManager = svm;
-		_stringRecord = stringRec;
 	}
 	/**
 	 * Sometimes the shared formula flag "seems" to be erroneously set (because the corresponding
diff --git a/src/testcases/org/apache/poi/hssf/record/aggregates/TestFormulaRecordAggregate.java b/src/testcases/org/apache/poi/hssf/record/aggregates/TestFormulaRecordAggregate.java
index 978f400fcd..a93bed20c2 100644
--- a/src/testcases/org/apache/poi/hssf/record/aggregates/TestFormulaRecordAggregate.java
+++ b/src/testcases/org/apache/poi/hssf/record/aggregates/TestFormulaRecordAggregate.java
@@ -17,23 +17,58 @@
 
 package org.apache.poi.hssf.record.aggregates;
 
+import junit.framework.AssertionFailedError;
 import junit.framework.TestCase;
 
 import org.apache.poi.hssf.record.FormulaRecord;
+import org.apache.poi.hssf.record.Record;
+import org.apache.poi.hssf.record.RecordFormatException;
 import org.apache.poi.hssf.record.StringRecord;
+import org.apache.poi.hssf.usermodel.RecordInspector.RecordCollector;
 
 /**
- *
- * @author  avik
+ * 
+ * @author avik
  */
 public final class TestFormulaRecordAggregate extends TestCase {
-    
-    public void testBasic() throws Exception {
-        FormulaRecord f = new FormulaRecord();
-        f.setCachedResultTypeString();
-        StringRecord s = new StringRecord();
-        s.setString("abc");
-        FormulaRecordAggregate fagg = new FormulaRecordAggregate(f, s, SharedValueManager.EMPTY);
-        assertEquals("abc", fagg.getStringValue());
-    }
+
+	public void testBasic() throws Exception {
+		FormulaRecord f = new FormulaRecord();
+		f.setCachedResultTypeString();
+		StringRecord s = new StringRecord();
+		s.setString("abc");
+		FormulaRecordAggregate fagg = new FormulaRecordAggregate(f, s, SharedValueManager.EMPTY);
+		assertEquals("abc", fagg.getStringValue());
+	}
+
+	/**
+	 * Sometimes a {@link StringRecord} appears after a {@link FormulaRecord} even though the
+	 * formula has evaluated to a text value.  This might be more likely to occur when the formula
+	 * <i>can</i> evaluate to a text value.<br/>
+	 * Bug 46213 attachment 22874 has such an extra {@link StringRecord} at stream offset 0x5765.
+	 * This file seems to open in Excel (2007) with no trouble.  When it is re-saved, Excel omits
+	 * the extra record.  POI should do the same.
+	 */
+	public void testExtraStringRecord_bug46213() {
+		FormulaRecord fr = new FormulaRecord();
+		fr.setValue(2.0);
+		StringRecord sr = new StringRecord();
+		sr.setString("NA");
+		SharedValueManager svm = SharedValueManager.EMPTY;
+		FormulaRecordAggregate fra;
+		
+		try {
+			fra = new FormulaRecordAggregate(fr, sr, svm);
+		} catch (RecordFormatException e) {
+			if ("String record was  supplied but formula record flag is not  set".equals(e.getMessage())) {
+				throw new AssertionFailedError("Identified bug 46213");
+			}
+			throw e;
+		}
+		RecordCollector rc = new RecordCollector();
+		fra.visitContainedRecords(rc);
+		Record[] vraRecs = rc.getRecords();
+		assertEquals(1, vraRecs.length);
+		assertEquals(fr, vraRecs[0]);
+	}
 }
diff --git a/src/testcases/org/apache/poi/hssf/usermodel/RecordInspector.java b/src/testcases/org/apache/poi/hssf/usermodel/RecordInspector.java
index 867c13db4a..c47fca7629 100644
--- a/src/testcases/org/apache/poi/hssf/usermodel/RecordInspector.java
+++ b/src/testcases/org/apache/poi/hssf/usermodel/RecordInspector.java
@@ -24,7 +24,7 @@ import org.apache.poi.hssf.record.Record;
 import org.apache.poi.hssf.record.aggregates.RecordAggregate.RecordVisitor;
 
 /**
- * Test utility class to get {@link Record}s out HSSF objects
+ * Test utility class to get {@link Record}s out of HSSF objects
  * 
  * @author Josh Micich
  */
@@ -34,12 +34,12 @@ public final class RecordInspector {
 		// no instances of this class
 	}
 
-	private static final class RecordCollector implements RecordVisitor {
+	public static final class RecordCollector implements RecordVisitor {
 
-		private List _list;
+		private List<Record> _list;
 
 		public RecordCollector() {
-			_list = new ArrayList(128);
+			_list = new ArrayList<Record>(128);
 		}
 
 		public void visitRecord(Record r) {