In my code I have
X_DEFAULT = ['a', 'long', 'list', 'of', 'values', 'that', 'is', 'really', 'ugly', 'to', 'see', 'over', 'and', 'over', 'again', 'every', 'time', 'it', 'is', 'referred', 'to', 'in', 'the', 'documentation']
and later
def some_function(..., x=X_DEFAULT, ...):
so that in my Sphinx documentation, using (e.g., using .. autofunction::
, etc.) I get the entire long and unwieldy value of X_DEFAULT
expanded in the signature for some_function
:
some_function(..., x=['a', 'long', 'list', 'of', 'values', 'that', 'is', 'really', 'ugly', 'to', 'see', 'over', 'and', 'over', 'again', 'every', 'time', 'it', 'is', 'referred', 'to', 'in', 'the', 'documentation'], ...)
Is there a way to suppress this substitution in the generated documentation, ideally with a link back to the definition of X_DEFAULT
:
some_function(..., x=X_DEFAULT, ...)
I'm aware that I can manually override the signature for each function and method that I explicitly list as arguments to Sphinx documentation directives, but that's not my goal here. I'm also aware that I could use autodoc_docstring_signature
and the first line of the docstring, but that would produce bad docstrings, really intended for cases where introspection fails (like C). I suspect that there's something I could do in autodoc-process-signature
that might be adequate (but not perfect), though I'm unsure how to proceed.
One approach, that will, for example, "revert" substitution of all values that have been defined at the module level as "public constants" (recognized by all-caps names with no leading underscore) for which a unique name can be found in the module in which it was defined:
def pretty_signature(app, what, name, obj, options, signature, return_annotation):
if what not in ('function', 'method', 'class'):
return
if signature is None:
return
import inspect
mod = inspect.getmodule(obj)
new_sig = signature
# Get all-caps names with no leading underscore
global_names = [name for name in dir(mod) if name.isupper() if name[0] != '_']
# Get only names of variables with distinct values
names_to_replace = [name for name in global_names
if [mod.__dict__[n] for n in global_names].count(mod.__dict__[name]) == 1]
# Substitute name for value in signature, including quotes in a string value
for var_name in names_to_replace:
var_value = mod.__dict__[var_name]
value_string = str(var_value) if type(var_value) is not str else "'{0}'".format(var_value)
new_sig = new_sig.replace(value_string, var_name)
return new_sig, return_annotation
def setup(app):
app.connect('autodoc-process-signature', pretty_signature)
An alternative is to simply take the docstring directly from the source:
import inspect
import re
def pretty_signature(app, what, name, obj, options, signature, return_annotation):
"""Prevent substitution of values for names in signatures by preserving source text."""
if what not in ('function', 'method', 'class') or signature is None:
return
new_sig = signature
if inspect.isfunction(obj) or inspect.isclass(obj) or inspect.ismethod(obj):
sig_obj = obj if not inspect.isclass(obj) else obj.__init__
sig_re = '\((self|cls)?,?\s*(.*?)\)\:'
new_sig = ' '.join(re.search(sig_re, inspect.getsource(sig_obj), re.S).group(2).replace('\n', '').split())
new_sig = '(' + new_sig + ')'
return new_sig, return_annotation
来源:https://stackoverflow.com/questions/21287477/can-i-suppress-variable-expansion-in-sphinx-documentation