da29ed5238
svn: r19170
499 lines
16 KiB
Python
499 lines
16 KiB
Python
#! /usr/bin/env python
|
|
#
|
|
# update_po - a gramps tool to update translations
|
|
#
|
|
# Copyright (C) 2006-2006 Kees Bakker
|
|
# Copyright (C) 2006 Brian Matherly
|
|
# Copyright (C) 2008 Stephen George
|
|
# Copyright (C) 2012
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation; either version 2 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# This program 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
|
|
import os
|
|
import sys
|
|
from optparse import OptionParser, OptionGroup
|
|
|
|
|
|
if sys.platform == 'win32':
|
|
# GetText Win 32 obtained from http://gnuwin32.sourceforge.net/packages/gettext.htm
|
|
# ....\gettext\bin\msgmerge.exe needs to be on the path
|
|
msgmergeCmd = os.path.join('C:', 'Program Files(x86)', 'gettext', 'bin', 'msgmerge.exe')
|
|
msgfmtCmd = os.path.join('C:', 'Program Files(x86)', 'gettext', 'bin', 'msgfmt.exe')
|
|
msgattribCmd = os.path.join('C:', 'Program Files(x86)', 'gettext', 'bin', 'msgattrib.exe')
|
|
xgettextCmd = os.path.join('C:', 'Program Files(x86)', 'gettext', 'bin', 'xgettext.exe')
|
|
pythonCmd = os.path.join(sys.prefix, 'bin', 'python.exe')
|
|
elif sys.platform == 'linux2' or os.name == 'darwin':
|
|
msgmergeCmd = 'msgmerge'
|
|
msgfmtCmd = 'msgfmt'
|
|
msgattribCmd = 'msgattrib'
|
|
xgettextCmd = 'xgettext'
|
|
pythonCmd = os.path.join(sys.prefix, 'bin', 'python')
|
|
|
|
def tests():
|
|
"""
|
|
Testing installed programs.
|
|
We made tests (-t flag) by displaying versions of tools if properly
|
|
installed. Cannot run all commands without 'gettext' and 'python'.
|
|
"""
|
|
|
|
try:
|
|
print("\n====='msgmerge'=(merge our translation)================\n")
|
|
os.system('''%(program)s -V''' % {'program': msgmergeCmd})
|
|
except:
|
|
print('Please, install %(program)s for updating your translation' % {'program': msgmergeCmd})
|
|
|
|
try:
|
|
print("\n==='msgfmt'=(format our translation for installation)==\n")
|
|
os.system('''%(program)s -V''' % {'program': msgfmtCmd})
|
|
except:
|
|
print('Please, install %(program)s for checking your translation' % {'program': msgfmtCmd})
|
|
|
|
try:
|
|
print("\n===='msgattrib'==(list groups of messages)=============\n")
|
|
os.system('''%(program)s -V''' % {'program': msgattribCmd})
|
|
except:
|
|
print('Please, install %(program)s for listing groups of messages' % {'program': msgattribCmd})
|
|
|
|
|
|
try:
|
|
print("\n===='xgettext' =(generate a new template)==============\n")
|
|
os.system('''%(program)s -V''' % {'program': xgettextCmd})
|
|
except:
|
|
print('Please, install %(program)s for generating a new template' % {'program': xgettextCmd})
|
|
|
|
try:
|
|
print("\n=================='python'=============================\n")
|
|
os.system('''%(program)s -V''' % {'program': pythonCmd})
|
|
except:
|
|
print('Please, install python')
|
|
|
|
|
|
# See also 'get_string' from Gramps 2.0 (sample with SAX)
|
|
|
|
def TipsParse(filename, mark):
|
|
"""
|
|
Experimental alternative to 'intltool-extract' for 'tips.xml'.
|
|
"""
|
|
|
|
# in progress ...
|
|
from xml.etree import ElementTree
|
|
|
|
tree = ElementTree.parse(filename)
|
|
root = tree.getroot()
|
|
|
|
python_v = sys.version_info
|
|
|
|
#if python_v[1] != 6:
|
|
|
|
# python 2.7
|
|
# iter() is the new name for getiterator;
|
|
# in ET 1.3, it is implemented as a generator method,
|
|
# but is otherwise identical
|
|
|
|
'''
|
|
<?xml version="1.0" encoding="UTF-8"?>
|
|
<tips>
|
|
<_tip number="1">
|
|
<b>Working with Dates</b>
|
|
<br/>
|
|
A range of dates can be given by using the format "between
|
|
January 4, 2000 and March 20, 2003". You can also indicate
|
|
the level of confidence in a date and even choose between seven
|
|
different calendars. Try the button next to the date field in the
|
|
Events Editor.
|
|
</_tip>
|
|
|
|
char *s = N_("<b>Working with Dates</b><br/>A range of dates can be
|
|
given by using the format "between January 4, 2000 and March 20,
|
|
2003". You can also indicate the level of confidence in a date
|
|
and even choose between seven different calendars. Try the button
|
|
next to the date field in the Events Editor.");
|
|
|
|
gramps.pot:
|
|
msgid ""
|
|
"<b>Working with Dates</b><br/>A range of dates can be given by using the "
|
|
"format "between January 4, 2000 and March 20, 2003". You can also "
|
|
"indicate the level of confidence in a date and even choose between seven "
|
|
"different calendars. Try the button next to the date field in the Events "
|
|
"Editor."
|
|
'''
|
|
|
|
tips = open('../src/data/tips.xml.in.h', 'w')
|
|
|
|
for key in root.getiterator(mark):
|
|
tip = ElementTree.tostring(key, encoding="UTF-8")
|
|
tip = tip.replace("<?xml version='1.0' encoding='UTF-8'?>", "")
|
|
tip = tip.replace('\n<_tip number="%(number)s">' % key.attrib, "")
|
|
tip = tip.replace("<br />", "<br/>")
|
|
tip = tip.replace("\n</_tip>\n", "</_tip>\n") # special case tip 7
|
|
tip = tip.replace("\n<b>", "<b>") # special case tip 18
|
|
tip = tip.replace("</_tip>\n\n", "")
|
|
tip = tip.replace('"', '"')
|
|
tips.write('char *s = N_("%s");\n' % tip)
|
|
|
|
tips.close()
|
|
|
|
root.clear()
|
|
|
|
def HolidaysParse(filename, mark):
|
|
"""
|
|
Experimental alternative to 'intltool-extract' for 'holidays.xml'.
|
|
"""
|
|
|
|
# in progress ...
|
|
from xml.etree import ElementTree
|
|
|
|
tree = ElementTree.parse(filename)
|
|
root = tree.getroot()
|
|
|
|
python_v = sys.version_info
|
|
|
|
#if python_v[1] != 6:
|
|
|
|
# python 2.7
|
|
# iter() is the new name for getiterator;
|
|
# in ET 1.3, it is implemented as a generator method,
|
|
# but is otherwise identical
|
|
|
|
'''
|
|
<?xml version="1.0" encoding="utf-8"?>
|
|
calendar>
|
|
<country _name="Bulgaria">
|
|
..
|
|
<country _name="Jewish Holidays">
|
|
<date _name="Yom Kippur" value="> passover(y)" offset="172"/>
|
|
|
|
char *s = N_("Bulgaria");
|
|
char *s = N_("Jewish Holidays");
|
|
char *s = N_("Yom Kippur");
|
|
|
|
gramps.pot:
|
|
msgid "Bulgaria"
|
|
msgid "Jewish Holidays"
|
|
msgid "Yom Kippur"
|
|
'''
|
|
|
|
holidays = open('../src/plugins/lib/holidays.xml.in.h', 'w')
|
|
|
|
for key in root.getiterator():
|
|
if key.attrib.get(mark):
|
|
line = key.attrib
|
|
string = line.items
|
|
name = 'char *s = N_("%(_name)s");\n' % line
|
|
holidays.write(name)
|
|
|
|
holidays.close()
|
|
|
|
root.clear()
|
|
|
|
|
|
def main():
|
|
"""
|
|
The utility for handling translation stuff.
|
|
What is need by Gramps, nothing more.
|
|
"""
|
|
|
|
parser = OptionParser(
|
|
description='This program generates a new template and '
|
|
'also provides some common features.',
|
|
usage='%prog [options]'
|
|
)
|
|
|
|
extract = OptionGroup(
|
|
parser,
|
|
"Extract Options",
|
|
"Everything around extraction for message strings."
|
|
)
|
|
parser.add_option_group(extract)
|
|
|
|
update = OptionGroup(
|
|
parser,
|
|
"Update Options",
|
|
"Everything around update for translation files."
|
|
)
|
|
parser.add_option_group(update)
|
|
|
|
trans = OptionGroup(
|
|
parser,
|
|
"Translation Options",
|
|
"Some informations around translation."
|
|
)
|
|
parser.add_option_group(trans)
|
|
|
|
parser.add_option("-t", "--test",
|
|
action="store_true", dest="test", default=False,
|
|
help="test if 'python' and 'gettext' are properly installed")
|
|
|
|
extract.add_option("-x", "--xml",
|
|
action="store_true", dest="xml", default=False,
|
|
help="extract messages from xml based file formats")
|
|
extract.add_option("-g", "--glade",
|
|
action="store_true", dest="glade", default=False,
|
|
help="extract messages from glade file format only")
|
|
extract.add_option("-c", "--clean",
|
|
action="store_true", dest="clean", default=False,
|
|
help="remove created files")
|
|
extract.add_option("-p", "--pot",
|
|
action="store_true", dest="catalog", default=False,
|
|
help="create a new catalog")
|
|
|
|
# need at least one argument (sv.po, de.po, etc ...)
|
|
update.add_option("-m", "--merge",
|
|
action="store_true", dest="merge", default=False,
|
|
help="merge lang.po files with last catalog")
|
|
update.add_option("-k", "--check",
|
|
action="store_true", dest="check", default=False,
|
|
help="check lang.po files")
|
|
|
|
# testing stage
|
|
trans.add_option("-u", "--untranslated",
|
|
action="store_true", dest="untranslated", default=False,
|
|
help="list untranslated messages")
|
|
trans.add_option("-f", "--fuzzy",
|
|
action="store_true", dest="fuzzy", default=False,
|
|
help="list fuzzy messages")
|
|
|
|
(options, args) = parser.parse_args()
|
|
|
|
if options.test:
|
|
tests()
|
|
|
|
if options.xml:
|
|
extract_xml()
|
|
|
|
if options.glade:
|
|
extract_glade()
|
|
|
|
if options.catalog:
|
|
retrieve()
|
|
|
|
if options.clean:
|
|
clean()
|
|
|
|
if options.merge:
|
|
merge(args)
|
|
|
|
if options.check:
|
|
check(args)
|
|
|
|
if options.untranslated:
|
|
untranslated(args)
|
|
|
|
if options.fuzzy:
|
|
fuzzy(args)
|
|
|
|
def listing(name, extension):
|
|
"""
|
|
List files according to extensions.
|
|
Parsing from a textual file (gramps) is faster and easy for maintenance.
|
|
Like POTFILES.in and POTFILES.skip
|
|
"""
|
|
|
|
f = open('gramps')
|
|
files = [file.strip() for file in f]
|
|
f.close()
|
|
|
|
temp = open(name, 'w')
|
|
|
|
for entry in files:
|
|
(module, ext) = os.path.splitext(entry)
|
|
if ext == extension:
|
|
temp.write(entry)
|
|
temp.write('\n')
|
|
|
|
temp.close()
|
|
|
|
|
|
def headers():
|
|
"""
|
|
Look at existing C file format headers.
|
|
Generated by 'intltool-extract' but want to get rid of this
|
|
dependency (perl, just a set of tools).
|
|
"""
|
|
|
|
headers = []
|
|
|
|
# in.h; extract_xml
|
|
if os.path.isfile('''../src/data/tips.xml.in.h'''):
|
|
headers.append('''../src/data/tips.xml.in.h''')
|
|
if os.path.isfile('''../src/plugins/lib/holidays.xml.in.h'''):
|
|
headers.append('''../src/plugins/lib/holidays.xml.in.h''')
|
|
|
|
# cosmetic
|
|
if os.path.isfile('''../data/gramps.xml.in.h'''):
|
|
headers.append('''../data/gramps.xml.in.h''')
|
|
if os.path.isfile('''../data/gramps.desktop.in.h'''):
|
|
headers.append('''../data/gramps.desktop.in.h''')
|
|
if os.path.isfile('''../data/gramps.keys.in.h'''):
|
|
headers.append('''../data/gramps.keys.in.h''')
|
|
|
|
return headers
|
|
|
|
|
|
def extract_xml():
|
|
"""
|
|
Extract translation strings from XML based, keys, mime and desktop
|
|
files. Still performed by 'intltool-update'.
|
|
Need to look at own XML files parsing and custom translation marks.
|
|
"""
|
|
|
|
#os.system('''intltool-extract --type=gettext/xml ../src/data/tips.xml.in''')
|
|
#os.system('''intltool-extract --type=gettext/xml ../src/plugins/lib/holidays.xml.in''')
|
|
|
|
TipsParse('../src/data/tips.xml.in', '_tip')
|
|
HolidaysParse('../src/plugins/lib/holidays.xml.in', '_name')
|
|
|
|
# cosmetic
|
|
# could be simple copies without .in extension
|
|
os.system('''intltool-extract --type=gettext/xml ../data/gramps.xml.in''')
|
|
os.system('''intltool-extract --type=gettext/ini ../data/gramps.desktop.in''')
|
|
os.system('''intltool-extract --type=gettext/keys ../data/gramps.keys.in''')
|
|
|
|
|
|
def create_template():
|
|
"""
|
|
Create a new file for template, if it does not exist.
|
|
"""
|
|
|
|
template = open('gramps.pot', 'w')
|
|
template.close()
|
|
|
|
|
|
def extract_glade():
|
|
"""
|
|
Extract messages from a temp file with all .glade
|
|
"""
|
|
|
|
if not os.path.isfile('gramps.pot'):
|
|
create_template()
|
|
|
|
listing('glade.txt', '.glade')
|
|
os.system('''%(xgettext)s --add-comments -j -L Glade '''
|
|
'''--from-code=UTF-8 -o gramps.pot --files-from=glade.txt'''
|
|
% {'xgettext': xgettextCmd}
|
|
)
|
|
|
|
|
|
def retrieve():
|
|
"""
|
|
Extract messages from all files used by Gramps (python, glade, xml)
|
|
"""
|
|
|
|
extract_xml()
|
|
|
|
if not os.path.isfile('gramps.pot'):
|
|
create_template()
|
|
|
|
listing('python.txt', '.py')
|
|
os.system('''%(xgettext)s --add-comments -j --directory=. -d gramps '''
|
|
'''-L Python -o gramps.pot --files-from=python.txt '''
|
|
'''--keyword=_ --keyword=ngettext '''
|
|
'''--keyword=sgettext --from-code=UTF-8''' % {'xgettext': xgettextCmd}
|
|
)
|
|
|
|
extract_glade()
|
|
|
|
# C format header (.h extension)
|
|
for h in headers():
|
|
print('xgettext for %s') % h
|
|
os.system('''%(xgettext)s --add-comments -j -o gramps.pot '''
|
|
'''--keyword=N_ --from-code=UTF-8 %(head)s'''
|
|
% {'xgettext': xgettextCmd, 'head': h}
|
|
)
|
|
|
|
clean()
|
|
|
|
|
|
def clean():
|
|
"""
|
|
Remove created files (C format headers, temp listings)
|
|
"""
|
|
|
|
for h in headers():
|
|
if os.path.isfile(h):
|
|
os.unlink(h)
|
|
print('Remove %(head)s' % {'head': h})
|
|
|
|
if os.path.isfile('python.txt'):
|
|
os.unlink('python.txt')
|
|
print("Remove 'python.txt'")
|
|
|
|
if os.path.isfile('glade.txt'):
|
|
os.unlink('glade.txt')
|
|
print("Remove 'glade.txt'")
|
|
|
|
|
|
def merge(args):
|
|
"""
|
|
Merge messages with 'gramps.pot'
|
|
"""
|
|
|
|
if not args:
|
|
print('Please, add at least one argument (sv.po, de.po).')
|
|
|
|
for arg in args:
|
|
if arg[-3:] == '.po':
|
|
print('Merge %(lang)s with current template' % {'lang': arg})
|
|
os.system('''%(msgmerge)s --no-wrap %(lang)s gramps.pot -o updated_%(lang)s''' \
|
|
% {'msgmerge': msgmergeCmd, 'lang': arg})
|
|
print("Updated file: 'updated_%(lang)s'." % {'lang': arg})
|
|
else:
|
|
print("Please, try to set an argument with .po extension like '%(arg)s.po'." % {'arg': arg})
|
|
|
|
|
|
def check(args):
|
|
"""
|
|
Check the translation file
|
|
"""
|
|
|
|
if not args:
|
|
print('Please, add at least one argument (sv.po, de.po).')
|
|
|
|
for arg in args:
|
|
if arg[-3:] == '.po':
|
|
print("Checked file: '%(lang.po)s'. See '%(txt)s.txt'." \
|
|
% {'lang.po': arg, 'txt': arg[:2]})
|
|
os.system('''%(python)s ./check_po --skip-fuzzy ./%(lang.po)s > %(lang)s.txt''' \
|
|
% {'python': pythonCmd, 'lang.po': arg, 'lang': arg[:2]})
|
|
os.system('''%(msgfmt)s -c -v %(lang.po)s''' % {'msgfmt': msgfmtCmd, 'lang.po': arg})
|
|
else:
|
|
print("Please, try to set an argument with .po extension like '%(arg)s.po'." % {'arg': arg})
|
|
|
|
def untranslated(args):
|
|
"""
|
|
List untranslated messages
|
|
"""
|
|
|
|
if len(args) > 1:
|
|
print('Please, use only one argument (ex: fr.po).')
|
|
return
|
|
|
|
os.system('''%(msgattrib)s --untranslated %(lang.po)s''' % {'msgattrib': msgattribCmd, 'lang.po': args[0]})
|
|
|
|
def fuzzy(args):
|
|
"""
|
|
List fuzzy messages
|
|
"""
|
|
|
|
if len(args) > 1:
|
|
print('Please, use only one argument (ex: fr.po).')
|
|
return
|
|
|
|
os.system('''%(msgattrib)s --only-fuzzy --no-obsolete %(lang.po)s''' % {'msgattrib': msgattribCmd, 'lang.po': args[0]})
|
|
|
|
if __name__ == "__main__":
|
|
main()
|