2010-12-21 18:05:42 +00:00
|
|
|
'''
|
|
|
|
Extension for enhancing sphinx documentation generation for cython module
|
|
|
|
'''
|
2010-12-21 14:57:47 +00:00
|
|
|
|
2011-06-28 10:06:26 +00:00
|
|
|
import re
|
|
|
|
import types
|
|
|
|
from sphinx.ext.autodoc import MethodDocumenter
|
|
|
|
|
|
|
|
class CythonMethodDocumenter(MethodDocumenter):
|
|
|
|
# XXX i don't understand the impact of having a priority more than the
|
|
|
|
# attribute or instance method but the things is, if it's a cython module,
|
|
|
|
# the attribute will be prefer over method.
|
|
|
|
priority = 12
|
|
|
|
|
2010-12-21 18:05:42 +00:00
|
|
|
def is_cython_extension(what, obj):
|
|
|
|
# try to check if the first line of the doc is a signature
|
|
|
|
doc = obj.__doc__
|
|
|
|
if not doc:
|
|
|
|
return False
|
|
|
|
doc = doc.split('\n')
|
|
|
|
if not len(doc):
|
|
|
|
return False
|
|
|
|
doc = doc[0]
|
|
|
|
|
2011-06-28 10:06:26 +00:00
|
|
|
# test for cython cpdef
|
|
|
|
if what in ('attribute', 'method') and hasattr(obj, '__objclass__'):
|
|
|
|
if not re.match('^([a-zA-Z_][a-zA-Z0-9_]*)\.([a-zA-Z_][a-zA-Z0-9_]*)\((.*)\)', doc):
|
|
|
|
return False
|
2010-12-21 18:05:42 +00:00
|
|
|
return True
|
2011-06-28 10:06:26 +00:00
|
|
|
# test for cython class
|
2010-12-21 18:05:42 +00:00
|
|
|
if what == 'class' and hasattr(obj, '__pyx_vtable__'):
|
2011-06-28 10:06:26 +00:00
|
|
|
if not re.match('^([a-zA-Z_][a-zA-Z0-9_]*)\((.*)\)', doc):
|
|
|
|
return False
|
|
|
|
return True
|
|
|
|
# test for python method in cython class
|
2011-07-06 16:13:32 +00:00
|
|
|
if what in ('method', 'function') and obj.__class__ == types.BuiltinFunctionType:
|
2011-06-28 10:06:26 +00:00
|
|
|
if not re.match('^([a-zA-Z_][a-zA-Z0-9_]*)\((.*)\)', doc):
|
|
|
|
return False
|
2010-12-21 18:05:42 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
def callback_docstring(app, what, name, obj, options, lines):
|
2010-12-21 14:57:47 +00:00
|
|
|
if what == 'module':
|
|
|
|
# remove empty lines
|
2011-06-28 10:06:26 +00:00
|
|
|
while len(lines):
|
|
|
|
line = lines[0].strip()
|
|
|
|
if not line.startswith('.. _') and line != '':
|
|
|
|
break
|
2010-12-21 14:57:47 +00:00
|
|
|
lines.pop(0)
|
|
|
|
|
|
|
|
# if we still have lines, remove the title
|
|
|
|
if len(lines):
|
|
|
|
lines.pop(0)
|
|
|
|
|
|
|
|
# if the title is followed by a separator, remove it.
|
|
|
|
if len(lines) and lines[0].startswith('=='):
|
|
|
|
lines.pop(0)
|
|
|
|
|
2011-09-28 20:58:15 +00:00
|
|
|
elif is_cython_extension(what, obj) and lines:
|
2010-12-21 18:05:42 +00:00
|
|
|
lines.pop(0)
|
2011-06-28 10:06:26 +00:00
|
|
|
line = lines.pop(0)
|
|
|
|
|
|
|
|
# trick to realign the first line to the second one.
|
|
|
|
# FIXME: fail if we finishing with ::
|
|
|
|
if line is not None and len(lines):
|
|
|
|
l = len(lines[0]) - len(lines[0].lstrip())
|
|
|
|
else:
|
|
|
|
l = 0
|
|
|
|
lines.insert(0, ' ' * l + line)
|
|
|
|
|
|
|
|
# calculate the minimum space available
|
|
|
|
min_space = 999
|
|
|
|
for line in lines:
|
|
|
|
if not line.strip():
|
|
|
|
continue
|
|
|
|
min_space = min(min_space, len(line) - len(line.lstrip()))
|
|
|
|
|
|
|
|
# remove that kind of space now.
|
|
|
|
if min_space > 0:
|
|
|
|
spaces = ' ' * min_space
|
|
|
|
for idx, line in enumerate(lines):
|
|
|
|
if not line.strip():
|
|
|
|
continue
|
|
|
|
if not line.startswith(spaces):
|
|
|
|
continue
|
|
|
|
lines[idx] = line[min_space:]
|
2010-12-21 18:05:42 +00:00
|
|
|
|
|
|
|
def callback_signature(app, what, name, obj, options, signature,
|
|
|
|
return_annotation):
|
2011-06-28 10:06:26 +00:00
|
|
|
# remove the first 'self' argument, because python autodoc don't
|
|
|
|
# add it for python method class. So do the same for cython class.
|
2010-12-21 18:05:42 +00:00
|
|
|
if is_cython_extension(what, obj):
|
|
|
|
try:
|
|
|
|
doc = obj.__doc__.split('\n').pop(0)
|
|
|
|
doc = '(%s' % doc.split('(')[1]
|
|
|
|
doc = doc.replace('(self, ', '(')
|
|
|
|
doc = doc.replace('(self)', '( )')
|
|
|
|
return (doc, None)
|
2011-06-28 10:06:26 +00:00
|
|
|
except AttributeError:
|
|
|
|
pass
|
2010-12-21 18:05:42 +00:00
|
|
|
except IndexError:
|
|
|
|
pass
|
|
|
|
|
2010-12-21 14:57:47 +00:00
|
|
|
def setup(app):
|
2011-06-28 10:06:26 +00:00
|
|
|
app.add_autodocumenter(CythonMethodDocumenter)
|
2010-12-21 18:05:42 +00:00
|
|
|
app.connect('autodoc-process-docstring', callback_docstring)
|
|
|
|
app.connect('autodoc-process-signature', callback_signature)
|
2010-12-21 14:57:47 +00:00
|
|
|
|