From 755f215590af5a9bc70917412b28cd710318ec63 Mon Sep 17 00:00:00 2001 From: Claude Paroz Date: Thu, 17 Jan 2013 13:33:04 +0100 Subject: [PATCH] Refactored makemessages command --- .../core/management/commands/makemessages.py | 634 +++++++++--------- 1 file changed, 317 insertions(+), 317 deletions(-) diff --git a/django/core/management/commands/makemessages.py b/django/core/management/commands/makemessages.py index 31971a9101..72128eb931 100644 --- a/django/core/management/commands/makemessages.py +++ b/django/core/management/commands/makemessages.py @@ -9,12 +9,127 @@ from subprocess import PIPE, Popen import django from django.core.management.base import CommandError, NoArgsCommand +from django.utils.functional import total_ordering from django.utils.text import get_text_list from django.utils.jslex import prepare_js_for_gettext plural_forms_re = re.compile(r'^(?P"Plural-Forms.+?\\n")\s*$', re.MULTILINE | re.DOTALL) STATUS_OK = 0 + +@total_ordering +class TranslatableFile(object): + def __init__(self, dirpath, file_name): + self.file = file_name + self.dirpath = dirpath + + def __repr__(self): + return "" % os.sep.join([self.dirpath, self.file]) + + def __eq__(self, other): + return self.dirpath == other.dirpath and self.file == other.file + + def __lt__(self, other): + if self.dirpath == other.dirpath: + return self.file < other.file + return self.dirpath < other.dirpath + + def process(self, command, potfile, domain, keep_pot=False): + """ + Extract translatable literals from self.file for :param domain: + creating or updating the :param potfile: POT file. + + Uses the xgettext GNU gettext utility. + """ + + from django.utils.translation import templatize + + if command.verbosity > 1: + command.stdout.write('processing file %s in %s\n' % (self.file, self.dirpath)) + _, file_ext = os.path.splitext(self.file) + if domain == 'djangojs' and file_ext in command.extensions: + is_templatized = True + orig_file = os.path.join(self.dirpath, self.file) + with open(orig_file) as fp: + src_data = fp.read() + src_data = prepare_js_for_gettext(src_data) + thefile = '%s.c' % self.file + work_file = os.path.join(self.dirpath, thefile) + with open(work_file, "w") as fp: + fp.write(src_data) + cmd = ( + 'xgettext -d %s -L C %s %s --keyword=gettext_noop ' + '--keyword=gettext_lazy --keyword=ngettext_lazy:1,2 ' + '--keyword=pgettext:1c,2 --keyword=npgettext:1c,2,3 ' + '--from-code UTF-8 --add-comments=Translators -o - "%s"' % + (domain, command.wrap, command.location, work_file)) + elif domain == 'django' and (file_ext == '.py' or file_ext in command.extensions): + thefile = self.file + orig_file = os.path.join(self.dirpath, self.file) + is_templatized = file_ext in command.extensions + if is_templatized: + with open(orig_file, "rU") as fp: + src_data = fp.read() + thefile = '%s.py' % self.file + content = templatize(src_data, orig_file[2:]) + with open(os.path.join(self.dirpath, thefile), "w") as fp: + fp.write(content) + work_file = os.path.join(self.dirpath, thefile) + cmd = ( + 'xgettext -d %s -L Python %s %s --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 --from-code UTF-8 ' + '--add-comments=Translators -o - "%s"' % + (domain, command.wrap, command.location, work_file)) + else: + return + msgs, errors, status = _popen(cmd) + if errors: + if status != STATUS_OK: + if is_templatized: + os.unlink(work_file) + if not keep_pot and os.path.exists(potfile): + os.unlink(potfile) + raise CommandError( + "errors happened while running xgettext on %s\n%s" % + (self.file, errors)) + elif command.verbosity > 0: + # Print warnings + command.stdout.write(errors) + if msgs: + if is_templatized: + old = '#: ' + work_file[2:] + new = '#: ' + orig_file[2:] + msgs = msgs.replace(old, new) + write_pot_file(potfile, msgs) + if is_templatized: + os.unlink(work_file) + + +def _popen(cmd): + """ + Friendly wrapper around Popen for Windows + """ + p = Popen(cmd, shell=True, stdout=PIPE, stderr=PIPE, close_fds=os.name != 'nt', universal_newlines=True) + output, errors = p.communicate() + return output, errors, p.returncode + +def write_pot_file(potfile, msgs): + """ + 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') + with open(potfile, 'a') as fp: + fp.write(msgs) + def handle_extensions(extensions=('html',), ignored=('py',)): """ Organizes multiple extensions that are separated with commas or passed by @@ -39,310 +154,12 @@ def handle_extensions(extensions=('html',), ignored=('py',)): ext_list[i] = '.%s' % ext_list[i] return set([x for x in ext_list if x.strip('.') not in ignored]) -def _popen(cmd): - """ - Friendly wrapper around Popen for Windows - """ - p = Popen(cmd, shell=True, stdout=PIPE, stderr=PIPE, close_fds=os.name != 'nt', universal_newlines=True) - output, errors = p.communicate() - return output, errors, p.returncode - -def find_files(root, ignore_patterns, verbosity, stdout=sys.stdout, symlinks=False): - """ - Helper function to get all files in the given root. - """ - dir_suffix = '%s*' % os.sep - norm_patterns = [p[:-len(dir_suffix)] if p.endswith(dir_suffix) else p for p in ignore_patterns] - all_files = [] - for dirpath, dirnames, filenames in os.walk(root, topdown=True, followlinks=symlinks): - for dirname in dirnames[:]: - if is_ignored(os.path.normpath(os.path.join(dirpath, dirname)), norm_patterns): - dirnames.remove(dirname) - if verbosity > 1: - stdout.write('ignoring directory %s\n' % dirname) - for filename in filenames: - if is_ignored(os.path.normpath(os.path.join(dirpath, filename)), ignore_patterns): - if verbosity > 1: - stdout.write('ignoring file %s in %s\n' % (filename, dirpath)) - else: - all_files.extend([(dirpath, filename)]) - all_files.sort() - return all_files - -def is_ignored(path, ignore_patterns): - """ - Helper function to check if the given path should be ignored or not. - """ - for pattern in ignore_patterns: - if fnmatch.fnmatchcase(path, pattern): - return True - return False - -def copy_plural_forms(msgs, locale, domain, verbosity, stdout=sys.stdout): - """ - 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. - """ - django_dir = os.path.normpath(os.path.join(os.path.dirname(django.__file__))) - if 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): - with open(django_po, 'rU') as fp: - m = plural_forms_re.search(fp.read()) - if m: - if verbosity > 1: - stdout.write("copying plural forms: %s\n" % m.group('value')) - lines = [] - seen = False - for line in msgs.split('\n'): - if not line and not seen: - line = '%s\n' % m.group('value') - seen = True - lines.append(line) - msgs = '\n'.join(lines) - break - return msgs - -def write_pot_file(potfile, msgs, file, work_file, is_templatized): - """ - Write the :param potfile: POT file with the :param msgs: contents, - previously making sure its format is valid. - """ - if is_templatized: - old = '#: ' + work_file[2:] - new = '#: ' + file[2:] - msgs = msgs.replace(old, new) - 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') - with open(potfile, 'a') as fp: - fp.write(msgs) - -def process_file(file, dirpath, potfile, domain, verbosity, - extensions, wrap, location, keep_pot, stdout=sys.stdout): - """ - Extract translatable literals from :param file: for :param domain: - creating or updating the :param potfile: POT file. - - Uses the xgettext GNU gettext utility. - """ - - from django.utils.translation import templatize - - if verbosity > 1: - stdout.write('processing file %s in %s\n' % (file, dirpath)) - _, file_ext = os.path.splitext(file) - if domain == 'djangojs' and file_ext in extensions: - is_templatized = True - orig_file = os.path.join(dirpath, file) - with open(orig_file) as fp: - src_data = fp.read() - src_data = prepare_js_for_gettext(src_data) - thefile = '%s.c' % file - work_file = os.path.join(dirpath, thefile) - with open(work_file, "w") as fp: - fp.write(src_data) - cmd = ( - 'xgettext -d %s -L C %s %s --keyword=gettext_noop ' - '--keyword=gettext_lazy --keyword=ngettext_lazy:1,2 ' - '--keyword=pgettext:1c,2 --keyword=npgettext:1c,2,3 ' - '--from-code UTF-8 --add-comments=Translators -o - "%s"' % - (domain, wrap, location, work_file)) - elif domain == 'django' and (file_ext == '.py' or file_ext in extensions): - thefile = file - orig_file = os.path.join(dirpath, file) - is_templatized = file_ext in extensions - if is_templatized: - with open(orig_file, "rU") as fp: - src_data = fp.read() - thefile = '%s.py' % file - content = templatize(src_data, orig_file[2:]) - with open(os.path.join(dirpath, thefile), "w") as fp: - fp.write(content) - work_file = os.path.join(dirpath, thefile) - cmd = ( - 'xgettext -d %s -L Python %s %s --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 --from-code UTF-8 ' - '--add-comments=Translators -o - "%s"' % - (domain, wrap, location, work_file)) - else: - return - msgs, errors, status = _popen(cmd) - if errors: - if status != STATUS_OK: - if is_templatized: - os.unlink(work_file) - if not keep_pot and os.path.exists(potfile): - os.unlink(potfile) - raise CommandError( - "errors happened while running xgettext on %s\n%s" % - (file, errors)) - elif verbosity > 0: - # Print warnings - stdout.write(errors) - if msgs: - write_pot_file(potfile, msgs, orig_file, work_file, is_templatized) - if is_templatized: - os.unlink(work_file) - -def write_po_file(pofile, potfile, domain, locale, verbosity, stdout, - copy_pforms, wrap, location, no_obsolete, keep_pot): - """ - Creates of updates the :param pofile: PO file for :param domain: and :param - locale:. Uses contents of the existing :param potfile:. - - Uses mguniq, msgmerge, and msgattrib GNU gettext utilities. - """ - msgs, errors, status = _popen('msguniq %s %s --to-code=utf-8 "%s"' % - (wrap, location, potfile)) - if errors: - if status != STATUS_OK: - if not keep_pot: - os.unlink(potfile) - raise CommandError( - "errors happened while running msguniq\n%s" % errors) - elif verbosity > 0: - stdout.write(errors) - - if os.path.exists(pofile): - with open(potfile, 'w') as fp: - fp.write(msgs) - msgs, errors, status = _popen('msgmerge %s %s -q "%s" "%s"' % - (wrap, location, pofile, potfile)) - if errors: - if status != STATUS_OK: - if not keep_pot: - os.unlink(potfile) - raise CommandError( - "errors happened while running msgmerge\n%s" % errors) - elif verbosity > 0: - stdout.write(errors) - elif copy_pforms: - msgs = copy_plural_forms(msgs, locale, domain, verbosity, stdout) - msgs = msgs.replace( - "#. #-#-#-#-# %s.pot (PACKAGE VERSION) #-#-#-#-#\n" % domain, "") - with open(pofile, 'w') as fp: - fp.write(msgs) - if no_obsolete: - msgs, errors, status = _popen( - 'msgattrib %s %s -o "%s" --no-obsolete "%s"' % - (wrap, location, pofile, pofile)) - if errors: - if status != STATUS_OK: - raise CommandError( - "errors happened while running msgattrib\n%s" % errors) - elif verbosity > 0: - stdout.write(errors) - -def make_messages(locale=None, domain='django', verbosity=1, all=False, - extensions=None, symlinks=False, ignore_patterns=None, no_wrap=False, - no_location=False, no_obsolete=False, stdout=sys.stdout, keep_pot=False): - """ - Uses the ``locale/`` directory from the Django Git tree or an - application/project to process all files with translatable literals for - the :param domain: domain and :param locale: locale. - """ - # Need to ensure that the i18n framework is enabled - from django.conf import settings - if settings.configured: - settings.USE_I18N = True - else: - settings.configure(USE_I18N = True) - - if ignore_patterns is None: - ignore_patterns = [] - - invoked_for_django = False - if os.path.isdir(os.path.join('conf', 'locale')): - localedir = os.path.abspath(os.path.join('conf', 'locale')) - invoked_for_django = True - # Ignoring all contrib apps - ignore_patterns += ['contrib/*'] - elif os.path.isdir('locale'): - localedir = os.path.abspath('locale') - else: - raise CommandError("This script should be run from the Django Git " - "tree or your project or app tree. If you did indeed run it " - "from the Git checkout or your project or application, " - "maybe you are just missing the conf/locale (in the django " - "tree) or locale (for project and application) directory? It " - "is not created automatically, you have to create it by hand " - "if you want to enable i18n for your project or application.") - - if domain not in ('django', 'djangojs'): - raise CommandError("currently makemessages only supports domains " - "'django' and 'djangojs'") - - if (locale is None and not all) or domain is None: - message = "Type '%s help %s' for usage information." % ( - os.path.basename(sys.argv[0]), sys.argv[1]) - raise CommandError(message) - - # We require gettext version 0.15 or newer. - output, errors, status = _popen('xgettext --version') - if status != STATUS_OK: - raise CommandError("Error running xgettext. Note that Django " - "internationalization requires GNU gettext 0.15 or newer.") - match = re.search(r'(?P\d+)\.(?P\d+)', output) - if match: - xversion = (int(match.group('major')), int(match.group('minor'))) - if xversion < (0, 15): - raise CommandError("Django internationalization requires GNU " - "gettext 0.15 or newer. You are using version %s, please " - "upgrade your gettext toolset." % match.group()) - - locales = [] - if locale is not None: - locales += locale.split(',') if not isinstance(locale, list) else locale - elif all: - locale_dirs = filter(os.path.isdir, glob.glob('%s/*' % localedir)) - locales = [os.path.basename(l) for l in locale_dirs] - - wrap = '--no-wrap' if no_wrap else '' - location = '--no-location' if no_location else '' - - potfile = os.path.join(localedir, '%s.pot' % str(domain)) - - if os.path.exists(potfile): - os.unlink(potfile) - - for dirpath, file in find_files(".", ignore_patterns, verbosity, - stdout, symlinks=symlinks): - process_file(file, dirpath, potfile, domain, verbosity, extensions, - wrap, location, keep_pot, stdout) - - for locale in locales: - if verbosity > 0: - stdout.write("processing language %s\n" % locale) - basedir = os.path.join(localedir, locale, 'LC_MESSAGES') - if not os.path.isdir(basedir): - os.makedirs(basedir) - - pofile = os.path.join(basedir, '%s.po' % str(domain)) - - if os.path.exists(potfile): - write_po_file(pofile, potfile, domain, locale, verbosity, stdout, - not invoked_for_django, wrap, location, no_obsolete, keep_pot) - - if not keep_pot: - os.unlink(potfile) - class Command(NoArgsCommand): option_list = NoArgsCommand.option_list + ( make_option('--locale', '-l', default=None, dest='locale', action='append', - help='Creates or updates the message files for the given locale(s) (e.g. pt_BR). Can be used multiple times, accepts a comma-separated list of locale names.'), + help='Creates or updates the message files for the given locale(s) (e.g. pt_BR). ' + 'Can be used multiple times, accepts a comma-separated list of locale names.'), make_option('--domain', '-d', default='django', dest='domain', help='The domain of the message files (default: "django").'), make_option('--all', '-a', action='store_true', dest='all', @@ -355,7 +172,7 @@ class Command(NoArgsCommand): make_option('--ignore', '-i', action='append', dest='ignore_patterns', default=[], metavar='PATTERN', help='Ignore files or directories matching this glob-style pattern. Use multiple times to ignore more.'), make_option('--no-default-ignore', action='store_false', dest='use_default_ignore_patterns', - default=True, help="Don't ignore the common glob-style patterns 'CVS', '.*' and '*~'."), + default=True, help="Don't ignore the common glob-style patterns 'CVS', '.*', '*~' and '*.pyc'."), make_option('--no-wrap', action='store_true', dest='no_wrap', default=False, help="Don't break long message lines into several lines"), make_option('--no-location', action='store_true', dest='no_location', @@ -376,29 +193,212 @@ class Command(NoArgsCommand): def handle_noargs(self, *args, **options): locale = options.get('locale') - domain = options.get('domain') - verbosity = int(options.get('verbosity')) + self.domain = options.get('domain') + self.verbosity = int(options.get('verbosity')) process_all = options.get('all') extensions = options.get('extensions') - symlinks = options.get('symlinks') + self.symlinks = options.get('symlinks') ignore_patterns = options.get('ignore_patterns') if options.get('use_default_ignore_patterns'): - ignore_patterns += ['CVS', '.*', '*~'] - ignore_patterns = list(set(ignore_patterns)) - no_wrap = options.get('no_wrap') - no_location = options.get('no_location') - no_obsolete = options.get('no_obsolete') - keep_pot = options.get('keep_pot') - if domain == 'djangojs': + ignore_patterns += ['CVS', '.*', '*~', '*.pyc'] + self.ignore_patterns = list(set(ignore_patterns)) + self.wrap = '--no-wrap' if options.get('no_wrap') else '' + self.location = '--no-location' if options.get('no_location') else '' + self.no_obsolete = options.get('no_obsolete') + self.keep_pot = options.get('keep_pot') + + if self.domain not in ('django', 'djangojs'): + raise CommandError("currently makemessages only supports domains " + "'django' and 'djangojs'") + if self.domain == 'djangojs': exts = extensions if extensions else ['js'] else: exts = extensions if extensions else ['html', 'txt'] - extensions = handle_extensions(exts) + self.extensions = handle_extensions(exts) - if verbosity > 1: + if (locale is None and not process_all) or self.domain is None: + raise CommandError("Type '%s help %s' for usage information." % ( + os.path.basename(sys.argv[0]), sys.argv[1])) + + if self.verbosity > 1: self.stdout.write('examining files with the extensions: %s\n' - % get_text_list(list(extensions), 'and')) + % get_text_list(list(self.extensions), 'and')) - make_messages(locale, domain, verbosity, process_all, extensions, - symlinks, ignore_patterns, no_wrap, no_location, - no_obsolete, self.stdout, keep_pot) + # Need to ensure that the i18n framework is enabled + from django.conf import settings + if settings.configured: + settings.USE_I18N = True + else: + settings.configure(USE_I18N = True) + + self.invoked_for_django = False + if os.path.isdir(os.path.join('conf', 'locale')): + localedir = os.path.abspath(os.path.join('conf', 'locale')) + self.invoked_for_django = True + # Ignoring all contrib apps + self.ignore_patterns += ['contrib/*'] + elif os.path.isdir('locale'): + localedir = os.path.abspath('locale') + else: + raise CommandError("This script should be run from the Django Git " + "tree or your project or app tree. If you did indeed run it " + "from the Git checkout or your project or application, " + "maybe you are just missing the conf/locale (in the django " + "tree) or locale (for project and application) directory? It " + "is not created automatically, you have to create it by hand " + "if you want to enable i18n for your project or application.") + + # We require gettext version 0.15 or newer. + output, errors, status = _popen('xgettext --version') + if status != STATUS_OK: + raise CommandError("Error running xgettext. Note that Django " + "internationalization requires GNU gettext 0.15 or newer.") + match = re.search(r'(?P\d+)\.(?P\d+)', output) + if match: + xversion = (int(match.group('major')), int(match.group('minor'))) + if xversion < (0, 15): + raise CommandError("Django internationalization requires GNU " + "gettext 0.15 or newer. You are using version %s, please " + "upgrade your gettext toolset." % match.group()) + + potfile = self.build_pot_file(localedir) + + # Build po files for each selected locale + locales = [] + if locale is not None: + locales += locale.split(',') if not isinstance(locale, list) else locale + elif process_all: + locale_dirs = filter(os.path.isdir, glob.glob('%s/*' % localedir)) + locales = [os.path.basename(l) for l in locale_dirs] + + try: + for locale in locales: + if self.verbosity > 0: + self.stdout.write("processing language %s\n" % locale) + self.write_po_file(potfile, locale) + finally: + if not self.keep_pot and os.path.exists(potfile): + os.unlink(potfile) + + def build_pot_file(self, localedir): + file_list = self.find_files(".") + + potfile = os.path.join(localedir, '%s.pot' % str(self.domain)) + if os.path.exists(potfile): + # Remove a previous undeleted potfile, if any + os.unlink(potfile) + + for f in file_list: + f.process(self, potfile, self.domain, self.keep_pot) + return potfile + + def find_files(self, root): + """ + Helper method to get all files in the given root. + """ + + def is_ignored(path, ignore_patterns): + """ + Check if the given path should be ignored or not. + """ + for pattern in ignore_patterns: + if fnmatch.fnmatchcase(path, pattern): + return True + return False + + dir_suffix = '%s*' % os.sep + norm_patterns = [p[:-len(dir_suffix)] if p.endswith(dir_suffix) else p for p in self.ignore_patterns] + all_files = [] + for dirpath, dirnames, filenames in os.walk(root, topdown=True, followlinks=self.symlinks): + for dirname in dirnames[:]: + if is_ignored(os.path.normpath(os.path.join(dirpath, dirname)), norm_patterns): + dirnames.remove(dirname) + if self.verbosity > 1: + self.stdout.write('ignoring directory %s\n' % dirname) + for filename in filenames: + if is_ignored(os.path.normpath(os.path.join(dirpath, filename)), self.ignore_patterns): + if self.verbosity > 1: + self.stdout.write('ignoring file %s in %s\n' % (filename, dirpath)) + else: + all_files.append(TranslatableFile(dirpath, filename)) + return sorted(all_files) + + 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:. + + Uses mguniq, msgmerge, and msgattrib GNU gettext utilities. + """ + msgs, errors, status = _popen('msguniq %s %s --to-code=utf-8 "%s"' % + (self.wrap, self.location, potfile)) + if errors: + if status != STATUS_OK: + raise CommandError( + "errors happened while running msguniq\n%s" % errors) + elif self.verbosity > 0: + self.stdout.write(errors) + + 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): + with open(potfile, 'w') as fp: + fp.write(msgs) + msgs, errors, status = _popen('msgmerge %s %s -q "%s" "%s"' % + (self.wrap, self.location, pofile, potfile)) + if errors: + if status != STATUS_OK: + raise CommandError( + "errors happened while running msgmerge\n%s" % errors) + elif self.verbosity > 0: + self.stdout.write(errors) + elif not self.invoked_for_django: + msgs = self.copy_plural_forms(msgs, locale) + msgs = msgs.replace( + "#. #-#-#-#-# %s.pot (PACKAGE VERSION) #-#-#-#-#\n" % self.domain, "") + with open(pofile, 'w') as fp: + fp.write(msgs) + + if self.no_obsolete: + msgs, errors, status = _popen( + 'msgattrib %s %s -o "%s" --no-obsolete "%s"' % + (wrap, location, pofile, pofile)) + 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. + """ + django_dir = os.path.normpath(os.path.join(os.path.dirname(django.__file__))) + 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): + with open(django_po, 'rU') as fp: + m = plural_forms_re.search(fp.read()) + if m: + if self.verbosity > 1: + self.stdout.write("copying plural forms: %s\n" % m.group('value')) + lines = [] + seen = False + for line in msgs.split('\n'): + if not line and not seen: + line = '%s\n' % m.group('value') + seen = True + lines.append(line) + msgs = '\n'.join(lines) + break + return msgs