I’m trying to find the actual class of a django-model object, when using model-inheritance.
Some code to describe the problem:
class Base(models.model): def basemethod(self): ... class Child_1(Base): pass class Child_2(Base): pass
If I create various objects of the two Child classes and the create a queryset containing them all:
Child_1().save() Child_2().save() (o1, o2) = Base.objects.all()
I want to determine if the object is of type Child_1 or Child_2 in basemethod, I can get to the child object via o1.child_1 and o2.child_2 but that reconquers knowledge about the childclasses in the baseclass.
I have come up with the following code:
def concrete_instance(self): instance = None for subclass in self._meta.get_all_related_objects(): acc_name = subclass.get_accessor_name() try: instance = self.__getattribute__(acc_name) return instance except Exception, e: pass
But it feels brittle and I’m not sure of what happens when if I inherit in more levels.
Django implements model inheritance with a OneToOneField between the parent model’s table and the child model’s table. When you do
Base.object.all(), Django is querying just the Base table, and so has no way of knowing what the child table is. Therefore, unfortunately, it’s not possible to go directly to the child model instance without additional queries.This snippet shows a common method of adding a ContentType field to the base model:
You can then say
if Base.content_type.model_class()to determine the type.Here is another snippet that adds a custom manager into the mix.
As you can see, both of these solutions have the potential to be extremely expensive. If you have a large number of instances, using the as_leaf_class() method will require one query on each item.
Instead, if you have a known set of child models, simply query each model separately and aggregate the instances into one list.