婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > Django REST framework 異常處理

Django REST framework 異常處理

熱門標簽:外賣地址有什么地圖標注 銀川電話機器人電話 如何地圖標注公司 長春極信防封電銷卡批發 預覽式外呼系統 煙臺電話外呼營銷系統 企業彩鈴地圖標注 電銷機器人錄音要學習什么 上海正規的外呼系統最新報價

寫在前面

這兩天一直在思索關于 DRF 還有哪些是項目必備的而且還沒有說到的基礎性的知識。這不昨天寫到日志相關的功能就直接想到還有異常處理相關的功能,其實在之前項目中初期是沒有統一的異常捕獲手段。可能是 DRF 自帶的異常 能滿足大多數功能,也可能是比較懶,就使用比較粗暴的方式,以狀態碼 500 的方式去拋出異常,然后在日志中可以看到所有的異常信息。這么做呢,代碼其實是不夠健壯的,前端在調用的時候莫名的 500 也是不夠友好的,所以今天就補充一下異常相關的知識。

DRF異常處理

1. DRF 常見的異常

  • AuthenticationFailed/ NotAuthenticated 一般該異常狀態碼為"401 Unauthenticated",主要是沒有登錄鑒權的時候會返回,可以用在自定義登錄的時候。
  • PermissionDenied 一般用在鑒權時候使用,一般狀態碼為"403 Forbidden"。
  • ValidationError 一般狀態碼為"400 Bad Request",主要是 serializers 中對字段的校驗,比如對字段類型的校驗、字段長度的校驗以及自定義字段格式的校驗。

2. 自定義異常

這里對異常的定義主要的想法來自 ValidationError,統一異常返回的格式,方便前端統一處理類似異常。

自定義異常

# 新建 utils/custom_exception.py

class CustomException(Exception):
    _default_code = 400

    def __init__(
        self,
        message: str = "",
        status_code=status.HTTP_400_BAD_REQUEST,
        data=None,
        code: int = _default_code,
    ):

        self.code = code
        self.status = status_code
        self.message = message
        if data is None:
            self.data = {"detail": message}
        else:
            self.data = data

    def __str__(self):
        return self.message

自定義異常處理

# utils/custom_exception.py
from rest_framework.views import exception_handler

def custom_exception_handler(exc, context):
    # Call REST framework's default exception handler first,
    # to get the standard error response.
    
    # 這里對自定義的 CustomException 直接返回,保證系統其他異常不受影響
    if isinstance(exc, CustomException):
        return Response(data=exc.data, status=exc.status)
    response = exception_handler(exc, context)
    return response

配置自定義異常處理類

REST_FRAMEWORK = {
    # ...
    "EXCEPTION_HANDLER": "utils.custom_exception.custom_exception_handler",
}

3. 使用自定義異常

使用之前文章的接口用來測試自定義異常的處理

class ArticleViewSet(viewsets.ModelViewSet):
    """
    允許用戶查看或編輯的API路徑。
    """
    queryset = Article.objects.all()
    serializer_class = ArticleSerializer

    @action(detail=False, methods=["get"], url_name="exception", url_path="exception")
    def exception(self, request, *args, **kwargs):
        # 日志使用 demo
        logger.error("自定義異常")
        raise CustomException(data={"detail": "自定義異常"})

4. 驗證結果

$ curl -H 'Accept: application/json; indent=4' -u admin:admin http://127.0.0.1:8000/api/article/exception/
{
    "detail": "自定義異常"
}

異常處理進階

上面的代碼雖說是可以滿足90%的需求,但是錯誤的定義太泛泛。難以集中定義管理錯誤,與常見項目中自定義的異常比較優點就是靈活,但是隨著代碼中拋出的異常越來越多加之散落在各個角落,不利于更新維護。所以下面對修改一下代碼,對異常有統一的定義,同時也支持自定義返回HTTP狀態碼。

1. 修改自定義異常

# utils/custom_exception.py

class CustomException(Exception):
    # 自定義code
    default_code = 400
    # 自定義 message
    default_message = None

    def __init__(
            self,
            status_code=status.HTTP_400_BAD_REQUEST,
            code: int = None,
            message: str = None,
            data=None,
    ):
        self.status = status_code
        self.code = self.default_code if code is None else code
        self.message = self.default_message if message is None else message

        if data is None:
            self.data = {"detail": self.message, "code": self.code}
        else:
            self.data = data

    def __str__(self):
        return str(self.code) + self.message

2. 自定義更多異常

class ExecuteError(CustomException):
    """執行出錯"""
    default_code = 500
    default_message = "執行出錯"


class UnKnowError(CustomException):
    """執行出錯"""
    default_code = 500
    default_message = "未知出錯"

3. 新增測試接口

