Γράφοντας το πρώτο σας Django app, μέρος 4

This tutorial begins where Tutorial 3 left off. We’re continuing the web-poll application and will focus on form processing and cutting down our code.

Που να ψάξετε για βοήθεια

If you’re having trouble going through this tutorial, please head over to the Getting Help section of the FAQ.

Write a minimal form

Ας αναβαθμίσουμε το HTML template («polls/detail.html»), από τον προηγούμενο οδηγό, στο οποίο εμφανίζουμε τις λεπτομέρειες της ψηφοφορίας. Αυτή τη φορά θα προσθέσουμε ένα HTML <form> element:

polls/templates/polls/detail.html
<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>

Εδώ συμβαίνουν τα εξής:

  • Το παραπάνω template εμφανίζει ένα radio button για κάθε επιλογή ερώτησης. Η τιμή (value) κάθε radio button είναι το ID αυτού. Το όνομα (name) σε όλα τα radio buttons είναι "choice". Αυτό σημαίνει ότι, όταν κάποιος επιλέγει κάποια από τις πιθανές απαντήσεις και κάνει submit τη φόρμα, τότε θα σταλούν (στον server) τα εξής δεδομένα με τη μέθοδο POST (POST data): choice=# όπου # είναι το ID της επιλεγμένης απάντησης (στέλνεται, δηλαδή, το name του input μαζί με το value του ίδιου σε μια μορφή key=value). Αυτό είναι το βασικό concept των HTML forms.
  • We set the form’s action to {% url 'polls:vote' question.id %}, and we set method="post". Using method="post" (as opposed to method="get") is very important, because the act of submitting this form will alter data server-side. Whenever you create a form that alters data server-side, use method="post". This tip isn’t specific to Django; it’s good web development practice in general.
  • Το forloop.counter δείχνει τον αριθμό που ο βρόγχος επανάληψης for έχει τρέξει. Αυτό το κάνουμε για να λάβει κάθε input element μοναδικό id.
  • Since we’re creating a POST form (which can have the effect of modifying data), we need to worry about Cross Site Request Forgeries. Thankfully, you don’t have to worry too hard, because Django comes with a helpful system for protecting against it. In short, all POST forms that are targeted at internal URLs should use the {% csrf_token %} template tag.

Ας δημιουργήσουμε τώρα ένα Django view το οποίο θα χειριστεί τα δεδομένα τα οποία έγιναν submit από τη φόρμα. Θυμηθείτε ότι στον Οδηγό 3, δημιουργήσαμε ένα URLconf, το οποίο περιλαμβάνει τη γραμμή:

polls/urls.py
path('<int:question_id>/vote/', views.vote, name='vote'),

Δημιουργήσαμε επίσης μια πρόχειρη υλοποίηση της συνάρτησης (view) vote(). Ας αναβαθμίσουμε και αυτό το view για να χειριστεί τα data. Προσθέστε τα ακόλουθα στο αρχείο polls/views.py:

polls/views.py
from django.http import HttpResponse, HttpResponseRedirect
from django.shortcuts import get_object_or_404, render
from django.urls import reverse

from .models import Choice, Question
# ...
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,)))

