| 
									
										
										
										
											1999-08-11 21:40:38 +00:00
										 |  |  |  | #! /usr/bin/env python | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  | # -*- coding: iso-8859-1 -*- | 
					
						
							| 
									
										
										
										
											2001-07-27 16:47:18 +00:00
										 |  |  |  | # Originally written by Barry Warsaw <barry@zope.com> | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  | # | 
					
						
							| 
									
										
										
										
											2003-04-16 18:08:23 +00:00
										 |  |  |  | # Minimally patched to make it even more xgettext compatible | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  | # by Peter Funk <pf@artcom-gmbh.de> | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  | # | 
					
						
							|  |  |  |  | # 2002-11-22 J<>rgen Hermann <jh@web.de> | 
					
						
							|  |  |  |  | # Added checks that _() only contains string literals, and | 
					
						
							|  |  |  |  | # command line args are resolved to module lists, i.e. you | 
					
						
							|  |  |  |  | # can now pass a filename, a module or package name, or a | 
					
						
							|  |  |  |  | # directory (including globbing chars, important for Win32). | 
					
						
							|  |  |  |  | # Made docstring fit in 80 chars wide displays using pydoc. | 
					
						
							|  |  |  |  | # | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  | # for selftesting | 
					
						
							|  |  |  |  | try: | 
					
						
							|  |  |  |  |     import fintl | 
					
						
							|  |  |  |  |     _ = fintl.gettext | 
					
						
							|  |  |  |  | except ImportError: | 
					
						
							|  |  |  |  |     _ = lambda s: s | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | __doc__ = _("""pygettext -- Python equivalent of xgettext(1)
 | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | Many systems (Solaris, Linux, Gnu) provide extensive tools that ease the | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  | internationalization of C programs. Most of these tools are independent of | 
					
						
							|  |  |  |  | the programming language and can be used from within Python programs. | 
					
						
							| 
									
										
										
										
											2003-04-16 18:08:23 +00:00
										 |  |  |  | Martin von Loewis' work[1] helps considerably in this regard. | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-11-03 18:47:52 +00:00
										 |  |  |  | There's one problem though; xgettext is the program that scans source code | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  | looking for message strings, but it groks only C (or C++). Python | 
					
						
							|  |  |  |  | introduces a few wrinkles, such as dual quoting characters, triple quoted | 
					
						
							| 
									
										
										
										
											2003-04-16 18:08:23 +00:00
										 |  |  |  | strings, and raw strings. xgettext understands none of this. | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | Enter pygettext, which uses Python's standard tokenize module to scan | 
					
						
							|  |  |  |  | Python source code, generating .pot files identical to what GNU xgettext[2] | 
					
						
							|  |  |  |  | generates for C and C++ code. From there, the standard GNU tools can be | 
					
						
							| 
									
										
										
										
											2003-04-16 18:08:23 +00:00
										 |  |  |  | used. | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | A word about marking Python strings as candidates for translation. GNU | 
					
						
							|  |  |  |  | xgettext recognizes the following keywords: gettext, dgettext, dcgettext, | 
					
						
							|  |  |  |  | and gettext_noop. But those can be a lot of text to include all over your | 
					
						
							|  |  |  |  | code. C and C++ have a trick: they use the C preprocessor. Most | 
					
						
							|  |  |  |  | internationalized C source includes a #define for gettext() to _() so that | 
					
						
							|  |  |  |  | what has to be written in the source is much less. Thus these are both | 
					
						
							| 
									
										
										
										
											2003-04-16 18:08:23 +00:00
										 |  |  |  | translatable strings: | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     gettext("Translatable String") | 
					
						
							|  |  |  |  |     _("Translatable String") | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | Python of course has no preprocessor so this doesn't work so well.  Thus, | 
					
						
							|  |  |  |  | pygettext searches only for _() by default, but see the -k/--keyword flag | 
					
						
							|  |  |  |  | below for how to augment this. | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |  [1] http://www.python.org/workshops/1997-10/proceedings/loewis.html | 
					
						
							|  |  |  |  |  [2] http://www.gnu.org/software/gettext/gettext.html | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  | NOTE: pygettext attempts to be option and feature compatible with GNU | 
					
						
							|  |  |  |  | xgettext where ever possible. However some options are still missing or are | 
					
						
							|  |  |  |  | not fully implemented. Also, xgettext's use of command line switches with | 
					
						
							|  |  |  |  | option arguments is broken, and in these cases, pygettext just defines | 
					
						
							| 
									
										
										
										
											2003-04-16 18:08:23 +00:00
										 |  |  |  | additional switches. | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  | Usage: pygettext [options] inputfile ... | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | Options: | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     -a | 
					
						
							|  |  |  |  |     --extract-all | 
					
						
							| 
									
										
										
										
											2001-07-27 16:47:18 +00:00
										 |  |  |  |         Extract all strings. | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  |     -d name | 
					
						
							|  |  |  |  |     --default-domain=name | 
					
						
							| 
									
										
										
										
											2001-07-27 16:47:18 +00:00
										 |  |  |  |         Rename the default output file from messages.pot to name.pot. | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     -E | 
					
						
							|  |  |  |  |     --escape | 
					
						
							| 
									
										
										
										
											2000-10-27 04:56:28 +00:00
										 |  |  |  |         Replace non-ASCII characters with octal escape sequences. | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     -D | 
					
						
							|  |  |  |  |     --docstrings | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  |         Extract module, class, method, and function docstrings.  These do | 
					
						
							|  |  |  |  |         not need to be wrapped in _() markers, and in fact cannot be for | 
					
						
							|  |  |  |  |         Python to consider them docstrings. (See also the -X option). | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     -h | 
					
						
							|  |  |  |  |     --help | 
					
						
							| 
									
										
										
										
											2001-07-27 16:47:18 +00:00
										 |  |  |  |         Print this help message and exit. | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |     -k word | 
					
						
							|  |  |  |  |     --keyword=word | 
					
						
							|  |  |  |  |         Keywords to look for in addition to the default set, which are: | 
					
						
							|  |  |  |  |         %(DEFAULTKEYWORDS)s | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         You can have multiple -k flags on the command line. | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |     -K | 
					
						
							|  |  |  |  |     --no-default-keywords | 
					
						
							|  |  |  |  |         Disable the default set of keywords (see above).  Any keywords | 
					
						
							|  |  |  |  |         explicitly added with the -k/--keyword option are still recognized. | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     --no-location | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |         Do not write filename/lineno location comments. | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |     -n | 
					
						
							|  |  |  |  |     --add-location | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |         Write filename/lineno location comments indicating where each | 
					
						
							|  |  |  |  |         extracted string is found in the source.  These lines appear before | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |         each msgid.  The style of comments is controlled by the -S/--style | 
					
						
							|  |  |  |  |         option.  This is the default. | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-10-27 04:56:28 +00:00
										 |  |  |  |     -o filename | 
					
						
							|  |  |  |  |     --output=filename | 
					
						
							|  |  |  |  |         Rename the default output file from messages.pot to filename.  If | 
					
						
							|  |  |  |  |         filename is `-' then the output is sent to standard out. | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     -p dir | 
					
						
							|  |  |  |  |     --output-dir=dir | 
					
						
							|  |  |  |  |         Output files will be placed in directory dir. | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |     -S stylename | 
					
						
							|  |  |  |  |     --style stylename | 
					
						
							|  |  |  |  |         Specify which style to use for location comments.  Two styles are | 
					
						
							|  |  |  |  |         supported: | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         Solaris  # File: filename, line: line-number | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |         GNU      #: filename:line | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |         The style name is case insensitive.  GNU style is the default. | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-11-03 18:47:52 +00:00
										 |  |  |  |     -v | 
					
						
							|  |  |  |  |     --verbose | 
					
						
							|  |  |  |  |         Print the names of the files being processed. | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  |     -V | 
					
						
							|  |  |  |  |     --version | 
					
						
							|  |  |  |  |         Print the version of pygettext and exit. | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     -w columns | 
					
						
							|  |  |  |  |     --width=columns | 
					
						
							|  |  |  |  |         Set width of output to columns. | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     -x filename | 
					
						
							|  |  |  |  |     --exclude-file=filename | 
					
						
							|  |  |  |  |         Specify a file that contains a list of strings that are not be | 
					
						
							|  |  |  |  |         extracted from the input files.  Each string to be excluded must | 
					
						
							|  |  |  |  |         appear on a line by itself in the file. | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-07-27 16:47:18 +00:00
										 |  |  |  |     -X filename | 
					
						
							|  |  |  |  |     --no-docstrings=filename | 
					
						
							|  |  |  |  |         Specify a file that contains a list of files (one per line) that | 
					
						
							|  |  |  |  |         should not have their docstrings extracted.  This is only useful in | 
					
						
							|  |  |  |  |         conjunction with the -D option above. | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-07-27 16:47:18 +00:00
										 |  |  |  | If `inputfile' is -, standard input is read. | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  | """)
 | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | import os | 
					
						
							| 
									
										
										
										
											2003-04-16 18:08:23 +00:00
										 |  |  |  | import imp | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | import sys | 
					
						
							| 
									
										
										
										
											2003-04-16 18:08:23 +00:00
										 |  |  |  | import glob | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | import time | 
					
						
							|  |  |  |  | import getopt | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  | import token | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | import tokenize | 
					
						
							| 
									
										
										
										
											2001-05-21 19:51:26 +00:00
										 |  |  |  | import operator | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  | __version__ = '1.5' | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  | default_keywords = ['_'] | 
					
						
							|  |  |  |  | DEFAULTKEYWORDS = ', '.join(default_keywords) | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  | EMPTYSTRING = '' | 
					
						
							| 
									
										
										
										
											1999-11-03 18:47:52 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |  | 
					
						
							| 
									
										
										
										
											2001-07-27 16:47:18 +00:00
										 |  |  |  | # The normal pot-file header. msgmerge and Emacs's po-mode work better if it's | 
					
						
							|  |  |  |  | # there. | 
					
						
							| 
									
										
										
										
											1999-11-03 18:47:52 +00:00
										 |  |  |  | pot_header = _('''\
 | 
					
						
							|  |  |  |  | # SOME DESCRIPTIVE TITLE. | 
					
						
							|  |  |  |  | # Copyright (C) YEAR ORGANIZATION | 
					
						
							|  |  |  |  | # FIRST AUTHOR <EMAIL@ADDRESS>, YEAR. | 
					
						
							|  |  |  |  | # | 
					
						
							|  |  |  |  | msgid "" | 
					
						
							|  |  |  |  | msgstr "" | 
					
						
							|  |  |  |  | "Project-Id-Version: PACKAGE VERSION\\n" | 
					
						
							| 
									
										
										
										
											2001-03-01 22:56:17 +00:00
										 |  |  |  | "POT-Creation-Date: %(time)s\\n" | 
					
						
							|  |  |  |  | "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\\n" | 
					
						
							| 
									
										
										
										
											1999-11-03 18:47:52 +00:00
										 |  |  |  | "Last-Translator: FULL NAME <EMAIL@ADDRESS>\\n" | 
					
						
							|  |  |  |  | "Language-Team: LANGUAGE <LL@li.org>\\n" | 
					
						
							|  |  |  |  | "MIME-Version: 1.0\\n" | 
					
						
							|  |  |  |  | "Content-Type: text/plain; charset=CHARSET\\n" | 
					
						
							|  |  |  |  | "Content-Transfer-Encoding: ENCODING\\n" | 
					
						
							|  |  |  |  | "Generated-By: pygettext.py %(version)s\\n" | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | ''')
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |  | 
					
						
							|  |  |  |  | def usage(code, msg=''): | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  |     print >> sys.stderr, __doc__ % globals() | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |     if msg: | 
					
						
							| 
									
										
										
										
											2000-10-27 04:56:28 +00:00
										 |  |  |  |         print >> sys.stderr, msg | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |     sys.exit(code) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |  | 
					
						
							| 
									
										
										
										
											1999-11-03 18:47:52 +00:00
										 |  |  |  | escapes = [] | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  | def make_escapes(pass_iso8859): | 
					
						
							|  |  |  |  |     global escapes | 
					
						
							| 
									
										
										
										
											2000-02-27 14:30:48 +00:00
										 |  |  |  |     if pass_iso8859: | 
					
						
							|  |  |  |  |         # Allow iso-8859 characters to pass through so that e.g. 'msgid | 
					
						
							|  |  |  |  |         # "H<>he"' would result not result in 'msgid "H\366he"'.  Otherwise we | 
					
						
							|  |  |  |  |         # escape any character outside the 32..126 range. | 
					
						
							|  |  |  |  |         mod = 128 | 
					
						
							|  |  |  |  |     else: | 
					
						
							|  |  |  |  |         mod = 256 | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  |     for i in range(256): | 
					
						
							| 
									
										
										
										
											2000-02-27 14:30:48 +00:00
										 |  |  |  |         if 32 <= (i % mod) <= 126: | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  |             escapes.append(chr(i)) | 
					
						
							|  |  |  |  |         else: | 
					
						
							|  |  |  |  |             escapes.append("\\%03o" % i) | 
					
						
							|  |  |  |  |     escapes[ord('\\')] = '\\\\' | 
					
						
							|  |  |  |  |     escapes[ord('\t')] = '\\t' | 
					
						
							|  |  |  |  |     escapes[ord('\r')] = '\\r' | 
					
						
							|  |  |  |  |     escapes[ord('\n')] = '\\n' | 
					
						
							|  |  |  |  |     escapes[ord('\"')] = '\\"' | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-11-03 18:47:52 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | def escape(s): | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  |     global escapes | 
					
						
							| 
									
										
										
										
											1999-11-03 18:47:52 +00:00
										 |  |  |  |     s = list(s) | 
					
						
							|  |  |  |  |     for i in range(len(s)): | 
					
						
							|  |  |  |  |         s[i] = escapes[ord(s[i])] | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |     return EMPTYSTRING.join(s) | 
					
						
							| 
									
										
										
										
											1999-11-03 18:47:52 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | def safe_eval(s): | 
					
						
							|  |  |  |  |     # unwrap quotes, safely | 
					
						
							|  |  |  |  |     return eval(s, {'__builtins__':{}}, {}) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | def normalize(s): | 
					
						
							|  |  |  |  |     # This converts the various Python string types into a format that is | 
					
						
							|  |  |  |  |     # appropriate for .po files, namely much closer to C style. | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |     lines = s.split('\n') | 
					
						
							| 
									
										
										
										
											1999-11-03 18:47:52 +00:00
										 |  |  |  |     if len(lines) == 1: | 
					
						
							|  |  |  |  |         s = '"' + escape(s) + '"' | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |     else: | 
					
						
							| 
									
										
										
										
											1999-11-03 18:47:52 +00:00
										 |  |  |  |         if not lines[-1]: | 
					
						
							|  |  |  |  |             del lines[-1] | 
					
						
							|  |  |  |  |             lines[-1] = lines[-1] + '\n' | 
					
						
							|  |  |  |  |         for i in range(len(lines)): | 
					
						
							|  |  |  |  |             lines[i] = escape(lines[i]) | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |         lineterm = '\\n"\n"' | 
					
						
							|  |  |  |  |         s = '""\n"' + lineterm.join(lines) + '"' | 
					
						
							| 
									
										
										
										
											1999-11-03 18:47:52 +00:00
										 |  |  |  |     return s | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  |  | 
					
						
							|  |  |  |  | def containsAny(str, set): | 
					
						
							| 
									
										
										
										
											2003-04-16 18:08:23 +00:00
										 |  |  |  |     """Check whether 'str' contains ANY of the chars in 'set'""" | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  |     return 1 in [c in str for c in set] | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | def _visit_pyfiles(list, dirname, names): | 
					
						
							| 
									
										
										
										
											2003-04-16 18:08:23 +00:00
										 |  |  |  |     """Helper for getFilesForName().""" | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  |     # get extension for python source files | 
					
						
							|  |  |  |  |     if not globals().has_key('_py_ext'): | 
					
						
							|  |  |  |  |         global _py_ext | 
					
						
							| 
									
										
										
										
											2003-04-16 18:08:23 +00:00
										 |  |  |  |         _py_ext = [triple[0] for triple in imp.get_suffixes() | 
					
						
							|  |  |  |  |                    if triple[2] == imp.PY_SOURCE][0] | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     # don't recurse into CVS directories | 
					
						
							|  |  |  |  |     if 'CVS' in names: | 
					
						
							|  |  |  |  |         names.remove('CVS') | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     # add all *.py files to list | 
					
						
							|  |  |  |  |     list.extend( | 
					
						
							| 
									
										
										
										
											2003-04-16 18:08:23 +00:00
										 |  |  |  |         [os.path.join(dirname, file) for file in names | 
					
						
							|  |  |  |  |          if os.path.splitext(file)[1] == _py_ext] | 
					
						
							|  |  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | def _get_modpkg_path(dotted_name, pathlist=None): | 
					
						
							| 
									
										
										
										
											2003-04-16 18:08:23 +00:00
										 |  |  |  |     """Get the filesystem path for a module or a package.
 | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-16 18:08:23 +00:00
										 |  |  |  |     Return the file system path to a file for a module, and to a directory for | 
					
						
							|  |  |  |  |     a package. Return None if the name is not found, or is a builtin or | 
					
						
							|  |  |  |  |     extension module. | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  |     """
 | 
					
						
							|  |  |  |  |     # split off top-most name | 
					
						
							|  |  |  |  |     parts = dotted_name.split('.', 1) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if len(parts) > 1: | 
					
						
							|  |  |  |  |         # we have a dotted path, import top-level package | 
					
						
							|  |  |  |  |         try: | 
					
						
							|  |  |  |  |             file, pathname, description = imp.find_module(parts[0], pathlist) | 
					
						
							|  |  |  |  |             if file: file.close() | 
					
						
							|  |  |  |  |         except ImportError: | 
					
						
							|  |  |  |  |             return None | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         # check if it's indeed a package | 
					
						
							|  |  |  |  |         if description[2] == imp.PKG_DIRECTORY: | 
					
						
							|  |  |  |  |             # recursively handle the remaining name parts | 
					
						
							|  |  |  |  |             pathname = _get_modpkg_path(parts[1], [pathname]) | 
					
						
							|  |  |  |  |         else: | 
					
						
							|  |  |  |  |             pathname = None | 
					
						
							|  |  |  |  |     else: | 
					
						
							|  |  |  |  |         # plain name | 
					
						
							|  |  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2003-04-16 18:08:23 +00:00
										 |  |  |  |             file, pathname, description = imp.find_module( | 
					
						
							|  |  |  |  |                 dotted_name, pathlist) | 
					
						
							|  |  |  |  |             if file: | 
					
						
							|  |  |  |  |                 file.close() | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  |             if description[2] not in [imp.PY_SOURCE, imp.PKG_DIRECTORY]: | 
					
						
							|  |  |  |  |                 pathname = None | 
					
						
							|  |  |  |  |         except ImportError: | 
					
						
							|  |  |  |  |             pathname = None | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return pathname | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | def getFilesForName(name): | 
					
						
							| 
									
										
										
										
											2003-04-16 18:08:23 +00:00
										 |  |  |  |     """Get a list of module files for a filename, a module or package name,
 | 
					
						
							|  |  |  |  |     or a directory. | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  |     """
 | 
					
						
							|  |  |  |  |     if not os.path.exists(name): | 
					
						
							|  |  |  |  |         # check for glob chars | 
					
						
							|  |  |  |  |         if containsAny(name, "*?[]"): | 
					
						
							|  |  |  |  |             files = glob.glob(name) | 
					
						
							|  |  |  |  |             list = [] | 
					
						
							|  |  |  |  |             for file in files: | 
					
						
							|  |  |  |  |                 list.extend(getFilesForName(file)) | 
					
						
							|  |  |  |  |             return list | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         # try to find module or package | 
					
						
							|  |  |  |  |         name = _get_modpkg_path(name) | 
					
						
							|  |  |  |  |         if not name: | 
					
						
							|  |  |  |  |             return [] | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if os.path.isdir(name): | 
					
						
							|  |  |  |  |         # find all python files in directory | 
					
						
							|  |  |  |  |         list = [] | 
					
						
							|  |  |  |  |         os.path.walk(name, _visit_pyfiles, list) | 
					
						
							|  |  |  |  |         return list | 
					
						
							|  |  |  |  |     elif os.path.exists(name): | 
					
						
							|  |  |  |  |         # a single file | 
					
						
							|  |  |  |  |         return [name] | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return [] | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |  | 
					
						
							|  |  |  |  | class TokenEater: | 
					
						
							|  |  |  |  |     def __init__(self, options): | 
					
						
							|  |  |  |  |         self.__options = options | 
					
						
							|  |  |  |  |         self.__messages = {} | 
					
						
							|  |  |  |  |         self.__state = self.__waiting | 
					
						
							|  |  |  |  |         self.__data = [] | 
					
						
							|  |  |  |  |         self.__lineno = -1 | 
					
						
							| 
									
										
										
										
											2000-10-27 04:56:28 +00:00
										 |  |  |  |         self.__freshmodule = 1 | 
					
						
							| 
									
										
										
										
											2001-07-27 16:47:18 +00:00
										 |  |  |  |         self.__curfile = None | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     def __call__(self, ttype, tstring, stup, etup, line): | 
					
						
							|  |  |  |  |         # dispatch | 
					
						
							| 
									
										
										
										
											2000-10-27 04:56:28 +00:00
										 |  |  |  | ##        import token | 
					
						
							|  |  |  |  | ##        print >> sys.stderr, 'ttype:', token.tok_name[ttype], \ | 
					
						
							|  |  |  |  | ##              'tstring:', tstring | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |         self.__state(ttype, tstring, stup[0]) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def __waiting(self, ttype, tstring, lineno): | 
					
						
							| 
									
										
										
										
											2001-07-27 16:47:18 +00:00
										 |  |  |  |         opts = self.__options | 
					
						
							| 
									
										
										
										
											2000-10-27 04:56:28 +00:00
										 |  |  |  |         # Do docstring extractions, if enabled | 
					
						
							| 
									
										
										
										
											2001-07-27 16:47:18 +00:00
										 |  |  |  |         if opts.docstrings and not opts.nodocstrings.get(self.__curfile): | 
					
						
							| 
									
										
										
										
											2000-10-27 04:56:28 +00:00
										 |  |  |  |             # module docstring? | 
					
						
							|  |  |  |  |             if self.__freshmodule: | 
					
						
							|  |  |  |  |                 if ttype == tokenize.STRING: | 
					
						
							| 
									
										
										
										
											2001-05-21 19:51:26 +00:00
										 |  |  |  |                     self.__addentry(safe_eval(tstring), lineno, isdocstring=1) | 
					
						
							| 
									
										
										
										
											2000-10-27 04:56:28 +00:00
										 |  |  |  |                     self.__freshmodule = 0 | 
					
						
							|  |  |  |  |                 elif ttype not in (tokenize.COMMENT, tokenize.NL): | 
					
						
							|  |  |  |  |                     self.__freshmodule = 0 | 
					
						
							|  |  |  |  |                 return | 
					
						
							|  |  |  |  |             # class docstring? | 
					
						
							|  |  |  |  |             if ttype == tokenize.NAME and tstring in ('class', 'def'): | 
					
						
							|  |  |  |  |                 self.__state = self.__suiteseen | 
					
						
							|  |  |  |  |                 return | 
					
						
							| 
									
										
										
										
											2001-07-27 16:47:18 +00:00
										 |  |  |  |         if ttype == tokenize.NAME and tstring in opts.keywords: | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |             self.__state = self.__keywordseen | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-10-27 04:56:28 +00:00
										 |  |  |  |     def __suiteseen(self, ttype, tstring, lineno): | 
					
						
							|  |  |  |  |         # ignore anything until we see the colon | 
					
						
							|  |  |  |  |         if ttype == tokenize.OP and tstring == ':': | 
					
						
							|  |  |  |  |             self.__state = self.__suitedocstring | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def __suitedocstring(self, ttype, tstring, lineno): | 
					
						
							|  |  |  |  |         # ignore any intervening noise | 
					
						
							|  |  |  |  |         if ttype == tokenize.STRING: | 
					
						
							| 
									
										
										
										
											2001-05-21 19:51:26 +00:00
										 |  |  |  |             self.__addentry(safe_eval(tstring), lineno, isdocstring=1) | 
					
						
							| 
									
										
										
										
											2000-10-27 04:56:28 +00:00
										 |  |  |  |             self.__state = self.__waiting | 
					
						
							|  |  |  |  |         elif ttype not in (tokenize.NEWLINE, tokenize.INDENT, | 
					
						
							|  |  |  |  |                            tokenize.COMMENT): | 
					
						
							|  |  |  |  |             # there was no class docstring | 
					
						
							|  |  |  |  |             self.__state = self.__waiting | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |     def __keywordseen(self, ttype, tstring, lineno): | 
					
						
							|  |  |  |  |         if ttype == tokenize.OP and tstring == '(': | 
					
						
							|  |  |  |  |             self.__data = [] | 
					
						
							|  |  |  |  |             self.__lineno = lineno | 
					
						
							|  |  |  |  |             self.__state = self.__openseen | 
					
						
							|  |  |  |  |         else: | 
					
						
							|  |  |  |  |             self.__state = self.__waiting | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def __openseen(self, ttype, tstring, lineno): | 
					
						
							|  |  |  |  |         if ttype == tokenize.OP and tstring == ')': | 
					
						
							|  |  |  |  |             # We've seen the last of the translatable strings.  Record the | 
					
						
							| 
									
										
										
										
											2003-04-16 18:08:23 +00:00
										 |  |  |  |             # line number of the first line of the strings and update the list | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |             # of messages seen.  Reset state for the next batch.  If there | 
					
						
							|  |  |  |  |             # were no strings inside _(), then just ignore this entry. | 
					
						
							|  |  |  |  |             if self.__data: | 
					
						
							| 
									
										
										
										
											2000-10-27 04:56:28 +00:00
										 |  |  |  |                 self.__addentry(EMPTYSTRING.join(self.__data)) | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |             self.__state = self.__waiting | 
					
						
							|  |  |  |  |         elif ttype == tokenize.STRING: | 
					
						
							| 
									
										
										
										
											1999-11-03 18:47:52 +00:00
										 |  |  |  |             self.__data.append(safe_eval(tstring)) | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  |         elif ttype not in [tokenize.COMMENT, token.INDENT, token.DEDENT, | 
					
						
							|  |  |  |  |                            token.NEWLINE, tokenize.NL]: | 
					
						
							|  |  |  |  |             # warn if we see anything else than STRING or whitespace | 
					
						
							| 
									
										
										
										
											2003-04-16 18:08:23 +00:00
										 |  |  |  |             print >> sys.stderr, _( | 
					
						
							|  |  |  |  |                 '*** %(file)s:%(lineno)s: Seen unexpected token "%(token)s"' | 
					
						
							|  |  |  |  |                 ) % { | 
					
						
							|  |  |  |  |                 'token': tstring, | 
					
						
							|  |  |  |  |                 'file': self.__curfile, | 
					
						
							|  |  |  |  |                 'lineno': self.__lineno | 
					
						
							|  |  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  |             self.__state = self.__waiting | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-21 19:51:26 +00:00
										 |  |  |  |     def __addentry(self, msg, lineno=None, isdocstring=0): | 
					
						
							| 
									
										
										
										
											2000-10-27 04:56:28 +00:00
										 |  |  |  |         if lineno is None: | 
					
						
							|  |  |  |  |             lineno = self.__lineno | 
					
						
							|  |  |  |  |         if not msg in self.__options.toexclude: | 
					
						
							|  |  |  |  |             entry = (self.__curfile, lineno) | 
					
						
							| 
									
										
										
										
											2001-05-21 19:51:26 +00:00
										 |  |  |  |             self.__messages.setdefault(msg, {})[entry] = isdocstring | 
					
						
							| 
									
										
										
										
											2000-10-27 04:56:28 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |     def set_filename(self, filename): | 
					
						
							|  |  |  |  |         self.__curfile = filename | 
					
						
							| 
									
										
										
										
											2001-07-27 16:47:18 +00:00
										 |  |  |  |         self.__freshmodule = 1 | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     def write(self, fp): | 
					
						
							|  |  |  |  |         options = self.__options | 
					
						
							| 
									
										
										
										
											2004-08-16 12:10:12 +00:00
										 |  |  |  |         timestamp = time.strftime('%Y-%m-%d %H:%M+%Z') | 
					
						
							| 
									
										
										
										
											2000-10-27 04:56:28 +00:00
										 |  |  |  |         # The time stamp in the header doesn't have the same format as that | 
					
						
							|  |  |  |  |         # generated by xgettext... | 
					
						
							|  |  |  |  |         print >> fp, pot_header % {'time': timestamp, 'version': __version__} | 
					
						
							| 
									
										
										
										
											2001-05-23 16:59:45 +00:00
										 |  |  |  |         # Sort the entries.  First sort each particular entry's keys, then | 
					
						
							|  |  |  |  |         # sort all the entries by their first item. | 
					
						
							|  |  |  |  |         reverse = {} | 
					
						
							| 
									
										
										
										
											2000-10-26 03:49:15 +00:00
										 |  |  |  |         for k, v in self.__messages.items(): | 
					
						
							| 
									
										
										
										
											2001-05-23 16:59:45 +00:00
										 |  |  |  |             keys = v.keys() | 
					
						
							|  |  |  |  |             keys.sort() | 
					
						
							| 
									
										
										
										
											2001-05-24 23:06:13 +00:00
										 |  |  |  |             reverse.setdefault(tuple(keys), []).append((k, v)) | 
					
						
							| 
									
										
										
										
											2001-05-23 16:59:45 +00:00
										 |  |  |  |         rkeys = reverse.keys() | 
					
						
							|  |  |  |  |         rkeys.sort() | 
					
						
							|  |  |  |  |         for rkey in rkeys: | 
					
						
							| 
									
										
										
										
											2001-05-24 23:06:13 +00:00
										 |  |  |  |             rentries = reverse[rkey] | 
					
						
							|  |  |  |  |             rentries.sort() | 
					
						
							|  |  |  |  |             for k, v in rentries: | 
					
						
							|  |  |  |  |                 # If the entry was gleaned out of a docstring, then add a | 
					
						
							|  |  |  |  |                 # comment stating so.  This is to aid translators who may wish | 
					
						
							|  |  |  |  |                 # to skip translating some unimportant docstrings. | 
					
						
							| 
									
										
										
										
											2006-08-22 00:21:25 +00:00
										 |  |  |  |                 isdocstring = any(v.values()) | 
					
						
							| 
									
										
										
										
											2001-05-24 23:06:13 +00:00
										 |  |  |  |                 # k is the message string, v is a dictionary-set of (filename, | 
					
						
							|  |  |  |  |                 # lineno) tuples.  We want to sort the entries in v first by | 
					
						
							|  |  |  |  |                 # file name and then by line number. | 
					
						
							|  |  |  |  |                 v = v.keys() | 
					
						
							|  |  |  |  |                 v.sort() | 
					
						
							|  |  |  |  |                 if not options.writelocations: | 
					
						
							|  |  |  |  |                     pass | 
					
						
							|  |  |  |  |                 # location comments are different b/w Solaris and GNU: | 
					
						
							|  |  |  |  |                 elif options.locationstyle == options.SOLARIS: | 
					
						
							|  |  |  |  |                     for filename, lineno in v: | 
					
						
							|  |  |  |  |                         d = {'filename': filename, 'lineno': lineno} | 
					
						
							|  |  |  |  |                         print >>fp, _( | 
					
						
							|  |  |  |  |                             '# File: %(filename)s, line: %(lineno)d') % d | 
					
						
							|  |  |  |  |                 elif options.locationstyle == options.GNU: | 
					
						
							|  |  |  |  |                     # fit as many locations on one line, as long as the | 
					
						
							|  |  |  |  |                     # resulting line length doesn't exceeds 'options.width' | 
					
						
							|  |  |  |  |                     locline = '#:' | 
					
						
							|  |  |  |  |                     for filename, lineno in v: | 
					
						
							|  |  |  |  |                         d = {'filename': filename, 'lineno': lineno} | 
					
						
							|  |  |  |  |                         s = _(' %(filename)s:%(lineno)d') % d | 
					
						
							|  |  |  |  |                         if len(locline) + len(s) <= options.width: | 
					
						
							|  |  |  |  |                             locline = locline + s | 
					
						
							|  |  |  |  |                         else: | 
					
						
							|  |  |  |  |                             print >> fp, locline | 
					
						
							|  |  |  |  |                             locline = "#:" + s | 
					
						
							|  |  |  |  |                     if len(locline) > 2: | 
					
						
							| 
									
										
										
										
											2000-10-27 04:56:28 +00:00
										 |  |  |  |                         print >> fp, locline | 
					
						
							| 
									
										
										
										
											2001-06-20 19:41:40 +00:00
										 |  |  |  |                 if isdocstring: | 
					
						
							|  |  |  |  |                     print >> fp, '#, docstring' | 
					
						
							| 
									
										
										
										
											2001-05-24 23:06:13 +00:00
										 |  |  |  |                 print >> fp, 'msgid', normalize(k) | 
					
						
							|  |  |  |  |                 print >> fp, 'msgstr ""\n' | 
					
						
							| 
									
										
										
										
											2000-10-27 04:56:28 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |  | 
					
						
							|  |  |  |  | def main(): | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |     global default_keywords | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |     try: | 
					
						
							|  |  |  |  |         opts, args = getopt.getopt( | 
					
						
							|  |  |  |  |             sys.argv[1:], | 
					
						
							| 
									
										
										
										
											2001-07-27 16:47:18 +00:00
										 |  |  |  |             'ad:DEhk:Kno:p:S:Vvw:x:X:', | 
					
						
							| 
									
										
										
										
											2001-05-21 19:58:23 +00:00
										 |  |  |  |             ['extract-all', 'default-domain=', 'escape', 'help', | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |              'keyword=', 'no-default-keywords', | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  |              'add-location', 'no-location', 'output=', 'output-dir=', | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |              'style=', 'verbose', 'version', 'width=', 'exclude-file=', | 
					
						
							| 
									
										
										
										
											2001-07-27 16:47:18 +00:00
										 |  |  |  |              'docstrings', 'no-docstrings', | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  |              ]) | 
					
						
							| 
									
										
										
										
											2007-01-10 16:19:56 +00:00
										 |  |  |  |     except getopt.error as msg: | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |         usage(1, msg) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     # for holding option values | 
					
						
							|  |  |  |  |     class Options: | 
					
						
							|  |  |  |  |         # constants | 
					
						
							|  |  |  |  |         GNU = 1 | 
					
						
							|  |  |  |  |         SOLARIS = 2 | 
					
						
							|  |  |  |  |         # defaults | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  |         extractall = 0 # FIXME: currently this option has no effect at all. | 
					
						
							|  |  |  |  |         escape = 0 | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |         keywords = [] | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  |         outpath = '' | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |         outfile = 'messages.pot' | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |         writelocations = 1 | 
					
						
							|  |  |  |  |         locationstyle = GNU | 
					
						
							| 
									
										
										
										
											1999-11-03 18:47:52 +00:00
										 |  |  |  |         verbose = 0 | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  |         width = 78 | 
					
						
							|  |  |  |  |         excludefilename = '' | 
					
						
							| 
									
										
										
										
											2000-10-27 04:56:28 +00:00
										 |  |  |  |         docstrings = 0 | 
					
						
							| 
									
										
										
										
											2001-07-27 16:47:18 +00:00
										 |  |  |  |         nodocstrings = {} | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     options = Options() | 
					
						
							|  |  |  |  |     locations = {'gnu' : options.GNU, | 
					
						
							|  |  |  |  |                  'solaris' : options.SOLARIS, | 
					
						
							|  |  |  |  |                  } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     # parse options | 
					
						
							|  |  |  |  |     for opt, arg in opts: | 
					
						
							|  |  |  |  |         if opt in ('-h', '--help'): | 
					
						
							|  |  |  |  |             usage(0) | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  |         elif opt in ('-a', '--extract-all'): | 
					
						
							|  |  |  |  |             options.extractall = 1 | 
					
						
							|  |  |  |  |         elif opt in ('-d', '--default-domain'): | 
					
						
							|  |  |  |  |             options.outfile = arg + '.pot' | 
					
						
							|  |  |  |  |         elif opt in ('-E', '--escape'): | 
					
						
							|  |  |  |  |             options.escape = 1 | 
					
						
							| 
									
										
										
										
											2000-10-27 04:56:28 +00:00
										 |  |  |  |         elif opt in ('-D', '--docstrings'): | 
					
						
							|  |  |  |  |             options.docstrings = 1 | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |         elif opt in ('-k', '--keyword'): | 
					
						
							|  |  |  |  |             options.keywords.append(arg) | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |         elif opt in ('-K', '--no-default-keywords'): | 
					
						
							|  |  |  |  |             default_keywords = [] | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |         elif opt in ('-n', '--add-location'): | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |             options.writelocations = 1 | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |         elif opt in ('--no-location',): | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |             options.writelocations = 0 | 
					
						
							|  |  |  |  |         elif opt in ('-S', '--style'): | 
					
						
							|  |  |  |  |             options.locationstyle = locations.get(arg.lower()) | 
					
						
							|  |  |  |  |             if options.locationstyle is None: | 
					
						
							|  |  |  |  |                 usage(1, _('Invalid value for --style: %s') % arg) | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  |         elif opt in ('-o', '--output'): | 
					
						
							|  |  |  |  |             options.outfile = arg | 
					
						
							|  |  |  |  |         elif opt in ('-p', '--output-dir'): | 
					
						
							|  |  |  |  |             options.outpath = arg | 
					
						
							| 
									
										
										
										
											1999-11-03 18:47:52 +00:00
										 |  |  |  |         elif opt in ('-v', '--verbose'): | 
					
						
							|  |  |  |  |             options.verbose = 1 | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  |         elif opt in ('-V', '--version'): | 
					
						
							|  |  |  |  |             print _('pygettext.py (xgettext for Python) %s') % __version__ | 
					
						
							|  |  |  |  |             sys.exit(0) | 
					
						
							|  |  |  |  |         elif opt in ('-w', '--width'): | 
					
						
							|  |  |  |  |             try: | 
					
						
							|  |  |  |  |                 options.width = int(arg) | 
					
						
							|  |  |  |  |             except ValueError: | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |                 usage(1, _('--width argument must be an integer: %s') % arg) | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  |         elif opt in ('-x', '--exclude-file'): | 
					
						
							|  |  |  |  |             options.excludefilename = arg | 
					
						
							| 
									
										
										
										
											2001-07-27 16:47:18 +00:00
										 |  |  |  |         elif opt in ('-X', '--no-docstrings'): | 
					
						
							|  |  |  |  |             fp = open(arg) | 
					
						
							|  |  |  |  |             try: | 
					
						
							|  |  |  |  |                 while 1: | 
					
						
							|  |  |  |  |                     line = fp.readline() | 
					
						
							|  |  |  |  |                     if not line: | 
					
						
							|  |  |  |  |                         break | 
					
						
							|  |  |  |  |                     options.nodocstrings[line[:-1]] = 1 | 
					
						
							|  |  |  |  |             finally: | 
					
						
							|  |  |  |  |                 fp.close() | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     # calculate escapes | 
					
						
							| 
									
										
										
										
											2000-02-27 14:30:48 +00:00
										 |  |  |  |     make_escapes(options.escape) | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     # calculate all keywords | 
					
						
							|  |  |  |  |     options.keywords.extend(default_keywords) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  |     # initialize list of strings to exclude | 
					
						
							|  |  |  |  |     if options.excludefilename: | 
					
						
							|  |  |  |  |         try: | 
					
						
							|  |  |  |  |             fp = open(options.excludefilename) | 
					
						
							|  |  |  |  |             options.toexclude = fp.readlines() | 
					
						
							|  |  |  |  |             fp.close() | 
					
						
							|  |  |  |  |         except IOError: | 
					
						
							| 
									
										
										
										
											2001-05-21 19:35:20 +00:00
										 |  |  |  |             print >> sys.stderr, _( | 
					
						
							|  |  |  |  |                 "Can't read --exclude-file: %s") % options.excludefilename | 
					
						
							| 
									
										
										
										
											2000-02-26 20:56:47 +00:00
										 |  |  |  |             sys.exit(1) | 
					
						
							|  |  |  |  |     else: | 
					
						
							|  |  |  |  |         options.toexclude = [] | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  |     # resolve args to module lists | 
					
						
							|  |  |  |  |     expanded = [] | 
					
						
							|  |  |  |  |     for arg in args: | 
					
						
							|  |  |  |  |         if arg == '-': | 
					
						
							|  |  |  |  |             expanded.append(arg) | 
					
						
							|  |  |  |  |         else: | 
					
						
							|  |  |  |  |             expanded.extend(getFilesForName(arg)) | 
					
						
							|  |  |  |  |     args = expanded | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  |     # slurp through all the files | 
					
						
							|  |  |  |  |     eater = TokenEater(options) | 
					
						
							|  |  |  |  |     for filename in args: | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |         if filename == '-': | 
					
						
							|  |  |  |  |             if options.verbose: | 
					
						
							|  |  |  |  |                 print _('Reading standard input') | 
					
						
							|  |  |  |  |             fp = sys.stdin | 
					
						
							|  |  |  |  |             closep = 0 | 
					
						
							|  |  |  |  |         else: | 
					
						
							|  |  |  |  |             if options.verbose: | 
					
						
							|  |  |  |  |                 print _('Working on %s') % filename | 
					
						
							|  |  |  |  |             fp = open(filename) | 
					
						
							|  |  |  |  |             closep = 1 | 
					
						
							|  |  |  |  |         try: | 
					
						
							|  |  |  |  |             eater.set_filename(filename) | 
					
						
							| 
									
										
										
										
											2001-02-26 04:46:53 +00:00
										 |  |  |  |             try: | 
					
						
							|  |  |  |  |                 tokenize.tokenize(fp.readline, eater) | 
					
						
							| 
									
										
										
										
											2007-01-10 16:19:56 +00:00
										 |  |  |  |             except tokenize.TokenError as e: | 
					
						
							| 
									
										
										
										
											2001-05-21 19:35:20 +00:00
										 |  |  |  |                 print >> sys.stderr, '%s: %s, line %d, column %d' % ( | 
					
						
							|  |  |  |  |                     e[0], filename, e[1][0], e[1][1]) | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |         finally: | 
					
						
							|  |  |  |  |             if closep: | 
					
						
							|  |  |  |  |                 fp.close() | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-08 15:18:35 +00:00
										 |  |  |  |     # write the output | 
					
						
							|  |  |  |  |     if options.outfile == '-': | 
					
						
							|  |  |  |  |         fp = sys.stdout | 
					
						
							|  |  |  |  |         closep = 0 | 
					
						
							|  |  |  |  |     else: | 
					
						
							|  |  |  |  |         if options.outpath: | 
					
						
							|  |  |  |  |             options.outfile = os.path.join(options.outpath, options.outfile) | 
					
						
							|  |  |  |  |         fp = open(options.outfile, 'w') | 
					
						
							|  |  |  |  |         closep = 1 | 
					
						
							|  |  |  |  |     try: | 
					
						
							|  |  |  |  |         eater.write(fp) | 
					
						
							|  |  |  |  |     finally: | 
					
						
							|  |  |  |  |         if closep: | 
					
						
							|  |  |  |  |             fp.close() | 
					
						
							| 
									
										
										
										
											1999-08-13 20:59:48 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |  | 
					
						
							|  |  |  |  | if __name__ == '__main__': | 
					
						
							|  |  |  |  |     main() | 
					
						
							| 
									
										
										
										
											2000-05-02 19:28:30 +00:00
										 |  |  |  |     # some more test strings | 
					
						
							|  |  |  |  |     _(u'a unicode string') | 
					
						
							| 
									
										
										
										
											2003-04-16 18:08:23 +00:00
										 |  |  |  |     # this one creates a warning | 
					
						
							|  |  |  |  |     _('*** Seen unexpected token "%(token)s"') % {'token': 'test'} | 
					
						
							| 
									
										
										
										
											2002-11-22 08:36:54 +00:00
										 |  |  |  |     _('more' 'than' 'one' 'string') |