class ArticleViewSet(viewsets.ModelViewSet):
    """
    允許用戶查看或編輯的API路徑。
    """
    queryset = Article.objects.all()
    serializer_class = ArticleSerializer

    @action(detail=False, methods=["get"], url_name="exception", url_path="exception")
    def exception(self, request, *args, **kwargs):
        # 日志使用 demo
        logger.error("自定義異常")
        raise CustomException(data={"detail": "自定義異常"})

    @action(detail=False, methods=["get"], url_name="unknown", url_path="unknown")
    def unknown(self, request, *args, **kwargs):
        # 日志使用 demo
        logger.error("未知錯誤")
        raise UnknownError()

    @action(detail=False, methods=["get"], url_name="execute", url_path="execute")
    def execute(self, request, *args, **kwargs):
        # 日志使用 demo
        logger.error("執行錯誤")
        raise ExecuteError()

4. 驗證結果

curl -H 'Accept: application/json; indent=4' -u admin:admin http://127.0.0.1:8000/api/article/unknown/
{
    "detail": "未知出錯",
    "code": 500
}
$ curl -H 'Accept: application/json; indent=4' -u admin:admin http://127.0.0.1:8000/api/article/execute/
{
    "detail": "執行出錯",
    "code": 500
}

總結

需要注意自定義的異常處理函數需要在處理完成自定義異常后繼續執行 rest_framework.views.exception_handler,因為這里的執行仍然需要兼容已有的異常處理;下面貼一下 DRF 有關的異常處理邏輯。

該處理函數默認處理 APIException以及 Django 內部的 Http404 PermissionDenied,其他的異常會返回 None ,會觸發 DRF 500 的錯誤。

def exception_handler(exc, context):
    """
    Returns the response that should be used for any given exception.

    By default we handle the REST framework `APIException`, and also
    Django's built-in `Http404` and `PermissionDenied` exceptions.

    Any unhandled exceptions may return `None`, which will cause a 500 error
    to be raised.
    """
    if isinstance(exc, Http404):
        exc = exceptions.NotFound()
    elif isinstance(exc, PermissionDenied):
        exc = exceptions.PermissionDenied()

    if isinstance(exc, exceptions.APIException):
        headers = {}
        if getattr(exc, 'auth_header', None):
            headers['WWW-Authenticate'] = exc.auth_header
        if getattr(exc, 'wait', None):
            headers['Retry-After'] = '%d' % exc.wait

        if isinstance(exc.detail, (list, dict)):
            data = exc.detail
        else:
            data = {'detail': exc.detail}

        set_rollback()
        return Response(data, status=exc.status_code, headers=headers)

    return None

參考資料

Django REST framework 異常文檔
Django 異常文檔

到此這篇關于Django REST framework 異常處理的文章就介紹到這了,更多相關Django REST framework 異常內容請搜索腳本之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:
  • 詳解Django rest_framework實現RESTful API
  • django rest framework 數據的查找、過濾、排序的示例
  • django-rest-framework解析請求參數過程詳解
  • Django Rest framework權限的詳細用法
  • Django rest framework基本介紹與代碼示例
  • Django Rest framework認證組件詳細用法
  • 淺談Django REST Framework限速

標簽:珠海 佳木斯 盤錦 湖北 潮州 上饒 西寧 宜昌