Ο παραπάνω κώδικας περιλαμβάνει μερικά πράγματα τα οποία δεν έχουμε καλύψει μέχρι τώρα στον οδηγό:

  • Το object request.POST έχει τη μορφή ενός dictionary το οποίο σας επιτρέπει να έχετε πρόσβαση στα submitted data μέσω του ονόματος του key (όπως ακριβώς λειτουργεί ένα dictionary στην Python). Στην περίπτωση μας, το request.POST['choice'] επιστρέφει το ID της επιλεγμένης απάντησης, υπό τη μορφή ενός string. Όλες οι τιμές μέσα το object request.POST έχουν τη μορφή string.

    Σημειώστε ότι το Django παρέχει επίσης ανάλογη πρόσβαση στα GET data (αν χρησιμοποιηθεί το method="get") μέσω του object request.GET – αλλά στο συγκεκριμένο παράδειγμα χρησιμοποιούμε αποκλειστικά την request.POST, για να σιγουρευτούμε ότι τα data θα αλλάξουν (στη βάση δεδομένων) μόνο από κάποιο POST call (και ότι τυχόν από κάπου αλλού).

  • Το object request.POST['choice'] θα κάνει raise το exception KeyError (όπως και ένα κοινό Python dictionary) αν το key choice δεν βρεθεί (δεν υπάρχει, δηλαδή, στα POST data). Ο παραπάνω κώδικας κάνει handle το exception αυτό με το να επανεμφανίσει τη φόρμα ψηφοφορίας με ένα ανάλογο μήνυμα σφάλματος (error message).

  • Αφού αυξήσουμε το πλήθος των ψήφων (κατά ένα, κάθε φορά), ο κώδικας επιστρέφει ένα object HttpResponseRedirect παρά ένα συνηθισμένο HttpResponse. Το object HttpResponseRedirect αρχικοποιείται (initializes) με ένα όρισμα (argument): το URL στο οποίο θα ανακατευθυνθεί ο χρήστης (δείτε τις ακόλουθες δύο παραγράφους σχετικά με το πως χτίζουμε το URL σε αυτή την περίπτωση).

    As the Python comment above points out, you should always return an HttpResponseRedirect after successfully dealing with POST data. This tip isn’t specific to Django; it’s good web development practice in general.

  • Χρησιμοποιούμε τη συνάρτηση reverse() στον constructor της κλάσης HttpResponseRedirect. Η συνάρτηση αυτή βοηθά στο να αποφύγουμε να γράφουμε ολόκληρα τα URLs (hardcode) μέσα στη συνάρτηση view (ή οπουδήποτε αλλού σε Python κώδικα). Σαν ορίσματα παίρνει το όνομα του URL (π.χ αυτό που έχουμε ορίσει ως name μέσα στη συνάρτηση url() στο αρχείο polls/urls.py, με πρόθεμα το όνομα του application για τυχόν namespacing) και τυχόν arguments (είτε positional είτε named) τα οποία έχουν περαστεί μέσα στο URL (αυτά που “αιχμαλωτίστηκαν” λόγω των regular expressions). Στη δικιά μας περίπτωση, χρησιμοποιώντας το URLconf που δημιουργήσαμε στον Οδηγό 3, η κλήση της συνάρτησης reverse() θα επιστρέψει ένα string όπως το παρακάτω:

    '/polls/3/results/'
    

    όπου 3 είναι η τιμή του question.id. Αυτό το URL θα καλέσει το view με το όνομα 'results' για να εμφανίσει τη σελίδα (το rendered template, δηλαδή, polls/detail.html). Όταν λέμε rendered template ας εξηγήσουμε τι εννοούμε. Είναι πολύ απλό. Μέσα σε ένα template (συνήθως μια HTML σελίδα), συνήθως βάζουμε περιεχόμενο το οποίο αναμένουμε να “γεμίσει” με μια τιμή από κάπου (συνήθως από το view που κάλεσε αυτό το template). Είδαμε κάτι τέτοιο στον Οδηγό 3 όπου στο template polls/detail.html βάλαμε την έκφραση {{ choice.choice_text }} μέσα σε ένα <li></li> HTML element. Αυτή η τιμή θα “γεμίσει” από το context που θα περάσει το view στο template (μέσω της render() φυσικά). Το context δεν είναι τίποτε άλλο παρά ένα dictionary όπου τα keys ονομάζονται context variables. Υπολογίστηκαν κάποιες τιμές στο view και με κάποιον τρόπο θα πρέπει να περαστούν στο template. Όταν το template γεμίσει με τις απαραίτητες τιμές τότε έχει γίνει rendered.

Όπως αναφέρθηκε στον Οδηγό 3, το request είναι ένα object της κλάσης HttpRequest. Για περισσότερες πληροφορίες σχετικά με τα objects της κλάσης HttpRequest, δείτε το άρθρο εγχειρίδιο του request και response.

Όταν κάποιος ψηφίσει στην ερώτηση, τότε το view vote() θα ανακατευθύνει το χρήστη στη σελίδα των αποτελεσμάτων για αυτή την ερώτηση. Ας γράψουμε αυτό το view:

