2014-05-24 17:51:57 +08:00
|
|
|
from __future__ import unicode_literals
|
|
|
|
|
2010-02-16 20:15:04 +08:00
|
|
|
import fnmatch
|
|
|
|
import glob
|
2013-10-21 11:22:32 +08:00
|
|
|
import io
|
2008-07-06 14:39:44 +08:00
|
|
|
import os
|
2010-02-16 20:15:04 +08:00
|
|
|
import re
|
2008-07-06 14:39:44 +08:00
|
|
|
import sys
|
2015-06-15 22:10:40 +08:00
|
|
|
from functools import total_ordering
|
2008-07-06 14:39:44 +08:00
|
|
|
from itertools import dropwhile
|
2008-10-05 19:39:58 +08:00
|
|
|
|
2012-02-05 02:27:24 +08:00
|
|
|
import django
|
2014-07-15 17:10:50 +08:00
|
|
|
from django.conf import settings
|
2015-08-25 07:28:23 +08:00
|
|
|
from django.core.files.temp import NamedTemporaryFile
|
2015-01-28 20:35:27 +08:00
|
|
|
from django.core.management.base import BaseCommand, CommandError
|
|
|
|
from django.core.management.utils import (
|
|
|
|
find_command, handle_extensions, popen_wrapper,
|
|
|
|
)
|
2015-02-16 07:59:39 +08:00
|
|
|
from django.utils._os import upath
|
2015-04-02 00:37:47 +08:00
|
|
|
from django.utils.encoding import DEFAULT_LOCALE_ENCODING, force_str
|
2015-06-15 22:10:40 +08:00
|
|
|
from django.utils.functional import cached_property
|
2011-06-08 00:11:25 +08:00
|
|
|
from django.utils.jslex import prepare_js_for_gettext
|
2015-01-28 20:35:27 +08:00
|
|
|
from django.utils.text import get_text_list
|
2008-07-06 14:39:44 +08:00
|
|
|
|
2010-02-16 20:15:41 +08:00
|
|
|
plural_forms_re = re.compile(r'^(?P<value>"Plural-Forms.+?\\n")\s*$', re.MULTILINE | re.DOTALL)
|
2012-07-19 00:34:13 +08:00
|
|
|
STATUS_OK = 0
|
2015-05-15 16:23:11 +08:00
|
|
|
NO_LOCALE_DIR = object()
|
2008-07-06 14:39:44 +08:00
|
|
|
|
2008-08-09 00:41:55 +08:00
|
|
|
|
2013-02-13 03:50:47 +08:00
|
|
|
def check_programs(*programs):
|
|
|
|
for program in programs:
|
|
|
|
if find_command(program) is None:
|
2016-03-29 06:33:29 +08:00
|
|
|
raise CommandError(
|
|
|
|
"Can't find %s. Make sure you have GNU gettext tools 0.15 or "
|
|
|
|
"newer installed." % program
|
|
|
|
)
|
2013-02-13 03:50:47 +08:00
|
|
|
|
|
|
|
|
2013-01-17 20:33:04 +08:00
|
|
|
@total_ordering
|
|
|
|
class TranslatableFile(object):
|
2013-11-30 17:53:08 +08:00
|
|
|
def __init__(self, dirpath, file_name, locale_dir):
|
2013-01-17 20:33:04 +08:00
|
|
|
self.file = file_name
|
|
|
|
self.dirpath = dirpath
|
2013-11-30 17:53:08 +08:00
|
|
|
self.locale_dir = locale_dir
|
2013-01-17 20:33:04 +08:00
|
|
|
|
|
|
|
def __repr__(self):
|
|
|
|
return "<TranslatableFile: %s>" % os.sep.join([self.dirpath, self.file])
|
|
|
|
|
|
|
|
def __eq__(self, other):
|
2013-11-30 17:53:08 +08:00
|
|
|
return self.path == other.path
|
2013-01-17 20:33:04 +08:00
|
|
|
|
|
|
|
def __lt__(self, other):
|
2013-11-30 17:53:08 +08:00
|
|
|
return self.path < other.path
|
2013-01-17 20:33:04 +08:00
|
|
|
|
2013-11-30 17:53:08 +08:00
|
|
|
@property
|
|
|
|
def path(self):
|
|
|
|
return os.path.join(self.dirpath, self.file)
|
|
|
|
|
2013-01-17 20:33:04 +08:00
|
|
|
|
2015-08-25 07:28:23 +08:00
|
|
|
class BuildFile(object):
|
|
|
|
"""
|
|
|
|
Represents the state of a translatable file during the build process.
|
|
|
|
"""
|
|
|
|
def __init__(self, command, domain, translatable):
|
|
|
|
self.command = command
|
|
|
|
self.domain = domain
|
|
|
|
self.translatable = translatable
|
|
|
|
|
|
|
|
@cached_property
|
|
|
|
def is_templatized(self):
|
|
|
|
if self.domain == 'djangojs':
|
|
|
|
return self.command.gettext_version < (0, 18, 3)
|
|
|
|
elif self.domain == 'django':
|
|
|
|
file_ext = os.path.splitext(self.translatable.file)[1]
|
|
|
|
return file_ext != '.py'
|
|
|
|
return False
|
|
|
|
|
|
|
|
@cached_property
|
|
|
|
def path(self):
|
|
|
|
return self.translatable.path
|
|
|
|
|
|
|
|
@cached_property
|
|
|
|
def work_path(self):
|
|
|
|
"""
|
|
|
|
Path to a file which is being fed into GNU gettext pipeline. This may
|
|
|
|
be either a translatable or its preprocessed version.
|
|
|
|
"""
|
|
|
|
if not self.is_templatized:
|
|
|
|
return self.path
|
|
|
|
extension = {
|
|
|
|
'djangojs': 'c',
|
|
|
|
'django': 'py',
|
|
|
|
}.get(self.domain)
|
|
|
|
filename = '%s.%s' % (self.translatable.file, extension)
|
|
|
|
return os.path.join(self.translatable.dirpath, filename)
|
|
|
|
|
|
|
|
def preprocess(self):
|
|
|
|
"""
|
|
|
|
Preprocess (if necessary) a translatable file before passing it to
|
|
|
|
xgettext GNU gettext utility.
|
2013-01-17 20:33:04 +08:00
|
|
|
"""
|
|
|
|
from django.utils.translation import templatize
|
|
|
|
|
2015-08-25 07:28:23 +08:00
|
|
|
if not self.is_templatized:
|
2013-01-17 20:33:04 +08:00
|
|
|
return
|
2013-11-30 17:53:08 +08:00
|
|
|
|
2015-08-25 07:28:23 +08:00
|
|
|
with io.open(self.path, 'r', encoding=settings.FILE_CHARSET) as fp:
|
|
|
|
src_data = fp.read()
|
|
|
|
|
|
|
|
if self.domain == 'djangojs':
|
|
|
|
content = prepare_js_for_gettext(src_data)
|
|
|
|
elif self.domain == 'django':
|
|
|
|
content = templatize(src_data, self.path[2:])
|
|
|
|
|
|
|
|
with io.open(self.work_path, 'w', encoding='utf-8') as fp:
|
|
|
|
fp.write(content)
|
|
|
|
|
|
|
|
def postprocess_messages(self, msgs):
|
|
|
|
"""
|
|
|
|
Postprocess messages generated by xgettext GNU gettext utility.
|
|
|
|
|
|
|
|
Transform paths as if these messages were generated from original
|
|
|
|
translatable files rather than from preprocessed versions.
|
|
|
|
"""
|
|
|
|
if not self.is_templatized:
|
|
|
|
return msgs
|
|
|
|
|
|
|
|
# Remove '.py' suffix
|
|
|
|
if os.name == 'nt':
|
|
|
|
# Preserve '.\' prefix on Windows to respect gettext behavior
|
2016-04-29 03:18:55 +08:00
|
|
|
old_path = self.work_path
|
|
|
|
new_path = self.path
|
2015-08-25 07:28:23 +08:00
|
|
|
else:
|
2016-04-29 03:18:55 +08:00
|
|
|
old_path = self.work_path[2:]
|
|
|
|
new_path = self.path[2:]
|
|
|
|
|
|
|
|
return re.sub(
|
|
|
|
r'^(#: .*)(' + re.escape(old_path) + r')',
|
2016-05-02 01:21:43 +08:00
|
|
|
lambda match: match.group().replace(old_path, new_path),
|
2016-04-29 03:18:55 +08:00
|
|
|
msgs,
|
|
|
|
flags=re.MULTILINE
|
|
|
|
)
|
2015-08-25 07:28:23 +08:00
|
|
|
|
|
|
|
def cleanup(self):
|
|
|
|
"""
|
|
|
|
Remove a preprocessed copy of a translatable file (if any).
|
|
|
|
"""
|
|
|
|
if self.is_templatized:
|
|
|
|
# This check is needed for the case of a symlinked file and its
|
|
|
|
# source being processed inside a single group (locale dir);
|
|
|
|
# removing either of those two removes both.
|
|
|
|
if os.path.exists(self.work_path):
|
|
|
|
os.unlink(self.work_path)
|
2008-08-09 00:41:55 +08:00
|
|
|
|
2013-11-03 04:12:09 +08:00
|
|
|
|
2013-01-17 20:33:04 +08:00
|
|
|
def write_pot_file(potfile, msgs):
|
2011-12-23 19:24:35 +08:00
|
|
|
"""
|
|
|
|
Write the :param potfile: POT file with the :param msgs: contents,
|
|
|
|
previously making sure its format is valid.
|
|
|
|
"""
|
|
|
|
if os.path.exists(potfile):
|
|
|
|
# Strip the header
|
|
|
|
msgs = '\n'.join(dropwhile(len, msgs.split('\n')))
|
|
|
|
else:
|
|
|
|
msgs = msgs.replace('charset=CHARSET', 'charset=UTF-8')
|
2014-05-24 17:51:57 +08:00
|
|
|
with io.open(potfile, 'a', encoding='utf-8') as fp:
|
2012-05-05 20:01:38 +08:00
|
|
|
fp.write(msgs)
|
2011-12-23 19:24:35 +08:00
|
|
|
|
2008-07-06 14:39:44 +08:00
|
|
|
|
2014-06-18 07:07:54 +08:00
|
|
|
class Command(BaseCommand):
|
2016-03-29 06:33:29 +08:00
|
|
|
help = (
|
|
|
|
"Runs over the entire source tree of the current directory and "
|
|
|
|
"pulls out all strings marked for translation. It creates (or updates) a message "
|
|
|
|
"file in the conf/locale (in the django tree) or locale (for projects and "
|
|
|
|
"applications) directory.\n\nYou must run this command with one of either the "
|
|
|
|
"--locale, --exclude, or --all options."
|
|
|
|
)
|
2008-07-06 14:39:44 +08:00
|
|
|
|
2015-08-25 07:28:23 +08:00
|
|
|
translatable_file_class = TranslatableFile
|
|
|
|
build_file_class = BuildFile
|
|
|
|
|
2014-01-20 10:45:21 +08:00
|
|
|
requires_system_checks = False
|
2013-02-04 07:53:48 +08:00
|
|
|
leave_locale_alone = True
|
2008-07-06 14:39:44 +08:00
|
|
|
|
2014-03-06 17:13:22 +08:00
|
|
|
msgmerge_options = ['-q', '--previous']
|
|
|
|
msguniq_options = ['--to-code=utf-8']
|
|
|
|
msgattrib_options = ['--no-obsolete']
|
|
|
|
xgettext_options = ['--from-code=UTF-8', '--add-comments=Translators']
|
|
|
|
|
2014-06-07 04:39:33 +08:00
|
|
|
def add_arguments(self, parser):
|
2016-03-29 06:33:29 +08:00
|
|
|
parser.add_argument(
|
|
|
|
'--locale', '-l', default=[], dest='locale', action='append',
|
2014-06-07 04:39:33 +08:00
|
|
|
help='Creates or updates the message files for the given locale(s) (e.g. pt_BR). '
|
2016-03-29 06:33:29 +08:00
|
|
|
'Can be used multiple times.',
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'--exclude', '-x', default=[], dest='exclude', action='append',
|
|
|
|
help='Locales to exclude. Default is none. Can be used multiple times.',
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'--domain', '-d', default='django', dest='domain',
|
|
|
|
help='The domain of the message files (default: "django").',
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'--all', '-a', action='store_true', dest='all', default=False,
|
|
|
|
help='Updates the message files for all existing locales.',
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'--extension', '-e', dest='extensions', action='append',
|
2015-07-14 17:31:02 +08:00
|
|
|
help='The file extension(s) to examine (default: "html,txt,py", or "js" '
|
2014-06-07 04:39:33 +08:00
|
|
|
'if the domain is "djangojs"). Separate multiple extensions with '
|
|
|
|
'commas, or use -e multiple times.',
|
2016-03-29 06:33:29 +08:00
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'--symlinks', '-s', action='store_true', dest='symlinks', default=False,
|
|
|
|
help='Follows symlinks to directories when examining source code '
|
|
|
|
'and templates for translation strings.',
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'--ignore', '-i', action='append', dest='ignore_patterns',
|
2014-06-07 04:39:33 +08:00
|
|
|
default=[], metavar='PATTERN',
|
|
|
|
help='Ignore files or directories matching this glob-style pattern. '
|
2016-03-29 06:33:29 +08:00
|
|
|
'Use multiple times to ignore more.',
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'--no-default-ignore', action='store_false', dest='use_default_ignore_patterns',
|
|
|
|
default=True, help="Don't ignore the common glob-style patterns 'CVS', '.*', '*~' and '*.pyc'.",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'--no-wrap', action='store_true', dest='no_wrap',
|
|
|
|
default=False, help="Don't break long message lines into several lines.",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'--no-location', action='store_true', dest='no_location',
|
|
|
|
default=False, help="Don't write '#: filename:line' lines.",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'--no-obsolete', action='store_true', dest='no_obsolete',
|
|
|
|
default=False, help="Remove obsolete message strings.",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'--keep-pot', action='store_true', dest='keep_pot',
|
|
|
|
default=False, help="Keep .pot file after making messages. Useful when debugging.",
|
|
|
|
)
|
2014-06-07 04:39:33 +08:00
|
|
|
|
2014-06-18 07:07:54 +08:00
|
|
|
def handle(self, *args, **options):
|
2016-03-03 10:01:36 +08:00
|
|
|
locale = options['locale']
|
|
|
|
exclude = options['exclude']
|
|
|
|
self.domain = options['domain']
|
|
|
|
self.verbosity = options['verbosity']
|
|
|
|
process_all = options['all']
|
|
|
|
extensions = options['extensions']
|
|
|
|
self.symlinks = options['symlinks']
|
2014-07-15 17:10:50 +08:00
|
|
|
|
|
|
|
# Need to ensure that the i18n framework is enabled
|
|
|
|
if settings.configured:
|
|
|
|
settings.USE_I18N = True
|
|
|
|
else:
|
|
|
|
settings.configure(USE_I18N=True)
|
|
|
|
|
2016-03-03 10:01:36 +08:00
|
|
|
ignore_patterns = options['ignore_patterns']
|
|
|
|
if options['use_default_ignore_patterns']:
|
2013-01-17 20:33:04 +08:00
|
|
|
ignore_patterns += ['CVS', '.*', '*~', '*.pyc']
|
|
|
|
self.ignore_patterns = list(set(ignore_patterns))
|
2014-03-06 17:13:22 +08:00
|
|
|
|
|
|
|
# Avoid messing with mutable class variables
|
2016-03-03 10:01:36 +08:00
|
|
|
if options['no_wrap']:
|
2014-03-06 17:13:22 +08:00
|
|
|
self.msgmerge_options = self.msgmerge_options[:] + ['--no-wrap']
|
|
|
|
self.msguniq_options = self.msguniq_options[:] + ['--no-wrap']
|
|
|
|
self.msgattrib_options = self.msgattrib_options[:] + ['--no-wrap']
|
|
|
|
self.xgettext_options = self.xgettext_options[:] + ['--no-wrap']
|
2016-03-03 10:01:36 +08:00
|
|
|
if options['no_location']:
|
2014-03-06 17:13:22 +08:00
|
|
|
self.msgmerge_options = self.msgmerge_options[:] + ['--no-location']
|
|
|
|
self.msguniq_options = self.msguniq_options[:] + ['--no-location']
|
|
|
|
self.msgattrib_options = self.msgattrib_options[:] + ['--no-location']
|
|
|
|
self.xgettext_options = self.xgettext_options[:] + ['--no-location']
|
|
|
|
|
2016-03-03 10:01:36 +08:00
|
|
|
self.no_obsolete = options['no_obsolete']
|
|
|
|
self.keep_pot = options['keep_pot']
|
2013-01-17 20:33:04 +08:00
|
|
|
|
|
|
|
if self.domain not in ('django', 'djangojs'):
|
|
|
|
raise CommandError("currently makemessages only supports domains "
|
|
|
|
"'django' and 'djangojs'")
|
|
|
|
if self.domain == 'djangojs':
|
2012-02-05 06:12:58 +08:00
|
|
|
exts = extensions if extensions else ['js']
|
2008-08-09 00:41:55 +08:00
|
|
|
else:
|
2014-11-17 05:16:41 +08:00
|
|
|
exts = extensions if extensions else ['html', 'txt', 'py']
|
2014-11-17 16:24:56 +08:00
|
|
|
self.extensions = handle_extensions(exts)
|
2008-08-09 00:41:55 +08:00
|
|
|
|
2014-03-24 21:03:06 +08:00
|
|
|
if (locale is None and not exclude and not process_all) or self.domain is None:
|
2016-03-29 06:33:29 +08:00
|
|
|
raise CommandError(
|
|
|
|
"Type '%s help %s' for usage information."
|
|
|
|
% (os.path.basename(sys.argv[0]), sys.argv[1])
|
|
|
|
)
|
2013-01-17 20:33:04 +08:00
|
|
|
|
2013-11-15 21:39:23 +08:00
|
|
|
if self.verbosity > 1:
|
2016-03-29 06:33:29 +08:00
|
|
|
self.stdout.write(
|
|
|
|
'examining files with the extensions: %s\n'
|
|
|
|
% get_text_list(list(self.extensions), 'and')
|
|
|
|
)
|
2013-11-15 21:39:23 +08:00
|
|
|
|
2013-01-17 20:33:04 +08:00
|
|
|
self.invoked_for_django = False
|
2013-11-30 17:53:08 +08:00
|
|
|
self.locale_paths = []
|
|
|
|
self.default_locale_path = None
|
2013-01-17 20:33:04 +08:00
|
|
|
if os.path.isdir(os.path.join('conf', 'locale')):
|
2013-11-30 17:53:08 +08:00
|
|
|
self.locale_paths = [os.path.abspath(os.path.join('conf', 'locale'))]
|
|
|
|
self.default_locale_path = self.locale_paths[0]
|
2013-01-17 20:33:04 +08:00
|
|
|
self.invoked_for_django = True
|
|
|
|
else:
|
2015-01-22 00:55:57 +08:00
|
|
|
self.locale_paths.extend(settings.LOCALE_PATHS)
|
2013-11-30 17:53:08 +08:00
|
|
|
# Allow to run makemessages inside an app dir
|
|
|
|
if os.path.isdir('locale'):
|
|
|
|
self.locale_paths.append(os.path.abspath('locale'))
|
|
|
|
if self.locale_paths:
|
|
|
|
self.default_locale_path = self.locale_paths[0]
|
|
|
|
if not os.path.exists(self.default_locale_path):
|
|
|
|
os.makedirs(self.default_locale_path)
|
|
|
|
|
|
|
|
# Build locale list
|
2014-03-24 21:03:06 +08:00
|
|
|
locale_dirs = filter(os.path.isdir, glob.glob('%s/*' % self.default_locale_path))
|
|
|
|
all_locales = map(os.path.basename, locale_dirs)
|
|
|
|
|
|
|
|
# Account for excluded locales
|
|
|
|
if process_all:
|
|
|
|
locales = all_locales
|
|
|
|
else:
|
|
|
|
locales = locale or all_locales
|
|
|
|
locales = set(locales) - set(exclude)
|
|
|
|
|
2013-02-13 03:50:47 +08:00
|
|
|
if locales:
|
|
|
|
check_programs('msguniq', 'msgmerge', 'msgattrib')
|
|
|
|
|
2013-11-30 17:53:08 +08:00
|
|
|
check_programs('xgettext')
|
|
|
|
|
2013-01-26 00:58:37 +08:00
|
|
|
try:
|
2013-11-30 17:53:08 +08:00
|
|
|
potfiles = self.build_potfiles()
|
|
|
|
|
|
|
|
# Build po files for each selected locale
|
2013-01-17 20:33:04 +08:00
|
|
|
for locale in locales:
|
|
|
|
if self.verbosity > 0:
|
2013-01-19 23:06:52 +08:00
|
|
|
self.stdout.write("processing locale %s\n" % locale)
|
2013-11-30 17:53:08 +08:00
|
|
|
for potfile in potfiles:
|
|
|
|
self.write_po_file(potfile, locale)
|
2013-01-17 20:33:04 +08:00
|
|
|
finally:
|
2013-11-30 17:53:08 +08:00
|
|
|
if not self.keep_pot:
|
|
|
|
self.remove_potfiles()
|
2013-01-17 20:33:04 +08:00
|
|
|
|
2014-11-11 16:40:03 +08:00
|
|
|
@cached_property
|
|
|
|
def gettext_version(self):
|
2015-03-21 19:41:41 +08:00
|
|
|
# Gettext tools will output system-encoded bytestrings instead of UTF-8,
|
|
|
|
# when looking up the version. It's especially a problem on Windows.
|
2015-11-05 04:50:16 +08:00
|
|
|
out, err, status = popen_wrapper(
|
2015-03-21 19:41:41 +08:00
|
|
|
['xgettext', '--version'],
|
2015-04-02 00:37:47 +08:00
|
|
|
stdout_encoding=DEFAULT_LOCALE_ENCODING,
|
2015-03-21 19:41:41 +08:00
|
|
|
)
|
2015-01-07 01:05:20 +08:00
|
|
|
m = re.search(r'(\d+)\.(\d+)\.?(\d+)?', out)
|
2014-11-11 16:40:03 +08:00
|
|
|
if m:
|
2014-12-18 21:38:46 +08:00
|
|
|
return tuple(int(d) for d in m.groups() if d is not None)
|
2014-11-11 16:40:03 +08:00
|
|
|
else:
|
|
|
|
raise CommandError("Unable to get gettext version. Is it installed?")
|
|
|
|
|
2013-11-30 17:53:08 +08:00
|
|
|
def build_potfiles(self):
|
|
|
|
"""
|
|
|
|
Build pot files and apply msguniq to them.
|
|
|
|
"""
|
2013-01-17 20:33:04 +08:00
|
|
|
file_list = self.find_files(".")
|
2013-11-30 17:53:08 +08:00
|
|
|
self.remove_potfiles()
|
2015-08-25 07:28:23 +08:00
|
|
|
self.process_files(file_list)
|
2013-11-30 17:53:08 +08:00
|
|
|
potfiles = []
|
|
|
|
for path in self.locale_paths:
|
|
|
|
potfile = os.path.join(path, '%s.pot' % str(self.domain))
|
|
|
|
if not os.path.exists(potfile):
|
|
|
|
continue
|
2014-03-06 17:13:22 +08:00
|
|
|
args = ['msguniq'] + self.msguniq_options + [potfile]
|
2015-11-05 04:50:16 +08:00
|
|
|
msgs, errors, status = popen_wrapper(args)
|
2013-11-30 17:53:08 +08:00
|
|
|
if errors:
|
|
|
|
if status != STATUS_OK:
|
|
|
|
raise CommandError(
|
|
|
|
"errors happened while running msguniq\n%s" % errors)
|
|
|
|
elif self.verbosity > 0:
|
|
|
|
self.stdout.write(errors)
|
2014-05-24 17:51:57 +08:00
|
|
|
with io.open(potfile, 'w', encoding='utf-8') as fp:
|
2013-11-30 17:53:08 +08:00
|
|
|
fp.write(msgs)
|
|
|
|
potfiles.append(potfile)
|
|
|
|
return potfiles
|
|
|
|
|
|
|
|
def remove_potfiles(self):
|
|
|
|
for path in self.locale_paths:
|
|
|
|
pot_path = os.path.join(path, '%s.pot' % str(self.domain))
|
|
|
|
if os.path.exists(pot_path):
|
|
|
|
os.unlink(pot_path)
|
2013-01-17 20:33:04 +08:00
|
|
|
|
|
|
|
def find_files(self, root):
|
|
|
|
"""
|
2013-11-30 17:53:08 +08:00
|
|
|
Helper method to get all files in the given root. Also check that there
|
|
|
|
is a matching locale dir for each file.
|
2013-01-17 20:33:04 +08:00
|
|
|
"""
|
|
|
|
|
|
|
|
def is_ignored(path, ignore_patterns):
|
|
|
|
"""
|
|
|
|
Check if the given path should be ignored or not.
|
|
|
|
"""
|
2013-05-17 00:29:18 +08:00
|
|
|
filename = os.path.basename(path)
|
2016-01-24 00:47:07 +08:00
|
|
|
|
|
|
|
def ignore(pattern):
|
|
|
|
return fnmatch.fnmatchcase(filename, pattern) or fnmatch.fnmatchcase(path, pattern)
|
|
|
|
|
2013-05-17 00:29:18 +08:00
|
|
|
return any(ignore(pattern) for pattern in ignore_patterns)
|
2013-01-17 20:33:04 +08:00
|
|
|
|
2014-08-16 03:19:04 +08:00
|
|
|
ignore_patterns = [os.path.normcase(p) for p in self.ignore_patterns]
|
|
|
|
dir_suffixes = {'%s*' % path_sep for path_sep in {'/', os.sep}}
|
|
|
|
norm_patterns = []
|
|
|
|
for p in ignore_patterns:
|
|
|
|
for dir_suffix in dir_suffixes:
|
|
|
|
if p.endswith(dir_suffix):
|
|
|
|
norm_patterns.append(p[:-len(dir_suffix)])
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
norm_patterns.append(p)
|
|
|
|
|
2013-01-17 20:33:04 +08:00
|
|
|
all_files = []
|
2014-10-27 16:35:01 +08:00
|
|
|
ignored_roots = [os.path.normpath(p) for p in (settings.MEDIA_ROOT, settings.STATIC_ROOT) if p]
|
2013-01-17 20:33:04 +08:00
|
|
|
for dirpath, dirnames, filenames in os.walk(root, topdown=True, followlinks=self.symlinks):
|
|
|
|
for dirname in dirnames[:]:
|
2014-10-18 18:00:38 +08:00
|
|
|
if (is_ignored(os.path.normpath(os.path.join(dirpath, dirname)), norm_patterns) or
|
|
|
|
os.path.join(os.path.abspath(dirpath), dirname) in ignored_roots):
|
2013-01-17 20:33:04 +08:00
|
|
|
dirnames.remove(dirname)
|
|
|
|
if self.verbosity > 1:
|
|
|
|
self.stdout.write('ignoring directory %s\n' % dirname)
|
2013-11-30 17:53:08 +08:00
|
|
|
elif dirname == 'locale':
|
|
|
|
dirnames.remove(dirname)
|
|
|
|
self.locale_paths.insert(0, os.path.join(os.path.abspath(dirpath), dirname))
|
2013-01-17 20:33:04 +08:00
|
|
|
for filename in filenames:
|
2013-11-30 17:53:08 +08:00
|
|
|
file_path = os.path.normpath(os.path.join(dirpath, filename))
|
2014-11-17 05:16:41 +08:00
|
|
|
file_ext = os.path.splitext(filename)[1]
|
2014-11-17 20:42:14 +08:00
|
|
|
if file_ext not in self.extensions or is_ignored(file_path, self.ignore_patterns):
|
2013-01-17 20:33:04 +08:00
|
|
|
if self.verbosity > 1:
|
|
|
|
self.stdout.write('ignoring file %s in %s\n' % (filename, dirpath))
|
|
|
|
else:
|
2013-11-30 17:53:08 +08:00
|
|
|
locale_dir = None
|
|
|
|
for path in self.locale_paths:
|
|
|
|
if os.path.abspath(dirpath).startswith(os.path.dirname(path)):
|
|
|
|
locale_dir = path
|
|
|
|
break
|
|
|
|
if not locale_dir:
|
|
|
|
locale_dir = self.default_locale_path
|
|
|
|
if not locale_dir:
|
2015-05-15 16:23:11 +08:00
|
|
|
locale_dir = NO_LOCALE_DIR
|
2015-08-25 07:28:23 +08:00
|
|
|
all_files.append(self.translatable_file_class(dirpath, filename, locale_dir))
|
2013-01-17 20:33:04 +08:00
|
|
|
return sorted(all_files)
|
|
|
|
|
2015-08-25 07:28:23 +08:00
|
|
|
def process_files(self, file_list):
|
|
|
|
"""
|
|
|
|
Group translatable files by locale directory and run pot file build
|
|
|
|
process for each group.
|
|
|
|
"""
|
|
|
|
file_groups = {}
|
|
|
|
for translatable in file_list:
|
|
|
|
file_group = file_groups.setdefault(translatable.locale_dir, [])
|
|
|
|
file_group.append(translatable)
|
|
|
|
for locale_dir, files in file_groups.items():
|
|
|
|
self.process_locale_dir(locale_dir, files)
|
|
|
|
|
|
|
|
def process_locale_dir(self, locale_dir, files):
|
|
|
|
"""
|
|
|
|
Extract translatable literals from the specified files, creating or
|
|
|
|
updating the POT file for a given locale directory.
|
|
|
|
|
|
|
|
Uses the xgettext GNU gettext utility.
|
|
|
|
"""
|
|
|
|
build_files = []
|
|
|
|
for translatable in files:
|
|
|
|
if self.verbosity > 1:
|
|
|
|
self.stdout.write('processing file %s in %s\n' % (
|
|
|
|
translatable.file, translatable.dirpath
|
|
|
|
))
|
|
|
|
if self.domain not in ('djangojs', 'django'):
|
|
|
|
continue
|
|
|
|
build_file = self.build_file_class(self, self.domain, translatable)
|
|
|
|
try:
|
|
|
|
build_file.preprocess()
|
|
|
|
except UnicodeDecodeError as e:
|
|
|
|
self.stdout.write(
|
|
|
|
'UnicodeDecodeError: skipped file %s in %s (reason: %s)' % (
|
|
|
|
translatable.file, translatable.dirpath, e,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
continue
|
|
|
|
build_files.append(build_file)
|
|
|
|
|
|
|
|
if self.domain == 'djangojs':
|
|
|
|
is_templatized = build_file.is_templatized
|
|
|
|
args = [
|
|
|
|
'xgettext',
|
|
|
|
'-d', self.domain,
|
|
|
|
'--language=%s' % ('C' if is_templatized else 'JavaScript',),
|
|
|
|
'--keyword=gettext_noop',
|
|
|
|
'--keyword=gettext_lazy',
|
|
|
|
'--keyword=ngettext_lazy:1,2',
|
|
|
|
'--keyword=pgettext:1c,2',
|
|
|
|
'--keyword=npgettext:1c,2,3',
|
|
|
|
'--output=-',
|
|
|
|
]
|
|
|
|
elif self.domain == 'django':
|
|
|
|
args = [
|
|
|
|
'xgettext',
|
|
|
|
'-d', self.domain,
|
|
|
|
'--language=Python',
|
|
|
|
'--keyword=gettext_noop',
|
|
|
|
'--keyword=gettext_lazy',
|
|
|
|
'--keyword=ngettext_lazy:1,2',
|
|
|
|
'--keyword=ugettext_noop',
|
|
|
|
'--keyword=ugettext_lazy',
|
|
|
|
'--keyword=ungettext_lazy:1,2',
|
|
|
|
'--keyword=pgettext:1c,2',
|
|
|
|
'--keyword=npgettext:1c,2,3',
|
|
|
|
'--keyword=pgettext_lazy:1c,2',
|
|
|
|
'--keyword=npgettext_lazy:1c,2,3',
|
|
|
|
'--output=-',
|
|
|
|
]
|
|
|
|
else:
|
|
|
|
return
|
|
|
|
|
|
|
|
input_files = [bf.work_path for bf in build_files]
|
|
|
|
with NamedTemporaryFile(mode='w+') as input_files_list:
|
|
|
|
input_files_list.write('\n'.join(input_files))
|
|
|
|
input_files_list.flush()
|
|
|
|
args.extend(['--files-from', input_files_list.name])
|
|
|
|
args.extend(self.xgettext_options)
|
2015-11-05 04:50:16 +08:00
|
|
|
msgs, errors, status = popen_wrapper(args)
|
2015-08-25 07:28:23 +08:00
|
|
|
|
|
|
|
if errors:
|
|
|
|
if status != STATUS_OK:
|
|
|
|
for build_file in build_files:
|
|
|
|
build_file.cleanup()
|
|
|
|
raise CommandError(
|
|
|
|
'errors happened while running xgettext on %s\n%s' %
|
|
|
|
('\n'.join(input_files), errors)
|
|
|
|
)
|
|
|
|
elif self.verbosity > 0:
|
|
|
|
# Print warnings
|
|
|
|
self.stdout.write(errors)
|
|
|
|
|
|
|
|
if msgs:
|
|
|
|
if locale_dir is NO_LOCALE_DIR:
|
|
|
|
file_path = os.path.normpath(build_files[0].path)
|
|
|
|
raise CommandError(
|
|
|
|
'Unable to find a locale path to store translations for '
|
|
|
|
'file %s' % file_path
|
|
|
|
)
|
|
|
|
for build_file in build_files:
|
|
|
|
msgs = build_file.postprocess_messages(msgs)
|
|
|
|
potfile = os.path.join(locale_dir, '%s.pot' % str(self.domain))
|
|
|
|
write_pot_file(potfile, msgs)
|
|
|
|
|
|
|
|
for build_file in build_files:
|
|
|
|
build_file.cleanup()
|
|
|
|
|
2013-01-17 20:33:04 +08:00
|
|
|
def write_po_file(self, potfile, locale):
|
|
|
|
"""
|
|
|
|
Creates or updates the PO file for self.domain and :param locale:.
|
|
|
|
Uses contents of the existing :param potfile:.
|
|
|
|
|
2013-11-30 17:53:08 +08:00
|
|
|
Uses msgmerge, and msgattrib GNU gettext utilities.
|
2013-01-17 20:33:04 +08:00
|
|
|
"""
|
|
|
|
basedir = os.path.join(os.path.dirname(potfile), locale, 'LC_MESSAGES')
|
|
|
|
if not os.path.isdir(basedir):
|
|
|
|
os.makedirs(basedir)
|
|
|
|
pofile = os.path.join(basedir, '%s.po' % str(self.domain))
|
|
|
|
|
|
|
|
if os.path.exists(pofile):
|
2014-03-06 17:13:22 +08:00
|
|
|
args = ['msgmerge'] + self.msgmerge_options + [pofile, potfile]
|
2015-11-05 04:50:16 +08:00
|
|
|
msgs, errors, status = popen_wrapper(args)
|
2013-01-17 20:33:04 +08:00
|
|
|
if errors:
|
|
|
|
if status != STATUS_OK:
|
|
|
|
raise CommandError(
|
|
|
|
"errors happened while running msgmerge\n%s" % errors)
|
|
|
|
elif self.verbosity > 0:
|
|
|
|
self.stdout.write(errors)
|
2013-11-30 17:53:08 +08:00
|
|
|
else:
|
2014-05-24 17:51:57 +08:00
|
|
|
with io.open(potfile, 'r', encoding='utf-8') as fp:
|
2013-11-30 18:30:56 +08:00
|
|
|
msgs = fp.read()
|
2013-11-30 17:53:08 +08:00
|
|
|
if not self.invoked_for_django:
|
|
|
|
msgs = self.copy_plural_forms(msgs, locale)
|
2013-01-17 20:33:04 +08:00
|
|
|
msgs = msgs.replace(
|
|
|
|
"#. #-#-#-#-# %s.pot (PACKAGE VERSION) #-#-#-#-#\n" % self.domain, "")
|
2014-05-24 17:51:57 +08:00
|
|
|
with io.open(pofile, 'w', encoding='utf-8') as fp:
|
2013-01-17 20:33:04 +08:00
|
|
|
fp.write(msgs)
|
|
|
|
|
|
|
|
if self.no_obsolete:
|
2014-03-06 17:13:22 +08:00
|
|
|
args = ['msgattrib'] + self.msgattrib_options + ['-o', pofile, pofile]
|
2015-11-05 04:50:16 +08:00
|
|
|
msgs, errors, status = popen_wrapper(args)
|
2013-01-17 20:33:04 +08:00
|
|
|
if errors:
|
|
|
|
if status != STATUS_OK:
|
|
|
|
raise CommandError(
|
|
|
|
"errors happened while running msgattrib\n%s" % errors)
|
|
|
|
elif self.verbosity > 0:
|
|
|
|
self.stdout.write(errors)
|
|
|
|
|
|
|
|
def copy_plural_forms(self, msgs, locale):
|
|
|
|
"""
|
|
|
|
Copies plural forms header contents from a Django catalog of locale to
|
|
|
|
the msgs string, inserting it at the right place. msgs should be the
|
|
|
|
contents of a newly created .po file.
|
|
|
|
"""
|
2015-02-16 07:59:39 +08:00
|
|
|
django_dir = os.path.normpath(os.path.join(os.path.dirname(upath(django.__file__))))
|
2013-01-17 20:33:04 +08:00
|
|
|
if self.domain == 'djangojs':
|
|
|
|
domains = ('djangojs', 'django')
|
|
|
|
else:
|
|
|
|
domains = ('django',)
|
|
|
|
for domain in domains:
|
|
|
|
django_po = os.path.join(django_dir, 'conf', 'locale', locale, 'LC_MESSAGES', '%s.po' % domain)
|
|
|
|
if os.path.exists(django_po):
|
2014-05-24 17:51:57 +08:00
|
|
|
with io.open(django_po, 'r', encoding='utf-8') as fp:
|
2013-01-17 20:33:04 +08:00
|
|
|
m = plural_forms_re.search(fp.read())
|
|
|
|
if m:
|
2013-10-21 11:22:32 +08:00
|
|
|
plural_form_line = force_str(m.group('value'))
|
2013-01-17 20:33:04 +08:00
|
|
|
if self.verbosity > 1:
|
2013-10-21 11:22:32 +08:00
|
|
|
self.stdout.write("copying plural forms: %s\n" % plural_form_line)
|
2013-01-17 20:33:04 +08:00
|
|
|
lines = []
|
2013-06-23 05:39:14 +08:00
|
|
|
found = False
|
2013-01-17 20:33:04 +08:00
|
|
|
for line in msgs.split('\n'):
|
2013-06-23 05:39:14 +08:00
|
|
|
if not found and (not line or plural_forms_re.search(line)):
|
2013-10-21 11:22:32 +08:00
|
|
|
line = '%s\n' % plural_form_line
|
2013-06-23 05:39:14 +08:00
|
|
|
found = True
|
2013-01-17 20:33:04 +08:00
|
|
|
lines.append(line)
|
|
|
|
msgs = '\n'.join(lines)
|
|
|
|
break
|
|
|
|
return msgs
|