Coder Social home page Coder Social logo

django-poll's Introduction

The Django Voting Poll App

Created following the official tutorial for learning Django.

Workflow

python manage.py runserver
python manage.py migrate
python manage.py makemigrations polls
python manage.py shell # Use quit() or Ctrl-Z plus Return to exit
python manage.py createsuperuser
python manage.py test polls # run the sample test

Site is served at: http://127.0.0.1:8000/

http://127.0.0.1:8000/admin/

Routes

/polls/
/polls/34
/polls/34/results/
/polls/34/vote/

The polls app

From the "Writing your first Django app" article on the official Django docs site, these are notes from the four step tutorial.

I will just provide notes on the steps here for review purposes and reference. The article itself should be used as the full guide.

Step 1

Add a new app:

python manage.py startapp polls

Update the mysitepolls/views.py with a response.

Create a new file polls.urls.py and unclude url patterns.

Update mysite/urls.py to add a path to the polls url.

Step 2

Part 2 of the official Django tutorial.

To use some of the default apps we need at least one database table.

python manage.py migrate
Operations to perform:
  Apply all migrations: admin, auth, contenttypes, sessions
Running migrations:
  Applying contenttypes.0001_initial... OK
  Applying auth.0001_initial... OK
  Applying admin.0001_initial... OK
  Applying admin.0002_logentry_remove_auto_add... OK
  Applying admin.0003_logentry_add_action_flag_choices... OK
  Applying contenttypes.0002_remove_content_type_name... OK
  Applying auth.0002_alter_permission_name_max_length... OK
  Applying auth.0003_alter_user_email_max_length... OK
  Applying auth.0004_alter_user_username_opts... OK
  Applying auth.0005_alter_user_last_login_null... OK
  Applying auth.0006_require_contenttypes_0002... OK
  Applying auth.0007_alter_validators_add_error_messages... OK
  Applying auth.0008_alter_user_username_max_length... OK
  Applying auth.0009_alter_user_last_name_max_length... OK
  Applying auth.0010_alter_group_name_max_length... OK
  Applying auth.0011_update_proxy_permissions... OK
  Applying auth.0012_alter_user_first_name_max_length... OK
  Applying sessions.0001_initial... OK

Create two models in from django.db import models in polls/models.py

class Question(models.Model):
    question_text = models.CharField(max_length=200)
    pub_date = models.DateTimeField("date published")

class Choice(models.Model):
    question = models.ForeignKey(Question, on_delete=models.CASCADE)
    choice_text = models.CharField(max_length=200)
    votes = models.IntegerField(default=0): Question and Choice.

Add a reference to its configuration class in the INSTALLED_APPS setting in mysite/settings.py.

INSTALLED_APPS = [
    "polls.apps.PollsConfig",
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
]

Run the migrations command:

python manage.py makemigrations polls
Migrations for 'polls':
  polls\migrations\0001_initial.py
    - Create model Question
    - Create model Choice

Run migrate again to create those model tables in your database:

python manage.py migrate

invoke the Python interactive Python shell:

python manage.py shell

Add

Register models

In polls/admin.py add the question:

polls/admin.pyfrom django.contrib import admin
from .models import Question

admin.site.register(Question)

Next, create user, answer the questions and run the server:

python manage.py createsuperuser
python manage.py runserver

Goto: http://127.0.0.1:8000/admin/

You can login and see the form which was automatically generated from the Question model.

Note if you see an error like this:

name 'HttpResponseRedirect' is not defined

That means you need to import that like this:

from django.http import HttpResponse, HttpResponseRedirect

I'm not going to repeat all the demo code shown on the tutorial, as this is just a quick reference to the official docs.

Step 3

Add views to polls/views.py such as this:

def detail(request, question_id):
    return HttpResponse("You're looking at question %s." % question_id)

Add the new view to the urlpatterns array in the polls/urls.py like this:

# ex: /polls/5/
path("<int:question_id>/", views.detail, name="detail"),

Then in polls/view.py, update the index() view to displays the latest 5 poll questions:

def index(request):
    latest_question_list = Question.objects.order_by("-pub_date")[:5]
    output = ", ".join([q.question_text for q in latest_question_list])
    return HttpResponse(output)

Django will look for templates in a directory within apps, se we create this file: polls/templates/polls/index.html

{% if latest_question_list %}
    <ul>
    {% for question in latest_question_list %}
        <li><a href="/polls/{{ question.id }}/">{{ question.question_text }}</a></li>
    {% endfor %}
    </ul>
{% else %}
    <p>No polls are available.</p>
{% endif %}

Next update polls/views.py to use the template:

from django.http import HttpResponse
from django.template import loader
from .models import Question

def index(request):
    latest_question_list = Question.objects.order_by("-pub_date")[:5]
    template = loader.get_template("polls/index.html")
    context = {
        "latest_question_list": latest_question_list,
    }
    return HttpResponse(template.render(context, request))

The render() shortcut

There is a render() shortcut to load a template, fill a context and return an HttpResponse object with the result of the rendered template.

