django 的其他功能


认证

可以在配置文件中配置全局默认的认证方案

REST_FRAMEWORK = {

    'DEFAULT_AUTHENTICATION_CLASSES': (

        'rest_framework.authentication.BasicAuthentication', # 基本认证         'rest_framework.authentication.SessionAuthentication', # session认证

)

}

也可以在每个视图中通过设置authentication_classess属性来设置

fromrest_framework.authenticationimportSessionAuthentication, BasicAuthentication

from rest_framework.views impor tAPIView

class ExampleView(APIView):

    authentication_classes = (SessionAuthentication, BasicAuthentication) 

 ...

认证失败会有两种可能的返回值:

401 Unauthorized 未认证

403 Permission Denied 权限被禁止

权限Permissions

权限控制可以限制用户对于视图的访问和对于具体数据对象的访问。

在执行视图的dispatch()方法前,会先进行视图访问权限的判断

在通过get_object()获取具体对象时,会进行对象访问权限的判断

使用可以在配置文件中设置默认的权限管理类,如

REST_FRAMEWORK = {

    'DEFAULT_PERMISSION_CLASSES': (

        'rest_framework.permissions.IsAuthenticated', 

 )

}

如果未指明,则采用如下默认配置

'DEFAULT_PERMISSION_CLASSES': (

    'rest_framework.permissions.AllowAny',

)

也可以在具体的视图中通过permission_classes属性来设置,如  

from  rest_framework.permissions import IsAuthenticated

fromrest_framework.views  importAPIView

class ExampleView(APIView):

    permission_classes = (IsAuthenticated,)  

  ...  

提供的权限AllowAny 

允许所有用户IsAuthenticated 

仅通过认证的用户IsAdminUser 

仅管理员用户IsAuthenticated

OrReadOnly 认证的用户可以完全操作,否则只能get读取举例 

from rest_framework.authentication import SessionAuthentication

from rest_framework.permissions import IsAuthenticated

from rest_framework.generics import RetrieveAPIView

class BookDetailView(RetrieveAPIView):

    queryset = BookInfo.objects.all()

    serializer_class = BookInfoSerializer

    authentication_classes = [SessionAuthentication]

    permission_classes = [IsAuthenticated] 

自定义权限

如需自定义权限,需继承rest_framework.permissions.BasePermission父类,并实现以下两个任何一个方法或全部

.has_permission(self, request, view)

是否可以访问视图, view表示当前视图对象

.has_object_permission(self, request, view, obj)

是否可以访问数据对象, view表示当前视图, obj为数据对象

例如:

class MyPermission(BasePermission):

    def has_object_permission(self, request, view, obj):

         """控制对obj对象的访问权限,此案例决绝所有对对象的访问"""

        return False

class BookInfoViewSet(ModelViewSet):

    queryset = BookInfo.objects.all()

    serializer_class = BookInfoSerializer

     permission_classes = [IsAuthenticated, MyPermission] 

限流Throttling

可以对接口访问的频次进行限制,以减轻服务器压力。

使用

可以在配置文件中,使用DEFAULT_THROTTLE_CLASSES 和 DEFAULT_THROTTLE_RATES进行全局配置,

REST_FRAMEWORK = {

    'DEFAULT_THROTTLE_CLASSES': (

        'rest_framework.throttling.AnonRateThrottle',

        'rest_framework.throttling.UserRateThrottle'

        ),

    'DEFAULT_THROTTLE_RATES': {

        'anon':'100/day','user':'1000/day'

                                                               }

}

DEFAULT_THROTTLE_RATES 可以使用 second, minute, hour 或day来指明周期。

也可以在具体视图中通过throttle_classess属性来配置,如

from rest_framework.throttling import UserRateThrottle

from rest_framework.views import APIView

class ExampleView(APIView):

    throttle_classes = (UserRateThrottle,) 

     ...

可选限流类

1) AnonRateThrottle

限制所有匿名未认证用户,使用IP区分用户。

使用DEFAULT_THROTTLE_RATES['anon'] 来设置频次

2)UserRateThrottle

限制认证用户,使用User id 来区分。

使用DEFAULT_THROTTLE_RATES['user'] 来设置频次

3)ScopedRateThrottle

限制用户对于每个视图的访问频次,使用ip或user id。

例如:

class ContactListView(APIView):

     throttle_scope = 'contacts'

     ...

class ContactDetailView(APIView):

    throttle_scope = 'contacts'

    ...

class UploadView(APIView):

    throttle_scope = 'uploads'

    ...

REST_FRAMEWORK = {

    'DEFAULT_THROTTLE_CLASSES': (

        'rest_framework.throttling.ScopedRateThrottle',

    ),

    'DEFAULT_THROTTLE_RATES': {

        'contacts': '1000/day',

        'uploads': '20/day'

    }

}

实例

from rest_framework.authentication import SessionAuthentication

from rest_framework.permissions import IsAuthenticated

from rest_framework.generics import RetrieveAPIView

from rest_framework.throttling import UserRateThrottle

class BookDetailView(RetrieveAPIView):

    queryset = BookInfo.objects.all()

     serializer_class = BookInfoSerializer

     authentication_classes = [SessionAuthentication]

     permission_classes = [IsAuthenticated]

     throttle_classes = (UserRateThrottle,)

过滤Filtering

对于列表数据可能需要根据字段进行过滤,我们可以通过添加django-fitlter扩展来增强支持。

pip install django-filter

