今天看了drf的五个组件的源码,可读性仍是很高的,只是读组件的时候要注意的是 大部分的组件都是由dispatch分发出去的,因此看源码的时候必定要抓住dispatch这条主线,一步一步看下去前端
drf的request是在wsgi的request的基础上进行再次封装python
**wsgi的request做为drf的request一个属性:_request(下面附源码解释)**django
#源码: #在rest-framework 的views.py文件中 def dispatch(self, request, *args, **kwargs): """ `.dispatch()` is pretty much the same as Django's regular dispatch, but with extra hooks for startup, finalize, and exception handling. """ self.args = args self.kwargs = kwargs # 在下面这行代码中 django原来的request传入 self.initialize_request 这个方法 request = self.initialize_request(request, *args, **kwargs) #self.initialize_request方法源码分析 def initialize_request(self, request, *args, **kwargs): """ Returns the initial request object. """ parser_context = self.get_parser_context(request) # 在下面这个函数的返回结果能够看出来 传入的原生django(wsgi)的request当作初始化参数传入Request这个类中 返回的就是Request这个类的实例化对象 return Request( request, parsers=self.get_parsers(), authenticators=self.get_authenticators(), negotiator=self.get_content_negotiator(), parser_context=parser_context ) # 因为Request是实例化 因此走的是 __init__方法 下面就是Request的__init__方法代码 def __init__(self, request, parsers=None, authenticators=None, negotiator=None, parser_context=None): assert isinstance(request, HttpRequest), ( 'The `request` argument must be an instance of ' '`django.http.HttpRequest`, not `{}.{}`.' .format(request.__class__.__module__, request.__class__.__name__) ) self._request = request #只看这一行 咱们就能够得出结果 Request实例传进来的原生的django(wsgi)的request在 Request的__init__方法中 从新赋值给self._request 而这个self就是咱们在rest-framework中的request
#源码 #在rest-framework的request.py中 def __getattr__(self, attr): """ If an attribute does not exist on this instance, then we also attempt to proxy it to the underlying HttpRequest object. """ try: return getattr(self._request, attr) # 经过反射获取属性,若是self._request存在,则使用self._request except AttributeError: # 不存在 则使用 rest-framework的request return self.__getattribute__(attr)
新的request对数据解析更规范化 :全部的拼接参数都解析到 requery_params中,全部数据包数据都被解析到了data中, query_params和 data属于QueryDict类型,可使用.dict()转化为原生的dict类型json
drf的APIView类:重写了as_view(),但主题逻辑仍是调用父类View的as_view()方法 ,最大的改动就是局部禁用了csrf认证(下面附源码解释)api
def as_view(cls, **initkwargs): #目录:rest-framework/views.py if isinstance(getattr(cls, 'queryset', None), models.query.QuerySet): def force_evaluation(): raise RuntimeError( 'Do not evaluate the `.queryset` attribute directly, ' 'as the result will be cached and reused between requests. ' 'Use `.all()` or call `.get_queryset()` instead.' ) cls.queryset._fetch_all = force_evaluation view = super().as_view(**initkwargs) # 调用父类View的as_view()方法 view.cls = cls view.initkwargs = initkwargs return csrf_exempt(view) # 局部禁用csrf
能够在视图;类中经过rendere_classes类类属性对该视图的数据响应渲染作配置 --局部配置服务器
能够在项目的配资文件的drf配置中经过DEFAULT_RENDERER_CLASSES对该视图的数据响应渲染作配置 --全局配置(下面附渲染模块源码分析)网络
注:若是在一个视图类在有全局配置下,还进行了局部配置,优先走本身的局部配置app
局部(views中) :renderer_classes = [JSONRenderer, BrowsableAPIRenderer]函数
全局(项目配置文件中) : DEFAULT_RENDERER_CLASSES = [JSONRenderer, BrowsableAPIRenderer]源码分析
""" 渲染模块源码分析 一、二次处理响应对象:APIView的dispatch方法 - self.finalize_response(request, response, *args, **kwargs) 二、获取渲染类对象:进入finalize_response方法 - self.perform_content_negotiation(request, force=True) 三、从配置文件中获得渲染类对象:perform_content_negotiation -> self.get_renderers() -> [renderer() for renderer in self.renderer_classes] """ """ 核心:能够全局和局部配置视图类支持的结果渲染:默承认以json和页面渲染,学习该模块的目的是开发能够全局只配置json方式渲染 """ #详细分析 #位置 rest-framework/views.py中的dispatch函数中 507行 # 下面这行代码利用finalize_response这个函数 对响应对象进行二次处理(规定返回数据的样式) self.response = self.finalize_response(request, response, *args, **kwargs) # 再来看看 finalize_response 这个函数作了什么 # 位置:rest-framework/views.py/finalize_response 414行 def finalize_response(self, request, response, *args, **kwargs): """ Returns the final response object. """ # Make the error obvious if a proper response is not returned # 下面这行代码的做用是:判断返回的是不是HttpResponseBase的对象,不是则报错 assert isinstance(response, HttpResponseBase), ( 'Expected a `Response`, `HttpResponse` or `HttpStreamingResponse` ' 'to be returned from the view, but received a `%s`' % type(response) ) if isinstance(response, Response): #先经过反射到request中寻找accepted_renderer是否存在 若是不存在,就用本身默认的 if not getattr(request, 'accepted_renderer', None): # 利用perform_content_negotiation函数 找到全局或者默认的渲染规则(项目中没有在 settings中设置全局的渲染规则的话 就用rest-framework默认的渲染规则) neg = self.perform_content_negotiation(request, force=True) request.accepted_renderer, request.accepted_media_type = neg response.accepted_renderer = request.accepted_renderer response.accepted_media_type = request.accepted_media_type response.renderer_context = self.get_renderer_context() #perform_content_negotiation 函数 # 位置 rest-framework/view.py/perform_content_negotiation 302行 def perform_content_negotiation(self, request, force=False): renderers = self.get_renderers() #经过get_renderers方法获取配置 # 位置 rest-framework/view.py/get_renderers 256行 def get_renderers(self): # 又经过renderer_classes 获取配置 经过列表推导式实例化返回 return [renderer() for renderer in self.renderer_classes] # 位置 rest-framework/view.py 107行 # 获取全局DEFAULT_RENDERER_CLASSES配置 renderer_classes = api_settings.DEFAULT_RENDERER_CLASSES
做用:服务与数据包数据
能够在视图类中经过parser_classes类属性对该视图的数据包解析作配置 --局部配置
可在项目的配置文件的drf配置中经过DEFAULT_PARSER_CLASSES对该视图的数据响应渲染作配置 --全局配置
解析模块源码分析
APIVIEW的dispatch方法:self.initialize_request(request, *args, **kwargs)内部还提供了数据解析
self.get_parser_context(request)提供要解析的数据,self.get_parsers() 提供解析的类对象(内部从配置中找到解析类)
在settings的drf配置中配置EXCEPTION_HANDLER,指向之定义的exception_handle函数
drf出现异常了,都会回调exception_handle函数,携带异常对象和异常相关信息内容,
在exception_handle函数完成异常信息的返回以及异常信息的logging日志
源码分析
在APIView的dispatch方法中,有一个超大的try...except..., 将代码运行异常都交给异常处理self.handle_exception(exc)
try: self.initial(request, *args, **kwargs) # Get the appropriate handler method if request.method.lower() in self.http_method_names: handler = getattr(self, request.method.lower(), self.http_method_not_allowed) else: handler = self.http_method_not_allowed response = handler(request, *args, **kwargs) except Exception as exc: #捕捉错误信息 将错误信息交给handle_exception response = self.handle_exception(exc)
从配置中映射出配置处理异常的函数(自定义异常模块就是自定义配置指向本身的函数): self.get_exception_handle()
def handle_exception(self, exc): """ Handle any exception that occurs, by returning an appropriate response, or re-raising the error. """ if isinstance(exc, (exceptions.NotAuthenticated, exceptions.AuthenticationFailed)): # WWW-Authenticate header for 401 responses, else coerce to 403 auth_header = self.get_authenticate_header(self.request) if auth_header: exc.auth_header = auth_header else: exc.status_code = status.HTTP_403_FORBIDDEN exception_handler = self.get_exception_handler() #获取错误处理函数 自定义了就用你自定义的 没有自定义就用系统的 context = self.get_exception_handler_context() response = exception_handler(exc, context) if response is None: self.raise_uncaught_exception(exc) response.exception = True return response # drf默认的错误处理函数 def exception_handler(exc, context): 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) # 处理不了的就返回none return None
异常函数exception_handle(exc,context)处理异常 就会走本身的先交给系统处理(客户端的异常),系统没处理(服务器异常),再本身处理
# 自定义异常处理文件能够放在项目任何位置,你找获得就行,不过建议放在根目录或者应用目录 from rest_framework.response import Response from rest_framework.views import exception_handler as drf_exception_handler from rest_framework import status def exception_handler(exc, context): # 重写错误处理方法 response = drf_exception_handler(exc, context) #先执行系统定义的错误函数 ,过滤掉请求错误 if response is None: # drf没有处理的异常(服务器异常) return Response(status=status.HTTP_500_INTERNAL_SERVER_ERROR, data={ 'status': 7, 'exc': '%s' % exc }) # 项目阶段,要记录到日志文件 return Response(status=response.status_code, data={ 'status': 7, # drf处理的客户端异常,原始处理方式是将异常信息放在response对象的data中,data的格式是{'datail': '具体的异常信息'} 'exc': '%s' % response.data.get('detail') })
Response类生成对象须要的参数,以及Response类的对象可使用的属性
1. 参数: Response(data=响应的数据,status=响应的网络状态码,headers=想经过响应头再携带不部分信息给前端)
属性:response.data response.status_code response.status_text
源码:Response类的_init_方法
def __init__(self, data=None, status=None, template_name=None, headers=None, exception=False, content_type=None): """ Alters the init arguments slightly. For example, drop 'template_name', and instead use 'data'. Setting 'renderer' and 'media_type' will typically be deferred, For example being set automatically by the `APIView`. """ super().__init__(None, status=status) if isinstance(data, Serializer): msg = ( 'You passed a Serializer instance as data, but ' 'probably meant to pass serialized `.data` or ' '`.error`. representation.' ) raise AssertionError(msg) self.data = data self.template_name = template_name self.exception = exception self.content_type = content_type if headers: for name, value in headers.items(): self[name] = value