MongoDB + Django Rest Framework CRUD Rest API example

Django MongoDB CRUD Rest API overview

We will build Rest Apis using Django Rest Framework that can create, retrieve, update, delete and find Tutorials by title or published status.

First, we setup Django Project with a MongoDB Connector. Next, we create Rest Api app, add it with Django Rest Framework to the project. Next, we define data model and migrate it to the database. Then we write API Views and define Routes for handling all CRUD operations (including custom finder).

The following table shows overview of the Rest APIs that will be exported:

MethodsUrlsActions
GETapi/tutorialsget all Tutorials
GETapi/tutorials/:idget Tutorial by id
POSTapi/tutorialsadd new Tutorial
PUTapi/tutorials/:idupdate Tutorial by id
DELETEapi/tutorials/:idremove Tutorial by id
DELETEapi/tutorialsremove all Tutorials
GETapi/tutorials/publishedfind all published Tutorials
GETapi/tutorials?title=[kw]find all Tutorials which title contains 'kw'

Finally, we’re gonna test the Rest Apis using Postman.

Architecture

This diagram shows the architecture of our Django CRUD Rest Apis App with MongoDB database:

django-mongodb-crud-rest-api-architecture
  • HTTP requests will be matched by Url Patterns and passed to the Views
  • Views processes the HTTP requests and returns HTTP responses (with the help of Serializer)
  • Serializer serializes/deserializes data model objects
  • Models contains essential fields and behaviors for CRUD Operations with MongoDB Database

Technology

  • Python 3.7
  • Django 2.1.15
  • Django Rest Framework 3.11.0
  • djongo 1.3.1
  • django-cors-headers 3.2.1
  • MongoDB 3.4 or higher

Project structure

This is the directory structure of our Django Project:

django-mongodb-crud-rest-api-project-structure

Let me explain it briefly.

  • tutorials/apps.py: declares TutorialsConfig class (subclass of django.apps.AppConfig) that represents Rest CRUD Apis app and its configuration.
  • DjangoRestApiMongoDB/settings.py: contains settings for our Django project: MongoDB Database engine, INSTALLED_APPS list with Django REST framework, Tutorials Application, CORS and MIDDLEWARE.
  • tutorials/models.py: defines Tutorial data model class (subclass of django.db.models.Model).
  • migrations/0001_initial.py: is created when we make migrations for the data model, and will be used for generating MongoDB database collection.
  • tutorials/serializers.py: manages serialization and deserialization with TutorialSerializer class (subclass of rest_framework.serializers.ModelSerializer).
  • tutorials/views.py: contains functions to process HTTP requests and produce HTTP responses (using TutorialSerializer).
  • tutorials/urls.py: defines URL patterns along with request functions in the Views.
  • DjangoRestApiMongoDB/urls.py: also has URL patterns that includes tutorials.urls, it is the root URL configurations.

Install Django REST framework

Django REST framework helps us to build RESTful Web Services flexibly.

To install this package, run command:
pip install djangorestframework

Setup new Django project

Let’s create a new Django project with command:

django-admin startproject DjangoRestApiMongoDB

You can see the following folder tree when the process is done.

django-mongodb-crud-rest-api-example-create-project

Now open settings.py and add Django REST framework to the INSTALLED_APPS array here.

