forked from VimPlug/jedi
207 lines
6.0 KiB
Python
207 lines
6.0 KiB
Python
from jedi._compatibility import Python3Method
|
|
from jedi.common import unite
|
|
from parso.python.tree import ExprStmt, CompFor
|
|
from jedi.parser_utils import clean_scope_docstring, get_doc_with_call_signature
|
|
|
|
|
|
class Context(object):
|
|
"""
|
|
Should be defined, otherwise the API returns empty types.
|
|
"""
|
|
|
|
"""
|
|
To be defined by subclasses.
|
|
"""
|
|
predefined_names = {}
|
|
tree_node = None
|
|
|
|
def __init__(self, evaluator, parent_context=None):
|
|
self.evaluator = evaluator
|
|
self.parent_context = parent_context
|
|
|
|
@property
|
|
def api_type(self):
|
|
# By default just lower name of the class. Can and should be
|
|
# overwritten.
|
|
return self.__class__.__name__.lower()
|
|
|
|
def get_root_context(self):
|
|
context = self
|
|
while True:
|
|
if context.parent_context is None:
|
|
return context
|
|
context = context.parent_context
|
|
|
|
def execute(self, arguments):
|
|
return self.evaluator.execute(self, arguments)
|
|
|
|
def execute_evaluated(self, *value_list):
|
|
"""
|
|
Execute a function with already executed arguments.
|
|
"""
|
|
from jedi.evaluate.param import ValuesArguments
|
|
arguments = ValuesArguments([[value] for value in value_list])
|
|
return self.execute(arguments)
|
|
|
|
def eval_node(self, node):
|
|
return self.evaluator.eval_element(self, node)
|
|
|
|
def eval_stmt(self, stmt, seek_name=None):
|
|
return self.evaluator.eval_statement(self, stmt, seek_name)
|
|
|
|
def eval_trailer(self, types, trailer):
|
|
return self.evaluator.eval_trailer(self, types, trailer)
|
|
|
|
@Python3Method
|
|
def py__getattribute__(self, name_or_str, name_context=None, position=None,
|
|
search_global=False, is_goto=False,
|
|
analysis_errors=True):
|
|
if name_context is None:
|
|
name_context = self
|
|
return self.evaluator.find_types(
|
|
self, name_or_str, name_context, position, search_global, is_goto,
|
|
analysis_errors)
|
|
|
|
def create_context(self, node, node_is_context=False, node_is_object=False):
|
|
return self.evaluator.create_context(self, node, node_is_context, node_is_object)
|
|
|
|
def is_class(self):
|
|
return False
|
|
|
|
def py__bool__(self):
|
|
"""
|
|
Since Wrapper is a super class for classes, functions and modules,
|
|
the return value will always be true.
|
|
"""
|
|
return True
|
|
|
|
def py__doc__(self, include_call_signature=False):
|
|
try:
|
|
self.tree_node.get_doc_node
|
|
except AttributeError:
|
|
return ''
|
|
else:
|
|
if include_call_signature:
|
|
return get_doc_with_call_signature(self.tree_node)
|
|
else:
|
|
return clean_scope_docstring(self.tree_node)
|
|
return None
|
|
|
|
|
|
class TreeContext(Context):
|
|
def __init__(self, evaluator, parent_context=None):
|
|
super(TreeContext, self).__init__(evaluator, parent_context)
|
|
self.predefined_names = {}
|
|
|
|
def __repr__(self):
|
|
return '<%s: %s>' % (self.__class__.__name__, self.tree_node)
|
|
|
|
|
|
class AbstractLazyContext(object):
|
|
def __init__(self, data):
|
|
self.data = data
|
|
|
|
def __repr__(self):
|
|
return '<%s: %s>' % (self.__class__.__name__, self.data)
|
|
|
|
def infer(self):
|
|
raise NotImplementedError
|
|
|
|
|
|
class LazyKnownContext(AbstractLazyContext):
|
|
"""data is a context."""
|
|
def infer(self):
|
|
return set([self.data])
|
|
|
|
|
|
class LazyKnownContexts(AbstractLazyContext):
|
|
"""data is a set of contexts."""
|
|
def infer(self):
|
|
return self.data
|
|
|
|
|
|
class LazyUnknownContext(AbstractLazyContext):
|
|
def __init__(self):
|
|
super(LazyUnknownContext, self).__init__(None)
|
|
|
|
def infer(self):
|
|
return set()
|
|
|
|
|
|
class LazyTreeContext(AbstractLazyContext):
|
|
def __init__(self, context, node):
|
|
super(LazyTreeContext, self).__init__(node)
|
|
self._context = context
|
|
# We need to save the predefined names. It's an unfortunate side effect
|
|
# that needs to be tracked otherwise results will be wrong.
|
|
self._predefined_names = dict(context.predefined_names)
|
|
|
|
def infer(self):
|
|
old, self._context.predefined_names = \
|
|
self._context.predefined_names, self._predefined_names
|
|
try:
|
|
return self._context.eval_node(self.data)
|
|
finally:
|
|
self._context.predefined_names = old
|
|
|
|
|
|
def get_merged_lazy_context(lazy_contexts):
|
|
if len(lazy_contexts) > 1:
|
|
return MergedLazyContexts(lazy_contexts)
|
|
else:
|
|
return lazy_contexts[0]
|
|
|
|
|
|
class MergedLazyContexts(AbstractLazyContext):
|
|
"""data is a list of lazy contexts."""
|
|
def infer(self):
|
|
return unite(l.infer() for l in self.data)
|
|
|
|
|
|
class ContextualizedNode(object):
|
|
def __init__(self, context, node):
|
|
self.context = context
|
|
self._node = node
|
|
|
|
def get_root_context(self):
|
|
return self.context.get_root_context()
|
|
|
|
def infer(self):
|
|
return self.context.eval_node(self._node)
|
|
|
|
|
|
class ContextualizedName(ContextualizedNode):
|
|
# TODO merge with TreeNameDefinition?!
|
|
@property
|
|
def name(self):
|
|
return self._node
|
|
|
|
def assignment_indexes(self):
|
|
"""
|
|
Returns an array of tuple(int, node) of the indexes that are used in
|
|
tuple assignments.
|
|
|
|
For example if the name is ``y`` in the following code::
|
|
|
|
x, (y, z) = 2, ''
|
|
|
|
would result in ``[(1, xyz_node), (0, yz_node)]``.
|
|
"""
|
|
indexes = []
|
|
node = self._node.parent
|
|
compare = self._node
|
|
while node is not None:
|
|
if node.type in ('testlist', 'testlist_comp', 'testlist_star_expr', 'exprlist'):
|
|
for i, child in enumerate(node.children):
|
|
if child == compare:
|
|
indexes.insert(0, (int(i / 2), node))
|
|
break
|
|
else:
|
|
raise LookupError("Couldn't find the assignment.")
|
|
elif isinstance(node, (ExprStmt, CompFor)):
|
|
break
|
|
|
|
compare = node
|
|
node = node.parent
|
|
return indexes
|