# Generic Relations

Built in app that **keeps track of models from the installed apps** of your Django application.

Use case: **create generic relationships between models**.

## **Installation**

Make sure you have it in your `INSTALLED_APPS`:

```python
INSTALLED_APPS = [
    ...
    'django.contrib.contenttypes',
    ...
]
```

## **Example Scenario**

Social app where the users can ask and answer questions, **up vote**, **down vote**, **favorite** a question, **like** a post in the website, etc.

To keep track of that we create a model named **Activity**. See below:

```python
class Activity(models.Model):
    FAVORITE = 'F'
    LIKE = 'L'
    UP_VOTE = 'U'
    DOWN_VOTE = 'D'
    ACTIVITY_TYPES = (
        (FAVORITE, 'Favorite'),
        (LIKE, 'Like'),
        (UP_VOTE, 'Up Vote'),
        (DOWN_VOTE, 'Down Vote'),
    )

    user = models.ForeignKey(User)
    activity_type = models.CharField(max_length=1, choices=ACTIVITY_TYPES)
    date = models.DateTimeField(auto_now_add=True)
    post = models.ForeignKey(Post, null=True)
    question = models.ForeignKey(Question, null=True)
    answer = models.ForeignKey(Answer, null=True)
    comment = models.ForeignKey(Comment, null=True)
```

So an **Activity** can possibly interact with a **Post**, **Question**, **Answer** or a **Comment** instance. In a practical scenario an **Activity** instance would represent a single interaction. For example, the User with ID 1 **up voted** a Question with ID 125:

```python
Activity.objects.create(user=1, activity_type='U', question=125)
```

And if I wanted to calculate how many **up votes** the Question 125 received, I could do something like that:

```python
question = Question.objects.get(pk=125)
up_votes = question.activity_set.filter(activity_type=Activity.UP_VOTE)

# Display how many up votes
count = up_votes.count()

# Display the names of users who up voted
up_voters = up_votes.values_list('user__first_name')
```

In a similar way we could work with the Post, Answer and Comment models.

### **Using the Generic Relations**

To achieve the same result using **Generic Relations**, here is what the **Activity** models should look like:

```python
from django.contrib.contenttypes.fields import GenericForeignKey
from django.contrib.contenttypes.models import ContentType

class Activity(models.Model):
    FAVORITE = 'F'
    LIKE = 'L'
    UP_VOTE = 'U'
    DOWN_VOTE = 'D'
    ACTIVITY_TYPES = (
        (FAVORITE, 'Favorite'),
        (LIKE, 'Like'),
        (UP_VOTE, 'Up Vote'),
        (DOWN_VOTE, 'Down Vote'),
    )

    user = models.ForeignKey(User)
    activity_type = models.CharField(max_length=1, choices=ACTIVITY_TYPES)
    date = models.DateTimeField(auto_now_add=True)

    # Below the mandatory fields for generic relation
    content_type = models.ForeignKey(ContentType, on_delete=models.CASCADE)
    object_id = models.PositiveIntegerField()
    content_object = GenericForeignKey()
```

**Now we are no longer keeping ForeignKey to other models** we want to track the favorite, like, up vote and down vote activities. Meaning we can now track those activities to any model we want without having to modify the Activity model.

The relation is created in the model you want to track the **Activity**:

```python
from django.db import models
from django.contrib.contenttypes.fields import GenericRelation

from activities.models import Activity


class Post(models.Model):
    ...
    likes = GenericRelation(Activity)

class Question(models.Model):
    ...
    activities = GenericRelation(Activity)

class Answer(models.Model):
    ...
    votes = GenericRelation(Activity)

class Comment(models.Model):
    ...
    likes = GenericRelation(Activity)
```

This also enables you to define a more meaningful name for the relations. For example, the **Users** can only interact with **Post** and **Comment** models to **like** it. While with the **Answer** model, they can only **up vote/down vote**. And finally with the **Question** model, the **Users** can **up vote/down vote** and **favorite** it.

Now if you want to **like** a **Post**, you could do something like this:

```python
# Get the post object
post = Post.objects.get(pk=1)

# Add a like activity
post.likes.create(activity_type=Activity.LIKE, user=request.user)

# Or in a similar way using the Activity model to add the like
Activity.objects.create(content_object=post, activity_type=Activity.LIKE, user=request.user)

# Get all Activity instances related to Post
post.likes.all()

# Count the number of likes
post.likes.count()

# Get the users who liked the post
post.likes.values_list('user__first_name', flat=True)
```

A good thing about it is that if have a new model that you wants to interact with **Activity**, you simply add a **GenericRelation**:

```python
from django.contrib.contenttypes.fields import GenericRelation

class Picture(models.Model):
    user = models.ForeignKey(User)
    picture_file = models.ImageField(upload_to='uploads/pictures')
    date = models.DateTimeField(auto_now_add=True)
    favorites = GenericRelation(Activity)
```

And it’s already ready to use:

```python
picture = Picture.objects.get(pk=1)

picture.favorites.create(activity_type=Activity.FAVORITE, user=request.user)
picture.favorites.count()
```

## **Reverse relations**

You may also define an extra parameter in the GenericRelation:

```python
from django.contrib.contenttypes.fields import GenericRelation

class Picture(models.Model):
    user = models.ForeignKey(User)
    picture_file = models.ImageField(upload_to='uploads/pictures')
    date = models.DateTimeField(auto_now_add=True)
    favorites = GenericRelation(Activity, related_query_name='pictures')
```

Then you can use it to query for example all favorited pictures that was uploaded by a given user:

```
user = User.objects.get(pk=1)
Activity.objects.filter(pictures__user=user)
```

## **Caveats**

Adds an **extra layer of complexity and will eventually make things slower**.

**GenericForeignKey** does not accept an **on\_delete** argument to customize this behavior. The default behavior will cascade all the relations.

One way to avoid the default behavior is to not define a **GenericRelation**. Example:

```python
class Comment(models.Model):
    text = models.CharField(max_length=500, blank=True)
    date = models.DateTimeField(auto_now_add=True)

# Add a new instance of Comment
comment = Comment.objects.create(text='This is a test comment')

# Like the comment
Activity.objects.create(content_object=comment, activity_type=Activity.LIKE, user=request.user)
```

Now to get the list of **likes** this **Comment** received you must use the **ContentType** class:

```python
from django.contrib.contenttypes.models import ContentType

# Pass the instance we created in the snippet above
ct = ContentType.objects.get_for_model(comment)

# Get the list of likes
Activity.objects.filter(content_type=ct, object_id=comment.id, activity_type=Activity.LIKE)
```

This is also an option if you want to interact with a model from Django’s contrib module or any third party model that you don’t have access to add a **GenericRelation**.<br>


---

# Agent Instructions: Querying This Documentation

If you need additional information that is not directly available in this page, you can query the documentation dynamically by asking a question.

Perform an HTTP GET request on the current page URL with the `ask` query parameter:

```
GET https://ricardomol.gitbook.io/notes/backend/django/generic-relations.md?ask=<question>
```

The question should be specific, self-contained, and written in natural language.
The response will contain a direct answer to the question and relevant excerpts and sources from the documentation.

Use this mechanism when the answer is not explicitly present in the current page, you need clarification or additional context, or you want to retrieve related documentation sections.
