source: osm/applications/editors/josm/i18n/i18n.pl@ 35025

Last change on this file since 35025 was 34996, checked in by stoecker, 6 years ago

fix #josm17679 - CDATA in translated texts

  • Property svn:executable set to *
File size: 12.6 KB
RevLine 
[19319]1#! /usr/bin/perl -w
[34025]2# fileformat documentation: JOSM I18n.java function load()
[19319]3
4use utf8;
[31111]5use strict;
[30169]6use open qw/:std :encoding(utf8)/;
[19319]7use Term::ReadKey;
8use Encode;
9
10my $waswarn = 0;
[31087]11my $lang_pattern = '([a-z]{2}_[A-Z]{2}|[a-z]{2,3}|[a-z]{2}\@[a-z]+)';
[31088]12my $lang_pattern_file = '([a-z]{2}_[A-Z]{2}|[a-z]{2,3}|[a-z]{2}-[a-z]+)';
[19319]13
14main();
15
16sub getdate
17{
18 my @t=gmtime();
19 return sprintf("%04d-%02d-%02d %02d:%02d+0000",
20 1900+$t[5],$t[4]+1,$t[3],$t[2],$t[1]);
21}
22
[31111]23sub loadpot($)
[19319]24{
[31111]25 my ($file) = @_;
26 my %all = ();
27 my %keys = ();
28 die "Could not open file $file." if(!open FILE,"<:utf8",$file);
29 my %postate = (last => "", type => "");
30 my $linenum = 0;
31 print "Reading file $file\n";
32 while(<FILE>)
33 {
34 ++$linenum;
35 my $fn = "$file:$linenum";
36 chomp;
37 if($_ =~ /^#/ || !$_)
38 {
39 checkpo(\%postate, \%all, "pot", "line $linenum in $file", \%keys, 1, undef);
40 $postate{fuzzy} = 1 if ($_ =~ /fuzzy/);
41 }
42 elsif($_ =~ /^"(.*)"$/) {$postate{last} .= $1;}
43 elsif($_ =~ /^(msg.+) "(.*)"$/)
44 {
45 my ($n, $d) = ($1, $2);
46 my $new = !${postate}{fuzzy} && (($n eq "msgid" && $postate{type} ne "msgctxt") || ($n eq "msgctxt"));
47 checkpo(\%postate, \%all, "pot", "line $linenum in $file", \%keys, $new, undef);
48 $postate{last} = $d;
49 $postate{type} = $n;
50 $postate{src} = $fn if $new;
51 }
52 else
53 {
54 die "Strange line $linenum in $file: $_.";
55 }
56 }
57 checkpo(\%postate, \%all, "pot", "line $linenum in $file", \%keys, 1, undef);
58 close(FILE);
59 return \%all;
60}
61
62sub loadfiles($$@)
63{
[19319]64 my $desc;
[31111]65 my %all = ();
66 my %keys = ();
67 my ($lang,$use,@files) = @_;
[19319]68 foreach my $file (@files)
69 {
70 die "Could not open file $file." if(!open FILE,"<:utf8",$file);
71
[31086]72 if($file =~ /\/$lang_pattern\.po$/)
[19319]73 {
74 my $l = $1;
75 ++$lang->{$l};
76 my %postate = (last => "", type => "");
77 my $linenum = 0;
[31087]78 print "Reading file $file (lang $l)\n";
[19319]79 while(<FILE>)
80 {
81 ++$linenum;
82 my $fn = "$file:$linenum";
83 chomp;
84 if($_ =~ /^#/ || !$_)
85 {
[31111]86 checkpo(\%postate, \%all, $l, "line $linenum in $file", \%keys, 1, $use);
[19319]87 $postate{fuzzy} = 1 if ($_ =~ /fuzzy/);
88 }
89 elsif($_ =~ /^"(.*)"$/) {$postate{last} .= $1;}
90 elsif($_ =~ /^(msg.+) "(.*)"$/)
91 {
92 my ($n, $d) = ($1, $2);
[19334]93 my $new = !${postate}{fuzzy} && (($n eq "msgid" && $postate{type} ne "msgctxt") || ($n eq "msgctxt"));
[31111]94 checkpo(\%postate, \%all, $l, "line $linenum in $file", \%keys, $new, $use);
[19319]95 $postate{last} = $d;
96 $postate{type} = $n;
97 $postate{src} = $fn if $new;
98 }
99 else
100 {
101 die "Strange line $linenum in $file: $_.";
102 }
103 }
[31111]104 checkpo(\%postate, \%all, $l, "line $linenum in $file", \%keys, 1, $use);
[19319]105 }
106 else
107 {
108 die "File format not supported for file $file.";
109 }
110 close(FILE);
111 }
112 return %all;
113}
114
115my $alwayspo = 0;
116my $alwaysup = 0;
117my $noask = 0;
[31111]118my %conflicts;
[19319]119sub copystring($$$$$$$)
120{
121 my ($data, $en, $l, $str, $txt, $context, $ispo) = @_;
122
123 $en = "___${context}___$en" if $context;
124
125 if(exists($data->{$en}{$l}) && $data->{$en}{$l} ne $str)
126 {
127 return if !$str;
128 if($l =~ /^_/)
129 {
130 $data->{$en}{$l} .= ";$str" if !($data->{$en}{$l} =~ /$str/);
131 }
132 elsif(!$data->{$en}{$l})
133 {
134 $data->{$en}{$l} = $str;
135 }
136 else
137 {
138 my $f = $data->{$en}{_file} || "";
139 $f = ($f ? "$f;".$data->{$en}{"_src.$l"} : $data->{$en}{"_src.$l"}) if $data->{$en}{"_src.$l"};
140 my $isotherpo = ($f =~ /\.po\:/);
141 my $pomode = ($ispo && !$isotherpo) || (!$ispo && $isotherpo);
142
143 my $mis = "String mismatch for '$en' **$str** ($txt) != **$data->{$en}{$l}** ($f)\n";
144 my $replace = 0;
145
146 if(($conflicts{$l}{$str} || "") eq $data->{$en}{$l}) {}
147 elsif($pomode && $alwaysup) { $replace=$isotherpo; }
148 elsif($pomode && $alwayspo) { $replace=$ispo; }
149 elsif($noask) { print $mis; ++$waswarn; }
150 else
151 {
152 ReadMode 4; # Turn off controls keys
153 my $arg = "(l)eft, (r)ight";
154 $arg .= ", (p)o, (u)pstream[ts/mat], all p(o), all up(s)tream" if $pomode;
155 $arg .= ", e(x)it: ";
156 print "$mis$arg";
157 while((my $c = getc()))
158 {
159 if($c eq "l") { $replace=1; }
160 elsif($c eq "r") {}
161 elsif($c eq "p" && $pomode) { $replace=$ispo; }
162 elsif($c eq "u" && $pomode) { $replace=$isotherpo; }
163 elsif($c eq "o" && $pomode) { $alwayspo = 1; $replace=$ispo; }
164 elsif($c eq "s" && $pomode) { $alwaysup = 1; $replace=$isotherpo; }
165 elsif($c eq "x") { $noask = 1; ++$waswarn; }
166 else { print "\n$arg"; next; }
167 last;
168 }
169 print("\n");
170 ReadMode 0; # Turn on controls keys
171 }
172 if(!$noask)
173 {
174 if($replace)
175 {
176 $data->{$en}{$l} = $str;
177 $conflicts{$l}{$data->{$en}{$l}} = $str;
178 }
179 else
180 {
181 $conflicts{$l}{$str} = $data->{$en}{$l};
182 }
183 }
184 }
185 }
186 else
187 {
188 $data->{$en}{$l} = $str;
189 }
190}
191
[31111]192# Check a current state for new data
193#
194# @param postate Pointer to current status hash
195# @param data Pointer to final data array
196# @param l current language
197# @param txt output text in case of error, usually file and line number
198# @param keys pointer to hash for info keys extracted from the first msgid "" entry
199# @param new whether a data set is finish or not yet complete
200# @param use hash to strings to use or undef for all strings
201#
202sub checkpo($$$$$$$)
[19319]203{
[31111]204 my ($postate, $data, $l, $txt, $keys, $new, $use) = @_;
[19319]205
206 if($postate->{type} eq "msgid") {$postate->{msgid} = $postate->{last};}
207 elsif($postate->{type} eq "msgid_plural") {$postate->{msgid_1} = $postate->{last};}
208 elsif($postate->{type} =~ /^msgstr(\[0\])?$/) {$postate->{msgstr} = $postate->{last};}
209 elsif($postate->{type} =~ /^msgstr\[(.+)\]$/) {$postate->{"msgstr_$1"} = $postate->{last};}
210 elsif($postate->{type} eq "msgctxt") {$postate->{context} = $postate->{last};}
211 elsif($postate->{type}) { die "Strange type $postate->{type} found\n" }
212
213 if($new)
214 {
[31111]215 my $en = $postate->{context} ? "___$postate->{context}___$postate->{msgid}" : $postate->{msgid};
216 if((!$postate->{fuzzy}) && ($l eq "pot" || $postate->{msgstr}) && $postate->{msgid}
217 && (!$use || $use->{$en}))
[19319]218 {
219 copystring($data, $postate->{msgid}, $l, $postate->{msgstr},$txt,$postate->{context}, 1);
[31111]220 if(!$use || $use->{$en}{"en.1"})
221 {
222 for(my $i = 1; exists($postate->{"msgstr_$i"}); ++$i)
223 { copystring($data, $postate->{msgid}, "$l.$i", $postate->{"msgstr_$i"},$txt,$postate->{context}, 1); }
224 if($postate->{msgid_1})
225 { copystring($data, $postate->{msgid}, "en.1", $postate->{msgid_1},$txt,$postate->{context}, 1); }
226 }
[19319]227 copystring($data, $postate->{msgid}, "_src.$l", $postate->{src},$txt,$postate->{context}, 1);
228 }
229 elsif($postate->{msgstr} && !$postate->{msgid})
230 {
231 my %k = ($postate->{msgstr} =~ /(.+?): +(.+?)\\n/g);
232 # take the first one!
233 for $a (sort keys %k)
234 {
235 $keys->{$l}{$a} = $k{$a} if !$keys->{$l}{$a};
236 }
237 }
238 foreach my $k (keys %{$postate})
239 {
240 delete $postate->{$k};
241 }
242 $postate->{type} = $postate->{last} = "";
243 }
244}
245
[19356]246sub makestring($)
247{
248 my ($str) = @_;
249 $str =~ s/\\"/"/g;
[21523]250 $str =~ s/\\\\/\\/g;
[19356]251 $str =~ s/\\n/\n/g;
252 $str = encode("utf8", $str);
[26338]253 return $str;
[19356]254}
255
[26338]256sub checkstring
257{
[26586]258 my ($la, $tr, $en, $cnt, $en1, $eq) = @_;
[26338]259 $tr = makestring($tr);
260 $en = makestring($en);
[26340]261 $cnt = $cnt || 0;
[26338]262 $en1 = makestring($en1) if defined($en1);
263 my $error = 0;
264
265 # Test one - are there single quotes which don't occur twice
266 my $v = $tr;
267 $v =~ s/''//g; # replace all twice occuring single quotes
[26344]268 $v =~ s/'[{}]'//g; # replace all bracketquoting single quotes
[26340]269 if($v =~ /'/)#&& $la ne "en")
[26338]270 {
[30169]271 warn "JAVA translation issue for language $la: Mismatching single quotes:\nTranslated text: ".decode("utf8",$tr)."\nOriginal text: ".decode("utf8",$en)."\n";
[26339]272 $error = 1;
[26338]273 }
[34996]274 if($tr =~ /<!\[CDATA\[/)#&& $la ne "en")
275 {
276 warn "JAVA translation issue for language $la: CDATA in string:\nTranslated text: ".decode("utf8",$tr)."\nOriginal text: ".decode("utf8",$en)."\n";
277 $error = 1;
278 }
[26338]279 # Test two - check if there are {..} which should not be
280 my @fmt = ();
281 my $fmt;
282 my $fmte;
283 my $fmte1 = "";
[26344]284 my $trt = $tr; $trt =~ s/'[{}]'//g;
[33152]285 while($trt =~ /\{(.*?)\}/g) {push @fmt,$1};
286 while($trt =~ /\%([a-z]+)\%/g) {push @fmt,$1};
287 $fmt = join("_", sort @fmt); @fmt = ();
[26344]288 my $ent = $en; $ent =~ s/'[{}]'//g;
[33152]289 while($ent =~ /\{(.*?)\}/g) {push @fmt,$1};
290 while($ent =~ /\%([a-z]+)\%/g) {push @fmt,$1};
291 $fmte = join("_", sort @fmt); @fmt = ();
[26344]292 if($en1)
293 {
294 my $en1t = $en1; $en1t =~ s/'[{}]'//g;
295 while($en1t =~ /\{(.*?)\}/g) {push @fmt,$1}; $fmte1 = join("_", sort @fmt);
296 }
[26338]297 if($fmt ne $fmte && $fmt ne $fmte1)
298 {
299 if(!($fmte eq '0' && $fmt eq "" && $cnt == 1)) # Don't warn when a single value is left for first multi-translation
300 {
[30169]301 warn "JAVA translation issue for language $la ($cnt): Mismatching format entries:\nTranslated text: ".decode("utf8",$tr)."\nOriginal text: ".decode("utf8",$en)."\n";
[26338]302 $error = 1;
303 }
304 }
305
306 #$tr = "" if($error && $la ne "en");
[26586]307 return pack("n",65534) if $eq;
[26338]308
309 return pack("n",length($tr)).$tr;
310}
311
[19319]312sub createlang($@)
313{
314 my ($data, @files) = @_;
[26174]315 my $maxlen = 0;
[19319]316 foreach my $file (@files)
317 {
[26174]318 my $len = length($file);
319 $maxlen = $len if $len > $maxlen;
320 }
[31111]321 my $maxcount = keys(%{$data});
[26174]322 foreach my $file (@files)
323 {
[19319]324 my $la;
[19496]325 my $cnt = 0;
[31088]326 if($file =~ /^(?:.*\/)?$lang_pattern_file\.lang$/)
[19319]327 {
328 $la = $1;
[31088]329 $la =~ s/-/\@/;
[19319]330 }
331 else
332 {
333 die "Language for file $file unknown.";
334 }
335 die "Could not open outfile $file\n" if !open FILE,">:raw",$file;
336
337 foreach my $en (sort keys %{$data})
338 {
339 next if $data->{$en}{"en.1"};
340 my $val;
[26586]341 my $eq;
[19319]342 if($la eq "en")
343 {
[19496]344 ++$cnt;
[19319]345 $val = $en;
346 $val =~ s/^___(.*)___/_:$1\n/;
347 }
348 else
349 {
350 my $ennoctx = $en;
351 $ennoctx =~ s/^___(.*)___//;
352 $val = (exists($data->{$en}{$la})) ? $data->{$en}{$la} : "";
[19496]353 ++$cnt if $val;
[26586]354 if($ennoctx eq $val)
355 {
356 $val = ""; $eq = 1;
357 }
[19319]358 }
[26586]359 print FILE checkstring($la, $val, $en, undef, undef, $eq);
[19319]360 }
361 print FILE pack "n",0xFFFF;
362 foreach my $en (sort keys %{$data})
363 {
364 next if !$data->{$en}{"en.1"};
365 my $num;
366 for($num = 1; exists($data->{$en}{"$la.$num"}); ++$num)
367 { }
368 my $val;
[31111]369 my $eq = 0;
[19319]370 if($la eq "en")
371 {
[19496]372 ++$cnt;
[19319]373 $val = $en;
374 $val =~ s/^___(.*)___/_:$1\n/;
375 }
376 else
377 {
378 $val = (exists($data->{$en}{$la})) ? $data->{$en}{$la} : "";
379 --$num if(!$val);
[19496]380 ++$cnt if $val;
[19319]381 if($num == 2)
382 {
383 my $ennoctx = $en;
384 $ennoctx =~ s/^___(.*)___//;
[26987]385 if($val eq $ennoctx && $data->{$en}{"$la.1"} eq $data->{$en}{"en.1"})
386 {
387 $num = 0;
388 $eq = 1;
389 }
[19319]390 }
391 }
392
[26987]393 print FILE pack "C",$eq ? 0xFE : $num;
[19319]394 if($num)
395 {
[26338]396 print FILE checkstring($la, $val, $en, 1, $data->{$en}{"en.1"});
[19319]397 for($num = 1; exists($data->{$en}{"$la.$num"}); ++$num)
398 {
[26338]399 print FILE checkstring($la, $data->{$en}{"$la.$num"}, $en, $num+1, $data->{$en}{"en.1"});
[19319]400 }
401 }
402 }
403 close FILE;
[25523]404 if(!$cnt)
405 {
406 unlink $file;
[26174]407 printf "Skipped file %-${maxlen}s: Contained 0 strings out of %5d.\n",$file,$maxcount;
[25523]408 }
409 else
410 {
[31956]411 printf "Created file %-${maxlen}s: Added %5d strings out of %5d (%5.1f%%).\n",$file,$cnt,$maxcount,,$cnt*100.0/$maxcount-5e-2;
[25523]412 }
[19319]413 }
414}
415
416sub main
417{
418 my %lang;
419 my @po;
[31111]420 my $potfile;
421 my $basename = "./";
422 foreach my $arg (@ARGV)
423 {
424 next if $arg !~ /^--/;
425 if($arg =~ /^--basedir=(.+)$/)
426 {
427 $basename = $1;
428 }
429 elsif($arg =~ /^--potfile=(.+)$/)
430 {
431 $potfile = $1;
432 }
433 else
434 {
435 die "Unknown argument $arg.";
436 }
437 }
[28423]438 $basename .= "/" if !($basename =~ /[\/\\:]$/);
[19319]439 foreach my $arg (@ARGV)
440 {
[31111]441 next if $arg =~ /^--/;
[19319]442 foreach my $f (glob $arg)
443 {
444 if($f =~ /\*/) { printf "Skipping $f\n"; }
445 elsif($f =~ /\.po$/) { push(@po, $f); }
446 else { die "unknown file extension."; }
447 }
448 }
[31111]449 my %data = loadfiles(\%lang,$potfile ? loadpot($potfile) : undef, @po);
450
[19319]451 my @clang;
452 foreach my $la (sort keys %lang)
453 {
[31088]454 $la =~ s/\@/-/;
[19319]455 push(@clang, "${basename}$la.lang");
456 }
457 push(@clang, "${basename}en.lang");
458 die "There have been warning. No output.\n" if $waswarn;
459
460 createlang(\%data, @clang);
461}
Note: See TracBrowser for help on using the repository browser.