1 // © 2016 and later: Unicode, Inc. and others.
2 // License & terms of use: http://www.unicode.org/copyright.html
3 /*
4 *******************************************************************************
5 * Copyright (C) 2014-2016, International Business Machines Corporation and
6 * others. All Rights Reserved.
7 *******************************************************************************
8 *
9 * simpleformattertest.cpp
10 *
11 ********************************************************************************
12 */
13 
14 #include "unicode/msgfmt.h"
15 #include "unicode/unistr.h"
16 #include "cstring.h"
17 #include "intltest.h"
18 #include "unicode/simpleformatter.h"
19 
20 class SimpleFormatterTest : public IntlTest {
21 public:
SimpleFormatterTest()22     SimpleFormatterTest() {
23     }
24     void TestNoArguments();
25     void TestSyntaxErrors();
26     void TestOneArgument();
27     void TestBigArgument();
28     void TestManyArguments();
29     void TestTooFewArgumentValues();
30     void TestBadArguments();
31     void TestTextWithNoArguments();
32     void TestFormatReplaceNoOptimization();
33     void TestFormatReplaceNoOptimizationLeadingText();
34     void TestFormatReplaceOptimization();
35     void TestFormatReplaceNoOptimizationLeadingArgumentUsedTwice();
36     void TestFormatReplaceOptimizationNoOffsets();
37     void TestFormatReplaceNoOptimizationNoOffsets();
38     void TestQuotingLikeMessageFormat();
39     void runIndexedTest(int32_t index, UBool exec, const char *&name, char *par=0);
40 private:
41     void verifyOffsets(
42             const int32_t *expected,
43             const int32_t *actual,
44             int32_t count);
45 };
46 
runIndexedTest(int32_t index,UBool exec,const char * & name,char *)47 void SimpleFormatterTest::runIndexedTest(int32_t index, UBool exec, const char* &name, char* /*par*/) {
48   TESTCASE_AUTO_BEGIN;
49   TESTCASE_AUTO(TestNoArguments);
50   TESTCASE_AUTO(TestSyntaxErrors);
51   TESTCASE_AUTO(TestOneArgument);
52   TESTCASE_AUTO(TestBigArgument);
53   TESTCASE_AUTO(TestManyArguments);
54   TESTCASE_AUTO(TestTooFewArgumentValues);
55   TESTCASE_AUTO(TestBadArguments);
56   TESTCASE_AUTO(TestTextWithNoArguments);
57   TESTCASE_AUTO(TestFormatReplaceNoOptimization);
58   TESTCASE_AUTO(TestFormatReplaceNoOptimizationLeadingText);
59   TESTCASE_AUTO(TestFormatReplaceOptimization);
60   TESTCASE_AUTO(TestFormatReplaceNoOptimizationLeadingArgumentUsedTwice);
61   TESTCASE_AUTO(TestFormatReplaceOptimizationNoOffsets);
62   TESTCASE_AUTO(TestFormatReplaceNoOptimizationNoOffsets);
63   TESTCASE_AUTO(TestQuotingLikeMessageFormat);
64   TESTCASE_AUTO_END;
65 }
66 
TestNoArguments()67 void SimpleFormatterTest::TestNoArguments() {
68     UErrorCode status = U_ZERO_ERROR;
69     SimpleFormatter fmt("This doesn''t have templates '{0}", status);
70     assertEquals("getArgumentLimit", 0, fmt.getArgumentLimit());
71     UnicodeString appendTo;
72     assertEquals(
73             "format",
74             "This doesn't have templates {0}",
75             fmt.format("unused", appendTo, status));
76     appendTo.remove();
77     int32_t offsets[] = { 0 };
78     assertEquals(
79             "formatAndAppend",
80             "This doesn't have templates {0}",
81             fmt.formatAndAppend(NULL, 0, appendTo, offsets, 1, status));
82     assertEquals("formatAndAppend offsets[0]", -1, offsets[0]);
83     assertEquals(
84             "formatAndReplace",
85             "This doesn't have templates {0}",
86             fmt.formatAndReplace(NULL, 0, appendTo, NULL, 0, status));
87     assertSuccess("Status", status);
88 }
89 
TestSyntaxErrors()90 void SimpleFormatterTest::TestSyntaxErrors() {
91     UErrorCode status = U_ZERO_ERROR;
92     SimpleFormatter fmt("{}", status);
93     assertEquals("syntax error {}", (int32_t)U_ILLEGAL_ARGUMENT_ERROR, status);
94     status = U_ZERO_ERROR;
95     fmt.applyPattern("{12d", status);
96     assertEquals("syntax error {12d", (int32_t)U_ILLEGAL_ARGUMENT_ERROR, status);
97 }
98 
TestOneArgument()99 void SimpleFormatterTest::TestOneArgument() {
100     UErrorCode status = U_ZERO_ERROR;
101     SimpleFormatter fmt;
102     fmt.applyPattern("{0} meter", status);
103     if (!assertSuccess("Status", status)) {
104         return;
105     }
106     assertEquals("getArgumentLimit", 1, fmt.getArgumentLimit());
107     UnicodeString appendTo;
108     assertEquals(
109             "format",
110             "1 meter",
111             fmt.format("1", appendTo, status));
112 
113     // assignment
114     SimpleFormatter s;
115     s = fmt;
116     appendTo.remove();
117     assertEquals(
118             "Assignment",
119             "1 meter",
120             s.format("1", appendTo, status));
121 
122     // Copy constructor
123     SimpleFormatter r(fmt);
124     appendTo.remove();
125     assertEquals(
126             "Copy constructor",
127             "1 meter",
128             r.format("1", appendTo, status));
129     assertSuccess("Status", status);
130 }
131 
TestBigArgument()132 void SimpleFormatterTest::TestBigArgument() {
133     UErrorCode status = U_ZERO_ERROR;
134     SimpleFormatter fmt("a{20}c", status);
135     if (!assertSuccess("Status", status)) {
136         return;
137     }
138     assertEquals("{20} count", 21, fmt.getArgumentLimit());
139     UnicodeString b("b");
140     UnicodeString *values[] = {
141         NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
142         NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
143         &b
144     };
145     UnicodeString result;
146     assertEquals("{20}=b", "abc", fmt.formatAndAppend(values, 21, result, NULL, 0, status));
147     assertSuccess("Status", status);
148 }
149 
TestManyArguments()150 void SimpleFormatterTest::TestManyArguments() {
151     UErrorCode status = U_ZERO_ERROR;
152     SimpleFormatter fmt;
153     fmt.applyPattern(
154             "Templates {2}{1}{5} and {4} are out of order.", status);
155     if (!assertSuccess("Status", status)) {
156         return;
157     }
158     assertEquals("getArgumentLimit", 6, fmt.getArgumentLimit());
159     UnicodeString values[] = {
160             "freddy", "tommy", "frog", "billy", "leg", "{0}"};
161     UnicodeString *params[] = {
162            &values[0], &values[1], &values[2], &values[3], &values[4], &values[5]};
163     int32_t offsets[6];
164     int32_t expectedOffsets[6] = {-1, 22, 18, -1, 35, 27};
165     UnicodeString appendTo("Prefix: ");
166     assertEquals(
167             "format",
168             "Prefix: Templates frogtommy{0} and leg are out of order.",
169             fmt.formatAndAppend(
170                     params,
171                     UPRV_LENGTHOF(params),
172                     appendTo,
173                     offsets,
174                     UPRV_LENGTHOF(offsets),
175                     status));
176     if (!assertSuccess("Status", status)) {
177         return;
178     }
179     verifyOffsets(expectedOffsets, offsets, UPRV_LENGTHOF(expectedOffsets));
180     appendTo.remove();
181 
182     // Ensure we don't write to offsets array beyond its length.
183     status = U_ZERO_ERROR;
184     offsets[UPRV_LENGTHOF(offsets) - 1] = 289;
185     appendTo.remove();
186     fmt.formatAndAppend(
187             params,
188             UPRV_LENGTHOF(params),
189             appendTo,
190             offsets,
191             UPRV_LENGTHOF(offsets) - 1,
192             status);
193     assertEquals("Offsets buffer length", 289, offsets[UPRV_LENGTHOF(offsets) - 1]);
194 
195     // Test assignment
196     SimpleFormatter s;
197     s = fmt;
198     appendTo.remove();
199     assertEquals(
200             "Assignment",
201             "Templates frogtommy{0} and leg are out of order.",
202             s.formatAndAppend(
203                     params,
204                     UPRV_LENGTHOF(params),
205                     appendTo,
206                     NULL,
207                     0,
208                     status));
209 
210     // Copy constructor
211     SimpleFormatter r(fmt);
212     appendTo.remove();
213     assertEquals(
214             "Copy constructor",
215             "Templates frogtommy{0} and leg are out of order.",
216             r.formatAndAppend(
217                     params,
218                     UPRV_LENGTHOF(params),
219                     appendTo,
220                     NULL,
221                     0,
222                     status));
223     r.applyPattern("{0} meter", status);
224     assertEquals("getArgumentLimit", 1, r.getArgumentLimit());
225     appendTo.remove();
226     assertEquals(
227             "Replace with new applyPattern",
228             "freddy meter",
229             r.format("freddy", appendTo, status));
230     r.applyPattern("{0}, {1}", status);
231     assertEquals("getArgumentLimit", 2, r.getArgumentLimit());
232     appendTo.remove();
233     assertEquals(
234             "2 arg",
235             "foo, bar",
236             r.format("foo", "bar", appendTo, status));
237     r.applyPattern("{0}, {1} and {2}", status);
238     assertEquals("getArgumentLimit", 3, r.getArgumentLimit());
239     appendTo.remove();
240     assertEquals(
241             "3 arg",
242             "foo, bar and baz",
243             r.format("foo", "bar", "baz", appendTo, status));
244     assertSuccess("Status", status);
245 }
246 
TestTooFewArgumentValues()247 void SimpleFormatterTest::TestTooFewArgumentValues() {
248     UErrorCode status = U_ZERO_ERROR;
249     SimpleFormatter fmt("{0} and {1}", status);
250     UnicodeString appendTo;
251     UnicodeString firstValue;
252     UnicodeString *params[] = {&firstValue};
253 
254     fmt.format(
255             firstValue, appendTo, status);
256     if (status != U_ILLEGAL_ARGUMENT_ERROR) {
257         errln("Expected U_ILLEGAL_ARGUMENT_ERROR");
258     }
259 
260     status = U_ZERO_ERROR;
261     fmt.formatAndAppend(
262             params, UPRV_LENGTHOF(params), appendTo, NULL, 0, status);
263     if (status != U_ILLEGAL_ARGUMENT_ERROR) {
264         errln("Expected U_ILLEGAL_ARGUMENT_ERROR");
265     }
266 
267     status = U_ZERO_ERROR;
268     fmt.formatAndReplace(
269             params, UPRV_LENGTHOF(params), appendTo, NULL, 0, status);
270     if (status != U_ILLEGAL_ARGUMENT_ERROR) {
271         errln("Expected U_ILLEGAL_ARGUMENT_ERROR");
272     }
273 }
274 
TestBadArguments()275 void SimpleFormatterTest::TestBadArguments() {
276     UErrorCode status = U_ZERO_ERROR;
277     SimpleFormatter fmt("pickle", status);
278     UnicodeString appendTo;
279 
280     // These succeed
281     fmt.formatAndAppend(
282             NULL, 0, appendTo, NULL, 0, status);
283     fmt.formatAndReplace(
284             NULL, 0, appendTo, NULL, 0, status);
285     assertSuccess("", status);
286     status = U_ZERO_ERROR;
287 
288     // fails
289     fmt.formatAndAppend(
290             NULL, 1, appendTo, NULL, 0, status);
291     if (status != U_ILLEGAL_ARGUMENT_ERROR) {
292         errln("Expected U_ILLEGAL_ARGUMENT_ERROR: formatAndAppend() values=NULL but length=1");
293     }
294     status = U_ZERO_ERROR;
295 
296     // fails
297     fmt.formatAndAppend(
298             NULL, 0, appendTo, NULL, 1, status);
299     if (status != U_ILLEGAL_ARGUMENT_ERROR) {
300         errln("Expected U_ILLEGAL_ARGUMENT_ERROR: formatAndAppend() offsets=NULL but length=1");
301     }
302     status = U_ZERO_ERROR;
303 
304     // fails because appendTo used as a parameter value
305     SimpleFormatter fmt2("Arguments {0} and {1}", status);
306     UnicodeString frog("frog");
307     const UnicodeString *params[] = { &appendTo, &frog };
308     fmt2.formatAndAppend(params, 2, appendTo, NULL, 0, status);
309     if (status != U_ILLEGAL_ARGUMENT_ERROR) {
310         errln("Expected U_ILLEGAL_ARGUMENT_ERROR: formatAndAppend() value=appendTo");
311     }
312     status = U_ZERO_ERROR;
313 
314 
315     // fails
316     fmt.formatAndReplace(
317             NULL, 1, appendTo, NULL, 0, status);
318     if (status != U_ILLEGAL_ARGUMENT_ERROR) {
319         errln("Expected U_ILLEGAL_ARGUMENT_ERROR: formatAndReplace() values=NULL but length=1");
320     }
321     status = U_ZERO_ERROR;
322 
323     // fails
324     fmt.formatAndReplace(
325             NULL, 0, appendTo, NULL, 1, status);
326     if (status != U_ILLEGAL_ARGUMENT_ERROR) {
327         errln("Expected U_ILLEGAL_ARGUMENT_ERROR: formatAndReplace() offsets=NULL but length=1");
328     }
329 }
330 
TestTextWithNoArguments()331 void SimpleFormatterTest::TestTextWithNoArguments() {
332     UErrorCode status = U_ZERO_ERROR;
333     SimpleFormatter fmt("{0} has no {1} arguments.", status);
334     assertEquals(
335             "", " has no  arguments.", fmt.getTextWithNoArguments());
336 }
337 
TestFormatReplaceNoOptimization()338 void SimpleFormatterTest::TestFormatReplaceNoOptimization() {
339     UErrorCode status = U_ZERO_ERROR;
340     SimpleFormatter fmt;
341     fmt.applyPattern("{2}, {0}, {1} and {3}", status);
342     if (!assertSuccess("Status", status)) {
343         return;
344     }
345     UnicodeString result("original");
346     int32_t offsets[4];
347     UnicodeString freddy("freddy");
348     UnicodeString frog("frog");
349     UnicodeString by("by");
350     const UnicodeString *params[] = {&result, &freddy, &frog, &by};
351     assertEquals(
352             "",
353             "frog, original, freddy and by",
354             fmt.formatAndReplace(
355                     params,
356                     UPRV_LENGTHOF(params),
357                     result,
358                     offsets,
359                     UPRV_LENGTHOF(offsets),
360                     status));
361     if (!assertSuccess("Status", status)) {
362         return;
363     }
364     int32_t expectedOffsets[] = {6, 16, 0, 27};
365     verifyOffsets(expectedOffsets, offsets, UPRV_LENGTHOF(expectedOffsets));
366 }
367 
TestFormatReplaceNoOptimizationLeadingText()368 void SimpleFormatterTest::TestFormatReplaceNoOptimizationLeadingText() {
369     UErrorCode status = U_ZERO_ERROR;
370     SimpleFormatter fmt;
371     fmt.applyPattern("boo {2}, {0}, {1} and {3}", status);
372     if (!assertSuccess("Status", status)) {
373         return;
374     }
375     UnicodeString result("original");
376     int32_t offsets[4];
377     UnicodeString freddy("freddy");
378     UnicodeString frog("frog");
379     UnicodeString by("by");
380     const UnicodeString *params[] = {&freddy, &frog, &result, &by};
381     assertEquals(
382             "",
383             "boo original, freddy, frog and by",
384             fmt.formatAndReplace(
385                     params,
386                     UPRV_LENGTHOF(params),
387                     result,
388                     offsets,
389                     UPRV_LENGTHOF(offsets),
390                     status));
391     if (!assertSuccess("Status", status)) {
392         return;
393     }
394     int32_t expectedOffsets[] = {14, 22, 4, 31};
395     verifyOffsets(expectedOffsets, offsets, UPRV_LENGTHOF(expectedOffsets));
396 }
397 
TestFormatReplaceOptimization()398 void SimpleFormatterTest::TestFormatReplaceOptimization() {
399     UErrorCode status = U_ZERO_ERROR;
400     SimpleFormatter fmt;
401     fmt.applyPattern("{2}, {0}, {1} and {3}", status);
402     if (!assertSuccess("Status", status)) {
403         return;
404     }
405     UnicodeString result("original");
406     int32_t offsets[4];
407     UnicodeString freddy("freddy");
408     UnicodeString frog("frog");
409     UnicodeString by("by");
410     const UnicodeString *params[] = {&freddy, &frog, &result, &by};
411     assertEquals(
412             "",
413             "original, freddy, frog and by",
414             fmt.formatAndReplace(
415                     params,
416                     UPRV_LENGTHOF(params),
417                     result,
418                     offsets,
419                     UPRV_LENGTHOF(offsets),
420                     status));
421     if (!assertSuccess("Status", status)) {
422         return;
423     }
424     int32_t expectedOffsets[] = {10, 18, 0, 27};
425     verifyOffsets(expectedOffsets, offsets, UPRV_LENGTHOF(expectedOffsets));
426 }
427 
TestFormatReplaceNoOptimizationLeadingArgumentUsedTwice()428 void SimpleFormatterTest::TestFormatReplaceNoOptimizationLeadingArgumentUsedTwice() {
429     UErrorCode status = U_ZERO_ERROR;
430     SimpleFormatter fmt;
431     fmt.applyPattern("{2}, {0}, {1} and {3} {2}", status);
432     if (!assertSuccess("Status", status)) {
433         return;
434     }
435     UnicodeString result("original");
436     int32_t offsets[4];
437     UnicodeString freddy("freddy");
438     UnicodeString frog("frog");
439     UnicodeString by("by");
440     const UnicodeString *params[] = {&freddy, &frog, &result, &by};
441     assertEquals(
442             "",
443             "original, freddy, frog and by original",
444             fmt.formatAndReplace(
445                     params,
446                     UPRV_LENGTHOF(params),
447                     result,
448                     offsets,
449                     UPRV_LENGTHOF(offsets),
450                     status));
451     if (!assertSuccess("Status", status)) {
452         return;
453     }
454     int32_t expectedOffsets[] = {10, 18, 30, 27};
455     verifyOffsets(expectedOffsets, offsets, UPRV_LENGTHOF(expectedOffsets));
456 }
457 
TestFormatReplaceOptimizationNoOffsets()458 void SimpleFormatterTest::TestFormatReplaceOptimizationNoOffsets() {
459     UErrorCode status = U_ZERO_ERROR;
460     SimpleFormatter fmt;
461     fmt.applyPattern("{2}, {0}, {1} and {3}", status);
462     if (!assertSuccess("Status", status)) {
463         return;
464     }
465     UnicodeString result("original");
466     UnicodeString freddy("freddy");
467     UnicodeString frog("frog");
468     UnicodeString by("by");
469     const UnicodeString *params[] = {&freddy, &frog, &result, &by};
470     assertEquals(
471             "",
472             "original, freddy, frog and by",
473             fmt.formatAndReplace(
474                     params,
475                     UPRV_LENGTHOF(params),
476                     result,
477                     NULL,
478                     0,
479                     status));
480     assertSuccess("Status", status);
481 }
482 
TestFormatReplaceNoOptimizationNoOffsets()483 void SimpleFormatterTest::TestFormatReplaceNoOptimizationNoOffsets() {
484     UErrorCode status = U_ZERO_ERROR;
485     SimpleFormatter fmt("Arguments {0} and {1}", status);
486     UnicodeString result("previous:");
487     UnicodeString frog("frog");
488     const UnicodeString *params[] = {&result, &frog};
489     assertEquals(
490             "",
491             "Arguments previous: and frog",
492             fmt.formatAndReplace(
493                     params,
494                     UPRV_LENGTHOF(params),
495                     result,
496                     NULL,
497                     0,
498                     status));
499     assertSuccess("Status", status);
500 }
501 
TestQuotingLikeMessageFormat()502 void SimpleFormatterTest::TestQuotingLikeMessageFormat() {
503 #if !UCONFIG_NO_FORMATTING
504     UErrorCode status = U_ZERO_ERROR;
505     UnicodeString pattern = "{0} don't can''t '{5}''}{a' again '}'{1} to the '{end";
506     SimpleFormatter spf(pattern, status);
507     MessageFormat mf(pattern, Locale::getRoot(), status);
508     UnicodeString expected = "X don't can't {5}'}{a again }Y to the {end";
509     UnicodeString x("X"), y("Y");
510     Formattable values[] = { x, y };
511     UnicodeString result;
512     FieldPosition ignore(FieldPosition::DONT_CARE);
513     assertEquals("MessageFormat", expected, mf.format(values, 2, result, ignore, status));
514     assertEquals("SimpleFormatter", expected, spf.format(x, y, result.remove(), status));
515 #endif /* !UCONFIG_NO_FORMATTING */
516 }
517 
verifyOffsets(const int32_t * expected,const int32_t * actual,int32_t count)518 void SimpleFormatterTest::verifyOffsets(
519         const int32_t *expected, const int32_t *actual, int32_t count) {
520     for (int32_t i = 0; i < count; ++i) {
521         if (expected[i] != actual[i]) {
522             errln("Expected %d, got %d", expected[i], actual[i]);
523         }
524     }
525 }
526 
createSimpleFormatterTest()527 extern IntlTest *createSimpleFormatterTest() {
528     return new SimpleFormatterTest();
529 }
530