Django-restframework 之认证源码分析

Django-restframework 源码分析之认证

前言

最近学习了 django 的一个 restframework 框架,对于里面的执行流程产生了兴趣,经过昨天一晚上初步搞清楚了执行流程(部分方法还不太清楚),于是想详细的总结一下当来一个请求时,在该框架里面是如何执行的?

启动项目时

昨天在调试django时,发现在 APIView 中打的断点没有断下来,而是打在 View 中的断点断下来了,调试了很多次,最后发现,在 django 项目启动时,会首先加载 urls 中的文件,执行 views 中类的 as_view方法,其实是继承自 APIView 的,APIView 继承自 django 原生 View 的as_view 方法。

里面一个参数叫 pattern_name,对应的值是admin:auth_group_change,如下图所示:

006tKfTcly1g0dqrtq01oj310m0cadjc.jpg

目前还不清楚这里面的具体流程是什么,但是并不妨碍阅读之后的源码,在这只要清楚一点,在 Django 项目启动时,路由所对应的CBV里面的相关方法的内存地址已经获取到。这样做的好处就是提高效率,坏处可能有一点点,会提升性能的消耗。

具体路由和逻辑代码

在这里假设来一个 GET 请求,urls 和 views里面的代码如下:

# urls.py

url(r'^book/(?P<id>\d+)/', views.Book.as_view()),
# views.py
class Book(APIView):

    def dispatch(self, request, *args, **kwargs):
        return super().dispatch(request, *args, **kwargs)

    def get(self, request, id):

        response = {'status': 100, 'msg': None}

        book = models.Book.objects.filter(pk=id).first()
        book_ser = BookSerib(book, many=False)
        print('book_ser.data', book_ser.data)
        response['books'] = book_ser.data
        response['msg'] = '获取图书成功'
        print('response', response)
        return Response(response)

urls 里面就是一个典型的 CBV 的路由配置,在 views 中一个是路由分发方法,一个是获取单本图书信息(通过 id)。

as_view 具体执行流程

在项目启动时,相应的函数内存地址已经获取到,那么具体是怎么获取到的呢?

在上面的代码中可以看到 Book类是继承自APIView类的,所以在路由配置里面执行的as_view方法如果Book类没有重写,那么执行的就是按照mro列表顺序查找到的第一个方法,在这里执行的是APIView类中的as_view方法。

查看该源码如下:

006tKfTcly1g0dr5k56eqj31kg0mwdop.jpg

APIView类的父类是View类,查看该类的as_view方法,源码如下:

006tKfTcly1g0dtkje60nj31b00u0k29.jpg

具体as_view就是将view函数的内存地址返回,以便请求来时直接调用。

initkwargs这个参数应该是由 django 内部传的一些参数,如下所示:

006tKfTcly1g0ds6hb8o9j311w0bmn1o.jpg

请求到来时

因为经过 django 先执行了as_view返回view的内存地址,所以会直接执行view函数,如下:

006tKfTcly1g0dttrs9t9j31m8082dls.jpg

在这里该 self 由于是 继承自APIView类的Book类的对象,所以dispatch方法首先去自身找,之后去父类找,在APIView类中找到dispatch方法,源码如下:

006tKfTcly1g0du0o9jnej31mg0tewsm.jpg

initialize_request方法:

源码如下:

006tKfTcly1g0du2ggqlaj31gy0ein1a.jpg

在实例化Request对象时,authenticators参数需要在认证时使用,所以先把这个参数的值找出来。步骤如下:

  • 调用 self.get_authenticators

006tKfTcly1g0du4kpwcij31la05iwh3.jpg

  • 找到 self.authentication_classes

006tKfTcly1g0du79kijtj31gc0fmjyl.jpg

  • 找到 api_settings,才能找到DEFAULT_AUTHENTICATION_CLASSES

006tKfTcly1g0du8ss6x4j31iw05atad.jpg

  • DEFAULT和IMPORT_STRINGS参数
  1. DEFAULT参数

006tKfTcly1g0duhenrzgj31hy0rwakr.jpg

  1. IMPORT_STRINGS参数

006tKfTcly1g0dulz5m9tj31fs0j2qa6.jpg

  • 实例化 APISettings 对象

006tKfTcly1g0dv587iu9j31e40pugul.jpg

  • 通过__getattr__方法获取值

006tKfTcly1g0dve7p0ntj31ko0nsqcj.jpg

  • 执行perform_import方法

006tKfTcly1g0dvpmnf7ij31lm0c879s.jpg

  • 执行import_from_string

006tKfTcly1g0dvul5mptj31jy0ceahi.jpg

  • 执行import_module

006tKfTcly1g0dvx80nw0j31fa0iigra.jpg

  • 执行_bootstrap._gcd_import

006tKfTcly1g0dw0ns9znj31h00cy43l.jpg

  • 执行_sanity_check

006tKfTcly1g0dw3ij3ivj31fm0e043v.jpg

  • _sanity_check返回,执行_find_and_load方法

006tKfTcly1g0dw6pw2ghj31f60dq79n.jpg

  • 所以通过__getattr__的到的是一个列表,里面是两个类,分别是[rest_framework_authentication_SessionAuthentication, rest_framework_authentication_BasicAuthentication]

到这里,initialize_request方法执行结束,得到一个经过 Request 类实例化的 request 对象,该对象里面含有原生 request 对象,可以通过request._request取到。

initial 方法:

源码如下:

006tKfTcly1g0dwg9bl40j31f00i00zn.jpg

这里传入的request参数是经过Request封装后的参数

  • 执行get_format_suffix

006tKfTcly1g0dxa7s5azj31bs06etbd.jpg

执行完这个方法,self_format_kwarg = None

  • 执行perform_content_negotiation

006tKfTcly1g0dxfetavbj31lq0ckq70.jpg

  1. 执行self.get_renders

006tKfTcly1g0dxq4mzrzj31ha05s0vq.jpg

  1. 执行self.get_content_negotiator

006tKfTcly1g0e02qiaoxj31hi07y0vv.jpg

  1. 执行 conneg.select_renderer方法

006tKfTcly1g0e0a0yqxyj315h0u0qhz.jpg

未完待续…...

转载于:https://www.cnblogs.com/zuanzuan/p/10412165.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值