from django.shortcuts import render

def index(request):
    latest_question_list = Question.objects.order_by("-pub_date")[:5]
    context = {"latest_question_list": latest_question_list}
    return render(request, "polls/index.html", context)

There is also a shortcut called get_object_or_404. We go from this detail page:

def detail(request, question_id):
    try:
        question = Question.objects.get(pk=question_id)
    except Question.DoesNotExist:
        raise Http404("Question does not exist")
    return render(request, "polls/detail.html", {"question": question})

To this:

def detail(request, question_id):
    question = get_object_or_404(Question, pk=question_id)
    return render(request, "polls/detail.html", {"question": question})

This helps de-couple the model layer to the view layer.

Template functions

This code snippet is shown to show the choices for a poll:

<ul>
{% for choice in question.choice_set.all %}
    <li>{{ choice.choice_text }}</li>
{% endfor %}
</ul>

That is actually the function question.choice_set.all() that returns an iterable of Choice objects and is suitable for use in the {% for %} tag.

There is a link to the templates documents also shown.

The {% url %} template tag

Remove a reliance on specific URL paths defined in your url configurations by using the {% url %} template tag.

<li><a href="{% url 'detail' question.id %}">{{ question.question_text }}</a></li>

In the urls.py we can add an app name to be used to namespace the url. Then update the url tag like this:

{% url 'polls:detail' question.id %}

Step 4

Step 4 begins with a form.

<form action="{% url 'polls:vote' question.id %}" method="post">
    {% csrf_token %}
    <fieldset>
        <legend><h1>{{ question.question_text }}</h1></legend>
        {% if error_message %}
        <p><strong>{{ error_message }}</strong></p>
        {% endif %} {% for choice in question.choice_set.all %}
        <input
            type="radio"
            name="choice"
            id="choice{{ forloop.counter }}"
            value="{{ choice.id }}"
        />
        <label for="choice{{ forloop.counter }}">{{ choice.choice_text }}</label
        ><br />
        {% endfor %}
    </fieldset>
    <input type="submit" value="Vote" />
</form>

The vote function in polls/views.py looks like this:

def vote(request, question_id):
    question = get_object_or_404(Question, pk=question_id)
    try:
        selected_choice = question.choice_set.get(pk=request.POST["choice"])
    except (KeyError, Choice.DoesNotExist):
        # Redisplay the question voting form.
        return render(
            request,
            "polls/detail.html",
            {
                "question": question,
                "error_message": "You didn't select a choice.",
            },
        )
    else:
        selected_choice.votes += 1
        selected_choice.save()
        # Always return an HttpResponseRedirect after successfully dealing
        # with POST data. This prevents data from being posted twice if a
        # user hits the Back button.
        return HttpResponseRedirect(reverse("polls:results", args=(question.id,)))

The last line avoids having to hardcode a URL in the view function.

See the official docs for the request is an HttpRequest object.

The above redirects to results in the same file:

def results(request, question_id):
    question = get_object_or_404(Question, pk=question_id)
    return render(request, "polls/results.html", {"question": question})

Then create a polls/results.html template:

<h1>{{ question.question_text }}</h1>

<ul>
{% for choice in question.choice_set.all %}
    <li>{{ choice.choice_text }} -- {{ choice.votes }} vote{{ choice.votes|pluralize }}</li>
{% endfor %}
</ul>

<a href="{% url 'polls:detail' question.id %}">Vote again?</a>

Generic views system

There has been some talk about avoiding hardcoding URLs.

The ListView and DetailView generic views abstract the concepts of “display a list of objects” and “display a detail page for a particular type of object” respectively.

Convert our poll app to use the generic views system.

For example in urls.py, this:

    path("", views.index, name="index"),
    path("<int:question_id>/", views.detail, name="detail"),
    path("<int:question_id>/results/", views.results, name="results"),

Becomes this:

    path("", views.IndexView.as_view(), name="index"),
    path("<int:pk>/", views.DetailView.as_view(), name="detail"),
    path("<int:pk>/results/", views.ResultsView.as_view(), name="results"),

Then remove index, detail, and results views and use Django’s generic class views instead.

So this:

def index(request):
    latest_question_list = Question.objects.order_by("-pub_date")[:5]
    context = {"latest_question_list": latest_question_list}
    return render(request, "polls/index.html", context)

becomes this:

class IndexView(generic.ListView):
    template_name = "polls/index.html"
    context_object_name = "latest_question_list"

    def get_queryset(self):
        """Return the last five published questions."""
        return Question.objects.order_by("-pub_date")[:5]

I thought there would be less code.

Part 5: automated testing

In this part we write a test to confirm a bug before fixing it.

The bug: Question.was_published_recently() method returns True if the Question was published within the last day (which is correct) but also if the Question’s pub_date field is in the future (not correct).

class QuestionModelTests(TestCase):
    def test_was_published_recently_with_future_question(self):
        """
        was_published_recently() returns False for questions whose pub_date
        is in the future.
        """
        time = timezone.now() + datetime.timedelta(days=30)
        future_question = Question(pub_date=time)
        self.assertIs(future_question.was_published_recently(), False)

