/Hack_O_DjangoREST_Tut

Basic Django REST tutorial with example project

Primary LanguagePython

Tutorial Setup

1. Clone repo

  • Make a directory and clone this repo inside
  • cd into the Hack_O_DjangoREST_Tut repo

2. Make a virtual environment

Complete these steps inside the Hack_O_DjangoREST_Tut repo

  • Setup a virtual environment:
    • Depending on where your version of Python 3 is installed you may need to alter the path in the command below
    • virtualenv -p /usr/local/bin/python3 venv3
  • Activate virtual environment:
    • source venv3/bin/activate
    • command above activates the virtual environment your command line prompt should start with (venv3) if it worked

3. Install depencencies with pip

In the top level of the cloned repo there is a requirements.txt file

  • Install depedencies from requirements.txt:
    • pip install -r requirements.txt

4. Start server

  • cd into the example_project directory. Should contain the manage.py file.
  • In directory with manage.py run this command:

Example file structure of a starter project

Below is what the file structure looked like after running a basic setup found here: new_project_setup tree structure of project

Basic building blocks of endpoints

tree structure of project

The image above outlines the pieces that need to be in place for an endpoint to work in the DRF assuming that the basic setup of the project has already been done.

Models

A model just represents a table in your database and the attributes of the model class match the fields in that table. By creating a model and then syncing it with the database Django will tell the database to create a table that matches the model you have created in your project. Below is an example of the Songs model that is in our example_project/example_app/models.py file.

from django.db import models

class Songs(models.Model):
    # no need to declare a primary key field it is auto added with the name of 'id' by Django
    artist_name = models.CharField(max_length=255, default='')
    song_title = models.CharField(max_length=255, default='')
    song_year = models.IntegerField(blank=True, null=True)
    song_lyrics = models.TextField()

    class Meta:
        db_table = 'song_lyrics'

Notice how each attribute in the model class has a datatype to which it is assigned. These data types are defined in the django.db.models module and offer a convienient way to assign data types the values in our models. The class Meta part of a model is a container that Django uses to define meta data about a class object you're using. You will see examples of class Meta throughout Django's classes.

Serializers

A way of serializing and deserializing data. This basically means just taking data and turning it into a different format like JSON or XML. Serializers are basically like translators that allow our Django REST API to talk to different programs using formats that both understand. There are many ways to define a serializer but we have included a simple example using Django's ModelSerializer class in the example_project/example_app/serializers.py file. See below.

class SongSerializer(serializers.ModelSerializer):
    class Meta:
        model = models.Songs
        fields = '__all__'

What we're doing is pointing this serializer at the Songs model we've already created and telling it serialize all of the fields in that model by declaring fields = '__all__' in the class Meta of the serializer. If you wanted to hide some of the fields that are returned to the end user this can be accomplished by declaring only the fields you want listed: fields = ('song_title', 'song_year') would only convert data from the song_title and song_year columns effectively hiding some of the informaton from the end user. You could also declare each attribute explicitly if you liked. Follow the instructions below to inspect this serializer in the Django shell.

  • python manage.py shell --> launches shell from command line

  • Inside shell use the commands below to import our song serializer and then inspect it. Remember to hit return after entering each line.

  from example_app.serializers import SongSerializer 

  serial_instance = SongSerializer()
  print(repr(serial_instance))
  
  # to quit shell
  quit()
  • Example output from shell
SongSerializer():
    id = IntegerField(read_only=True)
    artist_name = CharField(max_length=255, required=False)
    song_title = CharField(max_length=255, required=False)
    song_year = IntegerField(allow_null=True, required=False)
    song_lyrics = CharField(style={'base_template': 'textarea.html'})

We can see from the output that the serializer is taking every field in our Song model and declaring its data type. This is needed to let the serializer know how to translate the data in each field of our model.

Views

Views are where the logic of your API lives, they're primarily concerned with the request/reponse cycle. Each view is responsible for grabbing infromation out of your database by using your model and then serializing or translating that data and returning it to a user. There are many different ways to write views but we will show you two basic ones. The first is a bit longer than the second but will hopefully give you better intuition as to what a view is doing.

@api_view(['GET'])
def song_list(request):
    """
    A function based view that use the api_view decorator to add functionality
    to the view.   
    """
    if request.method == 'GET':
        songs = models.Songs.objects.all()
        serializer = serializers.SongSerializer(songs, many=True)
        return Response(serializer.data)

