1 /*
2 *******************************************************************************
3 * Copyright (C) 1996-2009, International Business Machines Corporation and *
4 * others. All Rights Reserved. *
5 *******************************************************************************
6 */
7
8 #include "unicode/utypes.h"
9
10 #if !UCONFIG_NO_FORMATTING
11
12 #include "itrbnfrt.h"
13
14 #include "unicode/fmtable.h"
15 #include <math.h> // fabs
16 #include <stdio.h>
17
18 // current macro not in icu1.8.1
19 #define TESTCASE(id,test) \
20 case id: \
21 name = #test; \
22 if (exec) { \
23 logln(#test "---"); \
24 logln(); \
25 test(); \
26 } \
27 break
28
runIndexedTest(int32_t index,UBool exec,const char * & name,char *)29 void RbnfRoundTripTest::runIndexedTest(int32_t index, UBool exec, const char* &name, char* /*par*/)
30 {
31 if (exec) logln("TestSuite RuleBasedNumberFormatRT");
32 switch (index) {
33 #if U_HAVE_RBNF
34 TESTCASE(0, TestEnglishSpelloutRT);
35 TESTCASE(1, TestDurationsRT);
36 TESTCASE(2, TestSpanishSpelloutRT);
37 TESTCASE(3, TestFrenchSpelloutRT);
38 TESTCASE(4, TestSwissFrenchSpelloutRT);
39 TESTCASE(5, TestItalianSpelloutRT);
40 TESTCASE(6, TestGermanSpelloutRT);
41 TESTCASE(7, TestSwedishSpelloutRT);
42 TESTCASE(8, TestDutchSpelloutRT);
43 TESTCASE(9, TestJapaneseSpelloutRT);
44 TESTCASE(10, TestRussianSpelloutRT);
45 TESTCASE(11, TestPortugueseSpelloutRT);
46 #else
47 TESTCASE(0, TestRBNFDisabled);
48 #endif
49 default:
50 name = "";
51 break;
52 }
53 }
54
55 #if U_HAVE_RBNF
56
57 /**
58 * Perform an exhaustive round-trip test on the English spellout rules
59 */
60 void
TestEnglishSpelloutRT()61 RbnfRoundTripTest::TestEnglishSpelloutRT()
62 {
63 UErrorCode status = U_ZERO_ERROR;
64 RuleBasedNumberFormat* formatter
65 = new RuleBasedNumberFormat(URBNF_SPELLOUT, Locale::getUS(), status);
66
67 if (U_FAILURE(status)) {
68 errcheckln(status, "failed to construct formatter - %s", u_errorName(status));
69 } else {
70 doTest(formatter, -12345678, 12345678);
71 }
72 delete formatter;
73 }
74
75 /**
76 * Perform an exhaustive round-trip test on the duration-formatting rules
77 */
78 void
TestDurationsRT()79 RbnfRoundTripTest::TestDurationsRT()
80 {
81 UErrorCode status = U_ZERO_ERROR;
82 RuleBasedNumberFormat* formatter
83 = new RuleBasedNumberFormat(URBNF_DURATION, Locale::getUS(), status);
84
85 if (U_FAILURE(status)) {
86 errcheckln(status, "failed to construct formatter - %s", u_errorName(status));
87 } else {
88 doTest(formatter, 0, 12345678);
89 }
90 delete formatter;
91 }
92
93 /**
94 * Perform an exhaustive round-trip test on the Spanish spellout rules
95 */
96 void
TestSpanishSpelloutRT()97 RbnfRoundTripTest::TestSpanishSpelloutRT()
98 {
99 UErrorCode status = U_ZERO_ERROR;
100 RuleBasedNumberFormat* formatter
101 = new RuleBasedNumberFormat(URBNF_SPELLOUT, Locale("es", "es"), status);
102
103 if (U_FAILURE(status)) {
104 errcheckln(status, "failed to construct formatter - %s", u_errorName(status));
105 } else {
106 doTest(formatter, -12345678, 12345678);
107 }
108 delete formatter;
109 }
110
111 /**
112 * Perform an exhaustive round-trip test on the French spellout rules
113 */
114 void
TestFrenchSpelloutRT()115 RbnfRoundTripTest::TestFrenchSpelloutRT()
116 {
117 UErrorCode status = U_ZERO_ERROR;
118 RuleBasedNumberFormat* formatter
119 = new RuleBasedNumberFormat(URBNF_SPELLOUT, Locale::getFrance(), status);
120
121 if (U_FAILURE(status)) {
122 errcheckln(status, "failed to construct formatter - %s", u_errorName(status));
123 } else {
124 doTest(formatter, -12345678, 12345678);
125 }
126 delete formatter;
127 }
128
129 /**
130 * Perform an exhaustive round-trip test on the Swiss French spellout rules
131 */
132 void
TestSwissFrenchSpelloutRT()133 RbnfRoundTripTest::TestSwissFrenchSpelloutRT()
134 {
135 UErrorCode status = U_ZERO_ERROR;
136 RuleBasedNumberFormat* formatter
137 = new RuleBasedNumberFormat(URBNF_SPELLOUT, Locale("fr", "CH"), status);
138
139 if (U_FAILURE(status)) {
140 errcheckln(status, "failed to construct formatter - %s", u_errorName(status));
141 } else {
142 doTest(formatter, -12345678, 12345678);
143 }
144 delete formatter;
145 }
146
147 /**
148 * Perform an exhaustive round-trip test on the Italian spellout rules
149 */
150 void
TestItalianSpelloutRT()151 RbnfRoundTripTest::TestItalianSpelloutRT()
152 {
153 UErrorCode status = U_ZERO_ERROR;
154 RuleBasedNumberFormat* formatter
155 = new RuleBasedNumberFormat(URBNF_SPELLOUT, Locale::getItalian(), status);
156
157 if (U_FAILURE(status)) {
158 errcheckln(status, "failed to construct formatter - %s", u_errorName(status));
159 } else {
160 doTest(formatter, -999999, 999999);
161 }
162 delete formatter;
163 }
164
165 /**
166 * Perform an exhaustive round-trip test on the German spellout rules
167 */
168 void
TestGermanSpelloutRT()169 RbnfRoundTripTest::TestGermanSpelloutRT()
170 {
171 UErrorCode status = U_ZERO_ERROR;
172 RuleBasedNumberFormat* formatter
173 = new RuleBasedNumberFormat(URBNF_SPELLOUT, Locale::getGermany(), status);
174
175 if (U_FAILURE(status)) {
176 errcheckln(status, "failed to construct formatter - %s", u_errorName(status));
177 } else {
178 doTest(formatter, 0, 12345678);
179 }
180 delete formatter;
181 }
182
183 /**
184 * Perform an exhaustive round-trip test on the Swedish spellout rules
185 */
186 void
TestSwedishSpelloutRT()187 RbnfRoundTripTest::TestSwedishSpelloutRT()
188 {
189 UErrorCode status = U_ZERO_ERROR;
190 RuleBasedNumberFormat* formatter
191 = new RuleBasedNumberFormat(URBNF_SPELLOUT, Locale("sv", "SE"), status);
192
193 if (U_FAILURE(status)) {
194 errcheckln(status, "failed to construct formatter - %s", u_errorName(status));
195 } else {
196 doTest(formatter, 0, 12345678);
197 }
198 delete formatter;
199 }
200
201 /**
202 * Perform an exhaustive round-trip test on the Dutch spellout rules
203 */
204 void
TestDutchSpelloutRT()205 RbnfRoundTripTest::TestDutchSpelloutRT()
206 {
207 UErrorCode status = U_ZERO_ERROR;
208 RuleBasedNumberFormat* formatter
209 = new RuleBasedNumberFormat(URBNF_SPELLOUT, Locale("nl", "NL"), status);
210
211 if (U_FAILURE(status)) {
212 errcheckln(status, "failed to construct formatter - %s", u_errorName(status));
213 } else {
214 doTest(formatter, -12345678, 12345678);
215 }
216 delete formatter;
217 }
218
219 /**
220 * Perform an exhaustive round-trip test on the Japanese spellout rules
221 */
222 void
TestJapaneseSpelloutRT()223 RbnfRoundTripTest::TestJapaneseSpelloutRT()
224 {
225 UErrorCode status = U_ZERO_ERROR;
226 RuleBasedNumberFormat* formatter
227 = new RuleBasedNumberFormat(URBNF_SPELLOUT, Locale::getJapan(), status);
228
229 if (U_FAILURE(status)) {
230 errcheckln(status, "failed to construct formatter - %s", u_errorName(status));
231 } else {
232 doTest(formatter, 0, 12345678);
233 }
234 delete formatter;
235 }
236
237 /**
238 * Perform an exhaustive round-trip test on the Russian spellout rules
239 */
240 void
TestRussianSpelloutRT()241 RbnfRoundTripTest::TestRussianSpelloutRT()
242 {
243 UErrorCode status = U_ZERO_ERROR;
244 RuleBasedNumberFormat* formatter
245 = new RuleBasedNumberFormat(URBNF_SPELLOUT, Locale("ru", "RU"), status);
246
247 if (U_FAILURE(status)) {
248 errcheckln(status, "failed to construct formatter - %s", u_errorName(status));
249 } else {
250 doTest(formatter, 0, 12345678);
251 }
252 delete formatter;
253 }
254
255 /**
256 * Perform an exhaustive round-trip test on the Portuguese spellout rules
257 */
258 void
TestPortugueseSpelloutRT()259 RbnfRoundTripTest::TestPortugueseSpelloutRT()
260 {
261 UErrorCode status = U_ZERO_ERROR;
262 RuleBasedNumberFormat* formatter
263 = new RuleBasedNumberFormat(URBNF_SPELLOUT, Locale("pt", "BR"), status);
264
265 if (U_FAILURE(status)) {
266 errcheckln(status, "failed to construct formatter - %s", u_errorName(status));
267 } else {
268 doTest(formatter, -12345678, 12345678);
269 }
270 delete formatter;
271 }
272
273 void
doTest(const RuleBasedNumberFormat * formatter,double lowLimit,double highLimit)274 RbnfRoundTripTest::doTest(const RuleBasedNumberFormat* formatter,
275 double lowLimit,
276 double highLimit)
277 {
278 char buf[128];
279
280 uint32_t count = 0;
281 double increment = 1;
282 for (double i = lowLimit; i <= highLimit; i += increment) {
283 if (count % 1000 == 0) {
284 sprintf(buf, "%.12g", i);
285 logln(buf);
286 }
287
288 if (fabs(i) < 5000)
289 increment = 1;
290 else if (fabs(i) < 500000)
291 increment = 2737;
292 else
293 increment = 267437;
294
295 UnicodeString formatResult;
296 formatter->format(i, formatResult);
297 UErrorCode status = U_ZERO_ERROR;
298 Formattable parseResult;
299 formatter->parse(formatResult, parseResult, status);
300 if (U_FAILURE(status)) {
301 sprintf(buf, "Round-trip status failure: %.12g, status: %d", i, status);
302 errln(buf);
303 return;
304 } else {
305 double rt = (parseResult.getType() == Formattable::kDouble) ?
306 parseResult.getDouble() :
307 (double)parseResult.getLong();
308
309 if (rt != i) {
310 sprintf(buf, "Round-trip failed: %.12g -> %.12g", i, rt);
311 errln(buf);
312 return;
313 }
314 }
315
316 ++count;
317 }
318
319 if (lowLimit < 0) {
320 double d = 1.234;
321 while (d < 1000) {
322 UnicodeString formatResult;
323 formatter->format(d, formatResult);
324 UErrorCode status = U_ZERO_ERROR;
325 Formattable parseResult;
326 formatter->parse(formatResult, parseResult, status);
327 if (U_FAILURE(status)) {
328 sprintf(buf, "Round-trip status failure: %.12g, status: %d", d, status);
329 errln(buf);
330 return;
331 } else {
332 double rt = (parseResult.getType() == Formattable::kDouble) ?
333 parseResult.getDouble() :
334 (double)parseResult.getLong();
335
336 if (rt != d) {
337 UnicodeString msg;
338 sprintf(buf, "Round-trip failed: %.12g -> ", d);
339 msg.append(buf);
340 msg.append(formatResult);
341 sprintf(buf, " -> %.12g", rt);
342 msg.append(buf);
343 errln(msg);
344 return;
345 }
346 }
347
348 d *= 10;
349 }
350 }
351 }
352
353 /* U_HAVE_RBNF */
354 #else
355
356 void
TestRBNFDisabled()357 RbnfRoundTripTest::TestRBNFDisabled() {
358 errln("*** RBNF currently disabled on this platform ***\n");
359 }
360
361 /* U_HAVE_RBNF */
362 #endif
363
364 #endif /* #if !UCONFIG_NO_FORMATTING */
365