视图中的 Django asyncio 调用不起作用

Posted

技术标签:

【中文标题】视图中的 Django asyncio 调用不起作用【英文标题】:Django asyncio call in views doesn't work 【发布时间】:2018-04-04 09:45:00 【问题描述】:

我在这个问题上挣扎了很长时间。 我正在尝试使用pyppeteer 以pdf 格式导出视图。这是我的功能:

  async def export_pdf(url):                                                                             
      browser = await launch()                                                                           
      page = await browser.newPage() 
      await page.goto(url) 
      pdf = await page.pdf(                                                                              
             
              'printBackground': True                                                                    
             
      )       
      await browser.close()

      return pdf

在我看来这样称呼它:

   response.content = asyncio.get_event_loop().run_until_complete(
      export_pdf(self.request.get_full_path())
  )

但我得到了这个错误

/export-pdf/1/2018/1/1/ 处的运行时错误 线程 'Thread-1' 中没有当前的事件循环。

经过一番研究,我认为有人解决了我的问题,我就这样称呼它(不太明白,但这是关于 django 的事情,我的函数没有在主线程中调用):

  loop = asyncio.new_event_loop()
  asyncio.set_event_loop(loop)
  response.content = loop.run_until_complete(
      export_pdf(
          asyncio.wait(
              export_pdf(self.request.get_full_path())
          )
      )
  )
  loop.close()

但现在我有这个错误:

/export-pdf/1/2018/1/1/ 处的类型错误 期望一个期货列表,而不是协程

我对 python 中的异步非常陌生,问题是,当我将完全相同的代码复制并粘贴到我的 ipython shell 中时,一切正常。

任何解释/灯光将不胜感激!

提前致谢。

编辑: 经过一些研究,我设法遇到了另一个错误,即

信号只在主线程中起作用

奇怪的是错误来自/usr/lib/python3.6/signal.py,甚至不是来自我的virtualenv。

【问题讨论】:

【参考方案1】:

你的方法是正确的,除了有一个额外的asyncio.wait 电话。 只需这样做:

coroutine = export_pdf(self.request.get_full_path())

# nothing is done yet, we need to give this coroutine to an event loop which will run it
loop = asyncio.new_event_loop()
asyncio.set_event_loop(loop)
try:
   data = loop.run_until_complete(coroutine)
finally:
    loop.close()

response.content = data

【讨论】:

是的,我已经尝试删除该 asyncio.wait() 但我得到了完全相同的错误。它来自信号中的/usr/lib/python3.6/signal.py,这很奇怪,因为它不是来自我的venv。我已经更新了我的问题 信号错误是不言自明的:您的export_pdf 函数正在尝试设置新的信号处理程序,而这不能从主线程外部完成。更具体地说,pyppeteer 正在尝试设置三个信号处理程序:github.com/miyakogi/pyppeteer/… 这意味着pyppeteer库不能真正在主线程之外使用,使用主线程在django中不是一个选项。所以,你可以让主线程运行这个export_pdf调用(非常糟糕的主意,你会在运行export_pdf时锁定你的http堆栈,如果另一个请求进来,它将无法正确处理), 或者您可以在启动 pyppeteer 时使用三个handleSIGINThandleSIGTERMhandleSIGHUB 选项,以确保它不会尝试添加信号处理程序(这可能 i> 工作,但由于 pyppeteer 绝对不是从线程中使用的,它最终可能会中断) 或者,迄今为止最简洁的选项:您可以使用 puppeteer(官方 nodeJS 库)创建一个新的内部 HTTP 服务,该服务将公开一个 GET pdf?url=... 端点,该服务将在 nodeJS 中运行,并进行实际的 PDF 渲染,然后您将能够在您的 django 应用程序中执行简单的requests.get('http://my-pdf-renderer/pdf?url=...')【参考方案2】:

也许将其作为 celery 任务执行,并在调用 celery 任务时使用 apply_async。这可能会有所帮助:https://www.youtube.com/watch?v=XjzyOyLbvN8

【讨论】:

【参考方案3】:

我试图做和你一样的事情。 但我没有用线程来搞乱我的 Django 服务器,而是听从 Arthur 的建议,而是决定使用我自己的 Puppeteer 微服务。

但是在这个过程中,我发现有人有同样的想法,并为此创建了一个准备部署的工具:https://github.com/alvarcarto/url-to-pdf-api

基本上,您只需部署此代码(如果您使用 Heroku,则单击按钮),然后使用您的应用程序的 url 开始生成 pdf。

【讨论】:

以上是关于视图中的 Django asyncio 调用不起作用的主要内容,如果未能解决你的问题,请参考以下文章

添加注释以从另一个视图映射

Django - 为啥视图中的“默认值”不起作用?

Django - 详细视图 URL 中的模型 ID,第一个 ID 不起作用

表单动作在 django 中不起作用

2018第17周总结

Django中的异步