This is an example of a function based view and can be found in the example_project/example_app/views.py file. The @api_view(['GET']) decorator is used to add some Django specific functionality to the song_list function. Inside this view function you see that we're giving instructions about what should happen if it recieves a GET request. We're telling the view to grab every object in the Songs model and then use the SongSerializer we previously made to translate all of those objects. We then use a Django's Response class to return the translated data from our serializer. While there is still a bit Django magic going on in this function the main takeaway should be that a view handles an incoming request and decides which data to grab and how to respond.

Another very useful bit of built in Django magic are class based views. The view below does basically the same thing as the functional view above but with far less code and has built in logic for different types of requests.

class ListSongs(generics.ListAPIView):
    """
    A class based view that inherits from the generics class. The generics
    class gives you a convinient way to declare views quickly when you only
    need basic functionality or simple CRUD operations.
    """
    
    queryset = models.Songs.objects.all()
    serializer_class = serializers.SongSerializer

This class based view can be found in the example_project/example_app/views.py file. With class based views they only require that you define a queryset and serializer_class. Remember that these attributes just represent our model and our serializer. Class based views using the generics class don't require that you explicitly return the serialized data in a response like the functional view did above. The methods that are inherited from the generics class return the correct data by plugging in your queryset and serializer_class where appropriate behind the scenes.

This view currently only accepts GET requests becasue it's inheriting from the generics.ListAPIView class. If we would like to allow this view to be able to accept POST requests it is as simple as changing the class it inherits from. By using the generics.ListCreateAPIView class instead the endpoint will now accept POST requests and allow us to load data into our database by hitting this view's endpoint.

Please see the docs for the generics classes the DRF offers: generic_views

Linking URLs to views

The last thing we must remember to do is link a URL endpoint to each view we add to our project. The way we're recommending you do this for Hack Oregon projects can be seen below.

This is an example of the contents of the example_project/example_app/urls.py file:

from django.conf.urls import url
from . import views


urlpatterns = [
    #function based view
    url(r'^funcsongs/$', views.song_list),

    # class based view
    url(r'^songs/$', views.ListSongs.as_view()),   
]

Notice the difference in how the functional view and class view are defined. The r'^endpoint_name/$' part of the url declaration is using regex to point to a url that a user will hit to trigger the view.

This urls.py file inside your example_app directory doesn't come standard with the project setup and must be created by you if you're making a new project. By having the urls declared for each app inside a project it gives you the ability to create a single point of entry into your API. This is valuable because as long as the user knows the entry endpoint they can find all the other endpoints easily.

Adjusting urls in the project directory

There is still one last piece that needs to be linked in order for the endpoints to work properly. The endpoints that are served up by Django all must come out of the example_project/example_project/urls.py file. We haven't touched the inner example_project directory much in this tutorial yet. This inner directory is where you adjust the settings of your project and has two files that interest us settings.py and urls.py. These files control all of the top level settings in your project. Every URL that is served in your project must come out of the urls.py file. In order to do this we need to include the URLs we've already declared in our app.

from django.conf.urls import url
from django.contrib import admin
from django.conf.urls import include

urlpatterns = [
    url(r'^admin/', admin.site.urls),
    url(r'^', include('example_app.urls', namespace='example_app')),
]

Notice how we're using the include function to include all of the URLs we previously declared in our app. Doing this gives us the option to add more apps in the future without worrying about URL conflicts. We also have the option of adding a prefix to all URLs for our app here. For example:

url(r'^example_prefix/', include('example_app.urls', namespace='example_app')),

Would cause every URL in our app to look someting like this: https://api_location.com/example_prefix/songs/

Wrapping up

You now know the four main pieces that you need to worry about when adding an endpoint to any project. Those pieces are the model, serializer, view, and url. We would now like eveyone to try their hand at loading some sample data from a csv into this Django project by adding these four pieces and then sending POST requests to load the data. Please follow this link: movie_loader and follow the instructions there.

Next steps

This is a list of links to directories in this repo that will let you explore other aspects of the DRF now that you know the basics

  • movie_loader --> walks through creating a model and loading fake data from a csv to your db using a POST endpoint.
  • filtering --> explores passing arguments to views and then filtering responses with those arguments.