Django3.0 异步通信初体验(小结)

此前博主曾经写过一篇博文,介绍了Django3.0的新特性,其中最主要的就是加入对ASGI的支持,实现全双工的异步通信。

2019年12月2日,Django终于正式发布了3.0版本。怀着无比的期待,我们来尝试一下吧!

(附ASGI官方文档地址:https://asgi.readthedocs.io/en/latest/extensions.html)

一、创建Django3工程

没毛病,可以把服务器关了!

二、学习官方文档

毕竟是非常重要和复杂的新特性,先到官网取取经,看看文档怎么写的。

找了一圈,What?就这么点东西?

忽视前面的css问题,重点在于这个ws请求,居然无法建立连接!返回状态码500,也就是服务器错误!

再看看Pycharm后台的信息:

127.0.0.1:5991 - - [04/Dec/2019:10:30:05] "WSCONNECTING /" - -
2019-12-04 10:30:05,246 ERROR  Exception inside application: Django can only handle ASGI/HTTP connections, not websocket.
 File "d:workfor_testdjango3venvlibsite-packagesdaphnecli.py", line 30, in asgi
  await self.app(scope, receive, send)
 File "d:workfor_testdjango3venvlibsite-packagesdjangocorehandlersasgi.py", line 146, in __call__
  % scope['type']
 Django can only handle ASGI/HTTP connections, not websocket.
127.0.0.1:5991 - - [04/Dec/2019:10:30:05] "WSDISCONNECT /" - -

重点在这句Django can only handle ASGI/HTTP connections, not websocket.

什么?Django不支持Websocket?说好的ASGI,说好的全双工异步通信呢?

难道是我哪里搞错了?不行,我得看看源码去!

五、Django3.0源码

一路点点点,翻了个遍,发现Django3.0与之前的2.2关于ASGI的区别就是多了下面两个文件:

django.core.asgi很简单,一看就懂,不多说:

import django
from django.core.handlers.asgi import ASGIHandler

def get_asgi_application():
  django.setup(set_prefix=False)
  return ASGIHandler()

关键是django.core.handlers.asgi这个文件,不到300行,总共就2个类,代码就不贴了:

  • ASGIRequest:继承了HttpRequest类,一看就知道是构造请求对象
  • ASGIHandler:继承了base.BaseHandler,这个就类似WSGIHandler,用于具体处理ASGI请求

让我们看看它的其中一段代码:

# Serve only HTTP connections.
# FIXME: Allow to override this.
if scope['type'] != 'http':
  raise ValueError(
    'Django can only handle ASGI/HTTP connections, not %s.'
    % scope['type']
  )

这就是我们前面在浏览器中发送ws请求,但是无法建立连接的原因!

如果scope的type属性不是http,那么弹出ValueError异常,并提示不支持该类连接!

再看看人家写的注释,明明白白的写着Serve only HTTP connections. FIXME: Allow to override this.。翻译过来就是只支持HTTP连接,请自己重写这部分内容修复这个缺陷。FIXME!FIXME!

好吧,我承认白高兴了一场。

六、总结

与Django3.0关于异步通信的初体验很不好,有下面几点猜测:

  • 可能我水平不行,不会用Django
  • 可能不是通过Websocket,而是别的手段与ASGI通信
  • Django真的目前只提供了个接口,内部实现还没做

由于相关资料太少,多方查找,据说:

  • Django会在后续的版本陆续实现完整的原生的异步通信能力,从同步机制切换到可兼容的异步机制
  • 目前3.0阶段,要与websocket通信依然得通过django-channel库,还不是原生支持

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持来客网。