2009-04-11 12:00:43 -03:00
|
|
|
"""distutils.command.check
|
|
|
|
|
|
|
|
Implements the Distutils 'check' command.
|
|
|
|
"""
|
|
|
|
__revision__ = "$Id$"
|
|
|
|
|
|
|
|
from distutils.core import Command
|
|
|
|
from distutils.errors import DistutilsSetupError
|
|
|
|
|
|
|
|
try:
|
|
|
|
# docutils is installed
|
|
|
|
from docutils.utils import Reporter
|
|
|
|
from docutils.parsers.rst import Parser
|
|
|
|
from docutils import frontend
|
|
|
|
from docutils import nodes
|
|
|
|
from StringIO import StringIO
|
|
|
|
|
|
|
|
class SilentReporter(Reporter):
|
|
|
|
|
|
|
|
def __init__(self, source, report_level, halt_level, stream=None,
|
|
|
|
debug=0, encoding='ascii', error_handler='replace'):
|
|
|
|
self.messages = []
|
|
|
|
Reporter.__init__(self, source, report_level, halt_level, stream,
|
|
|
|
debug, encoding, error_handler)
|
|
|
|
|
|
|
|
def system_message(self, level, message, *children, **kwargs):
|
|
|
|
self.messages.append((level, message, children, kwargs))
|
|
|
|
|
|
|
|
HAS_DOCUTILS = True
|
Merged revisions 70912,70944,70968,71033,71041,71208,71263,71286,71395-71396,71405-71406,71485,71492,71494 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
r70912 | georg.brandl | 2009-03-31 17:35:46 -0500 (Tue, 31 Mar 2009) | 1 line
#5617: add a handy function to print a unicode string to gdbinit.
........
r70944 | georg.brandl | 2009-03-31 23:32:39 -0500 (Tue, 31 Mar 2009) | 1 line
#5631: add upload to list of possible commands, which is presented in --help-commands.
........
r70968 | michael.foord | 2009-04-01 13:25:38 -0500 (Wed, 01 Apr 2009) | 1 line
Adding Wing project file
........
r71033 | brett.cannon | 2009-04-01 22:34:53 -0500 (Wed, 01 Apr 2009) | 3 lines
Fix two issues introduced by issue #71031 by changing the signature of
PyImport_AppendInittab() to take a const char *.
........
r71041 | jesse.noller | 2009-04-02 00:17:26 -0500 (Thu, 02 Apr 2009) | 1 line
Add custom initializer argument to multiprocess.Manager*, courtesy of lekma
........
r71208 | michael.foord | 2009-04-04 20:15:01 -0500 (Sat, 04 Apr 2009) | 4 lines
Change the way unittest.TestSuite use their tests to always access them through iteration. Non behavior changing, this allows you to create custom subclasses that override __iter__.
Issue #5693
........
r71263 | michael.foord | 2009-04-05 14:19:28 -0500 (Sun, 05 Apr 2009) | 4 lines
Adding assertIs and assertIsNot methods to unittest.TestCase
Issue #2578
........
r71286 | tarek.ziade | 2009-04-05 17:04:38 -0500 (Sun, 05 Apr 2009) | 1 line
added a simplest test to distutils.spawn._nt_quote_args
........
r71395 | benjamin.peterson | 2009-04-08 08:27:29 -0500 (Wed, 08 Apr 2009) | 1 line
these must be installed to correctly run tests
........
r71396 | benjamin.peterson | 2009-04-08 08:29:41 -0500 (Wed, 08 Apr 2009) | 1 line
fix syntax
........
r71405 | andrew.kuchling | 2009-04-09 06:22:47 -0500 (Thu, 09 Apr 2009) | 1 line
Add items
........
r71406 | andrew.kuchling | 2009-04-09 06:23:36 -0500 (Thu, 09 Apr 2009) | 1 line
Typo fixes
........
r71485 | andrew.kuchling | 2009-04-11 11:12:23 -0500 (Sat, 11 Apr 2009) | 1 line
Add various items
........
r71492 | georg.brandl | 2009-04-11 13:19:27 -0500 (Sat, 11 Apr 2009) | 1 line
Take credit for a patch of mine.
........
r71494 | benjamin.peterson | 2009-04-11 14:31:00 -0500 (Sat, 11 Apr 2009) | 1 line
ignore py3_test_grammar when compiling the library
........
2009-04-11 17:45:40 -03:00
|
|
|
except Exception:
|
|
|
|
# Catch all exceptions because exceptions besides ImportError probably
|
|
|
|
# indicate that docutils is not ported to Py3k.
|
2009-04-11 12:00:43 -03:00
|
|
|
HAS_DOCUTILS = False
|
|
|
|
|
|
|
|
class check(Command):
|
|
|
|
"""This command checks the meta-data of the package.
|
|
|
|
"""
|
|
|
|
description = ("perform some checks on the package")
|
|
|
|
user_options = [('metadata', 'm', 'Verify meta-data'),
|
|
|
|
('restructuredtext', 'r',
|
|
|
|
('Checks if long string meta-data syntax '
|
|
|
|
'are reStructuredText-compliant')),
|
|
|
|
('strict', 's',
|
|
|
|
'Will exit with an error if a check fails')]
|
|
|
|
|
|
|
|
boolean_options = ['metadata', 'restructuredtext', 'strict']
|
|
|
|
|
|
|
|
def initialize_options(self):
|
|
|
|
"""Sets default values for options."""
|
|
|
|
self.restructuredtext = 0
|
|
|
|
self.metadata = 1
|
|
|
|
self.strict = 0
|
|
|
|
self._warnings = 0
|
|
|
|
|
|
|
|
def finalize_options(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def warn(self, msg):
|
|
|
|
"""Counts the number of warnings that occurs."""
|
|
|
|
self._warnings += 1
|
|
|
|
return Command.warn(self, msg)
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
"""Runs the command."""
|
|
|
|
# perform the various tests
|
|
|
|
if self.metadata:
|
|
|
|
self.check_metadata()
|
|
|
|
if self.restructuredtext:
|
2009-04-11 12:17:04 -03:00
|
|
|
if HAS_DOCUTILS:
|
2009-04-11 12:00:43 -03:00
|
|
|
self.check_restructuredtext()
|
|
|
|
elif self.strict:
|
|
|
|
raise DistutilsSetupError('The docutils package is needed.')
|
|
|
|
|
|
|
|
# let's raise an error in strict mode, if we have at least
|
|
|
|
# one warning
|
2009-04-17 11:34:49 -03:00
|
|
|
if self.strict and self._warnings > 0:
|
2009-04-11 12:00:43 -03:00
|
|
|
raise DistutilsSetupError('Please correct your package.')
|
|
|
|
|
|
|
|
def check_metadata(self):
|
|
|
|
"""Ensures that all required elements of meta-data are supplied.
|
|
|
|
|
|
|
|
name, version, URL, (author and author_email) or
|
|
|
|
(maintainer and maintainer_email)).
|
|
|
|
|
|
|
|
Warns if any are missing.
|
|
|
|
"""
|
|
|
|
metadata = self.distribution.metadata
|
|
|
|
|
|
|
|
missing = []
|
|
|
|
for attr in ('name', 'version', 'url'):
|
|
|
|
if not (hasattr(metadata, attr) and getattr(metadata, attr)):
|
|
|
|
missing.append(attr)
|
|
|
|
|
|
|
|
if missing:
|
|
|
|
self.warn("missing required meta-data: %s" % ' ,'.join(missing))
|
|
|
|
if metadata.author:
|
|
|
|
if not metadata.author_email:
|
|
|
|
self.warn("missing meta-data: if 'author' supplied, " +
|
|
|
|
"'author_email' must be supplied too")
|
|
|
|
elif metadata.maintainer:
|
|
|
|
if not metadata.maintainer_email:
|
|
|
|
self.warn("missing meta-data: if 'maintainer' supplied, " +
|
|
|
|
"'maintainer_email' must be supplied too")
|
|
|
|
else:
|
|
|
|
self.warn("missing meta-data: either (author and author_email) " +
|
|
|
|
"or (maintainer and maintainer_email) " +
|
|
|
|
"must be supplied")
|
|
|
|
|
|
|
|
def check_restructuredtext(self):
|
|
|
|
"""Checks if the long string fields are reST-compliant."""
|
|
|
|
data = self.distribution.get_long_description()
|
|
|
|
for warning in self._check_rst_data(data):
|
|
|
|
line = warning[-1].get('line')
|
|
|
|
if line is None:
|
|
|
|
warning = warning[1]
|
|
|
|
else:
|
|
|
|
warning = '%s (line %s)' % (warning[1], line)
|
|
|
|
self.warn(warning)
|
|
|
|
|
|
|
|
def _check_rst_data(self, data):
|
|
|
|
"""Returns warnings when the provided data doesn't compile."""
|
|
|
|
source_path = StringIO()
|
|
|
|
parser = Parser()
|
|
|
|
settings = frontend.OptionParser().get_default_values()
|
|
|
|
settings.tab_width = 4
|
|
|
|
settings.pep_references = None
|
|
|
|
settings.rfc_references = None
|
|
|
|
reporter = SilentReporter(source_path,
|
|
|
|
settings.report_level,
|
|
|
|
settings.halt_level,
|
|
|
|
stream=settings.warning_stream,
|
|
|
|
debug=settings.debug,
|
|
|
|
encoding=settings.error_encoding,
|
|
|
|
error_handler=settings.error_encoding_error_handler)
|
|
|
|
|
|
|
|
document = nodes.document(settings, reporter, source=source_path)
|
|
|
|
document.note_source(source_path, -1)
|
|
|
|
try:
|
|
|
|
parser.parse(data, document)
|
|
|
|
except AttributeError:
|
|
|
|
reporter.messages.append((-1, 'Could not finish the parsing.',
|
|
|
|
'', {}))
|
|
|
|
|
|
|
|
return reporter.messages
|