Compare commits
23 Commits
18.0
...
14.0-one-d
Author | SHA1 | Date | |
---|---|---|---|
![]() |
eb1a10b199 | ||
![]() |
83d57089fa | ||
![]() |
c084b8cc66 | ||
![]() |
3e00b2dc0a | ||
![]() |
2c55f54d58 | ||
![]() |
04388d5a09 | ||
![]() |
53389b1992 | ||
![]() |
610f2abef3 | ||
![]() |
de2c41927f | ||
![]() |
7a965fa3af | ||
![]() |
722a60d730 | ||
![]() |
95b7e74956 | ||
![]() |
74c4be35bf | ||
![]() |
db3ab859e1 | ||
![]() |
672b487aca | ||
![]() |
3509521c76 | ||
![]() |
65b85f289d | ||
![]() |
9631934593 | ||
![]() |
60510d0a64 | ||
![]() |
6fcd48070a | ||
![]() |
f6d651d2db | ||
![]() |
5b283ac547 | ||
![]() |
fcec9a9d7f |
8
Makefile
8
Makefile
@ -21,8 +21,8 @@ ALLI18NSPHINXOPTS = -d $(BUILDDIR)/doctrees/$(LANG) $(PAPEROPT_$(PAPER)) $(SPHIN
|
|||||||
# the i18n builder cannot share the environment and doctrees with the others
|
# the i18n builder cannot share the environment and doctrees with the others
|
||||||
I18NSPHINXOPTS = $(PAPEROPT_$(PAPER)) $(SPHINXOPTS) .
|
I18NSPHINXOPTS = $(PAPEROPT_$(PAPER)) $(SPHINXOPTS) .
|
||||||
|
|
||||||
lessfiles = _extensions/odoo/static/*.less
|
lessfiles = _extensions/odoo_ext/static/*.less
|
||||||
_extensions/odoo/static/style.css: $(lessfiles)
|
_extensions/odoo_ext/static/style.css: $(lessfiles)
|
||||||
lessc $(LESSOPTS) $(subst .css,.less,$@) $@
|
lessc $(LESSOPTS) $(subst .css,.less,$@) $@
|
||||||
|
|
||||||
.PHONY: help clean html dirhtml singlehtml pickle json htmlhelp qthelp devhelp epub latex latexpdf text man changes linkcheck doctest gettext
|
.PHONY: help clean html dirhtml singlehtml pickle json htmlhelp qthelp devhelp epub latex latexpdf text man changes linkcheck doctest gettext
|
||||||
@ -60,12 +60,12 @@ clean:
|
|||||||
rm -rf $(BUILDDIR)/*
|
rm -rf $(BUILDDIR)/*
|
||||||
|
|
||||||
# These commands are used to create files or run tests
|
# These commands are used to create files or run tests
|
||||||
html: _extensions/odoo/static/style.css
|
html: _extensions/odoo_ext/static/style.css
|
||||||
$(SPHINXBUILD) -b html $(ALLSPHINXOPTS) $(BUILDDIR)/html
|
$(SPHINXBUILD) -b html $(ALLSPHINXOPTS) $(BUILDDIR)/html
|
||||||
@echo
|
@echo
|
||||||
@echo "Build finished. The HTML pages are in $(BUILDDIR)/html."
|
@echo "Build finished. The HTML pages are in $(BUILDDIR)/html."
|
||||||
|
|
||||||
i18nhtml: _extensions/odoo/static/style.css
|
i18nhtml: _extensions/odoo_ext/static/style.css
|
||||||
$(SPHINXBUILD) -b html $(ALLI18NSPHINXOPTS) $(BUILDDIR)/html/$(LANG)
|
$(SPHINXBUILD) -b html $(ALLI18NSPHINXOPTS) $(BUILDDIR)/html/$(LANG)
|
||||||
@echo
|
@echo
|
||||||
@echo "Build finished. The HTML pages are in $(BUILDDIR)/html/$(LANG)."
|
@echo "Build finished. The HTML pages are in $(BUILDDIR)/html/$(LANG)."
|
||||||
|
87
_extensions/autojsdoc/README.rst
Normal file
87
_extensions/autojsdoc/README.rst
Normal file
@ -0,0 +1,87 @@
|
|||||||
|
:orphan:
|
||||||
|
|
||||||
|
======================================
|
||||||
|
JSDoc parser/Sphinx extension for Odoo
|
||||||
|
======================================
|
||||||
|
|
||||||
|
Why?
|
||||||
|
====
|
||||||
|
|
||||||
|
Spent about a week trying to coerce "standard" javascript tools (jsdoc_ with
|
||||||
|
the hope of using sphinx-js_ for integration or `documentation.js`_) and
|
||||||
|
failed to ever get a sensible result: failed to get any result with the
|
||||||
|
current state of the documentation, significant changes/additions/fixes to
|
||||||
|
docstrings brought this up to "garbage output" level.
|
||||||
|
|
||||||
|
Bug reports and mailing list posts didn't show any path to improvement on the
|
||||||
|
ES5 codebase (if we ever go whole-hog on ES6 modules and classes things could
|
||||||
|
be different, in fact most of JSDoc's current effort seem focused on
|
||||||
|
ES6/ES2015 features) but both experience and looking at the mailing lists
|
||||||
|
told me that spending more time would be wasted.
|
||||||
|
|
||||||
|
Even more so as I was writing visitors/rewriters to generate documentation
|
||||||
|
from our existing structure, which broadly speaking is relatively strict, and
|
||||||
|
thus
|
||||||
|
|
||||||
|
What?
|
||||||
|
=====
|
||||||
|
|
||||||
|
If it were possible to generate JSDoc annotations from our relatively
|
||||||
|
well-defined code structures, it was obviously possible to extract documentary
|
||||||
|
information directly from it, hence this Odoo-specific package/extension
|
||||||
|
trying to do exactly that.
|
||||||
|
|
||||||
|
This package should eventually provide:
|
||||||
|
|
||||||
|
* a command-line interface which can be invoked via ``-m autojsdoc`` (assuming
|
||||||
|
your ``PYTHONPATH`` can find it) which should allow dumping the parsed AST
|
||||||
|
in a convenient-ish form, possibly doing searches through the AST, a
|
||||||
|
dependency graph extractor/analysis and a text dumper for the
|
||||||
|
documentation.
|
||||||
|
|
||||||
|
* a sphinx extension (``autojsdoc.sphinx``) which can be used to integrate the
|
||||||
|
parsed JSDoc information into the Sphinx doc.
|
||||||
|
|
||||||
|
How?
|
||||||
|
====
|
||||||
|
|
||||||
|
Sphinx-aside, the package relies on 3 libraries:
|
||||||
|
|
||||||
|
* pyjsparser_, an Esprima-compliant ES5.1 parser (with bits of ES6 support),
|
||||||
|
sadly it does not support comments in its current form so I had to fork it.
|
||||||
|
Fed a javascript source file, pyjsparser_ simply generates a bunch of nested
|
||||||
|
dicts representing an Esprima ast, ast-types_ does a reasonably good job of
|
||||||
|
describing it once you understand that "bases" are basically just structural
|
||||||
|
mixins.
|
||||||
|
|
||||||
|
Because the original does not, this package provides a ``visitor`` module
|
||||||
|
for pyjsparser_ ASTs.
|
||||||
|
|
||||||
|
* pyjsdoc_, a one-file "port" of jsdoc, can actually do much of the JS parsing
|
||||||
|
(using string munging) but its core semantics don't fit our needs so I'm
|
||||||
|
only using it to parse the actual JSDoc content, and the ``jsdoc`` module
|
||||||
|
contains some replacement classes, extensions & monkey patches for things
|
||||||
|
`pyjsdoc`_ itself does not support, at the time of this writing:
|
||||||
|
|
||||||
|
- a bug in FunctionDoc.return_val
|
||||||
|
- a type on FunctionDoc so it's compatible with ParamDoc
|
||||||
|
- a more reliable comments-parsing function
|
||||||
|
- a replacement ModuleDoc as the original does not materialise AMD modules
|
||||||
|
- a ClassDoc extension to support mixins
|
||||||
|
- two additional CommentDoc extensions for "namespaces" objects (bag of
|
||||||
|
attributes without any more information) and mixin objects
|
||||||
|
|
||||||
|
* pytest_ to configure and run the test suite, which you can run by invoking
|
||||||
|
``pytest doc/_extensions`` from the project top-level, the tests represent
|
||||||
|
both "happy path" things we want to parse and various code patterns which
|
||||||
|
tripped the happy path because e.g. they were matched and should not have,
|
||||||
|
they were not matched and should have, or they were more complex than the
|
||||||
|
happy path had expected
|
||||||
|
|
||||||
|
.. _ast-types: _https://github.com/benjamn/ast-types/blob/master/def/core.js
|
||||||
|
.. _documentation.js: http://documentation.js.org
|
||||||
|
.. _jsdoc: http://usejsdoc.org
|
||||||
|
.. _pyjsdoc: https://github.com/nostrademons/pyjsdoc
|
||||||
|
.. _pyjsparser: https://github.com/PiotrDabkowski/pyjsparser
|
||||||
|
.. _pytest: https://pytest.org/
|
||||||
|
.. _sphinx-js: https://sphinx-js-howto.readthedocs.io
|
1
_extensions/autojsdoc/__init__.py
Normal file
1
_extensions/autojsdoc/__init__.py
Normal file
@ -0,0 +1 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
154
_extensions/autojsdoc/__main__.py
Normal file
154
_extensions/autojsdoc/__main__.py
Normal file
@ -0,0 +1,154 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
from __future__ import print_function
|
||||||
|
import cgitb
|
||||||
|
import fnmatch
|
||||||
|
import io
|
||||||
|
import logging
|
||||||
|
|
||||||
|
import click
|
||||||
|
|
||||||
|
import pyjsparser
|
||||||
|
import sys
|
||||||
|
|
||||||
|
from .parser.parser import ModuleMatcher
|
||||||
|
from .parser.visitor import Visitor, SKIP
|
||||||
|
from .parser import jsdoc
|
||||||
|
|
||||||
|
class Printer(Visitor):
|
||||||
|
def __init__(self, level=0):
|
||||||
|
super(Printer, self).__init__()
|
||||||
|
self.level = level
|
||||||
|
|
||||||
|
def _print(self, text):
|
||||||
|
print(' ' * self.level, text)
|
||||||
|
|
||||||
|
def enter_generic(self, node):
|
||||||
|
self._print(node['type'])
|
||||||
|
self.level += 1
|
||||||
|
|
||||||
|
def exit_generic(self, node):
|
||||||
|
self.level -= 1
|
||||||
|
|
||||||
|
def enter_Identifier(self, node):
|
||||||
|
self._print(node['name'])
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
def enter_Literal(self, node):
|
||||||
|
self._print(node['value'])
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
def enter_BinaryExpression(self, node):
|
||||||
|
self._print(node['operator'])
|
||||||
|
self.level += 1
|
||||||
|
|
||||||
|
def visit_files(files, visitor, ctx):
|
||||||
|
for name in files:
|
||||||
|
with io.open(name) as f:
|
||||||
|
ctx.logger.info("%s", name)
|
||||||
|
try:
|
||||||
|
yield visitor().visit(pyjsparser.parse(f.read()))
|
||||||
|
except Exception as e:
|
||||||
|
if ctx.logger.isEnabledFor(logging.DEBUG):
|
||||||
|
ctx.logger.exception("while visiting %s", name)
|
||||||
|
else:
|
||||||
|
ctx.logger.error("%s while visiting %s", e, name)
|
||||||
|
|
||||||
|
# bunch of modules various bits depend on which are not statically defined
|
||||||
|
# (or are outside the scope of the system)
|
||||||
|
ABSTRACT_MODULES = [
|
||||||
|
jsdoc.ModuleDoc({
|
||||||
|
'module': 'web.web_client',
|
||||||
|
'dependency': {'web.AbstractWebClient'},
|
||||||
|
'exports': jsdoc.NSDoc({
|
||||||
|
'name': 'web_client',
|
||||||
|
'doc': 'instance of AbstractWebClient',
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
jsdoc.ModuleDoc({
|
||||||
|
'module': 'web.Tour',
|
||||||
|
'dependency': {'web_tour.TourManager'},
|
||||||
|
'exports': jsdoc.NSDoc({
|
||||||
|
'name': 'Tour',
|
||||||
|
'doc': 'maybe tourmanager instance?',
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
# OH FOR FUCK'S SAKE
|
||||||
|
jsdoc.ModuleDoc({
|
||||||
|
'module': 'summernote/summernote',
|
||||||
|
'exports': jsdoc.NSDoc({'doc': "totally real summernote"}),
|
||||||
|
})
|
||||||
|
]
|
||||||
|
|
||||||
|
@click.group(context_settings={'help_option_names': ['-h', '--help']})
|
||||||
|
@click.option('-v', '--verbose', count=True)
|
||||||
|
@click.option('-q', '--quiet', count=True)
|
||||||
|
@click.pass_context
|
||||||
|
def autojsdoc(ctx, verbose, quiet):
|
||||||
|
logging.basicConfig(
|
||||||
|
level=logging.INFO + (quiet - verbose) * 10,
|
||||||
|
format="[%(levelname)s %(created)f] %(message)s",
|
||||||
|
)
|
||||||
|
ctx.logger = logging.getLogger('autojsdoc')
|
||||||
|
ctx.visitor = None
|
||||||
|
ctx.files = []
|
||||||
|
ctx.kw = {}
|
||||||
|
|
||||||
|
@autojsdoc.command()
|
||||||
|
@click.argument('files', type=click.Path(exists=True), nargs=-1)
|
||||||
|
@click.pass_context
|
||||||
|
def ast(ctx, files):
|
||||||
|
""" Prints a structure tree of the provided files
|
||||||
|
"""
|
||||||
|
if not files:
|
||||||
|
print(ctx.get_help())
|
||||||
|
visit_files(files, lambda: Printer(level=1), ctx.parent)
|
||||||
|
|
||||||
|
@autojsdoc.command()
|
||||||
|
@click.option('-m', '--module', multiple=True, help="Only shows dependencies matching any of the patterns")
|
||||||
|
@click.argument('files', type=click.Path(exists=True), nargs=-1)
|
||||||
|
@click.pass_context
|
||||||
|
def dependencies(ctx, module, files):
|
||||||
|
""" Prints a dot file of all modules to stdout
|
||||||
|
"""
|
||||||
|
if not files:
|
||||||
|
print(ctx.get_help())
|
||||||
|
byname = {
|
||||||
|
mod.name: mod.dependencies
|
||||||
|
for mod in ABSTRACT_MODULES
|
||||||
|
}
|
||||||
|
for modules in visit_files(files, ModuleMatcher, ctx.parent):
|
||||||
|
for mod in modules:
|
||||||
|
byname[mod.name] = mod.dependencies
|
||||||
|
|
||||||
|
print('digraph dependencies {')
|
||||||
|
|
||||||
|
todo = set()
|
||||||
|
# if module filters, roots are only matching modules
|
||||||
|
if module:
|
||||||
|
for f in module:
|
||||||
|
todo.update(fnmatch.filter(byname.keys(), f))
|
||||||
|
|
||||||
|
for m in todo:
|
||||||
|
# set a different box for selected roots
|
||||||
|
print(' "%s" [color=orangered]' % m)
|
||||||
|
else:
|
||||||
|
# otherwise check all modules
|
||||||
|
todo.update(byname)
|
||||||
|
|
||||||
|
done = set()
|
||||||
|
while todo:
|
||||||
|
node = todo.pop()
|
||||||
|
if node in done:
|
||||||
|
continue
|
||||||
|
|
||||||
|
done.add(node)
|
||||||
|
deps = byname[node]
|
||||||
|
todo.update(deps - done)
|
||||||
|
for dep in deps:
|
||||||
|
print(' "%s" -> "%s";' % (node, dep))
|
||||||
|
print('}')
|
||||||
|
|
||||||
|
try:
|
||||||
|
autojsdoc.main(prog_name='autojsdoc')
|
||||||
|
except Exception:
|
||||||
|
print(cgitb.text(sys.exc_info()))
|
13
_extensions/autojsdoc/ext/__init__.py
Normal file
13
_extensions/autojsdoc/ext/__init__.py
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
from .directives import automodule_bound, autodirective_bound
|
||||||
|
from .extractor import _get_roots
|
||||||
|
|
||||||
|
|
||||||
|
def setup(app):
|
||||||
|
app.add_config_value('js_roots', _get_roots, 'env')
|
||||||
|
modules = {}
|
||||||
|
app.add_directive_to_domain('js', 'automodule', automodule_bound(app, modules)
|
||||||
|
)
|
||||||
|
autodirective = autodirective_bound(app, modules)
|
||||||
|
for n in ['autonamespace', 'automixin', 'autoclass', 'autofunction']:
|
||||||
|
app.add_directive_to_domain('js', n, autodirective)
|
719
_extensions/autojsdoc/ext/directives.py
Normal file
719
_extensions/autojsdoc/ext/directives.py
Normal file
@ -0,0 +1,719 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
import abc
|
||||||
|
import collections
|
||||||
|
import contextlib
|
||||||
|
import fnmatch
|
||||||
|
import io
|
||||||
|
import re
|
||||||
|
|
||||||
|
from docutils import nodes
|
||||||
|
from docutils.parsers.rst import Directive
|
||||||
|
from docutils.statemachine import StringList
|
||||||
|
from sphinx import addnodes
|
||||||
|
from sphinx.ext.autodoc import members_set_option, bool_option, ALL
|
||||||
|
|
||||||
|
from autojsdoc.ext.extractor import read_js
|
||||||
|
from ..parser import jsdoc, types
|
||||||
|
|
||||||
|
class DocumenterError(Exception): pass
|
||||||
|
|
||||||
|
@contextlib.contextmanager
|
||||||
|
def addto(parent, newnode):
|
||||||
|
assert isinstance(newnode, nodes.Node), \
|
||||||
|
"Expected newnode to be a Node, got %s" % (type(newnode))
|
||||||
|
yield newnode
|
||||||
|
parent.append(newnode)
|
||||||
|
|
||||||
|
|
||||||
|
def documenter_for(directive, doc):
|
||||||
|
if isinstance(doc, jsdoc.FunctionDoc):
|
||||||
|
return FunctionDocumenter(directive, doc)
|
||||||
|
if isinstance(doc, jsdoc.ClassDoc):
|
||||||
|
return ClassDocumenter(directive, doc)
|
||||||
|
if isinstance(doc, jsdoc.MixinDoc):
|
||||||
|
return MixinDocumenter(directive, doc)
|
||||||
|
if isinstance(doc, (jsdoc.PropertyDoc, jsdoc.LiteralDoc)):
|
||||||
|
return PropertyDocumenter(directive, doc)
|
||||||
|
if isinstance(doc, jsdoc.InstanceDoc):
|
||||||
|
return InstanceDocumenter(directive, doc)
|
||||||
|
if isinstance(doc, jsdoc.Unknown):
|
||||||
|
return UnknownDocumenter(directive, doc)
|
||||||
|
if isinstance(doc, jsdoc.NSDoc):
|
||||||
|
return NSDocumenter(directive, doc)
|
||||||
|
|
||||||
|
raise TypeError("No documenter for %s" % type(doc))
|
||||||
|
|
||||||
|
def to_list(doc, source=None):
|
||||||
|
return StringList([
|
||||||
|
line.rstrip('\n')
|
||||||
|
for line in io.StringIO(doc)
|
||||||
|
], source=source)
|
||||||
|
|
||||||
|
DIRECTIVE_OPTIONS = {
|
||||||
|
'members': members_set_option,
|
||||||
|
'undoc-members': bool_option,
|
||||||
|
'private-members': bool_option,
|
||||||
|
'undoc-matches': bool_option,
|
||||||
|
}
|
||||||
|
def automodule_bound(app, modules):
|
||||||
|
class AutoModuleDirective(Directive):
|
||||||
|
required_arguments = 1
|
||||||
|
has_content = True
|
||||||
|
option_spec = DIRECTIVE_OPTIONS
|
||||||
|
|
||||||
|
# self.state.nested_parse(string, offset, node) => parse context for sub-content (body which can contain RST data)
|
||||||
|
# => needed for doc (converted?) and for actual directive body
|
||||||
|
def run(self):
|
||||||
|
self.env = self.state.document.settings.env
|
||||||
|
modname = self.arguments[0].strip()
|
||||||
|
|
||||||
|
# TODO: cache/memoize modules & symbols?
|
||||||
|
if not modules:
|
||||||
|
read_js(app, modules)
|
||||||
|
|
||||||
|
mods = [
|
||||||
|
(name, mod)
|
||||||
|
for name, mod in modules.items()
|
||||||
|
if fnmatch.fnmatch(name, modname)
|
||||||
|
]
|
||||||
|
ret = []
|
||||||
|
for name, mod in mods:
|
||||||
|
if mod.is_private:
|
||||||
|
continue
|
||||||
|
if name != modname and not (mod.doc or mod.exports):
|
||||||
|
# this module has no documentation, no exports and was
|
||||||
|
# not specifically requested through automodule -> skip
|
||||||
|
# unless requested
|
||||||
|
if not self.options.get('undoc-matches'):
|
||||||
|
continue
|
||||||
|
modsource = mod['sourcefile']
|
||||||
|
if modsource:
|
||||||
|
self.env.note_dependency(modsource)
|
||||||
|
# not sure what that's used for as normal xrefs are resolved using the id directly
|
||||||
|
target = nodes.target('', '', ids=['module-' + name], ismod=True)
|
||||||
|
self.state.document.note_explicit_target(target)
|
||||||
|
|
||||||
|
documenter = ModuleDocumenter(self, mod)
|
||||||
|
|
||||||
|
ret.append(target)
|
||||||
|
ret.extend(documenter.generate())
|
||||||
|
return ret
|
||||||
|
|
||||||
|
return AutoModuleDirective
|
||||||
|
|
||||||
|
def autodirective_bound(app, modules):
|
||||||
|
documenters = {
|
||||||
|
'js:autoclass': ClassDocumenter,
|
||||||
|
'js:autonamespace': NSDocumenter,
|
||||||
|
'js:autofunction': FunctionDocumenter,
|
||||||
|
'js:automixin': MixinDocumenter,
|
||||||
|
}
|
||||||
|
class AutoDirective(Directive):
|
||||||
|
required_arguments = 1
|
||||||
|
has_content = True
|
||||||
|
option_spec = DIRECTIVE_OPTIONS
|
||||||
|
|
||||||
|
def run(self):
|
||||||
|
self.env = self.state.document.settings.env
|
||||||
|
|
||||||
|
objname = self.arguments[0].strip()
|
||||||
|
if not modules:
|
||||||
|
read_js(app, modules)
|
||||||
|
|
||||||
|
# build complete path to object
|
||||||
|
path = self.env.temp_data.get('autojs:prefix', []) + objname.split('.')
|
||||||
|
# look for module/object split
|
||||||
|
for i in range(1, len(path)):
|
||||||
|
modname, objpath = '.'.join(path[:-i]), path[-i:]
|
||||||
|
module = modules.get(modname)
|
||||||
|
if module:
|
||||||
|
break
|
||||||
|
else:
|
||||||
|
raise Exception("Found no valid module in " + '.'.join(path))
|
||||||
|
|
||||||
|
item = module
|
||||||
|
# deref' namespaces until we reach the object we're looking for
|
||||||
|
for k in objpath:
|
||||||
|
item = item.get_property(k)
|
||||||
|
|
||||||
|
docclass = documenters[self.name]
|
||||||
|
return docclass(self, item).generate()
|
||||||
|
|
||||||
|
return AutoDirective
|
||||||
|
|
||||||
|
class Documenter(object):
|
||||||
|
objtype = None
|
||||||
|
def __init__(self, directive, doc):
|
||||||
|
self.directive = directive
|
||||||
|
self.env = directive.env
|
||||||
|
self.item = doc
|
||||||
|
|
||||||
|
@property
|
||||||
|
def modname(self):
|
||||||
|
return self.env.temp_data.get('autojs:module', '')
|
||||||
|
@property
|
||||||
|
def classname(self):
|
||||||
|
return self.env.temp_data.get('autojs:class', '')
|
||||||
|
def generate(self, all_members=False):
|
||||||
|
"""
|
||||||
|
:rtype: List[nodes.Node]
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
return self._generate(all_members=all_members)
|
||||||
|
except Exception as e:
|
||||||
|
raise DocumenterError("Failed to document %s" % self.item) from e
|
||||||
|
def _generate(self, all_members=False):
|
||||||
|
objname = self.item.name
|
||||||
|
prefixed = (self.item['sourcemodule'].name + '.' + objname) if self.item['sourcemodule'] else None
|
||||||
|
objtype = self.objtype
|
||||||
|
assert objtype, '%s has no objtype' % type(self)
|
||||||
|
root = addnodes.desc(domain='js', desctype=objtype, objtype=objtype)
|
||||||
|
with addto(root, addnodes.desc_signature(
|
||||||
|
module=self.modname or '',
|
||||||
|
fullname=objname,
|
||||||
|
)) as s:
|
||||||
|
s['class'] = self.classname
|
||||||
|
|
||||||
|
s['ids'] = []
|
||||||
|
if objname:
|
||||||
|
s['ids'].append(objname)
|
||||||
|
if prefixed:
|
||||||
|
s['ids'].append(prefixed)
|
||||||
|
|
||||||
|
if objtype:
|
||||||
|
s += addnodes.desc_annotation(
|
||||||
|
objtype, objtype,
|
||||||
|
nodes.Text(' '),
|
||||||
|
)
|
||||||
|
|
||||||
|
env = self.env
|
||||||
|
if objname:
|
||||||
|
env.domaindata['js']['objects'][objname] = (env.docname, objtype)
|
||||||
|
if prefixed:
|
||||||
|
env.domaindata['js']['objects'][prefixed] = (env.docname, objtype)
|
||||||
|
|
||||||
|
# TODO: linkcode_resolve
|
||||||
|
s += self.make_signature()
|
||||||
|
with addto(root, addnodes.desc_content()) as c:
|
||||||
|
# must be here otherwise nested_parse(self.content) will not have
|
||||||
|
# the prefix set
|
||||||
|
self.env.temp_data.setdefault('autojs:prefix', []).append(self.item.name)
|
||||||
|
c += self.make_content(all_members=all_members)
|
||||||
|
self.env.temp_data['autojs:prefix'].pop()
|
||||||
|
return [root]
|
||||||
|
|
||||||
|
def make_signature(self):
|
||||||
|
"""
|
||||||
|
:rtype: List[nodes.Node]
|
||||||
|
"""
|
||||||
|
return [addnodes.desc_name(self.item.name, self.item.name)]
|
||||||
|
|
||||||
|
@abc.abstractmethod
|
||||||
|
def make_content(self, all_members):
|
||||||
|
"""
|
||||||
|
:rtype: List[nodes.Node]
|
||||||
|
"""
|
||||||
|
|
||||||
|
def document_subtypes(self, subtypes):
|
||||||
|
docs = []
|
||||||
|
with with_mapping_value(self.directive.options, 'undoc-members', True):
|
||||||
|
for cls in subtypes:
|
||||||
|
docs += ClassDocumenter(self.directive, cls).generate(all_members=True)
|
||||||
|
return docs
|
||||||
|
|
||||||
|
|
||||||
|
class NSDocumenter(Documenter):
|
||||||
|
objtype = 'namespace'
|
||||||
|
def make_content(self, all_members):
|
||||||
|
doc = self.item
|
||||||
|
ret = nodes.section()
|
||||||
|
|
||||||
|
if doc.doc:
|
||||||
|
self.directive.state.nested_parse(to_list(doc.doc), 0, ret)
|
||||||
|
|
||||||
|
self.directive.state.nested_parse(self.directive.content, 0, ret)
|
||||||
|
|
||||||
|
ret += self.document_properties(all_members)
|
||||||
|
return ret.children
|
||||||
|
|
||||||
|
def should_document(self, member, name, all_members):
|
||||||
|
"""
|
||||||
|
:type member: jsdoc.CommentDoc
|
||||||
|
:type name: str
|
||||||
|
:type all_members: bool
|
||||||
|
:rtype: bool
|
||||||
|
"""
|
||||||
|
options = self.directive.options
|
||||||
|
|
||||||
|
members = options.get('members') or []
|
||||||
|
if not (all_members or members is ALL):
|
||||||
|
# if a member is requested by name, it's always documented
|
||||||
|
return name in members
|
||||||
|
|
||||||
|
# ctor params are merged into the class doc
|
||||||
|
if member.is_constructor:
|
||||||
|
return False
|
||||||
|
|
||||||
|
# only document "private" members if option is set
|
||||||
|
if self.is_private(member, name) and not options.get('private-members'):
|
||||||
|
return False
|
||||||
|
|
||||||
|
# TODO: what if member doesn't have a description but has non-desc tags set?
|
||||||
|
# TODO: add @public to force documenting symbol? => useful for implicit typedef
|
||||||
|
return bool(member.doc or options.get('undoc-members'))
|
||||||
|
|
||||||
|
def is_private(self, member, name):
|
||||||
|
return member.is_private
|
||||||
|
|
||||||
|
def document_properties(self, all_members):
|
||||||
|
ret = nodes.section()
|
||||||
|
# TODO: :member-order: [alphabetical | groupwise | bysource]
|
||||||
|
for (n, p) in self.item.properties:
|
||||||
|
if not self.should_document(p, n, all_members):
|
||||||
|
continue
|
||||||
|
# FIXME: maybe should use property name as name inside?
|
||||||
|
ret += documenter_for(self.directive, p).generate(all_members=True)
|
||||||
|
return ret.children
|
||||||
|
|
||||||
|
_NONE = object()
|
||||||
|
@contextlib.contextmanager
|
||||||
|
def with_mapping_value(mapping, key, value, restore_to=_NONE):
|
||||||
|
""" Sets ``key`` to ``value`` for the duration of the context.
|
||||||
|
|
||||||
|
If ``restore_to`` is not provided, restores ``key``'s old value
|
||||||
|
afterwards, removes it entirely if there was no value for ``key`` in the
|
||||||
|
mapping.
|
||||||
|
|
||||||
|
.. warning:: for defaultdict & similar mappings, may restore the default
|
||||||
|
value (depends how the collections' .get behaves)
|
||||||
|
"""
|
||||||
|
if restore_to is _NONE:
|
||||||
|
restore_to = mapping.get(key, _NONE)
|
||||||
|
mapping[key] = value
|
||||||
|
try:
|
||||||
|
yield
|
||||||
|
finally:
|
||||||
|
if restore_to is _NONE:
|
||||||
|
del mapping[key]
|
||||||
|
else:
|
||||||
|
mapping[key] = restore_to
|
||||||
|
class ModuleDocumenter(NSDocumenter):
|
||||||
|
objtype = 'module'
|
||||||
|
def document_properties(self, all_members):
|
||||||
|
with with_mapping_value(self.env.temp_data, 'autojs:module', self.item.name, ''):
|
||||||
|
return super(ModuleDocumenter, self).document_properties(all_members)
|
||||||
|
|
||||||
|
def make_content(self, all_members):
|
||||||
|
doc = self.item
|
||||||
|
content = addnodes.desc_content()
|
||||||
|
|
||||||
|
if doc.exports or doc.dependencies:
|
||||||
|
with addto(content, nodes.field_list()) as fields:
|
||||||
|
if doc.exports:
|
||||||
|
with addto(fields, nodes.field()) as field:
|
||||||
|
field += nodes.field_name('Exports', 'Exports')
|
||||||
|
with addto(field, nodes.field_body()) as body:
|
||||||
|
ref = doc['exports'] # warning: not the same as doc.exports
|
||||||
|
label = ref or '<anonymous>'
|
||||||
|
link = addnodes.pending_xref(
|
||||||
|
ref, nodes.paragraph(ref, label),
|
||||||
|
refdomain='js',
|
||||||
|
reftype='any',
|
||||||
|
reftarget=ref,
|
||||||
|
)
|
||||||
|
link['js:module'] = doc.name
|
||||||
|
body += link
|
||||||
|
|
||||||
|
if doc.dependencies:
|
||||||
|
with addto(fields, nodes.field()) as field:
|
||||||
|
self.make_dependencies(field, doc)
|
||||||
|
|
||||||
|
if doc.doc:
|
||||||
|
# FIXME: source offset
|
||||||
|
self.directive.state.nested_parse(to_list(doc.doc, source=doc['sourcefile']), 0, content)
|
||||||
|
|
||||||
|
self.directive.state.nested_parse(self.directive.content, 0, content)
|
||||||
|
|
||||||
|
content += self.document_properties(all_members)
|
||||||
|
|
||||||
|
return content
|
||||||
|
|
||||||
|
def make_dependencies(self, field, doc):
|
||||||
|
field += nodes.field_name("Depends On", "Depends On")
|
||||||
|
with addto(field, nodes.field_body()) as body:
|
||||||
|
with addto(body, nodes.bullet_list()) as deps:
|
||||||
|
for dep in sorted(doc.dependencies):
|
||||||
|
ref = addnodes.pending_xref(
|
||||||
|
dep, nodes.paragraph(dep, dep),
|
||||||
|
refdomain='js',
|
||||||
|
reftype='module',
|
||||||
|
reftarget=dep,
|
||||||
|
)
|
||||||
|
deps += nodes.list_item(dep, ref)
|
||||||
|
|
||||||
|
def should_document(self, member, name, all_members):
|
||||||
|
# member can be Nothing?
|
||||||
|
if not member:
|
||||||
|
return False
|
||||||
|
modname = getattr(member['sourcemodule'], 'name', None)
|
||||||
|
doc = self.item
|
||||||
|
|
||||||
|
# always document exported symbol (regardless undoc, private, ...)
|
||||||
|
# otherwise things become... somewhat odd
|
||||||
|
if name == doc['exports']:
|
||||||
|
return True
|
||||||
|
|
||||||
|
# if doc['exports'] the module is exporting a "named" item which
|
||||||
|
# does not need to be documented twice, if not doc['exports'] it's
|
||||||
|
# exporting an anonymous item (e.g. object literal) which needs to
|
||||||
|
# be documented on its own
|
||||||
|
if name == '<exports>' and not doc['exports']:
|
||||||
|
return True
|
||||||
|
|
||||||
|
# TODO: :imported-members:
|
||||||
|
# FIXME: *directly* re-exported "foreign symbols"?
|
||||||
|
return (not modname or modname == doc.name) \
|
||||||
|
and super(ModuleDocumenter, self).should_document(member, name, all_members)
|
||||||
|
|
||||||
|
|
||||||
|
class ClassDocumenter(NSDocumenter):
|
||||||
|
objtype = 'class'
|
||||||
|
|
||||||
|
def document_properties(self, all_members):
|
||||||
|
with with_mapping_value(self.env.temp_data, 'autojs:class', self.item.name, ''):
|
||||||
|
return super(ClassDocumenter, self).document_properties(all_members)
|
||||||
|
|
||||||
|
def make_signature(self):
|
||||||
|
sig = super(ClassDocumenter, self).make_signature()
|
||||||
|
sig.append(self.make_parameters())
|
||||||
|
return sig
|
||||||
|
|
||||||
|
def make_parameters(self):
|
||||||
|
params = addnodes.desc_parameterlist('', '')
|
||||||
|
ctor = self.item.constructor
|
||||||
|
if ctor:
|
||||||
|
params += make_desc_parameters(ctor.params)
|
||||||
|
|
||||||
|
return params
|
||||||
|
|
||||||
|
def make_content(self, all_members):
|
||||||
|
doc = self.item
|
||||||
|
ret = nodes.section()
|
||||||
|
|
||||||
|
ctor = self.item.constructor
|
||||||
|
params = subtypes = []
|
||||||
|
if ctor:
|
||||||
|
check_parameters(self, ctor)
|
||||||
|
params, subtypes = extract_subtypes(doc.name, ctor)
|
||||||
|
|
||||||
|
fields = nodes.field_list()
|
||||||
|
fields += self.make_super()
|
||||||
|
fields += self.make_mixins()
|
||||||
|
fields += self.make_params(params)
|
||||||
|
if fields.children:
|
||||||
|
ret += fields
|
||||||
|
|
||||||
|
if doc.doc:
|
||||||
|
self.directive.state.nested_parse(to_list(doc.doc), 0, ret)
|
||||||
|
|
||||||
|
self.directive.state.nested_parse(self.directive.content, 0, ret)
|
||||||
|
|
||||||
|
ret += self.document_properties(all_members)
|
||||||
|
|
||||||
|
ret += self.document_subtypes(subtypes)
|
||||||
|
|
||||||
|
return ret.children
|
||||||
|
|
||||||
|
def is_private(self, member, name):
|
||||||
|
return name.startswith('_') or super(ClassDocumenter, self).is_private(member, name)
|
||||||
|
|
||||||
|
def make_super(self):
|
||||||
|
doc = self.item
|
||||||
|
if not doc.superclass:
|
||||||
|
return []
|
||||||
|
|
||||||
|
sup_link = addnodes.pending_xref(
|
||||||
|
doc.superclass.name, nodes.paragraph(doc.superclass.name, doc.superclass.name),
|
||||||
|
refdomain='js', reftype='class', reftarget=doc.superclass.name,
|
||||||
|
)
|
||||||
|
sup_link['js:module'] = doc.superclass['sourcemodule'].name
|
||||||
|
return nodes.field(
|
||||||
|
'',
|
||||||
|
nodes.field_name("Extends", "Extends"),
|
||||||
|
nodes.field_body(doc.superclass.name, sup_link),
|
||||||
|
)
|
||||||
|
|
||||||
|
def make_mixins(self):
|
||||||
|
doc = self.item
|
||||||
|
if not doc.mixins:
|
||||||
|
return []
|
||||||
|
|
||||||
|
ret = nodes.field('', nodes.field_name("Mixes", "Mixes"))
|
||||||
|
with addto(ret, nodes.field_body()) as body:
|
||||||
|
with addto(body, nodes.bullet_list()) as mixins:
|
||||||
|
for mixin in sorted(doc.mixins, key=lambda m: m.name):
|
||||||
|
mixin_link = addnodes.pending_xref(
|
||||||
|
mixin.name, nodes.paragraph(mixin.name, mixin.name),
|
||||||
|
refdomain='js', reftype='mixin', reftarget=mixin.name
|
||||||
|
)
|
||||||
|
mixin_link['js:module'] = mixin['sourcemodule'].name
|
||||||
|
mixins += nodes.list_item('', mixin_link)
|
||||||
|
return ret
|
||||||
|
|
||||||
|
def make_params(self, params):
|
||||||
|
if not params:
|
||||||
|
return []
|
||||||
|
|
||||||
|
ret = nodes.field('', nodes.field_name('Parameters', 'Parameters'))
|
||||||
|
with addto(ret, nodes.field_body()) as body,\
|
||||||
|
addto(body, nodes.bullet_list()) as holder:
|
||||||
|
holder += make_parameters(params, mod=self.modname)
|
||||||
|
return ret
|
||||||
|
|
||||||
|
class InstanceDocumenter(Documenter):
|
||||||
|
objtype = 'object'
|
||||||
|
def make_signature(self):
|
||||||
|
cls = self.item.cls
|
||||||
|
ret = super(InstanceDocumenter, self).make_signature()
|
||||||
|
if cls:
|
||||||
|
super_ref = addnodes.pending_xref(
|
||||||
|
cls.name, nodes.Text(cls.name, cls.name),
|
||||||
|
refdomain='js', reftype='class', reftarget=cls.name
|
||||||
|
)
|
||||||
|
super_ref['js:module'] = cls['sourcemodule'].name
|
||||||
|
ret.append(addnodes.desc_annotation(' instance of ', ' instance of '))
|
||||||
|
ret.append(addnodes.desc_type(cls.name, '', super_ref))
|
||||||
|
if not ret:
|
||||||
|
return [addnodes.desc_name('???', '???')]
|
||||||
|
return ret
|
||||||
|
|
||||||
|
def make_content(self, all_members):
|
||||||
|
ret = nodes.section()
|
||||||
|
|
||||||
|
if self.item.doc:
|
||||||
|
self.directive.state.nested_parse(to_list(self.item.doc), 0, ret)
|
||||||
|
|
||||||
|
self.directive.state.nested_parse(self.directive.content, 0, ret)
|
||||||
|
|
||||||
|
return ret.children
|
||||||
|
|
||||||
|
class FunctionDocumenter(Documenter):
|
||||||
|
@property
|
||||||
|
def objtype(self):
|
||||||
|
return 'method' if self.classname else 'function'
|
||||||
|
def make_signature(self):
|
||||||
|
ret = super(FunctionDocumenter, self).make_signature()
|
||||||
|
with addto(ret, addnodes.desc_parameterlist()) as params:
|
||||||
|
params += make_desc_parameters(self.item.params)
|
||||||
|
retval = self.item.return_val
|
||||||
|
if retval.type or retval.doc:
|
||||||
|
ret.append(addnodes.desc_returns(retval.type or '*', retval.type or '*'))
|
||||||
|
return ret
|
||||||
|
|
||||||
|
def make_content(self, all_members):
|
||||||
|
ret = nodes.section()
|
||||||
|
doc = self.item
|
||||||
|
|
||||||
|
if doc.doc:
|
||||||
|
self.directive.state.nested_parse(to_list(doc.doc), 0, ret)
|
||||||
|
|
||||||
|
self.directive.state.nested_parse(self.directive.content, 0, ret)
|
||||||
|
|
||||||
|
check_parameters(self, doc)
|
||||||
|
|
||||||
|
params, subtypes = extract_subtypes(self.item.name, self.item)
|
||||||
|
rdoc = doc.return_val.doc
|
||||||
|
rtype = doc.return_val.type
|
||||||
|
if params or rtype or rdoc:
|
||||||
|
with addto(ret, nodes.field_list()) as fields:
|
||||||
|
if params:
|
||||||
|
with addto(fields, nodes.field()) as field:
|
||||||
|
field += nodes.field_name('Parameters', 'Parameters')
|
||||||
|
with addto(field, nodes.field_body()) as body,\
|
||||||
|
addto(body, nodes.bullet_list()) as holder:
|
||||||
|
holder.extend(make_parameters(params, mod=doc['sourcemodule'].name))
|
||||||
|
if rdoc:
|
||||||
|
with addto(fields, nodes.field()) as field:
|
||||||
|
field += nodes.field_name("Returns", "Returns")
|
||||||
|
with addto(field, nodes.field_body()) as body,\
|
||||||
|
addto(body, nodes.paragraph()) as p:
|
||||||
|
p += nodes.inline(rdoc, rdoc)
|
||||||
|
if rtype:
|
||||||
|
with addto(fields, nodes.field()) as field:
|
||||||
|
field += nodes.field_name("Return Type", "Return Type")
|
||||||
|
with addto(field, nodes.field_body()) as body, \
|
||||||
|
addto(body, nodes.paragraph()) as p:
|
||||||
|
p += make_types(rtype, mod=doc['sourcemodule'].name)
|
||||||
|
|
||||||
|
ret += self.document_subtypes(subtypes)
|
||||||
|
|
||||||
|
return ret.children
|
||||||
|
|
||||||
|
def pascal_case_ify(name):
|
||||||
|
"""
|
||||||
|
Uppercase first letter of ``name``, or any letter following an ``_``. In
|
||||||
|
the latter case, also strips out the ``_``.
|
||||||
|
|
||||||
|
=> key_for becomes KeyFor
|
||||||
|
=> options becomes Options
|
||||||
|
"""
|
||||||
|
return re.sub(r'(^|_)\w', lambda m: m.group(0)[-1].upper(), name)
|
||||||
|
def extract_subtypes(parent_name, doc):
|
||||||
|
""" Extracts composite parameters (a.b) into sub-types for the parent
|
||||||
|
parameter, swaps the parent's type from whatever it is to the extracted
|
||||||
|
one, and returns the extracted type for inclusion into the parent.
|
||||||
|
|
||||||
|
:arg parent_name: name of the containing symbol (function, class), will
|
||||||
|
be used to compose subtype names
|
||||||
|
:type parent_name: str
|
||||||
|
:type doc: FunctionDoc
|
||||||
|
:rtype: (List[ParamDoc], List[ClassDoc])
|
||||||
|
"""
|
||||||
|
# map of {param_name: [ParamDoc]} (from complete doc)
|
||||||
|
subparams = collections.defaultdict(list)
|
||||||
|
for p in map(jsdoc.ParamDoc, doc.get_as_list('param')):
|
||||||
|
pair = p.name.split('.', 1)
|
||||||
|
if len(pair) == 2:
|
||||||
|
k, p.name = pair # remove prefix from param name
|
||||||
|
subparams[k].append(p)
|
||||||
|
|
||||||
|
# keep original params order as that's the order of formal parameters in
|
||||||
|
# the function signature
|
||||||
|
params = collections.OrderedDict((p.name, p) for p in doc.params)
|
||||||
|
subtypes = []
|
||||||
|
# now we can use the subparams map to extract "compound" parameter types
|
||||||
|
# and swap the new type for the original param's type
|
||||||
|
for param_name, subs in subparams.items():
|
||||||
|
typename = '%s%s' % (
|
||||||
|
pascal_case_ify(parent_name),
|
||||||
|
pascal_case_ify(param_name),
|
||||||
|
)
|
||||||
|
param = params[param_name]
|
||||||
|
param.type = typename
|
||||||
|
subtypes.append(jsdoc.ClassDoc({
|
||||||
|
'name': typename,
|
||||||
|
'doc': param.doc,
|
||||||
|
'_members': [
|
||||||
|
# TODO: add default value
|
||||||
|
(sub.name, jsdoc.PropertyDoc(dict(sub.to_dict(), sourcemodule=doc['sourcemodule'])))
|
||||||
|
for sub in subs
|
||||||
|
],
|
||||||
|
'sourcemodule': doc['sourcemodule'],
|
||||||
|
}))
|
||||||
|
return params.values(), subtypes
|
||||||
|
|
||||||
|
def check_parameters(documenter, doc):
|
||||||
|
"""
|
||||||
|
Check that all documented parameters match a formal parameter for the
|
||||||
|
function. Documented params which don't match the actual function may be
|
||||||
|
typos.
|
||||||
|
"""
|
||||||
|
guessed = set(doc['guessed_params'] or [])
|
||||||
|
if not guessed:
|
||||||
|
return
|
||||||
|
|
||||||
|
documented = {
|
||||||
|
# param name can be of the form [foo.bar.baz=default]\ndescription
|
||||||
|
jsdoc.ParamDoc(text).name.split('.')[0]
|
||||||
|
for text in doc.get_as_list('param')
|
||||||
|
}
|
||||||
|
odd = documented - guessed
|
||||||
|
if not odd:
|
||||||
|
return
|
||||||
|
|
||||||
|
app = documenter.directive.env.app
|
||||||
|
app.warn("Found documented params %s not in formal parameter list "
|
||||||
|
"of function %s in module %s (%s)" % (
|
||||||
|
', '.join(odd),
|
||||||
|
doc.name,
|
||||||
|
documenter.modname,
|
||||||
|
doc['sourcemodule']['sourcefile'],
|
||||||
|
))
|
||||||
|
|
||||||
|
def make_desc_parameters(params):
|
||||||
|
for p in params:
|
||||||
|
if '.' in p.name:
|
||||||
|
continue
|
||||||
|
|
||||||
|
node = addnodes.desc_parameter(p.name, p.name)
|
||||||
|
if p.optional:
|
||||||
|
node = addnodes.desc_optional('', '', node)
|
||||||
|
yield node
|
||||||
|
|
||||||
|
def make_parameters(params, mod=None):
|
||||||
|
for param in params:
|
||||||
|
p = nodes.paragraph('', '', nodes.strong(param.name, param.name))
|
||||||
|
if param.default is not None:
|
||||||
|
p += nodes.Text('=', '=')
|
||||||
|
p += nodes.emphasis(param.default, param.default)
|
||||||
|
if param.type:
|
||||||
|
p += nodes.Text(' (')
|
||||||
|
p += make_types(param.type, mod=mod)
|
||||||
|
p += nodes.Text(')')
|
||||||
|
if param.doc:
|
||||||
|
p += [
|
||||||
|
nodes.Text(' -- '),
|
||||||
|
nodes.inline(param.doc, param.doc)
|
||||||
|
]
|
||||||
|
yield p
|
||||||
|
|
||||||
|
|
||||||
|
def _format_value(v):
|
||||||
|
if v == '|':
|
||||||
|
return nodes.emphasis(' or ', ' or ')
|
||||||
|
if v == ',':
|
||||||
|
return nodes.Text(', ', ', ')
|
||||||
|
return nodes.Text(v, v)
|
||||||
|
def make_types(typespec, mod=None):
|
||||||
|
# TODO: in closure notation {type=} => optional, do we care?
|
||||||
|
def format_type(t):
|
||||||
|
ref = addnodes.pending_xref(
|
||||||
|
t, addnodes.literal_emphasis(t, t),
|
||||||
|
refdomain='js', reftype='class', reftarget=t,
|
||||||
|
)
|
||||||
|
if mod:
|
||||||
|
ref['js:module'] = mod
|
||||||
|
return ref
|
||||||
|
|
||||||
|
try:
|
||||||
|
return types.iterate(
|
||||||
|
types.parse(typespec),
|
||||||
|
format_type,
|
||||||
|
_format_value
|
||||||
|
)
|
||||||
|
except ValueError as e:
|
||||||
|
raise ValueError("%s in '%s'" % (e, typespec))
|
||||||
|
|
||||||
|
|
||||||
|
class MixinDocumenter(NSDocumenter):
|
||||||
|
objtype = 'mixin'
|
||||||
|
|
||||||
|
class PropertyDocumenter(Documenter):
|
||||||
|
objtype = 'attribute'
|
||||||
|
def make_signature(self):
|
||||||
|
ret = super(PropertyDocumenter, self).make_signature()
|
||||||
|
proptype = self.item.type
|
||||||
|
if proptype:
|
||||||
|
typeref = addnodes.pending_xref(
|
||||||
|
proptype, nodes.Text(proptype, proptype),
|
||||||
|
refdomain='js', reftype='class', reftarget=proptype
|
||||||
|
)
|
||||||
|
typeref['js:module'] = self.item['sourcemodule'].name
|
||||||
|
ret.append(nodes.Text(' '))
|
||||||
|
ret.append(typeref)
|
||||||
|
return ret
|
||||||
|
|
||||||
|
def make_content(self, all_members):
|
||||||
|
doc = self.item
|
||||||
|
ret = nodes.section()
|
||||||
|
|
||||||
|
self.directive.state.nested_parse(self.directive.content, 0, ret)
|
||||||
|
|
||||||
|
if doc.doc:
|
||||||
|
self.directive.state.nested_parse(to_list(doc.doc), 0, ret)
|
||||||
|
return ret.children
|
||||||
|
|
||||||
|
class UnknownDocumenter(Documenter):
|
||||||
|
objtype = 'unknown'
|
||||||
|
def make_content(self, all_members):
|
||||||
|
return []
|
87
_extensions/autojsdoc/ext/extractor.py
Normal file
87
_extensions/autojsdoc/ext/extractor.py
Normal file
@ -0,0 +1,87 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
import io
|
||||||
|
import os
|
||||||
|
|
||||||
|
import pyjsdoc
|
||||||
|
import pyjsparser
|
||||||
|
|
||||||
|
try:
|
||||||
|
from sphinx.util import status_iterator
|
||||||
|
def it(app): return status_iterator
|
||||||
|
except ImportError:
|
||||||
|
# 1.2: Builder.status_iterator
|
||||||
|
# 1.3: Application.status_iterator (with alias on Builder)
|
||||||
|
# 1.6: sphinx.util.status_iterator (with *deprecated* aliases on Application and Builder)
|
||||||
|
# 1.7: removed Application and Builder aliases
|
||||||
|
# => if no sphinx.util.status_iterator, fallback onto the builder one for
|
||||||
|
# 1.2 compatibility, remove this entirely if we ever require 1.6+
|
||||||
|
status_iterator = None
|
||||||
|
def it(app): return app.builder.status_iterator
|
||||||
|
|
||||||
|
from ..parser import jsdoc, parser
|
||||||
|
|
||||||
|
|
||||||
|
def _get_roots(conf):
|
||||||
|
env_roots = os.environ.get('AUTOJSDOC_ROOTS_PATH')
|
||||||
|
if env_roots:
|
||||||
|
return env_roots.split(':')
|
||||||
|
return []
|
||||||
|
|
||||||
|
def read_js(app, modules):
|
||||||
|
"""
|
||||||
|
:type app: sphinx.application.Sphinx
|
||||||
|
:type modules: Dict[str, jsdoc.ModuleDoc]
|
||||||
|
"""
|
||||||
|
roots = map(os.path.normpath, app.config.js_roots or [os.path.join(app.confdir, '..')])
|
||||||
|
files = [
|
||||||
|
os.path.join(r, f)
|
||||||
|
for root in roots
|
||||||
|
for r, _, fs in os.walk(root)
|
||||||
|
if 'static/src/js' in r
|
||||||
|
for f in fs
|
||||||
|
if f.endswith('.js')
|
||||||
|
]
|
||||||
|
|
||||||
|
modules.update((mod.name, mod) for mod in ABSTRACT_MODULES)
|
||||||
|
for name in it(app)(files, "Parsing javascript files...", length=len(files)):
|
||||||
|
with io.open(name) as f:
|
||||||
|
ast = pyjsparser.parse(f.read())
|
||||||
|
modules.update(
|
||||||
|
(mod.name, mod)
|
||||||
|
for mod in parser.ModuleMatcher(name).visit(ast)
|
||||||
|
)
|
||||||
|
_resolve_references(modules)
|
||||||
|
|
||||||
|
def _resolve_references(byname):
|
||||||
|
# must be done in topological order otherwise the dependent can't
|
||||||
|
# resolve non-trivial references to a dependency properly
|
||||||
|
for name in pyjsdoc.topological_sort(
|
||||||
|
*pyjsdoc.build_dependency_graph(
|
||||||
|
list(byname.keys()),
|
||||||
|
byname
|
||||||
|
)
|
||||||
|
):
|
||||||
|
byname[name].post_process(byname)
|
||||||
|
|
||||||
|
ABSTRACT_MODULES = [
|
||||||
|
jsdoc.ModuleDoc({
|
||||||
|
'module': u'web.web_client',
|
||||||
|
'dependency': {u'web.AbstractWebClient'},
|
||||||
|
'exports': jsdoc.NSDoc({
|
||||||
|
'name': u'web_client',
|
||||||
|
'doc': u'instance of AbstractWebClient',
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
jsdoc.ModuleDoc({
|
||||||
|
'module': u'web.Tour',
|
||||||
|
'dependency': {u'web_tour.TourManager'},
|
||||||
|
'exports': jsdoc.NSDoc({
|
||||||
|
'name': u'Tour',
|
||||||
|
'doc': u'maybe tourmanager instance?',
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
jsdoc.ModuleDoc({
|
||||||
|
'module': u'summernote/summernote',
|
||||||
|
'exports': jsdoc.NSDoc({'doc': u"totally real summernote"}),
|
||||||
|
})
|
||||||
|
]
|
1
_extensions/autojsdoc/parser/__init__.py
Normal file
1
_extensions/autojsdoc/parser/__init__.py
Normal file
@ -0,0 +1 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
357
_extensions/autojsdoc/parser/jsdoc.py
Normal file
357
_extensions/autojsdoc/parser/jsdoc.py
Normal file
@ -0,0 +1,357 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
import re
|
||||||
|
|
||||||
|
import collections
|
||||||
|
|
||||||
|
import pyjsdoc
|
||||||
|
|
||||||
|
def strip_stars(doc_comment):
|
||||||
|
"""
|
||||||
|
Version of jsdoc.strip_stars which always removes 1 space after * if
|
||||||
|
one is available.
|
||||||
|
"""
|
||||||
|
return re.sub('\n\s*?\*[\t ]?', '\n', doc_comment[3:-2]).strip()
|
||||||
|
|
||||||
|
class ParamDoc(pyjsdoc.ParamDoc):
|
||||||
|
"""
|
||||||
|
Replace ParamDoc because FunctionDoc doesn't properly handle optional
|
||||||
|
params or default values (TODO: or compounds) if guessed_params is used
|
||||||
|
|
||||||
|
=> augment paramdoc with "required" and "default" items to clean up name
|
||||||
|
"""
|
||||||
|
def __init__(self, text):
|
||||||
|
super(ParamDoc, self).__init__(text)
|
||||||
|
# param name and doc can be separated by - or :, strip it
|
||||||
|
self.doc = self.doc.strip().lstrip('-:').lstrip()
|
||||||
|
self.optional = False
|
||||||
|
self.default = None
|
||||||
|
# there may not be a space between the param name and the :, in which
|
||||||
|
# case the : gets attached to the name, strip *again*
|
||||||
|
# TODO: formal @param/@property parser to handle this crap properly once and for all
|
||||||
|
self.name = self.name.strip().rstrip(':')
|
||||||
|
if self.name.startswith('['):
|
||||||
|
self.name = self.name.strip('[]')
|
||||||
|
self.optional = True
|
||||||
|
if '=' in self.name:
|
||||||
|
self.name, self.default = self.name.rsplit('=', 1)
|
||||||
|
def to_dict(self):
|
||||||
|
d = super(ParamDoc, self).to_dict()
|
||||||
|
d['optional'] = self.optional
|
||||||
|
d['default'] = self.default
|
||||||
|
return d
|
||||||
|
pyjsdoc.ParamDoc = ParamDoc
|
||||||
|
|
||||||
|
class CommentDoc(pyjsdoc.CommentDoc):
|
||||||
|
namekey = object()
|
||||||
|
is_constructor = False
|
||||||
|
|
||||||
|
@property
|
||||||
|
def name(self):
|
||||||
|
return self[self.namekey] or self['name'] or self['guessed_name']
|
||||||
|
def set_name(self, name):
|
||||||
|
# not great...
|
||||||
|
if name != '<exports>':
|
||||||
|
self.parsed['guessed_name'] = name
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_private(self):
|
||||||
|
return 'private' in self.parsed
|
||||||
|
|
||||||
|
def to_dict(self):
|
||||||
|
d = super(CommentDoc, self).to_dict()
|
||||||
|
d['name'] = self.name
|
||||||
|
return d
|
||||||
|
|
||||||
|
# don't resolve already resolved docs (e.g. a literal dict being
|
||||||
|
# include-ed in two different classes because I don't even care anymore
|
||||||
|
def become(self, modules):
|
||||||
|
return self
|
||||||
|
|
||||||
|
class PropertyDoc(CommentDoc):
|
||||||
|
@classmethod
|
||||||
|
def from_param(cls, s, sourcemodule=None):
|
||||||
|
parsed = ParamDoc(s).to_dict()
|
||||||
|
parsed['sourcemodule'] = sourcemodule
|
||||||
|
return cls(parsed)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def type(self):
|
||||||
|
return self['type'].strip('{}')
|
||||||
|
|
||||||
|
def to_dict(self):
|
||||||
|
d = super(PropertyDoc, self).to_dict()
|
||||||
|
d['type'] = self.type
|
||||||
|
d['is_private'] = self.is_private
|
||||||
|
return d
|
||||||
|
|
||||||
|
class InstanceDoc(CommentDoc):
|
||||||
|
@property
|
||||||
|
def cls(self):
|
||||||
|
return self['cls']
|
||||||
|
|
||||||
|
def to_dict(self):
|
||||||
|
return dict(super(InstanceDoc, self).to_dict(), cls=self.cls)
|
||||||
|
|
||||||
|
class LiteralDoc(CommentDoc):
|
||||||
|
@property
|
||||||
|
def type(self):
|
||||||
|
if self['type']:
|
||||||
|
return self['type']
|
||||||
|
valtype = type(self['value'])
|
||||||
|
if valtype is bool:
|
||||||
|
return 'Boolean'
|
||||||
|
elif valtype is float:
|
||||||
|
return 'Number'
|
||||||
|
elif valtype is type(u''):
|
||||||
|
return 'String'
|
||||||
|
return ''
|
||||||
|
|
||||||
|
@property
|
||||||
|
def value(self):
|
||||||
|
return self['value']
|
||||||
|
|
||||||
|
def to_dict(self):
|
||||||
|
d = super(LiteralDoc, self).to_dict()
|
||||||
|
d['type'] = self.type
|
||||||
|
d['value'] = self.value
|
||||||
|
return d
|
||||||
|
|
||||||
|
class FunctionDoc(CommentDoc):
|
||||||
|
type = 'Function'
|
||||||
|
namekey = 'function'
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_constructor(self):
|
||||||
|
return self.name == 'init'
|
||||||
|
|
||||||
|
@property
|
||||||
|
def params(self):
|
||||||
|
tag_texts = self.get_as_list('param')
|
||||||
|
# turns out guessed_params is *almost* (?) always set to a list,
|
||||||
|
# if empty list of guessed params fall back to @params
|
||||||
|
if not self['guessed_params']:
|
||||||
|
# only get "primary" params (no "." in name)
|
||||||
|
return [
|
||||||
|
p for p in map(ParamDoc, tag_texts)
|
||||||
|
if '.' not in p.name
|
||||||
|
]
|
||||||
|
else:
|
||||||
|
param_dict = {}
|
||||||
|
for text in tag_texts:
|
||||||
|
param = ParamDoc(text)
|
||||||
|
param_dict[param.name] = param
|
||||||
|
return [param_dict.get(name) or ParamDoc('{} ' + name)
|
||||||
|
for name in self.get('guessed_params')]
|
||||||
|
@property
|
||||||
|
def return_val(self):
|
||||||
|
ret = self.get('return') or self.get('returns')
|
||||||
|
type = self.get('type')
|
||||||
|
if '{' in ret and '}' in ret:
|
||||||
|
if not '} ' in ret:
|
||||||
|
# Ensure that name is empty
|
||||||
|
ret = re.sub(r'\}\s*', '} ', ret)
|
||||||
|
return ParamDoc(ret)
|
||||||
|
if ret and type:
|
||||||
|
return ParamDoc('{%s} %s' % (type, ret))
|
||||||
|
return ParamDoc(ret)
|
||||||
|
|
||||||
|
def to_dict(self):
|
||||||
|
d = super(FunctionDoc, self).to_dict()
|
||||||
|
d['name'] = self.name
|
||||||
|
d['params'] = [param.to_dict() for param in self.params]
|
||||||
|
d['return_val']= self.return_val.to_dict()
|
||||||
|
return d
|
||||||
|
|
||||||
|
class NSDoc(CommentDoc):
|
||||||
|
namekey = 'namespace'
|
||||||
|
def __init__(self, parsed_comment):
|
||||||
|
super(NSDoc, self).__init__(parsed_comment)
|
||||||
|
self.members = collections.OrderedDict()
|
||||||
|
def add_member(self, name, member):
|
||||||
|
"""
|
||||||
|
:type name: str
|
||||||
|
:type member: CommentDoc
|
||||||
|
"""
|
||||||
|
member.set_name(name)
|
||||||
|
self.members[name] = member
|
||||||
|
|
||||||
|
@property
|
||||||
|
def properties(self):
|
||||||
|
if self.get('property'):
|
||||||
|
return [
|
||||||
|
(p.name, p)
|
||||||
|
for p in (
|
||||||
|
PropertyDoc.from_param(p, self['sourcemodule'])
|
||||||
|
for p in self.get_as_list('property')
|
||||||
|
)
|
||||||
|
]
|
||||||
|
return list(self.members.items()) or self['_members'] or []
|
||||||
|
|
||||||
|
def has_property(self, name):
|
||||||
|
return self.get_property(name) is not None
|
||||||
|
|
||||||
|
def get_property(self, name):
|
||||||
|
return next((p for n, p in self.properties if n == name), None)
|
||||||
|
|
||||||
|
def to_dict(self):
|
||||||
|
d = super(NSDoc, self).to_dict()
|
||||||
|
d['properties'] = [(n, p.to_dict()) for n, p in self.properties]
|
||||||
|
return d
|
||||||
|
|
||||||
|
class MixinDoc(NSDoc):
|
||||||
|
namekey = 'mixin'
|
||||||
|
|
||||||
|
class ModuleDoc(NSDoc):
|
||||||
|
namekey = 'module'
|
||||||
|
def __init__(self, parsed_comment):
|
||||||
|
super(ModuleDoc, self).__init__(parsed_comment)
|
||||||
|
#: callbacks to run with the modules mapping once every module is resolved
|
||||||
|
self._post_process = []
|
||||||
|
|
||||||
|
def post_process(self, modules):
|
||||||
|
for callback in self._post_process:
|
||||||
|
callback(modules)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def module(self):
|
||||||
|
return self # lol
|
||||||
|
|
||||||
|
@property
|
||||||
|
def dependencies(self):
|
||||||
|
"""
|
||||||
|
Returns the immediate dependencies of a module (only those explicitly
|
||||||
|
declared/used).
|
||||||
|
"""
|
||||||
|
return self.get('dependency', None) or set()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def exports(self):
|
||||||
|
"""
|
||||||
|
Returns the actual item exported from the AMD module, can be a
|
||||||
|
namespace, a class, a function, an instance, ...
|
||||||
|
"""
|
||||||
|
return self.get_property('<exports>')
|
||||||
|
|
||||||
|
def to_dict(self):
|
||||||
|
vars = super(ModuleDoc, self).to_dict()
|
||||||
|
vars['dependencies'] = self.dependencies
|
||||||
|
vars['exports'] = self.exports
|
||||||
|
return vars
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
s = super().__str__()
|
||||||
|
if self['sourcefile']:
|
||||||
|
s += " in file " + self['sourcefile']
|
||||||
|
return s
|
||||||
|
|
||||||
|
class ClassDoc(NSDoc):
|
||||||
|
namekey = 'class'
|
||||||
|
@property
|
||||||
|
def constructor(self):
|
||||||
|
return self.get_property('init')
|
||||||
|
|
||||||
|
@property
|
||||||
|
def superclass(self):
|
||||||
|
return self['extends'] or self['base']
|
||||||
|
|
||||||
|
def get_property(self, method_name):
|
||||||
|
if method_name == 'extend':
|
||||||
|
return FunctionDoc({
|
||||||
|
'doc': 'Create subclass for %s' % self.name,
|
||||||
|
'guessed_function': 'extend',
|
||||||
|
})
|
||||||
|
# FIXME: should ideally be a proxy namespace
|
||||||
|
if method_name == 'prototype':
|
||||||
|
return self
|
||||||
|
return super(ClassDoc, self).get_property(method_name)\
|
||||||
|
or (self.superclass and self.superclass.get_property(method_name))
|
||||||
|
|
||||||
|
@property
|
||||||
|
def mixins(self):
|
||||||
|
return self.get_as_list('mixes')
|
||||||
|
|
||||||
|
def to_dict(self):
|
||||||
|
d = super(ClassDoc, self).to_dict()
|
||||||
|
d['mixins'] = self.mixins
|
||||||
|
return d
|
||||||
|
|
||||||
|
DEFAULT = object()
|
||||||
|
class UnknownNS(NSDoc):
|
||||||
|
params = () # TODO: log warning when (somehow) trying to access / document an unknown object as ctor?
|
||||||
|
def get_property(self, name):
|
||||||
|
return super(UnknownNS, self).get_property(name) or \
|
||||||
|
UnknownNS({'name': '{}.{}'.format(self.name, name)})
|
||||||
|
|
||||||
|
def __getitem__(self, item):
|
||||||
|
if self._probably_not_property(item):
|
||||||
|
return super().__getitem__(item)
|
||||||
|
return self.get_property(item)
|
||||||
|
|
||||||
|
def _probably_not_property(self, item):
|
||||||
|
return (
|
||||||
|
not isinstance(item, str)
|
||||||
|
or item in (self.namekey, 'name', 'params')
|
||||||
|
or item.startswith(('_', 'guessed_'))
|
||||||
|
or item in self.parsed
|
||||||
|
)
|
||||||
|
|
||||||
|
class Unknown(CommentDoc):
|
||||||
|
@classmethod
|
||||||
|
def from_(cls, source):
|
||||||
|
def builder(parsed):
|
||||||
|
inst = cls(parsed)
|
||||||
|
inst.parsed['source'] = source
|
||||||
|
return inst
|
||||||
|
return builder
|
||||||
|
|
||||||
|
@property
|
||||||
|
def name(self):
|
||||||
|
return self['name'] + ' ' + self['source']
|
||||||
|
|
||||||
|
@property
|
||||||
|
def type(self):
|
||||||
|
return "Unknown"
|
||||||
|
|
||||||
|
def get_property(self, p):
|
||||||
|
return Unknown(dict(self.parsed, source=self.name, name=p + '<'))
|
||||||
|
|
||||||
|
def parse_comments(comments, doctype=None):
|
||||||
|
# find last comment which starts with a *
|
||||||
|
docstring = next((
|
||||||
|
c['value']
|
||||||
|
for c in reversed(comments or [])
|
||||||
|
if c['value'].startswith(u'*')
|
||||||
|
), None) or u""
|
||||||
|
|
||||||
|
# \n prefix necessary otherwise parse_comment fails to take first
|
||||||
|
# block comment parser strips delimiters, but strip_stars fails without
|
||||||
|
# them
|
||||||
|
extract = '\n' + strip_stars('/*' + docstring + '\n*/')
|
||||||
|
parsed = pyjsdoc.parse_comment(extract, u'')
|
||||||
|
|
||||||
|
if doctype == 'FunctionExpression':
|
||||||
|
doctype = FunctionDoc
|
||||||
|
elif doctype == 'ObjectExpression' or doctype is None:
|
||||||
|
doctype = guess
|
||||||
|
|
||||||
|
if doctype is guess:
|
||||||
|
return doctype(parsed)
|
||||||
|
|
||||||
|
# in case a specific doctype is given, allow overriding it anyway
|
||||||
|
return guess(parsed, default=doctype)
|
||||||
|
|
||||||
|
def guess(parsed, default=UnknownNS):
|
||||||
|
if 'class' in parsed:
|
||||||
|
return ClassDoc(parsed)
|
||||||
|
if 'function' in parsed:
|
||||||
|
return FunctionDoc(parsed)
|
||||||
|
if 'mixin' in parsed:
|
||||||
|
return MixinDoc(parsed)
|
||||||
|
if 'namespace' in parsed:
|
||||||
|
return NSDoc(parsed)
|
||||||
|
if 'module' in parsed:
|
||||||
|
return ModuleDoc(parsed)
|
||||||
|
if 'type' in parsed:
|
||||||
|
return PropertyDoc(parsed)
|
||||||
|
|
||||||
|
return default(parsed)
|
572
_extensions/autojsdoc/parser/parser.py
Normal file
572
_extensions/autojsdoc/parser/parser.py
Normal file
@ -0,0 +1,572 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
import collections
|
||||||
|
|
||||||
|
import pyjsdoc
|
||||||
|
|
||||||
|
from . import jsdoc
|
||||||
|
from . import utils
|
||||||
|
from .visitor import Visitor, SKIP
|
||||||
|
|
||||||
|
DECLARATOR_INIT_TO_REF = ('Literal', 'Identifier', 'MemberExpression')
|
||||||
|
|
||||||
|
|
||||||
|
class ModuleMatcher(Visitor):
|
||||||
|
"""Looks for structures of the form::
|
||||||
|
|
||||||
|
odoo.define($string, function ($name) {
|
||||||
|
|
||||||
|
These are *Odoo module definitions*, upon encountering one the
|
||||||
|
matcher:
|
||||||
|
|
||||||
|
* creates a module entry, optionally associated with the module comment
|
||||||
|
* spawns off a :class:`ModuleBodyMatcher` on the function's body with the
|
||||||
|
module name and the $name as "require" function
|
||||||
|
"""
|
||||||
|
def __init__(self, filename):
|
||||||
|
super(ModuleMatcher, self).__init__()
|
||||||
|
self.filename = filename
|
||||||
|
self.result = []
|
||||||
|
def enter_Program(self, node):
|
||||||
|
pass # allows visiting toplevel
|
||||||
|
def enter_ExpressionStatement(self, node):
|
||||||
|
# we're interested in expression statements (toplevel call)
|
||||||
|
if utils.match(node, {'expression': {
|
||||||
|
'callee': {
|
||||||
|
'object': {'name': 'odoo'},
|
||||||
|
'property': {'name': 'define'},
|
||||||
|
},
|
||||||
|
}}):
|
||||||
|
[module, *_, func] = node['expression']['arguments']
|
||||||
|
mod = jsdoc.parse_comments(node.get('comments'), jsdoc.ModuleDoc)
|
||||||
|
# set module name
|
||||||
|
mod.set_name(module['value'])
|
||||||
|
mod.parsed['sourcefile'] = self.filename
|
||||||
|
self.result.append(mod)
|
||||||
|
|
||||||
|
# get name of require parameter
|
||||||
|
require = None # a module can have no dependencies
|
||||||
|
if func['params']:
|
||||||
|
require = func['params'][0]['name']
|
||||||
|
mod.parsed['dependency'], post = ModuleExtractor(mod, require).visit(func['body'])
|
||||||
|
mod._post_process.extend(post)
|
||||||
|
# don't recurse since we've fired off a sub-visitor for the
|
||||||
|
# bits we're interested in
|
||||||
|
return SKIP
|
||||||
|
def enter_generic(self, node):
|
||||||
|
# skip all other toplevel statements
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
ref = collections.namedtuple('Ref', 'object property')
|
||||||
|
def _name(r):
|
||||||
|
bits = []
|
||||||
|
while isinstance(r, ref):
|
||||||
|
bits.append(str(r.property))
|
||||||
|
r = r.object
|
||||||
|
return '.'.join(reversed(bits))
|
||||||
|
def deref(item, prop=None):
|
||||||
|
assert isinstance(item, ref)
|
||||||
|
|
||||||
|
while isinstance(item, ref):
|
||||||
|
obj = item.object
|
||||||
|
if isinstance(obj, ref):
|
||||||
|
obj = deref(obj)
|
||||||
|
|
||||||
|
if isinstance(obj, (jsdoc.NSDoc, jsdoc.Unknown)):
|
||||||
|
item = obj.get_property(item.property)
|
||||||
|
elif isinstance(obj, dict):
|
||||||
|
item = obj[item.property]
|
||||||
|
elif isinstance(obj, jsdoc.PropertyDoc):
|
||||||
|
# f'n dynamic crap
|
||||||
|
item = jsdoc.Unknown(obj.to_dict()).get_property(item.property)
|
||||||
|
else:
|
||||||
|
raise ValueError("%r (%s) should be a dict or namespace" % (obj, type(obj)))
|
||||||
|
return item
|
||||||
|
|
||||||
|
def m2r(me, scope):
|
||||||
|
# create a ref to the scope in case it's a hoisted function declaration,
|
||||||
|
# this may false-positive on other hoisting but w/e
|
||||||
|
if me['type'] == 'Literal':
|
||||||
|
return jsdoc.LiteralDoc({'value': me['value']})
|
||||||
|
if me['type'] == 'Identifier':
|
||||||
|
return ref(scope, me['name'])
|
||||||
|
if me['type'] != 'MemberExpression':
|
||||||
|
raise ValueError(me)
|
||||||
|
return ref(
|
||||||
|
m2r(me['object'], scope),
|
||||||
|
utils._value(me['property'], strict=True)
|
||||||
|
)
|
||||||
|
|
||||||
|
NOTHING = object()
|
||||||
|
class Declaration(object):
|
||||||
|
__slots__ = ['id', 'comments']
|
||||||
|
def __init__(self, id=None, comments=NOTHING):
|
||||||
|
self.id = id
|
||||||
|
self.comments = [] if comments is NOTHING else comments
|
||||||
|
|
||||||
|
class ModuleContent(object):
|
||||||
|
__slots__ = ['dependencies', 'post']
|
||||||
|
def __init__(self, dependencies=NOTHING, post=NOTHING):
|
||||||
|
self.dependencies = set() if dependencies is NOTHING else dependencies
|
||||||
|
self.post = [] if post is NOTHING else post
|
||||||
|
def __iter__(self):
|
||||||
|
yield self.dependencies
|
||||||
|
yield self.post
|
||||||
|
|
||||||
|
class Nothing(object):
|
||||||
|
def __init__(self, name):
|
||||||
|
self.name = name
|
||||||
|
def __bool__(self):
|
||||||
|
return False
|
||||||
|
__nonzero__ = __bool__
|
||||||
|
|
||||||
|
class RefProxy(object):
|
||||||
|
def __init__(self, r):
|
||||||
|
self._ref = r
|
||||||
|
def become(self, modules):
|
||||||
|
s = self
|
||||||
|
other = deref(self._ref) or Nothing(_name(self._ref))
|
||||||
|
# ??? shouldn't all previous refs have been resolved?
|
||||||
|
if isinstance(other, (RefProxy, ModuleProxy)):
|
||||||
|
other = other.become(modules)
|
||||||
|
self.__class__ = other.__class__
|
||||||
|
self.__dict__ = other.__dict__
|
||||||
|
return self
|
||||||
|
def set_name(self, name):
|
||||||
|
pass # ???
|
||||||
|
class ModuleProxy(object):
|
||||||
|
def __init__(self, name):
|
||||||
|
self._name = name
|
||||||
|
|
||||||
|
# replace the ModuleProxy by the module's exports
|
||||||
|
def become(self, modules):
|
||||||
|
s = self
|
||||||
|
m = modules[self._name].get_property('<exports>') or Nothing(self._name)
|
||||||
|
self.__class__ = m.__class__
|
||||||
|
self.__dict__ = m.__dict__
|
||||||
|
return self
|
||||||
|
|
||||||
|
def set_name(self, name):
|
||||||
|
pass # FIXME: ???
|
||||||
|
|
||||||
|
jq = jsdoc.UnknownNS({
|
||||||
|
'name': u'jQuery',
|
||||||
|
'doc': u'<jQuery>',
|
||||||
|
})
|
||||||
|
window = jsdoc.UnknownNS({
|
||||||
|
'doc': '<window>',
|
||||||
|
'name': 'window',
|
||||||
|
})
|
||||||
|
|
||||||
|
class BaseScope(collections.defaultdict):
|
||||||
|
""" The base scope assumes anything it's asked for is just an unknown
|
||||||
|
(global) namespace of some sort. Can hold a bunch of predefined params but
|
||||||
|
avoids the variables inference system blowing up when new (browser)
|
||||||
|
globals get used in module bodies.
|
||||||
|
"""
|
||||||
|
def __missing__(self, key):
|
||||||
|
it = jsdoc.UnknownNS({
|
||||||
|
'name': key,
|
||||||
|
'doc': u'<%s>' % key,
|
||||||
|
})
|
||||||
|
self[key] = it
|
||||||
|
return it
|
||||||
|
BASE_SCOPE = BaseScope(None, {
|
||||||
|
'_': jsdoc.UnknownNS({'doc': u'<underscore.js>', 'name': u'_'}),
|
||||||
|
'$': jq, 'jQuery': jq,
|
||||||
|
'window': window,
|
||||||
|
'document': window.get_property('document'),
|
||||||
|
'Date': jsdoc.ClassDoc({
|
||||||
|
'name': u'Date',
|
||||||
|
'doc': u'',
|
||||||
|
}),
|
||||||
|
'Backbone': jsdoc.UnknownNS({
|
||||||
|
'_members': [
|
||||||
|
('Model', jsdoc.ClassDoc({
|
||||||
|
'name': u'Model',
|
||||||
|
'doc': u'',
|
||||||
|
})),
|
||||||
|
('Collection', jsdoc.ClassDoc({
|
||||||
|
'name': u'Collection',
|
||||||
|
'doc': u'',
|
||||||
|
})),
|
||||||
|
]
|
||||||
|
}),
|
||||||
|
'odoo': jsdoc.UnknownNS({
|
||||||
|
'name': u'odoo',
|
||||||
|
'doc': u"Odoo",
|
||||||
|
'_members': [
|
||||||
|
('name', jsdoc.PropertyDoc({'name': u'csrf_token', 'type': u'{String}'})),
|
||||||
|
]
|
||||||
|
}),
|
||||||
|
'undefined': jsdoc.LiteralDoc({'name': u'undefined', 'value': None}),
|
||||||
|
})
|
||||||
|
|
||||||
|
class Scope(object):
|
||||||
|
"""
|
||||||
|
Add hoc scope versioning/SSA such that rebinding a symbol in a module
|
||||||
|
scope does not screw everything up e.g. "Foo = Foo.extend({})" should not
|
||||||
|
have the final Foo extending itself...
|
||||||
|
"""
|
||||||
|
def __init__(self, mapping):
|
||||||
|
self._namemap = self._empty(mapping)
|
||||||
|
self._targets = []
|
||||||
|
for k, v in mapping.items():
|
||||||
|
self[k] = v
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _empty(mapping):
|
||||||
|
m = mapping.copy()
|
||||||
|
m.clear()
|
||||||
|
return m
|
||||||
|
|
||||||
|
def __setitem__(self, k, v):
|
||||||
|
self._namemap[k] = len(self._targets)
|
||||||
|
self._targets.append(v)
|
||||||
|
|
||||||
|
def freeze(self):
|
||||||
|
d = self._empty(self._namemap)
|
||||||
|
for k, v in self._namemap.items():
|
||||||
|
d[k] = self._targets[v]
|
||||||
|
return d
|
||||||
|
|
||||||
|
class ModuleExtractor(Visitor):
|
||||||
|
def __init__(self, module, requirefunc):
|
||||||
|
super(ModuleExtractor, self).__init__()
|
||||||
|
self.module = module
|
||||||
|
self.requirefunc = requirefunc
|
||||||
|
self.result = ModuleContent()
|
||||||
|
self.scope = Scope(BASE_SCOPE)
|
||||||
|
self.declaration = None
|
||||||
|
|
||||||
|
def enter_BlockStatement(self, node):
|
||||||
|
Hoistifier(self).visit(node)
|
||||||
|
def exit_BlockStatement(self, node):
|
||||||
|
for k, v in self.scope._namemap.items():
|
||||||
|
if k not in BASE_SCOPE:
|
||||||
|
self.module.add_member(k, self.scope._targets[v])
|
||||||
|
for t in TypedefMatcher(self).visit(node):
|
||||||
|
self.module.add_member(t.name, t)
|
||||||
|
|
||||||
|
def enter_VariableDeclaration(self, node):
|
||||||
|
self.declaration = Declaration(comments=node.get('comments'))
|
||||||
|
def enter_VariableDeclarator(self, node):
|
||||||
|
# otherwise we've already hoisted the declaration so the variable
|
||||||
|
# already exist initialised to undefined, and ValueExtractor does not
|
||||||
|
# handle a None input node so it returns None which makes no sense
|
||||||
|
if node['init']:
|
||||||
|
self.declaration.id = node['id']['name']
|
||||||
|
self.scope[self.declaration.id] = ValueExtractor(
|
||||||
|
self, self.declaration
|
||||||
|
).visit(node['init'] or [])
|
||||||
|
self.declaration.id = None
|
||||||
|
return SKIP
|
||||||
|
def exit_VariableDeclaration(self, node):
|
||||||
|
self.declaration = None
|
||||||
|
|
||||||
|
# as the name denotes, AssignmentExpression is an *expression*, which
|
||||||
|
# means at the module toplevel it is wrapped into an ExpressionStatement
|
||||||
|
# which is where the comments get attached
|
||||||
|
def enter_ExpressionStatement(self, node):
|
||||||
|
self.declaration = Declaration(comments=node.get('comments'))
|
||||||
|
def exit_ExpressionStatement(self, node):
|
||||||
|
self.declaration = None
|
||||||
|
def enter_AssignmentExpression(self, node):
|
||||||
|
target = node['left']
|
||||||
|
if target['type'] == 'Identifier':
|
||||||
|
self.declaration.id = target['name']
|
||||||
|
self.scope[self.declaration.id] = ValueExtractor(
|
||||||
|
self, self.declaration
|
||||||
|
).visit(node['right'])
|
||||||
|
self.declaration.id = None
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
if target['type'] != 'MemberExpression':
|
||||||
|
raise ValueError("Unhandled assign to %s" % target['type'])
|
||||||
|
|
||||||
|
# only assign to straight a.b.c patterns (OK and trivial literals)
|
||||||
|
if target['computed'] and target['property']['type'] != 'Literal':
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
name = utils._value(target['property'], strict=True)
|
||||||
|
if isinstance(name, type(u'')) and name.endswith('.extend'):
|
||||||
|
return SKIP # ignore overwrite of .extend (WTF)
|
||||||
|
self.declaration.id = name
|
||||||
|
it = ValueExtractor(
|
||||||
|
self, self.declaration
|
||||||
|
).visit(node['right'])
|
||||||
|
self.declaration.id = None
|
||||||
|
assert it, "assigned a non-value from %s to %s" % (node['right'], name)
|
||||||
|
|
||||||
|
@self.result.post.append
|
||||||
|
def _augment_module(modules):
|
||||||
|
try:
|
||||||
|
t = deref(m2r(target['object'], self.scope.freeze()))
|
||||||
|
except ValueError:
|
||||||
|
return # f'n extension of global libraries garbage
|
||||||
|
if not isinstance(t, jsdoc.NSDoc):
|
||||||
|
# function Foo(){}; Foo.prototype = bar
|
||||||
|
# fuck that yo
|
||||||
|
return
|
||||||
|
# TODO: note which module added this
|
||||||
|
m = it
|
||||||
|
if isinstance(m, jsdoc.LiteralDoc):
|
||||||
|
m = jsdoc.PropertyDoc(m.to_dict())
|
||||||
|
t.add_member(name, m)
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
def enter_FunctionDeclaration(self, node):
|
||||||
|
""" Already processed by hoistitifier
|
||||||
|
"""
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
def enter_ReturnStatement(self, node):
|
||||||
|
self.declaration = Declaration(comments=node.get('comments'))
|
||||||
|
if node['argument']:
|
||||||
|
export = ValueExtractor(self, self.declaration).visit(node['argument'])
|
||||||
|
if isinstance(export, RefProxy):
|
||||||
|
self.module.parsed['exports'] = _name(export._ref)
|
||||||
|
self.scope['<exports>'] = export
|
||||||
|
self.declaration = None
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
def enter_CallExpression(self, node):
|
||||||
|
if utils.match(node, {
|
||||||
|
'callee': {
|
||||||
|
'type': 'MemberExpression',
|
||||||
|
'object': lambda n: (
|
||||||
|
n['type'] in ('Identifier', 'MemberExpression')
|
||||||
|
# _.str.include
|
||||||
|
and not utils._name(n).startswith('_')
|
||||||
|
),
|
||||||
|
'property': {'name': 'include'},
|
||||||
|
}
|
||||||
|
}):
|
||||||
|
target = RefProxy(m2r(node['callee']['object'], self.scope.freeze()))
|
||||||
|
target_name = utils._name(node['callee']['object'])
|
||||||
|
items = ClassProcessor(self).visit(node['arguments'])
|
||||||
|
@self.result.post.append
|
||||||
|
def resolve_extension(modules):
|
||||||
|
t = target.become(modules)
|
||||||
|
if not isinstance(t, jsdoc.ClassDoc):
|
||||||
|
return # FIXME: log warning
|
||||||
|
# raise ValueError("include() subjects should be classes, %s is %s" % (target_name, type(t)))
|
||||||
|
# TODO: note which module added these
|
||||||
|
for it in items:
|
||||||
|
if isinstance(it, dict):
|
||||||
|
for n, member in it.items():
|
||||||
|
t.add_member(n, member)
|
||||||
|
else:
|
||||||
|
t.parsed.setdefault('mixes', []).append(it.become(modules))
|
||||||
|
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
def refify(self, node, also=None):
|
||||||
|
it = m2r(node, self.scope.freeze())
|
||||||
|
assert isinstance(it, ref), "Expected ref, got {}".format(it)
|
||||||
|
px = RefProxy(it)
|
||||||
|
@self.result.post.append
|
||||||
|
def resolve(modules):
|
||||||
|
p = px.become(modules)
|
||||||
|
if also: also(p)
|
||||||
|
return px
|
||||||
|
|
||||||
|
class ValueExtractor(Visitor):
|
||||||
|
def __init__(self, parent, declaration=None):
|
||||||
|
super(ValueExtractor, self).__init__()
|
||||||
|
self.parent = parent
|
||||||
|
self.declaration = declaration or Declaration()
|
||||||
|
|
||||||
|
def enter_generic(self, node):
|
||||||
|
self.result = jsdoc.parse_comments(
|
||||||
|
self.declaration.comments,
|
||||||
|
jsdoc.Unknown.from_(node['type'])
|
||||||
|
)
|
||||||
|
self._update_result_meta()
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
def _update_result_meta(self, name=None):
|
||||||
|
self.result.parsed['sourcemodule'] = self.parent.module
|
||||||
|
n = name or self.declaration.id
|
||||||
|
if n:
|
||||||
|
self.result.set_name(n)
|
||||||
|
|
||||||
|
def enter_Literal(self, node):
|
||||||
|
self.result = jsdoc.parse_comments(
|
||||||
|
self.declaration.comments, jsdoc.LiteralDoc)
|
||||||
|
self._update_result_meta()
|
||||||
|
self.result.parsed['value'] = node['value']
|
||||||
|
return SKIP
|
||||||
|
def enter_Identifier(self, node):
|
||||||
|
self.result = self.parent.refify(node)
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
def enter_MemberExpression(self, node):
|
||||||
|
self.result = RefProxy(ref(
|
||||||
|
ValueExtractor(self.parent).visit(node['object']),
|
||||||
|
utils._value(node['property'], strict=True)
|
||||||
|
))
|
||||||
|
self.parent.result.post.append(self.result.become)
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
def enter_FunctionExpression(self, node):
|
||||||
|
name, comments = (self.declaration.id, self.declaration.comments)
|
||||||
|
self.result = jsdoc.parse_comments(comments, jsdoc.FunctionDoc)
|
||||||
|
self.result.parsed['name'] = node['id'] and node['id']['name']
|
||||||
|
self._update_result_meta()
|
||||||
|
self.result.parsed['guessed_params'] = [p['name'] for p in node['params']]
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
def enter_NewExpression(self, node):
|
||||||
|
comments = self.declaration.comments if self.declaration else node.get('comments')
|
||||||
|
self.result = ns = jsdoc.parse_comments(comments, jsdoc.InstanceDoc)
|
||||||
|
self._update_result_meta()
|
||||||
|
|
||||||
|
def _update_contents(cls):
|
||||||
|
if not isinstance(cls, jsdoc.ClassDoc):
|
||||||
|
return
|
||||||
|
ns.parsed['cls'] = cls
|
||||||
|
self.parent.refify(node['callee'], also=_update_contents)
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
def enter_ObjectExpression(self, node):
|
||||||
|
self.result = obj = jsdoc.parse_comments(self.declaration.comments)
|
||||||
|
self._update_result_meta()
|
||||||
|
for n, p in MemberExtractor(parent=self.parent).visit(node['properties']).items():
|
||||||
|
obj.add_member(n, p)
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
def enter_CallExpression(self, node):
|
||||||
|
# require(a_module_name)
|
||||||
|
if utils.match(node, {'callee': {'type': 'Identifier', 'name': self.parent.requirefunc}}):
|
||||||
|
depname = node['arguments'][0]['value']
|
||||||
|
# TODO: clean this up
|
||||||
|
self.parent.result.dependencies.add(depname)
|
||||||
|
self.result = ModuleProxy(name=depname)
|
||||||
|
self.parent.result.post.append(self.result.become)
|
||||||
|
|
||||||
|
# Class.extend(..mixins, {})
|
||||||
|
elif utils.match(node, {
|
||||||
|
'callee': {
|
||||||
|
'type': 'MemberExpression',
|
||||||
|
'object': lambda n: (
|
||||||
|
n['type'] in ('Identifier', 'MemberExpression')
|
||||||
|
# $.extend, $.fn.extend, _.extend
|
||||||
|
and not utils._name(n).startswith(('_', '$'))
|
||||||
|
),
|
||||||
|
'property': {'name': 'extend'},
|
||||||
|
},
|
||||||
|
}): # creates a new class, but may not actually return it
|
||||||
|
obj = node['callee']['object']
|
||||||
|
comments = self.declaration.comments
|
||||||
|
self.result = cls = jsdoc.parse_comments(comments, jsdoc.ClassDoc)
|
||||||
|
cls.parsed['extends'] = self.parent.refify(obj)
|
||||||
|
self._update_result_meta()
|
||||||
|
items = ClassProcessor(self.parent).visit(node['arguments'])
|
||||||
|
|
||||||
|
@self.parent.result.post.append
|
||||||
|
def add_to_class(modules):
|
||||||
|
for item in items:
|
||||||
|
if isinstance(item, dict):
|
||||||
|
# methods/attributes
|
||||||
|
for n, method in item.items():
|
||||||
|
cls.add_member(n, method)
|
||||||
|
else:
|
||||||
|
cls.parsed.setdefault('mixes', []).append(item)
|
||||||
|
|
||||||
|
# other function calls
|
||||||
|
else:
|
||||||
|
self.result = jsdoc.parse_comments(self.declaration.comments, jsdoc.guess)
|
||||||
|
self._update_result_meta()
|
||||||
|
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
class ClassProcessor(Visitor):
|
||||||
|
def __init__(self, parent):
|
||||||
|
super(ClassProcessor, self).__init__()
|
||||||
|
self.result = []
|
||||||
|
self.parent = parent
|
||||||
|
|
||||||
|
def enter_generic(self, node):
|
||||||
|
self.result.append(self.parent.refify(node))
|
||||||
|
return SKIP
|
||||||
|
def enter_ObjectExpression(self, node):
|
||||||
|
self.result.append(MemberExtractor(parent=self.parent, for_class=True).visit(node['properties']))
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
String = type(u'')
|
||||||
|
class MemberExtractor(Visitor):
|
||||||
|
def __init__(self, parent, for_class=False):
|
||||||
|
super(MemberExtractor, self).__init__()
|
||||||
|
self.result = collections.OrderedDict()
|
||||||
|
self.parent = parent
|
||||||
|
self.for_class = for_class
|
||||||
|
|
||||||
|
def enter_Property(self, node):
|
||||||
|
name = utils._value(node['key'])
|
||||||
|
prop = ValueExtractor(
|
||||||
|
self.parent,
|
||||||
|
Declaration(id=name, comments=node.get('comments'))
|
||||||
|
).visit(node['value'])
|
||||||
|
|
||||||
|
if isinstance(prop, jsdoc.LiteralDoc):
|
||||||
|
prop = jsdoc.PropertyDoc(prop.to_dict())
|
||||||
|
|
||||||
|
# ValueExtractor can return a Ref, maybe this should be sent as part
|
||||||
|
# of the decl/comments?
|
||||||
|
if name.startswith('_') and hasattr(prop, 'parsed'):
|
||||||
|
prop.parsed['private'] = True
|
||||||
|
self.result[name] = prop
|
||||||
|
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
class Hoistifier(Visitor):
|
||||||
|
"""
|
||||||
|
Processor for variable and function declarations properly hoisting them
|
||||||
|
to the "top" of a module such that they are available with the relevant
|
||||||
|
value afterwards.
|
||||||
|
"""
|
||||||
|
def __init__(self, parent):
|
||||||
|
super(Hoistifier, self).__init__()
|
||||||
|
self.parent = parent
|
||||||
|
|
||||||
|
def enter_generic(self, node):
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
# nodes to straight recurse into, others are just skipped
|
||||||
|
enter_BlockStatement = enter_VariableDeclaration = lambda self, node: None
|
||||||
|
|
||||||
|
def enter_VariableDeclarator(self, node):
|
||||||
|
self.parent.scope[node['id']['name']] = BASE_SCOPE['undefined']
|
||||||
|
|
||||||
|
def enter_FunctionDeclaration(self, node):
|
||||||
|
funcname = node['id']['name']
|
||||||
|
self.parent.scope[funcname] = fn = jsdoc.parse_comments(
|
||||||
|
node.get('comments'),
|
||||||
|
jsdoc.FunctionDoc,
|
||||||
|
)
|
||||||
|
fn.parsed['sourcemodule'] = self.parent.module
|
||||||
|
fn.parsed['name'] = funcname
|
||||||
|
fn.parsed['guessed_params'] = [p['name'] for p in node['params']]
|
||||||
|
return SKIP
|
||||||
|
|
||||||
|
class TypedefMatcher(Visitor):
|
||||||
|
def __init__(self, parent):
|
||||||
|
super(TypedefMatcher, self).__init__()
|
||||||
|
self.parent = parent
|
||||||
|
self.result = []
|
||||||
|
|
||||||
|
enter_BlockStatement = lambda self, node: None
|
||||||
|
def enter_generic(self, node):
|
||||||
|
# just traverse all top-level statements, check their comments, and
|
||||||
|
# bail
|
||||||
|
for comment in node.get('comments') or []:
|
||||||
|
if '@typedef' in comment['value']:
|
||||||
|
extract = '\n' + jsdoc.strip_stars('/*' + comment['value'] + '\n*/')
|
||||||
|
parsed = pyjsdoc.parse_comment(extract, u'')
|
||||||
|
p = jsdoc.ParamDoc(parsed['typedef'])
|
||||||
|
parsed['name'] = p.name
|
||||||
|
parsed['sourcemodule'] = self.parent.module
|
||||||
|
# TODO: add p.type as superclass somehow? Builtin types not in scope :(
|
||||||
|
self.result.append(jsdoc.ClassDoc(parsed))
|
||||||
|
|
||||||
|
return SKIP
|
6
_extensions/autojsdoc/parser/tests/README.rst
Normal file
6
_extensions/autojsdoc/parser/tests/README.rst
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
:orphan:
|
||||||
|
|
||||||
|
These files should be run via pytest_, simply install pytest and from the top
|
||||||
|
of the Odoo project run ``pytest doc/_extensions``.
|
||||||
|
|
||||||
|
.. _pytest: https://pytest.org/
|
73
_extensions/autojsdoc/parser/tests/support.py
Normal file
73
_extensions/autojsdoc/parser/tests/support.py
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
import operator
|
||||||
|
|
||||||
|
import pyjsparser
|
||||||
|
|
||||||
|
from autojsdoc.parser import jsdoc, parser
|
||||||
|
|
||||||
|
params = operator.attrgetter('name', 'type', 'doc')
|
||||||
|
|
||||||
|
|
||||||
|
def parse(s, source=None):
|
||||||
|
tree = pyjsparser.parse(s)
|
||||||
|
mods = parser.ModuleMatcher(source).visit(tree)
|
||||||
|
post(mods)
|
||||||
|
return mods
|
||||||
|
|
||||||
|
def post(mods):
|
||||||
|
modules = dict(BASE_MODULES)
|
||||||
|
modules.update((m.name, m) for m in mods)
|
||||||
|
|
||||||
|
for mod in mods:
|
||||||
|
mod.post_process(modules)
|
||||||
|
|
||||||
|
BASE_MODULES = {
|
||||||
|
'other': jsdoc.ModuleDoc({
|
||||||
|
'module': 'other',
|
||||||
|
'_members': [
|
||||||
|
('<exports>', jsdoc.LiteralDoc({'name': 'value', 'value': "ok"})),
|
||||||
|
],
|
||||||
|
}),
|
||||||
|
'dep2': jsdoc.ModuleDoc({
|
||||||
|
'module': 'dep2',
|
||||||
|
'_members': [
|
||||||
|
('<exports>', jsdoc.LiteralDoc({'value': 42.})),
|
||||||
|
],
|
||||||
|
}),
|
||||||
|
'dep3': jsdoc.ModuleDoc({
|
||||||
|
'module': 'dep3',
|
||||||
|
'_members': [
|
||||||
|
('<exports>', jsdoc.LiteralDoc({'value': 56.})),
|
||||||
|
],
|
||||||
|
}),
|
||||||
|
'Class': jsdoc.ModuleDoc({
|
||||||
|
'module': 'Class',
|
||||||
|
'_members': [
|
||||||
|
('<exports>', jsdoc.ClassDoc({
|
||||||
|
'name': 'Class',
|
||||||
|
'doc': "Base Class"
|
||||||
|
})),
|
||||||
|
],
|
||||||
|
}),
|
||||||
|
'mixins': jsdoc.ModuleDoc({
|
||||||
|
'module': 'mixins',
|
||||||
|
'_members': [
|
||||||
|
('<exports>', jsdoc.NSDoc({
|
||||||
|
'name': 'mixins',
|
||||||
|
'_members': [
|
||||||
|
('Bob', jsdoc.ClassDoc({'class': "Bob"})),
|
||||||
|
]
|
||||||
|
})),
|
||||||
|
],
|
||||||
|
}),
|
||||||
|
'Mixin': jsdoc.ModuleDoc({
|
||||||
|
'module': 'Mixin',
|
||||||
|
'_members': [
|
||||||
|
('<exports>', jsdoc.MixinDoc({
|
||||||
|
'_members': [
|
||||||
|
('a', jsdoc.FunctionDoc({'function': 'a'})),
|
||||||
|
]
|
||||||
|
})),
|
||||||
|
],
|
||||||
|
})
|
||||||
|
}
|
227
_extensions/autojsdoc/parser/tests/test_class.py
Normal file
227
_extensions/autojsdoc/parser/tests/test_class.py
Normal file
@ -0,0 +1,227 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
from autojsdoc.parser import jsdoc
|
||||||
|
from support import params, parse
|
||||||
|
|
||||||
|
def test_classvar():
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('a.A', function(require) {
|
||||||
|
var Class = require('Class');
|
||||||
|
/**
|
||||||
|
* This is my class-kai
|
||||||
|
*/
|
||||||
|
var A = Class.extend({});
|
||||||
|
return A;
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
cls = mod.exports
|
||||||
|
assert type(cls) == jsdoc.ClassDoc
|
||||||
|
|
||||||
|
assert type(cls.superclass) == jsdoc.ClassDoc
|
||||||
|
assert cls.superclass.name == 'Class'
|
||||||
|
|
||||||
|
assert cls.name == 'A'
|
||||||
|
assert cls.constructor is None
|
||||||
|
assert cls.properties == []
|
||||||
|
assert cls['doc'] == 'This is my class-kai'
|
||||||
|
|
||||||
|
def test_classret():
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('a.A', function(require) {
|
||||||
|
var Class = require('Class');
|
||||||
|
/**
|
||||||
|
* This is my class-kai
|
||||||
|
*/
|
||||||
|
return Class.extend({});
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
cls = mod.exports
|
||||||
|
assert type(cls) == jsdoc.ClassDoc
|
||||||
|
|
||||||
|
assert cls.name == ''
|
||||||
|
assert cls.constructor is None
|
||||||
|
assert cls.properties == []
|
||||||
|
assert cls['doc'] == 'This is my class-kai'
|
||||||
|
|
||||||
|
def test_methods():
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('a.A', function(require) {
|
||||||
|
var Class = require('Class');
|
||||||
|
return Class.extend({
|
||||||
|
/**
|
||||||
|
* @param {Widget} parent
|
||||||
|
*/
|
||||||
|
init: function (parent) {},
|
||||||
|
/**
|
||||||
|
* @returns {Widget}
|
||||||
|
*/
|
||||||
|
itself: function () { return this; },
|
||||||
|
/**
|
||||||
|
* @param {MouseEvent} e
|
||||||
|
*/
|
||||||
|
_onValidate: function (e) {},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
cls = mod.exports
|
||||||
|
assert len(cls.properties) == 3
|
||||||
|
assert cls.constructor
|
||||||
|
# assume methods are in source order
|
||||||
|
[_, init] = cls.properties[0]
|
||||||
|
assert init == cls.constructor
|
||||||
|
assert init.name == 'init'
|
||||||
|
assert not init.is_private
|
||||||
|
assert init.is_constructor
|
||||||
|
[param] = init.params
|
||||||
|
assert params(param) == ('parent', 'Widget', '')
|
||||||
|
|
||||||
|
[_, itself] = cls.properties[1]
|
||||||
|
assert itself.name == 'itself'
|
||||||
|
assert not itself.is_private
|
||||||
|
assert not itself.is_constructor
|
||||||
|
assert not itself.params
|
||||||
|
assert params(itself.return_val) == ('', 'Widget', '')
|
||||||
|
|
||||||
|
[_, _on] = cls.properties[2]
|
||||||
|
assert _on.name == '_onValidate'
|
||||||
|
assert _on.is_private
|
||||||
|
assert not _on.is_constructor
|
||||||
|
[param] = _on.params
|
||||||
|
assert params(param) == ('e', 'MouseEvent', '')
|
||||||
|
|
||||||
|
def test_mixin_explicit():
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('a.A', function (require) {
|
||||||
|
var Class = require('Class');
|
||||||
|
var mixins = require('mixins');
|
||||||
|
/**
|
||||||
|
* This is my class-kai
|
||||||
|
* @mixes mixins.Bob
|
||||||
|
*/
|
||||||
|
return Class.extend({});
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
cls = mod.exports
|
||||||
|
# FIXME: ClassDoc may want to m2r(mixin, scope)?
|
||||||
|
assert cls.mixins == ['mixins.Bob']
|
||||||
|
|
||||||
|
def test_mixin_implicit():
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('a.A', function(require) {
|
||||||
|
var Class = require('Class');
|
||||||
|
var Mixin = require('Mixin');
|
||||||
|
/**
|
||||||
|
* This is my class-kai
|
||||||
|
*/
|
||||||
|
return Class.extend(Mixin, { foo: function() {} });
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
cls = mod.exports
|
||||||
|
[mixin] = cls.mixins
|
||||||
|
assert type(mixin) == jsdoc.MixinDoc
|
||||||
|
assert params(mixin.properties[0][1]) == ('a', 'Function', '')
|
||||||
|
assert params(mixin.get_property('a')) == ('a', 'Function', '')
|
||||||
|
|
||||||
|
assert params(cls.get_property('foo')) == ('foo', 'Function', '')
|
||||||
|
|
||||||
|
def test_instanciation():
|
||||||
|
[A, a] = parse("""
|
||||||
|
odoo.define('A', function (r) {
|
||||||
|
var Class = r('Class');
|
||||||
|
/**
|
||||||
|
* @class A
|
||||||
|
*/
|
||||||
|
return Class.extend({
|
||||||
|
foo: function () {}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
odoo.define('a', function (r) {
|
||||||
|
var A = r('A');
|
||||||
|
var a = new A;
|
||||||
|
return a;
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
assert type(a.exports) == jsdoc.InstanceDoc
|
||||||
|
assert a.exports.cls.name == A.exports.name
|
||||||
|
|
||||||
|
def test_non_function_properties():
|
||||||
|
[A] = parse("""
|
||||||
|
odoo.define('A', function (r) {
|
||||||
|
var Class = r('Class');
|
||||||
|
return Class.extend({
|
||||||
|
template: 'thing',
|
||||||
|
a_prop: [1, 2, 3],
|
||||||
|
'other': {a: 7}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
t = A.exports.get_property('template')
|
||||||
|
assert type(t) == jsdoc.PropertyDoc
|
||||||
|
assert params(t) == ('template', 'String', '')
|
||||||
|
assert not t.is_private
|
||||||
|
|
||||||
|
def test_non_extend_classes():
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('A', function () {
|
||||||
|
/**
|
||||||
|
* @class
|
||||||
|
*/
|
||||||
|
var Class = function () {}
|
||||||
|
return Class;
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
assert type(mod.exports) == jsdoc.ClassDoc
|
||||||
|
|
||||||
|
def test_extend():
|
||||||
|
[a, _] = parse("""
|
||||||
|
odoo.define('A', function (require) {
|
||||||
|
var Class = require('Class');
|
||||||
|
return Class.extend({});
|
||||||
|
});
|
||||||
|
odoo.define('B', function (require) {
|
||||||
|
var A = require('A');
|
||||||
|
A.include({
|
||||||
|
/** A property */
|
||||||
|
a: 3,
|
||||||
|
/** A method */
|
||||||
|
b: function () {}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
cls = a.exports
|
||||||
|
assert type(cls) == jsdoc.ClassDoc
|
||||||
|
a = cls.get_property('a')
|
||||||
|
assert type(a) == jsdoc.PropertyDoc
|
||||||
|
assert params(a) == ('a', 'Number', 'A property')
|
||||||
|
b = cls.get_property('b')
|
||||||
|
assert type(b) == jsdoc.FunctionDoc
|
||||||
|
assert params(b) == ('b', 'Function', 'A method')
|
||||||
|
|
||||||
|
# TODO: also support virtual members?
|
||||||
|
# TODO: computed properties?
|
||||||
|
@pytest.mark.skip(reason="Need to implement member/var-parsing?")
|
||||||
|
def test_members():
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('A', function (r) {
|
||||||
|
var Class = r('Class');
|
||||||
|
return Class.extend({
|
||||||
|
init: function () {
|
||||||
|
/**
|
||||||
|
* This is bob
|
||||||
|
* @var {Foo}
|
||||||
|
*/
|
||||||
|
this.foo = 3;
|
||||||
|
this.bar = 42;
|
||||||
|
/**
|
||||||
|
* @member {Baz}
|
||||||
|
*/
|
||||||
|
this.baz = null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
cls = mod.exports
|
||||||
|
assert params(cls.members[0]) == ('foo', 'Foo', 'This is bob')
|
||||||
|
assert params(cls.members[1]) == ('bar', '', '')
|
||||||
|
assert params(cls.members[2]) == ('baz', 'Baz', '')
|
75
_extensions/autojsdoc/parser/tests/test_crap.py
Normal file
75
_extensions/autojsdoc/parser/tests/test_crap.py
Normal file
@ -0,0 +1,75 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
"""
|
||||||
|
Test various crap patterns found in Odoo code to ensure they don't blow up
|
||||||
|
the parser thingie
|
||||||
|
"""
|
||||||
|
from autojsdoc.parser import jsdoc
|
||||||
|
from support import parse
|
||||||
|
|
||||||
|
def test_export_external():
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('module', function () {
|
||||||
|
return $.Deferred().reject();
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
assert isinstance(mod.exports, jsdoc.CommentDoc)
|
||||||
|
assert mod.exports.doc == ''
|
||||||
|
|
||||||
|
def test_extend_jq():
|
||||||
|
parse("""
|
||||||
|
odoo.define('a', function (r) {
|
||||||
|
$.extend($.expr[':'], { a: function () {} });
|
||||||
|
$.fn.extend({ a: function () {} });
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
|
||||||
|
def test_extend_dynamic():
|
||||||
|
parse("""
|
||||||
|
odoo.define('a', function () {
|
||||||
|
foo.bar.baz[qux + '_external'] = function () {};
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
|
||||||
|
def test_extend_deep():
|
||||||
|
parse("""
|
||||||
|
odoo.define('a', function () {
|
||||||
|
var eventHandler = $.summernote.eventHandler;
|
||||||
|
var dom = $.summernote.core.dom;
|
||||||
|
dom.thing = function () {};
|
||||||
|
|
||||||
|
var fn_editor_currentstyle = eventHandler.modules.editor.currentStyle;
|
||||||
|
eventHandler.modules.editor.currentStyle = function () {}
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
|
||||||
|
def test_arbitrary():
|
||||||
|
parse("""
|
||||||
|
odoo.define('bob', function () {
|
||||||
|
var page = window.location.href.replace(/^.*\/\/[^\/]+/, '');
|
||||||
|
var mailWidgets = ['mail_followers', 'mail_thread', 'mail_activity', 'kanban_activity'];
|
||||||
|
var bob;
|
||||||
|
var fldj = foo.getTemplate().baz;
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
|
||||||
|
def test_prototype():
|
||||||
|
[A, B] = parse("""
|
||||||
|
odoo.define('mod1', function () {
|
||||||
|
var exports = {};
|
||||||
|
exports.Foo = Backbone.Model.extend({});
|
||||||
|
exports.Bar = Backbone.Model.extend({});
|
||||||
|
var BarCollection = Backbone.Collection.extend({
|
||||||
|
model: exports.Bar,
|
||||||
|
});
|
||||||
|
exports.Baz = Backbone.Model.extend({});
|
||||||
|
return exports;
|
||||||
|
});
|
||||||
|
odoo.define('mod2', function (require) {
|
||||||
|
var models = require('mod1');
|
||||||
|
var _super_orderline = models.Bar.prototype;
|
||||||
|
models.Foo = models.Bar.extend({});
|
||||||
|
var _super_order = models.Baz.prototype;
|
||||||
|
models.Bar = models.Baz.extend({});
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
|
173
_extensions/autojsdoc/parser/tests/test_module.py
Normal file
173
_extensions/autojsdoc/parser/tests/test_module.py
Normal file
@ -0,0 +1,173 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
from autojsdoc.parser import jsdoc
|
||||||
|
from support import params, parse, BASE_MODULES
|
||||||
|
|
||||||
|
|
||||||
|
def test_single():
|
||||||
|
[mod] = parse("""
|
||||||
|
/**
|
||||||
|
* This is a super module!
|
||||||
|
*/
|
||||||
|
odoo.define('supermodule', function (req) {
|
||||||
|
var other = req('other');
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
assert mod.name == 'supermodule'
|
||||||
|
assert mod.dependencies == {'other'}
|
||||||
|
assert mod.exports is None
|
||||||
|
assert mod.doc == "This is a super module!"
|
||||||
|
m = mod.get_property('other')
|
||||||
|
assert m.name == 'value', "property is exported value not imported module"
|
||||||
|
|
||||||
|
def test_multiple():
|
||||||
|
[mod1, mod2, mod3] = parse("""
|
||||||
|
odoo.define('module1', function (req) {
|
||||||
|
return 1;
|
||||||
|
});
|
||||||
|
odoo.define('module2', function (req) {
|
||||||
|
return req('dep2');
|
||||||
|
});
|
||||||
|
odoo.define('module3', function (req) {
|
||||||
|
var r = req('dep3');
|
||||||
|
return r;
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
assert mod1.name == 'module1'
|
||||||
|
assert mod1.dependencies == set()
|
||||||
|
assert isinstance(mod1.exports, jsdoc.LiteralDoc)
|
||||||
|
assert mod1.exports.value == 1.0
|
||||||
|
assert mod1.exports['sourcemodule'] is mod1
|
||||||
|
assert mod1.doc == ""
|
||||||
|
|
||||||
|
assert mod2.name == 'module2'
|
||||||
|
assert mod2.dependencies == {'dep2'}
|
||||||
|
assert isinstance(mod2.exports, jsdoc.LiteralDoc)
|
||||||
|
assert mod2.exports.value == 42.0
|
||||||
|
assert mod2.doc == ""
|
||||||
|
|
||||||
|
assert mod3.name == 'module3'
|
||||||
|
assert mod3.dependencies == {'dep3'}
|
||||||
|
assert isinstance(mod3.exports, jsdoc.LiteralDoc)
|
||||||
|
assert mod3.exports.value == 56.0
|
||||||
|
assert mod3.doc == ''
|
||||||
|
|
||||||
|
def test_func():
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('module', function (d) {
|
||||||
|
/**
|
||||||
|
* @param {Foo} bar this is a bar
|
||||||
|
* @param {Baz} qux this is a qux
|
||||||
|
*/
|
||||||
|
return function (bar, qux) {
|
||||||
|
return 42;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
exports = mod.exports
|
||||||
|
assert type(exports) == jsdoc.FunctionDoc
|
||||||
|
assert exports['sourcemodule'] is mod
|
||||||
|
|
||||||
|
assert exports.name == ''
|
||||||
|
assert exports.is_constructor == False
|
||||||
|
assert exports.is_private == False
|
||||||
|
|
||||||
|
assert params(exports.params[0]) == ('bar', 'Foo', 'this is a bar')
|
||||||
|
assert params(exports.params[1]) == ('qux', 'Baz', 'this is a qux')
|
||||||
|
assert params(exports.return_val) == ('', '', '')
|
||||||
|
|
||||||
|
def test_hoist():
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('module', function() {
|
||||||
|
return foo;
|
||||||
|
/**
|
||||||
|
* @param a_thing
|
||||||
|
*/
|
||||||
|
function foo(a_thing) {
|
||||||
|
return 42;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
actual = mod.exports
|
||||||
|
assert type(actual) == jsdoc.FunctionDoc
|
||||||
|
[param] = actual.params
|
||||||
|
assert params(param) == ('a_thing', '', '')
|
||||||
|
|
||||||
|
def test_export_instance():
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('module', function (require) {
|
||||||
|
var Class = require('Class');
|
||||||
|
/**
|
||||||
|
* Provides an instance of Class
|
||||||
|
*/
|
||||||
|
return new Class();
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
assert type(mod.exports) == jsdoc.InstanceDoc
|
||||||
|
assert mod.exports.doc == 'Provides an instance of Class'
|
||||||
|
assert mod.exports['sourcemodule'] is mod
|
||||||
|
|
||||||
|
def test_bounce():
|
||||||
|
[m2, m1] = parse("""
|
||||||
|
odoo.define('m2', function (require) {
|
||||||
|
var Item = require('m1');
|
||||||
|
return {
|
||||||
|
Item: Item
|
||||||
|
};
|
||||||
|
});
|
||||||
|
odoo.define('m1', function (require) {
|
||||||
|
var Class = require('Class');
|
||||||
|
var Item = Class.extend({});
|
||||||
|
return Item;
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
assert type(m2.exports) == jsdoc.NSDoc
|
||||||
|
it = m2.exports.get_property('Item')
|
||||||
|
assert type(it) == jsdoc.ClassDoc
|
||||||
|
assert it['sourcemodule'] is m1
|
||||||
|
assert sorted([n for n, _ in m1.properties]) == ['<exports>', 'Class', 'Item']
|
||||||
|
|
||||||
|
def test_reassign():
|
||||||
|
[m] = parse("""
|
||||||
|
odoo.define('m', function (require) {
|
||||||
|
var Class = require('Class');
|
||||||
|
/** local class */
|
||||||
|
var Class = Class.extend({});
|
||||||
|
return Class
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
assert m.exports.doc == 'local class'
|
||||||
|
# can't use equality or identity so use class comment...
|
||||||
|
assert m.exports.superclass.doc == 'Base Class'
|
||||||
|
|
||||||
|
def test_attr():
|
||||||
|
[m1, m2] = parse("""
|
||||||
|
odoo.define('m1', function (require) {
|
||||||
|
var Class = require('Class');
|
||||||
|
var Item = Class.extend({});
|
||||||
|
return {Item: Item};
|
||||||
|
});
|
||||||
|
odoo.define('m2', function (require) {
|
||||||
|
var Item = require('m1').Item;
|
||||||
|
Item.include({});
|
||||||
|
return Item.extend({});
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
assert type(m2.exports) == jsdoc.ClassDoc
|
||||||
|
# that these two are resolved separately may be an issue at one point (?)
|
||||||
|
assert m2.exports.superclass.to_dict() == m1.exports.get_property('Item').to_dict()
|
||||||
|
|
||||||
|
def test_nothing_implicit():
|
||||||
|
[m] = parse("""
|
||||||
|
odoo.define('m', function () {
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
assert m.exports is None
|
||||||
|
|
||||||
|
def test_nothing_explicit():
|
||||||
|
[m] = parse("""
|
||||||
|
odoo.define('m', function () {
|
||||||
|
return;
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
assert m.exports is None
|
177
_extensions/autojsdoc/parser/tests/test_namespace.py
Normal file
177
_extensions/autojsdoc/parser/tests/test_namespace.py
Normal file
@ -0,0 +1,177 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
from autojsdoc.parser import jsdoc
|
||||||
|
from support import parse, params
|
||||||
|
|
||||||
|
|
||||||
|
def test_empty():
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('a.ns', function (r) {
|
||||||
|
return {};
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
assert type(mod.exports) == jsdoc.NSDoc
|
||||||
|
assert mod.exports.properties == []
|
||||||
|
|
||||||
|
def test_inline():
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('a.ns', function (r) {
|
||||||
|
return {
|
||||||
|
/**
|
||||||
|
* a thing
|
||||||
|
* @type {Boolean}
|
||||||
|
*/
|
||||||
|
a: true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
assert isinstance(mod.exports, jsdoc.NSDoc)
|
||||||
|
[(n, p)] = mod.exports.properties
|
||||||
|
assert n == 'a'
|
||||||
|
assert params(p) == ('a', 'Boolean', 'a thing')
|
||||||
|
|
||||||
|
def test_header():
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('a.ns', function (r) {
|
||||||
|
/**
|
||||||
|
* @property {Boolean} a a thing
|
||||||
|
*/
|
||||||
|
return { a: true }
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
assert isinstance(mod.exports, jsdoc.NSDoc)
|
||||||
|
[(_, p)] = mod.exports.properties
|
||||||
|
assert params(p) == ('a', 'Boolean', 'a thing')
|
||||||
|
|
||||||
|
def test_header_conflict():
|
||||||
|
""" should the header or the inline comment take precedence? """
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('a.ns', function (r) {
|
||||||
|
/**
|
||||||
|
* @property {Boolean} a a thing
|
||||||
|
*/
|
||||||
|
return {
|
||||||
|
/** @type {String} */
|
||||||
|
a: true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
assert isinstance(mod.exports, jsdoc.NSDoc)
|
||||||
|
[(_, p)] = mod.exports.properties
|
||||||
|
assert params(p) == ('a', 'Boolean', 'a thing')
|
||||||
|
|
||||||
|
def test_mixin():
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('a.mixin', function (r) {
|
||||||
|
/**
|
||||||
|
* @mixin
|
||||||
|
*/
|
||||||
|
return {
|
||||||
|
/**
|
||||||
|
* @returns {Number} a number
|
||||||
|
*/
|
||||||
|
do_thing: function other() { return 42; }
|
||||||
|
}
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
assert isinstance(mod.exports, jsdoc.MixinDoc)
|
||||||
|
[(n, p)] = mod.exports.properties
|
||||||
|
assert n == 'do_thing'
|
||||||
|
assert params(p) == ('other', 'Function', '')
|
||||||
|
assert params(p.return_val) == ('', 'Number', 'a number')
|
||||||
|
|
||||||
|
def test_literal():
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('a.ns', function (r) {
|
||||||
|
/** whop whop */
|
||||||
|
return {
|
||||||
|
'a': 1,
|
||||||
|
/** wheee */
|
||||||
|
'b': 2,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
assert mod.exports.doc == 'whop whop'
|
||||||
|
[(_1, a), (_2, b)] = mod.exports.properties
|
||||||
|
assert params(a) == ('a', 'Number', '')
|
||||||
|
assert params(b) == ('b', 'Number', 'wheee')
|
||||||
|
|
||||||
|
def test_fill_ns():
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('a.ns', function (r) {
|
||||||
|
var Class = r('Class');
|
||||||
|
var ns = {};
|
||||||
|
/** ok */
|
||||||
|
ns.a = 1;
|
||||||
|
/** @type {String} */
|
||||||
|
ns['b'] = 2;
|
||||||
|
/** Ike */
|
||||||
|
ns.c = Class.extend({});
|
||||||
|
ns.d = function () {}
|
||||||
|
return ns;
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
ns = mod.exports
|
||||||
|
assert type(ns) == jsdoc.NSDoc
|
||||||
|
[(_a, a), (_b, b), (_c, c), (_d, d)] = ns.properties
|
||||||
|
assert params(a) == ('a', 'Number', 'ok')
|
||||||
|
assert params(b) == ('b', 'String', '')
|
||||||
|
assert type(c) == jsdoc.ClassDoc
|
||||||
|
assert type(d) == jsdoc.FunctionDoc
|
||||||
|
|
||||||
|
def test_extend_other():
|
||||||
|
[o, b] = parse("""
|
||||||
|
odoo.define('a.ns', function () {
|
||||||
|
/** @name outer */
|
||||||
|
return {
|
||||||
|
/** @name inner */
|
||||||
|
a: {}
|
||||||
|
};
|
||||||
|
});
|
||||||
|
odoo.define('b', function (r) {
|
||||||
|
var o = r('a.ns');
|
||||||
|
var Class = r('Class');
|
||||||
|
/** Class 1 */
|
||||||
|
o.a.b = Class.extend({m_b: 1});
|
||||||
|
/** Class 2 */
|
||||||
|
o.a['c'] = Class.extend({m_c: 1});
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
[(_, m)] = o.exports.properties
|
||||||
|
assert type(m) == jsdoc.NSDoc
|
||||||
|
|
||||||
|
b = m.get_property('b')
|
||||||
|
assert type(b) == jsdoc.ClassDoc
|
||||||
|
assert b.get_property('m_b')
|
||||||
|
assert b.doc == 'Class 1'
|
||||||
|
|
||||||
|
c = m.get_property('c')
|
||||||
|
assert type(c) == jsdoc.ClassDoc
|
||||||
|
assert c.get_property('m_c')
|
||||||
|
assert c.doc == 'Class 2'
|
||||||
|
|
||||||
|
def test_ns_variables():
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('A', function (r) {
|
||||||
|
var Class = r('Class');
|
||||||
|
var Thing = Class.extend({});
|
||||||
|
return {
|
||||||
|
Thing: Thing
|
||||||
|
};
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
p = mod.exports.get_property('Thing')
|
||||||
|
assert type(p) == jsdoc.ClassDoc
|
||||||
|
|
||||||
|
def test_diff():
|
||||||
|
""" Have the NS key and the underlying object differ
|
||||||
|
"""
|
||||||
|
[mod] = parse("""
|
||||||
|
odoo.define('mod', function (r) {
|
||||||
|
var Class = r('Class');
|
||||||
|
var Foo = Class.extend({});
|
||||||
|
return { Class: Foo };
|
||||||
|
});
|
||||||
|
""")
|
||||||
|
c = mod.exports.get_property('Class')
|
||||||
|
assert type(c) == jsdoc.ClassDoc
|
||||||
|
assert c.name == 'Foo'
|
116
_extensions/autojsdoc/parser/tests/test_params.py
Normal file
116
_extensions/autojsdoc/parser/tests/test_params.py
Normal file
@ -0,0 +1,116 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
from autojsdoc.parser.jsdoc import ParamDoc
|
||||||
|
|
||||||
|
"Lorem ipsum dolor sit amet, consectetur adipiscing elit."
|
||||||
|
|
||||||
|
def test_basic():
|
||||||
|
d = ParamDoc("Lorem ipsum dolor sit amet, consectetur adipiscing elit.").to_dict()
|
||||||
|
assert d == {
|
||||||
|
'name': 'Lorem',
|
||||||
|
'type': '',
|
||||||
|
'optional': False,
|
||||||
|
'default': None,
|
||||||
|
'doc': 'ipsum dolor sit amet, consectetur adipiscing elit.',
|
||||||
|
}
|
||||||
|
|
||||||
|
d = ParamDoc("{Lorem} ipsum dolor sit amet, consectetur adipiscing elit.").to_dict()
|
||||||
|
assert d == {
|
||||||
|
'name': 'ipsum',
|
||||||
|
'type': 'Lorem',
|
||||||
|
'optional': False,
|
||||||
|
'default': None,
|
||||||
|
'doc': 'dolor sit amet, consectetur adipiscing elit.',
|
||||||
|
}
|
||||||
|
|
||||||
|
def test_optional():
|
||||||
|
d = ParamDoc("[Lorem] ipsum dolor sit amet, consectetur adipiscing elit.").to_dict()
|
||||||
|
assert d == {
|
||||||
|
'name': 'Lorem',
|
||||||
|
'type': '',
|
||||||
|
'optional': True,
|
||||||
|
'default': None,
|
||||||
|
'doc': 'ipsum dolor sit amet, consectetur adipiscing elit.',
|
||||||
|
}
|
||||||
|
|
||||||
|
d = ParamDoc("[Lorem=42] ipsum dolor sit amet, consectetur adipiscing elit.").to_dict()
|
||||||
|
assert d == {
|
||||||
|
'name': 'Lorem',
|
||||||
|
'type': '',
|
||||||
|
'optional': True,
|
||||||
|
'default': "42",
|
||||||
|
'doc': 'ipsum dolor sit amet, consectetur adipiscing elit.',
|
||||||
|
}
|
||||||
|
|
||||||
|
d = ParamDoc("{Lorem} [ipsum] dolor sit amet, consectetur adipiscing elit.").to_dict()
|
||||||
|
assert d == {
|
||||||
|
'name': 'ipsum',
|
||||||
|
'type': 'Lorem',
|
||||||
|
'optional': True,
|
||||||
|
'default': None,
|
||||||
|
'doc': 'dolor sit amet, consectetur adipiscing elit.',
|
||||||
|
}
|
||||||
|
|
||||||
|
d = ParamDoc("{Lorem} [ipsum=42] dolor sit amet, consectetur adipiscing elit.").to_dict()
|
||||||
|
assert d == {
|
||||||
|
'name': 'ipsum',
|
||||||
|
'type': 'Lorem',
|
||||||
|
'optional': True,
|
||||||
|
'default': '42',
|
||||||
|
'doc': 'dolor sit amet, consectetur adipiscing elit.',
|
||||||
|
}
|
||||||
|
|
||||||
|
def test_returns():
|
||||||
|
d = ParamDoc("{} ipsum dolor sit amet, consectetur adipiscing elit.").to_dict()
|
||||||
|
assert d == {
|
||||||
|
'name': '',
|
||||||
|
'type': '',
|
||||||
|
'optional': False,
|
||||||
|
'default': None,
|
||||||
|
'doc': 'ipsum dolor sit amet, consectetur adipiscing elit.',
|
||||||
|
}
|
||||||
|
d = ParamDoc("{Lorem} ipsum dolor sit amet, consectetur adipiscing elit.").to_dict()
|
||||||
|
assert d == {
|
||||||
|
'name': '',
|
||||||
|
'type': 'Lorem',
|
||||||
|
'optional': False,
|
||||||
|
'default': None,
|
||||||
|
'doc': 'ipsum dolor sit amet, consectetur adipiscing elit.',
|
||||||
|
}
|
||||||
|
|
||||||
|
def test_odd():
|
||||||
|
d = ParamDoc("{jQuery} [$target] the node where content will be prepended").to_dict()
|
||||||
|
assert d == {
|
||||||
|
'name': '$target',
|
||||||
|
'type': 'jQuery',
|
||||||
|
'optional': True,
|
||||||
|
'default': None,
|
||||||
|
'doc': 'the node where content will be prepended',
|
||||||
|
}
|
||||||
|
|
||||||
|
d = ParamDoc("""{htmlString} [content] DOM element,
|
||||||
|
array of elements, HTML string or jQuery object to prepend to $target""").to_dict()
|
||||||
|
assert d == {
|
||||||
|
'name': 'content',
|
||||||
|
'type': 'htmlString',
|
||||||
|
'optional': True,
|
||||||
|
'default': None,
|
||||||
|
'doc': "DOM element,\n array of elements, HTML string or jQuery object to prepend to $target",
|
||||||
|
}
|
||||||
|
|
||||||
|
d = ParamDoc("{Boolean} [options.in_DOM] true if $target is in the DOM").to_dict()
|
||||||
|
assert d == {
|
||||||
|
'name': 'options.in_DOM',
|
||||||
|
'type': 'Boolean',
|
||||||
|
'optional': True,
|
||||||
|
'default': None,
|
||||||
|
'doc': 'true if $target is in the DOM',
|
||||||
|
}
|
||||||
|
|
||||||
|
d = ParamDoc("Lorem\n ipsum dolor sit amet, consectetur adipiscing elit.").to_dict()
|
||||||
|
assert d['doc'] == 'ipsum dolor sit amet, consectetur adipiscing elit.'
|
||||||
|
|
||||||
|
d = ParamDoc("Lorem - ipsum dolor sit amet, consectetur adipiscing elit.").to_dict()
|
||||||
|
assert d['doc'] == 'ipsum dolor sit amet, consectetur adipiscing elit.'
|
||||||
|
|
||||||
|
d = ParamDoc("Lorem : ipsum dolor sit amet, consectetur adipiscing elit.").to_dict()
|
||||||
|
assert d['doc'] == 'ipsum dolor sit amet, consectetur adipiscing elit.'
|
104
_extensions/autojsdoc/parser/tests/test_typespec.py
Normal file
104
_extensions/autojsdoc/parser/tests/test_typespec.py
Normal file
@ -0,0 +1,104 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
from __future__ import unicode_literals
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
from autojsdoc.parser import types
|
||||||
|
|
||||||
|
def test_parser():
|
||||||
|
assert types.parse("MouseEvent|TouchEvent") == types.Alt([
|
||||||
|
types.Type('MouseEvent', []),
|
||||||
|
types.Type('TouchEvent', []),
|
||||||
|
])
|
||||||
|
|
||||||
|
assert types.parse('Deferred<Object>') == types.Alt([
|
||||||
|
types.Type('Deferred', [
|
||||||
|
types.Alt([types.Type('Object', [])])
|
||||||
|
])
|
||||||
|
])
|
||||||
|
|
||||||
|
assert types.parse('Object|undefined') == types.Alt([
|
||||||
|
types.Type('Object', []),
|
||||||
|
types.Literal('undefined'),
|
||||||
|
])
|
||||||
|
assert types.parse('any[]') == types.Alt([
|
||||||
|
types.Type('Array', [types.Type('any', [])])
|
||||||
|
])
|
||||||
|
assert types.parse("'xml' | 'less'") == types.Alt([
|
||||||
|
types.Literal('xml'),
|
||||||
|
types.Literal('less'),
|
||||||
|
])
|
||||||
|
assert types.parse('Foo<Bar | Bar[] | null>') == types.Alt([
|
||||||
|
types.Type('Foo', [types.Alt([
|
||||||
|
types.Type('Bar', []),
|
||||||
|
types.Type('Array', [types.Type('Bar', [])]),
|
||||||
|
types.Literal('null'),
|
||||||
|
])])
|
||||||
|
])
|
||||||
|
assert types.parse('Function<Array<Object[]>>') == types.Alt([
|
||||||
|
types.Type('Function', [types.Alt([
|
||||||
|
types.Type('Array', [types.Alt([
|
||||||
|
types.Type('Array', [
|
||||||
|
types.Type('Object', [])
|
||||||
|
])
|
||||||
|
])])
|
||||||
|
])])
|
||||||
|
])
|
||||||
|
|
||||||
|
|
||||||
|
def test_tokens():
|
||||||
|
toks = list(types.tokenize('A'))
|
||||||
|
assert toks == [(types.NAME, 'A')]
|
||||||
|
|
||||||
|
toks = list(types.tokenize('"foo" | "bar"'))
|
||||||
|
assert toks == [(types.LITERAL, 'foo'), (types.OP, '|'), (types.LITERAL, 'bar')]
|
||||||
|
|
||||||
|
toks = list(types.tokenize('1 or 2'))
|
||||||
|
assert toks == [(types.LITERAL, 1), (types.OP, '|'), (types.LITERAL, 2)]
|
||||||
|
|
||||||
|
toks = list(types.tokenize('a.b.c.d'))
|
||||||
|
assert toks == [
|
||||||
|
(types.NAME, 'a'), (types.OP, '.'),
|
||||||
|
(types.NAME, 'b'), (types.OP, '.'),
|
||||||
|
(types.NAME, 'c'), (types.OP, '.'),
|
||||||
|
(types.NAME, 'd'),
|
||||||
|
]
|
||||||
|
|
||||||
|
toks = list(types.tokenize('Function<String, Object>'))
|
||||||
|
assert toks == [
|
||||||
|
(types.NAME, 'Function'),
|
||||||
|
(types.OP, '<'),
|
||||||
|
(types.NAME, 'String'),
|
||||||
|
(types.OP, ','),
|
||||||
|
(types.NAME, 'Object'),
|
||||||
|
(types.OP, '>')
|
||||||
|
]
|
||||||
|
|
||||||
|
toks = list(types.tokenize('Function<Array<Object[]>>'))
|
||||||
|
assert toks == [
|
||||||
|
(types.NAME, 'Function'),
|
||||||
|
(types.OP, '<'),
|
||||||
|
(types.NAME, 'Array'),
|
||||||
|
(types.OP, '<'),
|
||||||
|
(types.NAME, 'Object'),
|
||||||
|
(types.OP, '['),
|
||||||
|
(types.OP, ']'),
|
||||||
|
(types.OP, '>'),
|
||||||
|
(types.OP, '>')
|
||||||
|
]
|
||||||
|
|
||||||
|
def test_peekable():
|
||||||
|
p = types.Peekable(range(5))
|
||||||
|
|
||||||
|
assert p.peek() == 0
|
||||||
|
assert next(p) == 0, "consuming should yield previously peeked value"
|
||||||
|
next(p)
|
||||||
|
next(p)
|
||||||
|
assert next(p) == 3
|
||||||
|
assert p.peek() == 4
|
||||||
|
next(p)
|
||||||
|
assert p.peek(None) is None
|
||||||
|
with pytest.raises(StopIteration):
|
||||||
|
p.peek()
|
||||||
|
with pytest.raises(StopIteration):
|
||||||
|
next(p)
|
196
_extensions/autojsdoc/parser/types.py
Normal file
196
_extensions/autojsdoc/parser/types.py
Normal file
@ -0,0 +1,196 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
import ast
|
||||||
|
import collections
|
||||||
|
import io
|
||||||
|
import token
|
||||||
|
from tokenize import generate_tokens
|
||||||
|
|
||||||
|
LITERAL_NAME = ('null', 'undefined', 'true', 'false')
|
||||||
|
|
||||||
|
__all__ = ['tokenize', 'parse', 'iterate', 'Alt', 'Type', 'Literal']
|
||||||
|
|
||||||
|
class _Marker(object):
|
||||||
|
__slots__ = ['name']
|
||||||
|
def __init__(self, name): self.name = name
|
||||||
|
def __repr__(self): return '<%s>' % self.name
|
||||||
|
|
||||||
|
OP = _Marker('OP')
|
||||||
|
NAME = _Marker('NAME')
|
||||||
|
LITERAL = _Marker('LITERAL')
|
||||||
|
def tokenize(typespec):
|
||||||
|
typespec = typespec.replace('$', 'jQuery')
|
||||||
|
for toktype, string, _, _, _ in generate_tokens(io.StringIO(typespec).readline):
|
||||||
|
if toktype == token.NAME:
|
||||||
|
if string == 'or': # special case "A or B"
|
||||||
|
# TODO: deprecation warning
|
||||||
|
yield (OP, '|')
|
||||||
|
else:
|
||||||
|
yield (NAME, string)
|
||||||
|
elif toktype == token.OP:
|
||||||
|
if string in '|<>[].,':
|
||||||
|
yield (OP, string)
|
||||||
|
elif string == '>>':
|
||||||
|
yield (OP, '>')
|
||||||
|
yield (OP, '>')
|
||||||
|
elif string == '*': # maybe?
|
||||||
|
yield (NAME, 'any')
|
||||||
|
elif string in '()':
|
||||||
|
# TODO: deprecation warning
|
||||||
|
# seems useless unless we add support for "..."
|
||||||
|
continue
|
||||||
|
else:
|
||||||
|
raise ValueError("Unknown typespec operator %r" % string)
|
||||||
|
elif toktype in (token.STRING, token.NUMBER): # enum-ish
|
||||||
|
yield (LITERAL, ast.literal_eval(string))
|
||||||
|
elif toktype == token.ENDMARKER:
|
||||||
|
return
|
||||||
|
elif toktype == token.NEWLINE:
|
||||||
|
pass
|
||||||
|
else:
|
||||||
|
raise ValueError("Unknown typespec token %s" % token.tok_name[toktype])
|
||||||
|
|
||||||
|
Alt = collections.namedtuple('Alt', ['params'])
|
||||||
|
Literal = collections.namedtuple('Literal', ['value'])
|
||||||
|
Type = collections.namedtuple('Type', ['name', 'params'])
|
||||||
|
|
||||||
|
def iterate(t, fortype, forliteral):
|
||||||
|
if isinstance(t, Alt):
|
||||||
|
ps = Peekable(t.params)
|
||||||
|
for param in ps:
|
||||||
|
for it in iterate(param, fortype, forliteral):
|
||||||
|
yield it
|
||||||
|
if ps.peek(None):
|
||||||
|
yield forliteral('|')
|
||||||
|
elif isinstance(t, Type):
|
||||||
|
yield fortype(t.name)
|
||||||
|
if not t.params:
|
||||||
|
return
|
||||||
|
yield forliteral('<')
|
||||||
|
ps = Peekable(t.params)
|
||||||
|
for param in ps:
|
||||||
|
for it in iterate(param, fortype, forliteral):
|
||||||
|
yield it
|
||||||
|
if ps.peek(None):
|
||||||
|
yield forliteral(',')
|
||||||
|
yield forliteral('>')
|
||||||
|
elif isinstance(t, Literal):
|
||||||
|
yield forliteral(t.value)
|
||||||
|
else:
|
||||||
|
raise ValueError('Unknown item %s' % t)
|
||||||
|
|
||||||
|
def parse(typespec):
|
||||||
|
tokens = Peekable(tokenize(typespec))
|
||||||
|
return parse_alt(tokens)
|
||||||
|
|
||||||
|
# alt = param
|
||||||
|
# | param '|' alt
|
||||||
|
def parse_alt(tokens):
|
||||||
|
alt = Alt([])
|
||||||
|
while True:
|
||||||
|
alt.params.append(parse_param(tokens))
|
||||||
|
if not next_is(tokens, (OP, '|')):
|
||||||
|
break
|
||||||
|
return alt
|
||||||
|
|
||||||
|
# param = 'null' | 'undefined' | 'true' | 'false'
|
||||||
|
# | number
|
||||||
|
# | string
|
||||||
|
# | typename '[]'
|
||||||
|
# X '[' alt ']'
|
||||||
|
# | typename '<' params '>'
|
||||||
|
# params = alt | alt ',' params
|
||||||
|
def parse_param(tokens):
|
||||||
|
t, v = tokens.peek()
|
||||||
|
if t == LITERAL or (t == NAME and v in LITERAL_NAME):
|
||||||
|
next(tokens)
|
||||||
|
return Literal(str(v))
|
||||||
|
|
||||||
|
# # [typespec] # should this be Array<T> or an n-uple allowing multiple items?
|
||||||
|
# if tok == (OP, '['):
|
||||||
|
# next(tokens)
|
||||||
|
# t = Type('Array', [parse_alt(tokens)])
|
||||||
|
# rest = next(tokens)
|
||||||
|
# expect(rest, (OP, ']'))
|
||||||
|
# return t
|
||||||
|
|
||||||
|
t = Type(parse_typename(tokens), [])
|
||||||
|
# type[]
|
||||||
|
peek = tokens.peek(None)
|
||||||
|
if peek == (OP, '['):
|
||||||
|
next(tokens) # skip peeked
|
||||||
|
expect(next(tokens), (OP, ']'))
|
||||||
|
return Type('Array', [t])
|
||||||
|
# type<typespec, ...>
|
||||||
|
if peek == (OP, '<'):
|
||||||
|
next(tokens) # skip peeked
|
||||||
|
while True:
|
||||||
|
t.params.append(parse_alt(tokens))
|
||||||
|
n = next(tokens)
|
||||||
|
if n == (OP, ','):
|
||||||
|
continue
|
||||||
|
if n == (OP, '>'):
|
||||||
|
break
|
||||||
|
raise ValueError("Expected OP ',' or OP ',', got %s '%s'" % n)
|
||||||
|
return t
|
||||||
|
|
||||||
|
# typename = name | name '.' typename
|
||||||
|
def parse_typename(tokens):
|
||||||
|
typename = []
|
||||||
|
while True:
|
||||||
|
(t, n) = next(tokens)
|
||||||
|
if t != NAME:
|
||||||
|
raise ValueError("Expected a name token, got %s '%s'" % (t, n))
|
||||||
|
if n in LITERAL_NAME:
|
||||||
|
raise ValueError("Expected a type name, got literal %s" % n)
|
||||||
|
typename.append(n)
|
||||||
|
if not next_is(tokens, (OP, '.')):
|
||||||
|
break
|
||||||
|
return '.'.join(typename)
|
||||||
|
|
||||||
|
def next_is(tokens, expected):
|
||||||
|
"""
|
||||||
|
Consumes the next token if it's `expected` otherwise does not touch the
|
||||||
|
tokens.
|
||||||
|
|
||||||
|
Returns whether it consumed a token
|
||||||
|
"""
|
||||||
|
if tokens.peek(None) == expected:
|
||||||
|
next(tokens)
|
||||||
|
return True
|
||||||
|
return False
|
||||||
|
|
||||||
|
def expect(actual, expected):
|
||||||
|
""" Raises ValueError if `actual` and `expected` are different
|
||||||
|
|
||||||
|
:type actual: (object, str)
|
||||||
|
:type expected: (object, str)
|
||||||
|
"""
|
||||||
|
if actual != expected:
|
||||||
|
raise ValueError("Expected %s '%s', got %s '%s'" % (expected + actual))
|
||||||
|
|
||||||
|
|
||||||
|
NONE = object()
|
||||||
|
class Peekable(object):
|
||||||
|
__slots__ = ['_head', '_it']
|
||||||
|
def __init__(self, iterable):
|
||||||
|
self._head = NONE
|
||||||
|
self._it = iter(iterable)
|
||||||
|
|
||||||
|
def __iter__(self):
|
||||||
|
return self
|
||||||
|
def __next__(self):
|
||||||
|
if self._head is not NONE:
|
||||||
|
r, self._head = self._head, NONE
|
||||||
|
return r
|
||||||
|
return next(self._it)
|
||||||
|
next = __next__
|
||||||
|
|
||||||
|
def peek(self, default=NONE):
|
||||||
|
if self._head is NONE:
|
||||||
|
try:
|
||||||
|
self._head = next(self._it)
|
||||||
|
except StopIteration:
|
||||||
|
if default is not NONE:
|
||||||
|
return default
|
||||||
|
raise
|
||||||
|
return self._head
|
295
_extensions/autojsdoc/parser/utils.py
Normal file
295
_extensions/autojsdoc/parser/utils.py
Normal file
@ -0,0 +1,295 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
def _name(node):
|
||||||
|
if node['type'] == 'Identifier':
|
||||||
|
return node['name']
|
||||||
|
if node['type'] == 'MemberExpression':
|
||||||
|
return "%s.%s" % (_name(node['object']), _name(node['property']))
|
||||||
|
raise ValueError("Unnamable node type %s" % node['type'])
|
||||||
|
|
||||||
|
def _value(node, strict=False):
|
||||||
|
t = node['type']
|
||||||
|
if t == 'Identifier':
|
||||||
|
return node['name']
|
||||||
|
elif t == 'Literal':
|
||||||
|
return node['value']
|
||||||
|
msg = '<%s has no value>' % _identify(node, {})
|
||||||
|
if strict:
|
||||||
|
raise ValueError(msg)
|
||||||
|
return msg
|
||||||
|
|
||||||
|
def _identify(valnode, ns):
|
||||||
|
if valnode is None:
|
||||||
|
return "None"
|
||||||
|
# already identified and re-set in the ns?
|
||||||
|
if isinstance(valnode, Printable):
|
||||||
|
return valnode
|
||||||
|
|
||||||
|
# check other non-empty returns
|
||||||
|
t = valnode['type']
|
||||||
|
if t == "Literal":
|
||||||
|
if valnode.get('regex'):
|
||||||
|
return valnode['regex']['pattern']
|
||||||
|
return valnode['value']
|
||||||
|
elif t == "Identifier":
|
||||||
|
n = valnode['name']
|
||||||
|
return ns.get(n, Global(n))
|
||||||
|
elif t == "NewExpression":
|
||||||
|
return Instance(_identify(valnode['callee'], ns))
|
||||||
|
elif t == "ObjectExpression":
|
||||||
|
return Namespace({
|
||||||
|
_value(prop['key']): _identify(prop['value'], ns)
|
||||||
|
for prop in valnode['properties']
|
||||||
|
})
|
||||||
|
elif t == "MemberExpression":
|
||||||
|
return Deref(
|
||||||
|
_identify(valnode['object'], ns),
|
||||||
|
_identify(valnode['property'], ns) if valnode['computed'] else valnode['property']['name'],
|
||||||
|
)
|
||||||
|
elif t == "CallExpression":
|
||||||
|
return Call(
|
||||||
|
_identify(valnode['callee'], ns),
|
||||||
|
[_identify(arg, ns) for arg in valnode['arguments']],
|
||||||
|
)
|
||||||
|
elif t == 'BinaryExpression':
|
||||||
|
return "%s %s %s" % (
|
||||||
|
_identify(valnode['left'], ns),
|
||||||
|
valnode['operator'],
|
||||||
|
_identify(valnode['right'], ns),
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
return t
|
||||||
|
|
||||||
|
class Counter(object):
|
||||||
|
__slots__ = ['_number']
|
||||||
|
def __init__(self):
|
||||||
|
self._number = 0
|
||||||
|
|
||||||
|
def __bool__(self):
|
||||||
|
return bool(self._number)
|
||||||
|
__nonzero__ = __bool__
|
||||||
|
def __int__(self):
|
||||||
|
return self._number
|
||||||
|
|
||||||
|
def get(self):
|
||||||
|
return self._number
|
||||||
|
def increment(self):
|
||||||
|
self._number += 1
|
||||||
|
return self._number
|
||||||
|
def decrement(self):
|
||||||
|
self._number -= 1
|
||||||
|
return self._number
|
||||||
|
|
||||||
|
def __enter__(self):
|
||||||
|
self._number += 1
|
||||||
|
return self._number
|
||||||
|
def __exit__(self, *args):
|
||||||
|
self._number -= 1
|
||||||
|
|
||||||
|
class Printable(object):
|
||||||
|
__slots__ = []
|
||||||
|
depth = Counter()
|
||||||
|
def __repr__(self):
|
||||||
|
with self.depth as i:
|
||||||
|
if i > 2:
|
||||||
|
return "%s(...)" % type(self).__name__
|
||||||
|
return "%s(%s)" % (type(self).__name__, ', '.join(
|
||||||
|
"%s=%r" % (k, getattr(self, k))
|
||||||
|
for k in self.__slots__
|
||||||
|
if not k.startswith('_')
|
||||||
|
if getattr(self, k)
|
||||||
|
))
|
||||||
|
|
||||||
|
|
||||||
|
def resolve(obj, store):
|
||||||
|
if obj is None:
|
||||||
|
return None
|
||||||
|
|
||||||
|
if isinstance(obj, (type(u''), bool, int, float)):
|
||||||
|
return obj
|
||||||
|
|
||||||
|
if isinstance(obj, ModuleProxy):
|
||||||
|
return resolve(obj.get(), store)
|
||||||
|
|
||||||
|
try:
|
||||||
|
if getattr(obj, 'resolved', False):
|
||||||
|
return obj
|
||||||
|
return obj.resolve(store)
|
||||||
|
except AttributeError:
|
||||||
|
raise TypeError("Unresolvable {!r}".format(obj))
|
||||||
|
|
||||||
|
class Resolvable(Printable):
|
||||||
|
"""
|
||||||
|
For types resolved in place
|
||||||
|
"""
|
||||||
|
__slots__ = ['resolved']
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super(Resolvable, self).__init__()
|
||||||
|
self.resolved = False
|
||||||
|
def resolve(self, store):
|
||||||
|
self.resolved = True
|
||||||
|
return self
|
||||||
|
|
||||||
|
class Namespace(Resolvable):
|
||||||
|
__slots__ = ['attrs']
|
||||||
|
def __init__(self, attrs):
|
||||||
|
super(Namespace, self).__init__()
|
||||||
|
self.attrs = attrs
|
||||||
|
def resolve(self, store):
|
||||||
|
r = super(Namespace, self).resolve(store)
|
||||||
|
self.attrs = {
|
||||||
|
k: resolve(v, store)
|
||||||
|
for k, v in self.attrs.items()
|
||||||
|
}
|
||||||
|
return r
|
||||||
|
def __getitem__(self, key):
|
||||||
|
assert isinstance(key, type(u'')), "%r is not a namespace key" % key
|
||||||
|
try:
|
||||||
|
return self.attrs[key]
|
||||||
|
except KeyError:
|
||||||
|
return Global(self)[key]
|
||||||
|
|
||||||
|
class Module(Resolvable):
|
||||||
|
__slots__ = ('name', 'comments', 'exports', 'augments', 'dependencies', '_store')
|
||||||
|
def __init__(self, name, store, comments=()):
|
||||||
|
super(Module, self).__init__()
|
||||||
|
self.name = name
|
||||||
|
self._store = store
|
||||||
|
self.comments = tuple(comments)
|
||||||
|
self.exports = None
|
||||||
|
self.augments = []
|
||||||
|
self.dependencies = set()
|
||||||
|
store[name] = self
|
||||||
|
def add_dependency(self, depname):
|
||||||
|
dep = ModuleProxy(depname, self._store)
|
||||||
|
self.dependencies.add(dep)
|
||||||
|
return dep
|
||||||
|
def get(self):
|
||||||
|
return self
|
||||||
|
def resolve(self, store=None):
|
||||||
|
r = super(Module, self).resolve(store)
|
||||||
|
self.exports = resolve(self.exports, self._store)
|
||||||
|
self.augments = [resolve(a, self._store) for a in self.augments]
|
||||||
|
self.dependencies = {
|
||||||
|
resolve(d, self._store)
|
||||||
|
for d in self.dependencies
|
||||||
|
}
|
||||||
|
return r
|
||||||
|
def __getitem__(self, k):
|
||||||
|
try:
|
||||||
|
return self.exports[k]
|
||||||
|
except KeyError:
|
||||||
|
return Global(self)[k]
|
||||||
|
|
||||||
|
class ModuleProxy(object):
|
||||||
|
def __init__(self, name, store):
|
||||||
|
self.name = name
|
||||||
|
self.store = store
|
||||||
|
def get(self):
|
||||||
|
# web.web_client is not an actual module
|
||||||
|
return self.store.get(self.name, '<unstored %s>' % self.name)
|
||||||
|
def __repr__(self):
|
||||||
|
return repr(self.get())
|
||||||
|
def __hash__(self):
|
||||||
|
return hash(self.name)
|
||||||
|
def __eq__(self, other):
|
||||||
|
return self.name == other.name
|
||||||
|
|
||||||
|
class Class(Resolvable):
|
||||||
|
__slots__ = ['name', 'members', 'extends', 'mixins', 'comments']
|
||||||
|
def __init__(self, name, extends):
|
||||||
|
super(Class, self).__init__()
|
||||||
|
self.name = name
|
||||||
|
self.extends = extends
|
||||||
|
self.members = {}
|
||||||
|
self.mixins = []
|
||||||
|
self.comments = []
|
||||||
|
|
||||||
|
def resolve(self, store):
|
||||||
|
r = super(Class, self).resolve(store)
|
||||||
|
self.extends = resolve(self.extends, store)
|
||||||
|
self.mixins = [resolve(m, store) for m in self.mixins]
|
||||||
|
return r
|
||||||
|
|
||||||
|
class Instance(Resolvable):
|
||||||
|
__slots__ = ['type']
|
||||||
|
def __init__(self, type):
|
||||||
|
super(Instance, self).__init__()
|
||||||
|
self.type = type
|
||||||
|
|
||||||
|
def resolve(self, store):
|
||||||
|
r = super(Instance, self).resolve(store)
|
||||||
|
self.type = resolve(self.type, store)
|
||||||
|
if isinstance(self.type, Module):
|
||||||
|
self.type = self.type.exports
|
||||||
|
return r
|
||||||
|
|
||||||
|
def __getitem__(self, key):
|
||||||
|
return Global(self)['key']
|
||||||
|
|
||||||
|
class Function(Resolvable):
|
||||||
|
__slots__ = ['comments']
|
||||||
|
def __init__(self, comments):
|
||||||
|
super(Function, self).__init__()
|
||||||
|
self.comments = comments
|
||||||
|
|
||||||
|
class Deref(Printable):
|
||||||
|
__slots__ = ['object', 'property']
|
||||||
|
def __init__(self, object, property):
|
||||||
|
self.object = object
|
||||||
|
self.property = property
|
||||||
|
def resolve(self, store):
|
||||||
|
return resolve(self.object, store)[self.property]
|
||||||
|
|
||||||
|
class Call(Resolvable):
|
||||||
|
__slots__ = ['callable']
|
||||||
|
def __init__(self, callable, arguments):
|
||||||
|
super(Call, self).__init__()
|
||||||
|
self.callable = callable
|
||||||
|
def resolve(self, store):
|
||||||
|
r = super(Call, self).resolve(store)
|
||||||
|
self.callable = resolve(self.callable, store)
|
||||||
|
return r
|
||||||
|
def __getitem__(self, item):
|
||||||
|
return Global(self)[item]
|
||||||
|
|
||||||
|
def get(node, it, *path):
|
||||||
|
if not path:
|
||||||
|
return node[it]
|
||||||
|
return get(node[it], *path)
|
||||||
|
|
||||||
|
|
||||||
|
def match(node, pattern):
|
||||||
|
"""Checks that ``pattern`` is a subset of ``node``.
|
||||||
|
|
||||||
|
If a sub-pattern is a callable it will be called with the current
|
||||||
|
sub-node and its result is the match's. Other sub-patterns are
|
||||||
|
checked by equality against the correspoding sub-node.
|
||||||
|
|
||||||
|
Can be used to quickly check the descendants of a node of suitable
|
||||||
|
type.
|
||||||
|
|
||||||
|
TODO: support checking list content? Would be convenient to
|
||||||
|
constraint e.g. CallExpression based on their parameters
|
||||||
|
|
||||||
|
"""
|
||||||
|
if node is None and pattern is not None:
|
||||||
|
return
|
||||||
|
if callable(pattern):
|
||||||
|
return pattern(node)
|
||||||
|
if isinstance(node, list):
|
||||||
|
return all(len(node) > i and match(node[i], v) for i, v in pattern.items())
|
||||||
|
if isinstance(pattern, dict):
|
||||||
|
return all(match(node.get(k), v) for k, v in pattern.items())
|
||||||
|
return node == pattern
|
||||||
|
|
||||||
|
|
||||||
|
class Global(object):
|
||||||
|
def __init__(self, name):
|
||||||
|
self.name = name
|
||||||
|
def __repr__(self):
|
||||||
|
return '<external %s>' % self.name
|
||||||
|
def __getitem__(self, name):
|
||||||
|
return Global('%s.%s' % (self.name, name))
|
||||||
|
def resolve(self, store):
|
||||||
|
return self
|
126
_extensions/autojsdoc/parser/visitor.py
Normal file
126
_extensions/autojsdoc/parser/visitor.py
Normal file
@ -0,0 +1,126 @@
|
|||||||
|
from pyjsparser.pyjsparserdata import Syntax
|
||||||
|
|
||||||
|
_binary = lambda n: [n['left'], n['right']]
|
||||||
|
_function = lambda n: n['params'] + n['defaults'] + [n['body']]
|
||||||
|
# yields children in visitation order
|
||||||
|
_children = {
|
||||||
|
Syntax.ArrayExpression: lambda n: n['elements'],
|
||||||
|
Syntax.ArrayPattern: lambda n: n['elements'],
|
||||||
|
Syntax.ArrowFunctionExpression: _function,
|
||||||
|
Syntax.AssignmentExpression: _binary,
|
||||||
|
Syntax.AssignmentPattern: _binary,
|
||||||
|
Syntax.BinaryExpression: _binary,
|
||||||
|
Syntax.BlockStatement: lambda n: n['body'],
|
||||||
|
Syntax.BreakStatement: lambda n: [],
|
||||||
|
Syntax.CallExpression: lambda n: [n['callee']] + n['arguments'],
|
||||||
|
Syntax.CatchClause: lambda n: [n['param'], n['body']],
|
||||||
|
Syntax.ClassBody: lambda n: [n['body']],
|
||||||
|
Syntax.ClassDeclaration: lambda n: [n['superClass'], n['body']],
|
||||||
|
Syntax.ClassExpression: lambda n: [n['superClass'], n['body']],
|
||||||
|
Syntax.ConditionalExpression: lambda n: [n['test'], n['consequent'], n['alternate']],
|
||||||
|
Syntax.ContinueStatement: lambda n: [],
|
||||||
|
Syntax.DebuggerStatement: lambda n: [],
|
||||||
|
Syntax.DoWhileStatement: lambda n: [n['body'], n['test']],
|
||||||
|
Syntax.EmptyStatement: lambda n: [],
|
||||||
|
Syntax.ExportAllDeclaration: lambda n: [n['source']],
|
||||||
|
Syntax.ExportDefaultDeclaration: lambda n: [n['declaration']],
|
||||||
|
Syntax.ExportNamedDeclaration: lambda n: ([n['declaration']] if n['declaration'] else n['specifiers']) + [n['source']],
|
||||||
|
Syntax.ExportSpecifier: lambda n: [n['local'], n['exported']],
|
||||||
|
Syntax.ExpressionStatement: lambda n: [n['expression']],
|
||||||
|
Syntax.ForStatement: lambda n: [n['init'], n['test'], n['update'], n['body']],
|
||||||
|
Syntax.ForInStatement: lambda n: [n['left'], n['right'], n['body']],
|
||||||
|
Syntax.FunctionDeclaration: _function,
|
||||||
|
Syntax.FunctionExpression: _function,
|
||||||
|
Syntax.Identifier: lambda n: [],
|
||||||
|
Syntax.IfStatement: lambda n: [n['test'], n['consequent'], n['alternate']],
|
||||||
|
Syntax.ImportDeclaration: lambda n: n['specifiers'] + [n['source']],
|
||||||
|
Syntax.ImportDefaultSpecifier: lambda n: [n['local']],
|
||||||
|
Syntax.ImportNamespaceSpecifier: lambda n: [n['local']],
|
||||||
|
Syntax.ImportSpecifier: lambda n: [n['local'], n['imported']],
|
||||||
|
Syntax.LabeledStatement: lambda n: [n['body']],
|
||||||
|
Syntax.Literal: lambda n: [],
|
||||||
|
Syntax.LogicalExpression: _binary,
|
||||||
|
Syntax.MemberExpression: lambda n: [n['object'], n['property']],
|
||||||
|
#Syntax.MethodDefinition: lambda n: [],
|
||||||
|
Syntax.NewExpression: lambda n: [n['callee']] + n['arguments'],
|
||||||
|
Syntax.ObjectExpression: lambda n: n['properties'],
|
||||||
|
Syntax.ObjectPattern: lambda n: n['properties'],
|
||||||
|
Syntax.Program: lambda n: n['body'],
|
||||||
|
Syntax.Property: lambda n: [n['key'], n['value']],
|
||||||
|
Syntax.RestElement: lambda n: [n['argument']],
|
||||||
|
Syntax.ReturnStatement: lambda n: [n['argument']],
|
||||||
|
Syntax.SequenceExpression: lambda n: n['expressions'],
|
||||||
|
Syntax.SpreadElement: lambda n: [n['argument']],
|
||||||
|
Syntax.Super: lambda n: [],
|
||||||
|
Syntax.SwitchCase: lambda n: [n['test'], n['consequent']],
|
||||||
|
Syntax.SwitchStatement: lambda n: [n['discriminant']] + n['cases'],
|
||||||
|
Syntax.TaggedTemplateExpression: lambda n: [n['tag'], n['quasi']],
|
||||||
|
Syntax.TemplateElement: lambda n: [],
|
||||||
|
Syntax.TemplateLiteral: lambda n: n['quasis'] + n['expressions'],
|
||||||
|
Syntax.ThisExpression: lambda n: [],
|
||||||
|
Syntax.ThrowStatement: lambda n: [n['argument']],
|
||||||
|
Syntax.TryStatement: lambda n: [n['block'], n['handler'], n['finalizer']],
|
||||||
|
Syntax.UnaryExpression: lambda n: [n['argument']],
|
||||||
|
Syntax.UpdateExpression: lambda n: [n['argument']],
|
||||||
|
Syntax.VariableDeclaration: lambda n: n['declarations'],
|
||||||
|
Syntax.VariableDeclarator: lambda n: [n['id'], n['init']],
|
||||||
|
Syntax.WhileStatement: lambda n: [n['test'], n['body']],
|
||||||
|
Syntax.WithStatement: lambda n: [n['object'], n['body']],
|
||||||
|
}
|
||||||
|
|
||||||
|
SKIP = object()
|
||||||
|
class Visitor(object):
|
||||||
|
"""
|
||||||
|
Generic visitor for the pyjsparser AST.
|
||||||
|
|
||||||
|
Visitation is driven by the ``visit`` method, which iterates the tree in
|
||||||
|
depth-first pre-order.
|
||||||
|
|
||||||
|
For each node, calls ``enter_$NODETYPE``, visits the children then calls
|
||||||
|
``exit_$NODETYPE``. If the enter or exit methods are not present on the
|
||||||
|
visitor, falls back on ``enter_generic`` and ``exit_generic``.
|
||||||
|
|
||||||
|
Any ``enter_`` method can return ``SKIP`` to suppress both the traversal
|
||||||
|
of the subtree *and* the call to the corresponding ``exit_`` method
|
||||||
|
(whether generic or specific).
|
||||||
|
|
||||||
|
For convenience, ``visit`` will return whatever is set as the visitor's
|
||||||
|
``result`` attribute, ``None`` by default.
|
||||||
|
|
||||||
|
``visit`` can be given multiple root nodes, and it can be called multiple
|
||||||
|
times. The ``result`` attribute is cleared at each call but not between
|
||||||
|
two roots of the same ``visit`` call.
|
||||||
|
"""
|
||||||
|
def __init__(self):
|
||||||
|
super(Visitor, self).__init__()
|
||||||
|
self.result = None
|
||||||
|
|
||||||
|
def enter_generic(self, node): pass
|
||||||
|
def exit_generic(self, node): pass
|
||||||
|
|
||||||
|
def visit(self, nodes):
|
||||||
|
if isinstance(nodes, dict):
|
||||||
|
nodes = [nodes]
|
||||||
|
# if multiple nodes are passed in, we need to reverse the order in
|
||||||
|
# order to traverse front-to-back rather than the other way around
|
||||||
|
nodes = list(reversed(nodes))
|
||||||
|
|
||||||
|
while nodes:
|
||||||
|
node = nodes.pop()
|
||||||
|
# should probably filter None descendants in _children...
|
||||||
|
if node is None:
|
||||||
|
continue
|
||||||
|
node_type = node['type']
|
||||||
|
if node_type == '_exit':
|
||||||
|
node = node['node']
|
||||||
|
getattr(self, 'exit_' + node['type'], self.exit_generic)(node)
|
||||||
|
continue
|
||||||
|
|
||||||
|
if getattr(self, 'enter_' + node_type, self.enter_generic)(node) is SKIP:
|
||||||
|
continue
|
||||||
|
|
||||||
|
nodes.append({'type': '_exit', 'node': node})
|
||||||
|
nodes.extend(reversed(_children[node_type](node)))
|
||||||
|
|
||||||
|
return self.result
|
||||||
|
|
24
_extensions/exercise_admonition.py
Normal file
24
_extensions/exercise_admonition.py
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
"""
|
||||||
|
Adds a new "exercise" admonition type
|
||||||
|
"""
|
||||||
|
|
||||||
|
def setup(app):
|
||||||
|
app.add_directive('exercise', Exercise)
|
||||||
|
app.add_node(exercise, html=(
|
||||||
|
lambda self, node: self.visit_admonition(node, 'exercise'),
|
||||||
|
lambda self, node: self.depart_admonition(node)
|
||||||
|
), latex=(
|
||||||
|
lambda self, node: self.visit_admonition(node),
|
||||||
|
lambda self, node: self.depart_admonition(node)
|
||||||
|
))
|
||||||
|
|
||||||
|
from docutils import nodes
|
||||||
|
from docutils.parsers.rst.directives import admonitions
|
||||||
|
class exercise(nodes.Admonition, nodes.Element): pass
|
||||||
|
class Exercise(admonitions.BaseAdmonition):
|
||||||
|
node_class = exercise
|
||||||
|
|
||||||
|
from sphinx.locale import admonitionlabels
|
||||||
|
admonitionlabels['exercise'] = 'Exercise'
|
@ -1,8 +1,8 @@
|
|||||||
import inspect
|
import inspect
|
||||||
import importlib
|
import importlib
|
||||||
import os.path
|
import os.path
|
||||||
import werkzeug
|
|
||||||
|
|
||||||
|
from werkzeug import urls
|
||||||
|
|
||||||
"""
|
"""
|
||||||
* adds github_link(mode) context variable: provides URL (in relevant mode) of
|
* adds github_link(mode) context variable: provides URL (in relevant mode) of
|
||||||
@ -22,7 +22,7 @@ Notes
|
|||||||
|
|
||||||
* provided ``linkcode_resolve`` only supports Python domain
|
* provided ``linkcode_resolve`` only supports Python domain
|
||||||
* generates https github links
|
* generates https github links
|
||||||
* explicitly imports ``openerp``, so useless for anyone else
|
* explicitly imports ``odoo``, so useless for anyone else
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def setup(app):
|
def setup(app):
|
||||||
@ -63,9 +63,9 @@ def setup(app):
|
|||||||
# obj doesn't have a module, or something
|
# obj doesn't have a module, or something
|
||||||
return None
|
return None
|
||||||
|
|
||||||
import openerp
|
import odoo
|
||||||
# FIXME: make finding project root project-independent
|
# FIXME: make finding project root project-independent
|
||||||
project_root = os.path.join(os.path.dirname(openerp.__file__), '..')
|
project_root = os.path.join(os.path.dirname(odoo.__file__), '..')
|
||||||
return make_github_link(
|
return make_github_link(
|
||||||
app,
|
app,
|
||||||
os.path.relpath(obj_source_path, project_root),
|
os.path.relpath(obj_source_path, project_root),
|
||||||
@ -82,25 +82,24 @@ def make_github_link(app, path, line=None, mode="blob"):
|
|||||||
path=path,
|
path=path,
|
||||||
mode=mode,
|
mode=mode,
|
||||||
)
|
)
|
||||||
return werkzeug.urls.url_unparse((
|
return urls.url_unparse((
|
||||||
'https',
|
'https',
|
||||||
'github.com',
|
'github.com',
|
||||||
urlpath,
|
urlpath,
|
||||||
'',
|
'',
|
||||||
'' if line is None else 'L%d' % line
|
'' if line is None else 'L%d' % line
|
||||||
))
|
))
|
||||||
|
|
||||||
def add_doc_link(app, pagename, templatename, context, doctree):
|
def add_doc_link(app, pagename, templatename, context, doctree):
|
||||||
""" Add github_link function linking to the current page on github """
|
""" Add github_link function linking to the current page on github """
|
||||||
if not app.config.github_user and app.config.github_project:
|
if not app.config.github_user and app.config.github_project:
|
||||||
return
|
return
|
||||||
|
|
||||||
# FIXME: find other way to recover current document's source suffix
|
|
||||||
# in Sphinx 1.3 it's possible to have mutliple source suffixes and that
|
|
||||||
# may be useful in the future
|
|
||||||
source_suffix = app.config.source_suffix
|
source_suffix = app.config.source_suffix
|
||||||
source_suffix = next(iter(source_suffix))
|
# in 1.3 source_suffix can be a list
|
||||||
# FIXME: odoo/odoo has a doc/ prefix which is incorrect for this
|
# in 1.8 source_suffix can be a mapping
|
||||||
# project, how to unify? Add new setting?
|
# FIXME: will break if we ever add support for !rst markdown documents maybe
|
||||||
|
if not isinstance(source_suffix, str):
|
||||||
|
source_suffix = next(iter(source_suffix))
|
||||||
|
# can't use functools.partial because 3rd positional is line not mode
|
||||||
context['github_link'] = lambda mode='edit': make_github_link(
|
context['github_link'] = lambda mode='edit': make_github_link(
|
||||||
app, '%s%s' % (pagename, source_suffix), mode=mode)
|
app, 'doc/%s%s' % (pagename, source_suffix), mode=mode)
|
||||||
|
@ -1,23 +0,0 @@
|
|||||||
{# warning: if doc structure change, these rules may have to change as well #}
|
|
||||||
|
|
||||||
{# ===== VARIABLES ====== #}
|
|
||||||
{% set master_doc_short_name = 'User Doc' %}
|
|
||||||
|
|
||||||
{% if pagename == master_doc %}
|
|
||||||
<li><a href="{{ pathto(master_doc) }}" class="active">{{ master_doc_short_name }}</a></li>
|
|
||||||
{% else %}
|
|
||||||
{% for parent in parents %}
|
|
||||||
{% if loop.length > 1%}
|
|
||||||
{% if loop.first %}
|
|
||||||
<li><a href="{{ pathto(master_doc) }}">{{ master_doc_short_name }}</a></li>
|
|
||||||
{% else %}
|
|
||||||
{% if loop.index == 2 %}
|
|
||||||
<li><a href="{{ parent.link|e }}">{{parent.title}}</a></li>
|
|
||||||
{% endif %}
|
|
||||||
{% endif %}
|
|
||||||
{% else %}
|
|
||||||
<li><a href="{{ pathto(master_doc) }}">{{ master_doc_short_name }}</a></li>
|
|
||||||
{% endif %}
|
|
||||||
{% endfor %}
|
|
||||||
<li class="active"><a href="#">{{ meta.get('main-title', title) }}</a></li>
|
|
||||||
{% endif %}
|
|
@ -1,364 +0,0 @@
|
|||||||
{% extends "basic/layout.html" %}
|
|
||||||
{% set html5_doctype = True %}
|
|
||||||
|
|
||||||
{%- block scripts %}
|
|
||||||
{{ super() }}
|
|
||||||
<script type="text/javascript" src="{{ pathto('_static/jquery.min.js', 1) }}"></script>
|
|
||||||
<script type="text/javascript" src="{{ pathto('_static/bootstrap.js', 1) }}"></script>
|
|
||||||
<script type="text/javascript" src="{{ pathto('_static/doc.js', 1) }}"></script>
|
|
||||||
<script type="text/javascript" src="{{ pathto('_static/jquery.noconflict.js', 1) }}"></script>
|
|
||||||
{%- endblock %}
|
|
||||||
|
|
||||||
{% set classes = [] %}
|
|
||||||
{% if pagename == master_doc %}
|
|
||||||
{% set classes = classes + ['index'] %}
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{% if 'code-column' in meta %}
|
|
||||||
{% set classes = classes + ['has_code_col'] %}
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{% if 'classes' in meta %}
|
|
||||||
{% set classes = classes + meta['classes'].split() %}
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{%- block linktags -%}
|
|
||||||
{% for code, url in language_codes %}
|
|
||||||
<link rel="alternate" hreflang="{{ code }}" href="{{ url }}" />
|
|
||||||
{%- endfor %}
|
|
||||||
<link rel="canonical" href="{{ canonical }}" />
|
|
||||||
{{ super() }}
|
|
||||||
{%- endblock -%}
|
|
||||||
|
|
||||||
{%- block sidebar1 -%}{%- endblock -%}
|
|
||||||
{%- block sidebar2 -%}{%- endblock -%}
|
|
||||||
{%- block relbar1 -%}{%- endblock -%}
|
|
||||||
{%- block relbar2 -%}{%- endblock -%}
|
|
||||||
|
|
||||||
{%- block footer -%}
|
|
||||||
{%- if google_analytics_key -%}
|
|
||||||
<script>
|
|
||||||
(function(i,s,o,g,r,a,m){i['GoogleAnalyticsObject']=r;i[r]=i[r]||function(){
|
|
||||||
(i[r].q=i[r].q||[]).push(arguments)},i[r].l=1*new Date();a=s.createElement(o),
|
|
||||||
m=s.getElementsByTagName(o)[0];a.async=1;a.src=g;m.parentNode.insertBefore(a,m)
|
|
||||||
})(window,document,'script','//www.google-analytics.com/analytics.js','ga');
|
|
||||||
|
|
||||||
ga('create', '{{ google_analytics_key }}', 'auto');
|
|
||||||
ga('set', 'anonymizeIp', true);
|
|
||||||
ga('send','pageview');
|
|
||||||
</script>
|
|
||||||
{%- endif -%}
|
|
||||||
{%- endblock -%}
|
|
||||||
|
|
||||||
{%- block header -%}
|
|
||||||
<header class="o_main_header o_has_sub_nav o_inverted {{ ' '.join(classes) }}">
|
|
||||||
<div class="o_main_header_main">
|
|
||||||
<a class="pull-left o_logo" href="/"></a>
|
|
||||||
<a href="#" class="o_mobile_menu_toggle visible-xs-block pull-right">
|
|
||||||
<span class="sr-only">Toggle navigation</span>
|
|
||||||
<span class="mdi-navigation-menu"></span>
|
|
||||||
</a>
|
|
||||||
<div class="o_header_buttons">
|
|
||||||
<a href="http://www.odoo.com/trial" class="btn btn-primary">Start Now</a>
|
|
||||||
</div>
|
|
||||||
<ul class="o_primary_nav">
|
|
||||||
<li class="dropdown">
|
|
||||||
<a href="#" class="dropdown-toggle">Apps</a>
|
|
||||||
<div class="dropdown-menu o_secondary_nav">
|
|
||||||
<div class="container">
|
|
||||||
<div class="row">
|
|
||||||
<div class="col-sm-3 o_website_apps">
|
|
||||||
<div class="o_nav_app_family">
|
|
||||||
<span></span> Websites
|
|
||||||
<div>Build great user experience</div>
|
|
||||||
</div>
|
|
||||||
<ul>
|
|
||||||
<li><a href="https://www.odoo.com/page/website-builder">Website Builder</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/e-commerce">eCommerce</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/blog-engine">Blogs</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/community-builder">Forums</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/learning-management-system">eLearning</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/live-chat">Live Chat</a></li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
<div class="col-sm-3 o_sale_apps">
|
|
||||||
<div class="o_nav_app_family">
|
|
||||||
<span></span> Sales
|
|
||||||
<div>Boost your success rate</div>
|
|
||||||
</div>
|
|
||||||
<ul>
|
|
||||||
<li><a href="https://www.odoo.com/page/sales">Sales</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/crm">CRM</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/billing">Invoicing</a></li>
|
|
||||||
<li class="dropdown">
|
|
||||||
<a href="#0" class="dropdown-toggle">Point of Sale</a>
|
|
||||||
<ul>
|
|
||||||
<li><a href="https://www.odoo.com/page/point-of-sale">Shops</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/pos-restaurant">Restaurants</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/point-of-sale-hardware">Hardware</a></li>
|
|
||||||
</ul>
|
|
||||||
</li>
|
|
||||||
<li><a href="https://www.odoo.com/page/subscriptions">Subscriptions</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/sign">Sign</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/rental">Rental</a></li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
<div class="col-sm-3 o_operation_apps">
|
|
||||||
<div class="o_nav_app_family">
|
|
||||||
<span></span> Operations
|
|
||||||
<div>It's all about efficiency</div>
|
|
||||||
</div>
|
|
||||||
<ul>
|
|
||||||
<li><a href="https://www.odoo.com/page/accounting/">Accounting</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/project-management/">Project</a></li>
|
|
||||||
<li class="dropdown">
|
|
||||||
<a href="#0" class="dropdown-toggle">Human Resources</a>
|
|
||||||
<ul>
|
|
||||||
<li><a href="https://www.odoo.com/page/referral">Referral</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/employees">Employees</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/expenses">Expenses</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/appraisal">Appraisal</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/fleet">Fleet</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/leaves">Time Off</a></li>
|
|
||||||
</ul>
|
|
||||||
</li>
|
|
||||||
<li><a href="https://www.odoo.com/page/warehouse">Inventory</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/purchase">Purchase</a></li>
|
|
||||||
<li class="dropdown">
|
|
||||||
<a href="#0" class="dropdown-toggle">Manufacturing</a>
|
|
||||||
<ul>
|
|
||||||
<li><a href="https://www.odoo.com/page/manufacturing">MRP</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/plm">PLM</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/maintenance">Maintenance</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/quality">Quality</a></li>
|
|
||||||
</ul>
|
|
||||||
</li>
|
|
||||||
<li><a href="https://www.odoo.com/page/helpdesk">Helpdesk</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/field-service-management">Field Service</a></li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
<div class="col-sm-3 o_productivity_apps">
|
|
||||||
<div class="o_nav_app_family">
|
|
||||||
<span></span> Productivity Tools
|
|
||||||
<div>Great Tools = Happy People</div>
|
|
||||||
</div>
|
|
||||||
<ul>
|
|
||||||
<li class="dropdown">
|
|
||||||
<a href="#0" class="dropdown-toggle">Communication</a>
|
|
||||||
<ul>
|
|
||||||
<li><a href="https://www.odoo.com/page/discuss">Discuss</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/discuss-groups">Mailing Lists</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/notes">Notes</a></li>
|
|
||||||
</ul>
|
|
||||||
</li>
|
|
||||||
<li><a href="https://www.odoo.com/page/timesheet">Timesheet</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/events">Events</a></li>
|
|
||||||
<li class="dropdown">
|
|
||||||
<a href="#0" class="dropdown-toggle">Marketing</a>
|
|
||||||
<ul>
|
|
||||||
<li><a href="https://www.odoo.com/page/marketing-automation">Automation</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/email-marketing">Email</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/social-marketing">Social</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/sms-marketing">SMS</a></li>
|
|
||||||
</ul>
|
|
||||||
</li>
|
|
||||||
<li><a href="https://www.odoo.com/page/survey">Survey</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/approval-workflow">Approvals</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/appointments">Appointments</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/documents">Documents</a></li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<a href="http://www.odoo.com/apps/modules" class="o_store_link"><i class="fa fa-cube fa-fw"></i> Third party apps</a>
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li><a href="https://www.odoo.com/page/tour">Tour</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/pricing">Pricing</a></li>
|
|
||||||
<li><a href="https://www.odoo.com/page/docs">Docs</a></li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
<nav class="navbar o_sub_nav">
|
|
||||||
<div class="container">
|
|
||||||
<div class="navbar-header visible-xs">
|
|
||||||
<button type="button" class="navbar-toggle collapsed text-left btn-block" data-toggle="collapse" data-target="#o_sub-menu" aria-expanded="false">
|
|
||||||
Navigate
|
|
||||||
<span class="mdi-hardware-keyboard-arrow-down pull-right"></span>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="collapse navbar-collapse" id="o_sub-menu">
|
|
||||||
<ol class="o_breadcrumb breadcrumb nav navbar-left">
|
|
||||||
{% block breadcrumb_desktop %}
|
|
||||||
{% include "breadcrumb_list.html" %}
|
|
||||||
{% endblock %}
|
|
||||||
</ol>
|
|
||||||
|
|
||||||
<div class="call-to-action navbar-right hidden-xs">
|
|
||||||
<a href="http://www.odoo.com/trial" class="btn btn-primary">Start Now</a>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<ul class="nav navbar-nav navbar-right">
|
|
||||||
{% if languages or versions %}
|
|
||||||
<li class="divider"></li>
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{% block sub_menu_desktop %}
|
|
||||||
{% include "sub-menu_list.html" %}
|
|
||||||
{% endblock %}
|
|
||||||
|
|
||||||
{% if languages or versions %}
|
|
||||||
<li class="divider"></li>
|
|
||||||
{% endif %}
|
|
||||||
</ul>
|
|
||||||
|
|
||||||
<ul class="navbar-nav navbar-right nav o_sub_nav_actions">
|
|
||||||
{% if pagename != master_doc %}
|
|
||||||
<li class="divider"></li>
|
|
||||||
{% endif%}
|
|
||||||
|
|
||||||
{% block switchers_desktop %}
|
|
||||||
{% include "switchers_list.html" %}
|
|
||||||
{% endblock %}
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</nav>
|
|
||||||
</header>
|
|
||||||
|
|
||||||
{%- endblock -%}
|
|
||||||
|
|
||||||
{%- block content -%}
|
|
||||||
<div id="wrap" class="{{' '.join(classes) }}">
|
|
||||||
{% if meta['banner'] %}
|
|
||||||
{% set has_banner = 'has_banner' %}
|
|
||||||
{% endif %}
|
|
||||||
<figure class="card top {{ has_banner }}">
|
|
||||||
<span class="card-img" {% if meta['banner'] %}style="background-image: url('{{ pathto('_static/' + meta['banner'], True) }}');"{% endif %}></span>
|
|
||||||
<div class="container text-center">
|
|
||||||
<h1> {{ meta.get('main-title', title) }} </h1>
|
|
||||||
</div>
|
|
||||||
</figure>
|
|
||||||
{% if 'code-column' in meta %}
|
|
||||||
{% set container = 'container-fluid' %}
|
|
||||||
{% else %}
|
|
||||||
{% set container = 'container' %}
|
|
||||||
{% endif %}
|
|
||||||
<main class="container {{ ' '.join(classes) }}">
|
|
||||||
{% if pagename != master_doc %}
|
|
||||||
<div class="o_content row">
|
|
||||||
{% if 'has-toc' not in meta and not (pagename in toc) %}
|
|
||||||
<aside>
|
|
||||||
<div class="navbar-aside text-center">
|
|
||||||
{{ toc }}
|
|
||||||
{% if github_link %}
|
|
||||||
<p class="gith-container"><a href="{{ github_link(mode='edit') }}" class="gith-link">
|
|
||||||
Edit on GitHub
|
|
||||||
</a></p>
|
|
||||||
{% endif %}
|
|
||||||
</div>
|
|
||||||
</aside>
|
|
||||||
{% endif %}
|
|
||||||
<article class="doc-body {% if 'has-toc' in meta %}doc-toc{% endif %}{% if pagename in toc%}index-category{% endif %}">
|
|
||||||
{% endif %}
|
|
||||||
{% block body %} {% endblock %}
|
|
||||||
{% if pagename != master_doc %}</article>
|
|
||||||
</div>
|
|
||||||
{% endif %}
|
|
||||||
<div id="mask"></div>
|
|
||||||
</main>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="floating_action_container">
|
|
||||||
<a id="floating_action" class="ripple" href="#">
|
|
||||||
<i class="mdi-action-explore"></i>
|
|
||||||
</a>
|
|
||||||
<div id="floating_action_menu">
|
|
||||||
<span class="bubble"></span>
|
|
||||||
<ul class="list-group content">
|
|
||||||
<li class="list-group-item ripple"><a>Cras justo odio</a></li>
|
|
||||||
<li class="list-group-item ripple"><a>Dapibus ac facilisis in</a></li>
|
|
||||||
<li class="list-group-item ripple"><a>Morbi leo risus</a></li>
|
|
||||||
<li class="list-group-item ripple"><a>Porta ac consectetur ac</a></li>
|
|
||||||
<li class="list-group-item ripple"><a>Vestibulum at eros</a></li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<footer>
|
|
||||||
<div id="footer" class="container">
|
|
||||||
<span class="o_logo o_logo_inverse center-block o_footer_logo"></span>
|
|
||||||
<div class="row">
|
|
||||||
<div class="col-sm-7 col-md-7 col-lg-6">
|
|
||||||
<div class="row">
|
|
||||||
<div class="col-xs-6 col-sm-4">
|
|
||||||
<span class="menu_title">Community</span>
|
|
||||||
<ul>
|
|
||||||
<li><a href="https://github.com/odoo/odoo">Github</a></li>
|
|
||||||
<li><a href="http://www.odoo.com/page/download">Download</a></li>
|
|
||||||
<li class="divider"></li>
|
|
||||||
<li><a href="http://runbot.odoo.com/runbot/repo/git-github-com-odoo-enterprise-7">Runbot</a></li>
|
|
||||||
<li><a href="https://github.com/odoo/odoo/wiki/Translations">Translations</a></li>
|
|
||||||
<li class="divider"></li>
|
|
||||||
<li><a href="http://www.odoo.com/page/odoo-community">Mailing Lists</a></li>
|
|
||||||
<li><a href="http://www.odoo.com/forum/help-1">Forum</a></li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
<div class="col-xs-6 col-sm-4">
|
|
||||||
<span class="menu_title">Services</span>
|
|
||||||
<ul>
|
|
||||||
<li><a href="https://www.odoo.sh">Odoo Cloud Platform</a></li>
|
|
||||||
<li class="divider"></li>
|
|
||||||
<li><a href="http://www.odoo.com/help">Support</a></li>
|
|
||||||
<li><a href="https://upgrade.odoo.com">Upgrade</a></li>
|
|
||||||
<li class="divider"></li>
|
|
||||||
<li><a href="http://www.odoo.com/partners">Find a partner</a></li>
|
|
||||||
<li><a href="http://www.odoo.com/page/become-a-partner">Become a partner</a></li>
|
|
||||||
<li class="divider"></li>
|
|
||||||
<li><a href="http://training.odoo.com/courses/odoo-functional">Training Center</a></li>
|
|
||||||
<li><a href="http://www.odoo.com/page/education-program">Education</a></li>
|
|
||||||
<li class="divider"></li>
|
|
||||||
<li><a href="http://www.odoo.com/page/security">Security</a></li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
<div class="col-xs-12 col-sm-4 mb64">
|
|
||||||
<span class="menu_title">About us</span>
|
|
||||||
<ul>
|
|
||||||
<li><a href="http://www.odoo.com/page/about-us">Our company</a></li>
|
|
||||||
<li><a href="http://www.odoo.com/page/contactus">Contact</a></li>
|
|
||||||
<li class="divider" />
|
|
||||||
<li><a href="http://www.odoo.com/event">Events</a></li>
|
|
||||||
<li><a href="http://www.odoo.com/blog">Blog</a></li>
|
|
||||||
<li><a href="http://www.odoo.com/blog/6">Customers</a></li>
|
|
||||||
<li class="divider" />
|
|
||||||
<li><a href="http://www.odoo.com/jobs">Jobs</a></li>
|
|
||||||
<li class="divider" />
|
|
||||||
<li><a href="http://www.odoo.com/page/legal">Legal</a> | <a href="http://www.odoo.com/privacy">Privacy</a></li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="col-sm-5 col-md-4 col-md-offset-1 col-lg-5 col-lg-offset-1">
|
|
||||||
<p>
|
|
||||||
<small>
|
|
||||||
Odoo is a suite of open source business apps that cover all your company needs: CRM, eCommerce, accounting, inventory, point of sale, project management, etc.
|
|
||||||
<br/><br/>
|
|
||||||
Odoo's unique value proposition is to be at the same time very easy to use and fully integrated.
|
|
||||||
</small>
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="o_footer_bottom">
|
|
||||||
<div class="container">
|
|
||||||
<a class="small" href="http://www.odoo.com/page/website-builder">Website made with <span class="o_logo o_logo_inverse o_logo_15"></span></a>
|
|
||||||
<div class="social-links pull-right">
|
|
||||||
<a href="http://www.odoo.com/web/about/facebook"><i class="fa fa-facebook"></i></a>
|
|
||||||
<a href="http://www.odoo.com/web/about/twitter"><i class="fa fa-twitter"></i></a>
|
|
||||||
<a href="http://www.odoo.com/web/about/linkedin"><i class="fa fa-linkedin"></i></a>
|
|
||||||
<a href="mailto:info@odoo.com"><i class="fa fa-envelope"></i></a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</footer>
|
|
||||||
{%- endblock -%}
|
|
||||||
|
|
@ -1,304 +0,0 @@
|
|||||||
{
|
|
||||||
"always-semicolon": true,
|
|
||||||
"block-indent": 2,
|
|
||||||
"color-case": "lower",
|
|
||||||
"color-shorthand": true,
|
|
||||||
"element-case": "lower",
|
|
||||||
"eof-newline": true,
|
|
||||||
"leading-zero": false,
|
|
||||||
"remove-empty-rulesets": true,
|
|
||||||
"space-after-colon": 1,
|
|
||||||
"space-after-combinator": 1,
|
|
||||||
"space-before-selector-delimiter": 0,
|
|
||||||
"space-between-declarations": "\n",
|
|
||||||
"space-after-opening-brace": "\n",
|
|
||||||
"space-before-closing-brace": "\n",
|
|
||||||
"space-before-colon": 0,
|
|
||||||
"space-before-combinator": 1,
|
|
||||||
"space-before-opening-brace": 1,
|
|
||||||
"strip-spaces": true,
|
|
||||||
"unitless-zero": true,
|
|
||||||
"vendor-prefix-align": true,
|
|
||||||
"sort-order": [
|
|
||||||
[
|
|
||||||
"position",
|
|
||||||
"top",
|
|
||||||
"right",
|
|
||||||
"bottom",
|
|
||||||
"left",
|
|
||||||
"z-index",
|
|
||||||
"display",
|
|
||||||
"float",
|
|
||||||
"width",
|
|
||||||
"min-width",
|
|
||||||
"max-width",
|
|
||||||
"height",
|
|
||||||
"min-height",
|
|
||||||
"max-height",
|
|
||||||
"-webkit-box-sizing",
|
|
||||||
"-moz-box-sizing",
|
|
||||||
"box-sizing",
|
|
||||||
"-webkit-appearance",
|
|
||||||
"padding",
|
|
||||||
"padding-top",
|
|
||||||
"padding-right",
|
|
||||||
"padding-bottom",
|
|
||||||
"padding-left",
|
|
||||||
"margin",
|
|
||||||
"margin-top",
|
|
||||||
"margin-right",
|
|
||||||
"margin-bottom",
|
|
||||||
"margin-left",
|
|
||||||
"overflow",
|
|
||||||
"overflow-x",
|
|
||||||
"overflow-y",
|
|
||||||
"-webkit-overflow-scrolling",
|
|
||||||
"-ms-overflow-x",
|
|
||||||
"-ms-overflow-y",
|
|
||||||
"-ms-overflow-style",
|
|
||||||
"clip",
|
|
||||||
"clear",
|
|
||||||
"font",
|
|
||||||
"font-family",
|
|
||||||
"font-size",
|
|
||||||
"font-style",
|
|
||||||
"font-weight",
|
|
||||||
"font-variant",
|
|
||||||
"font-size-adjust",
|
|
||||||
"font-stretch",
|
|
||||||
"font-effect",
|
|
||||||
"font-emphasize",
|
|
||||||
"font-emphasize-position",
|
|
||||||
"font-emphasize-style",
|
|
||||||
"font-smooth",
|
|
||||||
"-webkit-hyphens",
|
|
||||||
"-moz-hyphens",
|
|
||||||
"hyphens",
|
|
||||||
"line-height",
|
|
||||||
"color",
|
|
||||||
"text-align",
|
|
||||||
"-webkit-text-align-last",
|
|
||||||
"-moz-text-align-last",
|
|
||||||
"-ms-text-align-last",
|
|
||||||
"text-align-last",
|
|
||||||
"text-emphasis",
|
|
||||||
"text-emphasis-color",
|
|
||||||
"text-emphasis-style",
|
|
||||||
"text-emphasis-position",
|
|
||||||
"text-decoration",
|
|
||||||
"text-indent",
|
|
||||||
"text-justify",
|
|
||||||
"text-outline",
|
|
||||||
"-ms-text-overflow",
|
|
||||||
"text-overflow",
|
|
||||||
"text-overflow-ellipsis",
|
|
||||||
"text-overflow-mode",
|
|
||||||
"text-shadow",
|
|
||||||
"text-transform",
|
|
||||||
"text-wrap",
|
|
||||||
"-webkit-text-size-adjust",
|
|
||||||
"-ms-text-size-adjust",
|
|
||||||
"letter-spacing",
|
|
||||||
"-ms-word-break",
|
|
||||||
"word-break",
|
|
||||||
"word-spacing",
|
|
||||||
"-ms-word-wrap",
|
|
||||||
"word-wrap",
|
|
||||||
"-moz-tab-size",
|
|
||||||
"-o-tab-size",
|
|
||||||
"tab-size",
|
|
||||||
"white-space",
|
|
||||||
"vertical-align",
|
|
||||||
"list-style",
|
|
||||||
"list-style-position",
|
|
||||||
"list-style-type",
|
|
||||||
"list-style-image",
|
|
||||||
"pointer-events",
|
|
||||||
"-ms-touch-action",
|
|
||||||
"touch-action",
|
|
||||||
"cursor",
|
|
||||||
"visibility",
|
|
||||||
"zoom",
|
|
||||||
"flex-direction",
|
|
||||||
"flex-order",
|
|
||||||
"flex-pack",
|
|
||||||
"flex-align",
|
|
||||||
"table-layout",
|
|
||||||
"empty-cells",
|
|
||||||
"caption-side",
|
|
||||||
"border-spacing",
|
|
||||||
"border-collapse",
|
|
||||||
"content",
|
|
||||||
"quotes",
|
|
||||||
"counter-reset",
|
|
||||||
"counter-increment",
|
|
||||||
"resize",
|
|
||||||
"-webkit-user-select",
|
|
||||||
"-moz-user-select",
|
|
||||||
"-ms-user-select",
|
|
||||||
"-o-user-select",
|
|
||||||
"user-select",
|
|
||||||
"nav-index",
|
|
||||||
"nav-up",
|
|
||||||
"nav-right",
|
|
||||||
"nav-down",
|
|
||||||
"nav-left",
|
|
||||||
"background",
|
|
||||||
"background-color",
|
|
||||||
"background-image",
|
|
||||||
"-ms-filter:\\'progid:DXImageTransform.Microsoft.gradient",
|
|
||||||
"filter:progid:DXImageTransform.Microsoft.gradient",
|
|
||||||
"filter:progid:DXImageTransform.Microsoft.AlphaImageLoader",
|
|
||||||
"filter",
|
|
||||||
"background-repeat",
|
|
||||||
"background-attachment",
|
|
||||||
"background-position",
|
|
||||||
"background-position-x",
|
|
||||||
"background-position-y",
|
|
||||||
"-webkit-background-clip",
|
|
||||||
"-moz-background-clip",
|
|
||||||
"background-clip",
|
|
||||||
"background-origin",
|
|
||||||
"-webkit-background-size",
|
|
||||||
"-moz-background-size",
|
|
||||||
"-o-background-size",
|
|
||||||
"background-size",
|
|
||||||
"border",
|
|
||||||
"border-color",
|
|
||||||
"border-style",
|
|
||||||
"border-width",
|
|
||||||
"border-top",
|
|
||||||
"border-top-color",
|
|
||||||
"border-top-style",
|
|
||||||
"border-top-width",
|
|
||||||
"border-right",
|
|
||||||
"border-right-color",
|
|
||||||
"border-right-style",
|
|
||||||
"border-right-width",
|
|
||||||
"border-bottom",
|
|
||||||
"border-bottom-color",
|
|
||||||
"border-bottom-style",
|
|
||||||
"border-bottom-width",
|
|
||||||
"border-left",
|
|
||||||
"border-left-color",
|
|
||||||
"border-left-style",
|
|
||||||
"border-left-width",
|
|
||||||
"border-radius",
|
|
||||||
"border-top-left-radius",
|
|
||||||
"border-top-right-radius",
|
|
||||||
"border-bottom-right-radius",
|
|
||||||
"border-bottom-left-radius",
|
|
||||||
"-webkit-border-image",
|
|
||||||
"-moz-border-image",
|
|
||||||
"-o-border-image",
|
|
||||||
"border-image",
|
|
||||||
"-webkit-border-image-source",
|
|
||||||
"-moz-border-image-source",
|
|
||||||
"-o-border-image-source",
|
|
||||||
"border-image-source",
|
|
||||||
"-webkit-border-image-slice",
|
|
||||||
"-moz-border-image-slice",
|
|
||||||
"-o-border-image-slice",
|
|
||||||
"border-image-slice",
|
|
||||||
"-webkit-border-image-width",
|
|
||||||
"-moz-border-image-width",
|
|
||||||
"-o-border-image-width",
|
|
||||||
"border-image-width",
|
|
||||||
"-webkit-border-image-outset",
|
|
||||||
"-moz-border-image-outset",
|
|
||||||
"-o-border-image-outset",
|
|
||||||
"border-image-outset",
|
|
||||||
"-webkit-border-image-repeat",
|
|
||||||
"-moz-border-image-repeat",
|
|
||||||
"-o-border-image-repeat",
|
|
||||||
"border-image-repeat",
|
|
||||||
"outline",
|
|
||||||
"outline-width",
|
|
||||||
"outline-style",
|
|
||||||
"outline-color",
|
|
||||||
"outline-offset",
|
|
||||||
"-webkit-box-shadow",
|
|
||||||
"-moz-box-shadow",
|
|
||||||
"box-shadow",
|
|
||||||
"filter:progid:DXImageTransform.Microsoft.Alpha(Opacity",
|
|
||||||
"-ms-filter:\\'progid:DXImageTransform.Microsoft.Alpha",
|
|
||||||
"opacity",
|
|
||||||
"-ms-interpolation-mode",
|
|
||||||
"-webkit-transition",
|
|
||||||
"-moz-transition",
|
|
||||||
"-ms-transition",
|
|
||||||
"-o-transition",
|
|
||||||
"transition",
|
|
||||||
"-webkit-transition-delay",
|
|
||||||
"-moz-transition-delay",
|
|
||||||
"-ms-transition-delay",
|
|
||||||
"-o-transition-delay",
|
|
||||||
"transition-delay",
|
|
||||||
"-webkit-transition-timing-function",
|
|
||||||
"-moz-transition-timing-function",
|
|
||||||
"-ms-transition-timing-function",
|
|
||||||
"-o-transition-timing-function",
|
|
||||||
"transition-timing-function",
|
|
||||||
"-webkit-transition-duration",
|
|
||||||
"-moz-transition-duration",
|
|
||||||
"-ms-transition-duration",
|
|
||||||
"-o-transition-duration",
|
|
||||||
"transition-duration",
|
|
||||||
"-webkit-transition-property",
|
|
||||||
"-moz-transition-property",
|
|
||||||
"-ms-transition-property",
|
|
||||||
"-o-transition-property",
|
|
||||||
"transition-property",
|
|
||||||
"-webkit-transform",
|
|
||||||
"-moz-transform",
|
|
||||||
"-ms-transform",
|
|
||||||
"-o-transform",
|
|
||||||
"transform",
|
|
||||||
"-webkit-transform-origin",
|
|
||||||
"-moz-transform-origin",
|
|
||||||
"-ms-transform-origin",
|
|
||||||
"-o-transform-origin",
|
|
||||||
"transform-origin",
|
|
||||||
"-webkit-animation",
|
|
||||||
"-moz-animation",
|
|
||||||
"-ms-animation",
|
|
||||||
"-o-animation",
|
|
||||||
"animation",
|
|
||||||
"-webkit-animation-name",
|
|
||||||
"-moz-animation-name",
|
|
||||||
"-ms-animation-name",
|
|
||||||
"-o-animation-name",
|
|
||||||
"animation-name",
|
|
||||||
"-webkit-animation-duration",
|
|
||||||
"-moz-animation-duration",
|
|
||||||
"-ms-animation-duration",
|
|
||||||
"-o-animation-duration",
|
|
||||||
"animation-duration",
|
|
||||||
"-webkit-animation-play-state",
|
|
||||||
"-moz-animation-play-state",
|
|
||||||
"-ms-animation-play-state",
|
|
||||||
"-o-animation-play-state",
|
|
||||||
"animation-play-state",
|
|
||||||
"-webkit-animation-timing-function",
|
|
||||||
"-moz-animation-timing-function",
|
|
||||||
"-ms-animation-timing-function",
|
|
||||||
"-o-animation-timing-function",
|
|
||||||
"animation-timing-function",
|
|
||||||
"-webkit-animation-delay",
|
|
||||||
"-moz-animation-delay",
|
|
||||||
"-ms-animation-delay",
|
|
||||||
"-o-animation-delay",
|
|
||||||
"animation-delay",
|
|
||||||
"-webkit-animation-iteration-count",
|
|
||||||
"-moz-animation-iteration-count",
|
|
||||||
"-ms-animation-iteration-count",
|
|
||||||
"-o-animation-iteration-count",
|
|
||||||
"animation-iteration-count",
|
|
||||||
"-webkit-animation-direction",
|
|
||||||
"-moz-animation-direction",
|
|
||||||
"-ms-animation-direction",
|
|
||||||
"-o-animation-direction",
|
|
||||||
"animation-direction"
|
|
||||||
]
|
|
||||||
]
|
|
||||||
}
|
|
@ -1,19 +0,0 @@
|
|||||||
{
|
|
||||||
"adjoining-classes": false,
|
|
||||||
"box-sizing": false,
|
|
||||||
"box-model": false,
|
|
||||||
"compatible-vendor-prefixes": false,
|
|
||||||
"floats": false,
|
|
||||||
"font-sizes": false,
|
|
||||||
"gradients": false,
|
|
||||||
"important": false,
|
|
||||||
"known-properties": false,
|
|
||||||
"outline-none": false,
|
|
||||||
"qualified-headings": false,
|
|
||||||
"regex-selectors": false,
|
|
||||||
"shorthand": false,
|
|
||||||
"text-indent": false,
|
|
||||||
"unique-headings": false,
|
|
||||||
"universal-selector": false,
|
|
||||||
"unqualified-attributes": false
|
|
||||||
}
|
|
@ -21,6 +21,10 @@ def setup(app):
|
|||||||
location="odoo extension")
|
location="odoo extension")
|
||||||
app.config.html_translator_class = 'odoo.translator.BootstrapTranslator'
|
app.config.html_translator_class = 'odoo.translator.BootstrapTranslator'
|
||||||
|
|
||||||
|
add_js_file = getattr(app, 'add_js_file', None) or app.add_javascript
|
||||||
|
for f in ['jquery.min.js', 'bootstrap.js', 'doc.js', 'jquery.noconflict.js']:
|
||||||
|
add_js_file(f)
|
||||||
|
|
||||||
switcher.setup(app)
|
switcher.setup(app)
|
||||||
app.add_config_value('odoo_cover_default', None, 'env')
|
app.add_config_value('odoo_cover_default', None, 'env')
|
||||||
app.add_config_value('odoo_cover_external', {}, 'env')
|
app.add_config_value('odoo_cover_external', {}, 'env')
|
||||||
@ -28,9 +32,9 @@ def setup(app):
|
|||||||
app.connect('html-page-context', update_meta)
|
app.connect('html-page-context', update_meta)
|
||||||
|
|
||||||
def update_meta(app, pagename, templatename, context, doctree):
|
def update_meta(app, pagename, templatename, context, doctree):
|
||||||
if not context.get('meta'): # context['meta'] can be None
|
meta = context.get('meta')
|
||||||
context['meta'] = {}
|
if meta is None:
|
||||||
meta = context.setdefault('meta', {}) # we want {} by default
|
meta = context['meta'] = {}
|
||||||
meta.setdefault('banner', app.config.odoo_cover_default)
|
meta.setdefault('banner', app.config.odoo_cover_default)
|
||||||
|
|
||||||
def navbarify(node, navbar=None):
|
def navbarify(node, navbar=None):
|
||||||
@ -105,7 +109,7 @@ if toctree:
|
|||||||
# than functions on the BuildEnv & al
|
# than functions on the BuildEnv & al
|
||||||
@monkey(toctree.TocTree)
|
@monkey(toctree.TocTree)
|
||||||
def resolve(old_resolve, tree, docname, *args, **kwargs):
|
def resolve(old_resolve, tree, docname, *args, **kwargs):
|
||||||
if docname == tree.env.config.master_doc:
|
if docname in tree.env.config.banners_doc:
|
||||||
return resolve_content_toctree(tree.env, docname, *args, **kwargs)
|
return resolve_content_toctree(tree.env, docname, *args, **kwargs)
|
||||||
toc = old_resolve(tree, docname, *args, **kwargs)
|
toc = old_resolve(tree, docname, *args, **kwargs)
|
||||||
if toc is None:
|
if toc is None:
|
||||||
@ -120,6 +124,8 @@ def resolve_toctree(old_resolve, self, docname, *args, **kwargs):
|
|||||||
""" If navbar, bootstrapify TOC to yield a navbar
|
""" If navbar, bootstrapify TOC to yield a navbar
|
||||||
|
|
||||||
"""
|
"""
|
||||||
|
# VFE NOTE not called since sphinx 1.6
|
||||||
|
# bump the version and remove ?
|
||||||
navbar = kwargs.pop('navbar', None)
|
navbar = kwargs.pop('navbar', None)
|
||||||
if docname == self.config.master_doc and not navbar:
|
if docname == self.config.master_doc and not navbar:
|
||||||
return resolve_content_toctree(self, docname, *args, **kwargs)
|
return resolve_content_toctree(self, docname, *args, **kwargs)
|
20
_extensions/odoo_ext/breadcrumb_list.html
Normal file
20
_extensions/odoo_ext/breadcrumb_list.html
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
{# warning: if doc structure change, these rules may have to change as well #}
|
||||||
|
|
||||||
|
{# ===== VARIABLES ====== #}
|
||||||
|
{% set master_doc_short_name = 'Documentation' %}
|
||||||
|
<!-- VFE TODO use value from config ? -->
|
||||||
|
|
||||||
|
{% if pagename == master_doc %}
|
||||||
|
<li><a href="{{ pathto(master_doc) }}" class="active">{{ master_doc_short_name }}</a></li>
|
||||||
|
{% else %}
|
||||||
|
{# Do not show main TOC link when in user/dev doc subpages to shorten breadcrumb links #}
|
||||||
|
{% if 'show_main_toc_link' in meta %}
|
||||||
|
<li><a href="{{ pathto(master_doc) }}">{{ master_doc_short_name }}</a></li>
|
||||||
|
{% endif %}
|
||||||
|
{% for parent in parents %}
|
||||||
|
<li><a href="{{ parent.link|e }}">{{ parent.title }}</a></li>
|
||||||
|
{% endfor %}
|
||||||
|
<li class="active"><a href="#">{{ meta.get('main-title', title) }}</a></li>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
<!-- VFE TODO integrate "next" page logic to allow better navigation -->
|
177
_extensions/odoo_ext/layout.html
Normal file
177
_extensions/odoo_ext/layout.html
Normal file
@ -0,0 +1,177 @@
|
|||||||
|
{% extends "basic/layout.html" %}
|
||||||
|
{% set html5_doctype = True %}
|
||||||
|
|
||||||
|
{# ===== VARIABLES ====== #}
|
||||||
|
{% set banners_doc = 'user/index' %}
|
||||||
|
|
||||||
|
{% set classes = [] %}
|
||||||
|
{% if pagename == master_doc or pagename == banners_doc %}
|
||||||
|
{% set classes = classes + ['index'] %}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if 'code-column' in meta %}
|
||||||
|
{% set classes = classes + ['has_code_col'] %}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if 'classes' in meta %}
|
||||||
|
{% set classes = classes + meta['classes'].split() %}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{%- block linktags -%}
|
||||||
|
{% for code, url in language_codes %}
|
||||||
|
<link rel="alternate" hreflang="{{ code }}" href="{{ url }}" />
|
||||||
|
{%- endfor %}
|
||||||
|
<link rel="canonical" href="{{ canonical }}" />
|
||||||
|
{{ super() }}
|
||||||
|
{%- endblock -%}
|
||||||
|
|
||||||
|
{%- block sidebar1 -%}{%- endblock -%}
|
||||||
|
{%- block sidebar2 -%}{%- endblock -%}
|
||||||
|
{%- block relbar1 -%}{%- endblock -%}
|
||||||
|
{%- block relbar2 -%}{%- endblock -%}
|
||||||
|
|
||||||
|
{%- block footer -%}
|
||||||
|
{%- if google_analytics_key -%}
|
||||||
|
<script>
|
||||||
|
(function(i,s,o,g,r,a,m){i['GoogleAnalyticsObject']=r;i[r]=i[r]||function(){
|
||||||
|
(i[r].q=i[r].q||[]).push(arguments)},i[r].l=1*new Date();a=s.createElement(o),
|
||||||
|
m=s.getElementsByTagName(o)[0];a.async=1;a.src=g;m.parentNode.insertBefore(a,m)
|
||||||
|
})(window,document,'script','//www.google-analytics.com/analytics.js','ga');
|
||||||
|
|
||||||
|
ga('create', '{{ google_analytics_key }}', 'auto');
|
||||||
|
ga('set', 'anonymizeIp', true);
|
||||||
|
ga('send','pageview');
|
||||||
|
</script>
|
||||||
|
{%- endif -%}
|
||||||
|
{%- endblock -%}
|
||||||
|
|
||||||
|
{%- block header -%}
|
||||||
|
<header class="o_main_header o_has_sub_nav o_inverted {{ ' '.join(classes) }}">
|
||||||
|
{% include "odoo_header.html" %}
|
||||||
|
<nav class="navbar o_sub_nav">
|
||||||
|
<div class="container">
|
||||||
|
<div class="navbar-header visible-xs">
|
||||||
|
<button type="button" class="navbar-toggle collapsed text-left btn-block" data-toggle="collapse" data-target="#o_sub-menu" aria-expanded="false">
|
||||||
|
Navigate
|
||||||
|
<span class="mdi-hardware-keyboard-arrow-down pull-right"></span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="collapse navbar-collapse" id="o_sub-menu">
|
||||||
|
<ol class="o_breadcrumb breadcrumb nav navbar-left">
|
||||||
|
{% block breadcrumb_desktop %}
|
||||||
|
{% include "breadcrumb_list.html" %}
|
||||||
|
{% endblock %}
|
||||||
|
</ol>
|
||||||
|
|
||||||
|
<div class="call-to-action navbar-right hidden-xs">
|
||||||
|
<a href="https://www.odoo.com/trial" class="btn btn-primary">Start Now</a>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- VFE TODO remove/reduce master_doc logic in html templates
|
||||||
|
use the banners / other logic instead ?
|
||||||
|
-->
|
||||||
|
<ul class="navbar-nav navbar-right nav o_sub_nav_actions">
|
||||||
|
{% if pagename != master_doc %}
|
||||||
|
<li class="divider"></li>
|
||||||
|
{% endif%}
|
||||||
|
|
||||||
|
{% block switchers_desktop %}
|
||||||
|
{% include "switchers_list.html" %}
|
||||||
|
{% endblock %}
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<ul class="nav navbar-nav navbar-right">
|
||||||
|
{% if languages or versions %}
|
||||||
|
<li class="divider"></li>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% block sub_menu_desktop %}
|
||||||
|
{% include "sub-menu_list.html" %}
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% if languages or versions %}
|
||||||
|
<li class="divider"></li>
|
||||||
|
{% endif %}
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</nav>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
{%- endblock -%}
|
||||||
|
|
||||||
|
{%- block content -%}
|
||||||
|
<div id="wrap" class="{{' '.join(classes) }}">
|
||||||
|
{% if meta['banner'] %}
|
||||||
|
{% set has_banner = 'has_banner' %}
|
||||||
|
{% endif %}
|
||||||
|
<figure class="card top {{ has_banner }}">
|
||||||
|
<span class="card-img" {% if meta['banner'] %}style="background-image: url('{{ pathto('_static/' + meta['banner'], True) }}');"{% endif %}></span>
|
||||||
|
<div class="container text-center">
|
||||||
|
<h1> {{ meta.get('main-title', title) }} </h1>
|
||||||
|
</div>
|
||||||
|
</figure>
|
||||||
|
{% if 'code-column' in meta %}
|
||||||
|
{% set container = 'container-fluid' %}
|
||||||
|
{% else %}
|
||||||
|
{% set container = 'container' %}
|
||||||
|
{% endif %}
|
||||||
|
<main class="container {{ ' '.join(classes) }}">
|
||||||
|
{% if pagename == master_doc %}
|
||||||
|
<div class="o_content row">
|
||||||
|
<article class="doc-body {% if 'has-toc' in meta %}doc-toc{% endif %}{% if pagename in toc%}index-category{% endif %}">
|
||||||
|
HOHOHOHO
|
||||||
|
<!-- {{ toc }} -->
|
||||||
|
{{ body }}
|
||||||
|
</article>
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
{% if 'banners-display' not in meta %}
|
||||||
|
<div class="o_content row">
|
||||||
|
{% if 'has-toc' not in meta and not (pagename in toc) %}
|
||||||
|
<aside>
|
||||||
|
<div class="navbar-aside text-center">
|
||||||
|
{{ toc }}
|
||||||
|
{% if github_link %}
|
||||||
|
<p class="gith-container">
|
||||||
|
<a href="{{ github_link(mode='edit') }}" class="gith-link">
|
||||||
|
Edit on GitHub
|
||||||
|
</a>
|
||||||
|
</p>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</aside>
|
||||||
|
{% endif %}
|
||||||
|
<article class="doc-body {% if 'has-toc' in meta %}doc-toc{% endif %}{% if pagename in toc%}index-category{% endif %}">
|
||||||
|
{% endif %}
|
||||||
|
{% block body %} {% endblock %}
|
||||||
|
{% if 'banners-display' not in meta %}
|
||||||
|
</article>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
{% endif %}
|
||||||
|
<div id="mask"></div>
|
||||||
|
</main>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="floating_action_container">
|
||||||
|
<a id="floating_action" class="ripple" href="#">
|
||||||
|
<i class="mdi-action-explore"></i>
|
||||||
|
</a>
|
||||||
|
<div id="floating_action_menu">
|
||||||
|
<span class="bubble"></span>
|
||||||
|
<ul class="list-group content">
|
||||||
|
<!-- VFE FIXME wtf is this kinda lorem ipsum binz ? -->
|
||||||
|
<li class="list-group-item ripple"><a>Cras justo odio</a></li>
|
||||||
|
<li class="list-group-item ripple"><a>Dapibus ac facilisis in</a></li>
|
||||||
|
<li class="list-group-item ripple"><a>Morbi leo risus</a></li>
|
||||||
|
<li class="list-group-item ripple"><a>Porta ac consectetur ac</a></li>
|
||||||
|
<li class="list-group-item ripple"><a>Vestibulum at eros</a></li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<footer>
|
||||||
|
{% include "odoo_footer.html" %}
|
||||||
|
</footer>
|
||||||
|
{%- endblock -%}
|
77
_extensions/odoo_ext/odoo_footer.html
Normal file
77
_extensions/odoo_ext/odoo_footer.html
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
<div id="footer" class="container">
|
||||||
|
<span class="o_logo o_logo_inverse center-block o_footer_logo"></span>
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-sm-7 col-md-7 col-lg-6">
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-xs-6 col-sm-4">
|
||||||
|
<span class="menu_title">Community</span>
|
||||||
|
<ul>
|
||||||
|
<li><a href="https://github.com/odoo/odoo">Github</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/download">Download</a></li>
|
||||||
|
<li class="divider"></li>
|
||||||
|
<li><a href="https://runbot.odoo.com/">Runbot</a></li>
|
||||||
|
<li><a href="https://github.com/odoo/odoo/wiki/Translations">Translations</a></li>
|
||||||
|
<li class="divider"></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/odoo-community">Mailing Lists</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/forum/help-1">Forum</a></li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div class="col-xs-6 col-sm-4">
|
||||||
|
<span class="menu_title">Services</span>
|
||||||
|
<ul>
|
||||||
|
<li><a href="https://www.odoo.sh">Odoo Cloud Platform</a></li>
|
||||||
|
<li class="divider"></li>
|
||||||
|
<li><a href="https://www.odoo.com/help">Support</a></li>
|
||||||
|
<li><a href="https://upgrade.odoo.com">Upgrade</a></li>
|
||||||
|
<li class="divider"></li>
|
||||||
|
<li><a href="https://www.odoo.com/partners">Find a partner</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/become-a-partner">Become a partner</a></li>
|
||||||
|
<li class="divider"></li>
|
||||||
|
<li><a href="https://training.odoo.com/courses/odoo-functional">Training Center</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/education-program">Education</a></li>
|
||||||
|
<li class="divider"></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/security">Security</a></li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div class="col-xs-12 col-sm-4 mb64">
|
||||||
|
<span class="menu_title">About us</span>
|
||||||
|
<ul>
|
||||||
|
<li><a href="https://www.odoo.com/page/about-us">Our company</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/contactus">Contact</a></li>
|
||||||
|
<li class="divider" />
|
||||||
|
<li><a href="https://www.odoo.com/event">Events</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/blog">Blog</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/blog/6">Customers</a></li>
|
||||||
|
<li class="divider" />
|
||||||
|
<li><a href="https://www.odoo.com/jobs">Jobs</a></li>
|
||||||
|
<li class="divider" />
|
||||||
|
<li><a href="https://www.odoo.com/page/legal">Legal</a> | <a
|
||||||
|
href="https://www.odoo.com/privacy">Privacy</a></li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="col-sm-5 col-md-4 col-md-offset-1 col-lg-5 col-lg-offset-1">
|
||||||
|
<p>
|
||||||
|
<small>
|
||||||
|
Odoo is a suite of open source business apps that cover all your company needs: CRM, eCommerce,
|
||||||
|
accounting, inventory, point of sale, project management, etc.
|
||||||
|
<br /><br />
|
||||||
|
Odoo's unique value proposition is to be at the same time very easy to use and fully integrated.
|
||||||
|
</small>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="o_footer_bottom">
|
||||||
|
<div class="container">
|
||||||
|
<a class="small" href="https://www.odoo.com/page/website-builder">Website made with <span
|
||||||
|
class="o_logo o_logo_inverse o_logo_15"></span></a>
|
||||||
|
<div class="social-links pull-right">
|
||||||
|
<a href="https://www.odoo.com/web/about/facebook"><i class="fa fa-facebook"></i></a>
|
||||||
|
<a href="https://www.odoo.com/web/about/twitter"><i class="fa fa-twitter"></i></a>
|
||||||
|
<a href="https://www.odoo.com/web/about/linkedin"><i class="fa fa-linkedin"></i></a>
|
||||||
|
<a href="mailto:info@odoo.com"><i class="fa fa-envelope"></i></a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
114
_extensions/odoo_ext/odoo_header.html
Normal file
114
_extensions/odoo_ext/odoo_header.html
Normal file
@ -0,0 +1,114 @@
|
|||||||
|
<div class="o_main_header_main">
|
||||||
|
<a class="pull-left o_logo" href="/"></a>
|
||||||
|
<a href="#" class="o_mobile_menu_toggle visible-xs-block pull-right">
|
||||||
|
<span class="sr-only">Toggle navigation</span>
|
||||||
|
<span class="mdi-navigation-menu"></span>
|
||||||
|
</a>
|
||||||
|
<div class="o_header_buttons">
|
||||||
|
<a href="https://www.odoo.com/trial" class="btn btn-primary">Start Now</a>
|
||||||
|
</div>
|
||||||
|
<ul class="o_primary_nav">
|
||||||
|
<li class="dropdown">
|
||||||
|
<a href="#" class="dropdown-toggle">Apps</a>
|
||||||
|
<div class="dropdown-menu o_secondary_nav">
|
||||||
|
<div class="container">
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-sm-3 o_website_apps">
|
||||||
|
<div class="o_nav_app_family">
|
||||||
|
<span></span> Websites
|
||||||
|
<div>Build great user experience</div>
|
||||||
|
</div>
|
||||||
|
<ul>
|
||||||
|
<li><a href="https://www.odoo.com/page/website-builder">Website Builder</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/e-commerce">eCommerce</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/blog-engine">Blogs</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/community-builder">Forums</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/slides">Slides</a></li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div class="col-sm-3 o_sale_apps">
|
||||||
|
<div class="o_nav_app_family">
|
||||||
|
<span></span> Sales
|
||||||
|
<div>Boost your success rate</div>
|
||||||
|
</div>
|
||||||
|
<ul>
|
||||||
|
<li><a href="https://www.odoo.com/page/sales">Sales</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/crm">CRM</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/billing">Invoicing</a></li>
|
||||||
|
<li class="dropdown">
|
||||||
|
<a href="#0" class="dropdown-toggle">Point of Sale</a>
|
||||||
|
<ul>
|
||||||
|
<li><a href="https://www.odoo.com/page/point-of-sale">Shops</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/pos-restaurant">Restaurants</a></li>
|
||||||
|
</ul>
|
||||||
|
</li>
|
||||||
|
<li><a href="https://www.odoo.com/page/subscriptions">Subscriptions</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/sign">Sign</a></li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div class="col-sm-3 o_operation_apps">
|
||||||
|
<div class="o_nav_app_family">
|
||||||
|
<span></span> Operations
|
||||||
|
<div>It's all about efficiency</div>
|
||||||
|
</div>
|
||||||
|
<ul>
|
||||||
|
<li><a href="https://www.odoo.com/page/accounting">Accounting</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/project-management">Project</a></li>
|
||||||
|
<li class="dropdown">
|
||||||
|
<a href="#0" class="dropdown-toggle">Human Resources</a>
|
||||||
|
<ul>
|
||||||
|
<li><a href="https://www.odoo.com/page/recruitment">Recruitment</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/employees">Employees</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/expenses">Expenses</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/appraisal">Appraisal</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/fleet">Fleet</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/leaves">Leaves</a></li>
|
||||||
|
</ul>
|
||||||
|
</li>
|
||||||
|
<li><a href="https://www.odoo.com/page/warehouse">Inventory</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/purchase">Purchase</a></li>
|
||||||
|
<li class="dropdown">
|
||||||
|
<a href="#0" class="dropdown-toggle">Manufacturing</a>
|
||||||
|
<ul>
|
||||||
|
<li><a href="https://www.odoo.com/page/manufacturing">MRP</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/plm">PLM</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/maintenance">Maintenance</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/quality">Quality</a></li>
|
||||||
|
</ul>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div class="col-sm-3 o_productivity_apps">
|
||||||
|
<div class="o_nav_app_family">
|
||||||
|
<span></span> Productivity Tools
|
||||||
|
<div>Great Tools = Happy People</div>
|
||||||
|
</div>
|
||||||
|
<ul>
|
||||||
|
<li class="dropdown">
|
||||||
|
<a href="#0" class="dropdown-toggle">Communication</a>
|
||||||
|
<ul>
|
||||||
|
<li><a href="https://www.odoo.com/page/discuss">Discuss</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/discuss-groups">Mailing Lists</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/notes">Notes</a></li>
|
||||||
|
<li><a href="#">Help desk</a></li>
|
||||||
|
<li><a href="#">Appointment</a></li>
|
||||||
|
</ul>
|
||||||
|
</li>
|
||||||
|
<li><a href="https://www.odoo.com/page/timesheet">Timesheet</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/email-marketing">Email Marketing</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/events">Events</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/survey">Survey</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/live-chat">Live Chat</a></li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<a href="https://www.odoo.com/apps/modules" class="o_store_link"><i class="fa fa-cube fa-fw"></i> Third
|
||||||
|
party apps</a>
|
||||||
|
</div>
|
||||||
|
</li>
|
||||||
|
<li><a href="https://www.odoo.com/page/tour">Tour</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/pricing">Pricing</a></li>
|
||||||
|
<li><a href="https://www.odoo.com/page/docs">Docs</a></li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
@ -118,9 +118,9 @@ class OdooStyle(Style):
|
|||||||
Generic.Traceback: "",
|
Generic.Traceback: "",
|
||||||
}
|
}
|
||||||
|
|
||||||
import imp
|
import types
|
||||||
import sys
|
import sys
|
||||||
modname = 'pygments.styles.odoo'
|
modname = 'pygments.styles.odoo'
|
||||||
m = imp.new_module(modname)
|
m = types.ModuleType(modname)
|
||||||
m.OdooStyle = OdooStyle
|
m.OdooStyle = OdooStyle
|
||||||
sys.modules[modname] = m
|
sys.modules[modname] = m
|
@ -237,7 +237,7 @@ aside {
|
|||||||
}
|
}
|
||||||
|
|
||||||
&.active {
|
&.active {
|
||||||
.transform(~ "translateZ(0px) rotateZ(-180deg)");
|
.transform(~"translateZ(0px) rotateZ(-180deg)");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user