当前位置:首页 > 文章列表 > 文章 > python教程 > 使用自定义 Django 命令自动重新加载 Celery 工作线程

使用自定义 Django 命令自动重新加载 Celery 工作线程

来源:dev.to 2024-07-22 08:15:50 0浏览 收藏

golang学习网今天将给大家带来《使用自定义 Django 命令自动重新加载 Celery 工作线程》,感兴趣的朋友请继续看下去吧!以下内容将会涉及到等等知识点,如果你是正在学习文章或者已经是大佬级别了,都非常欢迎也希望大家都能给我建议评论哈~希望能帮助到大家!

使用自定义 Django 命令自动重新加载 Celery 工作线程

celery 之前有一个 --autoreload 标志,现已被删除。然而,django 在其manage.py runserver 命令中内置了自动重新加载功能。 celery workers 中缺乏自动重新加载会造成令人困惑的开发体验:更新 python 代码会导致 django 服务器使用当前代码重新加载,但服务器触发的任何任务都将在 celery workers 中运行过时的代码。

这篇文章将向您展示如何构建自定义的 manage.py runworker 命令,该命令在开发过程中自动重新加载 celery 工作线程。该命令将模仿 runserver,我们将看看 django 的自动重新加载是如何在幕后工作的。

在我们开始之前

这篇文章假设您有一个已经安装了 celery 的 django 应用程序(指南)。它还假设您了解 django 中的项目和应用程序之间的差异。

所有源代码和文档链接都适用于发布时(2024 年 7 月)当前版本的 django 和 celery。如果你在遥远的将来读到这篇文章,事情可能已经改变了。

最后,主项目目录将在帖子的示例中命名为 my_project。

解决方案:自定义命令

我们将创建一个名为 runworker 的自定义管理.py 命令。因为 django 通过其 runsever 命令提供自动重新加载,所以我们将使用 runserver 的源代码作为我们自定义命令的基础。

您可以通过在项目的任何应用程序中创建 management/commands/ 目录来在 django 中创建命令。创建目录后,您可以在该目录中放置一个带有您要创建的命令名称的 python 文件(文档)。

假设您的项目有一个名为 polls 的应用程序,我们将在 polls/management/commands/runworker.py 中创建一个文件并添加以下代码:

# polls/management/commands/runworker.py

import sys
from datetime import datetime

from celery.signals import worker_init

from django.conf import settings
from django.core.management.base import basecommand
from django.utils import autoreload

from my_project.celery import app as celery_app


class command(basecommand):
    help = "starts a celery worker instance with auto-reloading for development."

    # validation is called explicitly each time the worker instance is reloaded.
    requires_system_checks = []
    suppressed_base_arguments = {"--verbosity", "--traceback"}

    def add_arguments(self, parser):
        parser.add_argument(
            "--skip-checks",
            action="store_true",
            help="skip system checks.",
        )
        parser.add_argument(
            "--loglevel",
            choices=("debug", "info", "warning", "error", "critical", "fatal"),
            type=str.upper,  # transforms user input to uppercase.
            default="info",
        )

    def handle(self, *args, **options):
        autoreload.run_with_reloader(self.run_worker, **options)

    def run_worker(self, **options):
        # if an exception was silenced in managementutility.execute in order
        # to be raised in the child process, raise it now.
        autoreload.raise_last_exception()

        if not options["skip_checks"]:
            self.stdout.write("performing system checks...\n\n")
            self.check(display_num_errors=true)

        # need to check migrations here, so can't use the
        # requires_migrations_check attribute.
        self.check_migrations()

        # print django info to console when the worker initializes.
        worker_init.connect(self.on_worker_init)

        # start the celery worker.
        celery_app.worker_main(
            [
                "--app",
                "my_project",
                "--skip-checks",
                "worker",
                "--loglevel",
                options["loglevel"],
            ]
        )

    def on_worker_init(self, sender, **kwargs):
        quit_command = "ctrl-break" if sys.platform == "win32" else "control-c"

        now = datetime.now().strftime("%b %d, %y - %x")
        version = self.get_version()
        print(
            f"{now}\n"
            f"django version {version}, using settings {settings.settings_module!r}\n"
            f"quit the worker instance with {quit_command}.",
            file=self.stdout,
        )