在配置文件中增加过滤后端的设置:

INSTALLED_APPS = [

 ...'django_filters',

# 需要注册应用,

]

REST_FRAMEWORK = {

'DEFAULT_FILTER_BACKENDS': ('django_filters.rest_framework.DjangoFilterBackend',

)

}


在视图中添加filter_fields属性,指定可以过滤的字段

class BookListView(ListAPIView):

    queryset = BookInfo.objects.all()

     serializer_class = BookInfoSerializer

     filter_fields = ('btitle','bread')

# 127.0.0.1:8000/books/?btitle=西游记

排序

对于列表数据,REST framework提供了OrderingFilter过滤器来帮助我们快速指明数据按照指定字段进行排序。

使用方法:

在类视图中设置filter_backends,使用rest_framework.filters.OrderingFilter过滤器,REST framework会在请求的查询字符串参数中检查是否包含了ordering参数,如果包含了ordering参数,则按照ordering参数指明的排序字段对数据集进行排序。

前端可以传递的ordering参数的可选字段值需要在ordering_fields中指明。

示例:

class BookListView(ListAPIView):

    queryset = BookInfo.objects.all()

     serializer_class = BookInfoSerializer

     filter_backends = [OrderingFilter]

     ordering_fields = ('id','bread','bpub_date')

# 127.0.0.1:8000/books/?ordering=-bread

分页Pagination

REST framework提供了分页的支持。

我们可以在配置文件中设置全局的分页方式,如:

REST_FRAMEWORK = {'DEFAULT_PAGINATION_CLASS':'rest_framework.pagination.PageNumberPagination',

'PAGE_SIZE':100# 每页数目}

也可通过自定义Pagination类,来为视图添加不同分页行为。在视图中通过pagination_clas属性来指明。

class LargeResultsSetPagination(PageNumberPagination):

    page_size =1000

    page_size_query_param ='page_size'

    max_page_size =10000


class BookDetailView(RetrieveAPIView):

    queryset = BookInfo.objects.all()

    serializer_class = BookInfoSerializer

    pagination_class = LargeResultsSetPagination

注意:如果在视图内关闭分页功能,只需在视图内设置

pagination_class =None


可选分页器

1) PageNumberPagination

前端访问网址形式:

GET http://api.example.org/books/?page=4

可以在子类中定义的属性:

page_size 每页数目

page_query_param 前端发送的页数关键字名,默认为"page"

page_size_query_param 前端发送的每页数目关键字名,默认为None

max_page_size 前端最多能设置的每页数量

from rest_framework.pagination import PageNumberPagination

class StandardPageNumberPagination(PageNumberPagination):

    page_size_query_param ='page_size'

    max_page_size =10

class BookListView(ListAPIView):

    queryset = BookInfo.objects.all().order_by('id')

     serializer_class = BookInfoSerializer

     pagination_class = StandardPageNumberPagination

# 127.0.0.1/books/?page=1&page_size=2


2)LimitOffsetPagination

前端访问网址形式:

GET http://api.example.org/books/?limit=100&offset=400

可以在子类中定义的属性:

default_limit 默认限制,默认值与PAGE_SIZE设置一直

limit_query_param limit参数名,默认'limit'

offset_query_param offset参数名,默认'offset'

max_limit 最大limit限制,默认None

from rest_framework.pagination impor tLimitOffsetPagination

classBookListView(ListAPIView):

    queryset = BookInfo.objects.all().order_by('id')

    serializer_class = BookInfoSerializer

    pagination_class = LimitOffsetPagination

# 127.0.0.1:8000/books/?offset=3&limit=2


异常处理 Exceptions

REST framework提供了异常处理,我们可以自定义异常处理函数。

from rest_framework.views import exception_handler

def custom_exception_handler(exc, context):

# 先调用REST framework默认的异常处理方法获得标准错误响应对象

    response = exception_handler(exc, context)# 在此处补充自定义的异常处理

    if response is not None:        

        response.data['status_code'] = response.status_code

    return response

在配置文件中声明自定义的异常处理

REST_FRAMEWORK = {

'EXCEPTION_HANDLER':'my_project.my_app.utils.custom_exception_handler'

}

如果未声明,会采用默认的方式,如下

REST_FRAMEWORK = {

'EXCEPTION_HANDLER':'rest_framework.views.exception_handler'

}

例如:

补充上处理关于数据库的异常

from rest_framework.views import exception_handlerasdrf_exception_handler

from rest_framework import status

from django.db import DatabaseError

defcustom_exception_handler(exc,context):

    #0,调用drf的异常信息,返回响应证明有异常

    response = exception_handler(exc,context)

    #1,判断是否是数据库的异常

    if isinstance(exc,DatabaseError): 

         response = Response("数据库出异常了",status=status.HTTP_507_INSUFFICIENT_STORAGE)

#2,返回响应

    return response


REST framework定义的异常

APIException 所有异常的父类

ParseError 解析错误

AuthenticationFailed 认证失败

NotAuthenticated 尚未认证

PermissionDenied 权限决绝

NotFound 未找到

MethodNotAllowed 请求方式不支持

NotAcceptable 要获取的数据格式不支持

Throttled 超过限流次数

ValidationError 校验失败

后续继续更新

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,847评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,208评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,587评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,942评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,332评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,587评论 1 218
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,853评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,568评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,273评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,542评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,033评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,373评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,031评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,073评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,830评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,628评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,537评论 2 269

推荐阅读更多精彩内容