polls/views.py
from django.shortcuts import get_object_or_404, render


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

Αυτό το view είναι σχεδόν πανομοιότυπο με το detail() από τον Οδηγό 3. Η μόνη διαφορά είναι το όνομα του template. Θα φτιάξουμε αυτό τον πλεονασμό αργότερα. Μας αρέσει να ακολουθάμε την τακτική DRY (don’t repeat yourself).

Τώρα, δημιουργήστε το template με το όνομα polls/results.html:

polls/templates/polls/results.html
<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>

Μεταβείτε στη σελίδα /polls/1/ και ψηφίστε. Κάθε φορά που ψηφίζετε θα βλέπετε τη σελίδα με τα αποτελέσματα της ερώτησης. Αν κάνετε submit τη φόρμα δίχως να επιλέξετε κάποια απάντηση θα δείτε ένα μήνυμα σφάλματος.

Σημείωση

Ο κώδικας του view vote() έχει ένα μικρό προβληματάκι. Πρώτα ρωτάει την database (μέσω της get()) για την ύπαρξη του choice με το συγκεκριμένο ID και αν βρεθεί επιστρέφει το ανάλογο object που είναι instance της polls.models.Choice class. Το object αυτό το ονομάζουμε selected_choice. Κατόπιν υπολογίζουμε τη νέα τιμή του votes και μετά αποθηκεύουμε τη νέα τιμή (αυξημένη κατά ένα) πίσω στην βάση δεδομένων. Εδώ όμως δημιουργείται πρόβλημα. Αν δύο χρήστες προσπαθήσουν να ψηφίσουν στην ίδια ερώτηση ακριβώς την ίδια χρονική στιγμή, τότε δεν θα δουλέψει όπως πρέπει: Η ίδια τιμή, ας πούμε 42, θα είναι η αρχική (votes) και για τους δύο χρήστες. Τότε και στους δύο, η νέα τιμή 43 θα εμφανιστεί ως αποτέλεσμα, αλλά η τιμή 44 θα έπρεπε να είναι η αναμενόμενη (που δεν είναι).

Αυτό ονομάζεται race condition. Αν ενδιαφέρεστε μπορείτε να διαβάσετε την αναφορά στο αποφεύγοντας τα race conditions χρησιμοποιώντας την F για να μάθετε πως μπορείτε να αποφύγετε τέτοιες (ακραίες) καταστάσεις.

Χρήση των generic views: Ο λιγότερος κώδικας είναι καλύτερος

The detail() (from Tutorial 3) and results() views are very short – and, as mentioned above, redundant. The index() view, which displays a list of polls, is similar.

These views represent a common case of basic web development: getting data from the database according to a parameter passed in the URL, loading a template and returning the rendered template. Because this is so common, Django provides a shortcut, called the «generic views» system.

Τα generic views απλοποιούν αυτές τις τόσο συνηθισμένες τακτικές σε βαθμό που δεν χρειάζεται να γράψετε καθόλου Python κώδικα για να φτιάξετε ένα app.

Let’s convert our poll app to use the generic views system, so we can delete a bunch of our own code. We’ll have to take a few steps to make the conversion. We will:

  1. Μετατροπή των URLconf.
  2. Διαγραφή παλιού κώδικα, αχρησιμοποίητα views.
  3. Εισαγωγή σε νέου είδους views, βασισμένα πάνω στα generic views του Django.

Συνεχίστε να διαβάζετε για λεπτομέρειες.

Γιατί αυτή η σύγχυση με τον κώδικα;

Σε γενικές γραμμές, όταν γράφετε ένα Django app, θα καταλάβετε από την αρχή αν θα πρέπει να χρησιμοποιήσετε τα generic views (για την επίλυση του προβλήματος σας), παρά να φτάσετε στο τέλος και μετά να ξαναγράψετε τον κώδικα (όπως κάναμε εμείς τώρα). Εδώ ακολουθήσαμε αυτή την τακτική προκειμένου να σας δείξουμε πως γράφετε ένα view “με το δύσκολο τρόπο” προκειμένου να επικεντρωθείτε σε βασικά concepts.

