”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > 使用自定义 Django 命令自动重新加载 Celery 工作线程

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

发布于2024-08-05
浏览:384

Automatically reload Celery workers with a custom Django command

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

这篇文章将向您展示如何构建一个自定义的 manage.py runworker 命令,该命令在开发过程中自动重新加载 Celery 工作线程。该命令将按照 runserver 进行建模,我们将了解 Django 的自动重新加载在幕后是如何工作的。

在我们开始之前

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

所有源代码和文档链接均适用于发布时(2024 年 7 月)当前版本的 Django 和 Celery。如果您在遥远的将来阅读本文,事情可能会发生变化。

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

解决方案:自定义命令

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

您可以通过在项目的任何应用程序中创建 management/commands/ 目录来在 Django 中创建命令。创建目录后,您可以在该目录 (docs) 中放置一个带有您要创建的命令名称的 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 Worker,而不是向 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在本要点中所做的那样。

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

版本声明 本文转载于:https://dev.to/tylerlwsmith/automatically-reload-celery-workers-with-a-custom-django-command-1ojl?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • 如何使用PHP将斑点(图像)正确插入MySQL?
    如何使用PHP将斑点(图像)正确插入MySQL?
    在尝试将image存储在mysql数据库中时,您可能会遇到一个可能会遇到问题。本指南将提供成功存储您的图像数据的解决方案。 easudy values('$ this-> image_id','file_get_contents($ tmp_image)...
    编程 发布于2025-02-06
  • 为什么我的GO数据库/SQL查询要比直接Postgres PSQL查询要慢?
    为什么我的GO数据库/SQL查询要比直接Postgres PSQL查询要慢?
    使用数据库/sql的查询比直接查询数据库 QUERYing明显慢,尽管使用了相同的查询,但在执行A执行一个明显的性能差异使用Postgres的PSQL实用程序直接查询,并使用GO应用程序中的数据库/SQL软件包进行查询。这种差异在PSQL中毫无疑问的查询占GO中的数十毫秒。数据库/SQL初始化了一...
    编程 发布于2025-02-06
  • 版本5.6.5之前,使用current_timestamp与时间戳列的current_timestamp与时间戳列有什么限制?
    版本5.6.5之前,使用current_timestamp与时间戳列的current_timestamp与时间戳列有什么限制?
    在默认值中使用current_timestamp或mysql版本中的current_timestamp或在5.6.5 这种限制源于遗产实现的关注,这些限制需要为Current_timestamp功能提供特定的实现。消息和相关问题 `Productid` int(10)unsigned not ...
    编程 发布于2025-02-06
  • 如何在JavaScript对象中动态设置键?
    如何在JavaScript对象中动态设置键?
    如何为JavaScript对象变量创建动态键,尝试为JavaScript对象创建动态键,使用此Syntax jsObj['key' i] = 'example' 1;将不起作用。正确的方法采用方括号:他们维持一个长度属性,该属性反映了数字属性(索引)和一个数字属性的数量。标准对象没有模仿这...
    编程 发布于2025-02-06
  • 我可以将加密从McRypt迁移到OpenSSL,并使用OpenSSL迁移MCRYPT加密数据?
    我可以将加密从McRypt迁移到OpenSSL,并使用OpenSSL迁移MCRYPT加密数据?
    将我的加密库从mcrypt升级到openssl 问题:是否可以将我的加密库从McRypt升级到OpenSSL?如果是这样?使用openssl?答案:可以使用mcrypt数据加密数据,可以使用openssl。关于如何使用openssl对McRypt进行加密的数据: openssl_decrypt...
    编程 发布于2025-02-06
  • 如何通过JavaScript中的键找到嵌套对象?
    如何通过JavaScript中的键找到嵌套对象?
    通过键 recursive solutive ); 如果(结果){ 休息; } } } 别的 { 对于(theObject中的var Prop){ con...
    编程 发布于2025-02-06
  • 我如何使用Laravel \'s“ Orderby”关系订购相关的模型记录?
    我如何使用Laravel \'s“ Orderby”关系订购相关的模型记录?
    在Laravel中与Laravel的订单关系一起检索相关的模型记录,从相关模型访问数据时,可以对使用订单方法的结果。例如,以下代码检索作者的所有注释,并将它们显示在列表中:但是,该列表不可用所需的序列排序。要根据帖子ID订购结果,您可以扩展与查询函数的关系:'列'参数指定要进行排序...
    编程 发布于2025-02-06
  • 如何使用不同数量列的联合数据库表?
    如何使用不同数量列的联合数据库表?
    合并列数不同的表 当尝试合并列数不同的数据库表时,可能会遇到挑战。一种直接的方法是在列数较少的表中,为缺失的列追加空值。 例如,考虑两个表,表 A 和表 B,其中表 A 的列数多于表 B。为了合并这些表,同时处理表 B 中缺失的列,请按照以下步骤操作: 确定表 B 中缺失的列,并将它们添加到表的末...
    编程 发布于2025-02-06
  • PHP启动错误:为什么可以加载动态库?
    PHP启动错误:为什么可以加载动态库?
    [2遇到错误消息,表明未能加载动态库。这些错误可能会显着影响PHP功能,这对于迅速解决和解决这些错误至关重要。此问题的一个常见原因是试图加载未安装的PHP扩展程序。要确定相关扩展名,请搜索PHP配置文件中包含扩展名=的行。利用GREP命令在PHP配置目录中递归搜索:修改适当的配置文件,然后重新启动a...
    编程 发布于2025-02-06
  • 潜入系统编程:C的初学者指南
    潜入系统编程:C的初学者指南
    探索系统编程:C 语言初学者指南系统编程涉及与计算机底层硬件和软件交互。C 语言是系统编程的首选语言之一,因为它能够直接访问硬件资源。这篇指南将带你踏上系统编程之旅,从 C 语言基础到实际应用案例。C 语言基础变量和数据类型:变量用于存储数据。在 C 中,变量必须声明其数据类型,例如:int age...
    编程 发布于2025-02-06
  • 现代游戏开发人员的高级JavaScript游戏开发技术
    现代游戏开发人员的高级JavaScript游戏开发技术
    使用JavaScript构建游戏比以往任何时候都更令人兴奋。无论您是在编码经典平台游戏还是复杂的模拟,都知道如何充分利用工具,可以改变游戏规则。本指南深入研究了JavaScript游戏开发的基本策略和高级技术,这些技术可以帮助您提高自己的技巧。 1。游戏开发中的网络工作者 为什么要使...
    编程 发布于2025-02-06
  • 如何修复\“常规错误:2006 MySQL Server在插入数据时已经消失\”?
    如何修复\“常规错误:2006 MySQL Server在插入数据时已经消失\”?
    插入记录时如何解决“一般错误:2006 MySQL 服务器已消失”介绍:将数据插入 MySQL 数据库有时会导致错误“一般错误:2006 MySQL 服务器已消失”。当与服务器的连接丢失时会出现此错误,通常是由于 MySQL 配置中的两个变量之一所致。解决方案:解决此错误的关键是调整wait_tim...
    编程 发布于2025-02-06
  • 在映射到MySQL枚举列时,如何确保冬眠保留值?
    在映射到MySQL枚举列时,如何确保冬眠保留值?
    在hibernate中保存枚举值:故障排除错误的列type ,他们各自的映射至关重要。在Java中使用枚举类型时,至关重要的是,建立冬眠的方式如何映射到基础数据库。在您的情况下,您已将MySQL列定义为枚举,并在Java中创建了相应的枚举代码。但是,您遇到以下错误:“ MyApp中的错误列类型。...
    编程 发布于2025-02-06
  • 我可以在CSS中使用SVG作为伪元素吗?
    我可以在CSS中使用SVG作为伪元素吗?
    使用svgs用作pseudo-element content css content properts允许在使用元素之前或之后使用元素插入各种类型的内容伪元素,例如::之前和::之后。但是,对可以包括哪些内容有限制。可以将svgs用作pseudo-element Content?,现在可以使用s...
    编程 发布于2025-02-06
  • 对象拟合:IE和Edge中的封面失败,如何修复?
    对象拟合:IE和Edge中的封面失败,如何修复?
    解决此问题,我们采用了一个巧妙的CSS解决方案来解决问题:高度:100%; 高度:auto; 宽度:100%; //对于水平块 ,使用绝对定位将图像定位在中心,以object-fit:object-fit:cover in IE和edge消除了问题。现在,图像将按比例扩展,保持所需的效果而不会失...
    编程 发布于2025-02-06

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3