如何在不触发异步上下文检查的情况下在 Jupyter Notebook 中使用 django 3.0 ORM?

mic*_*ols 45 django ipython python-asyncio jupyter django-3.0

Django 3.0 正在添加asgi/async 支持,并用它来保护在异步上下文中发出同步请求。同时,IPython 刚刚添加了顶级 async/await 支持,它似乎在默认事件循环内运行整个解释器会话。

不幸的是,这两个伟大的补充的结合意味着 jupyter notebook 中的任何 django ORM 操作都会导致SynchronousOnlyOperation异常:

SynchronousOnlyOperation: You cannot call this from an async context - use a thread or sync_to_async.
Run Code Online (Sandbox Code Playgroud)

正如异常消息所说,可以将每个 ORM 调用包装成sync_to_async()类似的形式:

SynchronousOnlyOperation: You cannot call this from an async context - use a thread or sync_to_async.
Run Code Online (Sandbox Code Playgroud)

但这不是很方便,特别是对于通常在属性查找时隐式解析的相关字段。

(我试过%autoawait off魔术,但它并没有在工作中,从快速浏览的文档,我假设这是因为ipykernels始终处于ASYNCIO循环中运行)

那么有没有办法在 django 中禁用异步上下文检查中的同步或在同步上下文中运行 ipykernel?


对于上下文:我编写了一个数据科学包,它使用 django 作为后端服务器,但还在 ORM 之上公开了一个基于 jupyter 的界面,允许您清理/注释数据、跟踪机器学习实验并在 jupyter notebook 中运行所有训练作业.

Woj*_*iej 44

这个对我有用

os.environ["DJANGO_ALLOW_ASYNC_UNSAFE"] = "true"
Run Code Online (Sandbox Code Playgroud)

顺便说一句,我使用命令启动我的笔记本

./manage.py shell_plus --notebook
Run Code Online (Sandbox Code Playgroud)

  • 我把它记在笔记本上 (4认同)
  • 直到我将 Django 从 3.0.0 更新到 3.0.6 后,此解决方案才给我结果。 (3认同)

Sam*_*son 6

与其他答案相反,我建议从 shell 运行:

env DJANGO_ALLOW_ASYNC_UNSAFE=true ./manage.py shell_plus --notebook
Run Code Online (Sandbox Code Playgroud)

并且不修改任何配置文件或启动脚本。

这样做的好处是这些检查在几乎所有其他地方都启用时似乎仍然有用(例如,通过本地调试runserver或运行测试时)。通过文件禁用很容易在太多地方禁用它们,从而否定它们的优势。

请注意,大多数 shell 提供了调用以前调用的命令行的简单方法,例如在 Bash 或 Zsh 中,Ctrl+R然后notebook会找到上次运行具有“笔记本”的东西的时间。在 fish shell 下只需键入notebook并按向上箭头键即可开始反向搜索。


mic*_*ols 5

现在我计划只使用带有新设置分叉版本的 django 来跳过 async_unsafe check。一旦 ORM 获得异步支持,我可能不得不重写我的项目以支持它并删除标志。

编辑:现在有一个 PR 来添加一个 env 变量 ( DJANGO_ALLOW_ASYNC_UNSAFE) 来禁用检查 ( https://github.com/django/django/pull/12172 )

  • 此更改现已计划在 [3.0.1](https://docs.djangoproject.com/en/3.0/releases/3.0.1/) (2认同)