1 /*
2 **********************************************************************
3 * Copyright (C) 1999-2014, International Business Machines
4 * Corporation and others. All Rights Reserved.
5 **********************************************************************
6 * Date Name Description
7 * 11/10/99 aliu Creation.
8 **********************************************************************
9 */
10
11 #include "unicode/utypes.h"
12
13 #if !UCONFIG_NO_TRANSLITERATION
14
15 #include "transtst.h"
16 #include "unicode/locid.h"
17 #include "unicode/dtfmtsym.h"
18 #include "unicode/normlzr.h"
19 #include "unicode/translit.h"
20 #include "unicode/uchar.h"
21 #include "unicode/unifilt.h"
22 #include "unicode/uniset.h"
23 #include "unicode/ustring.h"
24 #include "unicode/usetiter.h"
25 #include "unicode/uscript.h"
26 #include "unicode/utf16.h"
27 #include "cpdtrans.h"
28 #include "nultrans.h"
29 #include "rbt.h"
30 #include "rbt_pars.h"
31 #include "anytrans.h"
32 #include "esctrn.h"
33 #include "name2uni.h"
34 #include "nortrans.h"
35 #include "remtrans.h"
36 #include "titletrn.h"
37 #include "tolowtrn.h"
38 #include "toupptrn.h"
39 #include "unesctrn.h"
40 #include "uni2name.h"
41 #include "cstring.h"
42 #include "cmemory.h"
43 #include <stdio.h>
44
45 /***********************************************************************
46
47 HOW TO USE THIS TEST FILE
48 -or-
49 How I developed on two platforms
50 without losing (too much of) my mind
51
52
53 1. Add new tests by copying/pasting/changing existing tests. On Java,
54 any public void method named Test...() taking no parameters becomes
55 a test. On C++, you need to modify the header and add a line to
56 the runIndexedTest() dispatch method.
57
58 2. Make liberal use of the expect() method; it is your friend.
59
60 3. The tests in this file exactly match those in a sister file on the
61 other side. The two files are:
62
63 icu4j: src/com/ibm/test/translit/TransliteratorTest.java
64 icu4c: source/test/intltest/transtst.cpp
65
66 ==> THIS IS THE IMPORTANT PART <==
67
68 When you add a test in this file, add it in TransliteratorTest.java
69 too. Give it the same name and put it in the same relative place.
70 This makes maintenance a lot simpler for any poor soul who ends up
71 trying to synchronize the tests between icu4j and icu4c.
72
73 4. If you MUST enter a test that is NOT paralleled in the sister file,
74 then add it in the special non-mirrored section. These are
75 labeled
76
77 "icu4j ONLY"
78
79 or
80
81 "icu4c ONLY"
82
83 Make sure you document the reason the test is here and not there.
84
85
86 Thank you.
87 The Management
88 ***********************************************************************/
89
90 // Define character constants thusly to be EBCDIC-friendly
91 enum {
92 LEFT_BRACE=((UChar)0x007B), /*{*/
93 PIPE =((UChar)0x007C), /*|*/
94 ZERO =((UChar)0x0030), /*0*/
95 UPPER_A =((UChar)0x0041) /*A*/
96 };
97
TransliteratorTest()98 TransliteratorTest::TransliteratorTest()
99 : DESERET_DEE((UChar32)0x10414),
100 DESERET_dee((UChar32)0x1043C)
101 {
102 }
103
~TransliteratorTest()104 TransliteratorTest::~TransliteratorTest() {}
105
106 void
runIndexedTest(int32_t index,UBool exec,const char * & name,char *)107 TransliteratorTest::runIndexedTest(int32_t index, UBool exec,
108 const char* &name, char* /*par*/) {
109 switch (index) {
110 TESTCASE(0,TestInstantiation);
111 TESTCASE(1,TestSimpleRules);
112 TESTCASE(2,TestRuleBasedInverse);
113 TESTCASE(3,TestKeyboard);
114 TESTCASE(4,TestKeyboard2);
115 TESTCASE(5,TestKeyboard3);
116 TESTCASE(6,TestArabic);
117 TESTCASE(7,TestCompoundKana);
118 TESTCASE(8,TestCompoundHex);
119 TESTCASE(9,TestFiltering);
120 TESTCASE(10,TestInlineSet);
121 TESTCASE(11,TestPatternQuoting);
122 TESTCASE(12,TestJ277);
123 TESTCASE(13,TestJ243);
124 TESTCASE(14,TestJ329);
125 TESTCASE(15,TestSegments);
126 TESTCASE(16,TestCursorOffset);
127 TESTCASE(17,TestArbitraryVariableValues);
128 TESTCASE(18,TestPositionHandling);
129 TESTCASE(19,TestHiraganaKatakana);
130 TESTCASE(20,TestCopyJ476);
131 TESTCASE(21,TestAnchors);
132 TESTCASE(22,TestInterIndic);
133 TESTCASE(23,TestFilterIDs);
134 TESTCASE(24,TestCaseMap);
135 TESTCASE(25,TestNameMap);
136 TESTCASE(26,TestLiberalizedID);
137 TESTCASE(27,TestCreateInstance);
138 TESTCASE(28,TestNormalizationTransliterator);
139 TESTCASE(29,TestCompoundRBT);
140 TESTCASE(30,TestCompoundFilter);
141 TESTCASE(31,TestRemove);
142 TESTCASE(32,TestToRules);
143 TESTCASE(33,TestContext);
144 TESTCASE(34,TestSupplemental);
145 TESTCASE(35,TestQuantifier);
146 TESTCASE(36,TestSTV);
147 TESTCASE(37,TestCompoundInverse);
148 TESTCASE(38,TestNFDChainRBT);
149 TESTCASE(39,TestNullInverse);
150 TESTCASE(40,TestAliasInverseID);
151 TESTCASE(41,TestCompoundInverseID);
152 TESTCASE(42,TestUndefinedVariable);
153 TESTCASE(43,TestEmptyContext);
154 TESTCASE(44,TestCompoundFilterID);
155 TESTCASE(45,TestPropertySet);
156 TESTCASE(46,TestNewEngine);
157 TESTCASE(47,TestQuantifiedSegment);
158 TESTCASE(48,TestDevanagariLatinRT);
159 TESTCASE(49,TestTeluguLatinRT);
160 TESTCASE(50,TestCompoundLatinRT);
161 TESTCASE(51,TestSanskritLatinRT);
162 TESTCASE(52,TestLocaleInstantiation);
163 TESTCASE(53,TestTitleAccents);
164 TESTCASE(54,TestLocaleResource);
165 TESTCASE(55,TestParseError);
166 TESTCASE(56,TestOutputSet);
167 TESTCASE(57,TestVariableRange);
168 TESTCASE(58,TestInvalidPostContext);
169 TESTCASE(59,TestIDForms);
170 TESTCASE(60,TestToRulesMark);
171 TESTCASE(61,TestEscape);
172 TESTCASE(62,TestAnchorMasking);
173 TESTCASE(63,TestDisplayName);
174 TESTCASE(64,TestSpecialCases);
175 #if !UCONFIG_NO_FILE_IO
176 TESTCASE(65,TestIncrementalProgress);
177 #endif
178 TESTCASE(66,TestSurrogateCasing);
179 TESTCASE(67,TestFunction);
180 TESTCASE(68,TestInvalidBackRef);
181 TESTCASE(69,TestMulticharStringSet);
182 TESTCASE(70,TestUserFunction);
183 TESTCASE(71,TestAnyX);
184 TESTCASE(72,TestSourceTargetSet);
185 TESTCASE(73,TestGurmukhiDevanagari);
186 TESTCASE(74,TestPatternWhiteSpace);
187 TESTCASE(75,TestAllCodepoints);
188 TESTCASE(76,TestBoilerplate);
189 TESTCASE(77,TestAlternateSyntax);
190 TESTCASE(78,TestBeginEnd);
191 TESTCASE(79,TestBeginEndToRules);
192 TESTCASE(80,TestRegisterAlias);
193 TESTCASE(81,TestRuleStripping);
194 TESTCASE(82,TestHalfwidthFullwidth);
195 TESTCASE(83,TestThai);
196 TESTCASE(84,TestAny);
197 default: name = ""; break;
198 }
199 }
200
201 /**
202 * Make sure every system transliterator can be instantiated.
203 *
204 * ALSO test that the result of toRules() for each rule is a valid
205 * rule. Do this here so we don't have to have another test that
206 * instantiates everything as well.
207 */
TestInstantiation()208 void TransliteratorTest::TestInstantiation() {
209 UErrorCode ec = U_ZERO_ERROR;
210 StringEnumeration* avail = Transliterator::getAvailableIDs(ec);
211 assertSuccess("getAvailableIDs()", ec);
212 assertTrue("getAvailableIDs()!=NULL", avail!=NULL);
213 int32_t n = Transliterator::countAvailableIDs();
214 assertTrue("getAvailableIDs().count()==countAvailableIDs()",
215 avail->count(ec) == n);
216 assertSuccess("count()", ec);
217 UnicodeString name;
218 for (int32_t i=0; i<n; ++i) {
219 const UnicodeString& id = *avail->snext(ec);
220 if (!assertSuccess("snext()", ec) ||
221 !assertTrue("snext()!=NULL", (&id)!=NULL, TRUE)) {
222 break;
223 }
224 UnicodeString id2 = Transliterator::getAvailableID(i);
225 if (id.length() < 1) {
226 errln(UnicodeString("FAIL: getAvailableID(") +
227 i + ") returned empty string");
228 continue;
229 }
230 if (id != id2) {
231 errln(UnicodeString("FAIL: getAvailableID(") +
232 i + ") != getAvailableIDs().snext()");
233 continue;
234 }
235 UParseError parseError;
236 UErrorCode status = U_ZERO_ERROR;
237 Transliterator* t = Transliterator::createInstance(id,
238 UTRANS_FORWARD, parseError,status);
239 name.truncate(0);
240 Transliterator::getDisplayName(id, name);
241 if (t == 0) {
242 #if UCONFIG_NO_BREAK_ITERATION
243 // If UCONFIG_NO_BREAK_ITERATION is on, then only Thai should fail.
244 if (id.compare((UnicodeString)"Thai-Latin") != 0)
245 #endif
246 dataerrln(UnicodeString("FAIL: Couldn't create ") + id +
247 /*", parse error " + parseError.code +*/
248 ", line " + parseError.line +
249 ", offset " + parseError.offset +
250 ", pre-context " + prettify(parseError.preContext, TRUE) +
251 ", post-context " +prettify(parseError.postContext,TRUE) +
252 ", Error: " + u_errorName(status));
253 // When createInstance fails, it deletes the failing
254 // entry from the available ID list. We detect this
255 // here by looking for a change in countAvailableIDs.
256 int32_t nn = Transliterator::countAvailableIDs();
257 if (nn == (n - 1)) {
258 n = nn;
259 --i; // Compensate for deleted entry
260 }
261 } else {
262 logln(UnicodeString("OK: ") + name + " (" + id + ")");
263
264 // Now test toRules
265 UnicodeString rules;
266 t->toRules(rules, TRUE);
267 Transliterator *u = Transliterator::createFromRules("x",
268 rules, UTRANS_FORWARD, parseError,status);
269 if (u == 0) {
270 errln(UnicodeString("FAIL: ") + id +
271 ".createFromRules() => bad rules" +
272 /*", parse error " + parseError.code +*/
273 ", line " + parseError.line +
274 ", offset " + parseError.offset +
275 ", context " + prettify(parseError.preContext, TRUE) +
276 ", rules: " + prettify(rules, TRUE));
277 } else {
278 delete u;
279 }
280 delete t;
281 }
282 }
283 assertTrue("snext()==NULL", avail->snext(ec)==NULL);
284 assertSuccess("snext()", ec);
285 delete avail;
286
287 // Now test the failure path
288 UParseError parseError;
289 UErrorCode status = U_ZERO_ERROR;
290 UnicodeString id("<Not a valid Transliterator ID>");
291 Transliterator* t = Transliterator::createInstance(id, UTRANS_FORWARD, parseError, status);
292 if (t != 0) {
293 errln("FAIL: " + id + " returned a transliterator");
294 delete t;
295 } else {
296 logln("OK: Bogus ID handled properly");
297 }
298 }
299
TestSimpleRules(void)300 void TransliteratorTest::TestSimpleRules(void) {
301 /* Example: rules 1. ab>x|y
302 * 2. yc>z
303 *
304 * []|eabcd start - no match, copy e to tranlated buffer
305 * [e]|abcd match rule 1 - copy output & adjust cursor
306 * [ex|y]cd match rule 2 - copy output & adjust cursor
307 * [exz]|d no match, copy d to transliterated buffer
308 * [exzd]| done
309 */
310 expect(UnicodeString("ab>x|y;", "") +
311 "yc>z",
312 "eabcd", "exzd");
313
314 /* Another set of rules:
315 * 1. ab>x|yzacw
316 * 2. za>q
317 * 3. qc>r
318 * 4. cw>n
319 *
320 * []|ab Rule 1
321 * [x|yzacw] No match
322 * [xy|zacw] Rule 2
323 * [xyq|cw] Rule 4
324 * [xyqn]| Done
325 */
326 expect(UnicodeString("ab>x|yzacw;") +
327 "za>q;" +
328 "qc>r;" +
329 "cw>n",
330 "ab", "xyqn");
331
332 /* Test categories
333 */
334 UErrorCode status = U_ZERO_ERROR;
335 UParseError parseError;
336 Transliterator *t = Transliterator::createFromRules(
337 "<ID>",
338 UnicodeString("$dummy=").append((UChar)0xE100) +
339 UnicodeString(";"
340 "$vowel=[aeiouAEIOU];"
341 "$lu=[:Lu:];"
342 "$vowel } $lu > '!';"
343 "$vowel > '&';"
344 "'!' { $lu > '^';"
345 "$lu > '*';"
346 "a > ERROR", ""),
347 UTRANS_FORWARD, parseError,
348 status);
349 if (U_FAILURE(status)) {
350 dataerrln("FAIL: RBT constructor failed - %s", u_errorName(status));
351 return;
352 }
353 expect(*t, "abcdefgABCDEFGU", "&bcd&fg!^**!^*&");
354 delete t;
355 }
356
357 /**
358 * Test inline set syntax and set variable syntax.
359 */
TestInlineSet(void)360 void TransliteratorTest::TestInlineSet(void) {
361 expect("{ [:Ll:] } x > y; [:Ll:] > z;", "aAbxq", "zAyzz");
362 expect("a[0-9]b > qrs", "1a7b9", "1qrs9");
363
364 expect(UnicodeString(
365 "$digit = [0-9];"
366 "$alpha = [a-zA-Z];"
367 "$alphanumeric = [$digit $alpha];" // ***
368 "$special = [^$alphanumeric];" // ***
369 "$alphanumeric > '-';"
370 "$special > '*';", ""),
371
372 "thx-1138", "---*----");
373 }
374
375 /**
376 * Create some inverses and confirm that they work. We have to be
377 * careful how we do this, since the inverses will not be true
378 * inverses -- we can't throw any random string at the composition
379 * of the transliterators and expect the identity function. F x
380 * F' != I. However, if we are careful about the input, we will
381 * get the expected results.
382 */
TestRuleBasedInverse(void)383 void TransliteratorTest::TestRuleBasedInverse(void) {
384 UnicodeString RULES =
385 UnicodeString("abc>zyx;") +
386 "ab>yz;" +
387 "bc>zx;" +
388 "ca>xy;" +
389 "a>x;" +
390 "b>y;" +
391 "c>z;" +
392
393 "abc<zyx;" +
394 "ab<yz;" +
395 "bc<zx;" +
396 "ca<xy;" +
397 "a<x;" +
398 "b<y;" +
399 "c<z;" +
400
401 "";
402
403 const char* DATA[] = {
404 // Careful here -- random strings will not work. If we keep
405 // the left side to the domain and the right side to the range
406 // we will be okay though (left, abc; right xyz).
407 "a", "x",
408 "abcacab", "zyxxxyy",
409 "caccb", "xyzzy",
410 };
411
412 int32_t DATA_length = (int32_t)(sizeof(DATA) / sizeof(DATA[0]));
413
414 UErrorCode status = U_ZERO_ERROR;
415 UParseError parseError;
416 Transliterator *fwd = Transliterator::createFromRules("<ID>", RULES,
417 UTRANS_FORWARD, parseError, status);
418 Transliterator *rev = Transliterator::createFromRules("<ID>", RULES,
419 UTRANS_REVERSE, parseError, status);
420 if (U_FAILURE(status)) {
421 errln("FAIL: RBT constructor failed");
422 return;
423 }
424 for (int32_t i=0; i<DATA_length; i+=2) {
425 expect(*fwd, DATA[i], DATA[i+1]);
426 expect(*rev, DATA[i+1], DATA[i]);
427 }
428 delete fwd;
429 delete rev;
430 }
431
432 /**
433 * Basic test of keyboard.
434 */
TestKeyboard(void)435 void TransliteratorTest::TestKeyboard(void) {
436 UParseError parseError;
437 UErrorCode status = U_ZERO_ERROR;
438 Transliterator *t = Transliterator::createFromRules("<ID>",
439 UnicodeString("psch>Y;")
440 +"ps>y;"
441 +"ch>x;"
442 +"a>A;",
443 UTRANS_FORWARD, parseError,
444 status);
445 if (U_FAILURE(status)) {
446 errln("FAIL: RBT constructor failed");
447 return;
448 }
449 const char* DATA[] = {
450 // insertion, buffer
451 "a", "A",
452 "p", "Ap",
453 "s", "Aps",
454 "c", "Apsc",
455 "a", "AycA",
456 "psch", "AycAY",
457 0, "AycAY", // null means finishKeyboardTransliteration
458 };
459
460 keyboardAux(*t, DATA, (int32_t)(sizeof(DATA)/sizeof(DATA[0])));
461 delete t;
462 }
463
464 /**
465 * Basic test of keyboard with cursor.
466 */
TestKeyboard2(void)467 void TransliteratorTest::TestKeyboard2(void) {
468 UParseError parseError;
469 UErrorCode status = U_ZERO_ERROR;
470 Transliterator *t = Transliterator::createFromRules("<ID>",
471 UnicodeString("ych>Y;")
472 +"ps>|y;"
473 +"ch>x;"
474 +"a>A;",
475 UTRANS_FORWARD, parseError,
476 status);
477 if (U_FAILURE(status)) {
478 errln("FAIL: RBT constructor failed");
479 return;
480 }
481 const char* DATA[] = {
482 // insertion, buffer
483 "a", "A",
484 "p", "Ap",
485 "s", "Aps", // modified for rollback - "Ay",
486 "c", "Apsc", // modified for rollback - "Ayc",
487 "a", "AycA",
488 "p", "AycAp",
489 "s", "AycAps", // modified for rollback - "AycAy",
490 "c", "AycApsc", // modified for rollback - "AycAyc",
491 "h", "AycAY",
492 0, "AycAY", // null means finishKeyboardTransliteration
493 };
494
495 keyboardAux(*t, DATA, (int32_t)(sizeof(DATA)/sizeof(DATA[0])));
496 delete t;
497 }
498
499 /**
500 * Test keyboard transliteration with back-replacement.
501 */
TestKeyboard3(void)502 void TransliteratorTest::TestKeyboard3(void) {
503 // We want th>z but t>y. Furthermore, during keyboard
504 // transliteration we want t>y then yh>z if t, then h are
505 // typed.
506 UnicodeString RULES("t>|y;"
507 "yh>z;");
508
509 const char* DATA[] = {
510 // Column 1: characters to add to buffer (as if typed)
511 // Column 2: expected appearance of buffer after
512 // keyboard xliteration.
513 "a", "a",
514 "b", "ab",
515 "t", "abt", // modified for rollback - "aby",
516 "c", "abyc",
517 "t", "abyct", // modified for rollback - "abycy",
518 "h", "abycz",
519 0, "abycz", // null means finishKeyboardTransliteration
520 };
521
522 UParseError parseError;
523 UErrorCode status = U_ZERO_ERROR;
524 Transliterator *t = Transliterator::createFromRules("<ID>", RULES, UTRANS_FORWARD, parseError, status);
525 if (U_FAILURE(status)) {
526 errln("FAIL: RBT constructor failed");
527 return;
528 }
529 keyboardAux(*t, DATA, (int32_t)(sizeof(DATA)/sizeof(DATA[0])));
530 delete t;
531 }
532
keyboardAux(const Transliterator & t,const char * DATA[],int32_t DATA_length)533 void TransliteratorTest::keyboardAux(const Transliterator& t,
534 const char* DATA[], int32_t DATA_length) {
535 UErrorCode status = U_ZERO_ERROR;
536 UTransPosition index={0, 0, 0, 0};
537 UnicodeString s;
538 for (int32_t i=0; i<DATA_length; i+=2) {
539 UnicodeString log;
540 if (DATA[i] != 0) {
541 log = s + " + "
542 + DATA[i]
543 + " -> ";
544 t.transliterate(s, index, DATA[i], status);
545 } else {
546 log = s + " => ";
547 t.finishTransliteration(s, index);
548 }
549 // Show the start index '{' and the cursor '|'
550 UnicodeString a, b, c;
551 s.extractBetween(0, index.contextStart, a);
552 s.extractBetween(index.contextStart, index.start, b);
553 s.extractBetween(index.start, s.length(), c);
554 log.append(a).
555 append((UChar)LEFT_BRACE).
556 append(b).
557 append((UChar)PIPE).
558 append(c);
559 if (s == DATA[i+1] && U_SUCCESS(status)) {
560 logln(log);
561 } else {
562 errln(UnicodeString("FAIL: ") + log + ", expected " + DATA[i+1]);
563 }
564 }
565 }
566
TestArabic(void)567 void TransliteratorTest::TestArabic(void) {
568 // Test disabled for 2.0 until new Arabic transliterator can be written.
569 // /*
570 // const char* DATA[] = {
571 // "Arabic", "\u062a\u062a\u0645\u062a\u0639\u0020"+
572 // "\u0627\u0644\u0644\u063a\u0629\u0020"+
573 // "\u0627\u0644\u0639\u0631\u0628\u0628\u064a\u0629\u0020"+
574 // "\u0628\u0628\u0646\u0638\u0645\u0020"+
575 // "\u0643\u062a\u0627\u0628\u0628\u064a\u0629\u0020"+
576 // "\u062c\u0645\u064a\u0644\u0629",
577 // };
578 // */
579 //
580 // UChar ar_raw[] = {
581 // 0x062a, 0x062a, 0x0645, 0x062a, 0x0639, 0x0020, 0x0627,
582 // 0x0644, 0x0644, 0x063a, 0x0629, 0x0020, 0x0627, 0x0644,
583 // 0x0639, 0x0631, 0x0628, 0x0628, 0x064a, 0x0629, 0x0020,
584 // 0x0628, 0x0628, 0x0646, 0x0638, 0x0645, 0x0020, 0x0643,
585 // 0x062a, 0x0627, 0x0628, 0x0628, 0x064a, 0x0629, 0x0020,
586 // 0x062c, 0x0645, 0x064a, 0x0644, 0x0629, 0
587 // };
588 // UnicodeString ar(ar_raw);
589 // UErrorCode status=U_ZERO_ERROR;
590 // UParseError parseError;
591 // Transliterator *t = Transliterator::createInstance("Latin-Arabic", UTRANS_FORWARD, parseError, status);
592 // if (t == 0) {
593 // errln("FAIL: createInstance failed");
594 // return;
595 // }
596 // expect(*t, "Arabic", ar);
597 // delete t;
598 }
599
600 /**
601 * Compose the Kana transliterator forward and reverse and try
602 * some strings that should come out unchanged.
603 */
TestCompoundKana(void)604 void TransliteratorTest::TestCompoundKana(void) {
605 UParseError parseError;
606 UErrorCode status = U_ZERO_ERROR;
607 Transliterator* t = Transliterator::createInstance("Latin-Hiragana;Hiragana-Latin", UTRANS_FORWARD, parseError, status);
608 if (t == 0) {
609 dataerrln("FAIL: construction of Latin-Hiragana;Hiragana-Latin failed - %s", u_errorName(status));
610 } else {
611 expect(*t, "aaaaa", "aaaaa");
612 delete t;
613 }
614 }
615
616 /**
617 * Compose the hex transliterators forward and reverse.
618 */
TestCompoundHex(void)619 void TransliteratorTest::TestCompoundHex(void) {
620 UParseError parseError;
621 UErrorCode status = U_ZERO_ERROR;
622 Transliterator* a = Transliterator::createInstance("Any-Hex", UTRANS_FORWARD, parseError, status);
623 Transliterator* b = Transliterator::createInstance("Hex-Any", UTRANS_FORWARD, parseError, status);
624 Transliterator* transab[] = { a, b };
625 Transliterator* transba[] = { b, a };
626 if (a == 0 || b == 0) {
627 errln("FAIL: construction failed");
628 delete a;
629 delete b;
630 return;
631 }
632 // Do some basic tests of a
633 expect(*a, "01", UnicodeString("\\u0030\\u0031", ""));
634 // Do some basic tests of b
635 expect(*b, UnicodeString("\\u0030\\u0031", ""), "01");
636
637 Transliterator* ab = new CompoundTransliterator(transab, 2);
638 UnicodeString s("abcde", "");
639 expect(*ab, s, s);
640
641 UnicodeString str(s);
642 a->transliterate(str);
643 Transliterator* ba = new CompoundTransliterator(transba, 2);
644 expect(*ba, str, str);
645
646 delete ab;
647 delete ba;
648 delete a;
649 delete b;
650 }
651
652 int gTestFilterClassID = 0;
653 /**
654 * Used by TestFiltering().
655 */
656 class TestFilter : public UnicodeFilter {
clone() const657 virtual UnicodeFunctor* clone() const {
658 return new TestFilter(*this);
659 }
contains(UChar32 c) const660 virtual UBool contains(UChar32 c) const {
661 return c != (UChar)0x0063 /*c*/;
662 }
663 // Stubs
toPattern(UnicodeString & result,UBool) const664 virtual UnicodeString& toPattern(UnicodeString& result,
665 UBool /*escapeUnprintable*/) const {
666 return result;
667 }
matchesIndexValue(uint8_t) const668 virtual UBool matchesIndexValue(uint8_t /*v*/) const {
669 return FALSE;
670 }
addMatchSetTo(UnicodeSet &) const671 virtual void addMatchSetTo(UnicodeSet& /*toUnionTo*/) const {}
672 public:
getDynamicClassID() const673 UClassID getDynamicClassID() const { return (UClassID)&gTestFilterClassID; }
674 };
675
676 /**
677 * Do some basic tests of filtering.
678 */
TestFiltering(void)679 void TransliteratorTest::TestFiltering(void) {
680 UParseError parseError;
681 UErrorCode status = U_ZERO_ERROR;
682 Transliterator* hex = Transliterator::createInstance("Any-Hex", UTRANS_FORWARD, parseError, status);
683 if (hex == 0) {
684 errln("FAIL: createInstance(Any-Hex) failed");
685 return;
686 }
687 hex->adoptFilter(new TestFilter());
688 UnicodeString s("abcde");
689 hex->transliterate(s);
690 UnicodeString exp("\\u0061\\u0062c\\u0064\\u0065", "");
691 if (s == exp) {
692 logln(UnicodeString("Ok: \"") + exp + "\"");
693 } else {
694 logln(UnicodeString("FAIL: \"") + s + "\", wanted \"" + exp + "\"");
695 }
696
697 // ICU4C ONLY. Do not find Transliterator.orphanFilter() in ICU4J.
698 UnicodeFilter *f = hex->orphanFilter();
699 if (f == NULL){
700 errln("FAIL: orphanFilter() should get a UnicodeFilter");
701 } else {
702 delete f;
703 }
704 delete hex;
705 }
706
707 /**
708 * Test anchors
709 */
TestAnchors(void)710 void TransliteratorTest::TestAnchors(void) {
711 expect(UnicodeString("^a > 0; a$ > 2 ; a > 1;", ""),
712 "aaa",
713 "012");
714 expect(UnicodeString("$s=[z$]; $s{a>0; a}$s>2; a>1;", ""),
715 "aaa",
716 "012");
717 expect(UnicodeString("^ab > 01 ;"
718 " ab > |8 ;"
719 " b > k ;"
720 " 8x$ > 45 ;"
721 " 8x > 77 ;", ""),
722
723 "ababbabxabx",
724 "018k7745");
725 expect(UnicodeString("$s = [z$] ;"
726 "$s{ab > 01 ;"
727 " ab > |8 ;"
728 " b > k ;"
729 " 8x}$s > 45 ;"
730 " 8x > 77 ;", ""),
731
732 "abzababbabxzabxabx",
733 "01z018k45z01x45");
734 }
735
736 /**
737 * Test pattern quoting and escape mechanisms.
738 */
TestPatternQuoting(void)739 void TransliteratorTest::TestPatternQuoting(void) {
740 // Array of 3n items
741 // Each item is <rules>, <input>, <expected output>
742 const UnicodeString DATA[] = {
743 UnicodeString(UChar(0x4E01)) + ">'[male adult]'",
744 UnicodeString(UChar(0x4E01)),
745 "[male adult]"
746 };
747
748 for (int32_t i=0; i<3; i+=3) {
749 logln(UnicodeString("Pattern: ") + prettify(DATA[i]));
750 UParseError parseError;
751 UErrorCode status = U_ZERO_ERROR;
752 Transliterator *t = Transliterator::createFromRules("<ID>", DATA[i], UTRANS_FORWARD, parseError, status);
753 if (U_FAILURE(status)) {
754 errln("RBT constructor failed");
755 } else {
756 expect(*t, DATA[i+1], DATA[i+2]);
757 }
758 delete t;
759 }
760 }
761
762 /**
763 * Regression test for bugs found in Greek transliteration.
764 */
TestJ277(void)765 void TransliteratorTest::TestJ277(void) {
766 UErrorCode status = U_ZERO_ERROR;
767 UParseError parseError;
768 Transliterator *gl = Transliterator::createInstance("Greek-Latin; NFD; [:M:]Remove; NFC", UTRANS_FORWARD, parseError, status);
769 if (gl == NULL) {
770 dataerrln("FAIL: createInstance(Greek-Latin) returned NULL - %s", u_errorName(status));
771 return;
772 }
773
774 UChar sigma = 0x3C3;
775 UChar upsilon = 0x3C5;
776 UChar nu = 0x3BD;
777 // UChar PHI = 0x3A6;
778 UChar alpha = 0x3B1;
779 // UChar omega = 0x3C9;
780 // UChar omicron = 0x3BF;
781 // UChar epsilon = 0x3B5;
782
783 // sigma upsilon nu -> syn
784 UnicodeString syn;
785 syn.append(sigma).append(upsilon).append(nu);
786 expect(*gl, syn, "syn");
787
788 // sigma alpha upsilon nu -> saun
789 UnicodeString sayn;
790 sayn.append(sigma).append(alpha).append(upsilon).append(nu);
791 expect(*gl, sayn, "saun");
792
793 // Again, using a smaller rule set
794 UnicodeString rules(
795 "$alpha = \\u03B1;"
796 "$nu = \\u03BD;"
797 "$sigma = \\u03C3;"
798 "$ypsilon = \\u03C5;"
799 "$vowel = [aeiouAEIOU$alpha$ypsilon];"
800 "s <> $sigma;"
801 "a <> $alpha;"
802 "u <> $vowel { $ypsilon;"
803 "y <> $ypsilon;"
804 "n <> $nu;",
805 "");
806 Transliterator *mini = Transliterator::createFromRules("mini", rules, UTRANS_REVERSE, parseError, status);
807 if (U_FAILURE(status)) { errln("FAIL: Transliterator constructor failed"); return; }
808 expect(*mini, syn, "syn");
809 expect(*mini, sayn, "saun");
810 delete mini;
811 mini = NULL;
812
813 #if !UCONFIG_NO_FORMATTING
814 // Transliterate the Greek locale data
815 Locale el("el");
816 DateFormatSymbols syms(el, status);
817 if (U_FAILURE(status)) { errln("FAIL: Transliterator constructor failed"); return; }
818 int32_t i, count;
819 const UnicodeString* data = syms.getMonths(count);
820 for (i=0; i<count; ++i) {
821 if (data[i].length() == 0) {
822 continue;
823 }
824 UnicodeString out(data[i]);
825 gl->transliterate(out);
826 UBool ok = TRUE;
827 if (data[i].length() >= 2 && out.length() >= 2 &&
828 u_isupper(data[i].charAt(0)) && u_islower(data[i].charAt(1))) {
829 if (!(u_isupper(out.charAt(0)) && u_islower(out.charAt(1)))) {
830 ok = FALSE;
831 }
832 }
833 if (ok) {
834 logln(prettify(data[i] + " -> " + out));
835 } else {
836 errln(UnicodeString("FAIL: ") + prettify(data[i] + " -> " + out));
837 }
838 }
839 #endif
840
841 delete gl;
842 }
843
844 /**
845 * Prefix, suffix support in hex transliterators
846 */
TestJ243(void)847 void TransliteratorTest::TestJ243(void) {
848 UErrorCode ec = U_ZERO_ERROR;
849
850 // Test default Hex-Any, which should handle
851 // \u, \U, u+, and U+
852 Transliterator *hex =
853 Transliterator::createInstance("Hex-Any", UTRANS_FORWARD, ec);
854 if (assertSuccess("getInstance", ec)) {
855 expect(*hex, UnicodeString("\\u0041+\\U00000042,U+0043uU+0044z", ""), "A+B,CuDz");
856 }
857 delete hex;
858
859 // // Try a custom Hex-Unicode
860 // // \uXXXX and &#xXXXX;
861 // ec = U_ZERO_ERROR;
862 // HexToUnicodeTransliterator hex2(UnicodeString("\\\\u###0;&\\#x###0\\;", ""), ec);
863 // expect(hex2, UnicodeString("\\u61\\u062\\u0063\\u00645\\u66x0123", ""),
864 // "abcd5fx0123");
865 // // Try custom Any-Hex (default is tested elsewhere)
866 // ec = U_ZERO_ERROR;
867 // UnicodeToHexTransliterator hex3(UnicodeString("&\\#x###0;", ""), ec);
868 // expect(hex3, "012", "012");
869 }
870
871 /**
872 * Parsers need better syntax error messages.
873 */
TestJ329(void)874 void TransliteratorTest::TestJ329(void) {
875
876 struct { UBool containsErrors; const char* rule; } DATA[] = {
877 { FALSE, "a > b; c > d" },
878 { TRUE, "a > b; no operator; c > d" },
879 };
880 int32_t DATA_length = (int32_t)(sizeof(DATA) / sizeof(DATA[0]));
881
882 for (int32_t i=0; i<DATA_length; ++i) {
883 UErrorCode status = U_ZERO_ERROR;
884 UParseError parseError;
885 Transliterator *rbt = Transliterator::createFromRules("<ID>",
886 DATA[i].rule,
887 UTRANS_FORWARD,
888 parseError,
889 status);
890 UBool gotError = U_FAILURE(status);
891 UnicodeString desc(DATA[i].rule);
892 desc.append(gotError ? " -> error" : " -> no error");
893 if (gotError) {
894 desc = desc + ", ParseError code=" + u_errorName(status) +
895 " line=" + parseError.line +
896 " offset=" + parseError.offset +
897 " context=" + parseError.preContext;
898 }
899 if (gotError == DATA[i].containsErrors) {
900 logln(UnicodeString("Ok: ") + desc);
901 } else {
902 errln(UnicodeString("FAIL: ") + desc);
903 }
904 delete rbt;
905 }
906 }
907
908 /**
909 * Test segments and segment references.
910 */
TestSegments(void)911 void TransliteratorTest::TestSegments(void) {
912 // Array of 3n items
913 // Each item is <rules>, <input>, <expected output>
914 UnicodeString DATA[] = {
915 "([a-z]) '.' ([0-9]) > $2 '-' $1",
916 "abc.123.xyz.456",
917 "ab1-c23.xy4-z56",
918
919 // nested
920 "(([a-z])([0-9])) > $1 '.' $2 '.' $3;",
921 "a1 b2",
922 "a1.a.1 b2.b.2",
923 };
924 int32_t DATA_length = (int32_t)(sizeof(DATA)/sizeof(*DATA));
925
926 for (int32_t i=0; i<DATA_length; i+=3) {
927 logln("Pattern: " + prettify(DATA[i]));
928 UParseError parseError;
929 UErrorCode status = U_ZERO_ERROR;
930 Transliterator *t = Transliterator::createFromRules("ID", DATA[i], UTRANS_FORWARD, parseError, status);
931 if (U_FAILURE(status)) {
932 errln("FAIL: RBT constructor");
933 } else {
934 expect(*t, DATA[i+1], DATA[i+2]);
935 }
936 delete t;
937 }
938 }
939
940 /**
941 * Test cursor positioning outside of the key
942 */
TestCursorOffset(void)943 void TransliteratorTest::TestCursorOffset(void) {
944 // Array of 3n items
945 // Each item is <rules>, <input>, <expected output>
946 UnicodeString DATA[] = {
947 "pre {alpha} post > | @ ALPHA ;"
948 "eALPHA > beta ;"
949 "pre {beta} post > BETA @@ | ;"
950 "post > xyz",
951
952 "prealphapost prebetapost",
953
954 "prbetaxyz preBETApost",
955 };
956 int32_t DATA_length = (int32_t)(sizeof(DATA)/sizeof(*DATA));
957
958 for (int32_t i=0; i<DATA_length; i+=3) {
959 logln("Pattern: " + prettify(DATA[i]));
960 UParseError parseError;
961 UErrorCode status = U_ZERO_ERROR;
962 Transliterator *t = Transliterator::createFromRules("<ID>", DATA[i], UTRANS_FORWARD, parseError, status);
963 if (U_FAILURE(status)) {
964 errln("FAIL: RBT constructor");
965 } else {
966 expect(*t, DATA[i+1], DATA[i+2]);
967 }
968 delete t;
969 }
970 }
971
972 /**
973 * Test zero length and > 1 char length variable values. Test
974 * use of variable refs in UnicodeSets.
975 */
TestArbitraryVariableValues(void)976 void TransliteratorTest::TestArbitraryVariableValues(void) {
977 // Array of 3n items
978 // Each item is <rules>, <input>, <expected output>
979 UnicodeString DATA[] = {
980 "$abe = ab;"
981 "$pat = x[yY]z;"
982 "$ll = 'a-z';"
983 "$llZ = [$ll];"
984 "$llY = [$ll$pat];"
985 "$emp = ;"
986
987 "$abe > ABE;"
988 "$pat > END;"
989 "$llZ > 1;"
990 "$llY > 2;"
991 "7$emp 8 > 9;"
992 "",
993
994 "ab xYzxyz stY78",
995 "ABE ENDEND 1129",
996 };
997 int32_t DATA_length = (int32_t)(sizeof(DATA)/sizeof(*DATA));
998
999 for (int32_t i=0; i<DATA_length; i+=3) {
1000 logln("Pattern: " + prettify(DATA[i]));
1001 UParseError parseError;
1002 UErrorCode status = U_ZERO_ERROR;
1003 Transliterator *t = Transliterator::createFromRules("<ID>", DATA[i], UTRANS_FORWARD, parseError, status);
1004 if (U_FAILURE(status)) {
1005 errln("FAIL: RBT constructor");
1006 } else {
1007 expect(*t, DATA[i+1], DATA[i+2]);
1008 }
1009 delete t;
1010 }
1011 }
1012
1013 /**
1014 * Confirm that the contextStart, contextLimit, start, and limit
1015 * behave correctly. J474.
1016 */
TestPositionHandling(void)1017 void TransliteratorTest::TestPositionHandling(void) {
1018 // Array of 3n items
1019 // Each item is <rules>, <input>, <expected output>
1020 const char* DATA[] = {
1021 "a{t} > SS ; {t}b > UU ; {t} > TT ;",
1022 "xtat txtb", // pos 0,9,0,9
1023 "xTTaSS TTxUUb",
1024
1025 "a{t} > SS ; {t}b > UU ; {t} > TT ; a > A ; b > B ;",
1026 "xtat txtb", // pos 2,9,3,8
1027 "xtaSS TTxUUb",
1028
1029 "a{t} > SS ; {t}b > UU ; {t} > TT ; a > A ; b > B ;",
1030 "xtat txtb", // pos 3,8,3,8
1031 "xtaTT TTxTTb",
1032 };
1033
1034 // Array of 4n positions -- these go with the DATA array
1035 // They are: contextStart, contextLimit, start, limit
1036 int32_t POS[] = {
1037 0, 9, 0, 9,
1038 2, 9, 3, 8,
1039 3, 8, 3, 8,
1040 };
1041
1042 int32_t n = (int32_t)(sizeof(DATA) / sizeof(DATA[0])) / 3;
1043 for (int32_t i=0; i<n; i++) {
1044 UErrorCode status = U_ZERO_ERROR;
1045 UParseError parseError;
1046 Transliterator *t = Transliterator::createFromRules("<ID>",
1047 DATA[3*i], UTRANS_FORWARD, parseError, status);
1048 if (U_FAILURE(status)) {
1049 delete t;
1050 errln("FAIL: RBT constructor");
1051 return;
1052 }
1053 UTransPosition pos;
1054 pos.contextStart= POS[4*i];
1055 pos.contextLimit = POS[4*i+1];
1056 pos.start = POS[4*i+2];
1057 pos.limit = POS[4*i+3];
1058 UnicodeString rsource(DATA[3*i+1]);
1059 t->transliterate(rsource, pos, status);
1060 if (U_FAILURE(status)) {
1061 delete t;
1062 errln("FAIL: transliterate");
1063 return;
1064 }
1065 t->finishTransliteration(rsource, pos);
1066 expectAux(DATA[3*i],
1067 DATA[3*i+1],
1068 rsource,
1069 DATA[3*i+2]);
1070 delete t;
1071 }
1072 }
1073
1074 /**
1075 * Test the Hiragana-Katakana transliterator.
1076 */
TestHiraganaKatakana(void)1077 void TransliteratorTest::TestHiraganaKatakana(void) {
1078 UParseError parseError;
1079 UErrorCode status = U_ZERO_ERROR;
1080 Transliterator* hk = Transliterator::createInstance("Hiragana-Katakana", UTRANS_FORWARD, parseError, status);
1081 Transliterator* kh = Transliterator::createInstance("Katakana-Hiragana", UTRANS_FORWARD, parseError, status);
1082 if (hk == 0 || kh == 0) {
1083 dataerrln("FAIL: createInstance failed - %s", u_errorName(status));
1084 delete hk;
1085 delete kh;
1086 return;
1087 }
1088
1089 // Array of 3n items
1090 // Each item is "hk"|"kh"|"both", <Hiragana>, <Katakana>
1091 const char* DATA[] = {
1092 "both",
1093 "\\u3042\\u3090\\u3099\\u3092\\u3050",
1094 "\\u30A2\\u30F8\\u30F2\\u30B0",
1095
1096 "kh",
1097 "\\u307C\\u3051\\u3060\\u3042\\u3093\\u30FC",
1098 "\\u30DC\\u30F6\\u30C0\\u30FC\\u30F3\\u30FC",
1099 };
1100 int32_t DATA_length = (int32_t)(sizeof(DATA) / sizeof(DATA[0]));
1101
1102 for (int32_t i=0; i<DATA_length; i+=3) {
1103 UnicodeString h = CharsToUnicodeString(DATA[i+1]);
1104 UnicodeString k = CharsToUnicodeString(DATA[i+2]);
1105 switch (*DATA[i]) {
1106 case 0x68: //'h': // Hiragana-Katakana
1107 expect(*hk, h, k);
1108 break;
1109 case 0x6B: //'k': // Katakana-Hiragana
1110 expect(*kh, k, h);
1111 break;
1112 case 0x62: //'b': // both
1113 expect(*hk, h, k);
1114 expect(*kh, k, h);
1115 break;
1116 }
1117 }
1118 delete hk;
1119 delete kh;
1120 }
1121
1122 /**
1123 * Test cloning / copy constructor of RBT.
1124 */
TestCopyJ476(void)1125 void TransliteratorTest::TestCopyJ476(void) {
1126 // The real test here is what happens when the destructors are
1127 // called. So we let one object get destructed, and check to
1128 // see that its copy still works.
1129 Transliterator *t2 = 0;
1130 {
1131 UParseError parseError;
1132 UErrorCode status = U_ZERO_ERROR;
1133 Transliterator *t1 = Transliterator::createFromRules("t1",
1134 "a>A;b>B;'foo'+>'bar'", UTRANS_FORWARD, parseError, status);
1135 if (U_FAILURE(status)) {
1136 errln("FAIL: RBT constructor");
1137 return;
1138 }
1139 t2 = t1->clone(); // Call copy constructor under the covers.
1140 expect(*t1, "abcfoofoo", "ABcbar");
1141 delete t1;
1142 }
1143 expect(*t2, "abcfoofoo", "ABcbar");
1144 delete t2;
1145 }
1146
1147 /**
1148 * Test inter-Indic transliterators. These are composed.
1149 * ICU4C Jitterbug 483.
1150 */
TestInterIndic(void)1151 void TransliteratorTest::TestInterIndic(void) {
1152 UnicodeString ID("Devanagari-Gujarati", "");
1153 UErrorCode status = U_ZERO_ERROR;
1154 UParseError parseError;
1155 Transliterator* dg = Transliterator::createInstance(ID, UTRANS_FORWARD, parseError, status);
1156 if (dg == 0) {
1157 dataerrln("FAIL: createInstance(" + ID + ") returned NULL - " + u_errorName(status));
1158 return;
1159 }
1160 UnicodeString id = dg->getID();
1161 if (id != ID) {
1162 errln("FAIL: createInstance(" + ID + ")->getID() => " + id);
1163 }
1164 UnicodeString dev = CharsToUnicodeString("\\u0901\\u090B\\u0925");
1165 UnicodeString guj = CharsToUnicodeString("\\u0A81\\u0A8B\\u0AA5");
1166 expect(*dg, dev, guj);
1167 delete dg;
1168 }
1169
1170 /**
1171 * Test filter syntax in IDs. (J918)
1172 */
TestFilterIDs(void)1173 void TransliteratorTest::TestFilterIDs(void) {
1174 // Array of 3n strings:
1175 // <id>, <inverse id>, <input>, <expected output>
1176 const char* DATA[] = {
1177 "[aeiou]Any-Hex", // ID
1178 "[aeiou]Hex-Any", // expected inverse ID
1179 "quizzical", // src
1180 "q\\u0075\\u0069zz\\u0069c\\u0061l", // expected ID.translit(src)
1181
1182 "[aeiou]Any-Hex;[^5]Hex-Any",
1183 "[^5]Any-Hex;[aeiou]Hex-Any",
1184 "quizzical",
1185 "q\\u0075izzical",
1186
1187 "[abc]Null",
1188 "[abc]Null",
1189 "xyz",
1190 "xyz",
1191 };
1192 enum { DATA_length = sizeof(DATA) / sizeof(DATA[0]) };
1193
1194 for (int i=0; i<DATA_length; i+=4) {
1195 UnicodeString ID(DATA[i], "");
1196 UnicodeString uID(DATA[i+1], "");
1197 UnicodeString data2(DATA[i+2], "");
1198 UnicodeString data3(DATA[i+3], "");
1199 UParseError parseError;
1200 UErrorCode status = U_ZERO_ERROR;
1201 Transliterator *t = Transliterator::createInstance(ID, UTRANS_FORWARD, parseError, status);
1202 if (t == 0) {
1203 errln("FAIL: createInstance(" + ID + ") returned NULL");
1204 return;
1205 }
1206 expect(*t, data2, data3);
1207
1208 // Check the ID
1209 if (ID != t->getID()) {
1210 errln("FAIL: createInstance(" + ID + ").getID() => " +
1211 t->getID());
1212 }
1213
1214 // Check the inverse
1215 Transliterator *u = t->createInverse(status);
1216 if (u == 0) {
1217 errln("FAIL: " + ID + ".createInverse() returned NULL");
1218 } else if (u->getID() != uID) {
1219 errln("FAIL: " + ID + ".createInverse().getID() => " +
1220 u->getID() + ", expected " + uID);
1221 }
1222
1223 delete t;
1224 delete u;
1225 }
1226 }
1227
1228 /**
1229 * Test the case mapping transliterators.
1230 */
TestCaseMap(void)1231 void TransliteratorTest::TestCaseMap(void) {
1232 UParseError parseError;
1233 UErrorCode status = U_ZERO_ERROR;
1234 Transliterator* toUpper =
1235 Transliterator::createInstance("Any-Upper[^xyzXYZ]", UTRANS_FORWARD, parseError, status);
1236 Transliterator* toLower =
1237 Transliterator::createInstance("Any-Lower[^xyzXYZ]", UTRANS_FORWARD, parseError, status);
1238 Transliterator* toTitle =
1239 Transliterator::createInstance("Any-Title[^xyzXYZ]", UTRANS_FORWARD, parseError, status);
1240 if (toUpper==0 || toLower==0 || toTitle==0) {
1241 errln("FAIL: createInstance returned NULL");
1242 delete toUpper;
1243 delete toLower;
1244 delete toTitle;
1245 return;
1246 }
1247
1248 expect(*toUpper, "The quick brown fox jumped over the lazy dogs.",
1249 "THE QUICK BROWN FOx JUMPED OVER THE LAzy DOGS.");
1250 expect(*toLower, "The quIck brown fOX jUMPED OVER THE LAzY dogs.",
1251 "the quick brown foX jumped over the lazY dogs.");
1252 expect(*toTitle, "the quick brown foX can't jump over the laZy dogs.",
1253 "The Quick Brown FoX Can't Jump Over The LaZy Dogs.");
1254
1255 delete toUpper;
1256 delete toLower;
1257 delete toTitle;
1258 }
1259
1260 /**
1261 * Test the name mapping transliterators.
1262 */
TestNameMap(void)1263 void TransliteratorTest::TestNameMap(void) {
1264 UParseError parseError;
1265 UErrorCode status = U_ZERO_ERROR;
1266 Transliterator* uni2name =
1267 Transliterator::createInstance("Any-Name[^abc]", UTRANS_FORWARD, parseError, status);
1268 Transliterator* name2uni =
1269 Transliterator::createInstance("Name-Any", UTRANS_FORWARD, parseError, status);
1270 if (uni2name==0 || name2uni==0) {
1271 errln("FAIL: createInstance returned NULL");
1272 delete uni2name;
1273 delete name2uni;
1274 return;
1275 }
1276
1277 // Careful: CharsToUS will convert "\\N" => "N"; use "\\\\N" for \N
1278 expect(*uni2name, CharsToUnicodeString("\\u00A0abc\\u4E01\\u00B5\\u0A81\\uFFFD\\u0004\\u0009\\u0081\\uFFFF"),
1279 CharsToUnicodeString("\\\\N{NO-BREAK SPACE}abc\\\\N{CJK UNIFIED IDEOGRAPH-4E01}\\\\N{MICRO SIGN}\\\\N{GUJARATI SIGN CANDRABINDU}\\\\N{REPLACEMENT CHARACTER}\\\\N{<control-0004>}\\\\N{<control-0009>}\\\\N{<control-0081>}\\\\N{<noncharacter-FFFF>}"));
1280 expect(*name2uni, UNICODE_STRING_SIMPLE("{\\N { NO-BREAK SPACE}abc\\N{ CJK UNIFIED IDEOGRAPH-4E01 }\\N{x\\N{MICRO SIGN}\\N{GUJARATI SIGN CANDRABINDU}\\N{REPLACEMENT CHARACTER}\\N{<control-0004>}\\N{<control-0009>}\\N{<control-0081>}\\N{<noncharacter-FFFF>}\\N{<control-0004>}\\N{"),
1281 CharsToUnicodeString("{\\u00A0abc\\u4E01\\\\N{x\\u00B5\\u0A81\\uFFFD\\u0004\\u0009\\u0081\\uFFFF\\u0004\\\\N{"));
1282
1283 delete uni2name;
1284 delete name2uni;
1285
1286 // round trip
1287 Transliterator* t =
1288 Transliterator::createInstance("Any-Name;Name-Any", UTRANS_FORWARD, parseError, status);
1289 if (t==0) {
1290 errln("FAIL: createInstance returned NULL");
1291 delete t;
1292 return;
1293 }
1294
1295 // Careful: CharsToUS will convert "\\N" => "N"; use "\\\\N" for \N
1296 UnicodeString s = CharsToUnicodeString("{\\u00A0abc\\u4E01\\\\N{x\\u00B5\\u0A81\\uFFFD\\u0004\\u0009\\u0081\\uFFFF\\u0004\\\\N{");
1297 expect(*t, s, s);
1298 delete t;
1299 }
1300
1301 /**
1302 * Test liberalized ID syntax. 1006c
1303 */
TestLiberalizedID(void)1304 void TransliteratorTest::TestLiberalizedID(void) {
1305 // Some test cases have an expected getID() value of NULL. This
1306 // means I have disabled the test case for now. This stuff is
1307 // still under development, and I haven't decided whether to make
1308 // getID() return canonical case yet. It will all get rewritten
1309 // with the move to Source-Target/Variant IDs anyway. [aliu]
1310 const char* DATA[] = {
1311 "latin-greek", NULL /*"Latin-Greek"*/, "case insensitivity",
1312 " Null ", "Null", "whitespace",
1313 " Latin[a-z]-Greek ", "[a-z]Latin-Greek", "inline filter",
1314 " null ; latin-greek ", NULL /*"Null;Latin-Greek"*/, "compound whitespace",
1315 };
1316 const int32_t DATA_length = sizeof(DATA)/sizeof(DATA[0]);
1317 UParseError parseError;
1318 UErrorCode status= U_ZERO_ERROR;
1319 for (int32_t i=0; i<DATA_length; i+=3) {
1320 Transliterator *t = Transliterator::createInstance(DATA[i], UTRANS_FORWARD, parseError, status);
1321 if (t == 0) {
1322 dataerrln(UnicodeString("FAIL: ") + DATA[i+2] +
1323 " cannot create ID \"" + DATA[i] + "\" - " + u_errorName(status));
1324 } else {
1325 UnicodeString exp;
1326 if (DATA[i+1]) {
1327 exp = UnicodeString(DATA[i+1], "");
1328 }
1329 // Don't worry about getID() if the expected char*
1330 // is NULL -- see above.
1331 if (exp.length() == 0 || exp == t->getID()) {
1332 logln(UnicodeString("Ok: ") + DATA[i+2] +
1333 " create ID \"" + DATA[i] + "\" => \"" +
1334 exp + "\"");
1335 } else {
1336 errln(UnicodeString("FAIL: ") + DATA[i+2] +
1337 " create ID \"" + DATA[i] + "\" => \"" +
1338 t->getID() + "\", exp \"" + exp + "\"");
1339 }
1340 delete t;
1341 }
1342 }
1343 }
1344
1345 /* test for Jitterbug 912 */
TestCreateInstance()1346 void TransliteratorTest::TestCreateInstance(){
1347 const char* FORWARD = "F";
1348 const char* REVERSE = "R";
1349 const char* DATA[] = {
1350 // Column 1: id
1351 // Column 2: direction
1352 // Column 3: expected ID, or "" if expect failure
1353 "Latin-Hangul", REVERSE, "Hangul-Latin", // JB#912
1354
1355 // JB#2689: bad compound causes crash
1356 "InvalidSource-InvalidTarget", FORWARD, "",
1357 "InvalidSource-InvalidTarget", REVERSE, "",
1358 "Hex-Any;InvalidSource-InvalidTarget", FORWARD, "",
1359 "Hex-Any;InvalidSource-InvalidTarget", REVERSE, "",
1360 "InvalidSource-InvalidTarget;Hex-Any", FORWARD, "",
1361 "InvalidSource-InvalidTarget;Hex-Any", REVERSE, "",
1362
1363 NULL
1364 };
1365
1366 for (int32_t i=0; DATA[i]; i+=3) {
1367 UParseError err;
1368 UErrorCode ec = U_ZERO_ERROR;
1369 UnicodeString id(DATA[i]);
1370 UTransDirection dir = (DATA[i+1]==FORWARD)?
1371 UTRANS_FORWARD:UTRANS_REVERSE;
1372 UnicodeString expID(DATA[i+2]);
1373 Transliterator* t =
1374 Transliterator::createInstance(id,dir,err,ec);
1375 UnicodeString newID;
1376 if (t) {
1377 newID = t->getID();
1378 }
1379 UBool ok = (newID == expID);
1380 if (!t) {
1381 newID = u_errorName(ec);
1382 }
1383 if (ok) {
1384 logln((UnicodeString)"Ok: createInstance(" +
1385 id + "," + DATA[i+1] + ") => " + newID);
1386 } else {
1387 dataerrln((UnicodeString)"FAIL: createInstance(" +
1388 id + "," + DATA[i+1] + ") => " + newID +
1389 ", expected " + expID);
1390 }
1391 delete t;
1392 }
1393 }
1394
1395 /**
1396 * Test the normalization transliterator.
1397 */
TestNormalizationTransliterator()1398 void TransliteratorTest::TestNormalizationTransliterator() {
1399 // THE FOLLOWING TWO TABLES ARE COPIED FROM com.ibm.test.normalizer.BasicTest
1400 // PLEASE KEEP THEM IN SYNC WITH BasicTest.
1401 const char* CANON[] = {
1402 // Input Decomposed Composed
1403 "cat", "cat", "cat" ,
1404 "\\u00e0ardvark", "a\\u0300ardvark", "\\u00e0ardvark" ,
1405
1406 "\\u1e0a", "D\\u0307", "\\u1e0a" , // D-dot_above
1407 "D\\u0307", "D\\u0307", "\\u1e0a" , // D dot_above
1408
1409 "\\u1e0c\\u0307", "D\\u0323\\u0307", "\\u1e0c\\u0307" , // D-dot_below dot_above
1410 "\\u1e0a\\u0323", "D\\u0323\\u0307", "\\u1e0c\\u0307" , // D-dot_above dot_below
1411 "D\\u0307\\u0323", "D\\u0323\\u0307", "\\u1e0c\\u0307" , // D dot_below dot_above
1412
1413 "\\u1e10\\u0307\\u0323", "D\\u0327\\u0323\\u0307","\\u1e10\\u0323\\u0307", // D dot_below cedilla dot_above
1414 "D\\u0307\\u0328\\u0323","D\\u0328\\u0323\\u0307","\\u1e0c\\u0328\\u0307", // D dot_above ogonek dot_below
1415
1416 "\\u1E14", "E\\u0304\\u0300", "\\u1E14" , // E-macron-grave
1417 "\\u0112\\u0300", "E\\u0304\\u0300", "\\u1E14" , // E-macron + grave
1418 "\\u00c8\\u0304", "E\\u0300\\u0304", "\\u00c8\\u0304" , // E-grave + macron
1419
1420 "\\u212b", "A\\u030a", "\\u00c5" , // angstrom_sign
1421 "\\u00c5", "A\\u030a", "\\u00c5" , // A-ring
1422
1423 "\\u00fdffin", "y\\u0301ffin", "\\u00fdffin" , //updated with 3.0
1424 "\\u00fd\\uFB03n", "y\\u0301\\uFB03n", "\\u00fd\\uFB03n" , //updated with 3.0
1425
1426 "Henry IV", "Henry IV", "Henry IV" ,
1427 "Henry \\u2163", "Henry \\u2163", "Henry \\u2163" ,
1428
1429 "\\u30AC", "\\u30AB\\u3099", "\\u30AC" , // ga (Katakana)
1430 "\\u30AB\\u3099", "\\u30AB\\u3099", "\\u30AC" , // ka + ten
1431 "\\uFF76\\uFF9E", "\\uFF76\\uFF9E", "\\uFF76\\uFF9E" , // hw_ka + hw_ten
1432 "\\u30AB\\uFF9E", "\\u30AB\\uFF9E", "\\u30AB\\uFF9E" , // ka + hw_ten
1433 "\\uFF76\\u3099", "\\uFF76\\u3099", "\\uFF76\\u3099" , // hw_ka + ten
1434
1435 "A\\u0300\\u0316", "A\\u0316\\u0300", "\\u00C0\\u0316" ,
1436 0 // end
1437 };
1438
1439 const char* COMPAT[] = {
1440 // Input Decomposed Composed
1441 "\\uFB4f", "\\u05D0\\u05DC", "\\u05D0\\u05DC" , // Alef-Lamed vs. Alef, Lamed
1442
1443 "\\u00fdffin", "y\\u0301ffin", "\\u00fdffin" , //updated for 3.0
1444 "\\u00fd\\uFB03n", "y\\u0301ffin", "\\u00fdffin" , // ffi ligature -> f + f + i
1445
1446 "Henry IV", "Henry IV", "Henry IV" ,
1447 "Henry \\u2163", "Henry IV", "Henry IV" ,
1448
1449 "\\u30AC", "\\u30AB\\u3099", "\\u30AC" , // ga (Katakana)
1450 "\\u30AB\\u3099", "\\u30AB\\u3099", "\\u30AC" , // ka + ten
1451
1452 "\\uFF76\\u3099", "\\u30AB\\u3099", "\\u30AC" , // hw_ka + ten
1453 0 // end
1454 };
1455
1456 int32_t i;
1457 UParseError parseError;
1458 UErrorCode status = U_ZERO_ERROR;
1459 Transliterator* NFD = Transliterator::createInstance("NFD", UTRANS_FORWARD, parseError, status);
1460 Transliterator* NFC = Transliterator::createInstance("NFC", UTRANS_FORWARD, parseError, status);
1461 if (!NFD || !NFC) {
1462 dataerrln("FAIL: createInstance failed: %s", u_errorName(status));
1463 delete NFD;
1464 delete NFC;
1465 return;
1466 }
1467 for (i=0; CANON[i]; i+=3) {
1468 UnicodeString in = CharsToUnicodeString(CANON[i]);
1469 UnicodeString expd = CharsToUnicodeString(CANON[i+1]);
1470 UnicodeString expc = CharsToUnicodeString(CANON[i+2]);
1471 expect(*NFD, in, expd);
1472 expect(*NFC, in, expc);
1473 }
1474 delete NFD;
1475 delete NFC;
1476
1477 Transliterator* NFKD = Transliterator::createInstance("NFKD", UTRANS_FORWARD, parseError, status);
1478 Transliterator* NFKC = Transliterator::createInstance("NFKC", UTRANS_FORWARD, parseError, status);
1479 if (!NFKD || !NFKC) {
1480 dataerrln("FAIL: createInstance failed");
1481 delete NFKD;
1482 delete NFKC;
1483 return;
1484 }
1485 for (i=0; COMPAT[i]; i+=3) {
1486 UnicodeString in = CharsToUnicodeString(COMPAT[i]);
1487 UnicodeString expkd = CharsToUnicodeString(COMPAT[i+1]);
1488 UnicodeString expkc = CharsToUnicodeString(COMPAT[i+2]);
1489 expect(*NFKD, in, expkd);
1490 expect(*NFKC, in, expkc);
1491 }
1492 delete NFKD;
1493 delete NFKC;
1494
1495 UParseError pe;
1496 status = U_ZERO_ERROR;
1497 Transliterator *t = Transliterator::createInstance("NFD; [x]Remove",
1498 UTRANS_FORWARD,
1499 pe, status);
1500 if (t == 0) {
1501 errln("FAIL: createInstance failed");
1502 }
1503 expect(*t, CharsToUnicodeString("\\u010dx"),
1504 CharsToUnicodeString("c\\u030C"));
1505 delete t;
1506 }
1507
1508 /**
1509 * Test compound RBT rules.
1510 */
TestCompoundRBT(void)1511 void TransliteratorTest::TestCompoundRBT(void) {
1512 // Careful with spacing and ';' here: Phrase this exactly
1513 // as toRules() is going to return it. If toRules() changes
1514 // with regard to spacing or ';', then adjust this string.
1515 UnicodeString rule("::Hex-Any;\n"
1516 "::Any-Lower;\n"
1517 "a > '.A.';\n"
1518 "b > '.B.';\n"
1519 "::[^t]Any-Upper;", "");
1520 UParseError parseError;
1521 UErrorCode status = U_ZERO_ERROR;
1522 Transliterator *t = Transliterator::createFromRules("Test", rule, UTRANS_FORWARD, parseError, status);
1523 if (t == 0) {
1524 errln("FAIL: createFromRules failed");
1525 return;
1526 }
1527 expect(*t, UNICODE_STRING_SIMPLE("\\u0043at in the hat, bat on the mat"),
1528 "C.A.t IN tHE H.A.t, .B..A.t ON tHE M.A.t");
1529 UnicodeString r;
1530 t->toRules(r, TRUE);
1531 if (r == rule) {
1532 logln((UnicodeString)"OK: toRules() => " + r);
1533 } else {
1534 errln((UnicodeString)"FAIL: toRules() => " + r +
1535 ", expected " + rule);
1536 }
1537 delete t;
1538
1539 // Now test toRules
1540 t = Transliterator::createInstance("Greek-Latin; Latin-Cyrillic", UTRANS_FORWARD, parseError, status);
1541 if (t == 0) {
1542 dataerrln("FAIL: createInstance failed - %s", u_errorName(status));
1543 return;
1544 }
1545 UnicodeString exp("::Greek-Latin;\n::Latin-Cyrillic;");
1546 t->toRules(r, TRUE);
1547 if (r != exp) {
1548 errln((UnicodeString)"FAIL: toRules() => " + r +
1549 ", expected " + exp);
1550 } else {
1551 logln((UnicodeString)"OK: toRules() => " + r);
1552 }
1553 delete t;
1554
1555 // Round trip the result of toRules
1556 t = Transliterator::createFromRules("Test", r, UTRANS_FORWARD, parseError, status);
1557 if (t == 0) {
1558 errln("FAIL: createFromRules #2 failed");
1559 return;
1560 } else {
1561 logln((UnicodeString)"OK: createFromRules(" + r + ") succeeded");
1562 }
1563
1564 // Test toRules again
1565 t->toRules(r, TRUE);
1566 if (r != exp) {
1567 errln((UnicodeString)"FAIL: toRules() => " + r +
1568 ", expected " + exp);
1569 } else {
1570 logln((UnicodeString)"OK: toRules() => " + r);
1571 }
1572
1573 delete t;
1574
1575 // Test Foo(Bar) IDs. Careful with spacing in id; make it conform
1576 // to what the regenerated ID will look like.
1577 UnicodeString id("Upper(Lower);(NFKC)", "");
1578 t = Transliterator::createInstance(id, UTRANS_FORWARD, parseError, status);
1579 if (t == 0) {
1580 errln("FAIL: createInstance #2 failed");
1581 return;
1582 }
1583 if (t->getID() == id) {
1584 logln((UnicodeString)"OK: created " + id);
1585 } else {
1586 errln((UnicodeString)"FAIL: createInstance(" + id +
1587 ").getID() => " + t->getID());
1588 }
1589
1590 Transliterator *u = t->createInverse(status);
1591 if (u == 0) {
1592 errln("FAIL: createInverse failed");
1593 delete t;
1594 return;
1595 }
1596 exp = "NFKC();Lower(Upper)";
1597 if (u->getID() == exp) {
1598 logln((UnicodeString)"OK: createInverse(" + id + ") => " +
1599 u->getID());
1600 } else {
1601 errln((UnicodeString)"FAIL: createInverse(" + id + ") => " +
1602 u->getID());
1603 }
1604 delete t;
1605 delete u;
1606 }
1607
1608 /**
1609 * Compound filter semantics were orginially not implemented
1610 * correctly. Originally, each component filter f(i) is replaced by
1611 * f'(i) = f(i) && g, where g is the filter for the compound
1612 * transliterator.
1613 *
1614 * From Mark:
1615 *
1616 * Suppose and I have a transliterator X. Internally X is
1617 * "Greek-Latin; Latin-Cyrillic; Any-Lower". I use a filter [^A].
1618 *
1619 * The compound should convert all greek characters (through latin) to
1620 * cyrillic, then lowercase the result. The filter should say "don't
1621 * touch 'A' in the original". But because an intermediate result
1622 * happens to go through "A", the Greek Alpha gets hung up.
1623 */
TestCompoundFilter(void)1624 void TransliteratorTest::TestCompoundFilter(void) {
1625 UParseError parseError;
1626 UErrorCode status = U_ZERO_ERROR;
1627 Transliterator *t = Transliterator::createInstance
1628 ("Greek-Latin; Latin-Greek; Lower", UTRANS_FORWARD, parseError, status);
1629 if (t == 0) {
1630 dataerrln("FAIL: createInstance failed - %s", u_errorName(status));
1631 return;
1632 }
1633 t->adoptFilter(new UnicodeSet("[^A]", status));
1634 if (U_FAILURE(status)) {
1635 errln("FAIL: UnicodeSet ct failed");
1636 delete t;
1637 return;
1638 }
1639
1640 // Only the 'A' at index 1 should remain unchanged
1641 expect(*t,
1642 CharsToUnicodeString("BA\\u039A\\u0391"),
1643 CharsToUnicodeString("\\u03b2A\\u03ba\\u03b1"));
1644 delete t;
1645 }
1646
TestRemove(void)1647 void TransliteratorTest::TestRemove(void) {
1648 UParseError parseError;
1649 UErrorCode status = U_ZERO_ERROR;
1650 Transliterator *t = Transliterator::createInstance("Remove[abc]", UTRANS_FORWARD, parseError, status);
1651 if (t == 0) {
1652 errln("FAIL: createInstance failed");
1653 return;
1654 }
1655
1656 expect(*t, "Able bodied baker's cats", "Ale odied ker's ts");
1657
1658 // extra test for RemoveTransliterator::clone(), which at one point wasn't
1659 // duplicating the filter
1660 Transliterator* t2 = t->clone();
1661 expect(*t2, "Able bodied baker's cats", "Ale odied ker's ts");
1662
1663 delete t;
1664 delete t2;
1665 }
1666
TestToRules(void)1667 void TransliteratorTest::TestToRules(void) {
1668 const char* RBT = "rbt";
1669 const char* SET = "set";
1670 static const char* DATA[] = {
1671 RBT,
1672 "$a=\\u4E61; [$a] > A;",
1673 "[\\u4E61] > A;",
1674
1675 RBT,
1676 "$white=[[:Zs:][:Zl:]]; $white{a} > A;",
1677 "[[:Zs:][:Zl:]]{a} > A;",
1678
1679 SET,
1680 "[[:Zs:][:Zl:]]",
1681 "[[:Zs:][:Zl:]]",
1682
1683 SET,
1684 "[:Ps:]",
1685 "[:Ps:]",
1686
1687 SET,
1688 "[:L:]",
1689 "[:L:]",
1690
1691 SET,
1692 "[[:L:]-[A]]",
1693 "[[:L:]-[A]]",
1694
1695 SET,
1696 "[~[:Lu:][:Ll:]]",
1697 "[~[:Lu:][:Ll:]]",
1698
1699 SET,
1700 "[~[a-z]]",
1701 "[~[a-z]]",
1702
1703 RBT,
1704 "$white=[:Zs:]; $black=[^$white]; $black{a} > A;",
1705 "[^[:Zs:]]{a} > A;",
1706
1707 RBT,
1708 "$a=[:Zs:]; $b=[[a-z]-$a]; $b{a} > A;",
1709 "[[a-z]-[:Zs:]]{a} > A;",
1710
1711 RBT,
1712 "$a=[:Zs:]; $b=[$a&[a-z]]; $b{a} > A;",
1713 "[[:Zs:]&[a-z]]{a} > A;",
1714
1715 RBT,
1716 "$a=[:Zs:]; $b=[x$a]; $b{a} > A;",
1717 "[x[:Zs:]]{a} > A;",
1718
1719 RBT,
1720 "$accentMinus = [ [\\u0300-\\u0345] & [:M:] - [\\u0338]] ;"
1721 "$macron = \\u0304 ;"
1722 "$evowel = [aeiouyAEIOUY] ;"
1723 "$iotasub = \\u0345 ;"
1724 "($evowel $macron $accentMinus *) i > | $1 $iotasub ;",
1725 "([AEIOUYaeiouy]\\u0304[[\\u0300-\\u0345]&[:M:]-[\\u0338]]*)i > | $1 \\u0345;",
1726
1727 RBT,
1728 "([AEIOUYaeiouy]\\u0304[[:M:]-[\\u0304\\u0345]]*)i > | $1 \\u0345;",
1729 "([AEIOUYaeiouy]\\u0304[[:M:]-[\\u0304\\u0345]]*)i > | $1 \\u0345;",
1730 };
1731 static const int32_t DATA_length = (int32_t)(sizeof(DATA) / sizeof(DATA[0]));
1732
1733 for (int32_t d=0; d < DATA_length; d+=3) {
1734 if (DATA[d] == RBT) {
1735 // Transliterator test
1736 UParseError parseError;
1737 UErrorCode status = U_ZERO_ERROR;
1738 Transliterator *t = Transliterator::createFromRules("ID",
1739 UnicodeString(DATA[d+1], -1, US_INV), UTRANS_FORWARD, parseError, status);
1740 if (t == 0) {
1741 dataerrln("FAIL: createFromRules failed - %s", u_errorName(status));
1742 return;
1743 }
1744 UnicodeString rules, escapedRules;
1745 t->toRules(rules, FALSE);
1746 t->toRules(escapedRules, TRUE);
1747 UnicodeString expRules = CharsToUnicodeString(DATA[d+2]);
1748 UnicodeString expEscapedRules(DATA[d+2], -1, US_INV);
1749 if (rules == expRules) {
1750 logln((UnicodeString)"Ok: " + UnicodeString(DATA[d+1], -1, US_INV) +
1751 " => " + rules);
1752 } else {
1753 errln((UnicodeString)"FAIL: " + UnicodeString(DATA[d+1], -1, US_INV) +
1754 " => " + rules + ", exp " + expRules);
1755 }
1756 if (escapedRules == expEscapedRules) {
1757 logln((UnicodeString)"Ok: " + UnicodeString(DATA[d+1], -1, US_INV) +
1758 " => " + escapedRules);
1759 } else {
1760 errln((UnicodeString)"FAIL: " + UnicodeString(DATA[d+1], -1, US_INV) +
1761 " => " + escapedRules + ", exp " + expEscapedRules);
1762 }
1763 delete t;
1764
1765 } else {
1766 // UnicodeSet test
1767 UErrorCode status = U_ZERO_ERROR;
1768 UnicodeString pat(DATA[d+1], -1, US_INV);
1769 UnicodeString expToPat(DATA[d+2], -1, US_INV);
1770 UnicodeSet set(pat, status);
1771 if (U_FAILURE(status)) {
1772 errln("FAIL: UnicodeSet ct failed");
1773 return;
1774 }
1775 // Adjust spacing etc. as necessary.
1776 UnicodeString toPat;
1777 set.toPattern(toPat);
1778 if (expToPat == toPat) {
1779 logln((UnicodeString)"Ok: " + pat +
1780 " => " + toPat);
1781 } else {
1782 errln((UnicodeString)"FAIL: " + pat +
1783 " => " + prettify(toPat, TRUE) +
1784 ", exp " + prettify(pat, TRUE));
1785 }
1786 }
1787 }
1788 }
1789
TestContext()1790 void TransliteratorTest::TestContext() {
1791 UTransPosition pos = {0, 2, 0, 1}; // cs cl s l
1792 expect("de > x; {d}e > y;",
1793 "de",
1794 "ye",
1795 &pos);
1796
1797 expect("ab{c} > z;",
1798 "xadabdabcy",
1799 "xadabdabzy");
1800 }
1801
TestSupplemental()1802 void TransliteratorTest::TestSupplemental() {
1803
1804 expect(CharsToUnicodeString("$a=\\U00010300; $s=[\\U00010300-\\U00010323];"
1805 "a > $a; $s > i;"),
1806 CharsToUnicodeString("ab\\U0001030Fx"),
1807 CharsToUnicodeString("\\U00010300bix"));
1808
1809 expect(CharsToUnicodeString("$a=[a-z\\U00010300-\\U00010323];"
1810 "$b=[A-Z\\U00010400-\\U0001044D];"
1811 "($a)($b) > $2 $1;"),
1812 CharsToUnicodeString("aB\\U00010300\\U00010400c\\U00010401\\U00010301D"),
1813 CharsToUnicodeString("Ba\\U00010400\\U00010300\\U00010401cD\\U00010301"));
1814
1815 // k|ax\\U00010300xm
1816
1817 // k|a\\U00010400\\U00010300xm
1818 // ky|\\U00010400\\U00010300xm
1819 // ky\\U00010400|\\U00010300xm
1820
1821 // ky\\U00010400|\\U00010300\\U00010400m
1822 // ky\\U00010400y|\\U00010400m
1823 expect(CharsToUnicodeString("$a=[a\\U00010300-\\U00010323];"
1824 "$a {x} > | @ \\U00010400;"
1825 "{$a} [^\\u0000-\\uFFFF] > y;"),
1826 CharsToUnicodeString("kax\\U00010300xm"),
1827 CharsToUnicodeString("ky\\U00010400y\\U00010400m"));
1828
1829 expectT("Any-Name",
1830 CharsToUnicodeString("\\U00010330\\U000E0061\\u00A0"),
1831 UNICODE_STRING_SIMPLE("\\N{GOTHIC LETTER AHSA}\\N{TAG LATIN SMALL LETTER A}\\N{NO-BREAK SPACE}"));
1832
1833 expectT("Any-Hex/Unicode",
1834 CharsToUnicodeString("\\U00010330\\U0010FF00\\U000E0061\\u00A0"),
1835 UNICODE_STRING_SIMPLE("U+10330U+10FF00U+E0061U+00A0"));
1836
1837 expectT("Any-Hex/C",
1838 CharsToUnicodeString("\\U00010330\\U0010FF00\\U000E0061\\u00A0"),
1839 UNICODE_STRING_SIMPLE("\\U00010330\\U0010FF00\\U000E0061\\u00A0"));
1840
1841 expectT("Any-Hex/Perl",
1842 CharsToUnicodeString("\\U00010330\\U0010FF00\\U000E0061\\u00A0"),
1843 UNICODE_STRING_SIMPLE("\\x{10330}\\x{10FF00}\\x{E0061}\\x{A0}"));
1844
1845 expectT("Any-Hex/Java",
1846 CharsToUnicodeString("\\U00010330\\U0010FF00\\U000E0061\\u00A0"),
1847 UNICODE_STRING_SIMPLE("\\uD800\\uDF30\\uDBFF\\uDF00\\uDB40\\uDC61\\u00A0"));
1848
1849 expectT("Any-Hex/XML",
1850 CharsToUnicodeString("\\U00010330\\U0010FF00\\U000E0061\\u00A0"),
1851 "𐌰􏼀󠁡 ");
1852
1853 expectT("Any-Hex/XML10",
1854 CharsToUnicodeString("\\U00010330\\U0010FF00\\U000E0061\\u00A0"),
1855 "𐌰􏼀󠁡 ");
1856
1857 expectT(UNICODE_STRING_SIMPLE("[\\U000E0000-\\U000E0FFF] Remove"),
1858 CharsToUnicodeString("\\U00010330\\U0010FF00\\U000E0061\\u00A0"),
1859 CharsToUnicodeString("\\U00010330\\U0010FF00\\u00A0"));
1860 }
1861
TestQuantifier()1862 void TransliteratorTest::TestQuantifier() {
1863
1864 // Make sure @ in a quantified anteContext works
1865 expect("a+ {b} > | @@ c; A > a; (a+ c) > '(' $1 ')';",
1866 "AAAAAb",
1867 "aaa(aac)");
1868
1869 // Make sure @ in a quantified postContext works
1870 expect("{b} a+ > c @@ |; (a+) > '(' $1 ')';",
1871 "baaaaa",
1872 "caa(aaa)");
1873
1874 // Make sure @ in a quantified postContext with seg ref works
1875 expect("{(b)} a+ > $1 @@ |; (a+) > '(' $1 ')';",
1876 "baaaaa",
1877 "baa(aaa)");
1878
1879 // Make sure @ past ante context doesn't enter ante context
1880 UTransPosition pos = {0, 5, 3, 5};
1881 expect("a+ {b} > | @@ c; x > y; (a+ c) > '(' $1 ')';",
1882 "xxxab",
1883 "xxx(ac)",
1884 &pos);
1885
1886 // Make sure @ past post context doesn't pass limit
1887 UTransPosition pos2 = {0, 4, 0, 2};
1888 expect("{b} a+ > c @@ |; x > y; a > A;",
1889 "baxx",
1890 "caxx",
1891 &pos2);
1892
1893 // Make sure @ past post context doesn't enter post context
1894 expect("{b} a+ > c @@ |; x > y; a > A;",
1895 "baxx",
1896 "cayy");
1897
1898 expect("(ab)? c > d;",
1899 "c abc ababc",
1900 "d d abd");
1901
1902 // NOTE: The (ab)+ when referenced just yields a single "ab",
1903 // not the full sequence of them. This accords with perl behavior.
1904 expect("(ab)+ {x} > '(' $1 ')';",
1905 "x abx ababxy",
1906 "x ab(ab) abab(ab)y");
1907
1908 expect("b+ > x;",
1909 "ac abc abbc abbbc",
1910 "ac axc axc axc");
1911
1912 expect("[abc]+ > x;",
1913 "qac abrc abbcs abtbbc",
1914 "qx xrx xs xtx");
1915
1916 expect("q{(ab)+} > x;",
1917 "qa qab qaba qababc qaba",
1918 "qa qx qxa qxc qxa");
1919
1920 expect("q(ab)* > x;",
1921 "qa qab qaba qababc",
1922 "xa x xa xc");
1923
1924 // NOTE: The (ab)+ when referenced just yields a single "ab",
1925 // not the full sequence of them. This accords with perl behavior.
1926 expect("q(ab)* > '(' $1 ')';",
1927 "qa qab qaba qababc",
1928 "()a (ab) (ab)a (ab)c");
1929
1930 // 'foo'+ and 'foo'* -- the quantifier should apply to the entire
1931 // quoted string
1932 expect("'ab'+ > x;",
1933 "bb ab ababb",
1934 "bb x xb");
1935
1936 // $foo+ and $foo* -- the quantifier should apply to the entire
1937 // variable reference
1938 expect("$var = ab; $var+ > x;",
1939 "bb ab ababb",
1940 "bb x xb");
1941 }
1942
1943 class TestTrans : public Transliterator {
1944 public:
TestTrans(const UnicodeString & id)1945 TestTrans(const UnicodeString& id) : Transliterator(id, 0) {
1946 }
clone(void) const1947 virtual Transliterator* clone(void) const {
1948 return new TestTrans(getID());
1949 }
handleTransliterate(Replaceable &,UTransPosition & offsets,UBool) const1950 virtual void handleTransliterate(Replaceable& /*text*/, UTransPosition& offsets,
1951 UBool /*isIncremental*/) const
1952 {
1953 offsets.start = offsets.limit;
1954 }
1955 virtual UClassID getDynamicClassID() const;
1956 static UClassID U_EXPORT2 getStaticClassID();
1957 };
UOBJECT_DEFINE_RTTI_IMPLEMENTATION(TestTrans)1958 UOBJECT_DEFINE_RTTI_IMPLEMENTATION(TestTrans)
1959
1960 /**
1961 * Test Source-Target/Variant.
1962 */
1963 void TransliteratorTest::TestSTV(void) {
1964 int32_t ns = Transliterator::countAvailableSources();
1965 if (ns < 0 || ns > 255) {
1966 errln((UnicodeString)"FAIL: Bad source count: " + ns);
1967 return;
1968 }
1969 int32_t i, j;
1970 for (i=0; i<ns; ++i) {
1971 UnicodeString source;
1972 Transliterator::getAvailableSource(i, source);
1973 logln((UnicodeString)"" + i + ": " + source);
1974 if (source.length() == 0) {
1975 errln("FAIL: empty source");
1976 continue;
1977 }
1978 int32_t nt = Transliterator::countAvailableTargets(source);
1979 if (nt < 0 || nt > 255) {
1980 errln((UnicodeString)"FAIL: Bad target count: " + nt);
1981 continue;
1982 }
1983 for (int32_t j=0; j<nt; ++j) {
1984 UnicodeString target;
1985 Transliterator::getAvailableTarget(j, source, target);
1986 logln((UnicodeString)" " + j + ": " + target);
1987 if (target.length() == 0) {
1988 errln("FAIL: empty target");
1989 continue;
1990 }
1991 int32_t nv = Transliterator::countAvailableVariants(source, target);
1992 if (nv < 0 || nv > 255) {
1993 errln((UnicodeString)"FAIL: Bad variant count: " + nv);
1994 continue;
1995 }
1996 for (int32_t k=0; k<nv; ++k) {
1997 UnicodeString variant;
1998 Transliterator::getAvailableVariant(k, source, target, variant);
1999 if (variant.length() == 0) {
2000 logln((UnicodeString)" " + k + ": <empty>");
2001 } else {
2002 logln((UnicodeString)" " + k + ": " + variant);
2003 }
2004 }
2005 }
2006 }
2007
2008 // Test registration
2009 const char* IDS[] = { "Fieruwer", "Seoridf-Sweorie", "Oewoir-Oweri/Vsie" };
2010 const char* FULL_IDS[] = { "Any-Fieruwer", "Seoridf-Sweorie", "Oewoir-Oweri/Vsie" };
2011 const char* SOURCES[] = { NULL, "Seoridf", "Oewoir" };
2012 for (i=0; i<3; ++i) {
2013 Transliterator *t = new TestTrans(IDS[i]);
2014 if (t == 0) {
2015 errln("FAIL: out of memory");
2016 return;
2017 }
2018 if (t->getID() != IDS[i]) {
2019 errln((UnicodeString)"FAIL: ID mismatch for " + IDS[i]);
2020 delete t;
2021 return;
2022 }
2023 Transliterator::registerInstance(t);
2024 UErrorCode status = U_ZERO_ERROR;
2025 t = Transliterator::createInstance(IDS[i], UTRANS_FORWARD, status);
2026 if (t == NULL) {
2027 errln((UnicodeString)"FAIL: Registration/creation failed for ID " +
2028 IDS[i]);
2029 } else {
2030 logln((UnicodeString)"Ok: Registration/creation succeeded for ID " +
2031 IDS[i]);
2032 delete t;
2033 }
2034 Transliterator::unregister(IDS[i]);
2035 t = Transliterator::createInstance(IDS[i], UTRANS_FORWARD, status);
2036 if (t != NULL) {
2037 errln((UnicodeString)"FAIL: Unregistration failed for ID " +
2038 IDS[i]);
2039 delete t;
2040 }
2041 }
2042
2043 // Make sure getAvailable API reflects removal
2044 int32_t n = Transliterator::countAvailableIDs();
2045 for (i=0; i<n; ++i) {
2046 UnicodeString id = Transliterator::getAvailableID(i);
2047 for (j=0; j<3; ++j) {
2048 if (id.caseCompare(FULL_IDS[j],0)==0) {
2049 errln((UnicodeString)"FAIL: unregister(" + id + ") failed");
2050 }
2051 }
2052 }
2053 n = Transliterator::countAvailableTargets("Any");
2054 for (i=0; i<n; ++i) {
2055 UnicodeString t;
2056 Transliterator::getAvailableTarget(i, "Any", t);
2057 if (t.caseCompare(IDS[0],0)==0) {
2058 errln((UnicodeString)"FAIL: unregister(Any-" + t + ") failed");
2059 }
2060 }
2061 n = Transliterator::countAvailableSources();
2062 for (i=0; i<n; ++i) {
2063 UnicodeString s;
2064 Transliterator::getAvailableSource(i, s);
2065 for (j=0; j<3; ++j) {
2066 if (SOURCES[j] == NULL) continue;
2067 if (s.caseCompare(SOURCES[j],0)==0) {
2068 errln((UnicodeString)"FAIL: unregister(" + s + "-*) failed");
2069 }
2070 }
2071 }
2072 }
2073
2074 /**
2075 * Test inverse of Greek-Latin; Title()
2076 */
TestCompoundInverse(void)2077 void TransliteratorTest::TestCompoundInverse(void) {
2078 UParseError parseError;
2079 UErrorCode status = U_ZERO_ERROR;
2080 Transliterator *t = Transliterator::createInstance
2081 ("Greek-Latin; Title()", UTRANS_REVERSE,parseError, status);
2082 if (t == 0) {
2083 dataerrln("FAIL: createInstance - %s", u_errorName(status));
2084 return;
2085 }
2086 UnicodeString exp("(Title);Latin-Greek");
2087 if (t->getID() == exp) {
2088 logln("Ok: inverse of \"Greek-Latin; Title()\" is \"" +
2089 t->getID());
2090 } else {
2091 errln("FAIL: inverse of \"Greek-Latin; Title()\" is \"" +
2092 t->getID() + "\", expected \"" + exp + "\"");
2093 }
2094 delete t;
2095 }
2096
2097 /**
2098 * Test NFD chaining with RBT
2099 */
TestNFDChainRBT()2100 void TransliteratorTest::TestNFDChainRBT() {
2101 UParseError pe;
2102 UErrorCode ec = U_ZERO_ERROR;
2103 Transliterator* t = Transliterator::createFromRules(
2104 "TEST", "::NFD; aa > Q; a > q;",
2105 UTRANS_FORWARD, pe, ec);
2106 if (t == NULL || U_FAILURE(ec)) {
2107 dataerrln("FAIL: Transliterator::createFromRules failed with %s", u_errorName(ec));
2108 return;
2109 }
2110 expect(*t, "aa", "Q");
2111 delete t;
2112
2113 // TEMPORARY TESTS -- BEING DEBUGGED
2114 //=- UnicodeString s, s2;
2115 //=- t = Transliterator::createInstance("Latin-Devanagari", UTRANS_FORWARD, pe, ec);
2116 //=- s = CharsToUnicodeString("rmk\\u1E63\\u0113t");
2117 //=- s2 = CharsToUnicodeString("\\u0930\\u094D\\u092E\\u094D\\u0915\\u094D\\u0937\\u0947\\u0924\\u094D");
2118 //=- expect(*t, s, s2);
2119 //=- delete t;
2120 //=-
2121 //=- t = Transliterator::createInstance("Devanagari-Latin", UTRANS_FORWARD, pe, ec);
2122 //=- expect(*t, s2, s);
2123 //=- delete t;
2124 //=-
2125 //=- t = Transliterator::createInstance("Latin-Devanagari;Devanagari-Latin", UTRANS_FORWARD, pe, ec);
2126 //=- s = CharsToUnicodeString("rmk\\u1E63\\u0113t");
2127 //=- expect(*t, s, s);
2128 //=- delete t;
2129
2130 // const char* source[] = {
2131 // /*
2132 // "\\u015Br\\u012Bmad",
2133 // "bhagavadg\\u012Bt\\u0101",
2134 // "adhy\\u0101ya",
2135 // "arjuna",
2136 // "vi\\u1E63\\u0101da",
2137 // "y\\u014Dga",
2138 // "dhr\\u0325tar\\u0101\\u1E63\\u1E6Dra",
2139 // "uv\\u0101cr\\u0325",
2140 // */
2141 // "rmk\\u1E63\\u0113t",
2142 // //"dharmak\\u1E63\\u0113tr\\u0113",
2143 // /*
2144 // "kuruk\\u1E63\\u0113tr\\u0113",
2145 // "samav\\u0113t\\u0101",
2146 // "yuyutsava-\\u1E25",
2147 // "m\\u0101mak\\u0101-\\u1E25",
2148 // // "p\\u0101\\u1E47\\u1E0Dav\\u0101\\u015Bcaiva",
2149 // "kimakurvata",
2150 // "san\\u0304java",
2151 // */
2152 //
2153 // 0
2154 // };
2155 // const char* expected[] = {
2156 // /*
2157 // "\\u0936\\u094d\\u0930\\u0940\\u092e\\u0926\\u094d",
2158 // "\\u092d\\u0917\\u0935\\u0926\\u094d\\u0917\\u0940\\u0924\\u093e",
2159 // "\\u0905\\u0927\\u094d\\u092f\\u093e\\u092f",
2160 // "\\u0905\\u0930\\u094d\\u091c\\u0941\\u0928",
2161 // "\\u0935\\u093f\\u0937\\u093e\\u0926",
2162 // "\\u092f\\u094b\\u0917",
2163 // "\\u0927\\u0943\\u0924\\u0930\\u093e\\u0937\\u094d\\u091f\\u094d\\u0930",
2164 // "\\u0909\\u0935\\u093E\\u091A\\u0943",
2165 // */
2166 // "\\u0927",
2167 // //"\\u0927\\u0930\\u094d\\u092e\\u0915\\u094d\\u0937\\u0947\\u0924\\u094d\\u0930\\u0947",
2168 // /*
2169 // "\\u0915\\u0941\\u0930\\u0941\\u0915\\u094d\\u0937\\u0947\\u0924\\u094d\\u0930\\u0947",
2170 // "\\u0938\\u092e\\u0935\\u0947\\u0924\\u093e",
2171 // "\\u092f\\u0941\\u092f\\u0941\\u0924\\u094d\\u0938\\u0935\\u0903",
2172 // "\\u092e\\u093e\\u092e\\u0915\\u093e\\u0903",
2173 // // "\\u092a\\u093e\\u0923\\u094d\\u0921\\u0935\\u093e\\u0936\\u094d\\u091a\\u0948\\u0935",
2174 // "\\u0915\\u093f\\u092e\\u0915\\u0941\\u0930\\u094d\\u0935\\u0924",
2175 // "\\u0938\\u0902\\u091c\\u0935",
2176 // */
2177 // 0
2178 // };
2179 // UErrorCode status = U_ZERO_ERROR;
2180 // UParseError parseError;
2181 // UnicodeString message;
2182 // Transliterator* latinToDevToLatin=Transliterator::createInstance("Latin-Devanagari;Devanagari-Latin", UTRANS_FORWARD, parseError, status);
2183 // Transliterator* devToLatinToDev=Transliterator::createInstance("Devanagari-Latin;Latin-Devanagari", UTRANS_FORWARD, parseError, status);
2184 // if(U_FAILURE(status)){
2185 // errln("FAIL: construction " + UnicodeString(" Error: ") + u_errorName(status));
2186 // errln("PreContext: " + prettify(parseError.preContext) + "PostContext: " + prettify( parseError.postContext) );
2187 // delete latinToDevToLatin;
2188 // delete devToLatinToDev;
2189 // return;
2190 // }
2191 // UnicodeString gotResult;
2192 // for(int i= 0; source[i] != 0; i++){
2193 // gotResult = source[i];
2194 // expect(*latinToDevToLatin,CharsToUnicodeString(source[i]),CharsToUnicodeString(source[i]));
2195 // expect(*devToLatinToDev,CharsToUnicodeString(expected[i]),CharsToUnicodeString(expected[i]));
2196 // }
2197 // delete latinToDevToLatin;
2198 // delete devToLatinToDev;
2199 }
2200
2201 /**
2202 * Inverse of "Null" should be "Null". (J21)
2203 */
TestNullInverse()2204 void TransliteratorTest::TestNullInverse() {
2205 UParseError pe;
2206 UErrorCode ec = U_ZERO_ERROR;
2207 Transliterator *t = Transliterator::createInstance("Null", UTRANS_FORWARD, pe, ec);
2208 if (t == 0 || U_FAILURE(ec)) {
2209 errln("FAIL: createInstance");
2210 return;
2211 }
2212 Transliterator *u = t->createInverse(ec);
2213 if (u == 0 || U_FAILURE(ec)) {
2214 errln("FAIL: createInverse");
2215 delete t;
2216 return;
2217 }
2218 if (u->getID() != "Null") {
2219 errln("FAIL: Inverse of Null should be Null");
2220 }
2221 delete t;
2222 delete u;
2223 }
2224
2225 /**
2226 * Check ID of inverse of alias. (J22)
2227 */
TestAliasInverseID()2228 void TransliteratorTest::TestAliasInverseID() {
2229 UnicodeString ID("Latin-Hangul", ""); // This should be any alias ID with an inverse
2230 UParseError pe;
2231 UErrorCode ec = U_ZERO_ERROR;
2232 Transliterator *t = Transliterator::createInstance(ID, UTRANS_FORWARD, pe, ec);
2233 if (t == 0 || U_FAILURE(ec)) {
2234 dataerrln("FAIL: createInstance - %s", u_errorName(ec));
2235 return;
2236 }
2237 Transliterator *u = t->createInverse(ec);
2238 if (u == 0 || U_FAILURE(ec)) {
2239 errln("FAIL: createInverse");
2240 delete t;
2241 return;
2242 }
2243 UnicodeString exp = "Hangul-Latin";
2244 UnicodeString got = u->getID();
2245 if (got != exp) {
2246 errln((UnicodeString)"FAIL: Inverse of " + ID + " is " + got +
2247 ", expected " + exp);
2248 }
2249 delete t;
2250 delete u;
2251 }
2252
2253 /**
2254 * Test IDs of inverses of compound transliterators. (J20)
2255 */
TestCompoundInverseID()2256 void TransliteratorTest::TestCompoundInverseID() {
2257 UnicodeString ID = "Latin-Jamo;NFC(NFD)";
2258 UParseError pe;
2259 UErrorCode ec = U_ZERO_ERROR;
2260 Transliterator *t = Transliterator::createInstance(ID, UTRANS_FORWARD, pe, ec);
2261 if (t == 0 || U_FAILURE(ec)) {
2262 dataerrln("FAIL: createInstance - %s", u_errorName(ec));
2263 return;
2264 }
2265 Transliterator *u = t->createInverse(ec);
2266 if (u == 0 || U_FAILURE(ec)) {
2267 errln("FAIL: createInverse");
2268 delete t;
2269 return;
2270 }
2271 UnicodeString exp = "NFD(NFC);Jamo-Latin";
2272 UnicodeString got = u->getID();
2273 if (got != exp) {
2274 errln((UnicodeString)"FAIL: Inverse of " + ID + " is " + got +
2275 ", expected " + exp);
2276 }
2277 delete t;
2278 delete u;
2279 }
2280
2281 /**
2282 * Test undefined variable.
2283
2284 */
TestUndefinedVariable()2285 void TransliteratorTest::TestUndefinedVariable() {
2286 UnicodeString rule = "$initial } a <> \\u1161;";
2287 UParseError pe;
2288 UErrorCode ec = U_ZERO_ERROR;
2289 Transliterator *t = Transliterator::createFromRules("<ID>", rule, UTRANS_FORWARD, pe, ec);
2290 delete t;
2291 if (U_FAILURE(ec)) {
2292 logln((UnicodeString)"OK: Got exception for " + rule + ", as expected: " +
2293 u_errorName(ec));
2294 return;
2295 }
2296 errln((UnicodeString)"Fail: bogus rule " + rule + " compiled with error " +
2297 u_errorName(ec));
2298 }
2299
2300 /**
2301 * Test empty context.
2302 */
TestEmptyContext()2303 void TransliteratorTest::TestEmptyContext() {
2304 expect(" { a } > b;", "xay a ", "xby b ");
2305 }
2306
2307 /**
2308 * Test compound filter ID syntax
2309 */
TestCompoundFilterID(void)2310 void TransliteratorTest::TestCompoundFilterID(void) {
2311 static const char* DATA[] = {
2312 // Col. 1 = ID or rule set (latter must start with #)
2313
2314 // = columns > 1 are null if expect col. 1 to be illegal =
2315
2316 // Col. 2 = direction, "F..." or "R..."
2317 // Col. 3 = source string
2318 // Col. 4 = exp result
2319
2320 "[abc]; [abc]", NULL, NULL, NULL, // multiple filters
2321 "Latin-Greek; [abc];", NULL, NULL, NULL, // misplaced filter
2322 "[b]; Latin-Greek; Upper; ([xyz])", "F", "abc", "a\\u0392c",
2323 "[b]; (Lower); Latin-Greek; Upper(); ([\\u0392])", "R", "\\u0391\\u0392\\u0393", "\\u0391b\\u0393",
2324 "#\n::[b]; ::Latin-Greek; ::Upper; ::([xyz]);", "F", "abc", "a\\u0392c",
2325 "#\n::[b]; ::(Lower); ::Latin-Greek; ::Upper(); ::([\\u0392]);", "R", "\\u0391\\u0392\\u0393", "\\u0391b\\u0393",
2326 NULL,
2327 };
2328
2329 for (int32_t i=0; DATA[i]; i+=4) {
2330 UnicodeString id = CharsToUnicodeString(DATA[i]);
2331 UTransDirection direction = (DATA[i+1] != NULL && DATA[i+1][0] == 'R') ?
2332 UTRANS_REVERSE : UTRANS_FORWARD;
2333 UnicodeString source;
2334 UnicodeString exp;
2335 if (DATA[i+2] != NULL) {
2336 source = CharsToUnicodeString(DATA[i+2]);
2337 exp = CharsToUnicodeString(DATA[i+3]);
2338 }
2339 UBool expOk = (DATA[i+1] != NULL);
2340 Transliterator* t = NULL;
2341 UParseError pe;
2342 UErrorCode ec = U_ZERO_ERROR;
2343 if (id.charAt(0) == 0x23/*#*/) {
2344 t = Transliterator::createFromRules("ID", id, direction, pe, ec);
2345 } else {
2346 t = Transliterator::createInstance(id, direction, pe, ec);
2347 }
2348 UBool ok = (t != NULL && U_SUCCESS(ec));
2349 UnicodeString transID;
2350 if (t!=0) {
2351 transID = t->getID();
2352 }
2353 else {
2354 transID = UnicodeString("NULL", "");
2355 }
2356 if (ok == expOk) {
2357 logln((UnicodeString)"Ok: " + id + " => " + transID + ", " +
2358 u_errorName(ec));
2359 if (source.length() != 0) {
2360 expect(*t, source, exp);
2361 }
2362 delete t;
2363 } else {
2364 dataerrln((UnicodeString)"FAIL: " + id + " => " + transID + ", " +
2365 u_errorName(ec));
2366 }
2367 }
2368 }
2369
2370 /**
2371 * Test new property set syntax
2372 */
TestPropertySet()2373 void TransliteratorTest::TestPropertySet() {
2374 expect(UNICODE_STRING_SIMPLE("a>A; \\p{Lu}>x; \\p{ANY}>y;"), "abcDEF", "Ayyxxx");
2375 expect("(.+)>'[' $1 ']';", " a stitch \n in time \r saves 9",
2376 "[ a stitch ]\n[ in time ]\r[ saves 9]");
2377 }
2378
2379 /**
2380 * Test various failure points of the new 2.0 engine.
2381 */
TestNewEngine()2382 void TransliteratorTest::TestNewEngine() {
2383 UParseError pe;
2384 UErrorCode ec = U_ZERO_ERROR;
2385 Transliterator *t = Transliterator::createInstance("Latin-Hiragana", UTRANS_FORWARD, pe, ec);
2386 if (t == 0 || U_FAILURE(ec)) {
2387 dataerrln("FAIL: createInstance Latin-Hiragana - %s", u_errorName(ec));
2388 return;
2389 }
2390 // Katakana should be untouched
2391 expect(*t, CharsToUnicodeString("a\\u3042\\u30A2"),
2392 CharsToUnicodeString("\\u3042\\u3042\\u30A2"));
2393
2394 delete t;
2395
2396 #if 1
2397 // This test will only work if Transliterator.ROLLBACK is
2398 // true. Otherwise, this test will fail, revealing a
2399 // limitation of global filters in incremental mode.
2400 Transliterator *a =
2401 Transliterator::createFromRules("a_to_A", "a > A;", UTRANS_FORWARD, pe, ec);
2402 Transliterator *A =
2403 Transliterator::createFromRules("A_to_b", "A > b;", UTRANS_FORWARD, pe, ec);
2404 if (U_FAILURE(ec)) {
2405 delete a;
2406 delete A;
2407 return;
2408 }
2409
2410 Transliterator* array[3];
2411 array[0] = a;
2412 array[1] = Transliterator::createInstance("NFD", UTRANS_FORWARD, pe, ec);
2413 array[2] = A;
2414 if (U_FAILURE(ec)) {
2415 errln("FAIL: createInstance NFD");
2416 delete a;
2417 delete A;
2418 delete array[1];
2419 return;
2420 }
2421
2422 t = new CompoundTransliterator(array, 3, new UnicodeSet("[:Ll:]", ec));
2423 if (U_FAILURE(ec)) {
2424 errln("FAIL: UnicodeSet constructor");
2425 delete a;
2426 delete A;
2427 delete array[1];
2428 delete t;
2429 return;
2430 }
2431
2432 expect(*t, "aAaA", "bAbA");
2433
2434 assertTrue("countElements", t->countElements() == 3);
2435 assertEquals("getElement(0)", t->getElement(0, ec).getID(), "a_to_A");
2436 assertEquals("getElement(1)", t->getElement(1, ec).getID(), "NFD");
2437 assertEquals("getElement(2)", t->getElement(2, ec).getID(), "A_to_b");
2438 assertSuccess("getElement", ec);
2439
2440 delete a;
2441 delete A;
2442 delete array[1];
2443 delete t;
2444 #endif
2445
2446 expect("$smooth = x; $macron = q; [:^L:] { ([aeiouyAEIOUY] $macron?) } [^aeiouyAEIOUY$smooth$macron] > | $1 $smooth ;",
2447 "a",
2448 "ax");
2449
2450 UnicodeString gr = CharsToUnicodeString(
2451 "$ddot = \\u0308 ;"
2452 "$lcgvowel = [\\u03b1\\u03b5\\u03b7\\u03b9\\u03bf\\u03c5\\u03c9] ;"
2453 "$rough = \\u0314 ;"
2454 "($lcgvowel+ $ddot?) $rough > h | $1 ;"
2455 "\\u03b1 <> a ;"
2456 "$rough <> h ;");
2457
2458 expect(gr, CharsToUnicodeString("\\u03B1\\u0314"), "ha");
2459 }
2460
2461 /**
2462 * Test quantified segment behavior. We want:
2463 * ([abc])+ > x $1 x; applied to "cba" produces "xax"
2464 */
TestQuantifiedSegment(void)2465 void TransliteratorTest::TestQuantifiedSegment(void) {
2466 // The normal case
2467 expect("([abc]+) > x $1 x;", "cba", "xcbax");
2468
2469 // The tricky case; the quantifier is around the segment
2470 expect("([abc])+ > x $1 x;", "cba", "xax");
2471
2472 // Tricky case in reverse direction
2473 expect("([abc])+ { q > x $1 x;", "cbaq", "cbaxax");
2474
2475 // Check post-context segment
2476 expect("{q} ([a-d])+ > '(' $1 ')';", "ddqcba", "dd(a)cba");
2477
2478 // Test toRule/toPattern for non-quantified segment.
2479 // Careful with spacing here.
2480 UnicodeString r("([a-c]){q} > x $1 x;");
2481 UParseError pe;
2482 UErrorCode ec = U_ZERO_ERROR;
2483 Transliterator* t = Transliterator::createFromRules("ID", r, UTRANS_FORWARD, pe, ec);
2484 if (U_FAILURE(ec)) {
2485 errln("FAIL: createFromRules");
2486 delete t;
2487 return;
2488 }
2489 UnicodeString rr;
2490 t->toRules(rr, TRUE);
2491 if (r != rr) {
2492 errln((UnicodeString)"FAIL: \"" + r + "\" x toRules() => \"" + rr + "\"");
2493 } else {
2494 logln((UnicodeString)"Ok: \"" + r + "\" x toRules() => \"" + rr + "\"");
2495 }
2496 delete t;
2497
2498 // Test toRule/toPattern for quantified segment.
2499 // Careful with spacing here.
2500 r = "([a-c])+{q} > x $1 x;";
2501 t = Transliterator::createFromRules("ID", r, UTRANS_FORWARD, pe, ec);
2502 if (U_FAILURE(ec)) {
2503 errln("FAIL: createFromRules");
2504 delete t;
2505 return;
2506 }
2507 t->toRules(rr, TRUE);
2508 if (r != rr) {
2509 errln((UnicodeString)"FAIL: \"" + r + "\" x toRules() => \"" + rr + "\"");
2510 } else {
2511 logln((UnicodeString)"Ok: \"" + r + "\" x toRules() => \"" + rr + "\"");
2512 }
2513 delete t;
2514 }
2515
2516 //======================================================================
2517 // Ram's tests
2518 //======================================================================
TestDevanagariLatinRT()2519 void TransliteratorTest::TestDevanagariLatinRT(){
2520 const int MAX_LEN= 52;
2521 const char* const source[MAX_LEN] = {
2522 "bh\\u0101rata",
2523 "kra",
2524 "k\\u1E63a",
2525 "khra",
2526 "gra",
2527 "\\u1E45ra",
2528 "cra",
2529 "chra",
2530 "j\\u00F1a",
2531 "jhra",
2532 "\\u00F1ra",
2533 "\\u1E6Dya",
2534 "\\u1E6Dhra",
2535 "\\u1E0Dya",
2536 //"r\\u0323ya", // \u095c is not valid in Devanagari
2537 "\\u1E0Dhya",
2538 "\\u1E5Bhra",
2539 "\\u1E47ra",
2540 "tta",
2541 "thra",
2542 "dda",
2543 "dhra",
2544 "nna",
2545 "pra",
2546 "phra",
2547 "bra",
2548 "bhra",
2549 "mra",
2550 "\\u1E49ra",
2551 //"l\\u0331ra",
2552 "yra",
2553 "\\u1E8Fra",
2554 //"l-",
2555 "vra",
2556 "\\u015Bra",
2557 "\\u1E63ra",
2558 "sra",
2559 "hma",
2560 "\\u1E6D\\u1E6Da",
2561 "\\u1E6D\\u1E6Dha",
2562 "\\u1E6Dh\\u1E6Dha",
2563 "\\u1E0D\\u1E0Da",
2564 "\\u1E0D\\u1E0Dha",
2565 "\\u1E6Dya",
2566 "\\u1E6Dhya",
2567 "\\u1E0Dya",
2568 "\\u1E0Dhya",
2569 // Not roundtrippable --
2570 // \\u0939\\u094d\\u094d\\u092E - hma
2571 // \\u0939\\u094d\\u092E - hma
2572 // CharsToUnicodeString("hma"),
2573 "hya",
2574 "\\u015Br\\u0325",
2575 "\\u015Bca",
2576 "\\u0115",
2577 "san\\u0304j\\u012Bb s\\u0113nagupta",
2578 "\\u0101nand vaddir\\u0101ju",
2579 "\\u0101",
2580 "a"
2581 };
2582 const char* const expected[MAX_LEN] = {
2583 "\\u092D\\u093E\\u0930\\u0924", /* bha\\u0304rata */
2584 "\\u0915\\u094D\\u0930", /* kra */
2585 "\\u0915\\u094D\\u0937", /* ks\\u0323a */
2586 "\\u0916\\u094D\\u0930", /* khra */
2587 "\\u0917\\u094D\\u0930", /* gra */
2588 "\\u0919\\u094D\\u0930", /* n\\u0307ra */
2589 "\\u091A\\u094D\\u0930", /* cra */
2590 "\\u091B\\u094D\\u0930", /* chra */
2591 "\\u091C\\u094D\\u091E", /* jn\\u0303a */
2592 "\\u091D\\u094D\\u0930", /* jhra */
2593 "\\u091E\\u094D\\u0930", /* n\\u0303ra */
2594 "\\u091F\\u094D\\u092F", /* t\\u0323ya */
2595 "\\u0920\\u094D\\u0930", /* t\\u0323hra */
2596 "\\u0921\\u094D\\u092F", /* d\\u0323ya */
2597 //"\\u095C\\u094D\\u092F", /* r\\u0323ya */ // \u095c is not valid in Devanagari
2598 "\\u0922\\u094D\\u092F", /* d\\u0323hya */
2599 "\\u0922\\u093C\\u094D\\u0930", /* r\\u0323hra */
2600 "\\u0923\\u094D\\u0930", /* n\\u0323ra */
2601 "\\u0924\\u094D\\u0924", /* tta */
2602 "\\u0925\\u094D\\u0930", /* thra */
2603 "\\u0926\\u094D\\u0926", /* dda */
2604 "\\u0927\\u094D\\u0930", /* dhra */
2605 "\\u0928\\u094D\\u0928", /* nna */
2606 "\\u092A\\u094D\\u0930", /* pra */
2607 "\\u092B\\u094D\\u0930", /* phra */
2608 "\\u092C\\u094D\\u0930", /* bra */
2609 "\\u092D\\u094D\\u0930", /* bhra */
2610 "\\u092E\\u094D\\u0930", /* mra */
2611 "\\u0929\\u094D\\u0930", /* n\\u0331ra */
2612 //"\\u0934\\u094D\\u0930", /* l\\u0331ra */
2613 "\\u092F\\u094D\\u0930", /* yra */
2614 "\\u092F\\u093C\\u094D\\u0930", /* y\\u0307ra */
2615 //"l-",
2616 "\\u0935\\u094D\\u0930", /* vra */
2617 "\\u0936\\u094D\\u0930", /* s\\u0301ra */
2618 "\\u0937\\u094D\\u0930", /* s\\u0323ra */
2619 "\\u0938\\u094D\\u0930", /* sra */
2620 "\\u0939\\u094d\\u092E", /* hma */
2621 "\\u091F\\u094D\\u091F", /* t\\u0323t\\u0323a */
2622 "\\u091F\\u094D\\u0920", /* t\\u0323t\\u0323ha */
2623 "\\u0920\\u094D\\u0920", /* t\\u0323ht\\u0323ha*/
2624 "\\u0921\\u094D\\u0921", /* d\\u0323d\\u0323a */
2625 "\\u0921\\u094D\\u0922", /* d\\u0323d\\u0323ha */
2626 "\\u091F\\u094D\\u092F", /* t\\u0323ya */
2627 "\\u0920\\u094D\\u092F", /* t\\u0323hya */
2628 "\\u0921\\u094D\\u092F", /* d\\u0323ya */
2629 "\\u0922\\u094D\\u092F", /* d\\u0323hya */
2630 // "hma", /* hma */
2631 "\\u0939\\u094D\\u092F", /* hya */
2632 "\\u0936\\u0943", /* s\\u0301r\\u0325a */
2633 "\\u0936\\u094D\\u091A", /* s\\u0301ca */
2634 "\\u090d", /* e\\u0306 */
2635 "\\u0938\\u0902\\u091C\\u0940\\u092C\\u094D \\u0938\\u0947\\u0928\\u0917\\u0941\\u092A\\u094D\\u0924",
2636 "\\u0906\\u0928\\u0902\\u0926\\u094D \\u0935\\u0926\\u094D\\u0926\\u093F\\u0930\\u093E\\u091C\\u0941",
2637 "\\u0906",
2638 "\\u0905",
2639 };
2640 UErrorCode status = U_ZERO_ERROR;
2641 UParseError parseError;
2642 UnicodeString message;
2643 Transliterator* latinToDev=Transliterator::createInstance("Latin-Devanagari", UTRANS_FORWARD, parseError, status);
2644 Transliterator* devToLatin=Transliterator::createInstance("Devanagari-Latin", UTRANS_FORWARD, parseError, status);
2645 if(U_FAILURE(status)){
2646 dataerrln("FAIL: construction " + UnicodeString(" Error: ") + u_errorName(status));
2647 dataerrln("PreContext: " + prettify(parseError.preContext) + " PostContext: " + prettify( parseError.postContext) );
2648 return;
2649 }
2650 UnicodeString gotResult;
2651 for(int i= 0; i<MAX_LEN; i++){
2652 gotResult = source[i];
2653 expect(*latinToDev,CharsToUnicodeString(source[i]),CharsToUnicodeString(expected[i]));
2654 expect(*devToLatin,CharsToUnicodeString(expected[i]),CharsToUnicodeString(source[i]));
2655 }
2656 delete latinToDev;
2657 delete devToLatin;
2658 }
2659
TestTeluguLatinRT()2660 void TransliteratorTest::TestTeluguLatinRT(){
2661 const int MAX_LEN=10;
2662 const char* const source[MAX_LEN] = {
2663 "raghur\\u0101m vi\\u015Bvan\\u0101dha", /* Raghuram Viswanadha */
2664 "\\u0101nand vaddir\\u0101ju", /* Anand Vaddiraju */
2665 "r\\u0101j\\u012Bv ka\\u015Barab\\u0101da", /* Rajeev Kasarabada */
2666 "san\\u0304j\\u012Bv ka\\u015Barab\\u0101da", /* sanjeev kasarabada */
2667 "san\\u0304j\\u012Bb sen'gupta", /* sanjib sengupata */
2668 "amar\\u0113ndra hanum\\u0101nula", /* Amarendra hanumanula */
2669 "ravi kum\\u0101r vi\\u015Bvan\\u0101dha", /* Ravi Kumar Viswanadha */
2670 "\\u0101ditya kandr\\u0113gula", /* Aditya Kandregula */
2671 "\\u015Br\\u012Bdhar ka\\u1E47\\u1E6Dama\\u015Be\\u1E6D\\u1E6Di",/* Shridhar Kantamsetty */
2672 "m\\u0101dhav de\\u015Be\\u1E6D\\u1E6Di" /* Madhav Desetty */
2673 };
2674
2675 const char* const expected[MAX_LEN] = {
2676 "\\u0c30\\u0c18\\u0c41\\u0c30\\u0c3e\\u0c2e\\u0c4d \\u0c35\\u0c3f\\u0c36\\u0c4d\\u0c35\\u0c28\\u0c3e\\u0c27",
2677 "\\u0c06\\u0c28\\u0c02\\u0c26\\u0c4d \\u0C35\\u0C26\\u0C4D\\u0C26\\u0C3F\\u0C30\\u0C3E\\u0C1C\\u0C41",
2678 "\\u0c30\\u0c3e\\u0c1c\\u0c40\\u0c35\\u0c4d \\u0c15\\u0c36\\u0c30\\u0c2c\\u0c3e\\u0c26",
2679 "\\u0c38\\u0c02\\u0c1c\\u0c40\\u0c35\\u0c4d \\u0c15\\u0c36\\u0c30\\u0c2c\\u0c3e\\u0c26",
2680 "\\u0c38\\u0c02\\u0c1c\\u0c40\\u0c2c\\u0c4d \\u0c38\\u0c46\\u0c28\\u0c4d\\u0c17\\u0c41\\u0c2a\\u0c4d\\u0c24",
2681 "\\u0c05\\u0c2e\\u0c30\\u0c47\\u0c02\\u0c26\\u0c4d\\u0c30 \\u0c39\\u0c28\\u0c41\\u0c2e\\u0c3e\\u0c28\\u0c41\\u0c32",
2682 "\\u0c30\\u0c35\\u0c3f \\u0c15\\u0c41\\u0c2e\\u0c3e\\u0c30\\u0c4d \\u0c35\\u0c3f\\u0c36\\u0c4d\\u0c35\\u0c28\\u0c3e\\u0c27",
2683 "\\u0c06\\u0c26\\u0c3f\\u0c24\\u0c4d\\u0c2f \\u0C15\\u0C02\\u0C26\\u0C4D\\u0C30\\u0C47\\u0C17\\u0C41\\u0c32",
2684 "\\u0c36\\u0c4d\\u0c30\\u0c40\\u0C27\\u0C30\\u0C4D \\u0c15\\u0c02\\u0c1f\\u0c2e\\u0c36\\u0c46\\u0c1f\\u0c4d\\u0c1f\\u0c3f",
2685 "\\u0c2e\\u0c3e\\u0c27\\u0c35\\u0c4d \\u0c26\\u0c46\\u0c36\\u0c46\\u0c1f\\u0c4d\\u0c1f\\u0c3f",
2686 };
2687
2688 UErrorCode status = U_ZERO_ERROR;
2689 UParseError parseError;
2690 UnicodeString message;
2691 Transliterator* latinToDev=Transliterator::createInstance("Latin-Telugu", UTRANS_FORWARD, parseError, status);
2692 Transliterator* devToLatin=Transliterator::createInstance("Telugu-Latin", UTRANS_FORWARD, parseError, status);
2693 if(U_FAILURE(status)){
2694 dataerrln("FAIL: construction " + UnicodeString(" Error: ") + u_errorName(status));
2695 dataerrln("PreContext: " + prettify(parseError.preContext) + " PostContext: " + prettify( parseError.postContext) );
2696 return;
2697 }
2698 UnicodeString gotResult;
2699 for(int i= 0; i<MAX_LEN; i++){
2700 gotResult = source[i];
2701 expect(*latinToDev,CharsToUnicodeString(source[i]),CharsToUnicodeString(expected[i]));
2702 expect(*devToLatin,CharsToUnicodeString(expected[i]),CharsToUnicodeString(source[i]));
2703 }
2704 delete latinToDev;
2705 delete devToLatin;
2706 }
2707
TestSanskritLatinRT()2708 void TransliteratorTest::TestSanskritLatinRT(){
2709 const int MAX_LEN =16;
2710 const char* const source[MAX_LEN] = {
2711 "rmk\\u1E63\\u0113t",
2712 "\\u015Br\\u012Bmad",
2713 "bhagavadg\\u012Bt\\u0101",
2714 "adhy\\u0101ya",
2715 "arjuna",
2716 "vi\\u1E63\\u0101da",
2717 "y\\u014Dga",
2718 "dhr\\u0325tar\\u0101\\u1E63\\u1E6Dra",
2719 "uv\\u0101cr\\u0325",
2720 "dharmak\\u1E63\\u0113tr\\u0113",
2721 "kuruk\\u1E63\\u0113tr\\u0113",
2722 "samav\\u0113t\\u0101",
2723 "yuyutsava\\u1E25",
2724 "m\\u0101mak\\u0101\\u1E25",
2725 // "p\\u0101\\u1E47\\u1E0Dav\\u0101\\u015Bcaiva",
2726 "kimakurvata",
2727 "san\\u0304java",
2728 };
2729 const char* const expected[MAX_LEN] = {
2730 "\\u0930\\u094D\\u092E\\u094D\\u0915\\u094D\\u0937\\u0947\\u0924\\u094D",
2731 "\\u0936\\u094d\\u0930\\u0940\\u092e\\u0926\\u094d",
2732 "\\u092d\\u0917\\u0935\\u0926\\u094d\\u0917\\u0940\\u0924\\u093e",
2733 "\\u0905\\u0927\\u094d\\u092f\\u093e\\u092f",
2734 "\\u0905\\u0930\\u094d\\u091c\\u0941\\u0928",
2735 "\\u0935\\u093f\\u0937\\u093e\\u0926",
2736 "\\u092f\\u094b\\u0917",
2737 "\\u0927\\u0943\\u0924\\u0930\\u093e\\u0937\\u094d\\u091f\\u094d\\u0930",
2738 "\\u0909\\u0935\\u093E\\u091A\\u0943",
2739 "\\u0927\\u0930\\u094d\\u092e\\u0915\\u094d\\u0937\\u0947\\u0924\\u094d\\u0930\\u0947",
2740 "\\u0915\\u0941\\u0930\\u0941\\u0915\\u094d\\u0937\\u0947\\u0924\\u094d\\u0930\\u0947",
2741 "\\u0938\\u092e\\u0935\\u0947\\u0924\\u093e",
2742 "\\u092f\\u0941\\u092f\\u0941\\u0924\\u094d\\u0938\\u0935\\u0903",
2743 "\\u092e\\u093e\\u092e\\u0915\\u093e\\u0903",
2744 //"\\u092a\\u093e\\u0923\\u094d\\u0921\\u0935\\u093e\\u0936\\u094d\\u091a\\u0948\\u0935",
2745 "\\u0915\\u093f\\u092e\\u0915\\u0941\\u0930\\u094d\\u0935\\u0924",
2746 "\\u0938\\u0902\\u091c\\u0935",
2747 };
2748 UErrorCode status = U_ZERO_ERROR;
2749 UParseError parseError;
2750 UnicodeString message;
2751 Transliterator* latinToDev=Transliterator::createInstance("Latin-Devanagari", UTRANS_FORWARD, parseError, status);
2752 Transliterator* devToLatin=Transliterator::createInstance("Devanagari-Latin", UTRANS_FORWARD, parseError, status);
2753 if(U_FAILURE(status)){
2754 dataerrln("FAIL: construction " + UnicodeString(" Error: ") + u_errorName(status));
2755 dataerrln("PreContext: " + prettify(parseError.preContext) + " PostContext: " + prettify( parseError.postContext) );
2756 return;
2757 }
2758 UnicodeString gotResult;
2759 for(int i= 0; i<MAX_LEN; i++){
2760 gotResult = source[i];
2761 expect(*latinToDev,CharsToUnicodeString(source[i]),CharsToUnicodeString(expected[i]));
2762 expect(*devToLatin,CharsToUnicodeString(expected[i]),CharsToUnicodeString(source[i]));
2763 }
2764 delete latinToDev;
2765 delete devToLatin;
2766 }
2767
2768
TestCompoundLatinRT()2769 void TransliteratorTest::TestCompoundLatinRT(){
2770 const char* const source[] = {
2771 "rmk\\u1E63\\u0113t",
2772 "\\u015Br\\u012Bmad",
2773 "bhagavadg\\u012Bt\\u0101",
2774 "adhy\\u0101ya",
2775 "arjuna",
2776 "vi\\u1E63\\u0101da",
2777 "y\\u014Dga",
2778 "dhr\\u0325tar\\u0101\\u1E63\\u1E6Dra",
2779 "uv\\u0101cr\\u0325",
2780 "dharmak\\u1E63\\u0113tr\\u0113",
2781 "kuruk\\u1E63\\u0113tr\\u0113",
2782 "samav\\u0113t\\u0101",
2783 "yuyutsava\\u1E25",
2784 "m\\u0101mak\\u0101\\u1E25",
2785 // "p\\u0101\\u1E47\\u1E0Dav\\u0101\\u015Bcaiva",
2786 "kimakurvata",
2787 "san\\u0304java"
2788 };
2789 const int MAX_LEN = sizeof(source)/sizeof(source[0]);
2790 const char* const expected[MAX_LEN] = {
2791 "\\u0930\\u094D\\u092E\\u094D\\u0915\\u094D\\u0937\\u0947\\u0924\\u094D",
2792 "\\u0936\\u094d\\u0930\\u0940\\u092e\\u0926\\u094d",
2793 "\\u092d\\u0917\\u0935\\u0926\\u094d\\u0917\\u0940\\u0924\\u093e",
2794 "\\u0905\\u0927\\u094d\\u092f\\u093e\\u092f",
2795 "\\u0905\\u0930\\u094d\\u091c\\u0941\\u0928",
2796 "\\u0935\\u093f\\u0937\\u093e\\u0926",
2797 "\\u092f\\u094b\\u0917",
2798 "\\u0927\\u0943\\u0924\\u0930\\u093e\\u0937\\u094d\\u091f\\u094d\\u0930",
2799 "\\u0909\\u0935\\u093E\\u091A\\u0943",
2800 "\\u0927\\u0930\\u094d\\u092e\\u0915\\u094d\\u0937\\u0947\\u0924\\u094d\\u0930\\u0947",
2801 "\\u0915\\u0941\\u0930\\u0941\\u0915\\u094d\\u0937\\u0947\\u0924\\u094d\\u0930\\u0947",
2802 "\\u0938\\u092e\\u0935\\u0947\\u0924\\u093e",
2803 "\\u092f\\u0941\\u092f\\u0941\\u0924\\u094d\\u0938\\u0935\\u0903",
2804 "\\u092e\\u093e\\u092e\\u0915\\u093e\\u0903",
2805 // "\\u092a\\u093e\\u0923\\u094d\\u0921\\u0935\\u093e\\u0936\\u094d\\u091a\\u0948\\u0935",
2806 "\\u0915\\u093f\\u092e\\u0915\\u0941\\u0930\\u094d\\u0935\\u0924",
2807 "\\u0938\\u0902\\u091c\\u0935"
2808 };
2809 if(MAX_LEN != sizeof(expected)/sizeof(expected[0])) {
2810 errln("error in TestCompoundLatinRT: source[] and expected[] have different lengths!");
2811 return;
2812 }
2813
2814 UErrorCode status = U_ZERO_ERROR;
2815 UParseError parseError;
2816 UnicodeString message;
2817 Transliterator* devToLatinToDev =Transliterator::createInstance("Devanagari-Latin;Latin-Devanagari", UTRANS_FORWARD, parseError, status);
2818 Transliterator* latinToDevToLatin=Transliterator::createInstance("Latin-Devanagari;Devanagari-Latin", UTRANS_FORWARD, parseError, status);
2819 Transliterator* devToTelToDev =Transliterator::createInstance("Devanagari-Telugu;Telugu-Devanagari", UTRANS_FORWARD, parseError, status);
2820 Transliterator* latinToTelToLatin=Transliterator::createInstance("Latin-Telugu;Telugu-Latin", UTRANS_FORWARD, parseError, status);
2821
2822 if(U_FAILURE(status)){
2823 dataerrln("FAIL: construction " + UnicodeString(" Error: ") + u_errorName(status));
2824 dataerrln("PreContext: " + prettify(parseError.preContext) + " PostContext: " + prettify( parseError.postContext) );
2825 return;
2826 }
2827 UnicodeString gotResult;
2828 for(int i= 0; i<MAX_LEN; i++){
2829 gotResult = source[i];
2830 expect(*devToLatinToDev,CharsToUnicodeString(expected[i]),CharsToUnicodeString(expected[i]));
2831 expect(*latinToDevToLatin,CharsToUnicodeString(source[i]),CharsToUnicodeString(source[i]));
2832 expect(*latinToTelToLatin,CharsToUnicodeString(source[i]),CharsToUnicodeString(source[i]));
2833
2834 }
2835 delete(latinToDevToLatin);
2836 delete(devToLatinToDev);
2837 delete(devToTelToDev);
2838 delete(latinToTelToLatin);
2839 }
2840
2841 /**
2842 * Test Gurmukhi-Devanagari Tippi and Bindi
2843 */
TestGurmukhiDevanagari()2844 void TransliteratorTest::TestGurmukhiDevanagari(){
2845 // the rule says:
2846 // (\u0902) (when preceded by vowel) ---> (\u0A02)
2847 // (\u0902) (when preceded by consonant) ---> (\u0A70)
2848 UErrorCode status = U_ZERO_ERROR;
2849 UnicodeSet vowel(UnicodeString("[\\u0905-\\u090A \\u090F\\u0910\\u0913\\u0914 \\u093e-\\u0942\\u0947\\u0948\\u094B\\u094C\\u094D]", -1, US_INV).unescape(), status);
2850 UnicodeSet non_vowel(UnicodeString("[\\u0915-\\u0928\\u092A-\\u0930]", -1, US_INV).unescape(), status);
2851 UParseError parseError;
2852
2853 UnicodeSetIterator vIter(vowel);
2854 UnicodeSetIterator nvIter(non_vowel);
2855 Transliterator* trans = Transliterator::createInstance("Devanagari-Gurmukhi",UTRANS_FORWARD, parseError, status);
2856 if(U_FAILURE(status)) {
2857 dataerrln("Error creating transliterator %s", u_errorName(status));
2858 delete trans;
2859 return;
2860 }
2861 UnicodeString src (" \\u0902", -1, US_INV);
2862 UnicodeString expected(" \\u0A02", -1, US_INV);
2863 src = src.unescape();
2864 expected= expected.unescape();
2865
2866 while(vIter.next()){
2867 src.setCharAt(0,(UChar) vIter.getCodepoint());
2868 expected.setCharAt(0,(UChar) (vIter.getCodepoint()+0x0100));
2869 expect(*trans,src,expected);
2870 }
2871
2872 expected.setCharAt(1,0x0A70);
2873 while(nvIter.next()){
2874 //src.setCharAt(0,(char) nvIter.codepoint);
2875 src.setCharAt(0,(UChar)nvIter.getCodepoint());
2876 expected.setCharAt(0,(UChar) (nvIter.getCodepoint()+0x0100));
2877 expect(*trans,src,expected);
2878 }
2879 delete trans;
2880 }
2881 /**
2882 * Test instantiation from a locale.
2883 */
TestLocaleInstantiation(void)2884 void TransliteratorTest::TestLocaleInstantiation(void) {
2885 UParseError pe;
2886 UErrorCode ec = U_ZERO_ERROR;
2887 Transliterator *t = Transliterator::createInstance("ru_RU-Latin", UTRANS_FORWARD, pe, ec);
2888 if (U_FAILURE(ec)) {
2889 dataerrln("FAIL: createInstance(ru_RU-Latin) - %s", u_errorName(ec));
2890 delete t;
2891 return;
2892 }
2893 expect(*t, CharsToUnicodeString("\\u0430"), "a");
2894 delete t;
2895
2896 t = Transliterator::createInstance("en-el", UTRANS_FORWARD, pe, ec);
2897 if (U_FAILURE(ec)) {
2898 errln("FAIL: createInstance(en-el)");
2899 delete t;
2900 return;
2901 }
2902 expect(*t, "a", CharsToUnicodeString("\\u03B1"));
2903 delete t;
2904 }
2905
2906 /**
2907 * Test title case handling of accent (should ignore accents)
2908 */
TestTitleAccents(void)2909 void TransliteratorTest::TestTitleAccents(void) {
2910 UParseError pe;
2911 UErrorCode ec = U_ZERO_ERROR;
2912 Transliterator *t = Transliterator::createInstance("Title", UTRANS_FORWARD, pe, ec);
2913 if (U_FAILURE(ec)) {
2914 errln("FAIL: createInstance(Title)");
2915 delete t;
2916 return;
2917 }
2918 expect(*t, CharsToUnicodeString("a\\u0300b can't abe"), CharsToUnicodeString("A\\u0300b Can't Abe"));
2919 delete t;
2920 }
2921
2922 /**
2923 * Basic test of a locale resource based rule.
2924 */
TestLocaleResource()2925 void TransliteratorTest::TestLocaleResource() {
2926 const char* DATA[] = {
2927 // id from to
2928 //"Latin-Greek/UNGEGN", "b", "\\u03bc\\u03c0",
2929 "Latin-el", "b", "\\u03bc\\u03c0",
2930 "Latin-Greek", "b", "\\u03B2",
2931 "Greek-Latin/UNGEGN", "\\u03B2", "v",
2932 "el-Latin", "\\u03B2", "v",
2933 "Greek-Latin", "\\u03B2", "b",
2934 };
2935 const int32_t DATA_length = sizeof(DATA) / sizeof(DATA[0]);
2936 for (int32_t i=0; i<DATA_length; i+=3) {
2937 UParseError pe;
2938 UErrorCode ec = U_ZERO_ERROR;
2939 Transliterator *t = Transliterator::createInstance(DATA[i], UTRANS_FORWARD, pe, ec);
2940 if (U_FAILURE(ec)) {
2941 dataerrln((UnicodeString)"FAIL: createInstance(" + DATA[i] + ") - " + u_errorName(ec));
2942 delete t;
2943 continue;
2944 }
2945 expect(*t, CharsToUnicodeString(DATA[i+1]),
2946 CharsToUnicodeString(DATA[i+2]));
2947 delete t;
2948 }
2949 }
2950
2951 /**
2952 * Make sure parse errors reference the right line.
2953 */
TestParseError()2954 void TransliteratorTest::TestParseError() {
2955 static const char* rule =
2956 "a > b;\n"
2957 "# more stuff\n"
2958 "d << b;";
2959 UErrorCode ec = U_ZERO_ERROR;
2960 UParseError pe;
2961 Transliterator *t = Transliterator::createFromRules("ID", rule, UTRANS_FORWARD, pe, ec);
2962 delete t;
2963 if (U_FAILURE(ec)) {
2964 UnicodeString err(pe.preContext);
2965 err.append((UChar)124/*|*/).append(pe.postContext);
2966 if (err.indexOf("d << b") >= 0) {
2967 logln("Ok: " + err);
2968 } else {
2969 errln("FAIL: " + err);
2970 }
2971 }
2972 else {
2973 errln("FAIL: no syntax error");
2974 }
2975 static const char* maskingRule =
2976 "a>x;\n"
2977 "# more stuff\n"
2978 "ab>y;";
2979 ec = U_ZERO_ERROR;
2980 delete Transliterator::createFromRules("ID", maskingRule, UTRANS_FORWARD, pe, ec);
2981 if (ec != U_RULE_MASK_ERROR) {
2982 errln("FAIL: returned %s instead of U_RULE_MASK_ERROR", u_errorName(ec));
2983 }
2984 else if (UnicodeString("a > x;") != UnicodeString(pe.preContext)) {
2985 errln("FAIL: did not get expected precontext");
2986 }
2987 else if (UnicodeString("ab > y;") != UnicodeString(pe.postContext)) {
2988 errln("FAIL: did not get expected postcontext");
2989 }
2990 }
2991
2992 /**
2993 * Make sure sets on output are disallowed.
2994 */
TestOutputSet()2995 void TransliteratorTest::TestOutputSet() {
2996 UnicodeString rule = "$set = [a-cm-n]; b > $set;";
2997 UErrorCode ec = U_ZERO_ERROR;
2998 UParseError pe;
2999 Transliterator *t = Transliterator::createFromRules("ID", rule, UTRANS_FORWARD, pe, ec);
3000 delete t;
3001 if (U_FAILURE(ec)) {
3002 UnicodeString err(pe.preContext);
3003 err.append((UChar)124/*|*/).append(pe.postContext);
3004 logln("Ok: " + err);
3005 return;
3006 }
3007 errln("FAIL: No syntax error");
3008 }
3009
3010 /**
3011 * Test the use variable range pragma, making sure that use of
3012 * variable range characters is detected and flagged as an error.
3013 */
TestVariableRange()3014 void TransliteratorTest::TestVariableRange() {
3015 UnicodeString rule = "use variable range 0x70 0x72; a > A; b > B; q > Q;";
3016 UErrorCode ec = U_ZERO_ERROR;
3017 UParseError pe;
3018 Transliterator *t = Transliterator::createFromRules("ID", rule, UTRANS_FORWARD, pe, ec);
3019 delete t;
3020 if (U_FAILURE(ec)) {
3021 UnicodeString err(pe.preContext);
3022 err.append((UChar)124/*|*/).append(pe.postContext);
3023 logln("Ok: " + err);
3024 return;
3025 }
3026 errln("FAIL: No syntax error");
3027 }
3028
3029 /**
3030 * Test invalid post context error handling
3031 */
TestInvalidPostContext()3032 void TransliteratorTest::TestInvalidPostContext() {
3033 UnicodeString rule = "a}b{c>d;";
3034 UErrorCode ec = U_ZERO_ERROR;
3035 UParseError pe;
3036 Transliterator *t = Transliterator::createFromRules("ID", rule, UTRANS_FORWARD, pe, ec);
3037 delete t;
3038 if (U_FAILURE(ec)) {
3039 UnicodeString err(pe.preContext);
3040 err.append((UChar)124/*|*/).append(pe.postContext);
3041 if (err.indexOf("a}b{c") >= 0) {
3042 logln("Ok: " + err);
3043 } else {
3044 errln("FAIL: " + err);
3045 }
3046 return;
3047 }
3048 errln("FAIL: No syntax error");
3049 }
3050
3051 /**
3052 * Test ID form variants
3053 */
TestIDForms()3054 void TransliteratorTest::TestIDForms() {
3055 const char* DATA[] = {
3056 "NFC", NULL, "NFD",
3057 "nfd", NULL, "NFC", // make sure case is ignored
3058 "Any-NFKD", NULL, "Any-NFKC",
3059 "Null", NULL, "Null",
3060 "-nfkc", "nfkc", "NFKD",
3061 "-nfkc/", "nfkc", "NFKD",
3062 "Latin-Greek/UNGEGN", NULL, "Greek-Latin/UNGEGN",
3063 "Greek/UNGEGN-Latin", "Greek-Latin/UNGEGN", "Latin-Greek/UNGEGN",
3064 "Bengali-Devanagari/", "Bengali-Devanagari", "Devanagari-Bengali",
3065 "Source-", NULL, NULL,
3066 "Source/Variant-", NULL, NULL,
3067 "Source-/Variant", NULL, NULL,
3068 "/Variant", NULL, NULL,
3069 "/Variant-", NULL, NULL,
3070 "-/Variant", NULL, NULL,
3071 "-/", NULL, NULL,
3072 "-", NULL, NULL,
3073 "/", NULL, NULL,
3074 };
3075 const int32_t DATA_length = sizeof(DATA)/sizeof(DATA[0]);
3076
3077 for (int32_t i=0; i<DATA_length; i+=3) {
3078 const char* ID = DATA[i];
3079 const char* expID = DATA[i+1];
3080 const char* expInvID = DATA[i+2];
3081 UBool expValid = (expInvID != NULL);
3082 if (expID == NULL) {
3083 expID = ID;
3084 }
3085 UParseError pe;
3086 UErrorCode ec = U_ZERO_ERROR;
3087 Transliterator *t =
3088 Transliterator::createInstance(ID, UTRANS_FORWARD, pe, ec);
3089 if (U_FAILURE(ec)) {
3090 if (!expValid) {
3091 logln((UnicodeString)"Ok: getInstance(" + ID +") => " + u_errorName(ec));
3092 } else {
3093 dataerrln((UnicodeString)"FAIL: Couldn't create " + ID + " - " + u_errorName(ec));
3094 }
3095 delete t;
3096 continue;
3097 }
3098 Transliterator *u = t->createInverse(ec);
3099 if (U_FAILURE(ec)) {
3100 errln((UnicodeString)"FAIL: Couldn't create inverse of " + ID);
3101 delete t;
3102 delete u;
3103 continue;
3104 }
3105 if (t->getID() == expID &&
3106 u->getID() == expInvID) {
3107 logln((UnicodeString)"Ok: " + ID + ".getInverse() => " + expInvID);
3108 } else {
3109 errln((UnicodeString)"FAIL: getInstance(" + ID + ") => " +
3110 t->getID() + " x getInverse() => " + u->getID() +
3111 ", expected " + expInvID);
3112 }
3113 delete t;
3114 delete u;
3115 }
3116 }
3117
3118 static const UChar SPACE[] = {32,0};
3119 static const UChar NEWLINE[] = {10,0};
3120 static const UChar RETURN[] = {13,0};
3121 static const UChar EMPTY[] = {0};
3122
checkRules(const UnicodeString & label,Transliterator & t2,const UnicodeString & testRulesForward)3123 void TransliteratorTest::checkRules(const UnicodeString& label, Transliterator& t2,
3124 const UnicodeString& testRulesForward) {
3125 UnicodeString rules2; t2.toRules(rules2, TRUE);
3126 //rules2 = TestUtility.replaceAll(rules2, new UnicodeSet("[' '\n\r]"), "");
3127 rules2.findAndReplace(SPACE, EMPTY);
3128 rules2.findAndReplace(NEWLINE, EMPTY);
3129 rules2.findAndReplace(RETURN, EMPTY);
3130
3131 UnicodeString testRules(testRulesForward); testRules.findAndReplace(SPACE, EMPTY);
3132
3133 if (rules2 != testRules) {
3134 errln(label);
3135 logln((UnicodeString)"GENERATED RULES: " + rules2);
3136 logln((UnicodeString)"SHOULD BE: " + testRulesForward);
3137 }
3138 }
3139
3140 /**
3141 * Mark's toRules test.
3142 */
TestToRulesMark()3143 void TransliteratorTest::TestToRulesMark() {
3144 const char* testRules =
3145 "::[[:Latin:][:Mark:]];"
3146 "::NFKD (NFC);"
3147 "::Lower (Lower);"
3148 "a <> \\u03B1;" // alpha
3149 "::NFKC (NFD);"
3150 "::Upper (Lower);"
3151 "::Lower ();"
3152 "::([[:Greek:][:Mark:]]);"
3153 ;
3154 const char* testRulesForward =
3155 "::[[:Latin:][:Mark:]];"
3156 "::NFKD(NFC);"
3157 "::Lower(Lower);"
3158 "a > \\u03B1;"
3159 "::NFKC(NFD);"
3160 "::Upper (Lower);"
3161 "::Lower ();"
3162 ;
3163 const char* testRulesBackward =
3164 "::[[:Greek:][:Mark:]];"
3165 "::Lower (Upper);"
3166 "::NFD(NFKC);"
3167 "\\u03B1 > a;"
3168 "::Lower(Lower);"
3169 "::NFC(NFKD);"
3170 ;
3171 UnicodeString source = CharsToUnicodeString("\\u00E1"); // a-acute
3172 UnicodeString target = CharsToUnicodeString("\\u03AC"); // alpha-acute
3173
3174 UParseError pe;
3175 UErrorCode ec = U_ZERO_ERROR;
3176 Transliterator *t2 = Transliterator::createFromRules("source-target", UnicodeString(testRules, -1, US_INV), UTRANS_FORWARD, pe, ec);
3177 Transliterator *t3 = Transliterator::createFromRules("target-source", UnicodeString(testRules, -1, US_INV), UTRANS_REVERSE, pe, ec);
3178
3179 if (U_FAILURE(ec)) {
3180 delete t2;
3181 delete t3;
3182 dataerrln((UnicodeString)"FAIL: createFromRules => " + u_errorName(ec));
3183 return;
3184 }
3185
3186 expect(*t2, source, target);
3187 expect(*t3, target, source);
3188
3189 checkRules("Failed toRules FORWARD", *t2, UnicodeString(testRulesForward, -1, US_INV));
3190 checkRules("Failed toRules BACKWARD", *t3, UnicodeString(testRulesBackward, -1, US_INV));
3191
3192 delete t2;
3193 delete t3;
3194 }
3195
3196 /**
3197 * Test Escape and Unescape transliterators.
3198 */
TestEscape()3199 void TransliteratorTest::TestEscape() {
3200 UParseError pe;
3201 UErrorCode ec;
3202 Transliterator *t;
3203
3204 ec = U_ZERO_ERROR;
3205 t = Transliterator::createInstance("Hex-Any", UTRANS_FORWARD, pe, ec);
3206 if (U_FAILURE(ec)) {
3207 errln((UnicodeString)"FAIL: createInstance");
3208 } else {
3209 expect(*t,
3210 UNICODE_STRING_SIMPLE("\\x{40}\\U000000312Q"),
3211 "@12Q");
3212 }
3213 delete t;
3214
3215 ec = U_ZERO_ERROR;
3216 t = Transliterator::createInstance("Any-Hex/C", UTRANS_FORWARD, pe, ec);
3217 if (U_FAILURE(ec)) {
3218 errln((UnicodeString)"FAIL: createInstance");
3219 } else {
3220 expect(*t,
3221 CharsToUnicodeString("A\\U0010BEEF\\uFEED"),
3222 UNICODE_STRING_SIMPLE("\\u0041\\U0010BEEF\\uFEED"));
3223 }
3224 delete t;
3225
3226 ec = U_ZERO_ERROR;
3227 t = Transliterator::createInstance("Any-Hex/Java", UTRANS_FORWARD, pe, ec);
3228 if (U_FAILURE(ec)) {
3229 errln((UnicodeString)"FAIL: createInstance");
3230 } else {
3231 expect(*t,
3232 CharsToUnicodeString("A\\U0010BEEF\\uFEED"),
3233 UNICODE_STRING_SIMPLE("\\u0041\\uDBEF\\uDEEF\\uFEED"));
3234 }
3235 delete t;
3236
3237 ec = U_ZERO_ERROR;
3238 t = Transliterator::createInstance("Any-Hex/Perl", UTRANS_FORWARD, pe, ec);
3239 if (U_FAILURE(ec)) {
3240 errln((UnicodeString)"FAIL: createInstance");
3241 } else {
3242 expect(*t,
3243 CharsToUnicodeString("A\\U0010BEEF\\uFEED"),
3244 UNICODE_STRING_SIMPLE("\\x{41}\\x{10BEEF}\\x{FEED}"));
3245 }
3246 delete t;
3247 }
3248
3249
TestAnchorMasking()3250 void TransliteratorTest::TestAnchorMasking(){
3251 UnicodeString rule ("^a > Q; a > q;");
3252 UErrorCode status= U_ZERO_ERROR;
3253 UParseError parseError;
3254
3255 Transliterator* t = Transliterator::createFromRules("ID", rule, UTRANS_FORWARD,parseError,status);
3256 if(U_FAILURE(status)){
3257 errln(UnicodeString("FAIL: ") + "ID" +
3258 ".createFromRules() => bad rules" +
3259 /*", parse error " + parseError.code +*/
3260 ", line " + parseError.line +
3261 ", offset " + parseError.offset +
3262 ", context " + prettify(parseError.preContext, TRUE) +
3263 ", rules: " + prettify(rule, TRUE));
3264 }
3265 delete t;
3266 }
3267
3268 /**
3269 * Make sure display names of variants look reasonable.
3270 */
TestDisplayName()3271 void TransliteratorTest::TestDisplayName() {
3272 #if UCONFIG_NO_FORMATTING
3273 logln("Skipping, UCONFIG_NO_FORMATTING is set\n");
3274 return;
3275 #else
3276 static const char* DATA[] = {
3277 // ID, forward name, reverse name
3278 // Update the text as necessary -- the important thing is
3279 // not the text itself, but how various cases are handled.
3280
3281 // Basic test
3282 "Any-Hex", "Any to Hex Escape", "Hex Escape to Any",
3283
3284 // Variants
3285 "Any-Hex/Perl", "Any to Hex Escape/Perl", "Hex Escape to Any/Perl",
3286
3287 // Target-only IDs
3288 "NFC", "Any to NFC", "Any to NFD",
3289 };
3290
3291 int32_t DATA_length = sizeof(DATA) / sizeof(DATA[0]);
3292
3293 Locale US("en", "US");
3294
3295 for (int32_t i=0; i<DATA_length; i+=3) {
3296 UnicodeString name;
3297 Transliterator::getDisplayName(DATA[i], US, name);
3298 if (name != DATA[i+1]) {
3299 dataerrln((UnicodeString)"FAIL: " + DATA[i] + ".getDisplayName() => " +
3300 name + ", expected " + DATA[i+1]);
3301 } else {
3302 logln((UnicodeString)"Ok: " + DATA[i] + ".getDisplayName() => " + name);
3303 }
3304 UErrorCode ec = U_ZERO_ERROR;
3305 UParseError pe;
3306 Transliterator *t = Transliterator::createInstance(DATA[i], UTRANS_REVERSE, pe, ec);
3307 if (U_FAILURE(ec)) {
3308 delete t;
3309 dataerrln("FAIL: createInstance failed - %s", u_errorName(ec));
3310 continue;
3311 }
3312 name = Transliterator::getDisplayName(t->getID(), US, name);
3313 if (name != DATA[i+2]) {
3314 dataerrln((UnicodeString)"FAIL: " + t->getID() + ".getDisplayName() => " +
3315 name + ", expected " + DATA[i+2]);
3316 } else {
3317 logln((UnicodeString)"Ok: " + t->getID() + ".getDisplayName() => " + name);
3318 }
3319 delete t;
3320 }
3321 #endif
3322 }
3323
TestSpecialCases(void)3324 void TransliteratorTest::TestSpecialCases(void) {
3325 const UnicodeString registerRules[] = {
3326 "Any-Dev1", "x > X; y > Y;",
3327 "Any-Dev2", "XY > Z",
3328 "Greek-Latin/FAKE",
3329 CharsToUnicodeString
3330 ("[^[:L:][:M:]] { \\u03bc\\u03c0 > b ; \\u03bc\\u03c0 } [^[:L:][:M:]] > b ; [^[:L:][:M:]] { [\\u039c\\u03bc][\\u03a0\\u03c0] > B ; [\\u039c\\u03bc][\\u03a0\\u03c0] } [^[:L:][:M:]] > B ;"),
3331 "" // END MARKER
3332 };
3333
3334 const UnicodeString testCases[] = {
3335 // NORMALIZATION
3336 // should add more test cases
3337 "NFD" , CharsToUnicodeString("a\\u0300 \\u00E0 \\u1100\\u1161 \\uFF76\\uFF9E\\u03D3"), "",
3338 "NFC" , CharsToUnicodeString("a\\u0300 \\u00E0 \\u1100\\u1161 \\uFF76\\uFF9E\\u03D3"), "",
3339 "NFKD", CharsToUnicodeString("a\\u0300 \\u00E0 \\u1100\\u1161 \\uFF76\\uFF9E\\u03D3"), "",
3340 "NFKC", CharsToUnicodeString("a\\u0300 \\u00E0 \\u1100\\u1161 \\uFF76\\uFF9E\\u03D3"), "",
3341
3342 // mp -> b BUG
3343 "Greek-Latin/UNGEGN", CharsToUnicodeString("(\\u03BC\\u03C0)"), "(b)",
3344 "Greek-Latin/FAKE", CharsToUnicodeString("(\\u03BC\\u03C0)"), "(b)",
3345
3346 // check for devanagari bug
3347 "nfd;Dev1;Dev2;nfc", "xy", "Z",
3348
3349 // ff, i, dotless-i, I, dotted-I, LJLjlj deseret deeDEE
3350 "Title", CharsToUnicodeString("ab'cD ffi\\u0131I\\u0130 \\u01C7\\u01C8\\u01C9 ") + DESERET_dee + DESERET_DEE,
3351 CharsToUnicodeString("Ab'cd Ffi\\u0131ii\\u0307 \\u01C8\\u01C9\\u01C9 ") + DESERET_DEE + DESERET_dee,
3352
3353 //TODO: enable this test once Titlecase works right
3354 /*
3355 "Title", CharsToUnicodeString("\\uFB00i\\u0131I\\u0130 \\u01C7\\u01C8\\u01C9 ") + DESERET_dee + DESERET_DEE,
3356 CharsToUnicodeString("Ffi\\u0131ii \\u01C8\\u01C9\\u01C9 ") + DESERET_DEE + DESERET_dee,
3357 */
3358 "Upper", CharsToUnicodeString("ab'cD \\uFB00i\\u0131I\\u0130 \\u01C7\\u01C8\\u01C9 ") + DESERET_dee + DESERET_DEE,
3359 CharsToUnicodeString("AB'CD FFIII\\u0130 \\u01C7\\u01C7\\u01C7 ") + DESERET_DEE + DESERET_DEE,
3360 "Lower", CharsToUnicodeString("ab'cD \\uFB00i\\u0131I\\u0130 \\u01C7\\u01C8\\u01C9 ") + DESERET_dee + DESERET_DEE,
3361 CharsToUnicodeString("ab'cd \\uFB00i\\u0131ii\\u0307 \\u01C9\\u01C9\\u01C9 ") + DESERET_dee + DESERET_dee,
3362
3363 "Upper", CharsToUnicodeString("ab'cD \\uFB00i\\u0131I\\u0130 \\u01C7\\u01C8\\u01C9 ") + DESERET_dee + DESERET_DEE, "",
3364 "Lower", CharsToUnicodeString("ab'cD \\uFB00i\\u0131I\\u0130 \\u01C7\\u01C8\\u01C9 ") + DESERET_dee + DESERET_DEE, "",
3365
3366 // FORMS OF S
3367 "Greek-Latin/UNGEGN", CharsToUnicodeString("\\u03C3 \\u03C3\\u03C2 \\u03C2\\u03C3"),
3368 CharsToUnicodeString("s ss s\\u0331s\\u0331") ,
3369 "Latin-Greek/UNGEGN", CharsToUnicodeString("s ss s\\u0331s\\u0331"),
3370 CharsToUnicodeString("\\u03C3 \\u03C3\\u03C2 \\u03C2\\u03C3") ,
3371 "Greek-Latin", CharsToUnicodeString("\\u03C3 \\u03C3\\u03C2 \\u03C2\\u03C3"),
3372 CharsToUnicodeString("s ss s\\u0331s\\u0331") ,
3373 "Latin-Greek", CharsToUnicodeString("s ss s\\u0331s\\u0331"),
3374 CharsToUnicodeString("\\u03C3 \\u03C3\\u03C2 \\u03C2\\u03C3"),
3375 // Tatiana bug
3376 // Upper: TAT\\u02B9\\u00C2NA
3377 // Lower: tat\\u02B9\\u00E2na
3378 // Title: Tat\\u02B9\\u00E2na
3379 "Upper", CharsToUnicodeString("tat\\u02B9\\u00E2na"),
3380 CharsToUnicodeString("TAT\\u02B9\\u00C2NA"),
3381 "Lower", CharsToUnicodeString("TAT\\u02B9\\u00C2NA"),
3382 CharsToUnicodeString("tat\\u02B9\\u00E2na"),
3383 "Title", CharsToUnicodeString("tat\\u02B9\\u00E2na"),
3384 CharsToUnicodeString("Tat\\u02B9\\u00E2na"),
3385
3386 "" // END MARKER
3387 };
3388
3389 UParseError pos;
3390 int32_t i;
3391 for (i = 0; registerRules[i].length()!=0; i+=2) {
3392 UErrorCode status = U_ZERO_ERROR;
3393
3394 Transliterator *t = Transliterator::createFromRules(registerRules[0+i],
3395 registerRules[i+1], UTRANS_FORWARD, pos, status);
3396 if (U_FAILURE(status)) {
3397 dataerrln("Fails: Unable to create the transliterator from rules. - %s", u_errorName(status));
3398 } else {
3399 Transliterator::registerInstance(t);
3400 }
3401 }
3402 for (i = 0; testCases[i].length()!=0; i+=3) {
3403 UErrorCode ec = U_ZERO_ERROR;
3404 UParseError pe;
3405 const UnicodeString& name = testCases[i];
3406 Transliterator *t = Transliterator::createInstance(name, UTRANS_FORWARD, pe, ec);
3407 if (U_FAILURE(ec)) {
3408 dataerrln((UnicodeString)"FAIL: Couldn't create " + name + " - " + u_errorName(ec));
3409 delete t;
3410 continue;
3411 }
3412 const UnicodeString& id = t->getID();
3413 const UnicodeString& source = testCases[i+1];
3414 UnicodeString target;
3415
3416 // Automatic generation of targets, to make it simpler to add test cases (and more fail-safe)
3417
3418 if (testCases[i+2].length() > 0) {
3419 target = testCases[i+2];
3420 } else if (0==id.caseCompare("NFD", U_FOLD_CASE_DEFAULT)) {
3421 Normalizer::normalize(source, UNORM_NFD, 0, target, ec);
3422 } else if (0==id.caseCompare("NFC", U_FOLD_CASE_DEFAULT)) {
3423 Normalizer::normalize(source, UNORM_NFC, 0, target, ec);
3424 } else if (0==id.caseCompare("NFKD", U_FOLD_CASE_DEFAULT)) {
3425 Normalizer::normalize(source, UNORM_NFKD, 0, target, ec);
3426 } else if (0==id.caseCompare("NFKC", U_FOLD_CASE_DEFAULT)) {
3427 Normalizer::normalize(source, UNORM_NFKC, 0, target, ec);
3428 } else if (0==id.caseCompare("Lower", U_FOLD_CASE_DEFAULT)) {
3429 target = source;
3430 target.toLower(Locale::getUS());
3431 } else if (0==id.caseCompare("Upper", U_FOLD_CASE_DEFAULT)) {
3432 target = source;
3433 target.toUpper(Locale::getUS());
3434 }
3435 if (U_FAILURE(ec)) {
3436 errln((UnicodeString)"FAIL: Internal error normalizing " + source);
3437 continue;
3438 }
3439
3440 expect(*t, source, target);
3441 delete t;
3442 }
3443 for (i = 0; registerRules[i].length()!=0; i+=2) {
3444 Transliterator::unregister(registerRules[i]);
3445 }
3446 }
3447
Char32ToEscapedChars(UChar32 ch,char * buffer)3448 char* Char32ToEscapedChars(UChar32 ch, char* buffer) {
3449 if (ch <= 0xFFFF) {
3450 sprintf(buffer, "\\u%04x", (int)ch);
3451 } else {
3452 sprintf(buffer, "\\U%08x", (int)ch);
3453 }
3454 return buffer;
3455 }
3456
TestSurrogateCasing(void)3457 void TransliteratorTest::TestSurrogateCasing (void) {
3458 // check that casing handles surrogates
3459 // titlecase is currently defective
3460 char buffer[20];
3461 UChar buffer2[20];
3462 UChar32 dee;
3463 U16_GET(DESERET_dee,0, 0, DESERET_dee.length(), dee);
3464 UnicodeString DEE(u_totitle(dee));
3465 if (DEE != DESERET_DEE) {
3466 err("Fails titlecase of surrogates");
3467 err(Char32ToEscapedChars(dee, buffer));
3468 err(", ");
3469 errln(Char32ToEscapedChars(DEE.char32At(0), buffer));
3470 }
3471
3472 UnicodeString deeDEETest=DESERET_dee + DESERET_DEE;
3473 UnicodeString deedeeTest = DESERET_dee + DESERET_dee;
3474 UnicodeString DEEDEETest = DESERET_DEE + DESERET_DEE;
3475 UErrorCode status= U_ZERO_ERROR;
3476
3477 u_strToUpper(buffer2, 20, deeDEETest.getBuffer(), deeDEETest.length(), NULL, &status);
3478 if (U_FAILURE(status) || (UnicodeString(buffer2)!= DEEDEETest)) {
3479 errln("Fails: Can't uppercase surrogates.");
3480 }
3481
3482 status= U_ZERO_ERROR;
3483 u_strToLower(buffer2, 20, deeDEETest.getBuffer(), deeDEETest.length(), NULL, &status);
3484 if (U_FAILURE(status) || (UnicodeString(buffer2)!= deedeeTest)) {
3485 errln("Fails: Can't lowercase surrogates.");
3486 }
3487 }
3488
_trans(Transliterator & t,const UnicodeString & src,UnicodeString & result)3489 static void _trans(Transliterator& t, const UnicodeString& src,
3490 UnicodeString& result) {
3491 result = src;
3492 t.transliterate(result);
3493 }
3494
_trans(const UnicodeString & id,const UnicodeString & src,UnicodeString & result,UErrorCode ec)3495 static void _trans(const UnicodeString& id, const UnicodeString& src,
3496 UnicodeString& result, UErrorCode ec) {
3497 UParseError pe;
3498 Transliterator *t = Transliterator::createInstance(id, UTRANS_FORWARD, pe, ec);
3499 if (U_SUCCESS(ec)) {
3500 _trans(*t, src, result);
3501 }
3502 delete t;
3503 }
3504
_findMatch(const UnicodeString & source,const UnicodeString * pairs)3505 static UnicodeString _findMatch(const UnicodeString& source,
3506 const UnicodeString* pairs) {
3507 UnicodeString empty;
3508 for (int32_t i=0; pairs[i].length() > 0; i+=2) {
3509 if (0==source.caseCompare(pairs[i], U_FOLD_CASE_DEFAULT)) {
3510 return pairs[i+1];
3511 }
3512 }
3513 return empty;
3514 }
3515
3516 // Check to see that incremental gets at least part way through a reasonable string.
3517
TestIncrementalProgress(void)3518 void TransliteratorTest::TestIncrementalProgress(void) {
3519 UErrorCode ec = U_ZERO_ERROR;
3520 UnicodeString latinTest = "The Quick Brown Fox.";
3521 UnicodeString devaTest;
3522 _trans("Latin-Devanagari", latinTest, devaTest, ec);
3523 UnicodeString kataTest;
3524 _trans("Latin-Katakana", latinTest, kataTest, ec);
3525 if (U_FAILURE(ec)) {
3526 errln("FAIL: Internal error");
3527 return;
3528 }
3529 const UnicodeString tests[] = {
3530 "Any", latinTest,
3531 "Latin", latinTest,
3532 "Halfwidth", latinTest,
3533 "Devanagari", devaTest,
3534 "Katakana", kataTest,
3535 "" // END MARKER
3536 };
3537
3538 UnicodeString test("The Quick Brown Fox Jumped Over The Lazy Dog.");
3539 int32_t i = 0, j=0, k=0;
3540 int32_t sources = Transliterator::countAvailableSources();
3541 for (i = 0; i < sources; i++) {
3542 UnicodeString source;
3543 Transliterator::getAvailableSource(i, source);
3544 UnicodeString test = _findMatch(source, tests);
3545 if (test.length() == 0) {
3546 logln((UnicodeString)"Skipping " + source + "-X");
3547 continue;
3548 }
3549 int32_t targets = Transliterator::countAvailableTargets(source);
3550 for (j = 0; j < targets; j++) {
3551 UnicodeString target;
3552 Transliterator::getAvailableTarget(j, source, target);
3553 int32_t variants = Transliterator::countAvailableVariants(source, target);
3554 for (k =0; k< variants; k++) {
3555 UnicodeString variant;
3556 UParseError err;
3557 UErrorCode status = U_ZERO_ERROR;
3558
3559 Transliterator::getAvailableVariant(k, source, target, variant);
3560 UnicodeString id = source + "-" + target + "/" + variant;
3561
3562 Transliterator *t = Transliterator::createInstance(id, UTRANS_FORWARD, err, status);
3563 if (U_FAILURE(status)) {
3564 dataerrln((UnicodeString)"FAIL: Could not create " + id);
3565 delete t;
3566 continue;
3567 }
3568 status = U_ZERO_ERROR;
3569 CheckIncrementalAux(t, test);
3570
3571 UnicodeString rev;
3572 _trans(*t, test, rev);
3573 Transliterator *inv = t->createInverse(status);
3574 if (U_FAILURE(status)) {
3575 #if UCONFIG_NO_BREAK_ITERATION
3576 // If UCONFIG_NO_BREAK_ITERATION is on, then only Thai should fail.
3577 if (id.compare((UnicodeString)"Latin-Thai/") != 0)
3578 #endif
3579 errln((UnicodeString)"FAIL: Could not create inverse of " + id);
3580
3581 delete t;
3582 delete inv;
3583 continue;
3584 }
3585 CheckIncrementalAux(inv, rev);
3586 delete t;
3587 delete inv;
3588 }
3589 }
3590 }
3591 }
3592
CheckIncrementalAux(const Transliterator * t,const UnicodeString & input)3593 void TransliteratorTest::CheckIncrementalAux(const Transliterator* t,
3594 const UnicodeString& input) {
3595 UErrorCode ec = U_ZERO_ERROR;
3596 UTransPosition pos;
3597 UnicodeString test = input;
3598
3599 pos.contextStart = 0;
3600 pos.contextLimit = input.length();
3601 pos.start = 0;
3602 pos.limit = input.length();
3603
3604 t->transliterate(test, pos, ec);
3605 if (U_FAILURE(ec)) {
3606 errln((UnicodeString)"FAIL: transliterate() error " + u_errorName(ec));
3607 return;
3608 }
3609 UBool gotError = FALSE;
3610 (void)gotError; // Suppress set but not used warning.
3611
3612 // we have a few special cases. Any-Remove (pos.start = 0, but also = limit) and U+XXXXX?X?
3613
3614 if (pos.start == 0 && pos.limit != 0 && t->getID() != "Hex-Any/Unicode") {
3615 errln((UnicodeString)"No Progress, " +
3616 t->getID() + ": " + formatInput(test, input, pos));
3617 gotError = TRUE;
3618 } else {
3619 logln((UnicodeString)"PASS Progress, " +
3620 t->getID() + ": " + formatInput(test, input, pos));
3621 }
3622 t->finishTransliteration(test, pos);
3623 if (pos.start != pos.limit) {
3624 errln((UnicodeString)"Incomplete, " +
3625 t->getID() + ": " + formatInput(test, input, pos));
3626 gotError = TRUE;
3627 }
3628 }
3629
TestFunction()3630 void TransliteratorTest::TestFunction() {
3631 // Careful with spacing and ';' here: Phrase this exactly
3632 // as toRules() is going to return it. If toRules() changes
3633 // with regard to spacing or ';', then adjust this string.
3634 UnicodeString rule =
3635 "([:Lu:]) > $1 '(' &Lower( $1 ) '=' &Hex( &Any-Lower( $1 ) ) ')';";
3636
3637 UParseError pe;
3638 UErrorCode ec = U_ZERO_ERROR;
3639 Transliterator *t = Transliterator::createFromRules("Test", rule, UTRANS_FORWARD, pe, ec);
3640 if (t == NULL) {
3641 dataerrln("FAIL: createFromRules failed - %s", u_errorName(ec));
3642 return;
3643 }
3644
3645 UnicodeString r;
3646 t->toRules(r, TRUE);
3647 if (r == rule) {
3648 logln((UnicodeString)"OK: toRules() => " + r);
3649 } else {
3650 errln((UnicodeString)"FAIL: toRules() => " + r +
3651 ", expected " + rule);
3652 }
3653
3654 expect(*t, "The Quick Brown Fox",
3655 UNICODE_STRING_SIMPLE("T(t=\\u0074)he Q(q=\\u0071)uick B(b=\\u0062)rown F(f=\\u0066)ox"));
3656
3657 delete t;
3658 }
3659
TestInvalidBackRef(void)3660 void TransliteratorTest::TestInvalidBackRef(void) {
3661 UnicodeString rule = ". > $1;";
3662 UnicodeString rule2 =CharsToUnicodeString("(.) <> &hex/unicode($1) &name($1); . > $1; [{}] >\\u0020;");
3663 UParseError pe;
3664 UErrorCode ec = U_ZERO_ERROR;
3665 Transliterator *t = Transliterator::createFromRules("Test", rule, UTRANS_FORWARD, pe, ec);
3666 Transliterator *t2 = Transliterator::createFromRules("Test2", rule2, UTRANS_FORWARD, pe, ec);
3667
3668 if (t != NULL) {
3669 errln("FAIL: createFromRules should have returned NULL");
3670 delete t;
3671 }
3672
3673 if (t2 != NULL) {
3674 errln("FAIL: createFromRules should have returned NULL");
3675 delete t2;
3676 }
3677
3678 if (U_SUCCESS(ec)) {
3679 errln("FAIL: Ok: . > $1; => no error");
3680 } else {
3681 logln((UnicodeString)"Ok: . > $1; => " + u_errorName(ec));
3682 }
3683 }
3684
TestMulticharStringSet()3685 void TransliteratorTest::TestMulticharStringSet() {
3686 // Basic testing
3687 const char* rule =
3688 " [{aa}] > x;"
3689 " a > y;"
3690 " [b{bc}] > z;"
3691 "[{gd}] { e > q;"
3692 " e } [{fg}] > r;" ;
3693
3694 UParseError pe;
3695 UErrorCode ec = U_ZERO_ERROR;
3696 Transliterator* t = Transliterator::createFromRules("Test", rule, UTRANS_FORWARD, pe, ec);
3697 if (t == NULL || U_FAILURE(ec)) {
3698 delete t;
3699 errln("FAIL: createFromRules failed");
3700 return;
3701 }
3702
3703 expect(*t, "a aa ab bc d gd de gde gdefg ddefg",
3704 "y x yz z d gd de gdq gdqfg ddrfg");
3705 delete t;
3706
3707 // Overlapped string test. Make sure that when multiple
3708 // strings can match that the longest one is matched.
3709 rule =
3710 " [a {ab} {abc}] > x;"
3711 " b > y;"
3712 " c > z;"
3713 " q [t {st} {rst}] { e > p;" ;
3714
3715 t = Transliterator::createFromRules("Test", rule, UTRANS_FORWARD, pe, ec);
3716 if (t == NULL || U_FAILURE(ec)) {
3717 delete t;
3718 errln("FAIL: createFromRules failed");
3719 return;
3720 }
3721
3722 expect(*t, "a ab abc qte qste qrste",
3723 "x x x qtp qstp qrstp");
3724 delete t;
3725 }
3726
3727 // vvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvv
3728 // BEGIN TestUserFunction support factory
3729
3730 Transliterator* _TUFF[4];
3731 UnicodeString* _TUFID[4];
3732
_TUFFactory(const UnicodeString &,Transliterator::Token context)3733 static Transliterator* U_EXPORT2 _TUFFactory(const UnicodeString& /*ID*/,
3734 Transliterator::Token context) {
3735 return _TUFF[context.integer]->clone();
3736 }
3737
_TUFReg(const UnicodeString & ID,Transliterator * t,int32_t n)3738 static void _TUFReg(const UnicodeString& ID, Transliterator* t, int32_t n) {
3739 _TUFF[n] = t;
3740 _TUFID[n] = new UnicodeString(ID);
3741 Transliterator::registerFactory(ID, _TUFFactory, Transliterator::integerToken(n));
3742 }
3743
_TUFUnreg(int32_t n)3744 static void _TUFUnreg(int32_t n) {
3745 if (_TUFF[n] != NULL) {
3746 Transliterator::unregister(*_TUFID[n]);
3747 delete _TUFF[n];
3748 delete _TUFID[n];
3749 }
3750 }
3751
3752 // END TestUserFunction support factory
3753 // ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
3754
3755 /**
3756 * Test that user-registered transliterators can be used under function
3757 * syntax.
3758 */
TestUserFunction()3759 void TransliteratorTest::TestUserFunction() {
3760
3761 Transliterator* t;
3762 UParseError pe;
3763 UErrorCode ec = U_ZERO_ERROR;
3764
3765 // Setup our factory
3766 int32_t i;
3767 for (i=0; i<4; ++i) {
3768 _TUFF[i] = NULL;
3769 }
3770
3771 // There's no need to register inverses if we don't use them
3772 t = Transliterator::createFromRules("gif",
3773 UNICODE_STRING_SIMPLE("'\\'u(..)(..) > '<img src=\"http://www.unicode.org/gifs/24/' $1 '/U' $1$2 '.gif\">';"),
3774 UTRANS_FORWARD, pe, ec);
3775 if (t == NULL || U_FAILURE(ec)) {
3776 dataerrln((UnicodeString)"FAIL: createFromRules gif " + u_errorName(ec));
3777 return;
3778 }
3779 _TUFReg("Any-gif", t, 0);
3780
3781 t = Transliterator::createFromRules("RemoveCurly",
3782 UNICODE_STRING_SIMPLE("[\\{\\}] > ; '\\N' > ;"),
3783 UTRANS_FORWARD, pe, ec);
3784 if (t == NULL || U_FAILURE(ec)) {
3785 errln((UnicodeString)"FAIL: createFromRules RemoveCurly " + u_errorName(ec));
3786 goto FAIL;
3787 }
3788 expect(*t, UNICODE_STRING_SIMPLE("\\N{name}"), "name");
3789 _TUFReg("Any-RemoveCurly", t, 1);
3790
3791 logln("Trying &hex");
3792 t = Transliterator::createFromRules("hex2",
3793 "(.) > &hex($1);",
3794 UTRANS_FORWARD, pe, ec);
3795 if (t == NULL || U_FAILURE(ec)) {
3796 errln("FAIL: createFromRules");
3797 goto FAIL;
3798 }
3799 logln("Registering");
3800 _TUFReg("Any-hex2", t, 2);
3801 t = Transliterator::createInstance("Any-hex2", UTRANS_FORWARD, ec);
3802 if (t == NULL || U_FAILURE(ec)) {
3803 errln((UnicodeString)"FAIL: createInstance Any-hex2 " + u_errorName(ec));
3804 goto FAIL;
3805 }
3806 expect(*t, "abc", UNICODE_STRING_SIMPLE("\\u0061\\u0062\\u0063"));
3807 delete t;
3808
3809 logln("Trying &gif");
3810 t = Transliterator::createFromRules("gif2",
3811 "(.) > &Gif(&Hex2($1));",
3812 UTRANS_FORWARD, pe, ec);
3813 if (t == NULL || U_FAILURE(ec)) {
3814 errln((UnicodeString)"FAIL: createFromRules gif2 " + u_errorName(ec));
3815 goto FAIL;
3816 }
3817 logln("Registering");
3818 _TUFReg("Any-gif2", t, 3);
3819 t = Transliterator::createInstance("Any-gif2", UTRANS_FORWARD, ec);
3820 if (t == NULL || U_FAILURE(ec)) {
3821 errln((UnicodeString)"FAIL: createInstance Any-gif2 " + u_errorName(ec));
3822 goto FAIL;
3823 }
3824 expect(*t, "ab", "<img src=\"http://www.unicode.org/gifs/24/00/U0061.gif\">"
3825 "<img src=\"http://www.unicode.org/gifs/24/00/U0062.gif\">");
3826 delete t;
3827
3828 // Test that filters are allowed after &
3829 t = Transliterator::createFromRules("test",
3830 "(.) > &Hex($1) ' ' &RemoveCurly(&Name($1)) ' ';",
3831 UTRANS_FORWARD, pe, ec);
3832 if (t == NULL || U_FAILURE(ec)) {
3833 errln((UnicodeString)"FAIL: createFromRules test " + u_errorName(ec));
3834 goto FAIL;
3835 }
3836 expect(*t, "abc",
3837 UNICODE_STRING_SIMPLE("\\u0061 LATIN SMALL LETTER A \\u0062 LATIN SMALL LETTER B \\u0063 LATIN SMALL LETTER C "));
3838 delete t;
3839
3840 FAIL:
3841 for (i=0; i<4; ++i) {
3842 _TUFUnreg(i);
3843 }
3844 }
3845
3846 /**
3847 * Test the Any-X transliterators.
3848 */
TestAnyX(void)3849 void TransliteratorTest::TestAnyX(void) {
3850 UParseError parseError;
3851 UErrorCode status = U_ZERO_ERROR;
3852 Transliterator* anyLatin =
3853 Transliterator::createInstance("Any-Latin", UTRANS_FORWARD, parseError, status);
3854 if (anyLatin==0) {
3855 dataerrln("FAIL: createInstance returned NULL - %s", u_errorName(status));
3856 delete anyLatin;
3857 return;
3858 }
3859
3860 expect(*anyLatin,
3861 CharsToUnicodeString("greek:\\u03B1\\u03B2\\u03BA\\u0391\\u0392\\u039A hiragana:\\u3042\\u3076\\u304F cyrillic:\\u0430\\u0431\\u0446"),
3862 CharsToUnicodeString("greek:abkABK hiragana:abuku cyrillic:abc"));
3863
3864 delete anyLatin;
3865 }
3866
3867 /**
3868 * Test Any-X transliterators with sample letters from all scripts.
3869 */
TestAny(void)3870 void TransliteratorTest::TestAny(void) {
3871 UErrorCode status = U_ZERO_ERROR;
3872 // Note: there is a lot of implict construction of UnicodeStrings from (char *) in
3873 // function call parameters going on in this test.
3874 UnicodeSet alphabetic("[:alphabetic:]", status);
3875 if (U_FAILURE(status)) {
3876 dataerrln("Failure: file %s, line %d, status = %s", __FILE__, __LINE__, u_errorName(status));
3877 return;
3878 }
3879 alphabetic.freeze();
3880
3881 UnicodeString testString;
3882 for (int32_t i = 0; i < USCRIPT_CODE_LIMIT; i++) {
3883 const char *scriptName = uscript_getShortName((UScriptCode)i);
3884 if (scriptName == NULL) {
3885 errln("Failure: file %s, line %d: Script Code %d is invalid, ", __FILE__, __LINE__, i);
3886 return;
3887 }
3888
3889 UnicodeSet sample;
3890 sample.applyPropertyAlias("script", scriptName, status);
3891 if (U_FAILURE(status)) {
3892 errln("Failure: file %s, line %d, status = %s", __FILE__, __LINE__, u_errorName(status));
3893 return;
3894 }
3895 sample.retainAll(alphabetic);
3896 for (int32_t count=0; count<5; count++) {
3897 UChar32 c = sample.charAt(count);
3898 if (c == -1) {
3899 break;
3900 }
3901 testString.append(c);
3902 }
3903 }
3904
3905 UParseError parseError;
3906 Transliterator* anyLatin =
3907 Transliterator::createInstance("Any-Latin", UTRANS_FORWARD, parseError, status);
3908 if (U_FAILURE(status)) {
3909 dataerrln("Failure: file %s, line %d, status = %s", __FILE__, __LINE__, u_errorName(status));
3910 return;
3911 }
3912
3913 logln(UnicodeString("Sample set for Any-Latin: ") + testString);
3914 anyLatin->transliterate(testString);
3915 logln(UnicodeString("Sample result for Any-Latin: ") + testString);
3916 delete anyLatin;
3917 }
3918
3919
3920 /**
3921 * Test the source and target set API. These are only implemented
3922 * for RBT and CompoundTransliterator at this time.
3923 */
TestSourceTargetSet()3924 void TransliteratorTest::TestSourceTargetSet() {
3925 UErrorCode ec = U_ZERO_ERROR;
3926
3927 // Rules
3928 const char* r =
3929 "a > b; "
3930 "r [x{lu}] > q;";
3931
3932 // Expected source
3933 UnicodeSet expSrc("[arx{lu}]", ec);
3934
3935 // Expected target
3936 UnicodeSet expTrg("[bq]", ec);
3937
3938 UParseError pe;
3939 Transliterator* t = Transliterator::createFromRules("test", r, UTRANS_FORWARD, pe, ec);
3940
3941 if (U_FAILURE(ec)) {
3942 delete t;
3943 errln("FAIL: Couldn't set up test");
3944 return;
3945 }
3946
3947 UnicodeSet src; t->getSourceSet(src);
3948 UnicodeSet trg; t->getTargetSet(trg);
3949
3950 if (src == expSrc && trg == expTrg) {
3951 UnicodeString a, b;
3952 logln((UnicodeString)"Ok: " +
3953 r + " => source = " + src.toPattern(a, TRUE) +
3954 ", target = " + trg.toPattern(b, TRUE));
3955 } else {
3956 UnicodeString a, b, c, d;
3957 errln((UnicodeString)"FAIL: " +
3958 r + " => source = " + src.toPattern(a, TRUE) +
3959 ", expected " + expSrc.toPattern(b, TRUE) +
3960 "; target = " + trg.toPattern(c, TRUE) +
3961 ", expected " + expTrg.toPattern(d, TRUE));
3962 }
3963
3964 delete t;
3965 }
3966
3967 /**
3968 * Test handling of Pattern_White_Space, for both RBT and UnicodeSet.
3969 */
TestPatternWhiteSpace()3970 void TransliteratorTest::TestPatternWhiteSpace() {
3971 // Rules
3972 const char* r = "a > \\u200E b;";
3973
3974 UErrorCode ec = U_ZERO_ERROR;
3975 UParseError pe;
3976 Transliterator* t = Transliterator::createFromRules("test", CharsToUnicodeString(r), UTRANS_FORWARD, pe, ec);
3977
3978 if (U_FAILURE(ec)) {
3979 errln("FAIL: Couldn't set up test");
3980 } else {
3981 expect(*t, "a", "b");
3982 }
3983 delete t;
3984
3985 // UnicodeSet
3986 ec = U_ZERO_ERROR;
3987 UnicodeSet set(CharsToUnicodeString("[a \\u200E]"), ec);
3988
3989 if (U_FAILURE(ec)) {
3990 errln("FAIL: Couldn't set up test");
3991 } else {
3992 if (set.contains(0x200E)) {
3993 errln("FAIL: U+200E not being ignored by UnicodeSet");
3994 }
3995 }
3996 }
3997 //======================================================================
3998 // this method is in TestUScript.java
3999 //======================================================================
TestAllCodepoints()4000 void TransliteratorTest::TestAllCodepoints(){
4001 UScriptCode code= USCRIPT_INVALID_CODE;
4002 char id[256]={'\0'};
4003 char abbr[256]={'\0'};
4004 char newId[256]={'\0'};
4005 char newAbbrId[256]={'\0'};
4006 char oldId[256]={'\0'};
4007 char oldAbbrId[256]={'\0'};
4008
4009 UErrorCode status =U_ZERO_ERROR;
4010 UParseError pe;
4011
4012 for(uint32_t i = 0; i<=0x10ffff; i++){
4013 code = uscript_getScript(i,&status);
4014 if(code == USCRIPT_INVALID_CODE){
4015 dataerrln("uscript_getScript for codepoint \\U%08X failed.", i);
4016 }
4017 const char* myId = uscript_getName(code);
4018 if(!myId) {
4019 dataerrln("Valid script code returned NULL name. Check your data!");
4020 return;
4021 }
4022 uprv_strcpy(id,myId);
4023 uprv_strcpy(abbr,uscript_getShortName(code));
4024
4025 uprv_strcpy(newId,"[:");
4026 uprv_strcat(newId,id);
4027 uprv_strcat(newId,":];NFD");
4028
4029 uprv_strcpy(newAbbrId,"[:");
4030 uprv_strcat(newAbbrId,abbr);
4031 uprv_strcat(newAbbrId,":];NFD");
4032
4033 if(uprv_strcmp(newId,oldId)!=0){
4034 Transliterator* t = Transliterator::createInstance(newId,UTRANS_FORWARD,pe,status);
4035 if(t==NULL || U_FAILURE(status)){
4036 dataerrln((UnicodeString)"FAIL: Could not create " + id + " - " + u_errorName(status));
4037 }
4038 delete t;
4039 }
4040 if(uprv_strcmp(newAbbrId,oldAbbrId)!=0){
4041 Transliterator* t = Transliterator::createInstance(newAbbrId,UTRANS_FORWARD,pe,status);
4042 if(t==NULL || U_FAILURE(status)){
4043 dataerrln((UnicodeString)"FAIL: Could not create " + id + " - " + u_errorName(status));
4044 }
4045 delete t;
4046 }
4047 uprv_strcpy(oldId,newId);
4048 uprv_strcpy(oldAbbrId, newAbbrId);
4049
4050 }
4051
4052 }
4053
4054 #define TEST_TRANSLIT_ID(id, cls) { \
4055 UErrorCode ec = U_ZERO_ERROR; \
4056 Transliterator* t = Transliterator::createInstance(id, UTRANS_FORWARD, ec); \
4057 if (U_FAILURE(ec)) { \
4058 dataerrln("FAIL: Couldn't create %s - %s", id, u_errorName(ec)); \
4059 } else { \
4060 if (t->getDynamicClassID() != cls::getStaticClassID()) { \
4061 errln("FAIL: " #cls " dynamic and static class ID mismatch"); \
4062 } \
4063 /* *t = *t; */ /*can't do this: coverage test for assignment op*/ \
4064 } \
4065 delete t; \
4066 }
4067
4068 #define TEST_TRANSLIT_RULE(rule, cls) { \
4069 UErrorCode ec = U_ZERO_ERROR; \
4070 UParseError pe; \
4071 Transliterator* t = Transliterator::createFromRules("_", rule, UTRANS_FORWARD, pe, ec); \
4072 if (U_FAILURE(ec)) { \
4073 errln("FAIL: Couldn't create " rule); \
4074 } else { \
4075 if (t->getDynamicClassID() != cls ::getStaticClassID()) { \
4076 errln("FAIL: " #cls " dynamic and static class ID mismatch"); \
4077 } \
4078 /* *t = *t; */ /*can't do this: coverage test for assignment op*/ \
4079 } \
4080 delete t; \
4081 }
4082
TestBoilerplate()4083 void TransliteratorTest::TestBoilerplate() {
4084 TEST_TRANSLIT_ID("Any-Latin", AnyTransliterator);
4085 TEST_TRANSLIT_ID("Any-Hex", EscapeTransliterator);
4086 TEST_TRANSLIT_ID("Hex-Any", UnescapeTransliterator);
4087 TEST_TRANSLIT_ID("Lower", LowercaseTransliterator);
4088 TEST_TRANSLIT_ID("Upper", UppercaseTransliterator);
4089 TEST_TRANSLIT_ID("Title", TitlecaseTransliterator);
4090 TEST_TRANSLIT_ID("Null", NullTransliterator);
4091 TEST_TRANSLIT_ID("Remove", RemoveTransliterator);
4092 TEST_TRANSLIT_ID("Any-Name", UnicodeNameTransliterator);
4093 TEST_TRANSLIT_ID("Name-Any", NameUnicodeTransliterator);
4094 TEST_TRANSLIT_ID("NFD", NormalizationTransliterator);
4095 TEST_TRANSLIT_ID("Latin-Greek", CompoundTransliterator);
4096 TEST_TRANSLIT_RULE("a>b;", RuleBasedTransliterator);
4097 }
4098
TestAlternateSyntax()4099 void TransliteratorTest::TestAlternateSyntax() {
4100 // U+2206 == &
4101 // U+2190 == <
4102 // U+2192 == >
4103 // U+2194 == <>
4104 expect(CharsToUnicodeString("a \\u2192 x; b \\u2190 y; c \\u2194 z"),
4105 "abc",
4106 "xbz");
4107 expect(CharsToUnicodeString("([:^ASCII:]) \\u2192 \\u2206Name($1);"),
4108 CharsToUnicodeString("<=\\u2190; >=\\u2192; <>=\\u2194; &=\\u2206"),
4109 UNICODE_STRING_SIMPLE("<=\\N{LEFTWARDS ARROW}; >=\\N{RIGHTWARDS ARROW}; <>=\\N{LEFT RIGHT ARROW}; &=\\N{INCREMENT}"));
4110 }
4111
4112 static const char* BEGIN_END_RULES[] = {
4113 // [0]
4114 "abc > xy;"
4115 "aba > z;",
4116
4117 // [1]
4118 /*
4119 "::BEGIN;"
4120 "abc > xy;"
4121 "::END;"
4122 "::BEGIN;"
4123 "aba > z;"
4124 "::END;",
4125 */
4126 "", // test case commented out below, this is here to keep from messing up the indexes
4127
4128 // [2]
4129 /*
4130 "abc > xy;"
4131 "::BEGIN;"
4132 "aba > z;"
4133 "::END;",
4134 */
4135 "", // test case commented out below, this is here to keep from messing up the indexes
4136
4137 // [3]
4138 /*
4139 "::BEGIN;"
4140 "abc > xy;"
4141 "::END;"
4142 "aba > z;",
4143 */
4144 "", // test case commented out below, this is here to keep from messing up the indexes
4145
4146 // [4]
4147 "abc > xy;"
4148 "::Null;"
4149 "aba > z;",
4150
4151 // [5]
4152 "::Upper;"
4153 "ABC > xy;"
4154 "AB > x;"
4155 "C > z;"
4156 "::Upper;"
4157 "XYZ > p;"
4158 "XY > q;"
4159 "Z > r;"
4160 "::Upper;",
4161
4162 // [6]
4163 "$ws = [[:Separator:][\\u0009-\\u000C]$];"
4164 "$delim = [\\-$ws];"
4165 "$ws $delim* > ' ';"
4166 "'-' $delim* > '-';",
4167
4168 // [7]
4169 "::Null;"
4170 "$ws = [[:Separator:][\\u0009-\\u000C]$];"
4171 "$delim = [\\-$ws];"
4172 "$ws $delim* > ' ';"
4173 "'-' $delim* > '-';",
4174
4175 // [8]
4176 "$ws = [[:Separator:][\\u0009-\\u000C]$];"
4177 "$delim = [\\-$ws];"
4178 "$ws $delim* > ' ';"
4179 "'-' $delim* > '-';"
4180 "::Null;",
4181
4182 // [9]
4183 "$ws = [[:Separator:][\\u0009-\\u000C]$];"
4184 "$delim = [\\-$ws];"
4185 "::Null;"
4186 "$ws $delim* > ' ';"
4187 "'-' $delim* > '-';",
4188
4189 // [10]
4190 /*
4191 "::BEGIN;"
4192 "$ws = [[:Separator:][\\u0009-\\u000C]$];"
4193 "$delim = [\\-$ws];"
4194 "::END;"
4195 "$ws $delim* > ' ';"
4196 "'-' $delim* > '-';",
4197 */
4198 "", // test case commented out below, this is here to keep from messing up the indexes
4199
4200 // [11]
4201 /*
4202 "$ws = [[:Separator:][\\u0009-\\u000C]$];"
4203 "$delim = [\\-$ws];"
4204 "::BEGIN;"
4205 "$ws $delim* > ' ';"
4206 "'-' $delim* > '-';"
4207 "::END;",
4208 */
4209 "", // test case commented out below, this is here to keep from messing up the indexes
4210
4211 // [12]
4212 /*
4213 "$ws = [[:Separator:][\\u0009-\\u000C]$];"
4214 "$delim = [\\-$ws];"
4215 "$ab = [ab];"
4216 "::BEGIN;"
4217 "$ws $delim* > ' ';"
4218 "'-' $delim* > '-';"
4219 "::END;"
4220 "::BEGIN;"
4221 "$ab { ' ' } $ab > '-';"
4222 "c { ' ' > ;"
4223 "::END;"
4224 "::BEGIN;"
4225 "'a-a' > a\\%|a;"
4226 "::END;",
4227 */
4228 "", // test case commented out below, this is here to keep from messing up the indexes
4229
4230 // [13]
4231 "$ws = [[:Separator:][\\u0009-\\u000C]$];"
4232 "$delim = [\\-$ws];"
4233 "$ab = [ab];"
4234 "::Null;"
4235 "$ws $delim* > ' ';"
4236 "'-' $delim* > '-';"
4237 "::Null;"
4238 "$ab { ' ' } $ab > '-';"
4239 "c { ' ' > ;"
4240 "::Null;"
4241 "'a-a' > a\\%|a;",
4242
4243 // [14]
4244 /*
4245 "::[abc];"
4246 "::BEGIN;"
4247 "abc > xy;"
4248 "::END;"
4249 "::BEGIN;"
4250 "aba > yz;"
4251 "::END;"
4252 "::Upper;",
4253 */
4254 "", // test case commented out below, this is here to keep from messing up the indexes
4255
4256 // [15]
4257 "::[abc];"
4258 "abc > xy;"
4259 "::Null;"
4260 "aba > yz;"
4261 "::Upper;",
4262
4263 // [16]
4264 /*
4265 "::[abc];"
4266 "::BEGIN;"
4267 "abc <> xy;"
4268 "::END;"
4269 "::BEGIN;"
4270 "aba <> yz;"
4271 "::END;"
4272 "::Upper(Lower);"
4273 "::([XYZ]);"
4274 */
4275 "", // test case commented out below, this is here to keep from messing up the indexes
4276
4277 // [17]
4278 "::[abc];"
4279 "abc <> xy;"
4280 "::Null;"
4281 "aba <> yz;"
4282 "::Upper(Lower);"
4283 "::([XYZ]);"
4284 };
4285
4286 /*
4287 (This entire test is commented out below and will need some heavy revision when we re-add
4288 the ::BEGIN/::END stuff)
4289 static const char* BOGUS_BEGIN_END_RULES[] = {
4290 // [7]
4291 "::BEGIN;"
4292 "abc > xy;"
4293 "::BEGIN;"
4294 "aba > z;"
4295 "::END;"
4296 "::END;",
4297
4298 // [8]
4299 "abc > xy;"
4300 " aba > z;"
4301 "::END;",
4302
4303 // [9]
4304 "::BEGIN;"
4305 "::Upper;"
4306 "::END;"
4307 };
4308 static const int32_t BOGUS_BEGIN_END_RULES_length = (int32_t)(sizeof(BOGUS_BEGIN_END_RULES) / sizeof(BOGUS_BEGIN_END_RULES[0]));
4309 */
4310
4311 static const char* BEGIN_END_TEST_CASES[] = {
4312 // rules input expected output
4313 BEGIN_END_RULES[0], "abc ababc aba", "xy zbc z",
4314 // BEGIN_END_RULES[1], "abc ababc aba", "xy abxy z",
4315 // BEGIN_END_RULES[2], "abc ababc aba", "xy abxy z",
4316 // BEGIN_END_RULES[3], "abc ababc aba", "xy abxy z",
4317 BEGIN_END_RULES[4], "abc ababc aba", "xy abxy z",
4318 BEGIN_END_RULES[5], "abccabaacababcbc", "PXAARXQBR",
4319
4320 BEGIN_END_RULES[6], "e e - e---e- e", "e e e-e-e",
4321 BEGIN_END_RULES[7], "e e - e---e- e", "e e e-e-e",
4322 BEGIN_END_RULES[8], "e e - e---e- e", "e e e-e-e",
4323 BEGIN_END_RULES[9], "e e - e---e- e", "e e e-e-e",
4324 // BEGIN_END_RULES[10], "e e - e---e- e", "e e e-e-e",
4325 // BEGIN_END_RULES[11], "e e - e---e- e", "e e e-e-e",
4326 // BEGIN_END_RULES[12], "e e - e---e- e", "e e e-e-e",
4327 // BEGIN_END_RULES[12], "a a a a", "a%a%a%a",
4328 // BEGIN_END_RULES[12], "a a-b c b a", "a%a-b cb-a",
4329 BEGIN_END_RULES[13], "e e - e---e- e", "e e e-e-e",
4330 BEGIN_END_RULES[13], "a a a a", "a%a%a%a",
4331 BEGIN_END_RULES[13], "a a-b c b a", "a%a-b cb-a",
4332
4333 // BEGIN_END_RULES[14], "abc xy ababc xyz aba", "XY xy ABXY xyz YZ",
4334 BEGIN_END_RULES[15], "abc xy ababc xyz aba", "XY xy ABXY xyz YZ",
4335 // BEGIN_END_RULES[16], "abc xy ababc xyz aba", "XY xy ABXY xyz YZ",
4336 BEGIN_END_RULES[17], "abc xy ababc xyz aba", "XY xy ABXY xyz YZ"
4337 };
4338 static const int32_t BEGIN_END_TEST_CASES_length = (int32_t)(sizeof(BEGIN_END_TEST_CASES) / sizeof(BEGIN_END_TEST_CASES[0]));
4339
TestBeginEnd()4340 void TransliteratorTest::TestBeginEnd() {
4341 // run through the list of test cases above
4342 int32_t i = 0;
4343 for (i = 0; i < BEGIN_END_TEST_CASES_length; i += 3) {
4344 expect((UnicodeString)"Test case #" + (i / 3),
4345 UnicodeString(BEGIN_END_TEST_CASES[i], -1, US_INV),
4346 UnicodeString(BEGIN_END_TEST_CASES[i + 1], -1, US_INV),
4347 UnicodeString(BEGIN_END_TEST_CASES[i + 2], -1, US_INV));
4348 }
4349
4350 // instantiate the one reversible rule set in the reverse direction and make sure it does the right thing
4351 UParseError parseError;
4352 UErrorCode status = U_ZERO_ERROR;
4353 Transliterator* reversed = Transliterator::createFromRules("Reversed", UnicodeString(BEGIN_END_RULES[17]),
4354 UTRANS_REVERSE, parseError, status);
4355 if (reversed == 0 || U_FAILURE(status)) {
4356 reportParseError(UnicodeString("FAIL: Couldn't create reversed transliterator"), parseError, status);
4357 } else {
4358 expect(*reversed, UnicodeString("xy XY XYZ yz YZ"), UnicodeString("xy abc xaba yz aba"));
4359 }
4360 delete reversed;
4361
4362 // finally, run through the list of syntactically-ill-formed rule sets above and make sure
4363 // that all of them cause errors
4364 /*
4365 (commented out until we have the real ::BEGIN/::END stuff in place
4366 for (i = 0; i < BOGUS_BEGIN_END_RULES_length; i++) {
4367 UParseError parseError;
4368 UErrorCode status = U_ZERO_ERROR;
4369 Transliterator* t = Transliterator::createFromRules("foo", UnicodeString(BOGUS_BEGIN_END_RULES[i]),
4370 UTRANS_FORWARD, parseError, status);
4371 if (!U_FAILURE(status)) {
4372 delete t;
4373 errln((UnicodeString)"Should have gotten syntax error from " + BOGUS_BEGIN_END_RULES[i]);
4374 }
4375 }
4376 */
4377 }
4378
TestBeginEndToRules()4379 void TransliteratorTest::TestBeginEndToRules() {
4380 // run through the same list of test cases we used above, but this time, instead of just
4381 // instantiating a Transliterator from the rules and running the test against it, we instantiate
4382 // a Transliterator from the rules, do toRules() on it, instantiate a Transliterator from
4383 // the resulting set of rules, and make sure that the generated rule set is semantically equivalent
4384 // to (i.e., does the same thing as) the original rule set
4385 for (int32_t i = 0; i < BEGIN_END_TEST_CASES_length; i += 3) {
4386 UParseError parseError;
4387 UErrorCode status = U_ZERO_ERROR;
4388 Transliterator* t = Transliterator::createFromRules("--", UnicodeString(BEGIN_END_TEST_CASES[i], -1, US_INV),
4389 UTRANS_FORWARD, parseError, status);
4390 if (U_FAILURE(status)) {
4391 reportParseError(UnicodeString("FAIL: Couldn't create transliterator"), parseError, status);
4392 } else {
4393 UnicodeString rules;
4394 t->toRules(rules, TRUE);
4395 Transliterator* t2 = Transliterator::createFromRules((UnicodeString)"Test case #" + (i / 3), rules,
4396 UTRANS_FORWARD, parseError, status);
4397 if (U_FAILURE(status)) {
4398 reportParseError(UnicodeString("FAIL: Couldn't create transliterator from generated rules"),
4399 parseError, status);
4400 delete t;
4401 } else {
4402 expect(*t2,
4403 UnicodeString(BEGIN_END_TEST_CASES[i + 1], -1, US_INV),
4404 UnicodeString(BEGIN_END_TEST_CASES[i + 2], -1, US_INV));
4405 delete t;
4406 delete t2;
4407 }
4408 }
4409 }
4410
4411 // do the same thing for the reversible test case
4412 UParseError parseError;
4413 UErrorCode status = U_ZERO_ERROR;
4414 Transliterator* reversed = Transliterator::createFromRules("Reversed", UnicodeString(BEGIN_END_RULES[17]),
4415 UTRANS_REVERSE, parseError, status);
4416 if (U_FAILURE(status)) {
4417 reportParseError(UnicodeString("FAIL: Couldn't create reversed transliterator"), parseError, status);
4418 } else {
4419 UnicodeString rules;
4420 reversed->toRules(rules, FALSE);
4421 Transliterator* reversed2 = Transliterator::createFromRules("Reversed", rules, UTRANS_FORWARD,
4422 parseError, status);
4423 if (U_FAILURE(status)) {
4424 reportParseError(UnicodeString("FAIL: Couldn't create reversed transliterator from generated rules"),
4425 parseError, status);
4426 delete reversed;
4427 } else {
4428 expect(*reversed2,
4429 UnicodeString("xy XY XYZ yz YZ"),
4430 UnicodeString("xy abc xaba yz aba"));
4431 delete reversed;
4432 delete reversed2;
4433 }
4434 }
4435 }
4436
TestRegisterAlias()4437 void TransliteratorTest::TestRegisterAlias() {
4438 UnicodeString longID("Lower;[aeiou]Upper");
4439 UnicodeString shortID("Any-CapVowels");
4440 UnicodeString reallyShortID("CapVowels");
4441
4442 Transliterator::registerAlias(shortID, longID);
4443
4444 UErrorCode err = U_ZERO_ERROR;
4445 Transliterator* t1 = Transliterator::createInstance(longID, UTRANS_FORWARD, err);
4446 if (U_FAILURE(err)) {
4447 errln("Failed to instantiate transliterator with long ID");
4448 Transliterator::unregister(shortID);
4449 return;
4450 }
4451 Transliterator* t2 = Transliterator::createInstance(reallyShortID, UTRANS_FORWARD, err);
4452 if (U_FAILURE(err)) {
4453 errln("Failed to instantiate transliterator with short ID");
4454 delete t1;
4455 Transliterator::unregister(shortID);
4456 return;
4457 }
4458
4459 if (t1->getID() != longID)
4460 errln("Transliterator instantiated with long ID doesn't have long ID");
4461 if (t2->getID() != reallyShortID)
4462 errln("Transliterator instantiated with short ID doesn't have short ID");
4463
4464 UnicodeString rules1;
4465 UnicodeString rules2;
4466
4467 t1->toRules(rules1, TRUE);
4468 t2->toRules(rules2, TRUE);
4469 if (rules1 != rules2)
4470 errln("Alias transliterators aren't the same");
4471
4472 delete t1;
4473 delete t2;
4474 Transliterator::unregister(shortID);
4475
4476 t1 = Transliterator::createInstance(shortID, UTRANS_FORWARD, err);
4477 if (U_SUCCESS(err)) {
4478 errln("Instantiation with short ID succeeded after short ID was unregistered");
4479 delete t1;
4480 }
4481
4482 // try the same thing again, but this time with something other than
4483 // an instance of CompoundTransliterator
4484 UnicodeString realID("Latin-Greek");
4485 UnicodeString fakeID("Latin-dlgkjdflkjdl");
4486 Transliterator::registerAlias(fakeID, realID);
4487
4488 err = U_ZERO_ERROR;
4489 t1 = Transliterator::createInstance(realID, UTRANS_FORWARD, err);
4490 if (U_FAILURE(err)) {
4491 dataerrln("Failed to instantiate transliterator with real ID - %s", u_errorName(err));
4492 Transliterator::unregister(realID);
4493 return;
4494 }
4495 t2 = Transliterator::createInstance(fakeID, UTRANS_FORWARD, err);
4496 if (U_FAILURE(err)) {
4497 errln("Failed to instantiate transliterator with fake ID");
4498 delete t1;
4499 Transliterator::unregister(realID);
4500 return;
4501 }
4502
4503 t1->toRules(rules1, TRUE);
4504 t2->toRules(rules2, TRUE);
4505 if (rules1 != rules2)
4506 errln("Alias transliterators aren't the same");
4507
4508 delete t1;
4509 delete t2;
4510 Transliterator::unregister(fakeID);
4511 }
4512
TestRuleStripping()4513 void TransliteratorTest::TestRuleStripping() {
4514 /*
4515 #
4516 \uE001>\u0C01; # SIGN
4517 */
4518 static const UChar rule[] = {
4519 0x0023,0x0020,0x000D,0x000A,
4520 0xE001,0x003E,0x0C01,0x003B,0x0020,0x0023,0x0020,0x0053,0x0049,0x0047,0x004E,0
4521 };
4522 static const UChar expectedRule[] = {
4523 0xE001,0x003E,0x0C01,0x003B,0
4524 };
4525 UChar result[sizeof(rule)/sizeof(rule[0])];
4526 UErrorCode status = U_ZERO_ERROR;
4527 int32_t len = utrans_stripRules(rule, (int32_t)(sizeof(rule)/sizeof(rule[0])), result, &status);
4528 if (len != u_strlen(expectedRule)) {
4529 errln("utrans_stripRules return len = %d", len);
4530 }
4531 if (u_strncmp(expectedRule, result, len) != 0) {
4532 errln("utrans_stripRules did not return expected string");
4533 }
4534 }
4535
4536 /**
4537 * Test the Halfwidth-Fullwidth transliterator (ticket 6281).
4538 */
TestHalfwidthFullwidth(void)4539 void TransliteratorTest::TestHalfwidthFullwidth(void) {
4540 UParseError parseError;
4541 UErrorCode status = U_ZERO_ERROR;
4542 Transliterator* hf = Transliterator::createInstance("Halfwidth-Fullwidth", UTRANS_FORWARD, parseError, status);
4543 Transliterator* fh = Transliterator::createInstance("Fullwidth-Halfwidth", UTRANS_FORWARD, parseError, status);
4544 if (hf == 0 || fh == 0) {
4545 dataerrln("FAIL: createInstance failed - %s", u_errorName(status));
4546 delete hf;
4547 delete fh;
4548 return;
4549 }
4550
4551 // Array of 2n items
4552 // Each item is
4553 // "hf"|"fh"|"both",
4554 // <Halfwidth>,
4555 // <Fullwidth>
4556 const char* DATA[] = {
4557 "both",
4558 "\\uFFE9\\uFFEA\\uFFEB\\uFFEC\\u0061\\uFF71\\u00AF\\u0020",
4559 "\\u2190\\u2191\\u2192\\u2193\\uFF41\\u30A2\\uFFE3\\u3000",
4560 };
4561 int32_t DATA_length = (int32_t)(sizeof(DATA) / sizeof(DATA[0]));
4562
4563 for (int32_t i=0; i<DATA_length; i+=3) {
4564 UnicodeString h = CharsToUnicodeString(DATA[i+1]);
4565 UnicodeString f = CharsToUnicodeString(DATA[i+2]);
4566 switch (*DATA[i]) {
4567 case 0x68: //'h': // Halfwidth-Fullwidth only
4568 expect(*hf, h, f);
4569 break;
4570 case 0x66: //'f': // Fullwidth-Halfwidth only
4571 expect(*fh, f, h);
4572 break;
4573 case 0x62: //'b': // both directions
4574 expect(*hf, h, f);
4575 expect(*fh, f, h);
4576 break;
4577 }
4578 }
4579 delete hf;
4580 delete fh;
4581 }
4582
4583
4584 /**
4585 * Test Thai. The text is the first paragraph of "What is Unicode" from the Unicode.org web site.
4586 * TODO: confirm that the expected results are correct.
4587 * For now, test just confirms that C++ and Java give identical results.
4588 */
TestThai(void)4589 void TransliteratorTest::TestThai(void) {
4590 #if !UCONFIG_NO_BREAK_ITERATION
4591 UParseError parseError;
4592 UErrorCode status = U_ZERO_ERROR;
4593 Transliterator* tr = Transliterator::createInstance("Any-Latin", UTRANS_FORWARD, parseError, status);
4594 if (tr == 0) {
4595 dataerrln("FAIL: createInstance failed - %s", u_errorName(status));
4596 return;
4597 }
4598 if (U_FAILURE(status)) {
4599 errln("FAIL: createInstance failed with %s", u_errorName(status));
4600 return;
4601 }
4602 const char *thaiText =
4603 "\\u0e42\\u0e14\\u0e22\\u0e1e\\u0e37\\u0e49\\u0e19\\u0e10\\u0e32\\u0e19\\u0e41\\u0e25\\u0e49\\u0e27, \\u0e04\\u0e2d"
4604 "\\u0e21\\u0e1e\\u0e34\\u0e27\\u0e40\\u0e15\\u0e2d\\u0e23\\u0e4c\\u0e08\\u0e30\\u0e40\\u0e01\\u0e35\\u0e48\\u0e22"
4605 "\\u0e27\\u0e02\\u0e49\\u0e2d\\u0e07\\u0e01\\u0e31\\u0e1a\\u0e40\\u0e23\\u0e37\\u0e48\\u0e2d\\u0e07\\u0e02\\u0e2d"
4606 "\\u0e07\\u0e15\\u0e31\\u0e27\\u0e40\\u0e25\\u0e02. \\u0e04\\u0e2d\\u0e21\\u0e1e\\u0e34\\u0e27\\u0e40\\u0e15\\u0e2d"
4607 "\\u0e23\\u0e4c\\u0e08\\u0e31\\u0e14\\u0e40\\u0e01\\u0e47\\u0e1a\\u0e15\\u0e31\\u0e27\\u0e2d\\u0e31\\u0e01\\u0e29"
4608 "\\u0e23\\u0e41\\u0e25\\u0e30\\u0e2d\\u0e31\\u0e01\\u0e02\\u0e23\\u0e30\\u0e2d\\u0e37\\u0e48\\u0e19\\u0e46 \\u0e42"
4609 "\\u0e14\\u0e22\\u0e01\\u0e32\\u0e23\\u0e01\\u0e33\\u0e2b\\u0e19\\u0e14\\u0e2b\\u0e21\\u0e32\\u0e22\\u0e40\\u0e25"
4610 "\\u0e02\\u0e43\\u0e2b\\u0e49\\u0e2a\\u0e33\\u0e2b\\u0e23\\u0e31\\u0e1a\\u0e41\\u0e15\\u0e48\\u0e25\\u0e30\\u0e15"
4611 "\\u0e31\\u0e27. \\u0e01\\u0e48\\u0e2d\\u0e19\\u0e2b\\u0e19\\u0e49\\u0e32\\u0e17\\u0e35\\u0e48\\u0e4a Unicode \\u0e08"
4612 "\\u0e30\\u0e16\\u0e39\\u0e01\\u0e2a\\u0e23\\u0e49\\u0e32\\u0e07\\u0e02\\u0e36\\u0e49\\u0e19, \\u0e44\\u0e14\\u0e49"
4613 "\\u0e21\\u0e35\\u0e23\\u0e30\\u0e1a\\u0e1a encoding \\u0e2d\\u0e22\\u0e39\\u0e48\\u0e2b\\u0e25\\u0e32\\u0e22\\u0e23"
4614 "\\u0e49\\u0e2d\\u0e22\\u0e23\\u0e30\\u0e1a\\u0e1a\\u0e2a\\u0e33\\u0e2b\\u0e23\\u0e31\\u0e1a\\u0e01\\u0e32\\u0e23"
4615 "\\u0e01\\u0e33\\u0e2b\\u0e19\\u0e14\\u0e2b\\u0e21\\u0e32\\u0e22\\u0e40\\u0e25\\u0e02\\u0e40\\u0e2b\\u0e25\\u0e48"
4616 "\\u0e32\\u0e19\\u0e35\\u0e49. \\u0e44\\u0e21\\u0e48\\u0e21\\u0e35 encoding \\u0e43\\u0e14\\u0e17\\u0e35\\u0e48"
4617 "\\u0e21\\u0e35\\u0e08\\u0e33\\u0e19\\u0e27\\u0e19\\u0e15\\u0e31\\u0e27\\u0e2d\\u0e31\\u0e01\\u0e02\\u0e23\\u0e30"
4618 "\\u0e21\\u0e32\\u0e01\\u0e40\\u0e1e\\u0e35\\u0e22\\u0e07\\u0e1e\\u0e2d: \\u0e22\\u0e01\\u0e15\\u0e31\\u0e27\\u0e2d"
4619 "\\u0e22\\u0e48\\u0e32\\u0e07\\u0e40\\u0e0a\\u0e48\\u0e19, \\u0e40\\u0e09\\u0e1e\\u0e32\\u0e30\\u0e43\\u0e19\\u0e01"
4620 "\\u0e25\\u0e38\\u0e48\\u0e21\\u0e2a\\u0e2b\\u0e20\\u0e32\\u0e1e\\u0e22\\u0e38\\u0e42\\u0e23\\u0e1b\\u0e40\\u0e1e"
4621 "\\u0e35\\u0e22\\u0e07\\u0e41\\u0e2b\\u0e48\\u0e07\\u0e40\\u0e14\\u0e35\\u0e22\\u0e27 \\u0e01\\u0e47\\u0e15\\u0e49"
4622 "\\u0e2d\\u0e07\\u0e01\\u0e32\\u0e23\\u0e2b\\u0e25\\u0e32\\u0e22 encoding \\u0e43\\u0e19\\u0e01\\u0e32\\u0e23\\u0e04"
4623 "\\u0e23\\u0e2d\\u0e1a\\u0e04\\u0e25\\u0e38\\u0e21\\u0e17\\u0e38\\u0e01\\u0e20\\u0e32\\u0e29\\u0e32\\u0e43\\u0e19"
4624 "\\u0e01\\u0e25\\u0e38\\u0e48\\u0e21. \\u0e2b\\u0e23\\u0e37\\u0e2d\\u0e41\\u0e21\\u0e49\\u0e41\\u0e15\\u0e48\\u0e43"
4625 "\\u0e19\\u0e20\\u0e32\\u0e29\\u0e32\\u0e40\\u0e14\\u0e35\\u0e48\\u0e22\\u0e27 \\u0e40\\u0e0a\\u0e48\\u0e19 \\u0e20"
4626 "\\u0e32\\u0e29\\u0e32\\u0e2d\\u0e31\\u0e07\\u0e01\\u0e24\\u0e29 \\u0e01\\u0e47\\u0e44\\u0e21\\u0e48\\u0e21\\u0e35"
4627 " encoding \\u0e43\\u0e14\\u0e17\\u0e35\\u0e48\\u0e40\\u0e1e\\u0e35\\u0e22\\u0e07\\u0e1e\\u0e2d\\u0e2a\\u0e33\\u0e2b"
4628 "\\u0e23\\u0e31\\u0e1a\\u0e17\\u0e38\\u0e01\\u0e15\\u0e31\\u0e27\\u0e2d\\u0e31\\u0e01\\u0e29\\u0e23, \\u0e40\\u0e04"
4629 "\\u0e23\\u0e37\\u0e48\\u0e2d\\u0e07\\u0e2b\\u0e21\\u0e32\\u0e22\\u0e27\\u0e23\\u0e23\\u0e04\\u0e15\\u0e2d\\u0e19"
4630 " \\u0e41\\u0e25\\u0e30\\u0e2a\\u0e31\\u0e0d\\u0e25\\u0e31\\u0e01\\u0e29\\u0e13\\u0e4c\\u0e17\\u0e32\\u0e07\\u0e40"
4631 "\\u0e17\\u0e04\\u0e19\\u0e34\\u0e04\\u0e17\\u0e35\\u0e48\\u0e43\\u0e0a\\u0e49\\u0e01\\u0e31\\u0e19\\u0e2d\\u0e22"
4632 "\\u0e39\\u0e48\\u0e17\\u0e31\\u0e48\\u0e27\\u0e44\\u0e1b.";
4633
4634 const char *latinText =
4635 "doy ph\\u1ee5\\u0304\\u0302n \\u1e6d\\u0304h\\u0101n l\\u00e6\\u0302w, khxmphiwtexr\\u0312 ca ke\\u012b\\u0300"
4636 "ywk\\u0304\\u0125xng k\\u1ea1b re\\u1ee5\\u0304\\u0300xng k\\u0304hxng t\\u1ea1wlek\\u0304h. khxmphiwtexr"
4637 "\\u0312 c\\u1ea1d k\\u0115b t\\u1ea1w x\\u1ea1ks\\u0304\\u02b9r l\\u00e6a x\\u1ea1kk\\u0304h ra x\\u1ee5\\u0304"
4638 "\\u0300n\\u00ab doy k\\u0101r k\\u1ea3h\\u0304nd h\\u0304m\\u0101ylek\\u0304h h\\u0304\\u0131\\u0302 s\\u0304"
4639 "\\u1ea3h\\u0304r\\u1ea1b t\\u00e6\\u0300la t\\u1ea1w. k\\u0300xn h\\u0304n\\u0302\\u0101 th\\u012b\\u0300\\u0301"
4640 " Unicode ca t\\u0304h\\u016bk s\\u0304r\\u0302\\u0101ng k\\u0304h\\u1ee5\\u0302n, d\\u1ecb\\u0302 m\\u012b "
4641 "rabb encoding xy\\u016b\\u0300 h\\u0304l\\u0101y r\\u0302xy rabb s\\u0304\\u1ea3h\\u0304r\\u1ea1b k\\u0101"
4642 "r k\\u1ea3h\\u0304nd h\\u0304m\\u0101ylek\\u0304h h\\u0304el\\u0300\\u0101 n\\u012b\\u0302. m\\u1ecb\\u0300m"
4643 "\\u012b encoding d\\u0131 th\\u012b\\u0300 m\\u012b c\\u1ea3nwn t\\u1ea1w x\\u1ea1kk\\u0304hra m\\u0101k p"
4644 "he\\u012byng phx: yk t\\u1ea1wx\\u1ef3\\u0101ng ch\\u00e8n, c\\u0304heph\\u0101a n\\u0131 kl\\u00f9m s\\u0304"
4645 "h\\u0304p\\u0323h\\u0101ph yurop phe\\u012byng h\\u0304\\u00e6\\u0300ng de\\u012byw k\\u0306 t\\u0302xngk\\u0101"
4646 "r h\\u0304l\\u0101y encoding n\\u0131 k\\u0101r khrxbkhlum thuk p\\u0323h\\u0101s\\u0304\\u02b9\\u0101 n\\u0131"
4647 " kl\\u00f9m. h\\u0304r\\u1ee5\\u0304x m\\u00e6\\u0302t\\u00e6\\u0300 n\\u0131 p\\u0323h\\u0101s\\u0304\\u02b9"
4648 "\\u0101 de\\u012b\\u0300yw ch\\u00e8n p\\u0323h\\u0101s\\u0304\\u02b9\\u0101 x\\u1ea1ngkvs\\u0304\\u02b9 k\\u0306"
4649 " m\\u1ecb\\u0300m\\u012b encoding d\\u0131 th\\u012b\\u0300 phe\\u012byng phx s\\u0304\\u1ea3h\\u0304r\\u1ea1"
4650 "b thuk t\\u1ea1w x\\u1ea1ks\\u0304\\u02b9r, kher\\u1ee5\\u0304\\u0300xngh\\u0304m\\u0101y wrrkh txn l\\u00e6"
4651 "a s\\u0304\\u1ea1\\u1ef5l\\u1ea1ks\\u0304\\u02b9\\u1e47\\u0312 th\\u0101ng thekhnikh th\\u012b\\u0300 ch\\u0131"
4652 "\\u0302 k\\u1ea1n xy\\u016b\\u0300 th\\u1ea1\\u0300wp\\u1ecb.";
4653
4654
4655 UnicodeString xlitText(thaiText);
4656 xlitText = xlitText.unescape();
4657 tr->transliterate(xlitText);
4658
4659 UnicodeString expectedText(latinText);
4660 expectedText = expectedText.unescape();
4661 expect(*tr, xlitText, expectedText);
4662
4663 delete tr;
4664 #endif
4665 }
4666
4667
4668 //======================================================================
4669 // Support methods
4670 //======================================================================
expectT(const UnicodeString & id,const UnicodeString & source,const UnicodeString & expectedResult)4671 void TransliteratorTest::expectT(const UnicodeString& id,
4672 const UnicodeString& source,
4673 const UnicodeString& expectedResult) {
4674 UErrorCode ec = U_ZERO_ERROR;
4675 UParseError pe;
4676 Transliterator *t = Transliterator::createInstance(id, UTRANS_FORWARD, pe, ec);
4677 if (U_FAILURE(ec)) {
4678 errln((UnicodeString)"FAIL: Could not create " + id + " - " + u_errorName(ec));
4679 delete t;
4680 return;
4681 }
4682 expect(*t, source, expectedResult);
4683 delete t;
4684 }
4685
reportParseError(const UnicodeString & message,const UParseError & parseError,const UErrorCode & status)4686 void TransliteratorTest::reportParseError(const UnicodeString& message,
4687 const UParseError& parseError,
4688 const UErrorCode& status) {
4689 dataerrln(message +
4690 /*", parse error " + parseError.code +*/
4691 ", line " + parseError.line +
4692 ", offset " + parseError.offset +
4693 ", pre-context " + prettify(parseError.preContext, TRUE) +
4694 ", post-context " + prettify(parseError.postContext,TRUE) +
4695 ", Error: " + u_errorName(status));
4696 }
4697
expect(const UnicodeString & rules,const UnicodeString & source,const UnicodeString & expectedResult,UTransPosition * pos)4698 void TransliteratorTest::expect(const UnicodeString& rules,
4699 const UnicodeString& source,
4700 const UnicodeString& expectedResult,
4701 UTransPosition *pos) {
4702 expect("<ID>", rules, source, expectedResult, pos);
4703 }
4704
expect(const UnicodeString & id,const UnicodeString & rules,const UnicodeString & source,const UnicodeString & expectedResult,UTransPosition * pos)4705 void TransliteratorTest::expect(const UnicodeString& id,
4706 const UnicodeString& rules,
4707 const UnicodeString& source,
4708 const UnicodeString& expectedResult,
4709 UTransPosition *pos) {
4710 UErrorCode status = U_ZERO_ERROR;
4711 UParseError parseError;
4712 Transliterator* t = Transliterator::createFromRules(id, rules, UTRANS_FORWARD, parseError, status);
4713 if (U_FAILURE(status)) {
4714 reportParseError(UnicodeString("Couldn't create transliterator from ") + rules, parseError, status);
4715 } else {
4716 expect(*t, source, expectedResult, pos);
4717 }
4718 delete t;
4719 }
4720
expect(const Transliterator & t,const UnicodeString & source,const UnicodeString & expectedResult,const Transliterator & reverseTransliterator)4721 void TransliteratorTest::expect(const Transliterator& t,
4722 const UnicodeString& source,
4723 const UnicodeString& expectedResult,
4724 const Transliterator& reverseTransliterator) {
4725 expect(t, source, expectedResult);
4726 expect(reverseTransliterator, expectedResult, source);
4727 }
4728
expect(const Transliterator & t,const UnicodeString & source,const UnicodeString & expectedResult,UTransPosition * pos)4729 void TransliteratorTest::expect(const Transliterator& t,
4730 const UnicodeString& source,
4731 const UnicodeString& expectedResult,
4732 UTransPosition *pos) {
4733 if (pos == 0) {
4734 UnicodeString result(source);
4735 t.transliterate(result);
4736 expectAux(t.getID() + ":String", source, result, expectedResult);
4737 }
4738 UTransPosition index={0, 0, 0, 0};
4739 if (pos != 0) {
4740 index = *pos;
4741 }
4742
4743 UnicodeString rsource(source);
4744 if (pos == 0) {
4745 t.transliterate(rsource);
4746 } else {
4747 // Do it all at once -- below we do it incrementally
4748 t.finishTransliteration(rsource, *pos);
4749 }
4750 expectAux(t.getID() + ":Replaceable", source, rsource, expectedResult);
4751
4752 // Test keyboard (incremental) transliteration -- this result
4753 // must be the same after we finalize (see below).
4754 UnicodeString log;
4755 rsource.remove();
4756 if (pos != 0) {
4757 rsource = source;
4758 formatInput(log, rsource, index);
4759 log.append(" -> ");
4760 UErrorCode status = U_ZERO_ERROR;
4761 t.transliterate(rsource, index, status);
4762 formatInput(log, rsource, index);
4763 } else {
4764 for (int32_t i=0; i<source.length(); ++i) {
4765 if (i != 0) {
4766 log.append(" + ");
4767 }
4768 log.append(source.charAt(i)).append(" -> ");
4769 UErrorCode status = U_ZERO_ERROR;
4770 t.transliterate(rsource, index, source.charAt(i), status);
4771 formatInput(log, rsource, index);
4772 }
4773 }
4774
4775 // As a final step in keyboard transliteration, we must call
4776 // transliterate to finish off any pending partial matches that
4777 // were waiting for more input.
4778 t.finishTransliteration(rsource, index);
4779 log.append(" => ").append(rsource);
4780
4781 expectAux(t.getID() + ":Keyboard", log,
4782 rsource == expectedResult,
4783 expectedResult);
4784 }
4785
4786
4787 /**
4788 * @param appendTo result is appended to this param.
4789 * @param input the string being transliterated
4790 * @param pos the index struct
4791 */
formatInput(UnicodeString & appendTo,const UnicodeString & input,const UTransPosition & pos)4792 UnicodeString& TransliteratorTest::formatInput(UnicodeString &appendTo,
4793 const UnicodeString& input,
4794 const UTransPosition& pos) {
4795 // Output a string of the form aaa{bbb|ccc|ddd}eee, where
4796 // the {} indicate the context start and limit, and the ||
4797 // indicate the start and limit.
4798 if (0 <= pos.contextStart &&
4799 pos.contextStart <= pos.start &&
4800 pos.start <= pos.limit &&
4801 pos.limit <= pos.contextLimit &&
4802 pos.contextLimit <= input.length()) {
4803
4804 UnicodeString a, b, c, d, e;
4805 input.extractBetween(0, pos.contextStart, a);
4806 input.extractBetween(pos.contextStart, pos.start, b);
4807 input.extractBetween(pos.start, pos.limit, c);
4808 input.extractBetween(pos.limit, pos.contextLimit, d);
4809 input.extractBetween(pos.contextLimit, input.length(), e);
4810 appendTo.append(a).append((UChar)123/*{*/).append(b).
4811 append((UChar)PIPE).append(c).append((UChar)PIPE).append(d).
4812 append((UChar)125/*}*/).append(e);
4813 } else {
4814 appendTo.append((UnicodeString)"INVALID UTransPosition {cs=" +
4815 pos.contextStart + ", s=" + pos.start + ", l=" +
4816 pos.limit + ", cl=" + pos.contextLimit + "} on " +
4817 input);
4818 }
4819 return appendTo;
4820 }
4821
expectAux(const UnicodeString & tag,const UnicodeString & source,const UnicodeString & result,const UnicodeString & expectedResult)4822 void TransliteratorTest::expectAux(const UnicodeString& tag,
4823 const UnicodeString& source,
4824 const UnicodeString& result,
4825 const UnicodeString& expectedResult) {
4826 expectAux(tag, source + " -> " + result,
4827 result == expectedResult,
4828 expectedResult);
4829 }
4830
expectAux(const UnicodeString & tag,const UnicodeString & summary,UBool pass,const UnicodeString & expectedResult)4831 void TransliteratorTest::expectAux(const UnicodeString& tag,
4832 const UnicodeString& summary, UBool pass,
4833 const UnicodeString& expectedResult) {
4834 if (pass) {
4835 logln(UnicodeString("(")+tag+") " + prettify(summary));
4836 } else {
4837 dataerrln(UnicodeString("FAIL: (")+tag+") "
4838 + prettify(summary)
4839 + ", expected " + prettify(expectedResult));
4840 }
4841 }
4842
4843 #endif /* #if !UCONFIG_NO_TRANSLITERATION */
4844