Исключения
Исключения
Исключения... позволяют чисто организовать обработку ошибок в центральном или высокоуровневом месте в структуре программы.
— Даг Хеллманн, Python Exception Handling Techniques
Обработка исключений в представлениях DRF
Представления DRF обрабатывают различные исключения и возвращают соответствующие ответы на ошибки.
Обрабатываемыми исключениями являются:
Подклассы
APIException
, возникающие внутри DRF.Исключение Django
Http404
.Исключение Django
PermissionDenied
.
В каждом случае DRF вернет ответ с соответствующим кодом состояния и типом содержимого. В теле ответа будут содержаться любые дополнительные сведения о характере ошибки.
Большинство ответов на ошибки будут содержать ключ detail
в теле ответа.
Например, следующий запрос:
Может быть получен ответ об ошибке, указывающий на то, что метод DELETE
не разрешен для данного ресурса:
Ошибки валидации обрабатываются несколько иначе, и в качестве ключей в ответе будут указаны имена полей. Если ошибка валидации не относится к конкретному полю, то будет использоваться ключ "non_field_errors", или любое строковое значение, установленное для параметра NON_FIELD_ERRORS_KEY
.
Пример ошибки валидации может выглядеть следующим образом:
Пользовательская обработка исключений
Вы можете реализовать пользовательскую обработку исключений, создав функцию-обработчик, которая преобразует исключения, возникающие в ваших представлениях API, в объекты ответа. Это позволяет вам контролировать стиль ответов на ошибки, используемый вашим API.
Функция должна принимать пару аргументов, первый из которых - обрабатываемое исключение, а второй - словарь, содержащий любой дополнительный контекст, например, обрабатываемое в данный момент представление. Функция обработчика исключения должна либо возвращать объект Response
, либо возвращать None
, если исключение не может быть обработано. Если обработчик возвращает None
, то исключение будет повторно поднято, и Django вернет стандартный ответ HTTP 500 "Server error"
.
Например, вы можете захотеть убедиться, что все ответы на ошибки включают код состояния HTTP в теле ответа, например, так:
Чтобы изменить стиль ответа, вы можете написать следующий пользовательский обработчик исключений:
Аргумент context
не используется обработчиком по умолчанию, но может быть полезен, если обработчику исключений нужна дополнительная информация, например, обрабатываемое в данный момент представление, доступ к которому можно получить как context['view']
.
Обработчик исключений также должен быть настроен в ваших настройках, используя ключ настройки EXCEPTION_HANDLER
. Например:
Если параметр 'EXCEPTION_HANDLER'
не указан, по умолчанию используется стандартный обработчик исключений, предоставляемый DRF:
Обратите внимание, что обработчик исключений будет вызываться только для ответов, сгенерированных поднятыми исключениями. Он не будет использоваться для ответов, возвращаемых непосредственно представлением, таких как ответы HTTP_400_BAD_REQUEST
, которые возвращаются общими представлениями при неудачной проверке сериализатора.
API Reference
APIException
Сигнатура: APIException()
.
базовый класс для всех исключений, возникающих внутри класса APIView
или @api_view
.
Чтобы предоставить пользовательское исключение, подкласс APIException
и установите атрибуты .status_code
, .default_detail
и .default_code
для класса.
Например, если ваш API полагается на сторонний сервис, который иногда может быть недоступен, вы можете захотеть реализовать исключение для кода ответа HTTP "503 Service Unavailable". Это можно сделать следующим образом:
Проверка исключений API
Существует ряд различных свойств, доступных для проверки состояния исключения API. Вы можете использовать их для создания пользовательской обработки исключений для вашего проекта.
Доступными атрибутами и методами являются:
.detail
- Возвращает текстовое описание ошибки..get_codes()
- Возвращает идентификатор кода ошибки..get_full_details()
- Возвращает как текстовое описание, так и идентификатор кода.
В большинстве случаев деталь ошибки будет простым элементом:
В случае ошибок валидации деталь ошибки будет представлять собой либо список, либо словарь элементов:
ParseError
Сигнатура: ParseError(detail=None, code=None)
.
Возникает, если запрос содержит неправильно сформированные данные при доступе к request.data
.
По умолчанию это исключение приводит к ответу с кодом состояния HTTP "400 Bad Request".
AuthenticationFailed
Сигнатура: AuthenticationFailed(detail=None, code=None)
.
Возникает, когда входящий запрос содержит неправильную аутентификацию.
По умолчанию это исключение приводит к ответу с кодом состояния HTTP "401 Unauthenticated", но оно также может привести к ответу "403 Forbidden", в зависимости от используемой схемы аутентификации. Более подробную информацию см. в документации authentication documentation.
NotAuthenticated
Сигнатура: NotAuthenticated(detail=None, code=None)
.
Возникает, когда неаутентифицированный запрос не прошел проверку на разрешение.
По умолчанию это исключение приводит к ответу с кодом состояния HTTP "401 Unauthenticated", но оно также может привести к ответу "403 Forbidden", в зависимости от используемой схемы аутентификации. Более подробную информацию см. в документации authentication documentation.
PermissionDenied
Сигнатура: PermissionDenied(detail=None, code=None)
.
Возникает, когда аутентифицированный запрос не прошел проверку на разрешение.
По умолчанию это исключение приводит к ответу с кодом состояния HTTP "403 Forbidden".
NotFound
Сигнатура: NotFound(detail=None, code=None)
.
Возникает, когда ресурс не существует по указанному URL. Это исключение эквивалентно стандартному исключению Http404
Django.
По умолчанию это исключение приводит к ответу с кодом состояния HTTP "404 Not Found".
MethodNotAllowed
Сигнатура: MethodNotAllowed(method, detail=None, code=None)
.
Возникает, когда происходит входящий запрос, который не сопоставлен с методом-обработчиком на представлении.
По умолчанию это исключение приводит к ответу с кодом состояния HTTP "405 Method Not Allowed".
Неприемлемо
Сигнатура: NotAcceptable(detail=None, code=None)
.
Возникает, когда поступает запрос с заголовком Accept
, который не может быть удовлетворен ни одним из доступных рендереров.
По умолчанию это исключение приводит к ответу с кодом состояния HTTP "406 Not Acceptable".
UnsupportedMediaType
Сигнатура: UnsupportedMediaType(media_type, detail=None, code=None)
.
Возникает, если при обращении к request.data
нет парсеров, способных обработать тип содержимого данных запроса.
По умолчанию это исключение приводит к ответу с кодом состояния HTTP "415 Unsupported Media Type".
Дроссель
Сигнатура: Throttled(wait=None, detail=None, code=None)
.
Возникает, когда входящий запрос не проходит проверку на дросселирование.
По умолчанию это исключение приводит к ответу с кодом состояния HTTP "429 Too Many Requests".
ValidationError
Сигнатура: ValidationError(detail=None, code=None)
.
Исключение ValidationError
немного отличается от других классов APIException
:
Аргумент
detail
может быть списком или словарем деталей ошибки, а также может быть вложенной структурой данных. Используя словарь, вы можете указать ошибки на уровне полей при выполнении проверки на уровне объектов в методеvalidate()
сериализатора. Например.raise serializers.ValidationError({'name': 'Please enter a valid name.'})
.По соглашению вы должны импортировать модуль
serializers
и использовать полностью определенныйValidationError
, чтобы отличить его от встроенной ошибки валидации Django. Например.raise serializers.ValidationError('Это поле должно быть целочисленным значением.')
.
Класс ValidationError
должен использоваться для сериализатора и валидации полей, а также классами валидаторов. Он также вызывается при вызове serializer.is_valid
с именованным аргументом raise_exception
:
Общие представления используют флаг raise_exception=True
, что означает, что вы можете переопределить стиль ответов на ошибки валидации глобально в вашем API. Для этого используйте пользовательский обработчик исключений, как описано выше.
По умолчанию это исключение приводит к ответу с кодом состояния HTTP "400 Bad Request".
Общие представления об ошибках
DRF предоставляет два представления ошибок, подходящих для предоставления общих JSON ответов 500
Server Error и 400
Bad Request. (Стандартные представления ошибок Django предоставляют HTML-ответы, которые могут не подойти для приложения, использующего только API).
Используйте их согласно Django's Customizing error views documentation.
rest_framework.exceptions.server_error
rest_framework.exceptions.server_error
Возвращает ответ с кодом состояния 500
и типом содержимого application/json
.
Устанавливается как handler500
:
rest_framework.exceptions.bad_request
rest_framework.exceptions.bad_request
Возвращает ответ с кодом статуса 400
и типом содержимого application/json
.
Устанавливается как handler400
:
Пакеты сторонних производителей
Также доступны следующие пакеты сторонних производителей.
Стандартизированные ошибки ДРФ
Пакет drf-standardized-errors предоставляет обработчик исключений, который генерирует одинаковый формат для всех ответов 4xx и 5xx. Он является заменой стандартного обработчика исключений и позволяет настраивать формат ответа на ошибку без переписывания всего обработчика исключений. Стандартизированный формат ответа на ошибку легче документировать и проще обрабатывать потребителям API.
Last updated