• python使用celery实现异步任务执行


    使用celery在django项目中实现异步发送短信

    在项目的目录下创建celery_tasks用于保存celery异步任务。

    在celery_tasks目录下创建config.py文件,用于保存celery的配置信息

      ```broker_url = "redis://127.0.0.1/14"```

    在celery_tasks目录下创建main.py文件,用于作为celery的启动文件

    from celery import Celery
     # 为celery使用django配置文件进行设置
    
    import os
    if not os.getenv('DJANGO_SETTINGS_MODULE'):
        os.environ['DJANGO_SETTINGS_MODULE'] = 'model.settings.dev'
    
     # 创建celery应用
    
    app = Celery('model')
    
     #导入celery配置
    
    app.config_from_object('celery_tasks.config')
     #自动注册celery任务
    app.autodiscover_tasks(['celery_tasks.sms'])

    在celery_tasks目录下创建sms目录,用于放置发送短信的异步任务相关代码。
    将提供的发送短信的云通讯SDK放到celery_tasks/sms/目录下。
    在celery_tasks/sms/目录下创建tasks.py(这个名字是固定的,非常重要,系统将会自动从这个文件中找任务队列)文件,用于保存发送短信的异步任务

     import logging
    
        from celery_tasks.main import app
        from .yuntongxun.sms import CCP
    
        logger = logging.getLogger("django")
    
         #验证码短信模板
        SMS_CODE_TEMP_ID = 1
    
        @app.task(name='send_sms_code')
            def send_sms_code(mobile, code, expires):
    
        发送短信验证码
        :param mobile: 手机号
        :param code: 验证码
        :param expires: 有效期
        :return: None
    
    
        try:
            ccp = CCP()
            result = ccp.send_template_sms(mobile, [code, expires], SMS_CODE_TEMP_ID)
        except Exception as e:
            logger.error("发送验证码短信[异常][ mobile: %s, message: %s ]" % (mobile, e))
        else:
            if result == 0:
                logger.info("发送验证码短信[正常][ mobile: %s ]" % mobile)
            else:
                logger.warning("发送验证码短信[失败][ mobile: %s ]" % mobile)

    在verifications/views.py中改写SMSCodeView视图,使用celery异步任务发送短信

    from celery_tasks.sms import tasks as sms_tasks
    
    class SMSCodeView(GenericAPIView):
        ...
            # 发送短信验证码 这是将时间转化为分钟,constants.SMS_CODE_REDIS_EXPIRES 是常量
            sms_code_expires = str(constants.SMS_CODE_REDIS_EXPIRES // 60)
    
            sms_tasks.send_sms_code.delay(mobile, sms_code, sms_code_expires)
    
            return Response({"message": "OK"})

     在django中使用celery的定时调度任务,执行讲redis的缓存内容定时同步到mysql数据库

      1、首先,在config文件中配置调度

    from celery.schedules import crontab
    from .main import app
    
    # 任务队列的链接地址
    broker_url = 'redis://test.zhuzhaoxia.com:6379/15'
    
    # 结果队列的链接地址
    result_backend = 'redis://test.zhuzhaoxia.com:6379/14'
    
    # 定时任务的调度列表,用于注册定时任务
    app.conf.beat_schedule = {
        'write_article_to_mysql': {
            # 本次定时调度的任务
            'task': 'write_article', # 这里的任务名称必须先到main.py中注册
            # 定时任务的调度周期
            # 'schedule': 60,   # 每120秒执行
            'schedule': crontab(hour=3),   # 每天凌晨3点
              # 'args': (16, 16),  # 注意:任务就是一个函数,所以如果有参数则需要传递
        },
    }

      2、在main.py中引入tasks文件

    # app.autodiscover_tasks(["任务1","任务2"])
    app.autodiscover_tasks(["mycelery.sms","mycelery.mail", "mycelery.article"])

      3、在tasks.py文件中写入任务:

    from mycelery.main import app
    from artical.models import Article
    from django_redis import get_redis_connection
    from datetime import datetime
    
    
    @app.task(name="write_article")
    def write_article():
        """把redis的缓存文章定时写入到mysql中"""
        redis_conn = get_redis_connection("article")
        history_list = redis_conn.keys("article_history_*")
        for user_history_list_bytes in history_list:
            user_history_list = user_history_list_bytes.decode()
            user_id = int( user_history_list.split("_")[-1] )
            artical_list = redis_conn.hgetall(user_history_list)   # 获取到用户的历史记录的最新文章和save_id
            for artical_id_bytes, save_id_bytes in artical_list.items():
                artical_id = artical_id_bytes.decode()
                save_id = save_id_bytes.decode()
                article_dict = redis_conn.hgetall("article_%s_%s_%s" % (user_id, artical_id, save_id)) # 获取到最新的文章保存信息
                try:
                    article = Article.objects.get(pk=artical_id)
                    article.title = article_dict["title".encode()].decode()
                    article.content = article_dict["content".encode()].decode()
                    article.save_id = save_id
                    timestamp = datetime.fromtimestamp(int(float(article_dict["update_time".encode()].decode())))
                    article.updated_time = timestamp
                    article.save()
    
                except Article.DoesNotExist:
                    pass

      4、启动beat调度节点

    celery beta -A mycelery.main  --loglevel=info

      5、重启异步调度节点

    celery -A mycelery.main worker  --loglevel=info
    世间安得双全法,不负如来不负卿
  • 相关阅读:
    Codeforces Round #652 (Div. 2) A. FashionabLee(几何)
    轻量应用服务器如何通过修改apache配置文件实现非https的访问多域名到不同子目录程序?
    在Windows环境下使用hexo搭建博客以及部署到gitee / github
    使用WordPress搭建个人手机博客(阿里云)
    访问自己服务器的ip地址
    php环境无法上传文件的解决方法
    SSRF漏洞
    CSRF全家桶(含义,防御,攻击)
    JS实现HTML实体与字符的相互转换
    CentOS系统下载及应用部署
  • 原文地址:https://www.cnblogs.com/shangguanruoling/p/12029562.html
Copyright © 2020-2023  润新知