重要提示: 请务必将 my_project 的所有实例替换为您的 django 项目的名称。

如果您想复制并粘贴此代码并继续编程,您可以安全地停在这里,而无需阅读本文的其余部分。这是一个优雅的解决方案,将在您开发 django 和 celery 项目时为您提供良好的服务。但是,如果您想了解更多有关其工作原理的信息,请继续阅读。

它是如何工作的(可选)

我不会逐行查看此代码,而是按主题讨论最有趣的部分。如果您还不熟悉 django 自定义命令,您可能需要在继续之前查看文档。

自动装弹

这部分感觉最神奇。在命令的handle()方法体内,调用了django的内部autoreload.run_with_reloader()。它接受一个回调函数,每次项目中的 python 文件发生更改时都会执行该函数。 实际上是如何运作的?

让我们看一下 autoreload.run_with_reloader() 函数源代码的简化版本。简化的函数重写、内联和删除代码,以使其操作更加清晰。

# NOTE: This has been dramatically pared down for clarity.

def run_with_reloader(callback_func, *args, **kwargs):
    # NOTE: This will evaluate to False the first time it is run.
    is_inside_subprocess = os.getenv("RUN_MAIN") == "true"

    if is_inside_subprocess:
        # The reloader watches for Python file changes.
        reloader = get_reloader()

        django_main_thread = threading.Thread(
            target=callback_func, args=args, kwargs=kwargs
        )
        django_main_thread.daemon = True
        django_main_thread.start()

        # When the code changes, the reloader exits with return code 3.
        reloader.run(django_main_thread)

    else:
        # Returns Python path and the arguments passed to the command.
        # Example output: ['/path/to/python', './manage.py', 'runworker']
        args = get_child_arguments()

        subprocess_env = {**os.environ, "RUN_MAIN": "true"}
        while True:
            # Rerun the manage.py command in a subprocess.
            p = subprocess.run(args, env=subprocess_env, close_fds=False)
            if p.returncode != 3:
                sys.exit(p.returncode)

当manage.py runworker在命令行中运行时,它会首先调用handle()方法,该方法会调用run_with_reloader()。

在 run_with_reloader() 内部,它将检查名为 run_main 的环境变量是否具有“true”值。当函数第一次被调用时,run_main 应该没有值。

当run_main没有设置为“true”时,run_with_reloader()会进入循环。在循环内,它将启动一个子进程,重新运行传入的manage.py [command_name],然后等待该子进程退出。如果子进程退出并返回代码 3,则循环的下一次迭代将启动一个新的子进程并等待。该循环将一直运行,直到子进程返回不为 3 的退出代码(或直到用户使用 ctrl + c 退出)。一旦得到非3的返回码,就会彻底退出程序。

生成的子进程再次运行manage.py命令(在我们的例子中是manage.py runworker),并且该命令将再次调用run_with_reloader()。这次,run_main 将被设置为“true”,因为该命令在子进程中运行。

现在 run_with_reloader() 知道它位于子进程中,它将获得一个监视文件更改的重新加载器,将提供的回调函数放入线程中,并将其传递给开始监视更改的重新加载器。

当重新加载器检测到文件更改时,它会运行 sys.exit(3)。这将退出子流程,从而触发生成子流程的代码的下一次循环迭代。反过来,会启动一个使用更新版本代码的新子流程。

系统检查和迁移

默认情况下,django 命令在运行其handle() 方法之前执行系统检查。但是,对于 runserver 和我们的自定义 runworker 命令,我们希望推迟运行这些命令,直到进入我们提供给 run_with_reloader() 的回调中。在我们的例子中,这是我们的 run_worker() 方法。这使我们能够运行自动重新加载的命令,同时修复损坏的系统检查。

