Django for Designers/Adding models
Part 3: Models and our database
Remember to make a new branch for section 3 based off the official branch!
# in django-for-designers/myproject
$ git branch my-branch-3 origin/pre-part-3
$ git checkout my-branch-3
Okay, being able to design your URLs however you want, without them having to correspond to your actual file structure, is pretty neat. But besides that, we haven't done anything yet that you couldn't do with just HTML, CSS, and JavaScript. Time to fix that.
One key thing that separates web apps from web pages is that apps typically store data somewhere. The code that runs the app chooses what data seems most important to show at the moment. When writing an app using Django, we configure the storage through Django models.
Every Django app comes with a models.py in which you list each kind of data you want to store. These are configured through Python classes (to name the kind of data) with a sequence of attributes (which control the pieces of data that make up the model). If you're familiar with, or interested in, SQL, these correspond to tables and columns respectively.
The models that you define here are built on top of Django's object-relational mapper, or ORM. Because you define your models as Python objects, you can effectively write queries against a relational database by instead coding in Python with reference to your objects. Django then maps your Python code into relational database queries in SQL.
By configuring your data access in Python, Django makes it easy to take advantage of your data layout in other places. For example, you'll see later how the ORM makes it easy to automatically generate forms that ask for exactly the information you need. It is also generally convenient to write your entire app in one language, rather than being required to switch to SQL to do queries. The ORM also handles "escaping," which makes your SQL queries handle Unicode and other strange characters properly. These conveniences let you avoid mojibake-style data corruption and SQL injection attacks effortlessly.Creating a basic model
Let's make it so we can store some real live data in our application! In our simple bookmarking app, we’ll create two models: Bookmarks and Tags. What information do we need to store about these objects?
A bookmark has:
- A URL
- A title (optionally)
- A timestamp for when the bookmark was made
A tag has:
- The tag slug/name
A tag also needs to know which bookmarks it applies to.
Open up bookmarks/models.py in your editor.
from django.db import models
class Bookmark(models.Model):
url = models.URLField()
timestamp = models.DateTimeField(auto_now_add=True)
title = models.CharField(max_length=200, blank=True, default="")
- We've made a Bookmark class that inherits from Django's model class.
- We gave it a url, which is a URLField (a field that expects a string that parses as a valid URL).
- We gave it a timestamp, which is a DateTimeField (a field that expects a Python datetime object). auto_now_add means that the field will automatically set itself to the time when the model is created.
- Finally, we gave it a title, which is a CharField (a field that expects a string of some length or less, in this case less than 200 characters). "blank=True" means that Django will permit this field to be blank. We also set a default of the empty string, so that if someone fails to specify a title we permit that.
Now let's add a tag model! In the same file, below the Bookmark model:
class Tag(models.Model):
bookmarks = models.ManyToManyField(Bookmark)
slug = models.CharField(max_length=50, unique=True)
In relational databases, there are two basic types of relationships. In a ForeignKey relationship, one model relates to one and only one other model. You could imagine this like a Car model and a Wheel model. Each Wheel instance belongs to one and only one Car, so it would have a ForeignKey field for its car. The car, of course, can have multiple wheels.
With a ManyToManyField, on the other hand, the relationship isn't exclusive for either of the models involved. For instance, you could imagine having a Pizza model and a Topping model. Each Pizza can have multiple Toppings, and each Topping can be on multiple Pizzas.
(There are other variants of these fields, but this is the basic concept. See https://docs.djangoproject.com/en/dev/ref/models/fields/#module-django.db.models.fields.related for more details for how these relationship fields work in Django.)
In this case, each Bookmark can have multiple Tags, and each Tag can apply to multiple Bookmarks. So we use a ManyToManyField to store that information.Creating database tables for your app (carefully, with South)
Now that we’ve made our first version of our models file, let’s set up our bookmarks app to work with South so that we can make migrations with it in the future!
On the command line, write:
# in django-for-designers/myproject
$ python manage.py schemamigration bookmarks --initial
As you can see, that’s created a migrations directory for us, and automatically made a new migration file inside it.
# in django-for-designers/myproject
$ ls bookmarks/migrations/
0001_initial.py __init__.py
All we need to do now is apply our new migration:
# in django-for-designers/myproject
$ python manage.py migrate bookmarks
Great! Now our database file knows about bookmarks and its new models, and if we need to change our models, South is set up to handle those changes. We’ll come back to South later.
python manage.py syncdb
Add and commit all your work, including the migrations folder that South generated for you!
Add some bookmarks via the command line
Now, let’s hop into the interactive Python shell and play around with the free API ("Application programming interface" -- APIs aren't just data doodads that hip web startups provide for you, they're an important concept in software architecture.) that Django gives you. To invoke the Python shell, use this command:
# in django-for-designers/myproject
$ python manage.py shell
- Making sure bookmarks (and any other apps you might have) are on the right path to be imported.
- Setting the DJANGO_SETTINGS_MODULE environment variable, which gives Django the path to your settings.py file.
Once you’re in the shell, let's explore the database API. Let's import the model classes we just wrote:
>>> from bookmarks.models import Bookmark, Tag
List all the current Bookmarks:
>>> Bookmark.objects.all()
[]
How many bookmarks is this?
Let's add a bookmark:
>>> b = Bookmark(url="http://www.bringinthecats.com/", title="Funny link")
Try getting the ID number of our new bookmark 'b' by typing:
>>> b.id
Python has given you back your prompt because the value of b.id is None.
Save your bookmark to the database. In Django, you have to call save() explicitly.
>>> b.save()
Now, try again to get the id of the Bookmark instance.
>>> b.id
1
Access the database columns (Fields, in Django parlance) as Python attributes:
>>> b.title
"Funny link"
>>> b.timestamp
datetime.datetime(2011, 12, 1, 3, 3, 55, 841929)
We can change the bookmark title by changing its title attribute, then calling save().
>>> b.title = "WHEEEE"
>>> b.save()
>>> b.title
"WHEEEE"
If we now ask Django to show a list of all the Bookmark objects available, we can see it's no longer an empty list!
>>> Bookmark.objects.all()
[<Bookmark: Bookmark object>]
Fix the hideous default model representation
Use your text editor to open the bookmarks/models.py file and adding a __unicode__() method to both Bookmark and Tag:
class Bookmark(models.Model):
url = models.URLField()
timestamp = models.DateTimeField(auto_now_add=True)
title = models.CharField(max_length=200, blank=True, default="")
def __unicode__(self):
return self.url
class Tag(models.Model):
bookmark = models.ManyToManyField(Bookmark)
slug = models.CharField(max_length=50, unique=True)
def __unicode__(self):
return self.slug
Let's see our new shiny __unicode__ methods in action. Start a new Python interactive shell by running:
# in django-for-designers/myproject
$ python manage.py shell
>>> from bookmarks.models import Bookmark, Tag
>>> Bookmark.objects.all()
[<Bookmark: http://www.bringinthecats.com/>]
Save and commit your changes.
Adding more data via the shell
>>> b = Bookmark(url="http://www.google.com")
>>> b.save()
>>> b = Bookmark(url="https://us.pycon.org/2013/", title="PyCon US website")
>>> b.save()
>>> b = Bookmark(url="https://www.djangoproject.com")
>>> b.save()
We've created a bunch of bookmarks, but no tags! Let's change that.
>>> b = Bookmark.objects.all()[0]
>>> b
<Bookmark: http://www.bringinthecats.com/>
>>> b.tag_set.create(slug="cats")
<Tag: cats>
>>> b.tag_set.create(slug="music")
<Tag: music>
>>> b.tag_set.create(slug="funny")
<Tag: funny>
>>> b.save()
Slicing and dicing Django data
There are many methods for searching across your Django models.
We can filter our bookmarks, for instance for ones with a particular URL:
>>> foo = Bookmark.objects.filter(url="http://www.bringinthecats.com/")
>>> foo
[<Bookmark: http://www.bringinthecats.com/>]
Or for bookmarks which have titles:
>>> titled_bookmarks = Bookmark.objects.exclude(title="")
>>> titled_bookmarks
[<Bookmark: http://www.bringinthecats.com/>, <Bookmark: https://us.pycon.org/2013/>]
If you try to use filter to search for a bookmark that does not exist, filter will give you the empty list.
>>> Bookmark.objects.filter(title="Who framed Roger Rabbit?")
[]
The get method, on the other hand, returns exactly one hit. If it finds zero matches, or more than one match, it will raise an exception.
>>> Bookmark.objects.get(id=1)
<Bookmark: http://www.bringinthecats.com/>
>>> Bookmark.objects.get(id=20)
Traceback (most recent call last):
...
DoesNotExist: Bookmark matching query does not exist.
More information on making queries with Django's ORM can be found in the Django docs at https://docs.djangoproject.com/en/dev/topics/db/queries/#retrieving-objects.
Part 3.5: Changing our mind and adding users
I don't mean like the number of people using it--I mean a way to store different users' accounts and keep track of who owns which bookmarks. So let's change our app to include this feature.
Lucky for us, Django comes with an app for user accounts and authentication from the get-go! In fact, it's already installed.If you look back at your settings.py file, you'll see that in INSTALLED_APPS, there is an entry for 'django.contrib.auth'.
INSTALLED_APPS = (
'django.contrib.auth',
'django.contrib.contenttypes',
'django.contrib.sessions',
'django.contrib.sites',
'django.contrib.messages',
'django.contrib.staticfiles',
# Uncomment the next line to enable the admin:
# 'django.contrib.admin',
# Uncomment the next line to enable admin documentation:
# 'django.contrib.admindocs',
'south',
'bookmarks',
)
That's our built-in authentication app!
If it isn't already open, let's open the Django shell and play with this app a bit.
# in django-for-designers/myproject
$ python manage.py shell
>>> from django.contrib.auth.models import User
>>> User.objects.all()
[<User: karen>]
There's already a User here. How can that be?
What is our user account's id number?
>>> me = User.objects.all()[0]
>>> me.id
1
Neato!
Add user field to bookmark
Now we need to create a relationship between the built-in User model and our Bookmark model.
First, in our models.py file, we need to import django.contrib.auth's built-in User model so that we can refer to it in our models.
from django.db import models
from django.contrib.auth.models import User
Now we need to think--what kind of relationship do users and bookmarks have?
class Bookmark(models.Model):
author = models.ForeignKey(User)
title = models.CharField(max_length=200, blank=True, default="")
url = models.URLField()
timestamp = models.DateTimeField(auto_now_add=True)
def __unicode__(self):
return self.url
While we're at it, let's update the __unicode__ method too to let us know to whom a bookmark belongs to.
class Bookmark(models.Model):
author = models.ForeignKey(User)
title = models.CharField(max_length=200, blank=True, default="")
url = models.URLField()
timestamp = models.DateTimeField(auto_now_add=True)
def __unicode__(self):
return "%s by %s" % (self.url, self.author.username)
Make a migration in South
Now that we've added a field to our model, we are going to need to create a database migration. South can help us do this!
To create our migration, run
# in django-for-designers/myproject
$ python manage.py schemamigration bookmarks --auto
Eep! Before it will make our migration file, South wants some information from us:
? The field 'Bookmark.author' does not have a default specified, yet is NOT NULL.
? Since you are adding this field, you MUST specify a default
? value to use for existing rows. Would you like to:
? 1. Quit now, and add a default to the field in models.py
? 2. Specify a one-off value to use for existing columns now
? Please select a choice: 2
We could in theory modify our models to specify a default value for author, or make it optional. But neither of those sound like good options.
So we'll choose 2, to set up a default value for our new author field just for the purposes of this migration.
? Please enter Python code for your one-off default value.
? The datetime module is available, so you can do e.g. datetime.date.today()
>>>
We then need to come up with a default value. Well, right now there's only one User in our system who the sample bookmarks we'd entered so far could belong to--our superuser account, which (if you don't remember) had an ID number of 1.
? Please enter Python code for your one-off default value.
? The datetime module is available, so you can do e.g. datetime.date.today()
>>> 1
+ Added field author on bookmarks.Bookmark
Created 0002_auto__add_field_bookmark_author.py. You can now apply this migration with: ./manage.py migrate bookmarks
Remember, the first step creates the migration, but doesn't run it. So let's do what South says and run a command to apply our migration!
# in django-for-designers/myproject
$ python manage.py migrate bookmarks
Running migrations for bookmarks:
- Migrating forwards to 0002_auto__add_field_bookmark_author.
> bookmarks:0002_auto__add_field_bookmark_author
- Loading initial data for bookmarks.
Installed 0 object(s) from 0 fixture(s)
Save and commit your work to add users to your bookmarks app!
Templates and links for login/logout/etc
Django's auth app comes with built-in views, which we can use to handle login and logout functionality for our users. Once we point some URLs at them, that is.
First, let's edit urls.py:
urlpatterns = patterns('',
url(r'^$', 'bookmarks.views.index', name='home'),
url(r'^bookmarks/$', 'bookmarks.views.index', name='bookmarks_view'),
url(r'^tags/(\w+)/$', 'bookmarks.views.tag'),
url(r'^login/$', 'django.contrib.auth.views.login'),
)
Now if anyone goes to localhost:8000/login/, the built-in login view will get triggered.
Run your dev server and try that. What error do you see?
While there is a built-in login view, there is no built-in login template. We need to build one for it. While we could put it anywhere and tell login explicitly where to look, by default the login view expects the template to reside at templates/registration/login.html. So we may as well put it there.
# in django-for-designers/myproject
$ mkdir bookmarks/templates/registration
Create a new file within the new registration directory called login.html, and put this inside:
{% extends "base.html" %}
{% block subheader %}Login{% endblock %}
{% block content %}
{% if form.errors %}
<p>Your username and password didn't match. Please try again.</p>
{% endif %}
<form method="post" action="{% url 'django.contrib.auth.views.login' %}">
{% csrf_token %}
<table>
<tr>
<td>{{ form.username.label_tag }}</td>
<td>{{ form.username }}</td>
</tr>
<tr>
<td>{{ form.password.label_tag }}</td>
<td>{{ form.password }}</td>
</tr>
</table>
<input type="submit" value="login" />
<input type="hidden" name="next" value="/" />
</form>
{% endblock %}
We need to add a logout view too! Let's do that. Back in urls.py:
url(r'^login/$', 'django.contrib.auth.views.login'),
url(r'^logout/$', 'django.contrib.auth.views.logout', {'next_page': '/'})
Now we need to add a login/logout link to our site so people can actually use these views! We want this link to be at the top of every page, so we'll edit our base template:
<body>
<div id="container">
<div id="header">
{% block bookmark_widget %}
{% endblock %}
<div id="authentication">
{% if user.is_authenticated %}
Hi {{user}}! <a href="{% url 'django.contrib.auth.views.logout' %}">Logout</a>
{% else %}
<a href="{% url 'django.contrib.auth.views.login' %}">Login</a>
{% endif %}
</div>
<h1><a href="/">My bookmarking app</a></h1>
</div>
[ ... ]
Check http://localhost:8000 and try logging in with the superadmin username and password you created before! It should work. :)
The answer is in the function we are using to render our templates, render(). render() automatically uses the request we sent it to create a Django RequestContext, which contains a bunch of extra context variables that get sent along to every view that uses a RequestContext. The Django auth app adds the current user to the RequestContext automatically. This is handy, since you don't want to have to look up the current user to every single view you ever write separately, just so the nav section on your website will work everywhere!
There are other functions that return an HTMLResponse, like render(), but don't include a RequestContext. render_to_response() is one common shortcut function that doesn't include it by default; another was the HttpResponse() function we used earlier! Just something to remember--if you're trying to send a piece of data to almost every page in your web app, 1.) you probably want to use render() in your views, and 2.) you want to find a way to add your data to your app's RequestContext. (https://docs.djangoproject.com/en/dev/ref/templates/api/#django.template.RequestContext says more about this!)Save and commit your changes in adding login/logout functionality.
Part 3.75: Modify views and templates to use model data
Let's edit our views and templates so they use real bookmark data from our database!
Add more data
First, let's add a bit more bookmark and tag data to our database. We'll use a Python loop to add a lot of bookmarks at once.
# in django-for-designers/myproject
$ python manage.py shell
That will bring us to the Django management shell. Within that special Python prompt, type the following:
(Remember, any time you see ">>>", you don't have to type those characters.)
>>> from django.contrib.auth.models import User
>>> me = User.objects.all()[0]
This tells Python we'll be accessing the user model, and then grabs the first user, storing it in a variable called me.
Keeping that Python prompt open, run the following commands:
>>> from bookmarks.models import Bookmark, Tag
>>> import random
>>> tag_names = ['kids', 'read_on_plane', 'send_to_dad']
>>> # Create the Tag objects
>>> for tag_name in tag_names:
... tag = Tag(slug=tag_name)
... tag.save()
...
>>> kids_urls = ['http://pbskids.org/', 'http://www.clubpenguin.com/', 'http://www.aplusmath.com/hh/index.html', 'http://kids.nationalgeographic.com/kids/activities/recipes/lucky-smoothie/', 'http://www.handwritingforkids.com/', 'http://pinterest.com/catfrilda/origami-for-kids/', 'http://richkidsofinstagram.tumblr.com/', 'http://www.dorkly.com/picture/50768/', 'http://www.whyzz.com/what-is-paint-made-of', 'http://yahooligans.com/']
>>> for kids_url in kids_urls:
... b = Bookmark(author=me, url=kids_url)
... b.save()
... how_many_tags = random.randrange(len(tag_names))
... for tag_name in random.sample(tag_names, how_many_tags):
... b.tag_set.add(Tag.objects.get(slug=tag_name))
...
>>> b = Bookmark.objects.get(url='http://yahooligans.com/')
>>> b.title = "Yahoooo!!!!!ligans"
>>> b.save()
This creates a number of bookmarks, tagged appropriately! You can exit the shell by typing:
>>> exit()
Get views.py to talk to our models
Then, we'll need to have our views.py file import the bookmark model and send data to the index view.
We'll add an import statement:
from bookmarks.models import Bookmark
And we'll edit index(request):
def index(request):
bookmarks = Bookmark.objects.all()
context = {
'bookmarks': bookmarks
}
return render(request, 'index.html', context)
Wait! We don't actually want to load every bookmark in our database when we go to the front page.
Instead, let's show the 10 most recent bookmarks:
def index(request):
bookmarks = Bookmark.objects.all().order_by('-timestamp')[:10]
context = {
'bookmarks': bookmarks
}
return render(request, 'index.html', context)
We also want our tag view to show real bookmarks. We only want to show bookmarks that have been tagged with the given tag. So we'll edit the import statement we just added:
from bookmarks.models import Bookmark, Tag
And alter the tag() definition as well:
def tag(request, tag_name):
tag = Tag.objects.get(slug=tag_name)
bookmarks = tag.bookmarks.all()
context = {
'tag': tag,
'bookmarks': bookmarks,
}
return render(request, 'tag.html', context)
Change templates to handle bookmark data
Now, let's modify our templates to use this data.
In index.html, update just block content per the following. Make sure to keep the extends directive and the block subheader directive.
{% block content %}
<ul class="bookmarks">
{% for bookmark in bookmarks %}
<li>
<a class="bookmark-link" href="">{{ bookmark }}</a>
<div class="metadata"><span class="author">Posted by Jane Smith</span> | <span class="timestamp">2012-2-29</span> | <span class="tags"><a href="">funny</a> <a href="">haha</a></span></div>
</li>
{% endfor %}
</ul>
{% endblock %}
Start your development server and look at http://localhost:8000 now. You'll see that instead of the fake HTML we had before, each link's text is the unicode representation of each bookmark in your database. Cool, huh?
That's okay, but the programmer-friendly unicode representation isn't really what we want to go there. Let's fix that:
# in index.html
<a class="bookmark-link" href="{{ bookmark.url }}">
{% if bookmark.title %}{{ bookmark.title }}{% else %}{{ bookmark.url }}{% endif %}
</a>
If the bookmark has a title, we'll make the title the link. Otherwise we'll show the URL. We also fill in the URL on the <a> tag, to make the link work.
We also want to fill in the other metadata, like tags and author info. To do that:
# in index.html
<div class="metadata"><span class="author">Posted by {{ bookmark.author }}</span> | <span class="timestamp">{{ bookmark.timestamp }}</span>
{% if bookmark.tag_set.all %}| <span class="tags">
{% for tag in bookmark.tag_set.all %}
<a href="{% url 'bookmarks.views.tag' tag.slug %}">{{ tag.slug }}</a></span>
{% endfor %}
{% endif %}
</div>
We check if there are any tags for the bookmark, and if so loop over the tags to put each of them in.
Additionally, we use Django's {% url %} tag to generate the URL for our tag's link.
Run
# in django-for-designers/myproject
$ python manage.py runserver
and confirm that your tag links working and our other changes are visible. It's easy to typo or make a mistake with {% url %} tags, so we want to make sure everything is working.
The default timestamp formatting that Django gave us is pretty neat, but in our original mockup we just used the date--not this long timestamp. Fortunately, Django's built-in filters make it easy to format a date or time any way we want! Since we just want to show the date, we'll use the date filter:
<span class="timestamp">{{ bookmark.timestamp|date:"Y-m-d" }}</span>
Spin up your dev server, if you haven't got it running already, and check out your changes! Then save and commit your precious work.
Updating the tag template, and dealing with errors
Let's update our tag.html template to use the same formatting as index.html. Since we're using the same formatting for the list of bookmarks, and there aren't any other content block differences between index.html and tag.html, we can simply modify our tag template to inherit from index.html. Edit tag.html to have only the following contents. (That means you can remove the mockup "block content" section in tags.html. Yay!)
{% extends 'index.html' %}
{% block subheader %}Bookmarks tagged {{ tag.slug }}{% endblock %}
Now we can go to http://localhost:8000/tags/funny and see a nicely styled list of bookmarks tagged with that tag!
What happens if we go to http://localhost:8000/tags/asdfjkl ?
Our application is erroring on one of our view lines:
tag = Tag.objects.get(slug=tag_name)
Fortunately, Django has a shortcut function that can help us -- get_object_or_404().
from django.shortcuts import render, get_object_or_404
from bookmarks.models import Bookmark, Tag
[...]
def tag(request, tag_name):
tag = get_object_or_404(Tag, slug=tag_name)
bookmarks = tag.bookmarks.all()
context = {
'tag': tag,
'bookmarks': bookmarks,
}
return render(request, 'tag.html', context)
Now http://localhost:8000/tags/asdfjkl will throw a nicer 404 error. We could even make a pretty 404.html template to handle such errors!
Save and commit your error-catching work.