1 // © 2016 and later: Unicode, Inc. and others.
2 // License & terms of use: http://www.unicode.org/copyright.html
3 /********************************************************************
4 * Copyright (c) 2001-2009, International Business Machines
5 * Corporation and others. All Rights Reserved.
6 *********************************************************************
7 * This test program is intended for testing error conditions of the
8 * transliterator APIs to make sure the exceptions are raised where
9 * necessary.
10 *
11 * Date Name Description
12 * 11/14/2001 hshih Creation.
13 *
14 ********************************************************************/
15
16 #include "unicode/utypes.h"
17
18 #if !UCONFIG_NO_TRANSLITERATION
19
20 #include "ittrans.h"
21 #include "trnserr.h"
22 #include "unicode/utypes.h"
23 #include "unicode/translit.h"
24 #include "unicode/uniset.h"
25 #include "unicode/unifilt.h"
26 #include "cpdtrans.h"
27 #include <string.h>
28 #include <stdio.h>
29 #include <stdlib.h>
30 #include "unicode/rep.h"
31 #include "unicode/locid.h"
32
33 //---------------------------------------------
34 // runIndexedTest
35 //---------------------------------------------
36
37 void
runIndexedTest(int32_t index,UBool exec,const char * & name,char *)38 TransliteratorErrorTest::runIndexedTest(int32_t index, UBool exec,
39 const char* &name, char* /*par*/) {
40 switch (index) {
41 TESTCASE(0,TestTransliteratorErrors);
42 TESTCASE(1, TestUnicodeSetErrors);
43 TESTCASE(2, TestRBTErrors);
44 TESTCASE(3, TestCoverage);
45 //TESTCASE(3, TestUniToHexErrors);
46 //TESTCASE(4, TestHexToUniErrors);
47 // TODO: Add a subclass to test clone().
48 default: name = ""; break;
49 }
50 }
51
52
TestTransliteratorErrors()53 void TransliteratorErrorTest::TestTransliteratorErrors() {
54 UnicodeString trans="Latin-Greek";
55 UnicodeString bogusID="LATINGREEK-GREEKLATIN";
56 UnicodeString newID="Bogus-Latin";
57 UnicodeString newIDRules="zzz > Z; f <> ph";
58 UnicodeString bogusRules="a } [b-g m-p ";
59 UParseError parseError;
60 UErrorCode status = U_ZERO_ERROR;
61 UnicodeString testString="A quick fox jumped over the lazy dog.";
62 UnicodeString insertString="cats and dogs";
63 int32_t stoppedAt = 0, len;
64 UTransPosition pos;
65
66 Transliterator* t= Transliterator::createInstance(trans, UTRANS_FORWARD, parseError, status);
67 if(t==0 || U_FAILURE(status)){
68 dataerrln("FAIL: construction of Latin-Greek - %s", u_errorName(status));
69 return;
70 }
71 pos.contextLimit = 0;
72 pos.contextStart = 0;
73 pos.limit = 0;
74 pos.start = 0;
75 len = testString.length();
76 stoppedAt = t->transliterate(testString, 0, 100);
77 if (stoppedAt != -1) {
78 errln("FAIL: Out of bounds check failed (1).");
79 } else if (testString.length() != len) {
80 testString="A quick fox jumped over the lazy dog.";
81 errln("FAIL: Transliterate fails and the target string was modified.");
82 }
83 stoppedAt = t->transliterate(testString, 100, testString.length()-1);
84 if (stoppedAt != -1)
85 errln("FAIL: Out of bounds check failed (2).");
86 else if (testString.length() != len) {
87 testString="A quick fox jumped over the lazy dog.";
88 errln("FAIL: Transliterate fails and the target string was modified.");
89 }
90 pos.start = 100;
91 pos.limit = testString.length();
92 t->transliterate(testString, pos, status);
93 if (U_SUCCESS(status)) {
94 errln("FAIL: Start offset is out of bounds, error not reported.\n");
95 }
96 status = U_ZERO_ERROR;
97 pos.limit = 100;
98 pos.start = 0;
99 t->transliterate(testString, pos, status);
100 if (U_SUCCESS(status)) {
101 errln("FAIL: Limit offset is out of bounds, error not reported.\n");
102 }
103 status = U_ZERO_ERROR;
104 len = pos.contextLimit = testString.length();
105 pos.contextStart = 0;
106 pos.limit = len - 1;
107 pos.start = 5;
108 t->transliterate(testString, pos, insertString, status);
109 if (len == pos.limit) {
110 errln("FAIL: Test insertion with string: the transliteration position limit didn't change as expected.");
111 if (U_SUCCESS(status)) {
112 errln("FAIL: Error code wasn't set either.");
113 }
114 }
115 status = U_ZERO_ERROR;
116 pos.contextStart = 0;
117 pos.contextLimit = testString.length();
118 pos.limit = testString.length() -1;
119 pos.start = 5;
120 t->transliterate(testString, pos, (UChar32)0x0061, status);
121 if (len == pos.limit) {
122 errln("FAIL: Test insertion with character: the transliteration position limit didn't change as expected.");
123 if (U_SUCCESS(status)) {
124 errln("FAIL: Error code wasn't set either.");
125 }
126 }
127 status = U_ZERO_ERROR;
128 len = pos.limit = testString.length();
129 pos.contextStart = 0;
130 pos.contextLimit = testString.length() - 1;
131 pos.start = 5;
132 t->transliterate(testString, pos, insertString, status);
133 if (U_SUCCESS(status)) {
134 errln("FAIL: Out of bounds check failed (3).");
135 if (testString.length() != len)
136 errln("FAIL: The input string was modified though the offsets were out of bounds.");
137 }
138 Transliterator* t1= Transliterator::createInstance(bogusID, UTRANS_FORWARD, parseError, status);
139 if(t1!=0 || U_SUCCESS(status)){
140 delete t1;
141 errln("FAIL: construction of bogus ID \"LATINGREEK-GREEKLATIN\"");
142 }
143 status = U_ZERO_ERROR;
144 Transliterator* t2 = Transliterator::createFromRules(newID, newIDRules, UTRANS_FORWARD, parseError, status);
145 if (U_SUCCESS(status)) {
146 Transliterator* t3 = t2->createInverse(status);
147 if (U_SUCCESS(status)) {
148 delete t3;
149 errln("FAIL: The newID transliterator was not registered so createInverse should fail.");
150 } else {
151 delete t3;
152 }
153 }
154 status = U_ZERO_ERROR;
155 Transliterator* t4 = Transliterator::createFromRules(newID, bogusRules, UTRANS_FORWARD, parseError, status);
156 if (t4 != NULL || U_SUCCESS(status)) {
157 errln("FAIL: The rules is malformed but error was not reported.");
158 if (parseError.offset != -1) {
159 errln("FAIL: The parse error offset isn't set correctly when fails.");
160 } else if (parseError.postContext[0] == 0 || parseError.preContext[0] == 0) {
161 errln("FAIL: The parse error pre/post context isn't reset properly.");
162 }
163 delete t4;
164 }
165 delete t;
166 delete t2;
167 }
168
TestUnicodeSetErrors()169 void TransliteratorErrorTest::TestUnicodeSetErrors() {
170 UnicodeString badPattern="[[:L:]-[0x0300-0x0400]";
171 UnicodeSet set;
172 UErrorCode status = U_ZERO_ERROR;
173 UnicodeString result;
174
175 if (!set.isEmpty()) {
176 errln("FAIL: The default ctor of UnicodeSet created a non-empty object.");
177 }
178 set.applyPattern(badPattern, status);
179 if (U_SUCCESS(status)) {
180 errln("FAIL: Applied a bad pattern to the UnicodeSet object okay.");
181 }
182 status = U_ZERO_ERROR;
183 UnicodeSet *set1 = new UnicodeSet(badPattern, status);
184 if (U_SUCCESS(status)) {
185 errln("FAIL: Created a UnicodeSet based on bad patterns.");
186 }
187 delete set1;
188 }
189
190 //void TransliteratorErrorTest::TestUniToHexErrors() {
191 // UErrorCode status = U_ZERO_ERROR;
192 // Transliterator *t = new UnicodeToHexTransliterator("", TRUE, NULL, status);
193 // if (U_SUCCESS(status)) {
194 // errln("FAIL: Created a UnicodeToHexTransliterator with an empty pattern.");
195 // }
196 // delete t;
197 //
198 // status = U_ZERO_ERROR;
199 // t = new UnicodeToHexTransliterator("\\x", TRUE, NULL, status);
200 // if (U_SUCCESS(status)) {
201 // errln("FAIL: Created a UnicodeToHexTransliterator with a bad pattern.");
202 // }
203 // delete t;
204 //
205 // status = U_ZERO_ERROR;
206 // t = new UnicodeToHexTransliterator();
207 // ((UnicodeToHexTransliterator*)t)->applyPattern("\\x", status);
208 // if (U_SUCCESS(status)) {
209 // errln("FAIL: UnicodeToHexTransliterator::applyPattern succeeded with a bad pattern.");
210 // }
211 // delete t;
212 //}
213
TestRBTErrors()214 void TransliteratorErrorTest::TestRBTErrors() {
215
216 UnicodeString rules="ab>y";
217 UnicodeString id="MyRandom-YReverse";
218 //UnicodeString goodPattern="[[:L:]&[\\u0000-\\uFFFF]]"; /* all BMP letters */
219 UErrorCode status = U_ZERO_ERROR;
220 UParseError parseErr;
221 /*UnicodeSet *set = new UnicodeSet(goodPattern, status);
222 if (U_FAILURE(status)) {
223 errln("FAIL: Was not able to create a good UnicodeSet based on valid patterns.");
224 return;
225 }*/
226 Transliterator *t = Transliterator::createFromRules(id, rules, UTRANS_REVERSE, parseErr, status);
227 if (U_FAILURE(status)) {
228 errln("FAIL: Was not able to create a good RBT to test registration.");
229 //delete set;
230 return;
231 }
232 Transliterator::registerInstance(t);
233 Transliterator::unregister(id);
234 status = U_ZERO_ERROR;
235 Transliterator* t1= Transliterator::createInstance(id, UTRANS_REVERSE, parseErr, status);
236 if(U_SUCCESS(status)){
237 delete t1;
238 errln("FAIL: construction of unregistered ID failed.");
239 }
240 }
241
242 //void TransliteratorErrorTest::TestHexToUniErrors() {
243 // UErrorCode status = U_ZERO_ERROR;
244 // Transliterator *t = new HexToUnicodeTransliterator("", NULL, status);
245 // if (U_FAILURE(status)) {
246 // errln("FAIL: Could not create a HexToUnicodeTransliterator with an empty pattern.");
247 // }
248 // delete t;
249 // status = U_ZERO_ERROR;
250 // t = new HexToUnicodeTransliterator("\\x", NULL, status);
251 // if (U_SUCCESS(status)) {
252 // errln("FAIL: Created a HexToUnicodeTransliterator with a bad pattern.");
253 // }
254 // delete t;
255 // status = U_ZERO_ERROR;
256 // t = new HexToUnicodeTransliterator();
257 // ((HexToUnicodeTransliterator*)t)->applyPattern("\\x", status);
258 // if (U_SUCCESS(status)) {
259 // errln("FAIL: HexToUnicodeTransliterator::applyPattern succeeded with a bad pattern.");
260 // }
261 // delete t;
262 //}
263
264 class StubTransliterator: public Transliterator{
265 public:
StubTransliterator()266 StubTransliterator(): Transliterator(UNICODE_STRING_SIMPLE("Any-Null"), 0) {}
handleTransliterate(Replaceable &,UTransPosition & offsets,UBool) const267 virtual void handleTransliterate(Replaceable& ,UTransPosition& offsets,UBool) const {
268 offsets.start = offsets.limit;
269 }
270
getDynamicClassID() const271 virtual UClassID getDynamicClassID() const{
272 static char classID = 0;
273 return (UClassID)&classID;
274 }
275 };
276
TestCoverage()277 void TransliteratorErrorTest::TestCoverage() {
278 StubTransliterator stub;
279
280 if (stub.clone() != NULL){
281 errln("FAIL: default Transliterator::clone() should return NULL");
282 }
283 }
284
285 #endif /* #if !UCONFIG_NO_TRANSLITERATION */
286