为了推迟运行系统检查,需要将requires_system_checks属性的值设置为空列表,并通过在run_worker()主体中调用self.check()来执行检查。与 runserver 一样,我们的自定义 runworker 命令也会检查所有迁移是否已运行,如果有待处理的迁移,它会显示警告。

因为我们已经在 run_worker() 方法中执行 django 的系统检查,所以我们通过向 celery 传递 --skip-checks 标志来禁用系统检查,以防止重复工作。

所有与系统检查和迁移相关的代码都是直接从 runserver 命令源代码中提取的。

celery_app.worker_main()

我们的实现使用 celery_app.worker_main() 直接从 python 启动 celery 工作程序,而不是向 celery 发起攻击。

on_worker_init()

此代码在工作进程初始化时执行,显示日期和时间、django 版本以及退出命令。它是根据 runserver 启动时显示的信息建模的。

其他 runserver 样板

以下行也从 runserver 源代码中删除:

  • suppressed_base_arguments = {"--verbosity", "--traceback"}
  • autoreload.raise_last_exception()

日志级别

我们的自定义命令具有可配置的日志级别,以防开发人员希望在不修改代码的情况下从 cli 调整设置。

更进一步

我研究了 django 和 celery 的源代码来构建这个实现,并且有很多扩展它的机会。您可以配置该命令以接受更多 celery 的工作参数。或者,您可以创建一个自定义的 manage.py 命令,它会自动重新加载任何 shell 命令,就像 david browne 在本要点中所做的那样。

如果您觉得本文有用,请随时留下点赞或评论。感谢您的阅读。

以上就是《使用自定义 Django 命令自动重新加载 Celery 工作线程》的详细内容,更多关于的资料请关注golang学习网公众号!

版本声明
本文转载于:dev.to 如有侵犯,请联系study_golang@163.com删除
使用 PHPUnit 在 Laravel 中测试您的 DOM使用 PHPUnit 在 Laravel 中测试您的 DOM
上一篇
使用 PHPUnit 在 Laravel 中测试您的 DOM
JavaScript 的 map() 方法
下一篇
JavaScript 的 map() 方法
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    542次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    508次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    497次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • AI Make Song:零门槛AI音乐创作平台,助你轻松制作个性化音乐
    AI Make Song
    AI Make Song是一款革命性的AI音乐生成平台,提供文本和歌词转音乐的双模式输入,支持多语言及商业友好版权体系。无论你是音乐爱好者、内容创作者还是广告从业者,都能在这里实现“用文字创造音乐”的梦想。平台已生成超百万首原创音乐,覆盖全球20个国家,用户满意度高达95%。
    2次使用
  • SongGenerator.io:零门槛AI音乐生成器,快速创作高质量音乐
    SongGenerator
    探索SongGenerator.io,零门槛、全免费的AI音乐生成器。无需注册,通过简单文本输入即可生成多风格音乐,适用于内容创作者、音乐爱好者和教育工作者。日均生成量超10万次,全球50国家用户信赖。
    2次使用
  •  BeArt AI换脸:免费在线工具,轻松实现照片、视频、GIF换脸
    BeArt AI换脸
    探索BeArt AI换脸工具,免费在线使用,无需下载软件,即可对照片、视频和GIF进行高质量换脸。体验快速、流畅、无水印的换脸效果,适用于娱乐创作、影视制作、广告营销等多种场景。
    2次使用
  • SEO标题协启动:AI驱动的智能对话与内容生成平台 - 提升创作效率
    协启动
    SEO摘要协启动(XieQiDong Chatbot)是由深圳协启动传媒有限公司运营的AI智能服务平台,提供多模型支持的对话服务、文档处理和图像生成工具,旨在提升用户内容创作与信息处理效率。平台支持订阅制付费,适合个人及企业用户,满足日常聊天、文案生成、学习辅助等需求。
    9次使用
  • Brev AI:零注册门槛的全功能免费AI音乐创作平台
    Brev AI
    探索Brev AI,一个无需注册即可免费使用的AI音乐创作平台,提供多功能工具如音乐生成、去人声、歌词创作等,适用于内容创作、商业配乐和个人创作,满足您的音乐需求。
    10次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码