Python: Разработка на фреймворке Django

Теория: Просмотр (CRUD)

Список сущностей готов, поэтому можно переходить к просмотру каждого элемента. Начинаем с маршрутизации:

from django.urls import path

from hexlet_django_blog.article.views import IndexView, ArticleView

urlpatterns = [
    path("", IndexView.as_view()),
    path("<int:id>/", ArticleView.as_view()),
]

В этом месте у маршрута появляется динамическая часть — идентификатор статьи. Подобный маршрут обрабатывает все страницы, имеющие вид /articles/<идентификатор статьи>/. Пара примеров:

/articles/1/
/articles/100/

Этот идентификатор используется в обработчике для выборки из базы нужной статьи.

from django.shortcuts import render, get_object_or_404
from django.views import View

from hexlet_django_blog.article.models import Article


class ArticleView(View):
    def get(self, request, *args, **kwargs):
        article = get_object_or_404(Article, id=kwargs["id"])
        return render(
            request,
            "articles/show.html",
            context={
                "article": article,
            },
        )

Параметр id, который определен в маршруте, приходит в обработчик в kwargs. Он представляет собой словарь. Если параметров в маршруте больше одного, то они передадутся в обработчик в том же порядке, в котором они определены в маршруте:

from django.urls import path

from hexlet_django_blog.article.views import ArticleCommentsView

urlpatterns = [
    ...
    path('<int:article_id>/comments/<int:id>/', ArticleCommentsView.as_view()),
]
# views.py


class ArticleCommentsView(View):
    def get(self, request, *args, **kwargs):
        comment = get_object_or_404(
            Comment, id=kwargs["id"], article__id=kwargs["article_id"]
        )

        return render(...)

Затем происходит выборка статьи из базы данных. Для нее используется функция get_object_or_404, а не get.

Дело в том, что просмотр конкретной сущности должен вернуть ошибку 404, если самой сущности не существует. Метод get не помогает обработать эту ситуацию, если сущности нет, то он сгенерирует ошибку. Дальше программисту придется делать проверку на существование и самому формировать правильный HTTP-ответ.

Так как задача очень частая, то разработчики Django решили ее внутри фреймворка. С одной стороны, они добавили функцию get_object_or_404, которая выбрасывает исключение в случае отсутствия записи. С другой — добавили специальную обработку данных исключений на уровне обработки запросов.

И последний шаг — передача статьи в шаблон. Здесь ничего нового:

{% extends "base.html" %}

{% block content %}
     <h1>{{ article.name }}</h1>
    <div>{{ article.body }}</div>
{% endblock %}