"The web framework for perfectionists with deadlines."
"Django was invented to meet fast-moving newsroom deadlines, while satisfying the tough requirements of experienced Web developers."
REpresentational State Transfer (REST) is an architectural style that describes six constraints.
To the extent that systems conform to the constraints of REST they can be called RESTful.
I love reading, and I use GoodReads to make it a somewhat social experience. But GoodReads isn't perfect. It could be better. Better than good. So I decided to start work on...
I love reading, and I use GoodReads to make it a somewhat social experience. But GoodReads isn't perfect. It could be better. Better than good. So I decided to start work on...
...BetterReads! Which will be better than GoodReads!
class Publisher(models.Model): name = models.CharField(max_length=50, unique=True) address = models.TextField()
class Publisher(models.Model): name = models.CharField(max_length=50, unique=True) address = models.TextField()
class PublisherSerializer(serializers.ModelSerializer): class Meta: model = models.Publisher
Note that the serializer knows the model class, and it knows all of the attributes on that class. The ModelSerializer default behavior automatically creates an appropriate serializer field for each attribute on the model
class Publisher(models.Model): name = models.CharField(max_length=50, unique=True) address = models.TextField()
class PublisherSerializer(serializers.ModelSerializer): class Meta: model = models.Publisher
{ 'address': '12345 S. State Street', 'id': 1, 'name': 'Del Rey'}
Note that the serializer knows the model class, and it knows all of the attributes on that class. The ModelSerializer default behavior automatically creates an appropriate serializer field for each attribute on the model
class Person(models.Model): first_name = models.CharField(max_length=50) last_name = models.CharField(max_length=50) email = models.CharField(max_length=50) class Meta: unique_together = ('first_name', 'last_name')
The unique_together constraint creates a multi-column index keeping all combinations of first and last names unique.
class Person(models.Model): first_name = models.CharField(max_length=50) last_name = models.CharField(max_length=50) email = models.CharField(max_length=50) class Meta: unique_together = ('first_name', 'last_name')
class PersonSerializer(serializers.ModelSerializer): email = serializers.EmailField() class Meta: model = models.Person
The unique_together constraint creates a multi-column index keeping all combinations of first and last names unique.
We can customize the way the email field is serialized by giving it a different serializer field. The EmailField serializer will check for a valid e-mail address before accepting the input.
class Tag(models.Model): name = models.CharField(max_length=20, unique=True)
class Tag(models.Model): name = models.CharField(max_length=20, unique=True)
class TagSerializer(serializers.ModelSerializer): books = serializers.SlugRelatedField( slug_field='name', many=True queryset=models.Book.objects.all()) class Meta: model = models.Tag
The slug related field alters the way a data relationship is displayed, by simply displaying one of the attributes of the related object. The attribute it shows is specified in the slug_field.
class Book(models.Model): name = models.CharField(max_length=100) authors = models.ManyToManyField(Person) tags = models.ManyToManyField(Tag) publisher = models.ForeignKey(Publisher, null=True) year = models.IntegerField(null=True) in_print = models.BooleanField(default=True)
class Book(models.Model): name = models.CharField(max_length=100) authors = models.ManyToManyField(Person) tags = models.ManyToManyField(Tag) publisher = models.ForeignKey(Publisher, null=True) year = models.IntegerField(null=True) in_print = models.BooleanField(default=True)
class BookSerializer(serializers.ModelSerializer): tags = serializers.SlugRelatedField( slug_field='name', many=True, queryset=models.Tag.objects.all()) publisher = serializers.SlugRelatedField( slug_field='name', allow_null=True, queryset=models.Publisher.objects.all()) url = serializers.HyperlinkedIdentityField( view_name='book-detail') class Meta: model = models.Book
The HyperlinkedIdentityField creates a link to the current object's representation. The url field is automatically created with the same field serializer if you subclass HyperlinkedModelSerializer instead of ModelSerializer.
{ 'authors': [ 1 ], 'id': 1, 'in_print': True, 'name': 'A Really Good Book', 'publisher': None, 'tags': [ 'Fun', 'Fantasy' ], 'url': 'http://localhost:8000/api/books/1/', 'year': 2016}
Note that authors and tags are both M2M relationships, but they render differently. Is anyone surprised by this?
We're used to talking in terms of HTTP methods
DRF talks in terms of "actions"
Detail view is for actions on an instance, or in other words, any time you have the primary key.
/api/books/1/
List view is for actions when you don't have a primary key.
/api/books/
DRF will happily create a view for every action you need with ViewSets.
class PersonViewSet(viewsets.ModelViewSet): queryset = models.Person.objects.all() serializer_class = serializers.PersonSerializer
DRF will happily create a view for every action you need with ViewSets.
class PersonViewSet(viewsets.ModelViewSet): queryset = models.Person.objects.all() serializer_class = serializers.PersonSerializer
Here's the most interesting thing my viewsets do:
class BookViewSet(viewsets.ModelViewSet): queryset = models.Book.objects.all() serializer_class = serializers.BookSerializer filter_fields = ('year', 'in_print', 'authors', 'name', 'tags', 'publisher')
Note that the ModelViewSet creates all five actions, but there are other viewsets that only create the actions for a read-only resource, or you can mix and match them to get the required functionality for a resource.
from django.conf.urls import include, urlfrom rest_framework import routersfrom api import viewsapi = routers.DefaultRouter()api.register('books', views.BookViewSet)api.register('tags', views.TagViewSet)api.register('publishers', views.PublisherViewSet)api.register('people', views.PersonViewSet)urlpatterns = [ url(r'^', include(api.urls))]
Python 3.5.1 (default, Dec 7 2015, 21:59:08)[GCC 4.2.1 Compatible Apple LLVM 7.0.0(clang-700.1.76)] on darwinType "help", "copyright", "credits" or"license" for more information.
Python 3.5.1 (default, Dec 7 2015, 21:59:08)[GCC 4.2.1 Compatible Apple LLVM 7.0.0(clang-700.1.76)] on darwinType "help", "copyright", "credits" or"license" for more information.
>>> import requests
Python 3.5.1 (default, Dec 7 2015, 21:59:08)[GCC 4.2.1 Compatible Apple LLVM 7.0.0(clang-700.1.76)] on darwinType "help", "copyright", "credits" or"license" for more information.
>>> import requests
>>> from pprint import pprint
Python 3.5.1 (default, Dec 7 2015, 21:59:08)[GCC 4.2.1 Compatible Apple LLVM 7.0.0(clang-700.1.76)] on darwinType "help", "copyright", "credits" or"license" for more information.
>>> import requests
>>> from pprint import pprint
>>> data = requests.get(... 'http://localhost:8000/api/people/'... ).json()
Python 3.5.1 (default, Dec 7 2015, 21:59:08)[GCC 4.2.1 Compatible Apple LLVM 7.0.0(clang-700.1.76)] on darwinType "help", "copyright", "credits" or"license" for more information.
>>> import requests
>>> from pprint import pprint
>>> data = requests.get(... 'http://localhost:8000/api/people/'... ).json()
>>> pprint(data)[{'email': 'kyle@mydomain.com', 'first_name': 'Kyle', 'id': 1, 'last_name': 'Rich'}]
Keyboard shortcuts
↑, ←, Pg Up, k | Go to previous slide |
↓, →, Pg Dn, Space, j | Go to next slide |
Home | Go to first slide |
End | Go to last slide |
b / m / f | Toggle blackout / mirrored / fullscreen mode |
c | Clone slideshow |
p | Toggle presenter mode |
t | Restart the presentation timer |
?, h | Toggle this help |
Esc | Back to slideshow |