2015-07-28 20:56:23 +02:00
|
|
|
#! /usr/bin/env python3
|
2012-03-12 16:43:12 +00:00
|
|
|
#
|
2012-03-15 17:11:31 +00:00
|
|
|
# update_po - a gramps tool to update translations
|
2012-03-12 16:43:12 +00:00
|
|
|
#
|
|
|
|
# Copyright (C) 2006-2006 Kees Bakker
|
2012-03-14 14:10:21 +00:00
|
|
|
# Copyright (C) 2006 Brian Matherly
|
|
|
|
# Copyright (C) 2008 Stephen George
|
2012-03-12 16:43:12 +00:00
|
|
|
# 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
|
2014-08-08 19:29:07 -07:00
|
|
|
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
2013-05-05 19:41:13 +00:00
|
|
|
#
|
2012-10-16 08:51:05 +00:00
|
|
|
|
|
|
|
"""
|
|
|
|
update_po.py for Gramps translations.
|
|
|
|
|
2016-05-18 05:47:34 -04:00
|
|
|
Examples:
|
2012-10-16 08:51:05 +00:00
|
|
|
python update_po.py -t
|
|
|
|
|
|
|
|
Tests if 'gettext' and 'python' are well configured.
|
|
|
|
|
|
|
|
python update_po.py -h
|
|
|
|
|
|
|
|
Calls help and command line interface.
|
|
|
|
|
|
|
|
python update_po.py -p
|
|
|
|
|
|
|
|
Generates a new template/catalog (gramps.pot).
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-10-16 08:51:05 +00:00
|
|
|
python update_po.py -m de.po
|
|
|
|
|
|
|
|
Merges 'de.po' file with 'gramps.pot'.
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-10-16 08:51:05 +00:00
|
|
|
python update_po.py -k de.po
|
|
|
|
|
|
|
|
Checks 'de.po' file, tests to compile and generates a textual resume.
|
|
|
|
"""
|
|
|
|
|
2012-10-03 12:16:06 +00:00
|
|
|
from __future__ import print_function
|
2012-03-12 16:43:12 +00:00
|
|
|
|
|
|
|
import os
|
|
|
|
import sys
|
2012-11-30 20:33:44 +00:00
|
|
|
from argparse import ArgumentParser
|
2012-03-12 16:43:12 +00:00
|
|
|
|
2012-12-13 20:43:32 +00:00
|
|
|
# Windows OS
|
2012-03-12 16:43:12 +00:00
|
|
|
|
2016-04-19 09:15:05 +10:00
|
|
|
if sys.platform == 'win32':
|
2012-03-15 17:11:31 +00:00
|
|
|
# GetText Win 32 obtained from http://gnuwin32.sourceforge.net/packages/gettext.htm
|
|
|
|
# ....\gettext\bin\msgmerge.exe needs to be on the path
|
2012-03-29 09:19:54 +00:00
|
|
|
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')
|
2012-03-19 18:46:50 +00:00
|
|
|
xgettextCmd = os.path.join('C:', 'Program Files(x86)', 'gettext', 'bin', 'xgettext.exe')
|
2015-07-29 11:06:46 +02:00
|
|
|
pythonCmd = os.path.join(sys.prefix, 'bin', 'python.exe')
|
2012-12-13 20:43:32 +00:00
|
|
|
|
|
|
|
# Others OS
|
|
|
|
|
2015-03-01 09:00:38 -08:00
|
|
|
elif sys.platform in ['linux', 'linux2', 'darwin', 'cygwin']:
|
2012-03-15 17:11:31 +00:00
|
|
|
msgmergeCmd = 'msgmerge'
|
|
|
|
msgfmtCmd = 'msgfmt'
|
2012-03-19 16:00:19 +00:00
|
|
|
msgattribCmd = 'msgattrib'
|
2012-03-16 18:35:35 +00:00
|
|
|
xgettextCmd = 'xgettext'
|
2015-07-28 20:56:23 +02:00
|
|
|
pythonCmd = os.path.join(sys.prefix, 'bin', 'python3')
|
2012-10-03 12:16:06 +00:00
|
|
|
else:
|
2013-01-20 19:35:35 +00:00
|
|
|
print("Found platform %s, OS %s" % (sys.platform, os.name))
|
|
|
|
print ("Update PO ERROR: unknown system, don't know msgmerge, ... commands")
|
2012-10-03 12:16:06 +00:00
|
|
|
sys.exit(0)
|
2012-12-13 20:43:32 +00:00
|
|
|
|
|
|
|
# List of available languages, useful for grouped actions
|
|
|
|
|
|
|
|
# need files with po extension
|
2012-12-02 17:22:40 +00:00
|
|
|
LANG = [file for file in os.listdir('.') if file.endswith('.po')]
|
2012-12-13 20:43:32 +00:00
|
|
|
# add a special 'all' argument (for 'check' and 'merge' arguments)
|
2012-12-02 17:22:40 +00:00
|
|
|
LANG.append("all")
|
2012-12-13 20:43:32 +00:00
|
|
|
# visual polish on the languages list
|
2012-12-02 17:22:40 +00:00
|
|
|
LANG.sort()
|
2012-03-15 17:11:31 +00:00
|
|
|
|
|
|
|
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:
|
2012-10-03 12:16:06 +00:00
|
|
|
print ("\n====='msgmerge'=(merge our translation)================\n")
|
2012-03-15 17:11:31 +00:00
|
|
|
os.system('''%(program)s -V''' % {'program': msgmergeCmd})
|
|
|
|
except:
|
2016-04-19 09:15:05 +10:00
|
|
|
print ('Please, install %(program)s for updating your translation'
|
2012-10-03 12:16:06 +00:00
|
|
|
% {'program': msgmergeCmd})
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-03-15 17:11:31 +00:00
|
|
|
try:
|
2012-10-03 12:16:06 +00:00
|
|
|
print ("\n==='msgfmt'=(format our translation for installation)==\n")
|
2012-03-15 17:11:31 +00:00
|
|
|
os.system('''%(program)s -V''' % {'program': msgfmtCmd})
|
|
|
|
except:
|
2016-04-19 09:15:05 +10:00
|
|
|
print ('Please, install %(program)s for checking your translation'
|
2012-10-03 12:16:06 +00:00
|
|
|
% {'program': msgfmtCmd})
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-03-15 17:11:31 +00:00
|
|
|
try:
|
2012-10-03 12:16:06 +00:00
|
|
|
print ("\n===='msgattrib'==(list groups of messages)=============\n")
|
2012-03-19 16:00:19 +00:00
|
|
|
os.system('''%(program)s -V''' % {'program': msgattribCmd})
|
|
|
|
except:
|
2016-04-19 09:15:05 +10:00
|
|
|
print ('Please, install %(program)s for listing groups of messages'
|
2012-10-03 12:16:06 +00:00
|
|
|
% {'program': msgattribCmd})
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-03-19 16:00:19 +00:00
|
|
|
try:
|
|
|
|
print("\n===='xgettext' =(generate a new template)==============\n")
|
2012-03-16 18:35:35 +00:00
|
|
|
os.system('''%(program)s -V''' % {'program': xgettextCmd})
|
|
|
|
except:
|
2016-04-19 09:15:05 +10:00
|
|
|
print ('Please, install %(program)s for generating a new template'
|
2012-10-03 12:16:06 +00:00
|
|
|
% {'program': xgettextCmd})
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-03-16 18:35:35 +00:00
|
|
|
try:
|
2012-03-19 16:00:19 +00:00
|
|
|
print("\n=================='python'=============================\n")
|
2012-03-15 17:11:31 +00:00
|
|
|
os.system('''%(program)s -V''' % {'program': pythonCmd})
|
|
|
|
except:
|
2012-10-03 12:16:06 +00:00
|
|
|
print ('Please, install python')
|
2012-12-13 20:43:32 +00:00
|
|
|
|
2012-03-29 06:29:55 +00:00
|
|
|
def TipsParse(filename, mark):
|
2012-03-15 17:11:31 +00:00
|
|
|
"""
|
2012-03-29 06:29:55 +00:00
|
|
|
Experimental alternative to 'intltool-extract' for 'tips.xml'.
|
2012-03-15 17:11:31 +00:00
|
|
|
"""
|
|
|
|
from xml.etree import ElementTree
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-03-15 17:11:31 +00:00
|
|
|
tree = ElementTree.parse(filename)
|
|
|
|
root = tree.getroot()
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-03-19 16:00:19 +00:00
|
|
|
'''
|
|
|
|
<?xml version="1.0" encoding="UTF-8"?>
|
|
|
|
<tips>
|
|
|
|
<_tip number="1">
|
|
|
|
<b>Working with Dates</b>
|
|
|
|
<br/>
|
2016-04-19 09:15:05 +10:00
|
|
|
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
|
2012-03-19 16:00:19 +00:00
|
|
|
different calendars. Try the button next to the date field in the
|
|
|
|
Events Editor.
|
|
|
|
</_tip>
|
2016-04-19 09:15:05 +10:00
|
|
|
|
|
|
|
char *s = N_("<b>Working with Dates</b><br/>A range of dates can be
|
2012-03-29 06:29:55 +00:00
|
|
|
given by using the format "between January 4, 2000 and March 20,
|
2016-04-19 09:15:05 +10:00
|
|
|
2003". You can also indicate the level of confidence in a date
|
|
|
|
and even choose between seven different calendars. Try the button
|
2012-03-29 06:29:55 +00:00
|
|
|
next to the date field in the Events Editor.");
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-03-19 16:00:19 +00:00
|
|
|
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."
|
|
|
|
'''
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2016-04-19 09:08:21 +10:00
|
|
|
with open('../data/tips.xml.in.h', 'w') as tips:
|
|
|
|
marklist = root.iter(mark)
|
|
|
|
for key in marklist:
|
|
|
|
tip = ElementTree.tostring(key, encoding="UTF-8", method="xml")
|
|
|
|
if sys.version_info[0] < 3:
|
|
|
|
tip = tip.replace("<?xml version='1.0' encoding='UTF-8'?>", "")
|
|
|
|
tip = tip.replace('\n<_tip number="%(number)s">' % key.attrib, "")
|
|
|
|
else: # python3 support
|
|
|
|
tip = tip.decode("utf-8")
|
|
|
|
tip = tip.replace('<_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)
|
2013-01-05 14:41:54 +00:00
|
|
|
print ('Wrote ../data/tips.xml.in.h')
|
2012-03-29 06:29:55 +00:00
|
|
|
root.clear()
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-03-29 06:29:55 +00:00
|
|
|
def HolidaysParse(filename, mark):
|
|
|
|
"""
|
|
|
|
Experimental alternative to 'intltool-extract' for 'holidays.xml'.
|
|
|
|
"""
|
|
|
|
from xml.etree import ElementTree
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-03-29 06:29:55 +00:00
|
|
|
tree = ElementTree.parse(filename)
|
|
|
|
root = tree.getroot()
|
2012-10-03 12:16:06 +00:00
|
|
|
ellist = root.iter()
|
2012-03-29 06:29:55 +00:00
|
|
|
|
2012-03-19 16:00:19 +00:00
|
|
|
'''
|
|
|
|
<?xml version="1.0" encoding="utf-8"?>
|
|
|
|
calendar>
|
|
|
|
<country _name="Bulgaria">
|
|
|
|
..
|
|
|
|
<country _name="Jewish Holidays">
|
|
|
|
<date _name="Yom Kippur" value="> passover(y)" offset="172"/>
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-03-29 06:29:55 +00:00
|
|
|
char *s = N_("Bulgaria");
|
|
|
|
char *s = N_("Jewish Holidays");
|
|
|
|
char *s = N_("Yom Kippur");
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-03-19 16:00:19 +00:00
|
|
|
gramps.pot:
|
|
|
|
msgid "Bulgaria"
|
|
|
|
msgid "Jewish Holidays"
|
|
|
|
msgid "Yom Kippur"
|
|
|
|
'''
|
2016-04-19 09:08:21 +10:00
|
|
|
with open('../data/holidays.xml.in.h', 'w') as holidays:
|
|
|
|
for key in ellist:
|
|
|
|
if key.attrib.get(mark):
|
|
|
|
line = key.attrib
|
|
|
|
string = line.items
|
|
|
|
# mapping via the line dict (_name is the key)
|
|
|
|
name = 'char *s = N_("%(_name)s");\n' % line
|
|
|
|
holidays.write(name)
|
2016-04-17 06:09:43 +10:00
|
|
|
print ('Wrote ../data/holidays.xml.in.h')
|
2012-03-29 06:29:55 +00:00
|
|
|
root.clear()
|
2012-03-12 16:43:12 +00:00
|
|
|
|
2012-10-04 18:35:40 +00:00
|
|
|
|
|
|
|
def XmlParse(filename, mark):
|
|
|
|
"""
|
2013-10-07 09:07:40 +00:00
|
|
|
Experimental alternative to 'intltool-extract' for 'file.xml.in'.
|
2012-10-04 18:35:40 +00:00
|
|
|
"""
|
|
|
|
from xml.etree import ElementTree
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-10-04 18:35:40 +00:00
|
|
|
tree = ElementTree.parse(filename)
|
|
|
|
root = tree.getroot()
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-10-04 18:35:40 +00:00
|
|
|
'''
|
|
|
|
<?xml version="1.0" encoding="UTF-8"?>
|
|
|
|
|
|
|
|
<mime-info xmlns="http://www.freedesktop.org/standards/shared-mime-info">
|
|
|
|
<mime-type type="application/x-gramps">
|
|
|
|
<_comment>Gramps database</_comment>
|
|
|
|
<glob pattern="*.grdb"/>
|
|
|
|
</mime-type>
|
|
|
|
<mime-type type="application/x-gedcom">
|
|
|
|
<_comment>GEDCOM</_comment>
|
|
|
|
<glob pattern="*.ged"/>
|
|
|
|
<glob pattern="*.gedcom"/>
|
|
|
|
<glob pattern="*.GED"/>
|
|
|
|
<glob pattern="*.GEDCOM"/>
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-10-04 18:35:40 +00:00
|
|
|
msgid "Gramps database"
|
|
|
|
msgid "GEDCOM"
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2016-05-16 12:09:30 +02:00
|
|
|
<description>
|
2013-10-07 09:07:40 +00:00
|
|
|
<_p> Gramps is a free software project and community.
|
|
|
|
We strive to produce a genealogy program that is both intuitive for hobbyists
|
|
|
|
and feature-complete for professional genealogists.
|
|
|
|
</p>
|
2016-05-16 12:09:30 +02:00
|
|
|
</description>
|
2012-10-04 18:35:40 +00:00
|
|
|
'''
|
2016-04-19 09:08:21 +10:00
|
|
|
|
2016-04-19 09:15:05 +10:00
|
|
|
with open(filename + '.h', 'w') as head:
|
2013-10-07 09:07:40 +00:00
|
|
|
for key in root.iter():
|
2016-04-19 09:08:21 +10:00
|
|
|
if key.tag == '{http://www.freedesktop.org/standards/shared-mime-info}%s' % mark:
|
2013-10-07 09:42:01 +00:00
|
|
|
comment = 'char *s = N_("%s");\n' % key.text
|
2016-05-16 12:09:30 +02:00
|
|
|
head.write(comment)
|
|
|
|
if key.tag == mark: #appdata
|
|
|
|
comment = 'char *s = N_("%s");\n' % key.text
|
2013-10-07 09:42:01 +00:00
|
|
|
head.write(comment)
|
2016-04-19 09:08:21 +10:00
|
|
|
|
|
|
|
if root.tag == 'application':
|
|
|
|
for key in root.iter():
|
|
|
|
if key.tag == mark:
|
|
|
|
comment = 'char *s = N_("%s");\n' % key.text
|
|
|
|
head.write(comment)
|
|
|
|
|
2013-10-07 09:07:40 +00:00
|
|
|
print ('Wrote %s' % filename)
|
2012-10-04 18:35:40 +00:00
|
|
|
root.clear()
|
|
|
|
|
|
|
|
def DesktopParse(filename):
|
|
|
|
"""
|
|
|
|
Experimental alternative to 'intltool-extract' for 'gramps.desktop'.
|
|
|
|
"""
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-10-04 18:35:40 +00:00
|
|
|
'''
|
|
|
|
[Desktop Entry]
|
|
|
|
_Name=Gramps
|
|
|
|
_GenericName=Genealogy System
|
|
|
|
_X-GNOME-FullName=Gramps Genealogy System
|
2016-04-19 09:15:05 +10:00
|
|
|
_Comment=Manage genealogical information,
|
2012-10-04 18:35:40 +00:00
|
|
|
perform genealogical research and analysis
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-10-04 18:35:40 +00:00
|
|
|
msgid "Gramps"
|
|
|
|
msgid "Genealogy System"
|
|
|
|
msgid "Gramps Genealogy System"
|
|
|
|
msgid ""
|
2016-05-18 05:47:34 -04:00
|
|
|
"Manage genealogical information,
|
2012-10-04 18:35:40 +00:00
|
|
|
perform genealogical research and analysis"
|
|
|
|
'''
|
2016-04-19 09:08:21 +10:00
|
|
|
with open('../data/gramps.desktop.in.h', 'w') as desktop:
|
|
|
|
|
|
|
|
with open(filename) as f:
|
|
|
|
lines = [file.strip() for file in f]
|
|
|
|
|
|
|
|
for line in lines:
|
|
|
|
if line[0] == '_':
|
|
|
|
for i in range(len(line)):
|
|
|
|
if line[i] == '=':
|
|
|
|
val = 'char *s = N_("%s");\n' % line[i+1:len(line)]
|
|
|
|
desktop.write(val)
|
|
|
|
|
2012-10-04 18:35:40 +00:00
|
|
|
print ('Wrote ../data/gramps.desktop.in.h')
|
|
|
|
|
|
|
|
def KeyParse(filename, mark):
|
|
|
|
"""
|
|
|
|
Experimental alternative to 'intltool-extract' for 'gramps.keys'.
|
|
|
|
"""
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-10-04 18:35:40 +00:00
|
|
|
'''
|
|
|
|
application/x-gramps-xml:
|
2012-12-13 20:43:32 +00:00
|
|
|
_description=Gramps XML database
|
|
|
|
default_action_type=application
|
|
|
|
short_list_application_ids=gramps
|
|
|
|
short_list_application_ids_for_novice_user_level=gramps
|
|
|
|
short_list_application_ids_for_intermediate_user_level=gramps
|
|
|
|
short_list_application_ids_for_advanced_user_level=gramps
|
|
|
|
category=Documents/Genealogy
|
|
|
|
icon-filename=/usr/share/gramps/gramps.png
|
|
|
|
open=gramps %f
|
2012-10-04 18:35:40 +00:00
|
|
|
|
|
|
|
application/x-gedcom:
|
2012-12-13 20:43:32 +00:00
|
|
|
_description=GEDCOM
|
|
|
|
default_action_type=application
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-10-04 18:35:40 +00:00
|
|
|
msgid "Gramps XML database"
|
|
|
|
msgid "GEDCOM"
|
|
|
|
'''
|
2016-04-19 09:08:21 +10:00
|
|
|
with open('../data/gramps.keys.in.h', 'w') as key:
|
|
|
|
|
|
|
|
with open(filename) as f:
|
|
|
|
lines = [file for file in f]
|
|
|
|
|
|
|
|
temp = []
|
|
|
|
|
|
|
|
for line in lines:
|
|
|
|
for i in range(len(line)):
|
|
|
|
if line[i:i+12] == mark:
|
|
|
|
temp.append(line.strip())
|
|
|
|
|
|
|
|
for t in temp:
|
|
|
|
for i in range(len(t)):
|
|
|
|
if t[i] == '=':
|
|
|
|
val = 'char *s = N_("%s");\n' % t[i+1:len(t)]
|
|
|
|
key.write(val)
|
|
|
|
|
2012-10-04 18:35:40 +00:00
|
|
|
print ('Wrote ../data/gramps.keys.in.h')
|
|
|
|
|
2012-03-12 16:43:12 +00:00
|
|
|
def main():
|
2012-03-15 17:11:31 +00:00
|
|
|
"""
|
|
|
|
The utility for handling translation stuff.
|
|
|
|
What is need by Gramps, nothing more.
|
|
|
|
"""
|
2016-04-19 09:15:05 +10:00
|
|
|
|
|
|
|
parser = ArgumentParser(
|
2012-03-15 17:11:31 +00:00
|
|
|
description='This program generates a new template and '
|
2016-04-19 09:15:05 +10:00
|
|
|
'also provides some common features.',
|
2012-03-12 16:43:12 +00:00
|
|
|
)
|
2012-11-30 20:33:44 +00:00
|
|
|
parser.add_argument("-t", "--test",
|
2012-12-01 09:45:14 +00:00
|
|
|
action="store_true", dest="test", default=True,
|
2012-10-03 12:16:06 +00:00
|
|
|
help="test if 'python' and 'gettext' are properly installed")
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-11-30 20:33:44 +00:00
|
|
|
parser.add_argument("-x", "--xml",
|
|
|
|
action="store_true", dest="xml", default=False,
|
2012-10-03 12:16:06 +00:00
|
|
|
help="extract messages from xml based file formats")
|
2012-11-30 20:33:44 +00:00
|
|
|
parser.add_argument("-g", "--glade",
|
|
|
|
action="store_true", dest="glade", default=False,
|
2012-10-03 12:16:06 +00:00
|
|
|
help="extract messages from glade file format only")
|
2012-11-30 20:33:44 +00:00
|
|
|
parser.add_argument("-c", "--clean",
|
|
|
|
action="store_true", dest="clean", default=False,
|
2012-10-03 12:16:06 +00:00
|
|
|
help="remove created files")
|
2016-05-18 05:47:34 -04:00
|
|
|
parser.add_argument("-p", "--pot",
|
2012-11-30 20:33:44 +00:00
|
|
|
action="store_true", dest="catalog", default=False,
|
2012-10-03 12:16:06 +00:00
|
|
|
help="create a new catalog")
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-11-30 20:33:44 +00:00
|
|
|
update = parser.add_argument_group('Update', 'Maintenance around translations')
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-03-15 17:11:31 +00:00
|
|
|
# need at least one argument (sv.po, de.po, etc ...)
|
2012-11-30 20:33:44 +00:00
|
|
|
|
2016-04-19 09:15:05 +10:00
|
|
|
# lang.po files maintenance
|
2012-12-02 17:29:13 +00:00
|
|
|
update.add_argument("-m", dest="merge",
|
2012-12-02 17:22:40 +00:00
|
|
|
choices=LANG,
|
2012-10-03 12:16:06 +00:00
|
|
|
help="merge lang.po files with last catalog")
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-12-02 17:29:13 +00:00
|
|
|
update.add_argument("-k", dest="check",
|
2012-12-02 17:22:40 +00:00
|
|
|
choices=LANG,
|
2012-10-03 12:16:06 +00:00
|
|
|
help="check lang.po files")
|
2016-04-19 09:15:05 +10:00
|
|
|
|
|
|
|
# testing stage
|
2012-12-02 14:09:44 +00:00
|
|
|
trans = parser.add_argument_group('Translation', 'Display content of translations file')
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-11-30 20:33:44 +00:00
|
|
|
# need one argument (eg, de.po)
|
2016-04-19 09:15:05 +10:00
|
|
|
|
|
|
|
trans.add_argument("-u", dest="untranslated",
|
2012-12-01 10:16:44 +00:00
|
|
|
choices=[file for file in os.listdir('.') if file.endswith('.po')],
|
2012-10-03 12:16:06 +00:00
|
|
|
help="list untranslated messages")
|
2012-12-02 17:29:13 +00:00
|
|
|
trans.add_argument("-f", dest="fuzzy",
|
2012-12-01 10:16:44 +00:00
|
|
|
choices=[file for file in os.listdir('.') if file.endswith('.po')],
|
2012-10-03 12:16:06 +00:00
|
|
|
help="list fuzzy messages")
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-11-30 20:33:44 +00:00
|
|
|
args = parser.parse_args()
|
2012-12-01 09:45:14 +00:00
|
|
|
namespace, extra = parser.parse_known_args()
|
2012-11-30 20:33:44 +00:00
|
|
|
|
|
|
|
if args.test:
|
2012-03-15 17:11:31 +00:00
|
|
|
tests()
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-11-30 20:33:44 +00:00
|
|
|
if args.xml:
|
2012-03-15 17:11:31 +00:00
|
|
|
extract_xml()
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-11-30 20:33:44 +00:00
|
|
|
if args.glade:
|
2012-10-03 15:55:36 +00:00
|
|
|
create_filesfile()
|
2012-03-15 17:11:31 +00:00
|
|
|
extract_glade()
|
2012-10-03 15:55:36 +00:00
|
|
|
if os.path.isfile('tmpfiles'):
|
|
|
|
os.unlink('tmpfiles')
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-11-30 20:33:44 +00:00
|
|
|
if args.catalog:
|
2012-03-15 17:11:31 +00:00
|
|
|
retrieve()
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-11-30 20:33:44 +00:00
|
|
|
if args.clean:
|
2012-03-15 17:11:31 +00:00
|
|
|
clean()
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-12-01 10:10:44 +00:00
|
|
|
if args.merge:
|
2012-12-13 20:43:32 +00:00
|
|
|
#retrieve() windows os?
|
2012-12-02 17:22:40 +00:00
|
|
|
if sys.argv[2:] == ['all']:
|
|
|
|
sys.argv[2:] = LANG
|
2012-11-30 20:33:44 +00:00
|
|
|
merge(sys.argv[2:])
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-12-01 10:10:44 +00:00
|
|
|
if args.check:
|
2012-12-13 20:43:32 +00:00
|
|
|
#retrieve() windows os?
|
2012-12-02 17:22:40 +00:00
|
|
|
if sys.argv[2:] == ['all']:
|
|
|
|
sys.argv[2:] = LANG
|
2012-11-30 20:33:44 +00:00
|
|
|
check(sys.argv[2:])
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-12-01 10:16:44 +00:00
|
|
|
if args.untranslated:
|
2012-11-30 20:33:44 +00:00
|
|
|
untranslated(sys.argv[2:])
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-12-01 10:16:44 +00:00
|
|
|
if args.fuzzy:
|
2012-11-30 20:33:44 +00:00
|
|
|
fuzzy(sys.argv[2:])
|
2012-10-03 15:37:44 +00:00
|
|
|
|
|
|
|
def create_filesfile():
|
|
|
|
"""
|
|
|
|
Create a file with all files that we should translate.
|
2016-04-19 09:15:05 +10:00
|
|
|
These are all python files not in POTFILES.skip added with those in
|
2012-10-03 15:37:44 +00:00
|
|
|
POTFILES.in
|
|
|
|
"""
|
|
|
|
dir = os.getcwd()
|
|
|
|
topdir = os.path.normpath(os.path.join(dir, '..', 'gramps'))
|
|
|
|
lentopdir = len(topdir)
|
2016-04-19 09:08:21 +10:00
|
|
|
with open('POTFILES.in') as f:
|
|
|
|
infiles = dict(['../' + file.strip(), None] for file in f if file.strip()
|
|
|
|
and not file[0]=='#')
|
|
|
|
|
|
|
|
with open('POTFILES.skip') as f:
|
|
|
|
notinfiles = dict(['../' + file.strip(), None] for file in f if file
|
|
|
|
and not file[0]=='#')
|
|
|
|
|
2012-10-03 15:37:44 +00:00
|
|
|
for (dirpath, dirnames, filenames) in os.walk(topdir):
|
|
|
|
root, subdir = os.path.split(dirpath)
|
|
|
|
if subdir.startswith("."):
|
|
|
|
#don't continue in this dir
|
|
|
|
dirnames[:] = []
|
|
|
|
continue
|
|
|
|
for dirname in dirnames:
|
|
|
|
# Skip hidden and system directories:
|
|
|
|
if dirname.startswith(".") or dirname in ["po", "locale"]:
|
|
|
|
dirnames.remove(dirname)
|
|
|
|
#add the files which are python or glade files
|
|
|
|
# if the directory does not exist or is a link, do nothing
|
|
|
|
if not os.path.isdir(dirpath) or os.path.islink(dirpath):
|
|
|
|
continue
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-10-03 15:37:44 +00:00
|
|
|
for filename in os.listdir(dirpath):
|
|
|
|
name = os.path.split(filename)[1]
|
|
|
|
if name.endswith('.py') or name.endswith('.glade'):
|
|
|
|
full_filename = os.path.join(dirpath, filename)
|
|
|
|
#Skip the file if in POTFILES.skip
|
|
|
|
if full_filename[lentopdir:] in notinfiles:
|
2012-10-16 18:39:40 +00:00
|
|
|
infiles['../gramps' + full_filename[lentopdir:]] = None
|
2012-10-03 15:37:44 +00:00
|
|
|
#now we write out all the files in form ../gramps/filename
|
2016-04-19 09:08:21 +10:00
|
|
|
with open('tmpfiles', 'w') as f:
|
|
|
|
for file in sorted(infiles.keys()):
|
|
|
|
f.write(file)
|
|
|
|
f.write('\n')
|
2012-10-03 15:37:44 +00:00
|
|
|
|
2012-10-03 12:16:06 +00:00
|
|
|
def listing(name, extensionlist):
|
2012-03-15 17:11:31 +00:00
|
|
|
"""
|
|
|
|
List files according to extensions.
|
|
|
|
Parsing from a textual file (gramps) is faster and easy for maintenance.
|
|
|
|
Like POTFILES.in and POTFILES.skip
|
|
|
|
"""
|
2016-04-19 09:08:21 +10:00
|
|
|
|
|
|
|
with open('tmpfiles') as f:
|
|
|
|
files = [file.strip() for file in f if file and not file[0]=='#']
|
|
|
|
|
|
|
|
with open(name, 'w') as temp:
|
|
|
|
for entry in files:
|
|
|
|
for ext in extensionlist:
|
|
|
|
if entry.endswith(ext):
|
|
|
|
temp.write(entry)
|
|
|
|
temp.write('\n')
|
|
|
|
break
|
|
|
|
|
2012-03-15 17:11:31 +00:00
|
|
|
def headers():
|
|
|
|
"""
|
|
|
|
Look at existing C file format headers.
|
2016-04-19 09:15:05 +10:00
|
|
|
Generated by 'intltool-extract' but want to get rid of this
|
2012-03-15 17:11:31 +00:00
|
|
|
dependency (perl, just a set of tools).
|
|
|
|
"""
|
|
|
|
headers = []
|
2012-10-03 12:16:06 +00:00
|
|
|
|
2012-03-15 17:11:31 +00:00
|
|
|
# in.h; extract_xml
|
2013-01-05 14:41:54 +00:00
|
|
|
if os.path.isfile('''../data/tips.xml.in.h'''):
|
|
|
|
headers.append('''../data/tips.xml.in.h''')
|
2016-04-16 16:38:43 +10:00
|
|
|
if os.path.isfile('''../data/holidays.xml.in.h'''):
|
|
|
|
headers.append('''../data/holidays.xml.in.h''')
|
2012-03-15 17:11:31 +00:00
|
|
|
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''')
|
2013-10-07 09:07:40 +00:00
|
|
|
if os.path.isfile('''../data/gramps.appdata.xml.in.h'''):
|
|
|
|
headers.append('''../data/gramps.appdata.xml.in.h''')
|
2013-05-08 09:25:10 +00:00
|
|
|
if os.path.isfile('''gtklist.h'''):
|
|
|
|
headers.append('''gtklist.h''')
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-03-15 17:11:31 +00:00
|
|
|
return headers
|
2012-10-03 12:16:06 +00:00
|
|
|
|
2012-03-15 17:11:31 +00:00
|
|
|
def extract_xml():
|
|
|
|
"""
|
|
|
|
Extract translation strings from XML based, keys, mime and desktop
|
2012-10-04 18:35:40 +00:00
|
|
|
files. Own XML files parsing and custom translation marks.
|
2012-03-15 17:11:31 +00:00
|
|
|
"""
|
2016-04-16 16:38:43 +10:00
|
|
|
HolidaysParse('../data/holidays.xml.in', '_name')
|
2013-01-05 14:41:54 +00:00
|
|
|
TipsParse('../data/tips.xml.in', '_tip')
|
2012-10-04 18:35:40 +00:00
|
|
|
XmlParse('../data/gramps.xml.in', '_comment')
|
2013-10-07 09:07:40 +00:00
|
|
|
XmlParse('../data/gramps.appdata.xml.in', '_p')
|
2012-10-04 18:35:40 +00:00
|
|
|
DesktopParse('../data/gramps.desktop.in')
|
|
|
|
KeyParse('../data/gramps.keys.in', '_description')
|
2016-04-19 09:08:21 +10:00
|
|
|
|
2012-03-15 17:11:31 +00:00
|
|
|
def create_template():
|
|
|
|
"""
|
|
|
|
Create a new file for template, if it does not exist.
|
|
|
|
"""
|
2016-04-19 09:08:21 +10:00
|
|
|
with open('gramps.pot', 'w') as template:
|
2016-04-19 09:15:05 +10:00
|
|
|
pass
|
2016-04-19 09:08:21 +10:00
|
|
|
|
2012-03-15 17:11:31 +00:00
|
|
|
def extract_glade():
|
|
|
|
"""
|
|
|
|
Extract messages from a temp file with all .glade
|
|
|
|
"""
|
|
|
|
if not os.path.isfile('gramps.pot'):
|
|
|
|
create_template()
|
2012-03-12 16:43:12 +00:00
|
|
|
|
2012-10-03 12:16:06 +00:00
|
|
|
listing('glade.txt', ['.glade'])
|
2014-04-17 16:43:46 +02:00
|
|
|
os.system('''%(xgettext)s -F --add-comments -j -L Glade '''
|
2012-03-15 17:11:31 +00:00
|
|
|
'''--from-code=UTF-8 -o gramps.pot --files-from=glade.txt'''
|
2012-03-16 18:35:35 +00:00
|
|
|
% {'xgettext': xgettextCmd}
|
2012-03-15 17:11:31 +00:00
|
|
|
)
|
|
|
|
|
2013-05-08 09:25:10 +00:00
|
|
|
def extract_gtkbuilder():
|
|
|
|
"""
|
|
|
|
Temp workaround for xgettext bug (< gettext 0.18.3)
|
|
|
|
https://savannah.gnu.org/bugs/index.php?29216
|
|
|
|
See bug reports #6595, #5621
|
|
|
|
"""
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2013-05-08 09:25:10 +00:00
|
|
|
from xml.etree import ElementTree
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2013-05-08 09:25:10 +00:00
|
|
|
'''
|
|
|
|
<?xml version="1.0" encoding="UTF-8"?>
|
|
|
|
<interface>
|
|
|
|
<!-- interface-requires gtk+ 3.0 -->
|
|
|
|
|
|
|
|
<object class="GtkListStore" id="model1">
|
|
|
|
<columns>
|
|
|
|
<!-- column-name gchararray -->
|
|
|
|
<column type="gchararray"/>
|
|
|
|
</columns>
|
|
|
|
<data>
|
|
|
|
<row>
|
|
|
|
<col id="0" translatable="yes">All rules must apply</col>
|
|
|
|
</row>
|
|
|
|
<row>
|
|
|
|
<col id="0" translatable="yes">At least one rule must apply</col>
|
|
|
|
</row>
|
|
|
|
<row>
|
|
|
|
<col id="0" translatable="yes">Exactly one rule must apply</col>
|
|
|
|
</row>
|
|
|
|
</data>
|
|
|
|
</object>
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2013-05-08 09:25:10 +00:00
|
|
|
msgid "All rules must apply"
|
|
|
|
msgid "At least one rule must apply"
|
|
|
|
msgid "Exactly one rule must apply"
|
|
|
|
'''
|
2016-04-19 09:08:21 +10:00
|
|
|
|
2013-05-08 09:25:10 +00:00
|
|
|
files = ['../gramps/plugins/importer/importgedcom.glade', '../gramps/gui/glade/rule.glade']
|
2016-04-19 09:08:21 +10:00
|
|
|
with open('gtklist.h', 'w') as temp:
|
|
|
|
for filename in files:
|
|
|
|
tree = ElementTree.parse(filename)
|
|
|
|
root = tree.getroot()
|
|
|
|
for line in root.iter():
|
|
|
|
att = line.attrib
|
|
|
|
if att == {'id': '0', 'translatable': 'yes'}:
|
|
|
|
col = 'char *s = N_("%s");\n' % line.text
|
|
|
|
temp.write(col)
|
|
|
|
root.clear()
|
|
|
|
|
2013-05-08 09:25:10 +00:00
|
|
|
print ('Wrote gtklist.h')
|
|
|
|
|
2012-03-15 17:11:31 +00:00
|
|
|
def retrieve():
|
|
|
|
"""
|
|
|
|
Extract messages from all files used by Gramps (python, glade, xml)
|
|
|
|
"""
|
|
|
|
extract_xml()
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2013-05-08 09:25:10 +00:00
|
|
|
extract_gtkbuilder()
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2013-05-04 15:54:14 +00:00
|
|
|
create_template()
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-10-03 15:55:36 +00:00
|
|
|
create_filesfile()
|
2012-10-03 12:16:06 +00:00
|
|
|
listing('python.txt', ['.py', '.py.in'])
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2013-05-05 19:41:13 +00:00
|
|
|
# additional keywords must always be kept in sync with those in genpot.sh
|
2014-04-17 16:43:46 +02:00
|
|
|
os.system('''%(xgettext)s -F -c -j --directory=./ -d gramps '''
|
2012-03-15 17:11:31 +00:00
|
|
|
'''-L Python -o gramps.pot --files-from=python.txt '''
|
2016-10-23 15:52:23 +02:00
|
|
|
'''--debug --keyword=_ --keyword=ngettext '''
|
2013-05-07 18:59:43 +00:00
|
|
|
'''--keyword=_T_ --keyword=trans_text '''
|
2012-03-16 18:35:35 +00:00
|
|
|
'''--keyword=sgettext --from-code=UTF-8''' % {'xgettext': xgettextCmd}
|
2012-03-15 17:11:31 +00:00
|
|
|
)
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-03-15 17:11:31 +00:00
|
|
|
extract_glade()
|
2016-04-19 09:15:05 +10:00
|
|
|
|
2012-03-15 17:11:31 +00:00
|
|
|
# C format header (.h extension)
|
|
|
|
for h in headers():
|
2012-10-03 12:16:06 +00:00
|
|
|
print ('xgettext for %s' % h)
|
2014-04-17 16:43:46 +02:00
|
|
|
os.system('''%(xgettext)s -F --add-comments -j -o gramps.pot '''
|
2016-04-19 09:15:05 +10:00
|
|
|
'''--keyword=N_ --from-code=UTF-8 %(head)s'''
|
2012-03-16 18:35:35 +00:00
|
|
|
% {'xgettext': xgettextCmd, 'head': h}
|
2012-03-15 17:11:31 +00:00
|
|
|
)
|
|
|
|
clean()
|
2012-10-03 12:16:06 +00:00
|
|
|
|
2012-03-15 17:11:31 +00:00
|
|
|
def clean():
|
|
|
|
"""
|
|
|
|
Remove created files (C format headers, temp listings)
|
|
|
|
"""
|
|
|
|
for h in headers():
|
|
|
|
if os.path.isfile(h):
|
2012-03-17 06:00:16 +00:00
|
|
|
os.unlink(h)
|
2012-10-03 12:16:06 +00:00
|
|
|
print ('Remove %(head)s' % {'head': h})
|
|
|
|
|
2012-03-15 17:11:31 +00:00
|
|
|
if os.path.isfile('python.txt'):
|
2012-03-16 18:35:35 +00:00
|
|
|
os.unlink('python.txt')
|
2012-10-03 12:16:06 +00:00
|
|
|
print ("Remove 'python.txt'")
|
|
|
|
|
2012-03-15 17:11:31 +00:00
|
|
|
if os.path.isfile('glade.txt'):
|
2012-03-16 18:35:35 +00:00
|
|
|
os.unlink('glade.txt')
|
2012-10-03 12:16:06 +00:00
|
|
|
print ("Remove 'glade.txt'")
|
|
|
|
|
2012-10-03 15:38:42 +00:00
|
|
|
if os.path.isfile('tmpfiles'):
|
|
|
|
os.unlink('tmpfiles')
|
|
|
|
print ("Remove 'tmpfiles'")
|
|
|
|
|
2012-12-02 17:22:40 +00:00
|
|
|
def merge(args):
|
2012-03-15 17:11:31 +00:00
|
|
|
"""
|
|
|
|
Merge messages with 'gramps.pot'
|
|
|
|
"""
|
2012-12-02 17:22:40 +00:00
|
|
|
for arg in args:
|
|
|
|
if arg == 'all':
|
2016-04-19 09:15:05 +10:00
|
|
|
continue
|
2012-12-02 17:22:40 +00:00
|
|
|
print ('Merge %(lang)s with current template' % {'lang': arg})
|
2015-03-25 19:53:20 +01:00
|
|
|
os.system('''%(msgmerge)s %(lang)s gramps.pot -o updated_%(lang)s''' \
|
2012-12-02 17:22:40 +00:00
|
|
|
% {'msgmerge': msgmergeCmd, 'lang': arg})
|
|
|
|
print ("Updated file: 'updated_%(lang)s'." % {'lang': arg})
|
2012-10-03 12:16:06 +00:00
|
|
|
|
2012-12-02 17:22:40 +00:00
|
|
|
def check(args):
|
2012-03-15 17:11:31 +00:00
|
|
|
"""
|
|
|
|
Check the translation file
|
|
|
|
"""
|
2012-12-02 17:22:40 +00:00
|
|
|
for arg in args:
|
|
|
|
if arg == 'all':
|
|
|
|
continue
|
|
|
|
print ("Checked file: '%(lang.po)s'. See '%(txt)s.txt'." \
|
|
|
|
% {'lang.po': arg, 'txt': arg[:-3]})
|
|
|
|
os.system('''%(python)s ./check_po -s %(lang.po)s > %(lang)s.txt''' \
|
|
|
|
% {'python': pythonCmd, 'lang.po': arg, 'lang': arg[:-3]})
|
2016-05-18 05:47:34 -04:00
|
|
|
os.system('''%(msgfmt)s -c -v %(lang.po)s'''
|
2012-12-02 17:22:40 +00:00
|
|
|
% {'msgfmt': msgfmtCmd, 'lang.po': arg})
|
2012-03-12 16:43:12 +00:00
|
|
|
|
2012-12-01 10:16:44 +00:00
|
|
|
def untranslated(arg):
|
2012-03-19 16:00:19 +00:00
|
|
|
"""
|
|
|
|
List untranslated messages
|
|
|
|
"""
|
2012-12-02 17:22:40 +00:00
|
|
|
os.system('''%(msgattrib)s --untranslated %(lang.po)s''' % {'msgattrib': msgattribCmd, 'lang.po': arg[0]})
|
2012-10-03 12:16:06 +00:00
|
|
|
|
2012-12-01 10:16:44 +00:00
|
|
|
def fuzzy(arg):
|
2012-03-19 16:00:19 +00:00
|
|
|
"""
|
|
|
|
List fuzzy messages
|
|
|
|
"""
|
2012-12-02 17:22:40 +00:00
|
|
|
os.system('''%(msgattrib)s --only-fuzzy --no-obsolete %(lang.po)s''' % {'msgattrib': msgattribCmd, 'lang.po': arg[0]})
|
2012-10-03 12:16:06 +00:00
|
|
|
|
2012-03-12 16:43:12 +00:00
|
|
|
if __name__ == "__main__":
|
2015-12-31 21:17:35 +00:00
|
|
|
main()
|