blob: 387980d8709cd867d28c0a99975dbbc76503673c [file] [log] [blame]
/* ====================================================================
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.ss.util;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertNotEquals;
import org.apache.poi.hssf.record.FormulaRecord;
import org.apache.poi.ss.formula.constant.ConstantValueParser;
import org.apache.poi.ss.formula.ptg.NumberPtg;
import org.apache.poi.ss.util.NumberToTextConversionExamples.ExampleConversion;
import org.junit.jupiter.api.Test;
/**
* Tests for {@link NumberToTextConverter}
*/
final class TestNumberToTextConverter {
/**
* Confirms that <tt>ExcelNumberToTextConverter.toText(d)</tt> produces the right results.
* As part of preparing this test class, the <tt>ExampleConversion</tt> instances should be set
* up to contain the rendering as produced by Excel.
*/
@Test
void testAll() {
ExampleConversion[] examples = NumberToTextConversionExamples.getExampleConversions();
for (ExampleConversion example : examples) {
if (example.isNaN()) {
confirmNaN(example.getRawDoubleBits(), example.getExcelRendering());
continue;
}
String actual = NumberToTextConverter.toText(example.getDoubleValue());
assertEquals(example.getExcelRendering(), actual);
}
}
/**
* Excel's abnormal rendering of NaNs is both difficult to test and even reproduce in java. In
* general, Excel does not attempt to use raw NaN in the IEEE sense. In {@link FormulaRecord}s,
* Excel uses the NaN bit pattern to flag non-numeric (text, boolean, error) cached results.
* If the formula result actually evaluates to raw NaN, Excel transforms it to <i>#NUM!</i>.
* In other places (e.g. {@link NumberRecord}, {@link NumberPtg}, array items (via {@link
* ConstantValueParser}), there seems to be no special NaN translation scheme. If a NaN bit
* pattern is somehow encoded into any of these places Excel actually attempts to render the
* values as a plain number. That is the unusual functionality that this method is testing.<p>
*
* There are multiple encodings (bit patterns) for NaN, and CPUs and applications can convert
* to a preferred NaN encoding (Java prefers <tt>0x7FF8000000000000L</tt>). Besides the
* special encoding in {@link FormulaRecord.SpecialCachedValue}, it is not known how/whether
* Excel attempts to encode NaN values.
*
* Observed NaN behaviour on HotSpot/Windows:
* <tt>Double.longBitsToDouble()</tt> will set one bit 51 (the NaN signaling flag) if it isn't
* already. <tt>Double.doubleToLongBits()</tt> will return a double with bit pattern
* <tt>0x7FF8000000000000L</tt> for any NaN bit pattern supplied.<br>
* Differences are likely to be observed with other architectures.<p>
*
* <p>
* The few test case examples calling this method represent functionality which may not be
* important for POI to support.
*/
private void confirmNaN(long l, String excelRep) {
double d = Double.longBitsToDouble(l);
assertEquals("NaN", Double.toString(d));
String strExcel = NumberToTextConverter.rawDoubleBitsToText(l);
assertEquals(excelRep, strExcel);
}
@Test
void testSimpleRendering_bug56156() {
double dResult = 0.05+0.01; // values chosen to produce rounding anomaly
String actualText = NumberToTextConverter.toText(dResult);
String jdkText = Double.toString(dResult);
// "0.060000000000000005"
assertNotEquals("Should not use default JDK IEEE double rendering", jdkText, actualText);
assertEquals("0.06", actualText);
}
}