经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 程序设计 » Django » 查看文章
详解配置Django的Celery异步之路踩坑
来源:jb51  时间:2018/11/27 9:58:33  对本文有异议

人生苦短,我用python。

看到这句话的时候,感觉可能确实是很深得人心,不过每每想学学,就又止步,年纪大了,感觉学什么东西都很慢,很难,精神啊注意力啊思维啊都跟不上。今天奶牛来分享自己今天踩的一个坑。

先说说配置过程吧,初学Django,啥都不懂,当然,python也很水,啥东西都得现查现用。Django安装还是很简单的。

  1. apt-get install python3
  2. pip3 install django

嗯,就是两条命令的事儿。

再说celery的安装:

  1. pip3 install celery
  2. pip3 install redis==2.10.6

目前奶牛所在的时间redis for python的版本是redis-3.0.1,为什么要用2.10.6呢?因为3.0.1压根配置就无法运行!!!

继续安装redis server

  1. apt-get install redis
  2. service redis start

然后就可以按照celery的官方教程走了,放个URL:http://docs.celeryproject.org/en/latest/django/index.html

  1. python3 manage.py startproject nenew
  2. cd nenew
  3. python3 manage.py startapp nenewapp
  4. touch ./nenew/celery.py
  5. touch ./nenewapp/tasks.py

然后增加nenew/nenew/celery.py内容为

  1. from __future__ import absolute_import, unicode_literals
  2. import os
  3. from celery import Celery
  4.  
  5. # set the default Django settings module for the 'celery' program.
  6. os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'nenew.settings')
  7.  
  8. app = Celery('nenew')
  9.  
  10. # Using a string here means the worker doesn't have to serialize
  11. # the configuration object to child processes.
  12. # - namespace='CELERY' means all celery-related configuration keys
  13. # should have a `CELERY_` prefix.
  14. app.config_from_object('django.conf:settings', namespace='CELERY')
  15.  
  16. # Load task modules from all registered Django app configs.
  17. app.autodiscover_tasks()
  18.  
  19.  
  20. @app.task(bind=True)
  21. def debug_task(self):
  22. print('Request: {0!r}'.format(self.request))
  23.  

增加nenew/nenew/__init__.py的内容

  1. from __future__ import absolute_import, unicode_literals
  2.  
  3. # This will make sure the app is always imported when
  4. # Django starts so that shared_task will use this app.
  5. from .celery import app as celery_app
  6.  
  7. __all__ = ('celery_app',)
  8.  

增加nenew/nenewtest/tasks.py的内容

  1. # Create your tasks here
  2. from __future__ import absolute_import, unicode_literals
  3. from celery import shared_task
  4.  
  5.  
  6. @shared_task
  7. def add(x, y):
  8. return x + y
  9.  
  10.  
  11. @shared_task
  12. def mul(x, y):
  13. return x * y
  14.  
  15.  
  16. @shared_task
  17. def xsum(numbers):
  18. return sum(numbers)
  19.  

在nenew/nenew/settings.py中增加和修改

  1. ...
  2.  
  3. ALLOWED_HOSTS = ['*']
  4. ....
  5.  
  6. INSTALLED_APPS = [
  7. ...
  8. 'nenewtest',
  9. ]
  10.  
  11. ...
  12.  
  13. CELERY_BROKER_URL = 'redis://localhost:6379/1'
  14. CELERY_RESULT_BACKEND = redis://localhost:6379/0'
  15.  

在nenew/nenewtest/views.py中增加或修改为

  1. from django.shortcuts import render
  2. from django.http import HttpResponse
  3. from .tasks import add
  4. # Create your views here.
  5. def nenewtest(request):
  6. result = add.delay('2','2')
  7. result.ready()
  8. return HttpResponse('nenew Django Celery worker run !')

最后把views添加到nenew/nenew/urls.py中

  1. from django.contrib import admin
  2. from django.urls import path
  3. from nenewtest import views
  4.  
  5. urlpatterns = [
  6. path('admin/', admin.site.urls),
  7. path('test/', views.nenewtest),
  8. ]
  9.  

然后在项目目录nenew执行

  1. celery -A nenew worker -l info

这时候worker启动正常会显示:

-------------- <a href="/cdn-cgi/l/email-protection" rel="external nofollow" data-cfemail="2c4f4940495e556c424942495b">[email protected]</a> v4.2.1 (windowlicker)
---- **** -----
--- * ***  * -- Linux-4.15.0-39-generic-x86_64-with-Ubuntu-18.04-bionic 2018-11-23 17:31:25
-- * - **** ---
- ** ---------- [config]
- ** ---------- .> app:         nenew:0x7fdc5a155cc0
- ** ---------- .> transport:   redis://localhost:6379/1
- ** ---------- .> results:     redis://localhost:6379/0
- *** --- * --- .> concurrency: 1 (prefork)
-- ******* ---- .> task events: OFF (enable -E to monitor tasks in this worker)
--- ***** -----
 -------------- [queues]
                .> celery           exchange=celery(direct) key=celery

这样类似的信息,然后我们启动项目,这里需要新开一个shell:

  1. python3 manage.py runserver 0:80

这样我们就可以通过80端口直接访问我们的web了。地址是http://locahost/test

当我们这里访问正常后,在worker界面会有

[2018-11-23 18:09:19,469: INFO/MainProcess] Received task: nenewtest.tasks.add[35faa0fe-dd48-4f8d-9559-516556a93a40]
[2018-11-23 18:09:19,470: INFO/ForkPoolWorker-1] Task nenewtest.tasks.add[35faa0fe-dd48-4f8d-9559-516556a93a40] succeeded in 0.00031037399821798317s: '22'

如下语句表示执行成功,这样子就表示通过Django的网页我们对celery任务的异步执行成功。

当然,按照我的方法是可以一步成功的,因为奶牛已经踩了一整天的坑了,被一个错误郁闷得不要不要的。

AttributeError: 'float' object has no attribute 'items'

就是这个错误,查遍国内的所有网站都没有结果,然后就去bing的国际版查,然后发现果然是有bug在啊,奶牛这一天浪费得可真是够了。

这是celery的一个issue,在地址https://github.com/celery/celery/issues/5175 ,issue里面提及在2018/11/22日 it's fixed in kombu and celery master。问题的根源是celery对redis3的支持不好,补救方法是

Solution: Roll back redis with pip: pip install redis==2.10.6

然后在commit里面找到是对requirements/extras/redis.txt文件进行版本限定。

  1. -redis>=2.10.5
  2. +redis>=2.10.5,<3

好了,踩坑总算结束了。毕竟是新手,很多问题可能就是潜意识认为是自己的代码有问题,实际。。。实际可能并不是这样子,得多关注源码的更迭和问题处理才好。

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

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号