Interesting commenting method there. As a Javascript developer, I'm used to the /* */ code blocks.

The self.assertIs(function.does_something(), True/False) is something I understand. Assertion libraries are the heart of testing.

In this case, future_question.was_published_recently() should be False because it's in the future.

Run the test:

$ python manage.py test polls
======================================================================
FAIL: test_was_published_recently_with_future_question (polls.tests.QuestionModelTests)
was_published_recently() returns False for questions whose pub_date
----------------------------------------------------------------------
Traceback (most recent call last):
  File "C:\Users\timof\repos\django\mysite\polls\tests.py", line 17, in test_was_published_recently_with_future_question
    self.assertIs(future_question.was_published_recently(), False)
AssertionError: True is not False
----------------------------------------------------------------------
Ran 1 test in 0.001s
FAILED (failures=1)

The function currently looks like this in the file: polls\models.py

class Question(models.Model):
    question_text = models.CharField(max_length=200)
    pub_date = models.DateTimeField("date published")
    def __str__(self):
        return self.question_text
    def was_published_recently(self):
        return self.pub_date >= timezone.now() - datetime.timedelta(days=1)

It should return False if its pub_date is in the future. Here is the fix:

def was_published_recently(self):
    now = timezone.now()
    return now - datetime.timedelta(days=1) <= self.pub_date <= now

This is a good example. Testing is hard to get right. When to test, what to test, how to setup a test, etc.

Writing a test to confirm a bug and then fix it creates coverage of a weak point in the code. There might be a time in the future where this starts to fail again in an odd circumstance. Having a test that will then fail will highlight this without manual testing, or worse a user discovering the bug.

The next section improves the polls view by applying the future logic to exclude future dates and limit to five polls.

In polls/views.pi:

class IndexView(generic.ListView):
    template_name = "polls/index.html"
    context_object_name = "latest_question_list"

    def get_queryset(self):
        return Question.objects.filter(pub_date__lte=timezone.now()).order_by("-pub_date")[
            :5
        ]

Remember the "latest_question_list" object is used in the file: polls\templates\polls\index.html

{% if latest_question_list %}
    <ul>
    {% for question in latest_question_list %}
        <li><a href="{% url 'polls:detail' question.id %}">{{ question.question_text }}</a></li>
    {% endfor %}

The formatting of the get_queryset function looks a little odd to me. Coming from a Javascript background, I would find something like this more readable:

def get_queryset(self):
    from_now = pub_date__lte=timezone.now()
    published_questions = Question.objects.filter(from_now)
    ordered_questions = published_questions.order_by("-pub_date")
    last_five = ordered_questions[:5]
    return last_five

That's probably more verbose than most would agree with, but just pointing that out here. The sample code actually includes a description in front of the function which I left out above since I just wanted to focus on the code:

def get_queryset(self):
    """
    Return the last five published questions (not including those set to be
    published in the future).
    """
    return Question.objects.filter(pub_date__lte=timezone.now()).order_by("-pub_date")[
        :5
    ]

For me the variable names become the documentation which is what I prefer. Variable names can be tough.

Five more tests

The test_future_question looks like this:

    def test_future_question(self):
        """
        Questions with a pub_date in the future aren't displayed on
        the index page.
        """
        create_question(question_text="Future question.", days=30)
        response = self.client.get(reverse("polls:index"))
        self.assertContains(response, "No polls are available.")
        self.assertQuerySetEqual(response.context["latest_question_list"], [])

However, I see this error:

======================================================================
ERROR: test_future_question (polls.tests.QuestionIndexViewTests)
Questions with a pub_date in the future aren't displayed on
----------------------------------------------------------------------
Traceback (most recent call last):
  File "C:\Users\timof\repos\django\mysite\polls\tests.py", line 75, in test_future_question
    self.assertQuerySetEqual(response.context["latest_question_list"], [])
AttributeError: 'QuestionIndexViewTests' object has no attribute 'assertQuerySetEqual'

And in fact, all five tests are failing. Given my experience with thise code, it would take a bit more to figure out why. Worth asking ChatGPT about the missing assertQuerySetEqual.

ChatGPT: The assertQuerySetEqual method is provided by the TestCase class from Django's test framework, but it seems that your QuestionIndexViewTests class is not inheriting from TestCase. To resolve this issue, you should make sure that QuestionIndexViewTests inherits from django.test.TestCase or a suitable test case class that provides the assertQuerySetEqual method.

However, the code was already doing this. Asking further, it appears there was some problems with the indentation. I'm not sure actually if my editor which is great for Javascript is appropriate for Python and its tabbing strictness. I feel a little lost with this kind of invisible error. It seems like the editor should be able to help with indentation errors...

Then ChatGPT suggests that assertQuerySetEqual should be assertQuerysetEqual. See the difference? The set is lowercase. I make this change and all the tests pass. However the link to the documentation for this function shows a capital S!

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    🖖 Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. 📊📈🎉

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google ❤️ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.