Θα πρέπει να γνωρίζετε βασικά μαθηματικά προτού χρησιμοποιήσετε το κομπιουτεράκι.

Βελτιώση του URLconf

Ανοίξτε πρώτα το URLconf αρχείο polls/urls.py και αλλάξτε το σε:

polls/urls.py
from django.urls import path

from . import views

app_name = 'polls'
urlpatterns = [
    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'),
    path('<int:question_id>/vote/', views.vote, name='vote'),
]

Note that the name of the matched pattern in the path strings of the second and third patterns has changed from <question_id> to <pk>.

Βελτίωση των views

Επόμενο βήμα είναι να αφαιρέσουμε τα παλιά (πλέον) index, detail και results views και να χρησιμοποιήσουμε τα generic views του Django. Για να γίνει αυτό, ανοίξτε το αρχείο polls/views.py και αλλάξτε το σε:

polls/views.py
from django.http import HttpResponseRedirect
from django.shortcuts import get_object_or_404, render
from django.urls import reverse
from django.views import generic

from .models import Choice, Question


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]


class DetailView(generic.DetailView):
    model = Question
    template_name = 'polls/detail.html'


class ResultsView(generic.DetailView):
    model = Question
    template_name = 'polls/results.html'


def vote(request, question_id):
    ... # same as above, no changes needed.

Χρησιμοποιούμε δύο generic views: Το ListView και το DetailView. Αντίστοιχα, αυτά τα δύο views απλοποιούν τα concepts της «προβολής μια λίστας από objects» και της «προβολής μιας σελίδας λεπτομερειών για ένα συγκεκριμένου τύπου object.»

  • Κάθε generic view χρειάζεται να γνωρίζει το μοντέλο με το οποίο θα συνεργαστεί. Αυτό επιτυγχάνεται με το attribute model.
  • Το DetailView generic view περιμένει μια τιμή ενός primary key (ID, αν θέλετε) η οποία έχει “αιχμαλωτιστεί” στο URL υπό το όνομα "pk". Αυτός είναι ο λόγος που μετονομάσαμε το question_id σε pk.

Από προεπιλογή, το generic view DetailView χρησιμοποιεί ένα template με το όνομα <appname>/<modelname>_detail.html. Στην περίπτωση μας θα χρησιμοποιήσει το template με το όνομα "polls/question_detail.html". Μπορούμε να παρακάμψουμε (override) αυτή τη συμπεριφορά, θέτοντας την τιμή του attribute template_name σε μια δική μας. Ορίζουμε, επίσης, ένα δικό μας template_name και για το view results – αυτό εξασφαλίζει ότι το results view και το detail view θα έχουν διαφορετική εμφάνιση όταν γίνουν rendered, παρά το γεγονός ότι και τα δύο είναι τύπου DetailView.

Ομοίως, το generic view ListView χρησιμοποιεί, εξ ορισμού, ένα template με το όνομα <appname>/<modelname>_list.html. Παρακάμπτουμε, και εδώ, τη συμπεριφορά αυτή θέτοντας τη τιμή του attribute template_name στο string "polls/index.html", το οποίο έχουμε ήδη υλοποιήσει.

In previous parts of the tutorial, the templates have been provided with a context that contains the question and latest_question_list context variables. For DetailView the question variable is provided automatically – since we’re using a Django model (Question), Django is able to determine an appropriate name for the context variable. However, for ListView, the automatically generated context variable is question_list. To override this we provide the context_object_name attribute, specifying that we want to use latest_question_list instead. As an alternative approach, you could change your templates to match the new default context variables – but it’s a lot easier to tell Django to use the variable you want.

Τρέξτε τον server και παίξτε με τη νέα σας εφαρμογή που είναι βασισμένη στα generic views.

Για όλες τις λεπτομέρειες πάνω στα generic views, δείτε το άρθρο εγχειρίδιο (documentation) των generic views.

Όταν είστε εξοικειωμένοι με τις φόρμες και τα generic views διαβάστε το πέμπτο μέρος αυτού του οδηγού για να μάθετε πως να κάνετε τεστ (test) στην εφαρμογή σας.