diff options
Diffstat (limited to 'rt/devel/tools/rt-message-catalog')
-rw-r--r-- | rt/devel/tools/rt-message-catalog | 224 |
1 files changed, 224 insertions, 0 deletions
diff --git a/rt/devel/tools/rt-message-catalog b/rt/devel/tools/rt-message-catalog new file mode 100644 index 000000000..f1a31587a --- /dev/null +++ b/rt/devel/tools/rt-message-catalog @@ -0,0 +1,224 @@ +#!/usr/bin/env perl +# BEGIN BPS TAGGED BLOCK {{{ +# +# COPYRIGHT: +# +# This software is Copyright (c) 1996-2015 Best Practical Solutions, LLC +# <sales@bestpractical.com> +# +# (Except where explicitly superseded by other copyright notices) +# +# +# LICENSE: +# +# This work is made available to you under the terms of Version 2 of +# the GNU General Public License. A copy of that license should have +# been provided with this software, but in any event can be snarfed +# from www.gnu.org. +# +# This work is distributed in the hope that it will be useful, but +# WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU +# General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program; if not, write to the Free Software +# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA +# 02110-1301 or visit their web page on the internet at +# http://www.gnu.org/licenses/old-licenses/gpl-2.0.html. +# +# +# CONTRIBUTION SUBMISSION POLICY: +# +# (The following paragraph is not intended to limit the rights granted +# to you to modify and distribute this software under the terms of +# the GNU General Public License and is only of importance to you if +# you choose to contribute your changes and enhancements to the +# community by submitting them to Best Practical Solutions, LLC.) +# +# By intentionally submitting any modifications, corrections or +# derivatives to this work, or any other work intended for use with +# Request Tracker, to Best Practical Solutions, LLC, you confirm that +# you are the copyright holder for those contributions and you grant +# Best Practical Solutions, LLC a nonexclusive, worldwide, irrevocable, +# royalty-free, perpetual, license to use, copy, create derivative +# works based on those contributions, and sublicense and distribute +# those contributions and any derivatives thereof. +# +# END BPS TAGGED BLOCK }}} +use strict; +use warnings; + +use Locale::PO; +use Getopt::Long; +use File::Temp 'tempdir'; + +use constant PO_DIR => 'share/po'; + +use constant BOUNDARY => 20; + +sub usage { + warn @_, "\n\n" if @_; + warn <<' USAGE'; +usages: + + rt-message-catalog stats [po-directory] + rt-message-catalog clean + rt-message-catalog rosetta download-url + rt-message-catalog extract [po-file ...] + +stats: Print stats for each translation. + +clean: Remove unused and identity translations + +rosetta: Merge translations from Launchpad's Rosetta; Requires a + Launchpad translations export url. + +extract: Extract message catalogs from source code and report common errors. + + If passed a specific translation file, only that file is updated. + (Not recommended except for debugging.) + + USAGE + exit 1; +} + +my $command = shift; +usage() unless $command; +usage("Unknown command '$command'") + unless main->can($command); + +main->can($command)->( @ARGV ); + +exit; + +sub stats { + my $dir = shift || PO_DIR; + + my $max = 0; + my %res = (); + + foreach my $po_file (<$dir/*.po>) { + my $array = Locale::PO->load_file_asarray( $po_file, "utf-8" ); + + $res{$po_file} = 0; + + my $size = 0; + foreach my $entry ( splice @$array, 1 ) { + next if $entry->obsolete; + next if $entry->reference && $entry->reference =~ /NOT FOUND IN SOURCE/; + $size++; + next unless length $entry->dequote( $entry->msgstr ); + $res{$po_file}++; + } + $max = $size if $max < $size; + } + + my $width = length($max); + foreach my $po_file ( sort { $res{$b} <=> $res{$a} } keys %res ) { + my $tr = $res{$po_file}; + my $perc = int($tr*1000/$max)/10; + printf "%-20s %${width}d/%${width}d (%.1f%%)\n", "$po_file:", $tr, $max, $perc; + } +} + +sub clean { + my $dir = shift || PO_DIR; + + foreach my $po_file (<$dir/*.po>) { + my $array = Locale::PO->load_file_asarray( $po_file, "utf-8" ); + foreach my $entry ( splice @$array, 1 ) { + # Replace identical translations with the empty string + $entry->msgstr("") if $entry->msgstr eq $entry->msgid; + + # Skip NOT FOUND IN SOURCE entries + next if $entry->obsolete; + next if $entry->reference && $entry->reference =~ /NOT FOUND IN SOURCE/; + + push @$array, $entry; + } + Locale::PO->save_file_fromarray($po_file, $array, "utf-8"); + } +} + +sub rosetta { + my $url = shift or die 'must provide Rosetta download url or directory with new po files'; + + my $dir; + if ( $url =~ m{^[a-z]+://} ) { + $dir = tempdir(); + my ($fname) = $url =~ m{([^/]+)$}; + + print "Downloading $url\n"; + require LWP::Simple; + LWP::Simple::getstore($url => "$dir/$fname"); + + print "Extracting $dir/$fname\n"; + require Archive::Extract; + my $ae = Archive::Extract->new(archive => "$dir/$fname"); + my $ok = $ae->extract( to => $dir ); + } + elsif ( -e $url && -d _ ) { + $dir = $url; + } + else { + die "Is not URL or directory: '$url'"; + } + + my @files = ( <$dir/*/*/*.po>, <$dir/*/*.po>, <$dir/*.po> ); + unless ( @files ) { + print STDERR "No files in $dir/rt/*.po and $dir/*.po\n"; + exit; + } + + for my $file ( @files ) { + my ($lang) = $file =~ m/([\w_]+)\.po/; + my $fn_orig = PO_DIR . "/$lang.po"; + + my $load_from = $fn_orig; + $load_from = PO_DIR . "/rt.pot" unless -e $load_from; + my $orig = Locale::PO->load_file_ashash( $fn_orig, "utf-8" ); + + print "$file -> $fn_orig\n"; + + my $rosetta = Locale::PO->load_file_asarray( $file, "utf-8" ); + + # We're merging in the current hash as fallbacks for the rosetta hash + my $translated = 0; + foreach my $entry ( splice @$rosetta, 1 ) { + # Skip no longer in source entries + next if $entry->obsolete; + next if $entry->reference && $entry->reference =~ /NOT FOUND IN SOURCE/; + + # Update to what the old po file had, if we have nothing + my $oldval = $orig->{$entry->msgid}; + if (not length $entry->dequote($entry->msgstr) and $oldval) { + $entry->msgstr($oldval->dequote($oldval->msgstr)); + } + + # Replace identical translations with the empty string + $entry->msgstr("") if $entry->msgstr eq $entry->msgid; + + # Drop "fuzzy" information + $entry->fuzzy_msgctxt(undef); + $entry->fuzzy_msgid(undef); + $entry->fuzzy_msgid_plural(undef); + + $translated++ if length $entry->dequote($entry->msgstr); + push @$rosetta, $entry; + } + + my $perc = int($translated/(@$rosetta - 1) * 100 + 0.5); + if ( $perc < BOUNDARY and $lang !~ /^en(_[A-Z]{2})?$/) { + unlink $fn_orig; + next; + } + + Locale::PO->save_file_fromarray($fn_orig, $rosetta, "utf-8"); + } + extract(); +} + +sub extract { + system($^X, 'devel/tools/extract-message-catalog', @_); +} |