1
0
forked from VimPlug/jedi

Make sure meta class filters can distinguish between classes and instances

This commit is contained in:
Dave Halter
2020-06-07 14:54:26 +02:00
parent 34cc8e9ad7
commit 9adcf3d233
4 changed files with 38 additions and 15 deletions

View File

@@ -193,7 +193,7 @@ class ClassMixin(object):
if include_metaclasses: if include_metaclasses:
metaclasses = self.get_metaclasses() metaclasses = self.get_metaclasses()
if metaclasses: if metaclasses:
for f in self.get_metaclass_filters(metaclasses): for f in self.get_metaclass_filters(metaclasses, is_instance):
yield f yield f
for cls in self.py__mro__(): for cls in self.py__mro__():
@@ -361,8 +361,8 @@ class ClassValue(use_metaclass(CachedMetaClass, ClassMixin, FunctionAndClassBase
return ValueSet({self}) return ValueSet({self})
@plugin_manager.decorate() @plugin_manager.decorate()
def get_metaclass_filters(self, metaclass): def get_metaclass_filters(self, metaclass, is_instance):
debug.dbg('Unprocessed metaclass %s', metaclass) debug.warning('Unprocessed metaclass %s', metaclass)
return [] return []
@inference_state_method_cache(default=NO_VALUES) @inference_state_method_cache(default=NO_VALUES)

View File

@@ -32,12 +32,21 @@ mapping = {
} }
def _infer_scalar_field(inference_state, field_name, field_tree_instance): def _get_deferred_attributes(inference_state):
return inference_state.import_module(
('django', 'db', 'models', 'query_utils')
).py__getattribute__('DeferredAttribute').execute_annotation()
def _infer_scalar_field(inference_state, field_name, field_tree_instance, is_instance):
try: try:
module_name, attribute_name = mapping[field_tree_instance.py__name__()] module_name, attribute_name = mapping[field_tree_instance.py__name__()]
except KeyError: except KeyError:
return None return None
if not is_instance:
return _get_deferred_attributes(inference_state)
if module_name is None: if module_name is None:
module = inference_state.builtins_module module = inference_state.builtins_module
else: else:
@@ -65,16 +74,20 @@ def _get_foreign_key_values(cls, field_tree_instance):
yield value yield value
def _infer_field(cls, field_name): def _infer_field(cls, field_name, is_instance):
inference_state = cls.inference_state inference_state = cls.inference_state
for field_tree_instance in field_name.infer(): for field_tree_instance in field_name.infer():
scalar_field = _infer_scalar_field(inference_state, field_name, field_tree_instance) scalar_field = _infer_scalar_field(
inference_state, field_name, field_tree_instance, is_instance)
if scalar_field is not None: if scalar_field is not None:
return scalar_field return scalar_field
name = field_tree_instance.py__name__() name = field_tree_instance.py__name__()
is_many_to_many = name == 'ManyToManyField' is_many_to_many = name == 'ManyToManyField'
if name in ('ForeignKey', 'OneToOneField') or is_many_to_many: if name in ('ForeignKey', 'OneToOneField') or is_many_to_many:
if not is_instance:
return _get_deferred_attributes(inference_state)
values = _get_foreign_key_values(cls, field_tree_instance) values = _get_foreign_key_values(cls, field_tree_instance)
if is_many_to_many: if is_many_to_many:
return ValueSet(filter(None, [ return ValueSet(filter(None, [
@@ -89,12 +102,13 @@ def _infer_field(cls, field_name):
class DjangoModelName(NameWrapper): class DjangoModelName(NameWrapper):
def __init__(self, cls, name): def __init__(self, cls, name, is_instance):
super(DjangoModelName, self).__init__(name) super(DjangoModelName, self).__init__(name)
self._cls = cls self._cls = cls
self._is_instance = is_instance
def infer(self): def infer(self):
return _infer_field(self._cls, self._wrapped_name) return _infer_field(self._cls, self._wrapped_name, self._is_instance)
def _create_manager_for(cls, manager_cls='BaseManager'): def _create_manager_for(cls, manager_cls='BaseManager'):
@@ -109,7 +123,7 @@ def _create_manager_for(cls, manager_cls='BaseManager'):
return None return None
def _new_dict_filter(cls): def _new_dict_filter(cls, is_instance):
def get_manager_name(filters): def get_manager_name(filters):
for f in filters: for f in filters:
names = f.get('objects') names = f.get('objects')
@@ -142,7 +156,7 @@ def _new_dict_filter(cls):
filters = list(cls.get_filters(is_instance=True, include_metaclasses=False)) filters = list(cls.get_filters(is_instance=True, include_metaclasses=False))
dct = { dct = {
name.string_name: DjangoModelName(cls, name) name.string_name: DjangoModelName(cls, name, is_instance)
for filter_ in reversed(filters) for filter_ in reversed(filters)
for name in filter_.values() for name in filter_.values()
} }
@@ -155,11 +169,11 @@ def _new_dict_filter(cls):
def get_metaclass_filters(func): def get_metaclass_filters(func):
def wrapper(cls, metaclasses): def wrapper(cls, metaclasses, is_instance):
for metaclass in metaclasses: for metaclass in metaclasses:
if metaclass.py__name__() == 'ModelBase' \ if metaclass.py__name__() == 'ModelBase' \
and metaclass.get_root_context().py__name__() == 'django.db.models.base': and metaclass.get_root_context().py__name__() == 'django.db.models.base':
return [_new_dict_filter(cls)] return [_new_dict_filter(cls, is_instance)]
return func(cls, metaclasses) return func(cls, metaclasses, is_instance)
return wrapper return wrapper

View File

@@ -801,7 +801,7 @@ _implemented = {
def get_metaclass_filters(func): def get_metaclass_filters(func):
def wrapper(cls, metaclasses): def wrapper(cls, metaclasses, is_instance):
for metaclass in metaclasses: for metaclass in metaclasses:
if metaclass.py__name__() == 'EnumMeta' \ if metaclass.py__name__() == 'EnumMeta' \
and metaclass.get_root_context().py__name__() == 'enum': and metaclass.get_root_context().py__name__() == 'enum':
@@ -809,7 +809,7 @@ def get_metaclass_filters(func):
return [DictFilter({ return [DictFilter({
name.string_name: EnumInstance(cls, name).name for name in filter_.values() name.string_name: EnumInstance(cls, name).name for name in filter_.values()
})] })]
return func(cls, metaclasses) return func(cls, metaclasses, is_instance)
return wrapper return wrapper

View File

@@ -4,6 +4,7 @@ import uuid
from django.db import models from django.db import models
from django.contrib.auth.models import User from django.contrib.auth.models import User
from django.db.models.query_utils import DeferredAttribute
class TagManager(models.Manager): class TagManager(models.Manager):
@@ -58,6 +59,9 @@ class BusinessModel(models.Model):
unidentifiable = NOT_FOUND unidentifiable = NOT_FOUND
#? models.IntegerField()
integer_field
def method(self): def method(self):
return 42 return 42
@@ -65,6 +69,11 @@ class BusinessModel(models.Model):
# Model attribute inference # Model attribute inference
# ----------------- # -----------------
#? DeferredAttribute()
BusinessModel.integer_field
#? DeferredAttribute()
BusinessModel.tags_m2m
model_instance = BusinessModel() model_instance = BusinessModel()
#? int() #? int()