巨人網絡通訊聲明:本文標題《Django REST framework 異常處理》,本文關鍵詞  Django,REST,framework,異常,處理,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Django REST framework 異常處理》相關的同類信息!
  • 本頁收集關于Django REST framework 異常處理的相關信息資訊供網民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    亚洲精品国产一区二区三区四区在线| 五月综合激情网| 在线成人小视频| 欧美mv日韩mv亚洲| 午夜精品福利一区二区蜜股av| 成人网页在线观看| 一区二区三区加勒比av| 欧美精品xxxxbbbb| 成人激情综合网站| 秋霞午夜av一区二区三区| 欧美不卡一二三| 高清不卡一二三区| 一区二区在线观看免费视频播放| 国产一区二区三区久久久 | 精品国产乱码久久久久久老虎| 久久这里只精品最新地址| 成人国产亚洲欧美成人综合网 | 欧美一区二区在线不卡| 一区二区三区不卡在线观看| 色婷婷综合久色| 麻豆视频观看网址久久| 亚洲视频一区二区在线观看| 99精品国产视频| 国产一区二三区好的| 蜜臀久久99精品久久久久宅男| 精品国产乱码久久久久久夜甘婷婷| 国产一区二区调教| 精品国产凹凸成av人导航| 日韩亚洲欧美一区二区三区| 99久久精品久久久久久清纯| 91福利社在线观看| 在线观看日韩一区| 亚洲成人午夜影院| 欧美一级夜夜爽| 久久99精品国产麻豆婷婷| 久久男人中文字幕资源站| 成人影视亚洲图片在线| 一区二区三区资源| 91精品在线一区二区| 国产乱人伦精品一区二区在线观看| 日本一区二区视频在线观看| 91麻豆swag| 蜜芽一区二区三区| 国产精品伦理一区二区| 欧美无砖专区一中文字| 国产一区欧美二区| 一区二区三区四区不卡在线| 日韩视频免费观看高清完整版在线观看 | 亚洲大片免费看| 精品美女被调教视频大全网站| 成人中文字幕电影| 一区二区三区成人| 久久天天做天天爱综合色| 色久优优欧美色久优优| 久久不见久久见中文字幕免费| 国产精品国产精品国产专区不片| 在线不卡免费欧美| 91丨九色丨蝌蚪富婆spa| 久久精品国内一区二区三区| 亚洲另类春色国产| 久久精品日产第一区二区三区高清版| 欧美性猛交一区二区三区精品| 国产麻豆成人传媒免费观看| 一区二区三区国产| 亚洲国产精品精华液2区45| 884aa四虎影成人精品一区| voyeur盗摄精品| 韩国毛片一区二区三区| 五月开心婷婷久久| 日韩理论在线观看| 久久精品人人做| 欧美一区二区视频在线观看2020| 99久久免费视频.com| 国产乱一区二区| 免费高清不卡av| 亚洲国产精品影院| 国产精品久久久久aaaa| 久久久www成人免费毛片麻豆| 91精品国产91久久久久久最新毛片| 色婷婷国产精品综合在线观看| 丁香六月综合激情| 国产精品1024久久| 狠狠色丁香久久婷婷综| 日韩国产成人精品| 午夜免费久久看| 亚洲国产精品一区二区www| 亚洲欧美一区二区三区国产精品| 欧美激情一区二区三区蜜桃视频| 久久精品人人做| 国产色产综合色产在线视频| 精品国偷自产国产一区| 欧美一级日韩一级| 91精品欧美久久久久久动漫| 欧美精品乱人伦久久久久久| 欧美性一二三区| 欧美日韩视频专区在线播放| 欧美系列一区二区| 欧美亚洲国产bt| 欧美视频一区二| 91污在线观看| 一本一道综合狠狠老| 色狠狠色噜噜噜综合网| 在线亚洲人成电影网站色www| 一本大道久久精品懂色aⅴ| av午夜一区麻豆| 国产99久久久国产精品免费看 | 91精品国产91久久久久久一区二区 | 亚洲久本草在线中文字幕| 国产精品乱码一区二区三区软件 | 最新国产成人在线观看| 中文字幕一区在线观看| 亚洲伦在线观看| 亚洲综合色在线| 全国精品久久少妇| 毛片av一区二区三区| 韩国v欧美v亚洲v日本v| 成人黄色电影在线| 色香蕉久久蜜桃| 91精品久久久久久久91蜜桃| 日韩精品一区国产麻豆| 久久综合狠狠综合久久激情| 国产日产亚洲精品系列| 最新国产成人在线观看| 无码av免费一区二区三区试看 | 视频一区二区三区入口| 老司机精品视频线观看86| 久久国产夜色精品鲁鲁99| 国产成人午夜99999| 日本高清不卡在线观看| 91精品福利在线一区二区三区 | 99热精品国产| 欧美日韩国产在线观看| 精品区一区二区| 国产精品久久久久久久第一福利 | 久久综合狠狠综合久久激情 | 狂野欧美性猛交blacked| 国产伦精品一区二区三区免费| 成人av片在线观看| 91传媒视频在线播放| 欧美成人一区二区三区片免费| 国产精品三级av| 丝袜美腿亚洲色图| 国产精品一区二区无线| 欧美三片在线视频观看| 国产日韩欧美精品在线| 亚洲一区二区五区| 国产 日韩 欧美大片| 在线综合视频播放| 中文字幕中文在线不卡住| 日韩精品成人一区二区在线| fc2成人免费人成在线观看播放| 亚洲一线二线三线视频| 亚洲一区二区三区中文字幕 | 色婷婷久久久综合中文字幕 | 久久综合久久综合久久| 久久久国产精品麻豆| 亚洲激情校园春色| 丝袜亚洲另类欧美综合| 丁香婷婷综合网| 7777精品伊人久久久大香线蕉最新版| 日韩欧美不卡一区| 日韩国产欧美在线播放| 一道本成人在线| 中文字幕亚洲欧美在线不卡| 奇米色一区二区| 欧美日韩1234| 日本亚洲天堂网| 欧美一区二视频| 日本午夜一本久久久综合| 99久久久免费精品国产一区二区 | 日韩欧美高清在线| 视频精品一区二区| 欧美三级电影网站| 亚洲电影你懂得| 欧美在线观看视频一区二区三区| 日韩欧美一区二区久久婷婷| 精品国产露脸精彩对白| 欧美不卡一区二区三区四区| 亚洲第一搞黄网站| 欧美系列日韩一区| 亚洲精品乱码久久久久久久久| 国产精品久久毛片av大全日韩| 综合久久久久久久| 秋霞电影一区二区| 播五月开心婷婷综合| 日本一区二区三区久久久久久久久不 | 欧美日韩在线亚洲一区蜜芽| 国产精品国产三级国产普通话三级| 欧美精品高清视频| 午夜精品福利一区二区蜜股av | 国产精品2024| wwwwxxxxx欧美| 色婷婷久久99综合精品jk白丝| 日本一区二区不卡视频| 国产99久久久国产精品| 一区二区三区.www| 欧美高清视频www夜色资源网| 日日骚欧美日韩| 国产精品国产三级国产三级人妇| 欧美福利一区二区|