tmp
/
pip-install-ghxuqwgs
/numpy_78e94bf2b6094bf9a1f3d92042f9bf46
/doc
/sphinxext
/numpydoc
/docscrape.py
"""Extract reference documentation from the NumPy source tree. | |
""" | |
from __future__ import division, absolute_import, print_function | |
import inspect | |
import textwrap | |
import re | |
import pydoc | |
from warnings import warn | |
import collections | |
import sys | |
class Reader(object): | |
"""A line-based string reader. | |
""" | |
def __init__(self, data): | |
""" | |
Parameters | |
---------- | |
data : str | |
String with lines separated by '\n'. | |
""" | |
if isinstance(data,list): | |
self._str = data | |
else: | |
self._str = data.split('\n') # store string as list of lines | |
self.reset() | |
def __getitem__(self, n): | |
return self._str[n] | |
def reset(self): | |
self._l = 0 # current line nr | |
def read(self): | |
if not self.eof(): | |
out = self[self._l] | |
self._l += 1 | |
return out | |
else: | |
return '' | |
def seek_next_non_empty_line(self): | |
for l in self[self._l:]: | |
if l.strip(): | |
break | |
else: | |
self._l += 1 | |
def eof(self): | |
return self._l >= len(self._str) | |
def read_to_condition(self, condition_func): | |
start = self._l | |
for line in self[start:]: | |
if condition_func(line): | |
return self[start:self._l] | |
self._l += 1 | |
if self.eof(): | |
return self[start:self._l+1] | |
return [] | |
def read_to_next_empty_line(self): | |
self.seek_next_non_empty_line() | |
def is_empty(line): | |
return not line.strip() | |
return self.read_to_condition(is_empty) | |
def read_to_next_unindented_line(self): | |
def is_unindented(line): | |
return (line.strip() and (len(line.lstrip()) == len(line))) | |
return self.read_to_condition(is_unindented) | |
def peek(self,n=0): | |
if self._l + n < len(self._str): | |
return self[self._l + n] | |
else: | |
return '' | |
def is_empty(self): | |
return not ''.join(self._str).strip() | |
class NumpyDocString(object): | |
def __init__(self, docstring, config={}): | |
docstring = textwrap.dedent(docstring).split('\n') | |
self._doc = Reader(docstring) | |
self._parsed_data = { | |
'Signature': '', | |
'Summary': [''], | |
'Extended Summary': [], | |
'Parameters': [], | |
'Returns': [], | |
'Raises': [], | |
'Warns': [], | |
'Other Parameters': [], | |
'Attributes': [], | |
'Methods': [], | |
'See Also': [], | |
'Notes': [], | |
'Warnings': [], | |
'References': '', | |
'Examples': '', | |
'index': {} | |
} | |
self._parse() | |
def __getitem__(self,key): | |
return self._parsed_data[key] | |
def __setitem__(self,key,val): | |
if key not in self._parsed_data: | |
warn("Unknown section %s" % key) | |
else: | |
self._parsed_data[key] = val | |
def _is_at_section(self): | |
self._doc.seek_next_non_empty_line() | |
if self._doc.eof(): | |
return False | |
l1 = self._doc.peek().strip() # e.g. Parameters | |
if l1.startswith('.. index::'): | |
return True | |
l2 = self._doc.peek(1).strip() # ---------- or ========== | |
return l2.startswith('-'*len(l1)) or l2.startswith('='*len(l1)) | |
def _strip(self,doc): | |
i = 0 | |
j = 0 | |
for i,line in enumerate(doc): | |
if line.strip(): break | |
for j,line in enumerate(doc[::-1]): | |
if line.strip(): break | |
return doc[i:len(doc)-j] | |
def _read_to_next_section(self): | |
section = self._doc.read_to_next_empty_line() | |
while not self._is_at_section() and not self._doc.eof(): | |
if not self._doc.peek(-1).strip(): # previous line was empty | |
section += [''] | |
section += self._doc.read_to_next_empty_line() | |
return section | |
def _read_sections(self): | |
while not self._doc.eof(): | |
data = self._read_to_next_section() | |
name = data[0].strip() | |
if name.startswith('..'): # index section | |
yield name, data[1:] | |
elif len(data) < 2: | |
yield StopIteration | |
else: | |
yield name, self._strip(data[2:]) | |
def _parse_param_list(self,content): | |
r = Reader(content) | |
params = [] | |
while not r.eof(): | |
header = r.read().strip() | |
if ' : ' in header: | |
arg_name, arg_type = header.split(' : ')[:2] | |
else: | |
arg_name, arg_type = header, '' | |
desc = r.read_to_next_unindented_line() | |
desc = dedent_lines(desc) | |
params.append((arg_name,arg_type,desc)) | |
return params | |
_name_rgx = re.compile(r"^\s*(:(?P<role>\w+):`(?P<name>[a-zA-Z0-9_.-]+)`|" | |
r" (?P<name2>[a-zA-Z0-9_.-]+))\s*", re.X) | |
def _parse_see_also(self, content): | |
""" | |
func_name : Descriptive text | |
continued text | |
another_func_name : Descriptive text | |
func_name1, func_name2, :meth:`func_name`, func_name3 | |
""" | |
items = [] | |
def parse_item_name(text): | |
"""Match ':role:`name`' or 'name'""" | |
m = self._name_rgx.match(text) | |
if m: | |
g = m.groups() | |
if g[1] is None: | |
return g[3], None | |
else: | |
return g[2], g[1] | |
raise ValueError("%s is not a item name" % text) | |
def push_item(name, rest): | |
if not name: | |
return | |
name, role = parse_item_name(name) | |
items.append((name, list(rest), role)) | |
del rest[:] | |
current_func = None | |
rest = [] | |
for line in content: | |
if not line.strip(): continue | |
m = self._name_rgx.match(line) | |
if m and line[m.end():].strip().startswith(':'): | |
push_item(current_func, rest) | |
current_func, line = line[:m.end()], line[m.end():] | |
rest = [line.split(':', 1)[1].strip()] | |
if not rest[0]: | |
rest = [] | |
elif not line.startswith(' '): | |
push_item(current_func, rest) | |
current_func = None | |
if ',' in line: | |
for func in line.split(','): | |
if func.strip(): | |
push_item(func, []) | |
elif line.strip(): | |
current_func = line | |
elif current_func is not None: | |
rest.append(line.strip()) | |
push_item(current_func, rest) | |
return items | |
def _parse_index(self, section, content): | |
""" | |
.. index: default | |
:refguide: something, else, and more | |
""" | |
def strip_each_in(lst): | |
return [s.strip() for s in lst] | |
out = {} | |
section = section.split('::') | |
if len(section) > 1: | |
out['default'] = strip_each_in(section[1].split(','))[0] | |
for line in content: | |
line = line.split(':') | |
if len(line) > 2: | |
out[line[1]] = strip_each_in(line[2].split(',')) | |
return out | |
def _parse_summary(self): | |
"""Grab signature (if given) and summary""" | |
if self._is_at_section(): | |
return | |
# If several signatures present, take the last one | |
while True: | |
summary = self._doc.read_to_next_empty_line() | |
summary_str = " ".join([s.strip() for s in summary]).strip() | |
if re.compile('^([\w., ]+=)?\s*[\w\.]+\(.*\)$').match(summary_str): | |
self['Signature'] = summary_str | |
if not self._is_at_section(): | |
continue | |
break | |
if summary is not None: | |
self['Summary'] = summary | |
if not self._is_at_section(): | |
self['Extended Summary'] = self._read_to_next_section() | |
def _parse(self): | |
self._doc.reset() | |
self._parse_summary() | |
for (section,content) in self._read_sections(): | |
if not section.startswith('..'): | |
section = ' '.join([s.capitalize() for s in section.split(' ')]) | |
if section in ('Parameters', 'Returns', 'Raises', 'Warns', | |
'Other Parameters', 'Attributes', 'Methods'): | |
self[section] = self._parse_param_list(content) | |
elif section.startswith('.. index::'): | |
self['index'] = self._parse_index(section, content) | |
elif section == 'See Also': | |
self['See Also'] = self._parse_see_also(content) | |
else: | |
self[section] = content | |
# string conversion routines | |
def _str_header(self, name, symbol='-'): | |
return [name, len(name)*symbol] | |
def _str_indent(self, doc, indent=4): | |
out = [] | |
for line in doc: | |
out += [' '*indent + line] | |
return out | |
def _str_signature(self): | |
if self['Signature']: | |
return [self['Signature'].replace('*','\*')] + [''] | |
else: | |
return [''] | |
def _str_summary(self): | |
if self['Summary']: | |
return self['Summary'] + [''] | |
else: | |
return [] | |
def _str_extended_summary(self): | |
if self['Extended Summary']: | |
return self['Extended Summary'] + [''] | |
else: | |
return [] | |
def _str_param_list(self, name): | |
out = [] | |
if self[name]: | |
out += self._str_header(name) | |
for param,param_type,desc in self[name]: | |
if param_type: | |
out += ['%s : %s' % (param, param_type)] | |
else: | |
out += [param] | |
out += self._str_indent(desc) | |
out += [''] | |
return out | |
def _str_section(self, name): | |
out = [] | |
if self[name]: | |
out += self._str_header(name) | |
out += self[name] | |
out += [''] | |
return out | |
def _str_see_also(self, func_role): | |
if not self['See Also']: return [] | |
out = [] | |
out += self._str_header("See Also") | |
last_had_desc = True | |
for func, desc, role in self['See Also']: | |
if role: | |
link = ':%s:`%s`' % (role, func) | |
elif func_role: | |
link = ':%s:`%s`' % (func_role, func) | |
else: | |
link = "`%s`_" % func | |
if desc or last_had_desc: | |
out += [''] | |
out += [link] | |
else: | |
out[-1] += ", %s" % link | |
if desc: | |
out += self._str_indent([' '.join(desc)]) | |
last_had_desc = True | |
else: | |
last_had_desc = False | |
out += [''] | |
return out | |
def _str_index(self): | |
idx = self['index'] | |
out = [] | |
out += ['.. index:: %s' % idx.get('default','')] | |
for section, references in idx.items(): | |
if section == 'default': | |
continue | |
out += [' :%s: %s' % (section, ', '.join(references))] | |
return out | |
def __str__(self, func_role=''): | |
out = [] | |
out += self._str_signature() | |
out += self._str_summary() | |
out += self._str_extended_summary() | |
for param_list in ('Parameters', 'Returns', 'Other Parameters', | |
'Raises', 'Warns'): | |
out += self._str_param_list(param_list) | |
out += self._str_section('Warnings') | |
out += self._str_see_also(func_role) | |
for s in ('Notes','References','Examples'): | |
out += self._str_section(s) | |
for param_list in ('Attributes', 'Methods'): | |
out += self._str_param_list(param_list) | |
out += self._str_index() | |
return '\n'.join(out) | |
def indent(str,indent=4): | |
indent_str = ' '*indent | |
if str is None: | |
return indent_str | |
lines = str.split('\n') | |
return '\n'.join(indent_str + l for l in lines) | |
def dedent_lines(lines): | |
"""Deindent a list of lines maximally""" | |
return textwrap.dedent("\n".join(lines)).split("\n") | |
def header(text, style='-'): | |
return text + '\n' + style*len(text) + '\n' | |
class FunctionDoc(NumpyDocString): | |
def __init__(self, func, role='func', doc=None, config={}): | |
self._f = func | |
self._role = role # e.g. "func" or "meth" | |
if doc is None: | |
if func is None: | |
raise ValueError("No function or docstring given") | |
doc = inspect.getdoc(func) or '' | |
NumpyDocString.__init__(self, doc) | |
if not self['Signature'] and func is not None: | |
func, func_name = self.get_func() | |
try: | |
# try to read signature | |
if sys.version_info[0] >= 3: | |
argspec = inspect.getfullargspec(func) | |
else: | |
argspec = inspect.getargspec(func) | |
argspec = inspect.formatargspec(*argspec) | |
argspec = argspec.replace('*','\*') | |
signature = '%s%s' % (func_name, argspec) | |
except TypeError as e: | |
signature = '%s()' % func_name | |
self['Signature'] = signature | |
def get_func(self): | |
func_name = getattr(self._f, '__name__', self.__class__.__name__) | |
if inspect.isclass(self._f): | |
func = getattr(self._f, '__call__', self._f.__init__) | |
else: | |
func = self._f | |
return func, func_name | |
def __str__(self): | |
out = '' | |
func, func_name = self.get_func() | |
signature = self['Signature'].replace('*', '\*') | |
roles = {'func': 'function', | |
'meth': 'method'} | |
if self._role: | |
if self._role not in roles: | |
print("Warning: invalid role %s" % self._role) | |
out += '.. %s:: %s\n \n\n' % (roles.get(self._role,''), | |
func_name) | |
out += super(FunctionDoc, self).__str__(func_role=self._role) | |
return out | |
class ClassDoc(NumpyDocString): | |
extra_public_methods = ['__call__'] | |
def __init__(self, cls, doc=None, modulename='', func_doc=FunctionDoc, | |
config={}): | |
if not inspect.isclass(cls) and cls is not None: | |
raise ValueError("Expected a class or None, but got %r" % cls) | |
self._cls = cls | |
if modulename and not modulename.endswith('.'): | |
modulename += '.' | |
self._mod = modulename | |
if doc is None: | |
if cls is None: | |
raise ValueError("No class or documentation string given") | |
doc = pydoc.getdoc(cls) | |
NumpyDocString.__init__(self, doc) | |
if config.get('show_class_members', True): | |
def splitlines_x(s): | |
if not s: | |
return [] | |
else: | |
return s.splitlines() | |
for field, items in [('Methods', self.methods), | |
('Attributes', self.properties)]: | |
if not self[field]: | |
doc_list = [] | |
for name in sorted(items): | |
try: | |
doc_item = pydoc.getdoc(getattr(self._cls, name)) | |
doc_list.append((name, '', splitlines_x(doc_item))) | |
except AttributeError: | |
pass # method doesn't exist | |
self[field] = doc_list | |
def methods(self): | |
if self._cls is None: | |
return [] | |
return [name for name,func in inspect.getmembers(self._cls) | |
if ((not name.startswith('_') | |
or name in self.extra_public_methods) | |
and isinstance(func, collections.Callable))] | |
def properties(self): | |
if self._cls is None: | |
return [] | |
return [name for name,func in inspect.getmembers(self._cls) | |
if not name.startswith('_') and | |
(func is None or isinstance(func, property) or | |
inspect.isgetsetdescriptor(func))] | |