1 // Copyright (C) 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 || 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, 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, 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, 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 = 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