sphinx-autoapi/autoapi/extension.py

175 lines
6.3 KiB
Python
Raw Normal View History

2015-03-27 19:50:56 +00:00
# -*- coding: utf-8 -*-
"""
Sphinx Auto-API Top-level Extension.
This extension allows you to automagically generate API documentation from your project.
2015-03-27 19:50:56 +00:00
"""
import os
2015-04-07 20:35:50 +00:00
import shutil
2015-03-27 19:50:56 +00:00
2017-02-08 14:53:19 +00:00
import sphinx
2015-06-10 18:35:54 +00:00
from sphinx.util.console import darkgreen, bold
from sphinx.addnodes import toctree
2015-08-03 20:55:33 +00:00
from sphinx.errors import ExtensionError
from docutils.parsers.rst import directives
2015-06-10 18:35:54 +00:00
2015-08-03 20:12:34 +00:00
from .backends import default_file_mapping, default_ignore_patterns, default_backend_mapping
2017-08-31 23:42:47 +00:00
from .directives import AutoapiSummary, NestedParse
from .settings import API_ROOT
from .toctree import add_domain_to_toctree
2015-04-01 00:01:41 +00:00
default_options = ['members', 'undoc-members', 'private-members', 'special-members']
2015-04-01 00:01:41 +00:00
2015-06-10 18:35:54 +00:00
def run_autoapi(app):
"""
Load AutoAPI data from the filesystem.
"""
if not app.config.autoapi_dirs:
2015-10-12 17:50:46 +00:00
raise ExtensionError('You must configure an autoapi_dirs setting')
2015-06-10 18:35:54 +00:00
# Make sure the paths are full
normalized_dirs = []
autoapi_dirs = app.config.autoapi_dirs
if isinstance(autoapi_dirs, str):
autoapi_dirs = [autoapi_dirs]
for path in autoapi_dirs:
if os.path.isabs(path):
normalized_dirs.append(path)
else:
normalized_dirs.append(
os.path.normpath(os.path.join(app.confdir, path))
)
for _dir in normalized_dirs:
if not os.path.exists(_dir):
raise ExtensionError(
2015-09-23 23:04:45 +00:00
'AutoAPI Directory `{dir}` not found. '
'Please check your `autoapi_dirs` setting.'.format(
dir=_dir
)
)
normalized_root = os.path.normpath(os.path.join(app.confdir, app.config.autoapi_root))
url_root = os.path.join('/', app.config.autoapi_root)
2016-11-04 20:42:15 +00:00
sphinx_mapper = default_backend_mapping[app.config.autoapi_type]
sphinx_mapper_obj = sphinx_mapper(app, template_dir=app.config.autoapi_template_dir,
url_root=url_root)
2017-08-31 23:42:47 +00:00
app.env.autoapi_mapper = sphinx_mapper_obj
2015-06-10 18:35:54 +00:00
2015-07-07 23:32:38 +00:00
if app.config.autoapi_file_patterns:
2015-09-22 15:02:34 +00:00
file_patterns = app.config.autoapi_file_patterns
else:
file_patterns = default_file_mapping.get(app.config.autoapi_type, [])
2015-07-20 18:37:31 +00:00
if app.config.autoapi_ignore:
ignore_patterns = app.config.autoapi_ignore
else:
ignore_patterns = default_ignore_patterns.get(app.config.autoapi_type, [])
2015-07-20 18:37:31 +00:00
2016-08-25 17:58:06 +00:00
if '.rst' in app.config.source_suffix:
out_suffix = '.rst'
elif '.txt' in app.config.source_suffix:
out_suffix = '.txt'
else:
# Fallback to first suffix listed
out_suffix = app.config.source_suffix[0]
# Actual meat of the run.
2015-06-10 18:35:54 +00:00
app.info(bold('[AutoAPI] ') + darkgreen('Loading Data'))
2016-11-04 20:42:15 +00:00
sphinx_mapper_obj.load(
patterns=file_patterns,
dirs=normalized_dirs,
2015-07-20 18:37:31 +00:00
ignore=ignore_patterns,
)
2015-06-10 18:35:54 +00:00
app.info(bold('[AutoAPI] ') + darkgreen('Mapping Data'))
2016-11-04 20:42:15 +00:00
sphinx_mapper_obj.map(options=app.config.autoapi_options)
2015-06-10 18:35:54 +00:00
app.info(bold('[AutoAPI] ') + darkgreen('Rendering Data'))
2016-11-04 20:42:15 +00:00
sphinx_mapper_obj.output_rst(
root=normalized_root,
2016-08-25 17:58:06 +00:00
source_suffix=out_suffix,
)
2015-03-27 19:50:56 +00:00
2015-04-07 20:35:50 +00:00
def build_finished(app, exception):
if not app.config.autoapi_keep_files:
normalized_root = os.path.normpath(os.path.join(app.confdir, app.config.autoapi_root))
2015-04-07 20:35:50 +00:00
if app.verbosity > 1:
2015-06-10 18:35:54 +00:00
app.info(bold('[AutoAPI] ') + darkgreen('Cleaning generated .rst files'))
shutil.rmtree(normalized_root)
2015-04-07 20:35:50 +00:00
2016-11-04 20:42:15 +00:00
sphinx_mapper = default_backend_mapping[app.config.autoapi_type]
if hasattr(sphinx_mapper, 'build_finished'):
sphinx_mapper.build_finished(app, exception)
2015-04-07 20:35:50 +00:00
def doctree_read(app, doctree):
"""
Inject AutoAPI into the TOC Tree dynamically.
"""
if app.env.docname == 'index':
all_docs = set()
insert = True
nodes = doctree.traverse(toctree)
2016-11-03 20:24:22 +00:00
toc_entry = '%s/index' % app.config.autoapi_root
if not nodes:
return
# Capture all existing toctree entries
2017-06-29 22:10:06 +00:00
root = nodes[0]
for node in nodes:
for entry in node['entries']:
all_docs.add(entry[1])
# Don't insert autoapi it's already present
for doc in all_docs:
2015-07-07 22:48:00 +00:00
if doc.find(app.config.autoapi_root) != -1:
insert = False
if insert and app.config.autoapi_add_toctree_entry:
if app.config.autoapi_add_api_root_toctree:
# Insert full API TOC in root TOC
for path in app.env.autoapi_toc_entries:
nodes[-1]['entries'].append(
(None, path[1:])
)
nodes[-1]['includefiles'].append(path)
else:
# Insert AutoAPI index
nodes[-1]['entries'].append(
(None, u'%s/index' % app.config.autoapi_root)
)
nodes[-1]['includefiles'].append(u'%s/index' % app.config.autoapi_root)
app.info(bold('[AutoAPI] ') +
darkgreen('Adding AutoAPI TOCTree [%s] to index.rst' % toc_entry)
)
2017-08-31 23:42:47 +00:00
def clear_env(app, env):
"""Clears the environment of the unpicklable objects that we left behind."""
env.autoapi_mapper = None
2015-03-27 19:50:56 +00:00
def setup(app):
2015-06-10 18:35:54 +00:00
app.connect('builder-inited', run_autoapi)
app.connect('doctree-read', doctree_read)
app.connect('doctree-resolved', add_domain_to_toctree)
app.connect('build-finished', build_finished)
2017-08-31 23:42:47 +00:00
app.connect('env-updated', clear_env)
app.add_config_value('autoapi_type', 'python', 'html')
app.add_config_value('autoapi_root', API_ROOT, 'html')
2015-07-20 18:37:31 +00:00
app.add_config_value('autoapi_ignore', [], 'html')
app.add_config_value('autoapi_options', default_options, 'html')
2015-07-07 23:32:38 +00:00
app.add_config_value('autoapi_file_patterns', None, 'html')
app.add_config_value('autoapi_dirs', [], 'html')
2015-06-10 18:35:54 +00:00
app.add_config_value('autoapi_keep_files', False, 'html')
app.add_config_value('autoapi_add_toctree_entry', True, 'html')
app.add_config_value('autoapi_add_api_root_toctree', False, 'html')
2017-08-31 23:43:44 +00:00
app.add_config_value('autoapi_template_dir', None, 'html')
app.add_config_value('autoapi_include_summaries', False, 'html')
2015-04-21 05:54:32 +00:00
app.add_stylesheet('autoapi.css')
directives.register_directive('autoapi-nested-parse', NestedParse)
2017-08-31 23:42:47 +00:00
directives.register_directive('autoapisummary', AutoapiSummary)
app.setup_extension('sphinx.ext.autosummary')