INSTALLED_APPS = [    ...    # Django REST framework     'rest_framework',]

Connect Django project to MongoDB database

We need a Django MongoDb connector to work with MongoDb database.
In this tutorial, we’re gonna use djongo.

Run the command: pip install djongo.

Then we need to setup MongoDb Database engine.
So open settings.py and change declaration of DATABASES:

DATABASES = {    'default': {        'ENGINE': 'djongo',        'NAME': 'bezkoder_db',        'HOST': '127.0.0.1',        'PORT': 27017,    }}

Setup new Django app for Rest CRUD Api

Run following commands to create new Django app tutorials:

cd DjangoRestApiMongoDBpython manage.py startapp tutorials

The project directory now looks like:

django-mongodb-crud-rest-api-example-create-app

Open tutorials/apps.py, you can see TutorialsConfig class (subclass of django.apps.AppConfig). This is the Django app and its configuration that we’ve just created.

from django.apps import AppConfigclass TutorialsConfig(AppConfig):    name = 'tutorials'

Don’t forget to add this app to INSTALLED_APPS array in settings.py:

INSTALLED_APPS = [    ...    # Tutorials application     'tutorials.apps.TutorialsConfig',]

Configure CORS and middleware

We need to allow requests to our Django application from other origins.
In this example, we’re gonna configure CORS to accept requests from localhost:8081.

First, install the django-cors-headers library:
pip install django-cors-headers

In settings.py, add configuration for CORS:

INSTALLED_APPS = [    ...    # CORS    'corsheaders',]

You also need to add a middleware class to listen in on responses:

MIDDLEWARE = [    ...    # CORS    'corsheaders.middleware.CorsMiddleware',    'django.middleware.common.CommonMiddleware',]

Note: CorsMiddleware should be placed as high as possible, especially before any middleware that can generate responses such as CommonMiddleware.

Next, set CORS_ORIGIN_ALLOW_ALL and add the host to CORS_ORIGIN_WHITELIST:

CORS_ORIGIN_ALLOW_ALL = FalseCORS_ORIGIN_WHITELIST = (    'http://localhost:8081',)
  • CORS_ORIGIN_ALLOW_ALL: If True, all origins will be accepted (not use the whitelist below). Defaults to False.
  • CORS_ORIGIN_WHITELIST: List of origins that are authorized to make cross-site HTTP requests. Defaults to [].

Define the Django Model

Open tutorials/models.py, add Tutorial class as subclass of django.db.models.Model.
There are 3 fields: titledescriptionpublished.

from django.db import modelsclass Tutorial(models.Model):    title = models.CharField(max_length=70, blank=False, default='')    description = models.CharField(max_length=200,blank=False, default='')    published = models.BooleanField(default=False)

Each field is specified as a class attribute, and each attribute maps to a database column.
id field is added automatically.

Migrate Data Model to MongoDB database

Run the Python script: python manage.py makemigrations tutorials.

The console will show:

Migrations for 'tutorials':  tutorials\migrations\0001_initial.py    - Create model Tutorial

New file 0001_initial.py has just been generated in tutorials/migrations folder. It includes code to create Tutorial data model:

# Generated by Django 2.1.15from django.db import migrations, modelsclass Migration(migrations.Migration):    initial = True    dependencies = [    ]    operations = [        migrations.CreateModel(            name='Tutorial',            fields=[                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),                ('title', models.CharField(default='', max_length=70)),                ('description', models.CharField(default='', max_length=200)),                ('published', models.BooleanField(default=False)),            ],        ),    ]

The generated code defines Migration class (subclass of the django.db.migrations.Migration).
It has operations array that contains operation for creating collection for Customer model: migrations.CreateModel().

The call to this will create a new model in the project history and a corresponding collection in the MongoDB database to match it.

Now run the following Python script to apply the generated migration above:
python manage.py migrate tutorials

The console will show:

Operations to perform:  Apply all migrations: tutorialsRunning migrations:  Applying tutorials.0001_initial... OK

Check MongoDB database, you can see that a collection for Tutorial model was generated automatically with the name: tutorials_tutorial:

django-mongodb-crud-rest-api-example-db-table

Create Serializer class for Data Model

Let’s create TutorialSerializer class that will manage serialization and deserialization from JSON.

It inherit from rest_framework.serializers.ModelSerializer superclass which automatically populates a set of fields and default validators. We need to specify the model class here.

tutorials/serializers.py

from rest_framework import serializers from tutorials.models import Tutorial  class TutorialSerializer(serializers.ModelSerializer):     class Meta:        model = Tutorial        fields = ('id',                  'title',                  'description',                  'published')

In the inner class Meta, we declare 2 attributes:

  • model: the model for Serializer
  • fields: a tuple of field names to be included in the serialization

Define Routes to Views functions

Let determine how the server will response for HTTP request (GET, POST, PUT, DELETE) with some endpoints. We’re gonna define the routes:

  • /api/tutorials: GET, POST, DELETE
  • /api/tutorials/:id: GET, PUT, DELETE
  • /api/tutorials/published: GET

Inside tutorials app, create urls.py file with urlpatterns containing urls to be matched with the functions in the views.py:

from django.conf.urls import url from tutorials import views  urlpatterns = [     url(r'^api/tutorials$', views.tutorial_list),    url(r'^api/tutorials/(?P<pk>[0-9]+)$', views.tutorial_detail),    url(r'^api/tutorials/published$', views.tutorial_list_published)]

Don’t forget to include this URL patterns in root URL configurations.
Open DjangoRestApiMongoDB/urls.py and modify the content with the following code:

from django.conf.urls import url, include  urlpatterns = [     url(r'^', include('tutorials.urls')),]

Write API Views

Now we create the functions that are pointed in Url Patterns above:
– tutorial_list(): GET list of tutorials, POST a new tutorial, DELETE all tutorials
– tutorial_detail(): GET / PUT / DELETE tutorial by ‘id’
– tutorial_list_published(): GET all published tutorials

These functions process HTTP requests and make CRUD Operations to database via Django Model.

Open tutorials/views.py and write following code:

from django.shortcuts import renderfrom django.http.response import JsonResponsefrom rest_framework.parsers import JSONParser from rest_framework import status from tutorials.models import Tutorialfrom tutorials.serializers import TutorialSerializerfrom rest_framework.decorators import api_view@api_view(['GET', 'POST', 'DELETE'])def tutorial_list(request):    # GET list of tutorials, POST a new tutorial, DELETE all tutorials  @api_view(['GET', 'PUT', 'DELETE'])def tutorial_detail(request, pk):    # find tutorial by pk (id)    try:         tutorial = Tutorial.objects.get(pk=pk)     except Tutorial.DoesNotExist:         return JsonResponse({'message': 'The tutorial does not exist'}, status=status.HTTP_404_NOT_FOUND)      # GET / PUT / DELETE tutorial            @api_view(['GET'])def tutorial_list_published(request):    # GET all published tutorials

Let’s implement these functions.

Create a new object

Create and Save a new Tutorial:

@api_view(['GET', 'POST', 'DELETE'])def tutorial_list(request):    ...     elif request.method == 'POST':        tutorial_data = JSONParser().parse(request)        tutorial_serializer = TutorialSerializer(data=tutorial_data)        if tutorial_serializer.is_valid():            tutorial_serializer.save()            return JsonResponse(tutorial_serializer.data, status=status.HTTP_201_CREATED)         return JsonResponse(tutorial_serializer.errors, status=status.HTTP_400_BAD_REQUEST)

Retrieve objects (with condition)

Retrieve all Tutorials/ find by title from MongoDB database:

@api_view(['GET', 'POST', 'DELETE'])def tutorial_list(request):    if request.method == 'GET':        tutorials = Tutorial.objects.all()                title = request.GET.get('title', None)        if title is not None:            tutorials = tutorials.filter(title__icontains=title)                tutorials_serializer = TutorialSerializer(tutorials, many=True)        return JsonResponse(tutorials_serializer.data, safe=False)        # 'safe=False' for objects serialization

Retrieve a single object

Find a single Tutorial with an id:

@api_view(['GET', 'PUT', 'DELETE'])def tutorial_detail(request, pk):    # ... tutorial = Tutorial.objects.get(pk=pk)     if request.method == 'GET':         tutorial_serializer = TutorialSerializer(tutorial)         return JsonResponse(tutorial_serializer.data) 

Update an object

Update a Tutorial by the id in the request:

@api_view(['GET', 'PUT', 'DELETE'])def tutorial_detail(request, pk):    # ... tutorial = Tutorial.objects.get(pk=pk)    # ...     elif request.method == 'PUT':         tutorial_data = JSONParser().parse(request)         tutorial_serializer = TutorialSerializer(tutorial, data=tutorial_data)         if tutorial_serializer.is_valid():             tutorial_serializer.save()             return JsonResponse(tutorial_serializer.data)         return JsonResponse(tutorial_serializer.errors, status=status.HTTP_400_BAD_REQUEST) 

Delete an object

Delete a Tutorial with the specified id:

@api_view(['GET', 'PUT', 'DELETE'])def tutorial_detail(request, pk):    # ... tutorial = Tutorial.objects.get(pk=pk)    # ...     elif request.method == 'DELETE':         tutorial.delete()         return JsonResponse({'message': 'Tutorial was deleted successfully!'}, status=status.HTTP_204_NO_CONTENT)

Delete all objects

Delete all Tutorials from the database:

@api_view(['GET', 'POST', 'DELETE'])def tutorial_list(request):    # ...        elif request.method == 'DELETE':        count = Tutorial.objects.all().delete()        return JsonResponse({'message': '{} Tutorials were deleted successfully!'.format(count[0])}, status=status.HTTP_204_NO_CONTENT)

Find all objects by condition

Find all Tutorials with published = True:

@api_view(['GET'])def tutorial_list_published(request):    tutorials = Tutorial.objects.filter(published=True)            if request.method == 'GET':         tutorials_serializer = TutorialSerializer(tutorials, many=True)        return JsonResponse(tutorials_serializer.data, safe=False)

Test the CRUD with APIs

Run our Django Project with command: python manage.py runserver 8080.
The console shows:

python manage.py runserver 8080Performing system checks...System check identified no issues (0 silenced).March 30, 2020 - 16:24:01Django version 2.1.15, using settings 'DjangoRestApiMongoDB.settings'Starting development server at http://127.0.0.1:8080/Quit the server with CTRL-BREAK.

Using Postman, we’re gonna test all the Apis above.

  1. Create a new Tutorial using POST /tutorials Api
  2. Retrieve all Tutorials using GET /tutorials Api
  3. Retrieve a single Tutorial by id using GET /tutorials/:id Api
  4. Update a Tutorial using PUT /tutorials/:id Api
  5. Find all Tutorials which title contains ‘ngo’: GET /tutorials?title=ngo
  6. Find all published Tutorials using GET /tutorials/published Api
  7. Delete a Tutorial using DELETE /tutorials/:id Api
  8. Delete all Tutorials using DELETE /tutorials Api

Conclusion

Today, we’ve learned how to create MongoDB & Django Rest Framework for Rest Apis CRUD example. We also know way to connect Django application with MongoDB database, create a Django Model, migrate it to database, write the Views and define Url patterns for handling all CRUD operations.

Happy learning! See you again.

Post a Comment

Previous Post Next Post