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) 2005-2016, International Business Machines Corporation and
6 * others. All Rights Reserved.
7 ********************************************************************/
8 #include "unicode/utypes.h"
9
10 #if !UCONFIG_NO_FORMATTING
11 #include "unicode/unum.h"
12 #include "unicode/ucurr.h"
13 #include "unicode/ustring.h"
14 #include "cintltst.h"
15 #include "cmemory.h"
16 #include "cstring.h"
17
expectInList(const char * isoCurrency,uint32_t currencyType,UBool isExpected)18 static void expectInList(const char *isoCurrency, uint32_t currencyType, UBool isExpected) {
19 UErrorCode status = U_ZERO_ERROR;
20 const char *foundCurrency = NULL;
21 const char *currentCurrency;
22 UEnumeration *en = ucurr_openISOCurrencies(currencyType, &status);
23 if (U_FAILURE(status)) {
24 log_err("Error: ucurr_openISOCurrencies returned %s\n", myErrorName(status));
25 return;
26 }
27
28 while ((currentCurrency = uenum_next(en, NULL, &status)) != NULL) {
29 if (strcmp(isoCurrency, currentCurrency) == 0) {
30 foundCurrency = currentCurrency;
31 break;
32 }
33 }
34
35 if ((foundCurrency != NULL) != isExpected) {
36 log_err("Error: could not find %s as expected. isExpected = %s type=0x%X\n",
37 isoCurrency, isExpected ? "TRUE" : "FALSE", currencyType);
38 }
39 uenum_close(en);
40 }
41
TestEnumList(void)42 static void TestEnumList(void) {
43 expectInList("ADP", UCURR_ALL, TRUE); /* First in list */
44 expectInList("ZWD", UCURR_ALL, TRUE); /* Last in list */
45
46 expectInList("USD", UCURR_ALL, TRUE);
47 expectInList("USD", UCURR_COMMON, TRUE);
48 expectInList("USD", UCURR_UNCOMMON, FALSE);
49 expectInList("USD", UCURR_DEPRECATED, FALSE);
50 expectInList("USD", UCURR_NON_DEPRECATED, TRUE);
51 expectInList("USD", UCURR_COMMON|UCURR_DEPRECATED, FALSE);
52 expectInList("USD", UCURR_COMMON|UCURR_NON_DEPRECATED, TRUE);
53 expectInList("USD", UCURR_UNCOMMON|UCURR_DEPRECATED, FALSE);
54 expectInList("USD", UCURR_UNCOMMON|UCURR_NON_DEPRECATED, FALSE);
55
56 expectInList("USN", UCURR_ALL, TRUE);
57 expectInList("USN", UCURR_COMMON, FALSE);
58 expectInList("USN", UCURR_UNCOMMON, TRUE);
59 expectInList("USN", UCURR_DEPRECATED, FALSE);
60 expectInList("USN", UCURR_NON_DEPRECATED, TRUE);
61 expectInList("USN", UCURR_COMMON|UCURR_DEPRECATED, FALSE);
62 expectInList("USN", UCURR_COMMON|UCURR_NON_DEPRECATED, FALSE);
63 expectInList("USN", UCURR_UNCOMMON|UCURR_DEPRECATED, FALSE);
64 expectInList("USN", UCURR_UNCOMMON|UCURR_NON_DEPRECATED, TRUE);
65
66 expectInList("DEM", UCURR_ALL, TRUE);
67 expectInList("DEM", UCURR_COMMON, TRUE);
68 expectInList("DEM", UCURR_UNCOMMON, FALSE);
69 expectInList("DEM", UCURR_DEPRECATED, TRUE);
70 expectInList("DEM", UCURR_NON_DEPRECATED, FALSE);
71 expectInList("DEM", UCURR_COMMON|UCURR_DEPRECATED, TRUE);
72 expectInList("DEM", UCURR_COMMON|UCURR_NON_DEPRECATED, FALSE);
73 expectInList("DEM", UCURR_UNCOMMON|UCURR_DEPRECATED, FALSE);
74 expectInList("DEM", UCURR_UNCOMMON|UCURR_NON_DEPRECATED, FALSE);
75
76 expectInList("XEU", UCURR_ALL, TRUE);
77 expectInList("XEU", UCURR_COMMON, FALSE);
78 expectInList("XEU", UCURR_UNCOMMON, TRUE);
79 expectInList("XEU", UCURR_DEPRECATED, TRUE);
80 expectInList("XEU", UCURR_NON_DEPRECATED, FALSE);
81 expectInList("XEU", UCURR_COMMON|UCURR_DEPRECATED, FALSE);
82 expectInList("XEU", UCURR_COMMON|UCURR_NON_DEPRECATED, FALSE);
83 expectInList("XEU", UCURR_UNCOMMON|UCURR_DEPRECATED, TRUE);
84 expectInList("XEU", UCURR_UNCOMMON|UCURR_NON_DEPRECATED, FALSE);
85
86 }
87
TestEnumListReset(void)88 static void TestEnumListReset(void) {
89 UErrorCode status = U_ZERO_ERROR;
90 const char *currency1;
91 const char *currency2;
92 UEnumeration *en = ucurr_openISOCurrencies(UCURR_ALL, &status);
93 if (U_FAILURE(status)) {
94 log_err("Error: ucurr_openISOCurrencies returned %s\n", myErrorName(status));
95 return;
96 }
97
98 currency1 = uenum_next(en, NULL, &status);
99 uenum_reset(en, &status);
100 currency2 = uenum_next(en, NULL, &status);
101 if (U_FAILURE(status)) {
102 log_err("Error: uenum_next or uenum_reset returned %s\n", myErrorName(status));
103 return;
104 }
105 /* The first item's pointer in the list should be the same between resets. */
106 if (currency1 != currency2) {
107 log_err("Error: reset doesn't work %s != %s\n", currency1, currency2);
108 }
109 uenum_close(en);
110 }
111
checkItemCount(uint32_t currencyType)112 static int32_t checkItemCount(uint32_t currencyType) {
113 UErrorCode status = U_ZERO_ERROR;
114 int32_t originalCount, count;
115 UEnumeration *en = ucurr_openISOCurrencies(currencyType, &status);
116 int32_t expectedLen = 3, len;
117 if (U_FAILURE(status)) {
118 log_err("Error: ucurr_openISOCurrencies returned %s\n", myErrorName(status));
119 return -1;
120 }
121
122 originalCount = uenum_count(en, &status);
123 for (count=0;;count++) {
124 const char *str = uenum_next(en, &len, &status);
125 if (str == NULL || len != expectedLen || (int32_t)strlen(str) != expectedLen) {
126 break;
127 }
128 }
129
130 if (originalCount != count) {
131 log_err("Error: uenum_count returned the wrong value (type = 0x%X). Got: %d Expected %d\n",
132 currencyType, count, originalCount);
133 }
134 if (U_FAILURE(status)) {
135 log_err("Error: uenum_next got an error: %s\n", u_errorName(status));
136 }
137 uenum_close(en);
138 return count;
139 }
140
TestEnumListCount(void)141 static void TestEnumListCount(void) {
142 checkItemCount(UCURR_ALL);
143 checkItemCount(UCURR_COMMON);
144 checkItemCount(UCURR_UNCOMMON);
145 checkItemCount(UCURR_DEPRECATED);
146 checkItemCount(UCURR_NON_DEPRECATED);
147 checkItemCount(UCURR_COMMON|UCURR_DEPRECATED);
148 checkItemCount(UCURR_COMMON|UCURR_NON_DEPRECATED);
149 checkItemCount(UCURR_UNCOMMON|UCURR_DEPRECATED);
150 checkItemCount(UCURR_UNCOMMON|UCURR_NON_DEPRECATED);
151
152 if (checkItemCount(UCURR_DEPRECATED|UCURR_NON_DEPRECATED) != 0) {
153 log_err("Error: UCURR_DEPRECATED|UCURR_NON_DEPRECATED should return 0 items\n");
154 }
155 if (checkItemCount(UCURR_COMMON|UCURR_UNCOMMON) != 0) {
156 log_err("Error: UCURR_DEPRECATED|UCURR_NON_DEPRECATED should return 0 items\n");
157 }
158 }
159
TestFractionDigitOverride(void)160 static void TestFractionDigitOverride(void) {
161 UErrorCode status = U_ZERO_ERROR;
162 UNumberFormat *fmt = unum_open(UNUM_CURRENCY, NULL, 0, "hu_HU", NULL, &status);
163 UChar buffer[256];
164 UChar expectedBuf[256];
165 const char expectedFirst[] = "123,46\\u00A0Ft"; /* changed to use 2 fraction digits */
166 const char expectedSecond[] = "123,46\\u00A0Ft";
167 const char expectedThird[] = "123,456\\u00A0Ft";
168 if (U_FAILURE(status)) {
169 log_data_err("Error: unum_open returned %s (Are you missing data?)\n", myErrorName(status));
170 return;
171 }
172 /* Make sure that you can format normal fraction digits. */
173 unum_formatDouble(fmt, 123.456, buffer, UPRV_LENGTHOF(buffer), NULL, &status);
174 u_unescape(expectedFirst, expectedBuf, (int32_t)strlen(expectedFirst)+1);
175 if (u_strcmp(buffer, expectedBuf) != 0) {
176 log_err("Error: unum_formatDouble didn't return %s\n", expectedFirst);
177 }
178 /* Make sure that you can format 2 fraction digits. */
179 unum_setAttribute(fmt, UNUM_FRACTION_DIGITS, 2);
180 unum_formatDouble(fmt, 123.456, buffer, UPRV_LENGTHOF(buffer), NULL, &status);
181 u_unescape(expectedSecond, expectedBuf, (int32_t)strlen(expectedSecond)+1);
182 if (u_strcmp(buffer, expectedBuf) != 0) {
183 log_err("Error: unum_formatDouble didn't return %s\n", expectedSecond);
184 }
185 /* Make sure that you can format more fraction digits. */
186 unum_setAttribute(fmt, UNUM_FRACTION_DIGITS, 3);
187 unum_formatDouble(fmt, 123.456, buffer, UPRV_LENGTHOF(buffer), NULL, &status);
188 u_unescape(expectedThird, expectedBuf, (int32_t)strlen(expectedThird)+1);
189 if (u_strcmp(buffer, expectedBuf) != 0) {
190 log_err("Error: unum_formatDouble didn't return %s\n", expectedThird);
191 }
192 unum_close(fmt);
193 }
194
TestPrefixSuffix(void)195 static void TestPrefixSuffix(void) {
196 int32_t pos;
197 UErrorCode status;
198 double result1 = 0.0, result2 = 0.0;
199 UNumberFormat* parser;
200 UChar buffer[4];
201 static const UChar TEST_NUMBER[] = {0x0024,0x0031,0x0032,0x002E,0x0030,0x0030,0}; /* $12.00 */
202 static const UChar NEG_PREFIX[] = {0x005B,0}; /* "[" */
203 static const UChar NEG_SUFFIX[] = {0x005D,0}; /* "]" */
204
205
206 status = U_ZERO_ERROR;
207 parser = unum_open(UNUM_CURRENCY, NULL, -1, "en_US", NULL, &status);
208 if (U_FAILURE(status)) {
209 log_data_err("Error: unum_open returned %s (Are you missing data?)\n", u_errorName(status));
210 return;
211 }
212
213 pos = 0;
214 status = U_ZERO_ERROR;
215 result1 = unum_parseDoubleCurrency(parser, TEST_NUMBER, -1, &pos, buffer, &status);
216
217 unum_setTextAttribute(parser, UNUM_NEGATIVE_SUFFIX, NEG_SUFFIX, -1, &status);
218 unum_setTextAttribute(parser, UNUM_NEGATIVE_PREFIX, NEG_PREFIX, -1, &status);
219 if (U_FAILURE(status)) {
220 log_err("Error: unum_setTextAttribute returned %s\n", u_errorName(status));
221 return;
222 }
223
224 pos = 0;
225 result2 = unum_parseDoubleCurrency(parser, TEST_NUMBER, -1, &pos, buffer, &status);
226 if (result1 != result2 || U_FAILURE(status)) {
227 log_err("Error: unum_parseDoubleCurrency didn't return the same value for same string %f %f %s\n",
228 result1, result2, u_errorName(status));
229 }
230 unum_close(parser);
231 }
232
233 typedef struct {
234 const char* alphaCode;
235 int32_t numericCode;
236 } NumCodeTestEntry;
237
238 static const NumCodeTestEntry NUMCODE_TESTDATA[] = {
239 {"USD", 840},
240 {"Usd", 840}, /* mixed casing */
241 {"EUR", 978},
242 {"JPY", 392},
243 {"XFU", 0}, /* XFU: no numeric code */
244 {"ZZZ", 0}, /* ZZZ: undefined ISO currency code */
245 {"bogus", 0}, /* bogus code */
246 {0, 0},
247 };
248
TestNumericCode(void)249 static void TestNumericCode(void) {
250 UChar code[8]; // at least one longer than the longest alphaCode
251 int32_t i;
252 int32_t numCode;
253
254 for (i = 0; NUMCODE_TESTDATA[i].alphaCode; i++) {
255 int32_t length = (int32_t)uprv_strlen(NUMCODE_TESTDATA[i].alphaCode);
256 u_charsToUChars(NUMCODE_TESTDATA[i].alphaCode, code, length + 1); // +1 includes the NUL
257 numCode = ucurr_getNumericCode(code);
258 if (numCode != NUMCODE_TESTDATA[i].numericCode) {
259 log_data_err("Error: ucurr_getNumericCode returned %d for currency %s, expected - %d\n",
260 numCode, NUMCODE_TESTDATA[i].alphaCode, NUMCODE_TESTDATA[i].numericCode);
261 }
262 }
263 }
264
265 void addCurrencyTest(TestNode** root);
266
267 #define TESTCASE(x) addTest(root, &x, "tsformat/currtest/" #x)
268
addCurrencyTest(TestNode ** root)269 void addCurrencyTest(TestNode** root)
270 {
271 TESTCASE(TestEnumList);
272 TESTCASE(TestEnumListReset);
273 TESTCASE(TestEnumListCount);
274 TESTCASE(TestFractionDigitOverride);
275 TESTCASE(TestPrefixSuffix);
276 TESTCASE(TestNumericCode);
277 }
278
279 #endif /* #if !UCONFIG_NO_FORMATTING */
280