Skip to content

Commit d22d890

Browse files
author
Justin Lu
committedFeb 26, 2024
8325898: ChoiceFormat returns erroneous result when formatting bad pattern
Reviewed-by: naoto
1 parent 93feda3 commit d22d890

File tree

2 files changed

+115
-112
lines changed

2 files changed

+115
-112
lines changed
 

‎src/java.base/share/classes/java/text/ChoiceFormat.java

+101-107
Original file line numberDiff line numberDiff line change
@@ -41,7 +41,9 @@
4141
import java.io.InvalidObjectException;
4242
import java.io.IOException;
4343
import java.io.ObjectInputStream;
44+
import java.util.ArrayList;
4445
import java.util.Arrays;
46+
import java.util.Objects;
4547

4648
/**
4749
* {@code ChoiceFormat} is a concrete subclass of {@code NumberFormat} that
@@ -238,6 +240,7 @@ public class ChoiceFormat extends NumberFormat {
238240
* @see #ChoiceFormat(String)
239241
*/
240242
public void applyPattern(String newPattern) {
243+
Objects.requireNonNull(newPattern, "newPattern must not be null");
241244
applyPatternImpl(newPattern);
242245
}
243246

@@ -249,86 +252,92 @@ public void applyPattern(String newPattern) {
249252
* further understanding of certain special characters: "#", "<", "\u2264", "|".
250253
*/
251254
private void applyPatternImpl(String newPattern) {
252-
StringBuilder[] segments = new StringBuilder[2];
253-
for (int i = 0; i < segments.length; ++i) {
254-
segments[i] = new StringBuilder();
255-
}
256-
double[] newChoiceLimits = new double[30];
257-
String[] newChoiceFormats = new String[30];
258-
int count = 0;
259-
int part = 0; // 0 denotes limit, 1 denotes format
260-
double startValue = 0;
261-
double oldStartValue = Double.NaN;
255+
// Set up components
256+
ArrayList<Double> limits = new ArrayList<>();
257+
ArrayList<String> formats = new ArrayList<>();
258+
StringBuilder[] segments = new StringBuilder[]{new StringBuilder(),
259+
new StringBuilder()};
260+
int part = 0; // 0 denotes LIMIT. 1 denotes FORMAT.
261+
double limit = 0;
262262
boolean inQuote = false;
263+
264+
// Parse the string, alternating the value of part
263265
for (int i = 0; i < newPattern.length(); ++i) {
264266
char ch = newPattern.charAt(i);
265-
if (ch=='\'') {
266-
// Check for "''" indicating a literal quote
267-
if ((i+1)<newPattern.length() && newPattern.charAt(i+1)==ch) {
267+
switch (ch) {
268+
case '\'':
269+
// Check for "''" indicating a literal quote
270+
if ((i + 1) < newPattern.length() && newPattern.charAt(i + 1) == ch) {
271+
segments[part].append(ch);
272+
++i;
273+
} else {
274+
inQuote = !inQuote;
275+
}
276+
break;
277+
case '<', '#', '\u2264':
278+
if (inQuote || part == 1) {
279+
// Don't interpret relational symbols if parsing the format
280+
segments[part].append(ch);
281+
} else {
282+
// Build the numerical value of the limit
283+
// and switch to parsing format
284+
if (segments[0].isEmpty()) {
285+
throw new IllegalArgumentException("Each interval must" +
286+
" contain a number before a format");
287+
}
288+
limit = stringToNum(segments[0].toString());
289+
if (ch == '<' && Double.isFinite(limit)) {
290+
limit = nextDouble(limit);
291+
}
292+
if (!limits.isEmpty() && limit <= limits.getLast()) {
293+
throw new IllegalArgumentException("Incorrect order " +
294+
"of intervals, must be in ascending order");
295+
}
296+
segments[0].setLength(0);
297+
part = 1;
298+
}
299+
break;
300+
case '|':
301+
if (inQuote) {
302+
segments[part].append(ch);
303+
} else {
304+
if (part != 1) {
305+
// Discard incorrect portion and finish building cFmt
306+
break;
307+
}
308+
// Insert an entry into the format and limit arrays
309+
// and switch to parsing limit
310+
limits.add(limit);
311+
formats.add(segments[1].toString());
312+
segments[1].setLength(0);
313+
part = 0;
314+
}
315+
break;
316+
default:
268317
segments[part].append(ch);
269-
++i;
270-
} else {
271-
inQuote = !inQuote;
272-
}
273-
} else if (inQuote) {
274-
segments[part].append(ch);
275-
} else if (part == 0 && (ch == '<' || ch == '#' || ch == '\u2264')) {
276-
// Only consider relational symbols if parsing the limit segment (part == 0).
277-
// Don't treat a relational symbol as syntactically significant
278-
// when parsing Format segment (part == 1)
279-
if (segments[0].length() == 0) {
280-
throw new IllegalArgumentException("Each interval must"
281-
+ " contain a number before a format");
282-
}
283-
284-
String tempBuffer = segments[0].toString();
285-
if (tempBuffer.equals("\u221E")) {
286-
startValue = Double.POSITIVE_INFINITY;
287-
} else if (tempBuffer.equals("-\u221E")) {
288-
startValue = Double.NEGATIVE_INFINITY;
289-
} else {
290-
startValue = Double.parseDouble(tempBuffer);
291-
}
292-
293-
if (ch == '<' && startValue != Double.POSITIVE_INFINITY &&
294-
startValue != Double.NEGATIVE_INFINITY) {
295-
startValue = nextDouble(startValue);
296-
}
297-
if (startValue <= oldStartValue) {
298-
throw new IllegalArgumentException("Incorrect order of"
299-
+ " intervals, must be in ascending order");
300-
}
301-
segments[0].setLength(0);
302-
part = 1;
303-
} else if (ch == '|') {
304-
if (count == newChoiceLimits.length) {
305-
newChoiceLimits = doubleArraySize(newChoiceLimits);
306-
newChoiceFormats = doubleArraySize(newChoiceFormats);
307-
}
308-
newChoiceLimits[count] = startValue;
309-
newChoiceFormats[count] = segments[1].toString();
310-
++count;
311-
oldStartValue = startValue;
312-
segments[1].setLength(0);
313-
part = 0;
314-
} else {
315-
segments[part].append(ch);
316318
}
317319
}
318-
// clean up last one
320+
321+
// clean up last one (SubPattern without trailing '|')
319322
if (part == 1) {
320-
if (count == newChoiceLimits.length) {
321-
newChoiceLimits = doubleArraySize(newChoiceLimits);
322-
newChoiceFormats = doubleArraySize(newChoiceFormats);
323-
}
324-
newChoiceLimits[count] = startValue;
325-
newChoiceFormats[count] = segments[1].toString();
326-
++count;
323+
limits.add(limit);
324+
formats.add(segments[1].toString());
327325
}
328-
choiceLimits = new double[count];
329-
System.arraycopy(newChoiceLimits, 0, choiceLimits, 0, count);
330-
choiceFormats = new String[count];
331-
System.arraycopy(newChoiceFormats, 0, choiceFormats, 0, count);
326+
choiceLimits = limits.stream().mapToDouble(d -> d).toArray();
327+
choiceFormats = formats.toArray(new String[0]);
328+
}
329+
330+
/**
331+
* Converts a string value to its double representation; this is used
332+
* to create the limit segment while applying a pattern.
333+
* Handles "\u221E", as specified by the pattern syntax.
334+
*/
335+
private static double stringToNum(String str) {
336+
return switch (str) {
337+
case "\u221E" -> Double.POSITIVE_INFINITY;
338+
case "-\u221E" -> Double.NEGATIVE_INFINITY;
339+
default -> Double.parseDouble(str);
340+
};
332341
}
333342

334343
/**
@@ -402,6 +411,7 @@ public String toPattern() {
402411
* @see #applyPattern
403412
*/
404413
public ChoiceFormat(String newPattern) {
414+
Objects.requireNonNull(newPattern, "newPattern must not be null");
405415
applyPatternImpl(newPattern);
406416
}
407417

@@ -574,6 +584,24 @@ public static final double nextDouble (double d) {
574584
return Math.nextUp(d);
575585
}
576586

587+
/**
588+
* Finds the least double greater than {@code d} (if {@code positive} is
589+
* {@code true}), or the greatest double less than {@code d} (if
590+
* {@code positive} is {@code false}).
591+
* If {@code NaN}, returns same value.
592+
*
593+
* @implNote This is equivalent to calling
594+
* {@code positive ? Math.nextUp(d) : Math.nextDown(d)}
595+
*
596+
* @param d the reference value
597+
* @param positive {@code true} if the least double is desired;
598+
* {@code false} otherwise
599+
* @return the least or greater double value
600+
*/
601+
public static double nextDouble (double d, boolean positive) {
602+
return positive ? Math.nextUp(d) : Math.nextDown(d);
603+
}
604+
577605
/**
578606
* Finds the greatest double less than {@code d}.
579607
* If {@code NaN}, returns same value.
@@ -593,8 +621,7 @@ public static final double previousDouble (double d) {
593621
* Overrides Cloneable
594622
*/
595623
@Override
596-
public Object clone()
597-
{
624+
public Object clone() {
598625
ChoiceFormat other = (ChoiceFormat) super.clone();
599626
// for primitives or immutables, shallow clone is enough
600627
other.choiceLimits = choiceLimits.clone();
@@ -685,37 +712,4 @@ private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundE
685712
* @serial
686713
*/
687714
private String[] choiceFormats;
688-
689-
/**
690-
* Finds the least double greater than {@code d} (if {@code positive} is
691-
* {@code true}), or the greatest double less than {@code d} (if
692-
* {@code positive} is {@code false}).
693-
* If {@code NaN}, returns same value.
694-
*
695-
* @implNote This is equivalent to calling
696-
* {@code positive ? Math.nextUp(d) : Math.nextDown(d)}
697-
*
698-
* @param d the reference value
699-
* @param positive {@code true} if the least double is desired;
700-
* {@code false} otherwise
701-
* @return the least or greater double value
702-
*/
703-
public static double nextDouble (double d, boolean positive) {
704-
return positive ? Math.nextUp(d) : Math.nextDown(d);
705-
}
706-
707-
private static double[] doubleArraySize(double[] array) {
708-
int oldSize = array.length;
709-
double[] newArray = new double[oldSize * 2];
710-
System.arraycopy(array, 0, newArray, 0, oldSize);
711-
return newArray;
712-
}
713-
714-
private String[] doubleArraySize(String[] array) {
715-
int oldSize = array.length;
716-
String[] newArray = new String[oldSize * 2];
717-
System.arraycopy(array, 0, newArray, 0, oldSize);
718-
return newArray;
719-
}
720-
721715
}

‎test/jdk/java/text/Format/ChoiceFormat/PatternsTest.java

+14-5
Original file line numberDiff line numberDiff line change
@@ -23,22 +23,21 @@
2323

2424
/*
2525
* @test
26-
* @bug 6285888 6801704
26+
* @bug 6285888 6801704 8325898
2727
* @summary Test the expected behavior for a wide range of patterns (both
2828
* correct and incorrect). This test documents the behavior of incorrect
2929
* ChoiceFormat patterns either throwing an exception, or discarding
3030
* the incorrect portion of a pattern.
3131
* @run junit PatternsTest
3232
*/
3333

34-
import java.text.ChoiceFormat;
35-
3634
import org.junit.jupiter.api.Test;
3735
import org.junit.jupiter.params.ParameterizedTest;
3836
import org.junit.jupiter.params.provider.Arguments;
3937
import org.junit.jupiter.params.provider.MethodSource;
4038

41-
import static org.junit.jupiter.api.Assertions.assertDoesNotThrow;
39+
import java.text.ChoiceFormat;
40+
4241
import static org.junit.jupiter.api.Assertions.assertEquals;
4342
import static org.junit.jupiter.api.Assertions.assertThrows;
4443
import static org.junit.jupiter.params.provider.Arguments.arguments;
@@ -97,6 +96,7 @@ public void invalidPatternsThrowsTest(String pattern, String errMsg) {
9796
// an exception.
9897
private static Arguments[] invalidPatternsThrowsTest() {
9998
return new Arguments[] {
99+
arguments("#", ERR1), // Only relation
100100
arguments("#foo", ERR1), // No Limit
101101
arguments("0#foo|#|1#bar", ERR1), // Missing Relation in SubPattern
102102
arguments("#|", ERR1), // Missing Limit
@@ -127,11 +127,20 @@ public void invalidPatternsDiscardedTest(String brokenPattern, String actualPatt
127127
// after discarding occurs.
128128
private static Arguments[] invalidPatternsDiscardedTest() {
129129
return new Arguments[] {
130+
// Incomplete SubPattern (limit only) at end of Pattern
131+
arguments("1#bar|2", "1#bar"),
130132
// Incomplete SubPattern at the end of the Pattern
131133
arguments("0#foo|1#bar|baz", "0#foo|1#bar"),
134+
// Incomplete SubPattern with trailing | at the end of the Pattern
135+
// Prior to 6801704, it created the broken "0#foo|1#bar|1#"
136+
// which caused formatting 1 to return an empty string
137+
arguments("0#foo|1#bar|baz|", "0#foo|1#bar"),
138+
// Same as previous, with additional incomplete subPatterns
139+
arguments("0#foo|1#bar|baz|quux", "0#foo|1#bar"),
132140

133141
// --- These throw an ArrayIndexOutOfBoundsException
134-
// when attempting to format with them ---
142+
// when attempting to format with them as the incomplete patterns
143+
// are discarded, initializing the cFmt with empty limits and formats ---
135144
// SubPattern with only a Limit (which is interpreted as a Format)
136145
arguments("0", ""),
137146
// SubPattern with only a Format

0 commit comments

Comments
 (0)
Please sign in to comment.