Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How do I find the "concrete class" of a django model baseclass

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.

like image 619
Mr Shark Avatar asked Dec 08 '08 11:12

Mr Shark


People also ask

What is the class of instance object in Django?

The two main concepts of OOP are classes and objects: Class: Class is basically a blueprint or a template for creating objects. Object: Collection of arguments and methods which can be performed on those data. An object is nothing but an instance of the class.

What class does a Django model class inherit?

The basics: Each model is a Python class that subclasses django.db.models.Model . Each attribute of the model represents a database field. With all of this, Django gives you an automatically-generated database-access API; see Making queries.

How will you define the model classes in Django?

When you make a model class in Django, consider that class the data-table, each individual instance of that class the table rows, and the attributes(e.g: title) of each table the columns. In the definition of the class Book, title seems to be a class attribute.

How do I find models in Django?

In Django 1.7+ it is better to use get_model() on the Django app registry, which is available via django. apps. apps. get_model(model_name) .


1 Answers

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:

from django.contrib.contenttypes.models import ContentType

class Base(models.Model):
    content_type = models.ForeignKey(ContentType,editable=False,null=True)

    def save(self):
        if(not self.content_type):
            self.content_type = ContentType.objects.get_for_model(self.__class__)
        self.save_base()

    def as_leaf_class(self):
        content_type = self.content_type
        model = content_type.model_class()
        if(model == Base):
            return self
        return model.objects.get(id=self.id)

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.

like image 55
Daniel Naab Avatar answered Oct 28 '22 17:10

Daniel Naab