]> git.saurik.com Git - apple/icu.git/blob - icuSources/test/intltest/msfmrgts.cpp
ICU-62123.0.1.tar.gz
[apple/icu.git] / icuSources / test / intltest / msfmrgts.cpp
1 // © 2016 and later: Unicode, Inc. and others.
2 // License & terms of use: http://www.unicode.org/copyright.html
3 /***********************************************************************
4 * COPYRIGHT:
5 * Copyright (c) 1997-2016, International Business Machines Corporation
6 * and others. All Rights Reserved.
7 ***********************************************************************/
8
9 #include "unicode/utypes.h"
10
11 #if !UCONFIG_NO_FORMATTING
12
13 #include "msfmrgts.h"
14
15 #include "unicode/format.h"
16 #include "unicode/decimfmt.h"
17 #include "unicode/locid.h"
18 #include "unicode/msgfmt.h"
19 #include "unicode/numfmt.h"
20 #include "unicode/choicfmt.h"
21 #include "unicode/gregocal.h"
22 #include "cmemory.h"
23 #include "putilimp.h"
24
25 // *****************************************************************************
26 // class MessageFormatRegressionTest
27 // *****************************************************************************
28
29 #define CASE(id,test) case id: name = #test; if (exec) { logln(#test "---"); logln((UnicodeString)""); test(); } break;
30
31 void
32 MessageFormatRegressionTest::runIndexedTest( int32_t index, UBool exec, const char* &name, char* /*par*/ )
33 {
34 TESTCASE_AUTO_BEGIN;
35 TESTCASE_AUTO(Test4074764)
36 //TESTCASE_AUTO(Test4058973) -- disabled/obsolete in ICU 4.8
37 TESTCASE_AUTO(Test4031438)
38 TESTCASE_AUTO(Test4052223)
39 TESTCASE_AUTO(Test4104976)
40 TESTCASE_AUTO(Test4106659)
41 TESTCASE_AUTO(Test4106660)
42 TESTCASE_AUTO(Test4111739)
43 TESTCASE_AUTO(Test4114743)
44 TESTCASE_AUTO(Test4116444)
45 TESTCASE_AUTO(Test4114739)
46 TESTCASE_AUTO(Test4113018)
47 TESTCASE_AUTO(Test4106661)
48 TESTCASE_AUTO(Test4094906)
49 TESTCASE_AUTO(Test4118592)
50 TESTCASE_AUTO(Test4118594)
51 TESTCASE_AUTO(Test4105380)
52 TESTCASE_AUTO(Test4120552)
53 TESTCASE_AUTO(Test4142938)
54 TESTCASE_AUTO(TestChoicePatternQuote)
55 TESTCASE_AUTO(Test4112104)
56 TESTCASE_AUTO(TestAPI)
57 TESTCASE_AUTO_END;
58 }
59
60 UBool
61 MessageFormatRegressionTest::failure(UErrorCode status, const char* msg, UBool possibleDataError)
62 {
63 if(U_FAILURE(status)) {
64 if (possibleDataError) {
65 dataerrln(UnicodeString("FAIL: ") + msg + " failed, error " + u_errorName(status));
66 } else {
67 errln(UnicodeString("FAIL: ") + msg + " failed, error " + u_errorName(status));
68 }
69 return TRUE;
70 }
71
72 return FALSE;
73 }
74
75 /* @bug 4074764
76 * Null exception when formatting pattern with MessageFormat
77 * with no parameters.
78 */
79 void MessageFormatRegressionTest::Test4074764() {
80 UnicodeString pattern [] = {
81 "Message without param",
82 "Message with param:{0}",
83 "Longer Message with param {0}"
84 };
85 //difference between the two param strings are that
86 //in the first one, the param position is within the
87 //length of the string without param while it is not so
88 //in the other case.
89
90 UErrorCode status = U_ZERO_ERROR;
91 MessageFormat *messageFormatter = new MessageFormat("", status);
92
93 failure(status, "couldn't create MessageFormat");
94
95 //try {
96 //Apply pattern with param and print the result
97 messageFormatter->applyPattern(pattern[1], status);
98 failure(status, "messageFormat->applyPattern");
99 //Object[] params = {new UnicodeString("BUG"), new Date()};
100 Formattable params [] = {
101 Formattable(UnicodeString("BUG")),
102 Formattable(0, Formattable::kIsDate)
103 };
104 UnicodeString tempBuffer;
105 FieldPosition pos(FieldPosition::DONT_CARE);
106 tempBuffer = messageFormatter->format(params, 2, tempBuffer, pos, status);
107 if( tempBuffer != "Message with param:BUG" || failure(status, "messageFormat->format"))
108 errln("MessageFormat with one param test failed.");
109 logln("Formatted with one extra param : " + tempBuffer);
110
111 //Apply pattern without param and print the result
112 messageFormatter->applyPattern(pattern[0], status);
113 failure(status, "messageFormatter->applyPattern");
114
115 // {sfb} how much does this apply in C++?
116 // do we want to verify that the Formattable* array is not NULL,
117 // or is that the user's responsibility?
118 // additionally, what should be the item count?
119 // for bug testing purposes, assume that something was set to
120 // NULL by mistake, and that the length should be non-zero
121
122 //tempBuffer = messageFormatter->format(NULL, 1, tempBuffer, FieldPosition(FieldPosition::DONT_CARE), status);
123 tempBuffer.remove();
124 tempBuffer = messageFormatter->format(NULL, 0, tempBuffer, pos, status);
125
126 if( tempBuffer != "Message without param" || failure(status, "messageFormat->format"))
127 errln("MessageFormat with no param test failed.");
128 logln("Formatted with no params : " + tempBuffer);
129
130 tempBuffer.remove();
131 tempBuffer = messageFormatter->format(params, 2, tempBuffer, pos, status);
132 if (tempBuffer != "Message without param" || failure(status, "messageFormat->format"))
133 errln("Formatted with arguments > subsitution failed. result = " + tempBuffer);
134 logln("Formatted with extra params : " + tempBuffer);
135 //This statement gives an exception while formatting...
136 //If we use pattern[1] for the message with param,
137 //we get an NullPointerException in MessageFormat.java(617)
138 //If we use pattern[2] for the message with param,
139 //we get an StringArrayIndexOutOfBoundsException in MessageFormat.java(614)
140 //Both are due to maxOffset not being reset to -1
141 //in applyPattern() when the pattern does not
142 //contain any param.
143 /*} catch (Exception foo) {
144 errln("Exception when formatting with no params.");
145 }*/
146
147 delete messageFormatter;
148 }
149
150 /* @bug 4058973
151 * MessageFormat.toPattern has weird rounding behavior.
152 *
153 * ICU 4.8: This test is commented out because toPattern() has been changed to return
154 * the original pattern string, rather than reconstituting a new (equivalent) one.
155 * This trivially eliminates issues with rounding or any other pattern string differences.
156 */
157 /*
158 void MessageFormatRegressionTest::Test4058973()
159 {
160 UErrorCode status = U_ZERO_ERROR;
161 MessageFormat *fmt = new MessageFormat("{0,choice,0#no files|1#one file|1< {0,number,integer} files}", status);
162 failure(status, "new MessageFormat");
163
164 UnicodeString pat;
165 pat = fmt->toPattern(pat);
166 UnicodeString exp("{0,choice,0#no files|1#one file|1< {0,number,integer} files}");
167 if (pat != exp) {
168 errln("MessageFormat.toPattern failed");
169 errln("Exp: " + exp);
170 errln("Got: " + pat);
171 }
172
173 delete fmt;
174 }*/
175 /* @bug 4031438
176 * More robust message formats.
177 */
178 void MessageFormatRegressionTest::Test4031438()
179 {
180 UErrorCode status = U_ZERO_ERROR;
181
182 UnicodeString pattern1("Impossible {1} has occurred -- status code is {0} and message is {2}.");
183 UnicodeString pattern2("Double '' Quotes {0} test and quoted '{1}' test plus 'other {2} stuff'.");
184
185 MessageFormat *messageFormatter = new MessageFormat("", status);
186 failure(status, "new MessageFormat");
187
188 const UBool possibleDataError = TRUE;
189
190 //try {
191 logln("Apply with pattern : " + pattern1);
192 messageFormatter->applyPattern(pattern1, status);
193 failure(status, "messageFormat->applyPattern");
194 //Object[] params = {new Integer(7)};
195 Formattable params []= {
196 Formattable((int32_t)7)
197 };
198 UnicodeString tempBuffer;
199 FieldPosition pos(FieldPosition::DONT_CARE);
200 tempBuffer = messageFormatter->format(params, 1, tempBuffer, pos, status);
201 if(tempBuffer != "Impossible {1} has occurred -- status code is 7 and message is {2}." || failure(status, "MessageFormat::format"))
202 dataerrln("Tests arguments < substitution failed");
203 logln("Formatted with 7 : " + tempBuffer);
204 ParsePosition pp(0);
205 int32_t count = 0;
206 Formattable *objs = messageFormatter->parse(tempBuffer, pp, count);
207 //if(objs[7/*params.length*/] != NULL)
208 // errln("Parse failed with more than expected arguments");
209
210 NumberFormat *fmt = 0;
211 UnicodeString temp, temp1;
212
213 for (int i = 0; i < count; i++) {
214
215 // convert to string if not already
216 Formattable obj = objs[i];
217 temp.remove();
218 if(obj.getType() == Formattable::kString)
219 temp = obj.getString(temp);
220 else {
221 fmt = NumberFormat::createInstance(status);
222 switch (obj.getType()) {
223 case Formattable::kLong: fmt->format(obj.getLong(), temp); break;
224 case Formattable::kInt64: fmt->format(obj.getInt64(), temp); break;
225 case Formattable::kDouble: fmt->format(obj.getDouble(), temp); break;
226 default: break;
227 }
228 }
229
230 // convert to string if not already
231 Formattable obj1 = params[i];
232 temp1.remove();
233 if(obj1.getType() == Formattable::kString)
234 temp1 = obj1.getString(temp1);
235 else {
236 fmt = NumberFormat::createInstance(status);
237 switch (obj1.getType()) {
238 case Formattable::kLong: fmt->format(obj1.getLong(), temp1); break;
239 case Formattable::kInt64: fmt->format(obj1.getInt64(), temp1); break;
240 case Formattable::kDouble: fmt->format(obj1.getDouble(), temp1); break;
241 default: break;
242 }
243 }
244
245 //if (objs[i] != NULL && objs[i].getString(temp1) != params[i].getString(temp2)) {
246 if (temp != temp1) {
247 errln("Parse failed on object " + objs[i].getString(temp1) + " at index : " + i);
248 }
249 }
250
251 delete fmt;
252 delete [] objs;
253
254 // {sfb} does this apply? no way to really pass a null Formattable,
255 // only a null array
256
257 /*tempBuffer = messageFormatter->format(null, tempBuffer, FieldPosition(FieldPosition::DONT_CARE), status);
258 if (tempBuffer != "Impossible {1} has occurred -- status code is {0} and message is {2}." || failure(status, "messageFormat->format"))
259 errln("Tests with no arguments failed");
260 logln("Formatted with null : " + tempBuffer);*/
261 logln("Apply with pattern : " + pattern2);
262 messageFormatter->applyPattern(pattern2, status);
263 failure(status, "messageFormatter->applyPattern", possibleDataError);
264 tempBuffer.remove();
265 tempBuffer = messageFormatter->format(params, 1, tempBuffer, pos, status);
266 if (tempBuffer != "Double ' Quotes 7 test and quoted {1} test plus 'other {2} stuff'.")
267 dataerrln("quote format test (w/ params) failed. - %s", u_errorName(status));
268 logln("Formatted with params : " + tempBuffer);
269
270 /*tempBuffer = messageFormatter->format(null);
271 if (!tempBuffer.equals("Double ' Quotes {0} test and quoted {1} test plus other {2} stuff."))
272 errln("quote format test (w/ null) failed.");
273 logln("Formatted with null : " + tempBuffer);
274 logln("toPattern : " + messageFormatter.toPattern());*/
275 /*} catch (Exception foo) {
276 errln("Exception when formatting in bug 4031438. "+foo.getMessage());
277 }*/
278 delete messageFormatter;
279 }
280
281 void MessageFormatRegressionTest::Test4052223()
282 {
283
284 ParsePosition pos(0);
285 if (pos.getErrorIndex() != -1) {
286 errln("ParsePosition.getErrorIndex initialization failed.");
287 }
288
289 UErrorCode status = U_ZERO_ERROR;
290 MessageFormat *fmt = new MessageFormat("There are {0} apples growing on the {1} tree.", status);
291 failure(status, "new MessageFormat");
292 UnicodeString str("There is one apple growing on the peach tree.");
293
294 int32_t count = 0;
295 fmt->parse(str, pos, count);
296
297 logln(UnicodeString("unparsable string , should fail at ") + pos.getErrorIndex());
298 if (pos.getErrorIndex() == -1)
299 errln("Bug 4052223 failed : parsing string " + str);
300 pos.setErrorIndex(4);
301 if (pos.getErrorIndex() != 4)
302 errln(UnicodeString("setErrorIndex failed, got ") + pos.getErrorIndex() + " instead of 4");
303
304 ChoiceFormat *f = new ChoiceFormat(
305 "-1#are negative|0#are no or fraction|1#is one|1.0<is 1+|2#are two|2<are more than 2.", status);
306 failure(status, "new ChoiceFormat");
307 pos.setIndex(0);
308 pos.setErrorIndex(-1);
309 Formattable obj;
310 f->parse("are negative", obj, pos);
311 if (pos.getErrorIndex() != -1 && obj.getDouble() == -1.0)
312 errln(UnicodeString("Parse with \"are negative\" failed, at ") + pos.getErrorIndex());
313 pos.setIndex(0);
314 pos.setErrorIndex(-1);
315 f->parse("are no or fraction ", obj, pos);
316 if (pos.getErrorIndex() != -1 && obj.getDouble() == 0.0)
317 errln(UnicodeString("Parse with \"are no or fraction\" failed, at ") + pos.getErrorIndex());
318 pos.setIndex(0);
319 pos.setErrorIndex(-1);
320 f->parse("go postal", obj, pos);
321 if (pos.getErrorIndex() == -1 && ! uprv_isNaN(obj.getDouble()))
322 errln(UnicodeString("Parse with \"go postal\" failed, at ") + pos.getErrorIndex());
323
324 delete fmt;
325 delete f;
326 }
327 /* @bug 4104976
328 * ChoiceFormat.equals(null) throws NullPointerException
329 */
330
331 // {sfb} not really applicable in C++?? (kind of silly)
332
333 void MessageFormatRegressionTest::Test4104976()
334 {
335 double limits [] = {1, 20};
336 UnicodeString formats [] = {
337 UnicodeString("xyz"),
338 UnicodeString("abc")
339 };
340 int32_t formats_length = UPRV_LENGTHOF(formats);
341 UErrorCode status = U_ZERO_ERROR;
342 ChoiceFormat *cf = new ChoiceFormat(limits, formats, formats_length);
343 failure(status, "new ChoiceFormat");
344 //try {
345 log("Compares to null is always false, returned : ");
346 logln(cf == NULL ? "TRUE" : "FALSE");
347 /*} catch (Exception foo) {
348 errln("ChoiceFormat.equals(null) throws exception.");
349 }*/
350
351 delete cf;
352 }
353
354 /* @bug 4106659
355 * ChoiceFormat.ctor(double[], String[]) doesn't check
356 * whether lengths of input arrays are equal.
357 */
358
359 // {sfb} again, not really applicable in C++
360
361 void MessageFormatRegressionTest::Test4106659()
362 {
363 /*
364 double limits [] = {
365 1, 2, 3
366 };
367 UnicodeString formats [] = {
368 "one", "two"
369 };
370 ChoiceFormat *cf = NULL;
371 //try {
372 // cf = new ChoiceFormat(limits, formats, 3);
373 //} catch (Exception foo) {
374 // logln("ChoiceFormat constructor should check for the array lengths");
375 // cf = null;
376 //}
377 //if (cf != null)
378 // errln(cf->format(5));
379 //
380 delete cf;
381 */
382 }
383
384 /* @bug 4106660
385 * ChoiceFormat.ctor(double[], String[]) allows unordered double array.
386 * This is not a bug, added javadoc to emphasize the use of limit
387 * array must be in ascending order.
388 */
389 void MessageFormatRegressionTest::Test4106660()
390 {
391 double limits [] = {3, 1, 2};
392 UnicodeString formats [] = {
393 UnicodeString("Three"),
394 UnicodeString("One"),
395 UnicodeString("Two")
396 };
397 ChoiceFormat *cf = new ChoiceFormat(limits, formats, 3);
398 double d = 5.0;
399 UnicodeString str;
400 FieldPosition pos(FieldPosition::DONT_CARE);
401 str = cf->format(d, str, pos);
402 if (str != "Two")
403 errln( (UnicodeString) "format(" + d + ") = " + str);
404
405 delete cf;
406 }
407
408 /* @bug 4111739
409 * MessageFormat is incorrectly serialized/deserialized.
410 */
411
412 // {sfb} doesn't apply in C++
413
414 void MessageFormatRegressionTest::Test4111739()
415 {
416 /*MessageFormat format1 = null;
417 MessageFormat format2 = null;
418 ObjectOutputStream ostream = null;
419 ByteArrayOutputStream baos = null;
420 ObjectInputStream istream = null;
421
422 try {
423 baos = new ByteArrayOutputStream();
424 ostream = new ObjectOutputStream(baos);
425 } catch(IOException e) {
426 errln("Unexpected exception : " + e.getMessage());
427 return;
428 }
429
430 try {
431 format1 = new MessageFormat("pattern{0}");
432 ostream.writeObject(format1);
433 ostream.flush();
434
435 byte bytes[] = baos.toByteArray();
436
437 istream = new ObjectInputStream(new ByteArrayInputStream(bytes));
438 format2 = (MessageFormat)istream.readObject();
439 } catch(Exception e) {
440 errln("Unexpected exception : " + e.getMessage());
441 }
442
443 if (!format1.equals(format2)) {
444 errln("MessageFormats before and after serialization are not" +
445 " equal\nformat1 = " + format1 + "(" + format1.toPattern() + ")\nformat2 = " +
446 format2 + "(" + format2.toPattern() + ")");
447 } else {
448 logln("Serialization for MessageFormat is OK.");
449 }*/
450 }
451 /* @bug 4114743
452 * MessageFormat.applyPattern allows illegal patterns.
453 */
454 void MessageFormatRegressionTest::Test4114743()
455 {
456 UnicodeString originalPattern("initial pattern");
457 UErrorCode status = U_ZERO_ERROR;
458 MessageFormat *mf = new MessageFormat(originalPattern, status);
459 failure(status, "new MessageFormat");
460 //try {
461 UnicodeString illegalPattern("ab { '}' de");
462 mf->applyPattern(illegalPattern, status);
463 if( ! U_FAILURE(status))
464 errln("illegal pattern: \"" + illegalPattern + "\"");
465 /*} catch (IllegalArgumentException foo) {
466 if (!originalPattern.equals(mf.toPattern()))
467 errln("pattern after: \"" + mf.toPattern() + "\"");
468 }*/
469 delete mf;
470 }
471
472 /* @bug 4116444
473 * MessageFormat.parse has different behavior in case of null.
474 */
475 void MessageFormatRegressionTest::Test4116444()
476 {
477 UnicodeString patterns [] = {
478 (UnicodeString)"",
479 (UnicodeString)"one",
480 (UnicodeString) "{0,date,short}"
481 };
482
483 UErrorCode status = U_ZERO_ERROR;
484 MessageFormat *mf = new MessageFormat("", status);
485 failure(status, "new MessageFormat");
486
487 for (int i = 0; i < 3; i++) {
488 UnicodeString pattern = patterns[i];
489 mf->applyPattern(pattern, status);
490 failure(status, "mf->applyPattern", TRUE);
491
492 //try {
493 int32_t count = 0;
494 ParsePosition pp(0);
495 Formattable *array = mf->parse(UnicodeString(""), pp, count);
496 logln("pattern: \"" + pattern + "\"");
497 log(" parsedObjects: ");
498 if (array != NULL) {
499 log("{");
500 for (int j = 0; j < count; j++) {
501 //if (array[j] != null)
502 UnicodeString dummy;
503 dataerrln("\"" + array[j].getString(dummy) + "\"");
504 //else
505 // log("null");
506 if (j < count- 1)
507 log(",");
508 }
509 log("}") ;
510 delete[] array;
511 } else {
512 log("null");
513 }
514 logln("");
515 /*} catch (Exception e) {
516 errln("pattern: \"" + pattern + "\"");
517 errln(" Exception: " + e.getMessage());
518 }*/
519 }
520
521 delete mf;
522 }
523 /* @bug 4114739 (FIX and add javadoc)
524 * MessageFormat.format has undocumented behavior about empty format objects.
525 */
526
527 // {sfb} doesn't apply in C++?
528 void MessageFormatRegressionTest::Test4114739()
529 {
530
531 UErrorCode status = U_ZERO_ERROR;
532 MessageFormat *mf = new MessageFormat("<{0}>", status);
533 failure(status, "new MessageFormat");
534
535 Formattable *objs1 = NULL;
536 //Formattable objs2 [] = {};
537 //Formattable *objs3 [] = {NULL};
538 //try {
539 UnicodeString pat;
540 UnicodeString res;
541 logln("pattern: \"" + mf->toPattern(pat) + "\"");
542 log("format(null) : ");
543 FieldPosition pos(FieldPosition::DONT_CARE);
544 logln("\"" + mf->format(objs1, 0, res, pos, status) + "\"");
545 failure(status, "mf->format");
546 /*log("format({}) : ");
547 logln("\"" + mf->format(objs2, 0, res, FieldPosition(FieldPosition::DONT_CARE), status) + "\"");
548 failure(status, "mf->format");
549 log("format({null}) :");
550 logln("\"" + mf->format(objs3, 0, res, FieldPosition(FieldPosition::DONT_CARE), status) + "\"");
551 failure(status, "mf->format");*/
552 /*} catch (Exception e) {
553 errln("Exception thrown for null argument tests.");
554 }*/
555
556 delete mf;
557 }
558
559 /* @bug 4113018
560 * MessageFormat.applyPattern works wrong with illegal patterns.
561 */
562 void MessageFormatRegressionTest::Test4113018()
563 {
564 UnicodeString originalPattern("initial pattern");
565 UErrorCode status = U_ZERO_ERROR;
566 MessageFormat *mf = new MessageFormat(originalPattern, status);
567 failure(status, "new messageFormat");
568 UnicodeString illegalPattern("format: {0, xxxYYY}");
569 UnicodeString pat;
570 logln("pattern before: \"" + mf->toPattern(pat) + "\"");
571 logln("illegal pattern: \"" + illegalPattern + "\"");
572 //try {
573 mf->applyPattern(illegalPattern, status);
574 if( ! U_FAILURE(status))
575 errln("Should have thrown IllegalArgumentException for pattern : " + illegalPattern);
576 /*} catch (IllegalArgumentException e) {
577 if (!originalPattern.equals(mf.toPattern()))
578 errln("pattern after: \"" + mf.toPattern() + "\"");
579 }*/
580 delete mf;
581 }
582
583 /* @bug 4106661
584 * ChoiceFormat is silent about the pattern usage in javadoc.
585 */
586 void MessageFormatRegressionTest::Test4106661()
587 {
588 UErrorCode status = U_ZERO_ERROR;
589 ChoiceFormat *fmt = new ChoiceFormat(
590 "-1#are negative| 0#are no or fraction | 1#is one |1.0<is 1+ |2#are two |2<are more than 2.", status);
591 failure(status, "new ChoiceFormat");
592 UnicodeString pat;
593 logln("Formatter Pattern : " + fmt->toPattern(pat));
594
595 FieldPosition bogus(FieldPosition::DONT_CARE);
596 UnicodeString str;
597
598 // Will this work for -inf?
599 logln("Format with -INF : " + fmt->format(Formattable(-uprv_getInfinity()), str, bogus, status));
600 failure(status, "fmt->format");
601 str.remove();
602 logln("Format with -1.0 : " + fmt->format(Formattable(-1.0), str, bogus, status));
603 failure(status, "fmt->format");
604 str.remove();
605 logln("Format with -1.0 : " + fmt->format(Formattable(-1.0), str, bogus, status));
606 failure(status, "fmt->format");
607 str.remove();
608 logln("Format with 0 : " + fmt->format(Formattable((int32_t)0), str, bogus, status));
609 failure(status, "fmt->format");
610 str.remove();
611 logln("Format with 0.9 : " + fmt->format(Formattable(0.9), str, bogus, status));
612 failure(status, "fmt->format");
613 str.remove();
614 logln("Format with 1.0 : " + fmt->format(Formattable(1.0), str, bogus, status));
615 failure(status, "fmt->format");
616 str.remove();
617 logln("Format with 1.5 : " + fmt->format(Formattable(1.5), str, bogus, status));
618 failure(status, "fmt->format");
619 str.remove();
620 logln("Format with 2 : " + fmt->format(Formattable((int32_t)2), str, bogus, status));
621 failure(status, "fmt->format");
622 str.remove();
623 logln("Format with 2.1 : " + fmt->format(Formattable(2.1), str, bogus, status));
624 failure(status, "fmt->format");
625 str.remove();
626 logln("Format with NaN : " + fmt->format(Formattable(uprv_getNaN()), str, bogus, status));
627 failure(status, "fmt->format");
628 str.remove();
629 logln("Format with +INF : " + fmt->format(Formattable(uprv_getInfinity()), str, bogus, status));
630 failure(status, "fmt->format");
631
632 delete fmt;
633 }
634
635 /* @bug 4094906
636 * ChoiceFormat should accept \u221E as eq. to INF.
637 */
638 void MessageFormatRegressionTest::Test4094906()
639 {
640 UErrorCode status = U_ZERO_ERROR;
641 UnicodeString pattern("-");
642 pattern += (UChar) 0x221E;
643 pattern += "<are negative|0<are no or fraction|1#is one|1<is 1+|";
644 pattern += (UChar) 0x221E;
645 pattern += "<are many.";
646
647 ChoiceFormat *fmt = new ChoiceFormat(pattern, status);
648 failure(status, "new ChoiceFormat");
649 UnicodeString pat;
650 if (fmt->toPattern(pat) != pattern) {
651 errln( (UnicodeString) "Formatter Pattern : " + pat);
652 errln( (UnicodeString) "Expected Pattern : " + pattern);
653 }
654 FieldPosition bogus(FieldPosition::DONT_CARE);
655 UnicodeString str;
656
657 // Will this work for -inf?
658 logln("Format with -INF : " + fmt->format(Formattable(-uprv_getInfinity()), str, bogus, status));
659 failure(status, "fmt->format");
660 str.remove();
661 logln("Format with -1.0 : " + fmt->format(Formattable(-1.0), str, bogus, status));
662 failure(status, "fmt->format");
663 str.remove();
664 logln("Format with -1.0 : " + fmt->format(Formattable(-1.0), str, bogus, status));
665 failure(status, "fmt->format");
666 str.remove();
667 logln("Format with 0 : " + fmt->format(Formattable((int32_t)0), str, bogus, status));
668 failure(status, "fmt->format");
669 str.remove();
670 logln("Format with 0.9 : " + fmt->format(Formattable(0.9), str, bogus, status));
671 failure(status, "fmt->format");
672 str.remove();
673 logln("Format with 1.0 : " + fmt->format(Formattable(1.0), str, bogus, status));
674 failure(status, "fmt->format");
675 str.remove();
676 logln("Format with 1.5 : " + fmt->format(Formattable(1.5), str, bogus, status));
677 failure(status, "fmt->format");
678 str.remove();
679 logln("Format with 2 : " + fmt->format(Formattable((int32_t)2), str, bogus, status));
680 failure(status, "fmt->format");
681 str.remove();
682 logln("Format with 2.1 : " + fmt->format(Formattable(2.1), str, bogus, status));
683 failure(status, "fmt->format");
684 str.remove();
685 logln("Format with NaN : " + fmt->format(Formattable(uprv_getNaN()), str, bogus, status));
686 failure(status, "fmt->format");
687 str.remove();
688 logln("Format with +INF : " + fmt->format(Formattable(uprv_getInfinity()), str, bogus, status));
689 failure(status, "fmt->format");
690
691 delete fmt;
692 }
693
694 /* @bug 4118592
695 * MessageFormat.parse fails with ChoiceFormat.
696 */
697 void MessageFormatRegressionTest::Test4118592()
698 {
699 UErrorCode status = U_ZERO_ERROR;
700 MessageFormat *mf = new MessageFormat("", status);
701 failure(status, "new messageFormat");
702 UnicodeString pattern("{0,choice,1#YES|2#NO}");
703 UnicodeString prefix("");
704 Formattable *objs = 0;
705
706 for (int i = 0; i < 5; i++) {
707 UnicodeString formatted;
708 formatted = prefix + "YES";
709 mf->applyPattern(prefix + pattern, status);
710 failure(status, "mf->applyPattern");
711 prefix += "x";
712 //Object[] objs = mf.parse(formatted, new ParsePosition(0));
713 int32_t count = 0;
714 ParsePosition pp(0);
715 objs = mf->parse(formatted, pp, count);
716 UnicodeString pat;
717 logln(UnicodeString("") + i + ". pattern :\"" + mf->toPattern(pat) + "\"");
718 log(" \"" + formatted + "\" parsed as ");
719 if (objs == NULL)
720 logln(" null");
721 else {
722 UnicodeString temp;
723 if(objs[0].getType() == Formattable::kString)
724 logln((UnicodeString)" " + objs[0].getString(temp));
725 else
726 logln((UnicodeString)" " + (objs[0].getType() == Formattable::kLong ? objs[0].getLong() : objs[0].getDouble()));
727 delete[] objs;
728
729 }
730 }
731
732 delete mf;
733 }
734 /* @bug 4118594
735 * MessageFormat.parse fails for some patterns.
736 */
737 void MessageFormatRegressionTest::Test4118594()
738 {
739 UErrorCode status = U_ZERO_ERROR;
740 const UBool possibleDataError = TRUE;
741 MessageFormat *mf = new MessageFormat("{0}, {0}, {0}", status);
742 failure(status, "new MessageFormat");
743 UnicodeString forParsing("x, y, z");
744 //Object[] objs = mf.parse(forParsing, new ParsePosition(0));
745 int32_t count = 0;
746 ParsePosition pp(0);
747 Formattable *objs = mf->parse(forParsing, pp, count);
748 UnicodeString pat;
749 logln("pattern: \"" + mf->toPattern(pat) + "\"");
750 logln("text for parsing: \"" + forParsing + "\"");
751 UnicodeString str;
752 if (objs[0].getString(str) != "z")
753 errln("argument0: \"" + objs[0].getString(str) + "\"");
754 mf->applyPattern("{0,number,#.##}, {0,number,#.#}", status);
755 failure(status, "mf->applyPattern", possibleDataError);
756 //Object[] oldobjs = {new Double(3.1415)};
757 Formattable oldobjs [] = {Formattable(3.1415)};
758 UnicodeString result;
759 FieldPosition pos(FieldPosition::DONT_CARE);
760 result = mf->format( oldobjs, 1, result, pos, status );
761 failure(status, "mf->format", possibleDataError);
762 pat.remove();
763 logln("pattern: \"" + mf->toPattern(pat) + "\"");
764 logln("text for parsing: \"" + result + "\"");
765 // result now equals "3.14, 3.1"
766 if (result != "3.14, 3.1")
767 dataerrln("result = " + result + " - " + u_errorName(status));
768 //Object[] newobjs = mf.parse(result, new ParsePosition(0));
769 int32_t count1 = 0;
770 pp.setIndex(0);
771 Formattable *newobjs = mf->parse(result, pp, count1);
772 // newobjs now equals {new Double(3.1)}
773 if (newobjs == NULL) {
774 dataerrln("Error calling MessageFormat::parse");
775 } else {
776 if (newobjs[0].getDouble() != 3.1)
777 errln( UnicodeString("newobjs[0] = ") + newobjs[0].getDouble());
778 }
779
780 delete [] objs;
781 delete [] newobjs;
782 delete mf;
783 }
784 /* @bug 4105380
785 * When using ChoiceFormat, MessageFormat is not good for I18n.
786 */
787 void MessageFormatRegressionTest::Test4105380()
788 {
789 UnicodeString patternText1("The disk \"{1}\" contains {0}.");
790 UnicodeString patternText2("There are {0} on the disk \"{1}\"");
791 UErrorCode status = U_ZERO_ERROR;
792 const UBool possibleDataError = TRUE;
793 MessageFormat *form1 = new MessageFormat(patternText1, status);
794 failure(status, "new MessageFormat");
795 MessageFormat *form2 = new MessageFormat(patternText2, status);
796 failure(status, "new MessageFormat");
797 double filelimits [] = {0,1,2};
798 UnicodeString filepart [] = {
799 (UnicodeString)"no files",
800 (UnicodeString)"one file",
801 (UnicodeString)"{0,number} files"
802 };
803 ChoiceFormat *fileform = new ChoiceFormat(filelimits, filepart, 3);
804 form1->setFormat(1, *fileform);
805 form2->setFormat(0, *fileform);
806 //Object[] testArgs = {new Long(12373), "MyDisk"};
807 Formattable testArgs [] = {
808 Formattable((int32_t)12373),
809 Formattable((UnicodeString)"MyDisk")
810 };
811
812 FieldPosition bogus(FieldPosition::DONT_CARE);
813
814 UnicodeString result;
815 logln(form1->format(testArgs, 2, result, bogus, status));
816 failure(status, "form1->format", possibleDataError);
817 result.remove();
818 logln(form2->format(testArgs, 2, result, bogus, status));
819 failure(status, "form1->format", possibleDataError);
820
821 delete form1;
822 delete form2;
823 delete fileform;
824 }
825 /* @bug 4120552
826 * MessageFormat.parse incorrectly sets errorIndex.
827 */
828 void MessageFormatRegressionTest::Test4120552()
829 {
830 UErrorCode status = U_ZERO_ERROR;
831 MessageFormat *mf = new MessageFormat("pattern", status);
832 failure(status, "new MessageFormat");
833 UnicodeString texts[] = {
834 (UnicodeString)"pattern",
835 (UnicodeString)"pat",
836 (UnicodeString)"1234"
837 };
838 UnicodeString pat;
839 logln("pattern: \"" + mf->toPattern(pat) + "\"");
840 for (int i = 0; i < 3; i++) {
841 ParsePosition pp(0);
842 //Object[] objs = mf.parse(texts[i], pp);
843 int32_t count = 0;
844 Formattable *objs = mf->parse(texts[i], pp, count);
845 log(" text for parsing: \"" + texts[i] + "\"");
846 if (objs == NULL) {
847 logln(" (incorrectly formatted string)");
848 if (pp.getErrorIndex() == -1)
849 errln(UnicodeString("Incorrect error index: ") + pp.getErrorIndex());
850 } else {
851 logln(" (correctly formatted string)");
852 delete[] objs;
853 }
854 }
855 delete mf;
856 }
857
858 /**
859 * @bug 4142938
860 * MessageFormat handles single quotes in pattern wrong.
861 * This is actually a problem in ChoiceFormat; it doesn't
862 * understand single quotes.
863 */
864 void MessageFormatRegressionTest::Test4142938()
865 {
866 UnicodeString pat = CharsToUnicodeString("''Vous'' {0,choice,0#n''|1#}avez s\\u00E9lectionn\\u00E9 "
867 "{0,choice,0#aucun|1#{0}} client{0,choice,0#s|1#|2#s} "
868 "personnel{0,choice,0#s|1#|2#s}.");
869 UErrorCode status = U_ZERO_ERROR;
870 MessageFormat *mf = new MessageFormat(pat, status);
871 failure(status, "new MessageFormat");
872
873 UnicodeString PREFIX [] = {
874 CharsToUnicodeString("'Vous' n'avez s\\u00E9lectionn\\u00E9 aucun clients personnels."),
875 CharsToUnicodeString("'Vous' avez s\\u00E9lectionn\\u00E9 "),
876 CharsToUnicodeString("'Vous' avez s\\u00E9lectionn\\u00E9 ")
877 };
878 UnicodeString SUFFIX [] = {
879 UnicodeString(),
880 UNICODE_STRING(" client personnel.", 18),
881 UNICODE_STRING(" clients personnels.", 20)
882 };
883
884 for (int i=0; i<3; i++) {
885 UnicodeString out;
886 //out = mf->format(new Object[]{new Integer(i)});
887 Formattable objs [] = {
888 Formattable((int32_t)i)
889 };
890 FieldPosition pos(FieldPosition::DONT_CARE);
891 out = mf->format(objs, 1, out, pos, status);
892 if (!failure(status, "mf->format", TRUE)) {
893 if (SUFFIX[i] == "") {
894 if (out != PREFIX[i])
895 errln((UnicodeString)"" + i + ": Got \"" + out + "\"; Want \"" + PREFIX[i] + "\"");
896 }
897 else {
898 if (!out.startsWith(PREFIX[i]) ||
899 !out.endsWith(SUFFIX[i]))
900 errln((UnicodeString)"" + i + ": Got \"" + out + "\"; Want \"" + PREFIX[i] + "\"...\"" +
901 SUFFIX[i] + "\"");
902 }
903 }
904 }
905
906 delete mf;
907 }
908
909 /**
910 * @bug 4142938
911 * Test the applyPattern and toPattern handling of single quotes
912 * by ChoiceFormat. (This is in here because this was a bug reported
913 * against MessageFormat.) The single quote is used to quote the
914 * pattern characters '|', '#', '<', and '\u2264'. Two quotes in a row
915 * is a quote literal.
916 */
917 void MessageFormatRegressionTest::TestChoicePatternQuote()
918 {
919 // ICU 4.8 ChoiceFormat (like PluralFormat & SelectFormat)
920 // returns the chosen string unmodified, so that it is usable in a MessageFormat.
921 // We modified the test strings accordingly.
922 // Note: Without further formatting/trimming/etc., it is not possible
923 // to get a single apostrophe as the last character of a non-final choice sub-message
924 // because the single apostrophe before the pipe '|' would start quoted text.
925 // Normally, ChoiceFormat is used inside a MessageFormat, where a double apostrophe
926 // can be used in that case and will be formatted as a single one.
927 // (Better: Use a "real" apostrophe, U+2019.)
928 UnicodeString DATA [] = {
929 // Pattern 0 value 1 value
930 // {sfb} hacked - changed \u2264 to = (copied from Character Map)
931 "0#can't|1#can", "can't", "can",
932 "0#pound(#)='#''|1#xyz", "pound(#)='#''", "xyz",
933 "0#1<2 '| 1=1'|1#'", "1<2 '| 1=1'", "'",
934 };
935 for (int i=0; i<9; i+=3) {
936 //try {
937 UErrorCode status = U_ZERO_ERROR;
938 ChoiceFormat *cf = new ChoiceFormat(DATA[i], status);
939 failure(status, "new ChoiceFormat");
940 for (int j=0; j<=1; ++j) {
941 UnicodeString out;
942 FieldPosition pos(FieldPosition::DONT_CARE);
943 out = cf->format((double)j, out, pos);
944 if (out != DATA[i+1+j])
945 errln("Fail: Pattern \"" + DATA[i] + "\" x "+j+" -> " +
946 out + "; want \"" + DATA[i+1+j] + "\"");
947 }
948 UnicodeString pat;
949 pat = cf->toPattern(pat);
950 UnicodeString pat2;
951 ChoiceFormat *cf2 = new ChoiceFormat(pat, status);
952 pat2 = cf2->toPattern(pat2);
953 if (pat != pat2)
954 errln("Fail: Pattern \"" + DATA[i] + "\" x toPattern -> \"" + pat + "\"");
955 else
956 logln("Ok: Pattern \"" + DATA[i] + "\" x toPattern -> \"" + pat + "\"");
957 /*}
958 catch (IllegalArgumentException e) {
959 errln("Fail: Pattern \"" + DATA[i] + "\" -> " + e);
960 }*/
961
962 delete cf;
963 delete cf2;
964 }
965 }
966
967 /**
968 * @bug 4112104
969 * MessageFormat.equals(null) throws a NullPointerException. The JLS states
970 * that it should return false.
971 */
972 void MessageFormatRegressionTest::Test4112104()
973 {
974 UErrorCode status = U_ZERO_ERROR;
975 MessageFormat *format = new MessageFormat("", status);
976 failure(status, "new MessageFormat");
977 //try {
978 // This should NOT throw an exception
979 if (format == NULL) {
980 // It also should return false
981 errln("MessageFormat.equals(null) returns false");
982 }
983 /*}
984 catch (NullPointerException e) {
985 errln("MessageFormat.equals(null) throws " + e);
986 }*/
987 delete format;
988 }
989
990 void MessageFormatRegressionTest::TestAPI() {
991 UErrorCode status = U_ZERO_ERROR;
992 MessageFormat *format = new MessageFormat("", status);
993 failure(status, "new MessageFormat");
994
995 // Test adoptFormat
996 MessageFormat *fmt = new MessageFormat("",status);
997 format->adoptFormat("some_name",fmt,status); // Must at least pass a valid identifier.
998 failure(status, "adoptFormat");
999
1000 // Test getFormat
1001 format->setFormat((int32_t)0,*fmt);
1002 format->getFormat("some_other_name",status); // Must at least pass a valid identifier.
1003 failure(status, "getFormat");
1004 delete format;
1005 }
1006
1007 #endif /* #if !UCONFIG_NO_FORMATTING */