This is a live mirror of the Perl 5 development currently hosted at https://github.com/perl/perl5
Unicode::UCD::prop_invmap() compress digit results
[perl5.git] / lib / Unicode / UCD.t
1 #!perl -w
2 BEGIN {
3     if (ord("A") != 65) {
4         print "1..0 # Skip: EBCDIC\n";
5         exit 0;
6     }
7     chdir 't' if -d 't';
8     @INC = '../lib';
9     require Config; import Config;
10     if ($Config{'extensions'} !~ /\bStorable\b/) {
11         print "1..0 # Skip: Storable was not built; Unicode::UCD uses Storable\n";
12         exit 0;
13     }
14 }
15
16 use strict;
17 use Unicode::UCD;
18 use Test::More;
19
20 use Unicode::UCD 'charinfo';
21
22 $/ = 7;
23
24 my $charinfo;
25
26 is(charinfo(0x110000), undef, "Verify charinfo() of non-unicode is undef");
27
28 $charinfo = charinfo(0);    # Null is often problematic, so test it.
29
30 is($charinfo->{code},           '0000', '<control>');
31 is($charinfo->{name},           '<control>');
32 is($charinfo->{category},       'Cc');
33 is($charinfo->{combining},      '0');
34 is($charinfo->{bidi},           'BN');
35 is($charinfo->{decomposition},  '');
36 is($charinfo->{decimal},        '');
37 is($charinfo->{digit},          '');
38 is($charinfo->{numeric},        '');
39 is($charinfo->{mirrored},       'N');
40 is($charinfo->{unicode10},      'NULL');
41 is($charinfo->{comment},        '');
42 is($charinfo->{upper},          '');
43 is($charinfo->{lower},          '');
44 is($charinfo->{title},          '');
45 is($charinfo->{block},          'Basic Latin');
46 is($charinfo->{script},         'Common');
47
48 $charinfo = charinfo(0x41);
49
50 is($charinfo->{code},           '0041', 'LATIN CAPITAL LETTER A');
51 is($charinfo->{name},           'LATIN CAPITAL LETTER A');
52 is($charinfo->{category},       'Lu');
53 is($charinfo->{combining},      '0');
54 is($charinfo->{bidi},           'L');
55 is($charinfo->{decomposition},  '');
56 is($charinfo->{decimal},        '');
57 is($charinfo->{digit},          '');
58 is($charinfo->{numeric},        '');
59 is($charinfo->{mirrored},       'N');
60 is($charinfo->{unicode10},      '');
61 is($charinfo->{comment},        '');
62 is($charinfo->{upper},          '');
63 is($charinfo->{lower},          '0061');
64 is($charinfo->{title},          '');
65 is($charinfo->{block},          'Basic Latin');
66 is($charinfo->{script},         'Latin');
67
68 $charinfo = charinfo(0x100);
69
70 is($charinfo->{code},           '0100', 'LATIN CAPITAL LETTER A WITH MACRON');
71 is($charinfo->{name},           'LATIN CAPITAL LETTER A WITH MACRON');
72 is($charinfo->{category},       'Lu');
73 is($charinfo->{combining},      '0');
74 is($charinfo->{bidi},           'L');
75 is($charinfo->{decomposition},  '0041 0304');
76 is($charinfo->{decimal},        '');
77 is($charinfo->{digit},          '');
78 is($charinfo->{numeric},        '');
79 is($charinfo->{mirrored},       'N');
80 is($charinfo->{unicode10},      'LATIN CAPITAL LETTER A MACRON');
81 is($charinfo->{comment},        '');
82 is($charinfo->{upper},          '');
83 is($charinfo->{lower},          '0101');
84 is($charinfo->{title},          '');
85 is($charinfo->{block},          'Latin Extended-A');
86 is($charinfo->{script},         'Latin');
87
88 # 0x0590 is in the Hebrew block but unused.
89
90 $charinfo = charinfo(0x590);
91
92 is($charinfo->{code},          undef,   '0x0590 - unused Hebrew');
93 is($charinfo->{name},          undef);
94 is($charinfo->{category},      undef);
95 is($charinfo->{combining},     undef);
96 is($charinfo->{bidi},          undef);
97 is($charinfo->{decomposition}, undef);
98 is($charinfo->{decimal},       undef);
99 is($charinfo->{digit},         undef);
100 is($charinfo->{numeric},       undef);
101 is($charinfo->{mirrored},      undef);
102 is($charinfo->{unicode10},     undef);
103 is($charinfo->{comment},       undef);
104 is($charinfo->{upper},         undef);
105 is($charinfo->{lower},         undef);
106 is($charinfo->{title},         undef);
107 is($charinfo->{block},         undef);
108 is($charinfo->{script},        undef);
109
110 # 0x05d0 is in the Hebrew block and used.
111
112 $charinfo = charinfo(0x5d0);
113
114 is($charinfo->{code},           '05D0', '05D0 - used Hebrew');
115 is($charinfo->{name},           'HEBREW LETTER ALEF');
116 is($charinfo->{category},       'Lo');
117 is($charinfo->{combining},      '0');
118 is($charinfo->{bidi},           'R');
119 is($charinfo->{decomposition},  '');
120 is($charinfo->{decimal},        '');
121 is($charinfo->{digit},          '');
122 is($charinfo->{numeric},        '');
123 is($charinfo->{mirrored},       'N');
124 is($charinfo->{unicode10},      '');
125 is($charinfo->{comment},        '');
126 is($charinfo->{upper},          '');
127 is($charinfo->{lower},          '');
128 is($charinfo->{title},          '');
129 is($charinfo->{block},          'Hebrew');
130 is($charinfo->{script},         'Hebrew');
131
132 # An open syllable in Hangul.
133
134 $charinfo = charinfo(0xAC00);
135
136 is($charinfo->{code},           'AC00', 'HANGUL SYLLABLE U+AC00');
137 is($charinfo->{name},           'HANGUL SYLLABLE GA');
138 is($charinfo->{category},       'Lo');
139 is($charinfo->{combining},      '0');
140 is($charinfo->{bidi},           'L');
141 is($charinfo->{decomposition},  '1100 1161');
142 is($charinfo->{decimal},        '');
143 is($charinfo->{digit},          '');
144 is($charinfo->{numeric},        '');
145 is($charinfo->{mirrored},       'N');
146 is($charinfo->{unicode10},      '');
147 is($charinfo->{comment},        '');
148 is($charinfo->{upper},          '');
149 is($charinfo->{lower},          '');
150 is($charinfo->{title},          '');
151 is($charinfo->{block},          'Hangul Syllables');
152 is($charinfo->{script},         'Hangul');
153
154 # A closed syllable in Hangul.
155
156 $charinfo = charinfo(0xAE00);
157
158 is($charinfo->{code},           'AE00', 'HANGUL SYLLABLE U+AE00');
159 is($charinfo->{name},           'HANGUL SYLLABLE GEUL');
160 is($charinfo->{category},       'Lo');
161 is($charinfo->{combining},      '0');
162 is($charinfo->{bidi},           'L');
163 is($charinfo->{decomposition},  "1100 1173 11AF");
164 is($charinfo->{decimal},        '');
165 is($charinfo->{digit},          '');
166 is($charinfo->{numeric},        '');
167 is($charinfo->{mirrored},       'N');
168 is($charinfo->{unicode10},      '');
169 is($charinfo->{comment},        '');
170 is($charinfo->{upper},          '');
171 is($charinfo->{lower},          '');
172 is($charinfo->{title},          '');
173 is($charinfo->{block},          'Hangul Syllables');
174 is($charinfo->{script},         'Hangul');
175
176 $charinfo = charinfo(0x1D400);
177
178 is($charinfo->{code},           '1D400', 'MATHEMATICAL BOLD CAPITAL A');
179 is($charinfo->{name},           'MATHEMATICAL BOLD CAPITAL A');
180 is($charinfo->{category},       'Lu');
181 is($charinfo->{combining},      '0');
182 is($charinfo->{bidi},           'L');
183 is($charinfo->{decomposition},  '<font> 0041');
184 is($charinfo->{decimal},        '');
185 is($charinfo->{digit},          '');
186 is($charinfo->{numeric},        '');
187 is($charinfo->{mirrored},       'N');
188 is($charinfo->{unicode10},      '');
189 is($charinfo->{comment},        '');
190 is($charinfo->{upper},          '');
191 is($charinfo->{lower},          '');
192 is($charinfo->{title},          '');
193 is($charinfo->{block},          'Mathematical Alphanumeric Symbols');
194 is($charinfo->{script},         'Common');
195
196 $charinfo = charinfo(0x9FBA);   #Bug 58428
197
198 is($charinfo->{code},           '9FBA', 'U+9FBA');
199 is($charinfo->{name},           'CJK UNIFIED IDEOGRAPH-9FBA');
200 is($charinfo->{category},       'Lo');
201 is($charinfo->{combining},      '0');
202 is($charinfo->{bidi},           'L');
203 is($charinfo->{decomposition},  '');
204 is($charinfo->{decimal},        '');
205 is($charinfo->{digit},          '');
206 is($charinfo->{numeric},        '');
207 is($charinfo->{mirrored},       'N');
208 is($charinfo->{unicode10},      '');
209 is($charinfo->{comment},        '');
210 is($charinfo->{upper},          '');
211 is($charinfo->{lower},          '');
212 is($charinfo->{title},          '');
213 is($charinfo->{block},          'CJK Unified Ideographs');
214 is($charinfo->{script},         'Han');
215
216 use Unicode::UCD qw(charblock charscript);
217
218 # 0x0590 is in the Hebrew block but unused.
219
220 is(charblock(0x590),          'Hebrew', '0x0590 - Hebrew unused charblock');
221 is(charscript(0x590),         'Unknown',    '0x0590 - Hebrew unused charscript');
222 is(charblock(0x1FFFF),        'No_Block', '0x1FFFF - unused charblock');
223
224 $charinfo = charinfo(0xbe);
225
226 is($charinfo->{code},           '00BE', 'VULGAR FRACTION THREE QUARTERS');
227 is($charinfo->{name},           'VULGAR FRACTION THREE QUARTERS');
228 is($charinfo->{category},       'No');
229 is($charinfo->{combining},      '0');
230 is($charinfo->{bidi},           'ON');
231 is($charinfo->{decomposition},  '<fraction> 0033 2044 0034');
232 is($charinfo->{decimal},        '');
233 is($charinfo->{digit},          '');
234 is($charinfo->{numeric},        '3/4');
235 is($charinfo->{mirrored},       'N');
236 is($charinfo->{unicode10},      'FRACTION THREE QUARTERS');
237 is($charinfo->{comment},        '');
238 is($charinfo->{upper},          '');
239 is($charinfo->{lower},          '');
240 is($charinfo->{title},          '');
241 is($charinfo->{block},          'Latin-1 Supplement');
242 is($charinfo->{script},         'Common');
243
244 # This is to test a case where both simple and full lowercases exist and
245 # differ
246 $charinfo = charinfo(0x130);
247
248 is($charinfo->{code},           '0130', 'LATIN CAPITAL LETTER I WITH DOT ABOVE');
249 is($charinfo->{name},           'LATIN CAPITAL LETTER I WITH DOT ABOVE');
250 is($charinfo->{category},       'Lu');
251 is($charinfo->{combining},      '0');
252 is($charinfo->{bidi},           'L');
253 is($charinfo->{decomposition},  '0049 0307');
254 is($charinfo->{decimal},        '');
255 is($charinfo->{digit},          '');
256 is($charinfo->{numeric},        '');
257 is($charinfo->{mirrored},       'N');
258 is($charinfo->{unicode10},      'LATIN CAPITAL LETTER I DOT');
259 is($charinfo->{comment},        '');
260 is($charinfo->{upper},          '');
261 is($charinfo->{lower},          '0069');
262 is($charinfo->{title},          '');
263 is($charinfo->{block},          'Latin Extended-A');
264 is($charinfo->{script},         'Latin');
265
266 # This is to test a case where both simple and full uppercases exist and
267 # differ
268 $charinfo = charinfo(0x1F80);
269
270 is($charinfo->{code},           '1F80', 'GREEK SMALL LETTER ALPHA WITH PSILI AND YPOGEGRAMMENI');
271 is($charinfo->{name},           'GREEK SMALL LETTER ALPHA WITH PSILI AND YPOGEGRAMMENI');
272 is($charinfo->{category},       'Ll');
273 is($charinfo->{combining},      '0');
274 is($charinfo->{bidi},           'L');
275 is($charinfo->{decomposition},  '1F00 0345');
276 is($charinfo->{decimal},        '');
277 is($charinfo->{digit},          '');
278 is($charinfo->{numeric},        '');
279 is($charinfo->{mirrored},       'N');
280 is($charinfo->{unicode10},      '');
281 is($charinfo->{comment},        '');
282 is($charinfo->{upper},          '1F88');
283 is($charinfo->{lower},          '');
284 is($charinfo->{title},          '1F88');
285 is($charinfo->{block},          'Greek Extended');
286 is($charinfo->{script},         'Greek');
287
288 use Unicode::UCD qw(charblocks charscripts);
289
290 my $charblocks = charblocks();
291
292 ok(exists $charblocks->{Thai}, 'Thai charblock exists');
293 is($charblocks->{Thai}->[0]->[0], hex('0e00'));
294 ok(!exists $charblocks->{PigLatin}, 'PigLatin charblock does not exist');
295
296 my $charscripts = charscripts();
297
298 ok(exists $charscripts->{Armenian}, 'Armenian charscript exists');
299 is($charscripts->{Armenian}->[0]->[0], hex('0531'));
300 ok(!exists $charscripts->{PigLatin}, 'PigLatin charscript does not exist');
301
302 my $charscript;
303
304 $charscript = charscript("12ab");
305 is($charscript, 'Ethiopic', 'Ethiopic charscript');
306
307 $charscript = charscript("0x12ab");
308 is($charscript, 'Ethiopic');
309
310 $charscript = charscript("U+12ab");
311 is($charscript, 'Ethiopic');
312
313 my $ranges;
314
315 $ranges = charscript('Ogham');
316 is($ranges->[0]->[0], hex('1680'), 'Ogham charscript');
317 is($ranges->[0]->[1], hex('169C'));
318
319 use Unicode::UCD qw(charinrange);
320
321 $ranges = charscript('Cherokee');
322 ok(!charinrange($ranges, "139f"), 'Cherokee charscript');
323 ok( charinrange($ranges, "13a0"));
324 ok( charinrange($ranges, "13f4"));
325 ok(!charinrange($ranges, "13f5"));
326
327 use Unicode::UCD qw(general_categories);
328
329 my $gc = general_categories();
330
331 ok(exists $gc->{L}, 'has L');
332 is($gc->{L}, 'Letter', 'L is Letter');
333 is($gc->{Lu}, 'UppercaseLetter', 'Lu is UppercaseLetter');
334
335 use Unicode::UCD qw(bidi_types);
336
337 my $bt = bidi_types();
338
339 ok(exists $bt->{L}, 'has L');
340 is($bt->{L}, 'Left-to-Right', 'L is Left-to-Right');
341 is($bt->{AL}, 'Right-to-Left Arabic', 'AL is Right-to-Left Arabic');
342
343 # If this fails, then maybe one should look at the Unicode changes to see
344 # what else might need to be updated.
345 is(Unicode::UCD::UnicodeVersion, '6.1.0', 'UnicodeVersion');
346
347 use Unicode::UCD qw(compexcl);
348
349 ok(!compexcl(0x0100), 'compexcl');
350 ok(!compexcl(0xD801), 'compexcl of surrogate');
351 ok(!compexcl(0x110000), 'compexcl of non-Unicode code point');
352 ok( compexcl(0x0958));
353
354 use Unicode::UCD qw(casefold);
355
356 my $casefold;
357
358 $casefold = casefold(0x41);
359
360 is($casefold->{code}, '0041', 'casefold 0x41 code');
361 is($casefold->{status}, 'C', 'casefold 0x41 status');
362 is($casefold->{mapping}, '0061', 'casefold 0x41 mapping');
363 is($casefold->{full}, '0061', 'casefold 0x41 full');
364 is($casefold->{simple}, '0061', 'casefold 0x41 simple');
365 is($casefold->{turkic}, "", 'casefold 0x41 turkic');
366
367 $casefold = casefold(0xdf);
368
369 is($casefold->{code}, '00DF', 'casefold 0xDF code');
370 is($casefold->{status}, 'F', 'casefold 0xDF status');
371 is($casefold->{mapping}, '0073 0073', 'casefold 0xDF mapping');
372 is($casefold->{full}, '0073 0073', 'casefold 0xDF full');
373 is($casefold->{simple}, "", 'casefold 0xDF simple');
374 is($casefold->{turkic}, "", 'casefold 0xDF turkic');
375
376 # Do different tests depending on if version <= 3.1, or not.
377 (my $version = Unicode::UCD::UnicodeVersion) =~ /^(\d+)\.(\d+)/;
378 if (defined $1 && ($1 <= 2 || $1 == 3 && defined $2 && $2 <= 1)) {
379         $casefold = casefold(0x130);
380
381         is($casefold->{code}, '0130', 'casefold 0x130 code');
382         is($casefold->{status}, 'I' , 'casefold 0x130 status');
383         is($casefold->{mapping}, '0069', 'casefold 0x130 mapping');
384         is($casefold->{full}, '0069', 'casefold 0x130 full');
385         is($casefold->{simple}, "0069", 'casefold 0x130 simple');
386         is($casefold->{turkic}, "0069", 'casefold 0x130 turkic');
387
388         $casefold = casefold(0x131);
389
390         is($casefold->{code}, '0131', 'casefold 0x131 code');
391         is($casefold->{status}, 'I' , 'casefold 0x131 status');
392         is($casefold->{mapping}, '0069', 'casefold 0x131 mapping');
393         is($casefold->{full}, '0069', 'casefold 0x131 full');
394         is($casefold->{simple}, "0069", 'casefold 0x131 simple');
395         is($casefold->{turkic}, "0069", 'casefold 0x131 turkic');
396 } else {
397         $casefold = casefold(0x49);
398
399         is($casefold->{code}, '0049', 'casefold 0x49 code');
400         is($casefold->{status}, 'C' , 'casefold 0x49 status');
401         is($casefold->{mapping}, '0069', 'casefold 0x49 mapping');
402         is($casefold->{full}, '0069', 'casefold 0x49 full');
403         is($casefold->{simple}, "0069", 'casefold 0x49 simple');
404         is($casefold->{turkic}, "0131", 'casefold 0x49 turkic');
405
406         $casefold = casefold(0x130);
407
408         is($casefold->{code}, '0130', 'casefold 0x130 code');
409         is($casefold->{status}, 'F' , 'casefold 0x130 status');
410         is($casefold->{mapping}, '0069 0307', 'casefold 0x130 mapping');
411         is($casefold->{full}, '0069 0307', 'casefold 0x130 full');
412         is($casefold->{simple}, "", 'casefold 0x130 simple');
413         is($casefold->{turkic}, "0069", 'casefold 0x130 turkic');
414 }
415
416 $casefold = casefold(0x1F88);
417
418 is($casefold->{code}, '1F88', 'casefold 0x1F88 code');
419 is($casefold->{status}, 'S' , 'casefold 0x1F88 status');
420 is($casefold->{mapping}, '1F80', 'casefold 0x1F88 mapping');
421 is($casefold->{full}, '1F00 03B9', 'casefold 0x1F88 full');
422 is($casefold->{simple}, '1F80', 'casefold 0x1F88 simple');
423 is($casefold->{turkic}, "", 'casefold 0x1F88 turkic');
424
425 ok(!casefold(0x20));
426
427 use Unicode::UCD qw(casespec);
428
429 my $casespec;
430
431 ok(!casespec(0x41));
432
433 $casespec = casespec(0xdf);
434
435 ok($casespec->{code} eq '00DF' &&
436    $casespec->{lower} eq '00DF'  &&
437    $casespec->{title} eq '0053 0073'  &&
438    $casespec->{upper} eq '0053 0053' &&
439    !defined $casespec->{condition}, 'casespec 0xDF');
440
441 $casespec = casespec(0x307);
442
443 ok($casespec->{az}->{code} eq '0307' &&
444    !defined $casespec->{az}->{lower} &&
445    $casespec->{az}->{title} eq '0307'  &&
446    $casespec->{az}->{upper} eq '0307' &&
447    $casespec->{az}->{condition} eq 'az After_I',
448    'casespec 0x307');
449
450 # perl #7305 UnicodeCD::compexcl is weird
451
452 for (1) {my $a=compexcl $_}
453 ok(1, 'compexcl read-only $_: perl #7305');
454 map {compexcl $_} %{{1=>2}};
455 ok(1, 'compexcl read-only hash: perl #7305');
456
457 is(Unicode::UCD::_getcode('123'),     123, "_getcode(123)");
458 is(Unicode::UCD::_getcode('0123'),  0x123, "_getcode(0123)");
459 is(Unicode::UCD::_getcode('0x123'), 0x123, "_getcode(0x123)");
460 is(Unicode::UCD::_getcode('0X123'), 0x123, "_getcode(0X123)");
461 is(Unicode::UCD::_getcode('U+123'), 0x123, "_getcode(U+123)");
462 is(Unicode::UCD::_getcode('u+123'), 0x123, "_getcode(u+123)");
463 is(Unicode::UCD::_getcode('U+1234'),   0x1234, "_getcode(U+1234)");
464 is(Unicode::UCD::_getcode('U+12345'), 0x12345, "_getcode(U+12345)");
465 is(Unicode::UCD::_getcode('123x'),    undef, "_getcode(123x)");
466 is(Unicode::UCD::_getcode('x123'),    undef, "_getcode(x123)");
467 is(Unicode::UCD::_getcode('0x123x'),  undef, "_getcode(x123)");
468 is(Unicode::UCD::_getcode('U+123x'),  undef, "_getcode(x123)");
469
470 {
471     my $r1 = charscript('Latin');
472     my $n1 = @$r1;
473     is($n1, 30, "number of ranges in Latin script (Unicode 6.1.0)");
474     shift @$r1 while @$r1;
475     my $r2 = charscript('Latin');
476     is(@$r2, $n1, "modifying results should not mess up internal caches");
477 }
478
479 {
480         is(charinfo(0xdeadbeef), undef, "[perl #23273] warnings in Unicode::UCD");
481 }
482
483 use Unicode::UCD qw(namedseq);
484
485 is(namedseq("KATAKANA LETTER AINU P"), "\x{31F7}\x{309A}", "namedseq");
486 is(namedseq("KATAKANA LETTER AINU Q"), undef);
487 is(namedseq(), undef);
488 is(namedseq(qw(foo bar)), undef);
489 my @ns = namedseq("KATAKANA LETTER AINU P");
490 is(scalar @ns, 2);
491 is($ns[0], 0x31F7);
492 is($ns[1], 0x309A);
493 my %ns = namedseq();
494 is($ns{"KATAKANA LETTER AINU P"}, "\x{31F7}\x{309A}");
495 @ns = namedseq(42);
496 is(@ns, 0);
497
498 use Unicode::UCD qw(num);
499 use charnames ":full";
500
501 is(num("0"), 0, 'Verify num("0") == 0');
502 is(num("98765"), 98765, 'Verify num("98765") == 98765');
503 ok(! defined num("98765\N{FULLWIDTH DIGIT FOUR}"), 'Verify num("98765\N{FULLWIDTH DIGIT FOUR}") isnt defined');
504 is(num("\N{NEW TAI LUE DIGIT TWO}\N{NEW TAI LUE DIGIT ONE}"), 21, 'Verify num("\N{NEW TAI LUE DIGIT TWO}\N{NEW TAI LUE DIGIT ONE}") == 21');
505 ok(! defined num("\N{NEW TAI LUE DIGIT TWO}\N{NEW TAI LUE THAM DIGIT ONE}"), 'Verify num("\N{NEW TAI LUE DIGIT TWO}\N{NEW TAI LUE THAM DIGIT ONE}") isnt defined');
506 is(num("\N{CHAM DIGIT ZERO}\N{CHAM DIGIT THREE}"), 3, 'Verify num("\N{CHAM DIGIT ZERO}\N{CHAM DIGIT THREE}") == 3');
507 ok(! defined num("\N{CHAM DIGIT ZERO}\N{JAVANESE DIGIT NINE}"), 'Verify num("\N{CHAM DIGIT ZERO}\N{JAVANESE DIGIT NINE}") isnt defined');
508 is(num("\N{SUPERSCRIPT TWO}"), 2, 'Verify num("\N{SUPERSCRIPT TWO} == 2');
509 is(num("\N{ETHIOPIC NUMBER TEN THOUSAND}"), 10000, 'Verify num("\N{ETHIOPIC NUMBER TEN THOUSAND}") == 10000');
510 is(num("\N{NORTH INDIC FRACTION ONE HALF}"), .5, 'Verify num("\N{NORTH INDIC FRACTION ONE HALF}") == .5');
511 is(num("\N{U+12448}"), 9, 'Verify num("\N{U+12448}") == 9');
512
513 # Create a user-defined property
514 sub InKana {<<'END'}
515 3040    309F
516 30A0    30FF
517 END
518
519 use Unicode::UCD qw(prop_aliases);
520
521 is(prop_aliases(undef), undef, "prop_aliases(undef) returns <undef>");
522 is(prop_aliases("unknown property"), undef,
523                 "prop_aliases(<unknown property>) returns <undef>");
524 is(prop_aliases("InKana"), undef,
525                 "prop_aliases(<user-defined property>) returns <undef>");
526 is(prop_aliases("Perl_Decomposition_Mapping"), undef, "prop_aliases('Perl_Decomposition_Mapping') returns <undef> since internal-Perl-only");
527 is(prop_aliases("Perl_Charnames"), undef,
528     "prop_aliases('Perl_Charnames') returns <undef> since internal-Perl-only");
529 is(prop_aliases("isgc"), undef,
530     "prop_aliases('isgc') returns <undef> since is not covered Perl extension");
531 is(prop_aliases("Is_Is_Any"), undef,
532                 "prop_aliases('Is_Is_Any') returns <undef> since two is's");
533
534 require 'utf8_heavy.pl';
535 require "unicore/Heavy.pl";
536
537 # Keys are lists of properties. Values are defined if have been tested.
538 my %props;
539
540 # To test for loose matching, add in the characters that are ignored there.
541 my $extra_chars = "-_ ";
542
543 # The one internal property we accept
544 $props{'Perl_Decimal_Digit'} = 1;
545 my @list = prop_aliases("perldecimaldigit");
546 is_deeply(\@list,
547           [ "Perl_Decimal_Digit",
548             "Perl_Decimal_Digit"
549           ], "prop_aliases('perldecimaldigit') returns Perl_Decimal_Digit as both short and full names");
550
551 # Get the official Unicode property name synonyms and test them.
552 open my $props, "<", "../lib/unicore/PropertyAliases.txt"
553                 or die "Can't open Unicode PropertyAliases.txt";
554 $/ = "\n";
555 while (<$props>) {
556     s/\s*#.*//;           # Remove comments
557     next if /^\s* $/x;    # Ignore empty and comment lines
558
559     chomp;
560     my $count = 0;  # 0th field in line is short name; 1th is long name
561     my $short_name;
562     my $full_name;
563     my @names_via_short;
564     foreach my $alias (split /\s*;\s*/) {    # Fields are separated by
565                                              # semi-colons
566         # Add in the characters that are supposed to be ignored, to test loose
567         # matching, which the tested function does on all inputs.
568         my $mod_name = "$extra_chars$alias";
569
570         my $loose = &utf8::_loose_name(lc $alias);
571
572         # Indicate we have tested this.
573         $props{$loose} = 1;
574
575         my @all_names = prop_aliases($mod_name);
576         if (grep { $_ eq $loose } @Unicode::UCD::suppressed_properties) {
577             is(@all_names, 0, "prop_aliases('$mod_name') returns undef since $alias is not installed");
578             next;
579         }
580         elsif (! @all_names) {
581             fail("prop_aliases('$mod_name')");
582             diag("'$alias' is unknown to prop_aliases()");
583             next;
584         }
585
586         if ($count == 0) {  # Is short name
587
588             @names_via_short = prop_aliases($mod_name);
589
590             # If the 0th test fails, no sense in continuing with the others
591             last unless is($names_via_short[0], $alias,
592                     "prop_aliases: '$alias' is the short name for '$mod_name'");
593             $short_name = $alias;
594         }
595         elsif ($count == 1) {   # Is full name
596
597             # Some properties have the same short and full name; no sense
598             # repeating the test if the same.
599             if ($alias ne $short_name) {
600                 my @names_via_full = prop_aliases($mod_name);
601                 is_deeply(\@names_via_full, \@names_via_short, "prop_aliases() returns the same list for both '$short_name' and '$mod_name'");
602             }
603
604             # Tests scalar context
605             is(prop_aliases($short_name), $alias,
606                 "prop_aliases: '$alias' is the long name for '$short_name'");
607         }
608         else {  # Is another alias
609             is_deeply(\@all_names, \@names_via_short, "prop_aliases() returns the same list for both '$short_name' and '$mod_name'");
610             ok((grep { $_ =~ /^$alias$/i } @all_names),
611                 "prop_aliases: '$alias' is listed as an alias for '$mod_name'");
612         }
613
614         $count++;
615     }
616 }
617
618 # Now test anything we can find that wasn't covered by the tests of the
619 # official properties.  We have no way of knowing if mktables omitted a Perl
620 # extension or not, but we do the best we can from its generated lists
621
622 foreach my $alias (keys %utf8::loose_to_file_of) {
623     next if $alias =~ /=/;
624     my $lc_name = lc $alias;
625     my $loose = &utf8::_loose_name($lc_name);
626     next if exists $props{$loose};  # Skip if already tested
627     $props{$loose} = 1;
628     my $mod_name = "$extra_chars$alias";    # Tests loose matching
629     my @aliases = prop_aliases($mod_name);
630     my $found_it = grep { &utf8::_loose_name(lc $_) eq $lc_name } @aliases;
631     if ($found_it) {
632         pass("prop_aliases: '$lc_name' is listed as an alias for '$mod_name'");
633     }
634     elsif ($lc_name =~ /l[_&]$/) {
635
636         # These two names are special in that they don't appear in the
637         # returned list because they are discouraged from use.  Verify
638         # that they return the same list as a non-discouraged version.
639         my @LC = prop_aliases('Is_LC');
640         is_deeply(\@aliases, \@LC, "prop_aliases: '$lc_name' returns the same list as 'Is_LC'");
641     }
642     else {
643         my $stripped = $lc_name =~ s/^is//;
644
645         # Could be that the input includes a prefix 'is', which is rarely
646         # returned as an alias, so having successfully stripped it off above,
647         # try again.
648         if ($stripped) {
649             $found_it = grep { &utf8::_loose_name(lc $_) eq $lc_name } @aliases;
650         }
651
652         # If that didn't work, it could be that it's a block, which is always
653         # returned with a leading 'In_' to avoid ambiguity.  Try comparing
654         # with that stripped off.
655         if (! $found_it) {
656             $found_it = grep { &utf8::_loose_name(s/^In_(.*)/\L$1/r) eq $lc_name }
657                               @aliases;
658             # Could check that is a real block, but tests for invmap will
659             # likely pickup any errors, since this will be tested there.
660             $lc_name = "in$lc_name" if $found_it;   # Change for message below
661         }
662         my $message = "prop_aliases: '$lc_name' is listed as an alias for '$mod_name'";
663         ($found_it) ? pass($message) : fail($message);
664     }
665 }
666
667 my $done_equals = 0;
668 foreach my $alias (keys %utf8::stricter_to_file_of) {
669     if ($alias =~ /=/) {    # Only test one case where there is an equals
670         next if $done_equals;
671         $done_equals = 1;
672     }
673     my $lc_name = lc $alias;
674     my @list = prop_aliases($alias);
675     if ($alias =~ /^_/) {
676         is(@list, 0, "prop_aliases: '$lc_name' returns an empty list since it is internal_only");
677     }
678     elsif ($alias =~ /=/) {
679         is(@list, 0, "prop_aliases: '$lc_name' returns an empty list since is illegal property name");
680     }
681     else {
682         ok((grep { lc $_ eq $lc_name } @list),
683                 "prop_aliases: '$lc_name' is listed as an alias for '$alias'");
684     }
685 }
686
687 use Unicode::UCD qw(prop_value_aliases);
688
689 is(prop_value_aliases("unknown property", "unknown value"), undef,
690     "prop_value_aliases(<unknown property>, <unknown value>) returns <undef>");
691 is(prop_value_aliases(undef, undef), undef,
692                            "prop_value_aliases(undef, undef) returns <undef>");
693 is((prop_value_aliases("na", "A")), "A", "test that prop_value_aliases returns its input for properties that don't have synonyms");
694 is(prop_value_aliases("isgc", "C"), undef, "prop_value_aliases('isgc', 'C') returns <undef> since is not covered Perl extension");
695 is(prop_value_aliases("gc", "isC"), undef, "prop_value_aliases('gc', 'isC') returns <undef> since is not covered Perl extension");
696
697 # We have no way of knowing if mktables omitted a Perl extension that it
698 # shouldn't have, but we can check if it omitted an official Unicode property
699 # name synonym.  And for those, we can check if the short and full names are
700 # correct.
701
702 my %pva_tested;   # List of things already tested.
703 open my $propvalues, "<", "../lib/unicore/PropValueAliases.txt"
704      or die "Can't open Unicode PropValueAliases.txt";
705 while (<$propvalues>) {
706     s/\s*#.*//;           # Remove comments
707     next if /^\s* $/x;    # Ignore empty and comment lines
708     chomp;
709
710     my @fields = split /\s*;\s*/; # Fields are separated by semi-colons
711     my $prop = shift @fields;   # 0th field is the property,
712     my $count = 0;  # 0th field in line (after shifting off the property) is
713                     # short name; 1th is long name
714     my $short_name;
715     my @names_via_short;    # Saves the values between iterations
716
717     # The property on the lhs of the = is always loosely matched.  Add in
718     # characters that are ignored under loose matching to test that
719     my $mod_prop = "$extra_chars$prop";
720
721     if ($fields[0] eq 'n/a') {  # See comments in input file, essentially
722                                 # means full name and short name are identical
723         $fields[0] = $fields[1];
724     }
725     elsif ($fields[0] ne $fields[1]
726            && &utf8::_loose_name(lc $fields[0])
727                eq &utf8::_loose_name(lc $fields[1])
728            && $fields[1] !~ /[[:upper:]]/)
729     {
730         # Also, there is a bug in the file in which "n/a" is omitted, and
731         # the two fields are identical except for case, and the full name
732         # is all lower case.  Copy the "short" name unto the full one to
733         # give it some upper case.
734
735         $fields[1] = $fields[0];
736     }
737
738     # The ccc property in the file is special; has an extra numeric field
739     # (0th), which should go at the end, since we use the next two fields as
740     # the short and full names, respectively.  See comments in input file.
741     splice (@fields, 0, 0, splice(@fields, 1, 2)) if $prop eq 'ccc';
742
743     my $loose_prop = &utf8::_loose_name(lc $prop);
744     my $suppressed = grep { $_ eq $loose_prop }
745                           @Unicode::UCD::suppressed_properties;
746     foreach my $value (@fields) {
747         if ($suppressed) {
748             is(prop_value_aliases($prop, $value), undef, "prop_value_aliases('$prop', '$value') returns undef for suppressed property $prop");
749             next;
750         }
751         elsif (grep { $_ eq ("$loose_prop=" . &utf8::_loose_name(lc $value)) } @Unicode::UCD::suppressed_properties) {
752             is(prop_value_aliases($prop, $value), undef, "prop_value_aliases('$prop', '$value') returns undef for suppressed property $prop=$value");
753             next;
754         }
755
756         # Add in test for loose matching.
757         my $mod_value = "$extra_chars$value";
758
759         # If the value is a number, optionally negative, including a floating
760         # point or rational numer, it should be only strictly matched, so the
761         # loose matching should fail.
762         if ($value =~ / ^ -? \d+ (?: [\/.] \d+ )? $ /x) {
763             is(prop_value_aliases($mod_prop, $mod_value), undef, "prop_value_aliases('$mod_prop', '$mod_value') returns undef because '$mod_value' should be strictly matched");
764
765             # And reset so below tests just the strict matching.
766             $mod_value = $value;
767         }
768
769         if ($count == 0) {
770
771             @names_via_short = prop_value_aliases($mod_prop, $mod_value);
772
773             # If the 0th test fails, no sense in continuing with the others
774             last unless is($names_via_short[0], $value, "prop_value_aliases: In '$prop', '$value' is the short name for '$mod_value'");
775             $short_name = $value;
776         }
777         elsif ($count == 1) {
778
779             # Some properties have the same short and full name; no sense
780             # repeating the test if the same.
781             if ($value ne $short_name) {
782                 my @names_via_full =
783                             prop_value_aliases($mod_prop, $mod_value);
784                 is_deeply(\@names_via_full, \@names_via_short, "In '$prop', prop_value_aliases() returns the same list for both '$short_name' and '$mod_value'");
785             }
786
787             # Tests scalar context
788             is(prop_value_aliases($prop, $short_name), $value, "'$value' is the long name for prop_value_aliases('$prop', '$short_name')");
789         }
790         else {
791             my @all_names = prop_value_aliases($mod_prop, $mod_value);
792             is_deeply(\@all_names, \@names_via_short, "In '$prop', prop_value_aliases() returns the same list for both '$short_name' and '$mod_value'");
793             ok((grep { &utf8::_loose_name(lc $_) eq &utf8::_loose_name(lc $value) } prop_value_aliases($prop, $short_name)), "'$value' is listed as an alias for prop_value_aliases('$prop', '$short_name')");
794         }
795
796         $pva_tested{&utf8::_loose_name(lc $prop) . "=" . &utf8::_loose_name(lc $value)} = 1;
797         $count++;
798     }
799 }
800
801 # And test as best we can, the non-official pva's that mktables generates.
802 foreach my $hash (\%utf8::loose_to_file_of, \%utf8::stricter_to_file_of) {
803     foreach my $test (keys %$hash) {
804         next if exists $pva_tested{$test};  # Skip if already tested
805
806         my ($prop, $value) = split "=", $test;
807         next unless defined $value; # prop_value_aliases() requires an input
808                                     # 'value'
809         my $mod_value;
810         if ($hash == \%utf8::loose_to_file_of) {
811
812             # Add extra characters to test loose-match rhs value
813             $mod_value = "$extra_chars$value";
814         }
815         else { # Here value is strictly matched.
816
817             # Extra elements are added by mktables to this hash so that
818             # something like "age=6.0" has a synonym of "age=6".  It's not
819             # clear to me (khw) if we should be encouraging those synonyms, so
820             # don't test for them.
821             next if $value !~ /\D/ && exists $hash->{"$prop=$value.0"};
822
823             # Verify that loose matching fails when only strict is called for.
824             next unless is(prop_value_aliases($prop, "$extra_chars$value"), undef,
825                         "prop_value_aliases('$prop', '$extra_chars$value') returns undef since '$value' should be strictly matched"),
826
827             # Strict matching does allow for underscores between digits.  Test
828             # for that.
829             $mod_value = $value;
830             while ($mod_value =~ s/(\d)(\d)/$1_$2/g) {}
831         }
832
833         # The lhs property is always loosely matched, so add in extra
834         # characters to test that.
835         my $mod_prop = "$extra_chars$prop";
836
837         if ($prop eq 'gc' && $value =~ /l[_&]$/) {
838             # These two names are special in that they don't appear in the
839             # returned list because they are discouraged from use.  Verify
840             # that they return the same list as a non-discouraged version.
841             my @LC = prop_value_aliases('gc', 'lc');
842             my @l_ = prop_value_aliases($mod_prop, $mod_value);
843             is_deeply(\@l_, \@LC, "prop_value_aliases('$mod_prop', '$mod_value) returns the same list as prop_value_aliases('gc', 'lc')");
844         }
845         else {
846             ok((grep { &utf8::_loose_name(lc $_) eq &utf8::_loose_name(lc $value) }
847                 prop_value_aliases($mod_prop, $mod_value)),
848                 "'$value' is listed as an alias for prop_value_aliases('$mod_prop', '$mod_value')");
849         }
850     }
851 }
852
853 undef %pva_tested;
854
855 no warnings 'once'; # We use some values once from 'required' modules.
856
857 use Unicode::UCD qw(prop_invlist prop_invmap MAX_CP);
858
859 # There were some problems with caching interfering with prop_invlist() vs
860 # prop_invmap() on binary properties, and also between the 3 properties where
861 # Perl used the same 'To' name as another property (see utf8_heavy.pl).
862 # So, before testing all of prop_invlist(),
863 #   1)  call prop_invmap() to try both orders of these name issues.  This uses
864 #       up two of the 3 properties;  the third will be left so that invlist()
865 #       on it gets called before invmap()
866 #   2)  call prop_invmap() on a generic binary property, ahead of invlist().
867 # This should test that the caching works in both directions.
868
869 # These properties are not stable between Unicode versions, but the first few
870 # elements are; just look at the first element to see if are getting the
871 # distinction right.  The general inversion map testing below will test the
872 # whole thing.
873 my $prop = "uc";
874 my ($invlist_ref, $invmap_ref, $format, $missing) = prop_invmap($prop);
875 is($format, 'cl', "prop_invmap() format of '$prop' is 'cl'");
876 is($missing, '0', "prop_invmap() missing of '$prop' is '0'");
877 is($invlist_ref->[1], 0x61, "prop_invmap('$prop') list[1] is 0x61");
878 is($invmap_ref->[1], -32, "prop_invmap('$prop') map[1] is -32");
879
880 $prop = "upper";
881 ($invlist_ref, $invmap_ref, $format, $missing) = prop_invmap($prop);
882 is($format, 's', "prop_invmap() format of '$prop' is 'cl'");
883 is($missing, 'N', "prop_invmap() missing of '$prop' is '<code point>'");
884 is($invlist_ref->[1], 0x41, "prop_invmap('$prop') list[1] is 0x41");
885 is($invmap_ref->[1], 'Y', "prop_invmap('$prop') map[1] is 'Y'");
886
887 $prop = "lower";
888 ($invlist_ref, $invmap_ref, $format, $missing) = prop_invmap($prop);
889 is($format, 's', "prop_invmap() format of '$prop' is 'cl'");
890 is($missing, 'N', "prop_invmap() missing of '$prop' is '<code point>'");
891 is($invlist_ref->[1], 0x61, "prop_invmap('$prop') list[1] is 0x61");
892 is($invmap_ref->[1], 'Y', "prop_invmap('$prop') map[1] is 'Y'");
893
894 $prop = "lc";
895 ($invlist_ref, $invmap_ref, $format, $missing) = prop_invmap($prop);
896 is($format, 'cl', "prop_invmap() format of '$prop' is 'cl'");
897 is($missing, '0', "prop_invmap() missing of '$prop' is '0'");
898 is($invlist_ref->[1], 0x41, "prop_invmap('$prop') list[1] is 0x41");
899 is($invmap_ref->[1], 32, "prop_invmap('$prop') map[1] is 32");
900
901 # This property is stable and small, so can test all of it
902 $prop = "ASCII_Hex_Digit";
903 ($invlist_ref, $invmap_ref, $format, $missing) = prop_invmap($prop);
904 is($format, 's', "prop_invmap() format of '$prop' is 's'");
905 is($missing, 'N', "prop_invmap() missing of '$prop' is 'N'");
906 is_deeply($invlist_ref, [ 0x0000, 0x0030, 0x003A, 0x0041,
907                           0x0047, 0x0061, 0x0067, 0x110000 ],
908           "prop_invmap('$prop') code point list is correct");
909 is_deeply($invmap_ref, [ 'N', 'Y', 'N', 'Y', 'N', 'Y', 'N', 'N' ] ,
910           "prop_invmap('$prop') map list is correct");
911
912 is(prop_invlist("Unknown property"), undef, "prop_invlist(<Unknown property>) returns undef");
913 is(prop_invlist(undef), undef, "prop_invlist(undef) returns undef");
914 is(prop_invlist("Any"), 2, "prop_invlist('Any') returns the number of elements in scalar context");
915 my @invlist = prop_invlist("Is_Any");
916 is_deeply(\@invlist, [ 0, 0x110000 ], "prop_invlist works on 'Is_' prefixes");
917 is(prop_invlist("Is_Is_Any"), undef, "prop_invlist('Is_Is_Any') returns <undef> since two is's");
918
919 use Storable qw(dclone);
920
921 is(prop_invlist("InKana"), undef, "prop_invlist(<user-defined property returns undef>)");
922
923 # The way both the tests for invlist and invmap work is that they take the
924 # lists returned by the functions and construct from them what the original
925 # file should look like, which are then compared with the file.  If they are
926 # identical, the test passes.  What this tests isn't that the results are
927 # correct, but that invlist and invmap haven't introduced errors beyond what
928 # are there in the files.  As a small hedge against that, test some
929 # prop_invlist() tables fully with the known correct result.  We choose
930 # ASCII_Hex_Digit again, as it is stable.
931 @invlist = prop_invlist("AHex");
932 is_deeply(\@invlist, [ 0x0030, 0x003A, 0x0041,
933                                  0x0047, 0x0061, 0x0067 ],
934           "prop_invlist('AHex') is exactly the expected set of points");
935 @invlist = prop_invlist("AHex=f");
936 is_deeply(\@invlist, [ 0x0000, 0x0030, 0x003A, 0x0041,
937                                  0x0047, 0x0061, 0x0067 ],
938           "prop_invlist('AHex=f') is exactly the expected set of points");
939
940 sub fail_with_diff ($$$$) {
941     # For use below to output better messages
942     my ($prop, $official, $constructed, $tested_function_name) = @_;
943
944     is($constructed, $official, "$tested_function_name('$prop')");
945     diag("Comment out lines " . (__LINE__ - 1) . " through " . (__LINE__ + 1) . " in '$0' on Un*x-like systems to see just the differences.  Uses the 'diff' first in your \$PATH");
946     return;
947
948     fail("$tested_function_name('$prop')");
949
950     require File::Temp;
951     my $off = File::Temp->new();
952     chomp $official;
953     print $off $official, "\n";
954     close $off || die "Can't close official";
955
956     chomp $constructed;
957     my $gend = File::Temp->new();
958     print $gend $constructed, "\n";
959     close $gend || die "Can't close gend";
960
961     my $diff = File::Temp->new();
962     system("diff $off $gend > $diff");
963
964     open my $fh, "<", $diff || die "Can't open $diff";
965     my @diffs = <$fh>;
966     diag("In the diff output below '<' marks lines from the filesystem tables;\n'>' are from $tested_function_name()");
967     diag(@diffs);
968 }
969
970 my %tested_invlist;
971
972 # Look at everything we think that mktables tells us exists, both loose and
973 # strict
974 foreach my $set_of_tables (\%utf8::stricter_to_file_of, \%utf8::loose_to_file_of)
975 {
976     foreach my $table (keys %$set_of_tables) {
977
978         my $mod_table;
979         my ($prop_only, $value) = split "=", $table;
980         if (defined $value) {
981
982             # If this is to be loose matched, add in characters to test that.
983             if ($set_of_tables == \%utf8::loose_to_file_of) {
984                 $value = "$extra_chars$value";
985             }
986             else {  # Strict match
987
988                 # Verify that loose matching fails when only strict is called
989                 # for.
990                 next unless is(prop_invlist("$prop_only=$extra_chars$value"), undef, "prop_invlist('$prop_only=$extra_chars$value') returns undef since should be strictly matched");
991
992                 # Strict matching does allow for underscores between digits.
993                 # Test for that.
994                 while ($value =~ s/(\d)(\d)/$1_$2/g) {}
995             }
996
997             # The property portion in compound form specifications always
998             # matches loosely
999             $mod_table = "$extra_chars$prop_only = $value";
1000         }
1001         else {  # Single-form.
1002
1003             # Like above, use looose if required, and insert underscores
1004             # between digits if strict.
1005             if ($set_of_tables == \%utf8::loose_to_file_of) {
1006                 $mod_table = "$extra_chars$table";
1007             }
1008             else {
1009                 $mod_table = $table;
1010                 while ($mod_table =~ s/(\d)(\d)/$1_$2/g) {}
1011             }
1012         }
1013
1014         my @tested = prop_invlist($mod_table);
1015         if ($table =~ /^_/) {
1016             is(@tested, 0, "prop_invlist('$mod_table') returns an empty list since is internal-only");
1017             next;
1018         }
1019
1020         # If we have already tested a property that uses the same file, this
1021         # list should be identical to the one that was tested, and can bypass
1022         # everything else.
1023         my $file = $set_of_tables->{$table};
1024         if (exists $tested_invlist{$file}) {
1025             is_deeply(\@tested, $tested_invlist{$file}, "prop_invlist('$mod_table') gave same results as its name synonym");
1026             next;
1027         }
1028         $tested_invlist{$file} = dclone \@tested;
1029
1030         # A leading '!' in the file name means that it is to be inverted.
1031         my $invert = $file =~ s/^!//;
1032         my $official = do "unicore/lib/$file.pl";
1033
1034         # Get rid of any trailing space and comments in the file.
1035         $official =~ s/\s*(#.*)?$//mg;
1036         chomp $official;
1037
1038         # If we are to test against an inverted file, it is easier to invert
1039         # our array than the file.
1040         # The file only is valid for Unicode code points, while the inversion
1041         # list is valid for all possible code points.  Therefore, we must test
1042         # just the Unicode part against the file.  Later we will test for
1043         # the non-Unicode part.
1044
1045         my $before_invert;  # Saves the pre-inverted table.
1046         if ($invert) {
1047             $before_invert = dclone \@tested;
1048             if (@tested && $tested[0] == 0) {
1049                 shift @tested;
1050             } else {
1051                 unshift @tested, 0;
1052             }
1053             if (@tested && $tested[-1] == 0x110000) {
1054                 pop @tested;
1055             }
1056             else {
1057                 push @tested, 0x110000;
1058             }
1059         }
1060
1061         # Now construct a string from the list that should match the file.
1062         # The file gives ranges of code points with starting and ending values
1063         # in hex, like this:
1064         # 0041\t005A
1065         # 0061\t007A
1066         # 00AA
1067         # Our list has even numbered elements start ranges that are in the
1068         # list, and odd ones that aren't in the list.  Therefore the odd
1069         # numbered ones are one beyond the end of the previous range, but
1070         # otherwise don't get reflected in the file.
1071         my $tested = "";
1072         my $i = 0;
1073         for (; $i < @tested - 1; $i += 2) {
1074             my $start = $tested[$i];
1075             my $end = $tested[$i+1] - 1;
1076             if ($start == $end) {
1077                 $tested .= sprintf("%04X\n", $start);
1078             }
1079             else {
1080                 $tested .= sprintf "%04X\t%04X\n", $start, $end;
1081             }
1082         }
1083
1084         # As mentioned earlier, the disk files only go up through Unicode,
1085         # whereas the prop_invlist() ones go as high as necessary.  The
1086         # comparison is only valid through max Unicode.
1087         if ($i == @tested - 1 && $tested[$i] <= 0x10FFFF) {
1088             $tested .= sprintf("%04X\t10FFFF\n", $tested[$i]);
1089         }
1090         chomp $tested;
1091         if ($tested ne $official) {
1092             fail_with_diff($mod_table, $official, $tested, "prop_invlist");
1093             next;
1094         }
1095
1096         # Here, it matched the table.  Now need to check for if it is correct
1097         # for beyond Unicode.  First, calculate if is the default table or
1098         # not.  This is the same algorithm as used internally in
1099         # prop_invlist(), so if it is wrong there, this test won't catch it.
1100         my $prop = lc $table;
1101         ($prop_only, $table) = split /\s*[:=]\s*/, $prop;
1102         if (defined $table) {
1103
1104             # May have optional prefixed 'is'
1105             $prop = &utf8::_loose_name($prop_only) =~ s/^is//r;
1106             $prop = $utf8::loose_property_name_of{$prop};
1107             $prop .= "=" . &utf8::_loose_name($table);
1108         }
1109         else {
1110             $prop = &utf8::_loose_name($prop);
1111         }
1112         my $is_default = exists $Unicode::UCD::loose_defaults{$prop};
1113
1114         @tested = @$before_invert if $invert;    # Use the original
1115         if (@tested % 2 == 0) {
1116
1117             # If there are an even number of elements, the final one starts a
1118             # range (going to infinity) of code points that are not in the
1119             # list.
1120             if ($is_default) {
1121                 fail("prop_invlist('$mod_table')");
1122                 diag("default table doesn't goto infinity");
1123                 use Data::Dumper;
1124                 diag Dumper \@tested;
1125                 next;
1126             }
1127         }
1128         else {
1129             # An odd number of elements means the final one starts a range
1130             # (going to infinity of code points that are in the list.
1131             if (! $is_default) {
1132                 fail("prop_invlist('$mod_table')");
1133                 diag("non-default table needs to stop in the Unicode range");
1134                 use Data::Dumper;
1135                 diag Dumper \@tested;
1136                 next;
1137             }
1138         }
1139
1140         pass("prop_invlist('$mod_table')");
1141     }
1142 }
1143
1144 # Now test prop_invmap().
1145
1146 @list = prop_invmap("Unknown property");
1147 is (@list, 0, "prop_invmap(<Unknown property>) returns an empty list");
1148 @list = prop_invmap(undef);
1149 is (@list, 0, "prop_invmap(undef) returns an empty list");
1150 ok (! eval "prop_invmap('gc')" && $@ ne "",
1151                                 "prop_invmap('gc') dies in scalar context");
1152 @list = prop_invmap("_X_Begin");
1153 is (@list, 0, "prop_invmap(<internal property>) returns an empty list");
1154 @list = prop_invmap("InKana");
1155 is(@list, 0, "prop_invmap(<user-defined property returns undef>)");
1156 @list = prop_invmap("Perl_Decomposition_Mapping"), undef,
1157 is(@list, 0, "prop_invmap('Perl_Decomposition_Mapping') returns <undef> since internal-Perl-only");
1158 @list = prop_invmap("Perl_Charnames"), undef,
1159 is(@list, 0, "prop_invmap('Perl_Charnames') returns <undef> since internal-Perl-only");
1160 @list = prop_invmap("Is_Is_Any");
1161 is(@list, 0, "prop_invmap('Is_Is_Any') returns <undef> since two is's");
1162
1163 # The set of properties to test on has already been compiled into %props by
1164 # the prop_aliases() tests.
1165
1166 my %tested_invmaps;
1167
1168 # Like prop_invlist(), prop_invmap() is tested by comparing the results
1169 # returned by the function with the tables that mktables generates.  Some of
1170 # these tables are directly stored as files on disk, in either the unicore or
1171 # unicore/To directories, and most should be listed in the mktables generated
1172 # hash %utf8::loose_property_to_file_of, with a few additional ones that this
1173 # handles specially.  For these, the files are read in directly, massaged, and
1174 # compared with what invmap() returns.  The SPECIALS hash in some of these
1175 # files overrides values in the main part of the file.
1176 #
1177 # The other properties are tested indirectly by generating all the possible
1178 # inversion lists for the property, and seeing if those match the inversion
1179 # lists returned by prop_invlist(), which has already been tested.
1180
1181 PROPERTY:
1182 foreach my $prop (keys %props) {
1183     my $loose_prop = &utf8::_loose_name(lc $prop);
1184     my $suppressed = grep { $_ eq $loose_prop }
1185                           @Unicode::UCD::suppressed_properties;
1186
1187     # Find the short and full names that this property goes by
1188     my ($name, $full_name) = prop_aliases($prop);
1189     if (! $name) {
1190         if (! $suppressed) {
1191             fail("prop_invmap('$prop')");
1192             diag("is unknown to prop_aliases(), and we need it in order to test prop_invmap");
1193         }
1194         next PROPERTY;
1195     }
1196
1197     # Normalize the short name, as it is stored in the hashes under the
1198     # normalized version.
1199     $name = &utf8::_loose_name(lc $name);
1200
1201     # Add in the characters that are supposed to be ignored to test loose
1202     # matching, which the tested function applies to all properties
1203     my $mod_prop = "$extra_chars$prop";
1204
1205     my ($invlist_ref, $invmap_ref, $format, $missing) = prop_invmap($mod_prop);
1206     my $return_ref = [ $invlist_ref, $invmap_ref, $format, $missing ];
1207
1208     # If have already tested this property under a different name, merely
1209     # compare the return from now with the saved one from before.
1210     if (exists $tested_invmaps{$name}) {
1211         is_deeply($return_ref, $tested_invmaps{$name}, "prop_invmap('$mod_prop') gave same results as its synonym, '$name'");
1212         next PROPERTY;
1213     }
1214     $tested_invmaps{$name} = dclone $return_ref;
1215
1216     # If prop_invmap() returned nothing, is ok iff is a property whose file is
1217     # not generated.
1218     if ($suppressed) {
1219         if (defined $format) {
1220             fail("prop_invmap('$mod_prop')");
1221             diag("did not return undef for suppressed property $prop");
1222         }
1223         next PROPERTY;
1224     }
1225     elsif (!defined $format) {
1226         fail("prop_invmap('$mod_prop')");
1227         diag("'$prop' is unknown to prop_invmap()");
1228         next PROPERTY;
1229     }
1230
1231     # The two parallel arrays must have the same number of elements.
1232     if (@$invlist_ref != @$invmap_ref) {
1233         fail("prop_invmap('$mod_prop')");
1234         diag("invlist has "
1235              . scalar @$invlist_ref
1236              . " while invmap has "
1237              . scalar @$invmap_ref
1238              . " elements");
1239         next PROPERTY;
1240     }
1241
1242     # The last element must be for the above-Unicode code points, and must be
1243     # for the default value.
1244     if ($invlist_ref->[-1] != 0x110000) {
1245         fail("prop_invmap('$mod_prop')");
1246         diag("The last inversion list element is not 0x110000");
1247         next PROPERTY;
1248     }
1249     if ($invmap_ref->[-1] ne $missing) {
1250         fail("prop_invmap('$mod_prop')");
1251         diag("The last inversion list element is '$invmap_ref->[-1]', and should be '$missing'");
1252         next PROPERTY;
1253     }
1254
1255     if ($name eq 'bmg') {   # This one has an atypical $missing
1256         if ($missing ne "") {
1257             fail("prop_invmap('$mod_prop')");
1258             diag("The missings should be \"\"; got '$missing'");
1259             next PROPERTY;
1260         }
1261     }
1262     elsif ($format =~ /^ c /x) {
1263         if ($full_name eq 'Perl_Decimal_Digit') {
1264             if ($missing ne "") {
1265                 fail("prop_invmap('$mod_prop')");
1266                 diag("The missings should be \"\"; got '$missing'");
1267                 next PROPERTY;
1268             }
1269         }
1270         elsif ($missing ne "0") {
1271             fail("prop_invmap('$mod_prop')");
1272             diag("The missings should be '0'; got '$missing'");
1273             next PROPERTY;
1274         }
1275     }
1276     elsif ($format =~ /^ d /x) {
1277         if ($missing ne "0") {
1278             fail("prop_invmap('$mod_prop')");
1279             diag("The missings should be '<code point>'; got '$missing'");
1280             next PROPERTY;
1281         }
1282     }
1283     elsif ($missing =~ /[<>]/) {
1284         fail("prop_invmap('$mod_prop')");
1285         diag("The missings should NOT be something with <...>'");
1286         next PROPERTY;
1287
1288         # I don't want to hard code in what all the missings should be, so
1289         # those don't get fully tested.
1290     }
1291
1292     # Certain properties don't have their own files, but must be constructed
1293     # using proxies.
1294     my $proxy_prop = $name;
1295     if ($full_name eq 'Present_In') {
1296         $proxy_prop = "age";    # The maps for these two props are identical
1297     }
1298     elsif ($full_name eq 'Simple_Case_Folding'
1299            || $full_name =~ /Simple_ (.) .*? case_Mapping  /x)
1300     {
1301         if ($full_name eq 'Simple_Case_Folding') {
1302             $proxy_prop = 'cf';
1303         }
1304         else {
1305             # We captured the U, L, or T, leading to uc, lc, or tc.
1306             $proxy_prop = lc $1 . "c";
1307         }
1308         if ($format ne "c") {
1309             fail("prop_invmap('$mod_prop')");
1310             diag("The format should be 'c'; got '$format'");
1311             next PROPERTY;
1312         }
1313     }
1314
1315     my $base_file;
1316     my $official;
1317
1318     # Handle the properties that have full disk files for them (except the
1319     # Name property which is structurally enough different that it is handled
1320     # separately below.)
1321     if ($name ne 'na'
1322         && ($name eq 'blk'
1323             || defined
1324                     ($base_file = $utf8::loose_property_to_file_of{$proxy_prop})
1325             || exists $utf8::loose_to_file_of{$proxy_prop}
1326             || $name eq "dm"))
1327     {
1328         # In the above, blk is done unconditionally, as we need to test that
1329         # the old-style block names are returned, even if mktables has
1330         # generated a file for the new-style; the test for dm comes afterward,
1331         # so that if a file has been generated for it explicitly, we use that
1332         # file (which is valid, unlike blk) instead of the combo
1333         # Decomposition.pl files.
1334         my $file;
1335         my $is_binary = 0;
1336         if ($name eq 'blk') {
1337
1338             # The blk property is special.  The original file with old block
1339             # names is retained, and the default is to not write out a
1340             # new-name file.  What we do is get the old names into a data
1341             # structure, and from that create what the new file would look
1342             # like.  $base_file is needed to be defined, just to avoid a
1343             # message below.
1344             $base_file = "This is a dummy name";
1345             my $blocks_ref = charblocks();
1346             $official = "";
1347             for my $range (sort { $a->[0][0] <=> $b->[0][0] }
1348                            values %$blocks_ref)
1349             {
1350                 # Translate the charblocks() data structure to what the file
1351                 # would like.
1352                 $official .= sprintf"%04X\t%04X\t%s\n",
1353                              $range->[0][0],
1354                              $range->[0][1],
1355                              $range->[0][2];
1356             }
1357         }
1358         else {
1359             $base_file = "Decomposition" if $format eq 'd';
1360
1361             # Above leaves $base_file undefined only if it came from the hash
1362             # below.  This should happen only when it is a binary property
1363             # (and are accessing via a single-form name, like 'In_Latin1'),
1364             # and so it is stored in a different directory than the To ones.
1365             # XXX Currently, the only cases where it is complemented are the
1366             # ones that have no code points.  And it works out for these that
1367             # 1) complementing them, and then 2) adding or subtracting the
1368             # initial 0 and final 110000 cancel each other out.  But further
1369             # work would be needed in the unlikely event that an inverted
1370             # property comes along without these characteristics
1371             if (!defined $base_file) {
1372                 $base_file = $utf8::loose_to_file_of{$proxy_prop};
1373                 $is_binary = ($base_file =~ s/^!//) ? -1 : 1;
1374                 $base_file = "lib/$base_file";
1375             }
1376
1377             # Read in the file
1378             $file = "unicore/$base_file.pl";
1379             $official = do $file;
1380
1381             # Get rid of any trailing space and comments in the file.
1382             $official =~ s/\s*(#.*)?$//mg;
1383
1384             if ($format eq 'd') {
1385                 my @official = split /\n/, $official;
1386                 $official = "";
1387                 foreach my $line (@official) {
1388                     my ($start, $end, $value)
1389                                     = $line =~ / ^ (.+?) \t (.*?) \t (.+?)
1390                                                 \s* ( \# .* )? $ /x;
1391                     # Decomposition.pl also has the <compatible> types in it,
1392                     # which should be removed.
1393                     $value =~ s/<.*?> //;
1394                     $official .= "$start\t\t$value\n";
1395
1396                     # If this is a multi-char range, we turn it into as many
1397                     # single character ranges as necessary.  This makes things
1398                     # easier below.
1399                     if ($end ne "") {
1400                         for my $i (hex($start) + 1 .. hex $end) {
1401                             $official .= sprintf "%04X\t\t%s\n", $i, $value;
1402                         }
1403                     }
1404                 }
1405             }
1406         }
1407         chomp $official;
1408
1409         # If there are any special elements, get a reference to them.
1410         my $swash_name = $utf8::file_to_swash_name{$base_file};
1411         my $specials_ref;
1412         if ($swash_name) {
1413             $specials_ref = $utf8::SwashInfo{$swash_name}{'specials_name'};
1414             if ($specials_ref) {
1415
1416                 # Convert from the name to the actual reference.
1417                 no strict 'refs';
1418                 $specials_ref = \%{$specials_ref};
1419             }
1420         }
1421
1422         # Certain of the proxy properties have to be adjusted to match the
1423         # real ones.
1424         if ($full_name =~ /^(Case_Folding|(Lower|Title|Upper)case_Mapping)/) {
1425
1426             # Here we have either
1427             #   1) Case_Folding; or
1428             #   2) a proxy that is a full mapping, which means that what the
1429             #      real property is is the equivalent simple mapping.
1430             # In both cases, the file will have a standard list containing
1431             # simple mappings (to a single code point), and a specials hash
1432             # which contains all the mappings that are to multiple code
1433             # points.  First, extract a list containing all the file's simple
1434             # mappings.
1435             my @list;
1436             for (split "\n", $official) {
1437                 my ($start, $end, $value) = / ^ (.+?) \t (.*?) \t (.+?)
1438                                                 \s* ( \# .* )? $ /x;
1439                 $end = $start if $end eq "";
1440                 push @list, [ hex $start, hex $end, $value ];
1441             }
1442
1443             # For these mappings, the file contains all the simple mappings,
1444             # including the ones that are overridden by the specials.  These
1445             # need to be removed as the list is for just the full ones.
1446
1447             # Go through any special mappings one by one.  They are packed.
1448             my $i = 0;
1449             foreach my $utf8_cp (sort keys %$specials_ref) {
1450                 my $cp = unpack("C0U", $utf8_cp);
1451
1452                 # Find the spot in the @list of simple mappings that this
1453                 # special applies to; uses a linear search.
1454                 while ($i < @list -1 ) {
1455                     last if  $cp <= $list[$i][1];
1456                     $i++;
1457                 }
1458
1459                 # Here $i is such that it points to the first range which ends
1460                 # at or above cp, and hence is the only range that could
1461                 # possibly contain it.
1462
1463                 # If not in this range, no range contains it: nothing to
1464                 # remove.
1465                 next if $cp < $list[$i][0];
1466
1467                 # Otherwise, remove the existing entry.  If it is the first
1468                 # element of the range...
1469                 if ($cp == $list[$i][0]) {
1470
1471                     # ... and there are other elements in the range, just shorten
1472                     # the range to exclude this code point.
1473                     if ($list[$i][1] > $list[$i][0]) {
1474                         $list[$i][0]++;
1475                     }
1476
1477                     # ... but if it is the only element in the range, remove
1478                     # it entirely.
1479                     else {
1480                         splice @list, $i, 1;
1481                     }
1482                 }
1483                 else { # Is somewhere in the middle of the range
1484                     # Split the range into two, excluding this one in the
1485                     # middle
1486                     splice @list, $i, 1,
1487                            [ $list[$i][0], $cp - 1, $list[$i][2] ],
1488                            [ $cp + 1, $list[$i][1], $list[$i][2] ];
1489                 }
1490             }
1491
1492             # Here, have gone through all the specials, modifying @list as
1493             # needed.  Turn it back into what the file should look like.
1494             $official = "";
1495             for my $element (@list) {
1496                 $official .= "\n" if $official;
1497                 if ($element->[1] == $element->[0]) {
1498                     $official .= sprintf "%04X\t\t%s", $element->[0], $element->[2];
1499                 }
1500                 else {
1501                     $official .= sprintf "%04X\t%04X\t%s", $element->[0], $element->[1], $element->[2];
1502                 }
1503             }
1504         }
1505         elsif ($full_name =~ /Simple_(Case_Folding|(Lower|Title|Upper)case_Mapping)/)
1506         {
1507
1508             # These properties have everything in the regular array, and the
1509             # specials are superfluous.
1510             undef $specials_ref;
1511         }
1512         elsif ($name eq 'bmg') {
1513
1514             # For this property, the file is output using hex notation for the
1515             # map, with all ranges equal to length 1.  Convert from hex to
1516             # decimal.
1517             my @lines = split "\n", $official;
1518             foreach my $line (@lines) {
1519                 my ($code_point, $map) = split "\t\t", $line;
1520                 $line = $code_point . "\t\t" . hex $map;
1521             }
1522             $official = join "\n", @lines;
1523         }
1524
1525         # Here, in $official, we have what the file looks like, or should like
1526         # if we've had to fix it up.  Now take the invmap() output and reverse
1527         # engineer from that what the file should look like.  Each iteration
1528         # appends the next line to the running string.
1529         my $tested_map = "";
1530
1531         # Create a copy of the file's specials hash.  (It has been undef'd if
1532         # we know it isn't relevant to this property, so if it exists, it's an
1533         # error or is relevant).  As we go along, we delete from that copy.
1534         # If a delete fails, or something is left over after we are done,
1535         # it's an error
1536         my %specials = %$specials_ref if $specials_ref;
1537
1538         # The extra -1 is because the final element has been tested above to
1539         # be for anything above Unicode.  The file doesn't go that high.
1540         for (my $i = 0; $i <  @$invlist_ref - 1; $i++) {
1541
1542             # If the map element is a reference, have to stringify it (but
1543             # don't do so if the format doesn't allow references, so that an
1544             # improper format will generate an error.
1545             if (ref $invmap_ref->[$i]
1546                 && ($format eq 'd' || $format =~ /^ . l /x))
1547             {
1548                 # The stringification depends on the format.
1549                 if ($format eq 'sl') {
1550
1551                     # At the time of this writing, there are two types of 'sl'
1552                     # format  One, in Name_Alias, has multiple separate entries
1553                     # for each code point; the other, in Script_Extension, is space
1554                     # separated.  Assume the latter for non-Name_Alias.
1555                     if ($full_name ne 'Name_Alias') {
1556                         $invmap_ref->[$i] = join " ", @{$invmap_ref->[$i]};
1557                     }
1558                     else {
1559                         # For Name_Alias, we emulate the file.  Entries with
1560                         # just one value don't need any changes, but we
1561                         # convert the list entries into a series of lines for
1562                         # the file, starting with the first name.  The
1563                         # succeeding entries are on separate lines, with the
1564                         # code point repeated for each one and then two tabs,
1565                         # then the value.  Code at the end of the loop will
1566                         # set up the first line with its code point and two
1567                         # tabs before the value, just as it does for every
1568                         # other property; thus the special handling of the
1569                         # first line.
1570                         if (ref $invmap_ref->[$i]) {
1571                             my $hex_cp = sprintf("%04X", $invlist_ref->[$i]);
1572                             my $concatenated = $invmap_ref->[$i][0];
1573                             for (my $j = 1; $j < @{$invmap_ref->[$i]}; $j++) {
1574                                 $concatenated .= "\n$hex_cp\t\t" . $invmap_ref->[$i][$j];
1575                             }
1576                             $invmap_ref->[$i] = $concatenated;
1577                         }
1578                     }
1579                 }
1580                 elsif ($format =~ / ^ cl e? $/x) {
1581
1582                     # For a cl property, the stringified result should be in
1583                     # the specials hash.  The key is the packed code point,
1584                     # and the value is the packed map.
1585                     my $value;
1586                     if (! defined ($value = delete $specials{pack("C0U", $invlist_ref->[$i]) })) {
1587                         fail("prop_invmap('$mod_prop')");
1588                         diag(sprintf "There was no specials element for %04X", $invlist_ref->[$i]);
1589                         next PROPERTY;
1590                     }
1591                     my $packed = pack "U*", @{$invmap_ref->[$i]};
1592                     if ($value ne $packed) {
1593                         fail("prop_invmap('$mod_prop')");
1594                         diag(sprintf "For %04X, expected the mapping to be '$packed', but got '$value'");
1595                         next PROPERTY;
1596                     }
1597
1598                     # As this doesn't get tested when we later compare with
1599                     # the actual file, it could be out of order and we
1600                     # wouldn't know it.
1601                     if (($i > 0 && $invlist_ref->[$i] <= $invlist_ref->[$i-1])
1602                         || $invlist_ref->[$i] >= $invlist_ref->[$i+1])
1603                     {
1604                         fail("prop_invmap('$mod_prop')");
1605                         diag(sprintf "Range beginning at %04X is out-of-order.", $invlist_ref->[$i]);
1606                         next PROPERTY;
1607                     }
1608                     next;
1609                 }
1610                 elsif ($format eq 'd') {
1611
1612                     # The decomposition mapping file has the code points as
1613                     # a string of space-separated hex constants.
1614                     $invmap_ref->[$i] = join " ", map { sprintf "%04X", $_ } @{$invmap_ref->[$i]};
1615                 }
1616                 else {
1617                     fail("prop_invmap('$mod_prop')");
1618                     diag("Can't handle format '$format'");
1619                     next PROPERTY;
1620                 }
1621             }
1622             elsif ($format eq 'd' || $format eq 'cle') {
1623
1624                 # The numerics in the returned map are stored as deltas.  The
1625                 # defaults are 0, and don't appear in $official, and are
1626                 # excluded later, but the elements must be converted back to
1627                 # their real code point values before comparing with
1628                 # $official, as these files, for backwards compatibility, are
1629                 # not stored as deltas.  (There currently is only one cle
1630                 # property, nfkccf.  If that changed this would also have to.)
1631                 if ($invmap_ref->[$i] =~ / ^ -? \d+ $ /x
1632                     && $invmap_ref->[$i] != 0)
1633                 {
1634                     my $delta = $invmap_ref->[$i];
1635                     $invmap_ref->[$i] += $invlist_ref->[$i];
1636
1637                     # If there are other elements with this same delta, they
1638                     # must individually be re-mapped.  Do this by splicing in
1639                     # a new element into the list and the map containing the
1640                     # remainder of the range.  Next time through we will look
1641                     # at that (possibly splicing again until the whole range
1642                     # is processed).
1643                     if ($invlist_ref->[$i+1] > $invlist_ref->[$i] + 1) {
1644                         splice @$invlist_ref, $i+1, 0,
1645                                 $invlist_ref->[$i] + 1;
1646                         splice @$invmap_ref, $i+1, 0, $delta;
1647                     }
1648                 }
1649                 if ($format eq 'cle' && $invmap_ref->[$i] eq "") {
1650
1651                     # cle properties have maps to the empty string that also
1652                     # should be in the specials hash, with the key the packed
1653                     # code point, and the map just empty.
1654                     my $value;
1655                     if (! defined ($value = delete $specials{pack("C0U", $invlist_ref->[$i]) })) {
1656                         fail("prop_invmap('$mod_prop')");
1657                         diag(sprintf "There was no specials element for %04X", $invlist_ref->[$i]);
1658                         next PROPERTY;
1659                     }
1660                     if ($value ne "") {
1661                         fail("prop_invmap('$mod_prop')");
1662                         diag(sprintf "For %04X, expected the mapping to be \"\", but got '$value'", $invlist_ref->[$i]);
1663                         next PROPERTY;
1664                     }
1665
1666                     # As this doesn't get tested when we later compare with
1667                     # the actual file, it could be out of order and we
1668                     # wouldn't know it.
1669                     if (($i > 0 && $invlist_ref->[$i] <= $invlist_ref->[$i-1])
1670                         || $invlist_ref->[$i] >= $invlist_ref->[$i+1])
1671                     {
1672                         fail("prop_invmap('$mod_prop')");
1673                         diag(sprintf "Range beginning at %04X is out-of-order.", $invlist_ref->[$i]);
1674                         next PROPERTY;
1675                     }
1676                     next;
1677                 }
1678             }
1679             elsif ($is_binary) { # These binary files don't have an explicit Y
1680                 $invmap_ref->[$i] =~ s/Y//;
1681             }
1682
1683             # The file doesn't include entries that map to $missing, so don't
1684             # include it in the built-up string.  But make sure that it is in
1685             # the correct order in the input.
1686             if ($invmap_ref->[$i] eq $missing) {
1687                 if (($i > 0 && $invlist_ref->[$i] <= $invlist_ref->[$i-1])
1688                     || $invlist_ref->[$i] >= $invlist_ref->[$i+1])
1689                 {
1690                     fail("prop_invmap('$mod_prop')");
1691                     diag(sprintf "Range beginning at %04X is out-of-order.", $invlist_ref->[$i]);
1692                     next PROPERTY;
1693                 }
1694                 next;
1695             }
1696
1697             # The 'd' property and 'c' properties whose underlying format is
1698             # hexadecimal have the mapping expressed in hex in the file
1699             if ($format eq 'd'
1700                 || ($format =~ /^c/
1701                     && $swash_name
1702                     && $utf8::SwashInfo{$swash_name}{'format'} eq 'x'))
1703             {
1704
1705                 # The d property has one entry which isn't in the file.
1706                 # Ignore it, but make sure it is in order.
1707                 if ($format eq 'd'
1708                     && $invmap_ref->[$i] eq '<hangul syllable>'
1709                     && $invlist_ref->[$i] == 0xAC00)
1710                 {
1711                     if (($i > 0 && $invlist_ref->[$i] <= $invlist_ref->[$i-1])
1712                         || $invlist_ref->[$i] >= $invlist_ref->[$i+1])
1713                     {
1714                         fail("prop_invmap('$mod_prop')");
1715                         diag(sprintf "Range beginning at %04X is out-of-order.", $invlist_ref->[$i]);
1716                         next PROPERTY;
1717                     }
1718                     next;
1719                 }
1720                 $invmap_ref->[$i] = sprintf("%04X", $invmap_ref->[$i])
1721                                   if $invmap_ref->[$i] =~ / ^ [A-Fa-f0-9]+ $/x;
1722             }
1723
1724             # Finally have figured out what the map column in the file should
1725             # be.  Append the line to the running string.
1726             my $start = $invlist_ref->[$i];
1727             my $end = $invlist_ref->[$i+1] - 1;
1728             $end = ($start == $end) ? "" : sprintf("%04X", $end);
1729             if ($invmap_ref->[$i] ne "") {
1730                 $tested_map .= sprintf "%04X\t%s\t%s\n", $start, $end, $invmap_ref->[$i];
1731             }
1732             elsif ($end ne "") {
1733                 $tested_map .= sprintf "%04X\t%s\n", $start, $end;
1734             }
1735             else {
1736                 $tested_map .= sprintf "%04X\n", $start;
1737             }
1738         } # End of looping over all elements.
1739
1740         # Here are done with generating what the file should look like
1741
1742         chomp $tested_map;
1743
1744         # And compare.
1745         if ($tested_map ne $official) {
1746             fail_with_diff($mod_prop, $official, $tested_map, "prop_invmap");
1747             next PROPERTY;
1748         }
1749
1750         # There shouldn't be any specials unaccounted for.
1751         if (keys %specials) {
1752             fail("prop_invmap('$mod_prop')");
1753             diag("Unexpected specials: " . join ", ", keys %specials);
1754             next PROPERTY;
1755         }
1756     }
1757     elsif ($format eq 'n') {
1758
1759         # Handle the Name property similar to the above.  But the file is
1760         # sufficiently different that it is more convenient to make a special
1761         # case for it.  It is a combination of the Name, Unicode1_Name, and
1762         # Name_Alias properties, and named sequences.  We need to remove all
1763         # but the Name in order to do the comparison.
1764
1765         if ($missing ne "") {
1766             fail("prop_invmap('$mod_prop')");
1767             diag("The missings should be \"\"; got \"missing\"");
1768             next PROPERTY;
1769         }
1770
1771         $official = do "unicore/Name.pl";
1772
1773         # Get rid of the named sequences portion of the file.  These don't
1774         # have a tab before the first blank on a line.
1775         $official =~ s/ ^ [^\t]+ \  .*? \n //xmg;
1776
1777         # And get rid of the controls.  These are named in the file, but
1778         # shouldn't be in the property.  This gets rid of the two ranges in
1779         # one fell swoop, and also all the Unicode1_Name values that may not
1780         # be in Name_Alias.
1781         $official =~ s/ 00000 \t .* 0001F .*? \n//xs;
1782         $official =~ s/ 0007F \t .* 0009F .*? \n//xs;
1783
1784         # And remove the aliases.  We read in the Name_Alias property, and go
1785         # through them one by one.
1786         my ($aliases_code_points, $aliases_maps, undef, undef)
1787                                                 = &prop_invmap('Name_Alias');
1788         for (my $i = 0; $i < @$aliases_code_points; $i++) {
1789             my $code_point = $aliases_code_points->[$i];
1790
1791             # Already removed these above.
1792             next if $code_point <= 0x1F
1793                     || ($code_point >= 0x7F && $code_point <= 0x9F);
1794
1795             my $hex_code_point = sprintf "%05X", $code_point;
1796
1797             # Convert to a list if not already to make the following loop
1798             # control uniform.
1799             $aliases_maps->[$i] = [ $aliases_maps->[$i] ]
1800                                                 if ! ref $aliases_maps->[$i];
1801
1802             # Remove each alias for this code point from the file
1803             foreach my $alias (@{$aliases_maps->[$i]}) {
1804
1805                 # Remove the alias type from the entry, retaining just the name.
1806                 $alias =~ s/:.*//;
1807
1808                 $alias = quotemeta($alias);
1809                 $official =~ s/$hex_code_point \t $alias \n //x;
1810             }
1811         }
1812         chomp $official;
1813
1814         # Here have adjusted the file.  We also have to adjust the returned
1815         # inversion map by checking and deleting all the lines in it that
1816         # won't be in the file.  These are the lines that have generated
1817         # things, like <hangul syllable>.
1818         my $tested_map = "";        # Current running string
1819         my @code_point_in_names =
1820                                @Unicode::UCD::code_points_ending_in_code_point;
1821
1822         for my $i (0 .. @$invlist_ref - 1 - 1) {
1823             my $start = $invlist_ref->[$i];
1824             my $end = $invlist_ref->[$i+1] - 1;
1825             if ($invmap_ref->[$i] eq $missing) {
1826                 if (($i > 0 && $invlist_ref->[$i] <= $invlist_ref->[$i-1])
1827                     || $invlist_ref->[$i] >= $invlist_ref->[$i+1])
1828                 {
1829                     fail("prop_invmap('$mod_prop')");
1830                     diag(sprintf "Range beginning at %04X is out-of-order.", $invlist_ref->[$i]);
1831                     next PROPERTY;
1832                 }
1833                 next;
1834             }
1835             if ($invmap_ref->[$i] =~ / (.*) ( < .*? > )/x) {
1836                 my $name = $1;
1837                 my $type = $2;
1838                 if (($i > 0 && $invlist_ref->[$i] <= $invlist_ref->[$i-1])
1839                     || $invlist_ref->[$i] >= $invlist_ref->[$i+1])
1840                 {
1841                     fail("prop_invmap('$mod_prop')");
1842                     diag(sprintf "Range beginning at %04X is out-of-order.", $invlist_ref->[$i]);
1843                     next PROPERTY;
1844                 }
1845                 if ($type eq "<hangul syllable>") {
1846                     if ($name ne "") {
1847                         fail("prop_invmap('$mod_prop')");
1848                         diag("Unexpected text in $invmap_ref->[$i]");
1849                         next PROPERTY;
1850                     }
1851                     if ($start != 0xAC00) {
1852                         fail("prop_invmap('$mod_prop')");
1853                         diag(sprintf("<hangul syllables> should begin at 0xAC00, got %04X", $start));
1854                         next PROPERTY;
1855                     }
1856                     if ($end != $start + 11172 - 1) {
1857                         fail("prop_invmap('$mod_prop')");
1858                         diag(sprintf("<hangul syllables> should end at %04X, got %04X", $start + 11172 -1, $end));
1859                         next PROPERTY;
1860                     }
1861                 }
1862                 elsif ($type ne "<code point>") {
1863                     fail("prop_invmap('$mod_prop')");
1864                     diag("Unexpected text '$type' in $invmap_ref->[$i]");
1865                     next PROPERTY;
1866                 }
1867                 else {
1868
1869                     # Look through the array of names that end in code points,
1870                     # and look for this start and end.  If not found is an
1871                     # error.  If found, delete it, and at the end, make sure
1872                     # have deleted everything.
1873                     for my $i (0 .. @code_point_in_names - 1) {
1874                         my $hash = $code_point_in_names[$i];
1875                         if ($hash->{'low'} == $start
1876                             && $hash->{'high'} == $end
1877                             && "$hash->{'name'}-" eq $name)
1878                         {
1879                             splice @code_point_in_names, $i, 1;
1880                             last;
1881                         }
1882                         else {
1883                             fail("prop_invmap('$mod_prop')");
1884                             diag("Unexpected code-point-in-name line '$invmap_ref->[$i]'");
1885                             next PROPERTY;
1886                         }
1887                     }
1888                 }
1889
1890                 next;
1891             }
1892
1893             # Have adjusted the map, as needed.  Append to running string.
1894             $end = ($start == $end) ? "" : sprintf("%05X", $end);
1895             $tested_map .= sprintf "%05X\t%s\n", $start, $invmap_ref->[$i];
1896         }
1897
1898         # Finished creating the string from the inversion map.  Can compare
1899         # with what the file is.
1900         chomp $tested_map;
1901         if ($tested_map ne $official) {
1902             fail_with_diff($mod_prop, $official, $tested_map, "prop_invmap");
1903             next PROPERTY;
1904         }
1905         if (@code_point_in_names) {
1906             fail("prop_invmap('$mod_prop')");
1907             use Data::Dumper;
1908             diag("Missing code-point-in-name line(s)" . Dumper \@code_point_in_names);
1909             next PROPERTY;
1910         }
1911     }
1912     elsif ($format eq 's' || $format eq 'r') {
1913
1914         # Here the map is not more or less directly from a file stored on
1915         # disk.  We try a different tack.  These should all be properties that
1916         # have just a few possible values (most of them are  binary).  We go
1917         # through the map list, sorting each range into buckets, one for each
1918         # map value.  Thus for binary properties there will be a bucket for Y
1919         # and one for N.  The buckets are inversion lists.  We compare each
1920         # constructed inversion list with what we would get for it using
1921         # prop_invlist(), which has already been tested.  If they all match,
1922         # the whole map must have matched.
1923         my %maps;
1924         my $previous_map;
1925
1926         # (The extra -1 is to not look at the final element in the loop, which
1927         # we know is the one that starts just beyond Unicode and goes to
1928         # infinity.)
1929         for my $i (0 .. @$invlist_ref - 1 - 1) {
1930             my $range_start = $invlist_ref->[$i];
1931
1932             # Because we are sorting into buckets, things could be
1933             # out-of-order here, and still be in the correct order in the
1934             # bucket, and hence wouldn't show up as an error; so have to
1935             # check.
1936             if (($i > 0 && $range_start <= $invlist_ref->[$i-1])
1937                 || $range_start >= $invlist_ref->[$i+1])
1938             {
1939                 fail("prop_invmap('$mod_prop')");
1940                 diag(sprintf "Range beginning at %04X is out-of-order.", $invlist_ref->[$i]);
1941                 next PROPERTY;
1942             }
1943
1944             # This new range closes out the range started in the previous
1945             # iteration.
1946             push @{$maps{$previous_map}}, $range_start if defined $previous_map;
1947
1948             # And starts a range which will be closed in the next iteration.
1949             $previous_map = $invmap_ref->[$i];
1950             push @{$maps{$previous_map}}, $range_start;
1951         }
1952
1953         # The range we just started hasn't been closed, and we didn't look at
1954         # the final element of the loop.  If that range is for the default
1955         # value, it shouldn't be closed, as it is to extend to infinity.  But
1956         # otherwise, it should end at the final Unicode code point, and the
1957         # list that maps to the default value should have another element that
1958         # does go to infinity for every above Unicode code point.
1959
1960         if (@$invlist_ref > 1) {
1961             my $penultimate_map = $invmap_ref->[-2];
1962             if ($penultimate_map ne $missing) {
1963
1964                 # The -1th element contains the first non-Unicode code point.
1965                 push @{$maps{$penultimate_map}}, $invlist_ref->[-1];
1966                 push @{$maps{$missing}}, $invlist_ref->[-1];
1967             }
1968         }
1969
1970         # Here, we have the buckets (inversion lists) all constructed.  Go
1971         # through each and verify that matches what prop_invlist() returns.
1972         # We could use is_deeply() for the comparison, but would get multiple
1973         # messages for each $prop.
1974         foreach my $map (keys %maps) {
1975             my @off_invlist = prop_invlist("$prop = $map");
1976             my $min = (@off_invlist >= @{$maps{$map}})
1977                        ? @off_invlist
1978                        : @{$maps{$map}};
1979             for my $i (0 .. $min- 1) {
1980                 if ($i > @off_invlist - 1) {
1981                     fail("prop_invmap('$mod_prop')");
1982                     diag("There is no element [$i] for $prop=$map from prop_invlist(), while [$i] in the implicit one constructed from prop_invmap() is '$maps{$map}[$i]'");
1983                     next PROPERTY;
1984                 }
1985                 elsif ($i > @{$maps{$map}} - 1) {
1986                     fail("prop_invmap('$mod_prop')");
1987                     diag("There is no element [$i] from the implicit $prop=$map constructed from prop_invmap(), while [$i] in the one from prop_invlist() is '$off_invlist[$i]'");
1988                     next PROPERTY;
1989                 }
1990                 elsif ($maps{$map}[$i] ne $off_invlist[$i]) {
1991                     fail("prop_invmap('$mod_prop')");
1992                     diag("Element [$i] of the implicit $prop=$map constructed from prop_invmap() is '$maps{$map}[$i]', and the one from prop_invlist() is '$off_invlist[$i]'");
1993                     next PROPERTY;
1994                 }
1995             }
1996         }
1997     }
1998     else {  # Don't know this property nor format.
1999
2000         fail("prop_invmap('$mod_prop')");
2001         diag("Unknown format '$format'");
2002     }
2003
2004     pass("prop_invmap('$mod_prop')");
2005 }
2006
2007 done_testing();