1
0
forked from VimPlug/jedi

Refactoring of the contexts to properly use inheritance

This commit is contained in:
Dave Halter
2019-08-23 15:27:15 +02:00
parent 7573e2033a
commit c393a406ee
3 changed files with 132 additions and 106 deletions

View File

@@ -21,7 +21,7 @@ from jedi.inference.compiled.access import compiled_objects_cache, \
ALLOWED_GETITEM_TYPES, get_api_type ALLOWED_GETITEM_TYPES, get_api_type
from jedi.inference.compiled.value import create_cached_compiled_object from jedi.inference.compiled.value import create_cached_compiled_object
from jedi.inference.gradual.conversion import to_stub from jedi.inference.gradual.conversion import to_stub
from jedi.inference.context import CompiledContext from jedi.inference.context import CompiledContext, TreeContextMixin
_sentinel = object() _sentinel = object()
@@ -81,7 +81,7 @@ class MixedObject(ValueWrapper):
) )
class MixedContext(CompiledContext): class MixedContext(CompiledContext, TreeContextMixin):
@property @property
def compiled_object(self): def compiled_object(self):
return self._value.compiled_object return self._value.compiled_object

View File

@@ -6,21 +6,131 @@ from jedi import parser_utils
class AbstractContext(object): class AbstractContext(object):
""" # Must be defined: inference_state and tree_node and parent_context as an attribute/property
Should be defined, otherwise the API returns empty types.
"""
predefined_names = {}
def __init__(self, value): def __init__(self, inference_state):
self.inference_state = value.inference_state self.inference_state = inference_state
self._value = value self.predefined_names = {}
@abstractmethod @abstractmethod
def get_filters(self, until_position=None, origin_scope=None): def get_filters(self, until_position=None, origin_scope=None):
raise NotImplementedError raise NotImplementedError
def goto(self, name_or_str, position):
from jedi.inference import finder
f = finder.NameFinder(self.inference_state, self, self, name_or_str, position)
filters = f.get_global_filters()
return f.filter_name(filters)
def py__getattribute__(self, name_or_str, name_value=None, position=None,
analysis_errors=True):
"""
:param position: Position of the last statement -> tuple of line, column
"""
if name_value is None:
name_value = self
from jedi.inference import finder
f = finder.NameFinder(self.inference_state, self, name_value, name_or_str,
position, analysis_errors=analysis_errors)
filters = f.get_global_filters()
return f.find(filters, attribute_lookup=False)
def get_root_context(self): def get_root_context(self):
return self._value.get_root_context() parent_context = self.parent_context
if parent_context is None:
return self
return parent_context.get_root_context()
def is_module(self):
return False
def is_builtins_module(self):
return False
def is_class(self):
return False
def is_stub(self):
return False
def is_instance(self):
return False
def is_compiled(self):
return False
@abstractmethod
def py__name__(self):
raise NotImplementedError
@property
def name(self):
return None
def get_qualified_names(self):
return ()
def py__doc__(self):
return ''
def __repr__(self):
return '%s(%s)' % (self.__class__.__name__, self._value)
class ValueContext(AbstractContext):
"""
Should be defined, otherwise the API returns empty types.
"""
def __init__(self, value):
super(ValueContext, self).__init__(value.inference_state)
self._value = value
@property
def tree_node(self):
return self._value.tree_node
@property
def parent_context(self):
return self._value.parent_context
def is_module(self):
return self._value.is_module()
def is_builtins_module(self):
return self._value == self.inference_state.builtins_module
def is_class(self):
return self._value.is_class()
def is_stub(self):
return self._value.is_stub()
def is_instance(self):
return self._value.is_instance()
def is_compiled(self):
return self._value.is_compiled()
def py__name__(self):
return self._value.py__name__()
@property
def name(self):
return self._value.name
def get_qualified_names(self):
return self._value.get_qualified_names()
def py__doc__(self):
return self._value.py__doc__()
def __repr__(self):
return '%s(%s)' % (self.__class__.__name__, self._value)
class TreeContextMixin(object):
def infer_node(self, node):
return self.inference_state.infer_element(self, node)
def create_value(self, node): def create_value(self, node):
from jedi.inference import value from jedi.inference import value
@@ -86,74 +196,8 @@ class AbstractContext(object):
scope_node = parent_scope(scope_node) scope_node = parent_scope(scope_node)
return from_scope_node(scope_node, is_nested=True) return from_scope_node(scope_node, is_nested=True)
def goto(self, name_or_str, position):
from jedi.inference import finder
f = finder.NameFinder(self.inference_state, self, self, name_or_str, position)
filters = f.get_global_filters()
return f.filter_name(filters)
def py__getattribute__(self, name_or_str, name_value=None, position=None, class FunctionContext(TreeContextMixin, ValueContext):
analysis_errors=True):
"""
:param position: Position of the last statement -> tuple of line, column
"""
if name_value is None:
name_value = self
from jedi.inference import finder
f = finder.NameFinder(self.inference_state, self, name_value, name_or_str,
position, analysis_errors=analysis_errors)
filters = f.get_global_filters()
return f.find(filters, attribute_lookup=False)
@property
def tree_node(self):
return self._value.tree_node
@property
def parent_context(self):
return self._value.parent_context
def is_module(self):
return self._value.is_module()
def is_builtins_module(self):
return self._value == self.inference_state.builtins_module
def is_class(self):
return self._value.is_class()
def is_stub(self):
return self._value.is_stub()
def is_instance(self):
return self._value.is_instance()
def is_compiled(self):
return self._value.is_compiled()
def py__name__(self):
return self._value.py__name__()
@property
def name(self):
if self._value is None:
return None
return self._value.name
def get_qualified_names(self):
return self._value.get_qualified_names()
def py__doc__(self):
return self._value.py__doc__()
def infer_node(self, node):
return self.inference_state.infer_element(self, node)
def __repr__(self):
return '%s(%s)' % (self.__class__.__name__, self._value)
class FunctionContext(AbstractContext):
def get_filters(self, until_position=None, origin_scope=None): def get_filters(self, until_position=None, origin_scope=None):
yield ParserTreeFilter( yield ParserTreeFilter(
self.inference_state, self.inference_state,
@@ -163,7 +207,7 @@ class FunctionContext(AbstractContext):
) )
class ModuleContext(AbstractContext): class ModuleContext(TreeContextMixin, ValueContext):
def py__file__(self): def py__file__(self):
return self._value.py__file__() return self._value.py__file__()
@@ -202,12 +246,12 @@ class ModuleContext(AbstractContext):
""" """
This is the only function that converts a context back to a value. This is the only function that converts a context back to a value.
This is necessary for stub -> python conversion and vice versa. However This is necessary for stub -> python conversion and vice versa. However
this method shouldn't be move to AbstractContext. this method shouldn't be moved to AbstractContext.
""" """
return self._value return self._value
class NamespaceContext(AbstractContext): class NamespaceContext(TreeContextMixin, ValueContext):
def get_filters(self, until_position=None, origin_scope=None): def get_filters(self, until_position=None, origin_scope=None):
return self._value.get_filters() return self._value.get_filters()
@@ -215,7 +259,7 @@ class NamespaceContext(AbstractContext):
return self._value.py__file__() return self._value.py__file__()
class ClassContext(AbstractContext): class ClassContext(TreeContextMixin, ValueContext):
def get_filters(self, until_position=None, origin_scope=None): def get_filters(self, until_position=None, origin_scope=None):
yield self.get_global_filter(until_position, origin_scope) yield self.get_global_filter(until_position, origin_scope)
@@ -227,35 +271,17 @@ class ClassContext(AbstractContext):
) )
class CompForContext(AbstractContext): class CompForContext(TreeContextMixin, AbstractContext):
def __init__(self, parent_context, comp_for): def __init__(self, parent_context, comp_for):
self._value = None super(CompForContext, self).__init__(parent_context.inference_state)
self._parent_context = parent_context self.tree_node = comp_for
self.inference_state = parent_context.inference_state self.parent_context = parent_context
self._tree_node = comp_for
@property
def parent_context(self):
return self._parent_context
def get_root_context(self):
return self._parent_context.get_root_context()
def is_instance(self):
return False
def is_compiled(self):
return False
@property
def tree_node(self):
return self._tree_node
def get_filters(self, until_position=None, origin_scope=None): def get_filters(self, until_position=None, origin_scope=None):
yield ParserTreeFilter(self) yield ParserTreeFilter(self)
class CompiledContext(AbstractContext): class CompiledContext(ValueContext):
def get_filters(self, until_position=None, origin_scope=None): def get_filters(self, until_position=None, origin_scope=None):
return self._value.get_filters() return self._value.get_filters()

View File

@@ -16,7 +16,7 @@ from jedi.inference.base_value import ContextualizedNode, NO_VALUES, \
ValueSet, TreeValue, ValueWrapper ValueSet, TreeValue, ValueWrapper
from jedi.inference.lazy_value import LazyKnownValues, LazyKnownValue, \ from jedi.inference.lazy_value import LazyKnownValues, LazyKnownValue, \
LazyTreeValue LazyTreeValue
from jedi.inference.context import AbstractContext from jedi.inference.context import ValueContext, TreeContextMixin
from jedi.inference.value import iterable from jedi.inference.value import iterable
from jedi import parser_utils from jedi import parser_utils
from jedi.inference.parser_cache import get_yield_exprs from jedi.inference.parser_cache import get_yield_exprs
@@ -164,7 +164,7 @@ class MethodValue(FunctionValue):
return names + (self.py__name__(),) return names + (self.py__name__(),)
class FunctionExecutionContext(AbstractContext): class FunctionExecutionContext(ValueContext, TreeContextMixin):
function_execution_filter = FunctionExecutionFilter function_execution_filter = FunctionExecutionFilter
def __init__(self, function_value, var_args): def __init__(self, function_value, var_args):