Flask/Celery:AttributeError(“无法腌制本地对象 'celery_init_app.<locals>.FlaskTask'”)

Rob*_*ens 5 python attributeerror redis celery flask

python 3.10运行venvWindows 10 pro.

我正在尝试按照教程进行Celery集成Flaskhttps://flask.palletsprojects.com/en/latest/patterns/celery/

# example.py

from celery import Celery, Task
from flask import Flask


def celery_init_app(app: Flask) -> Celery:
    class FlaskTask(Task):
        def __call__(self, *args: object, **kwargs: object) -> object:
            with app.app_context():
                return self.run(*args, **kwargs)

    celery_app = Celery(app.name, task_cls=FlaskTask)
    celery_app.config_from_object(app.config["CELERY"])
    celery_app.set_default()
    app.extensions["celery"] = celery_app
    return celery_app


def create_app() -> Flask:
    app = Flask(__name__)
    app.config.from_mapping(
        CELERY=dict(
            # Redis Docker container connection string
            broker_url="redis://default:redispw@localhost:55000",
            result_backend="redis://default:redispw@localhost:55000",
            task_ignore_result=True,
        ),
    )
    app.config.from_prefixed_env()
    celery_init_app(app)
    return app
Run Code Online (Sandbox Code Playgroud)
# make_celery.py

from example import create_app

flask_app = create_app()
celery_app = flask_app.extensions["celery"]
Run Code Online (Sandbox Code Playgroud)

命令行调用: celery -A make_celery worker --loglevel INFO

跑进:

Unrecoverable error: AttributeError("Can't pickle local object 'celery_init_app.<locals>.FlaskTask'")
[...]
.venv\lib\site-packages\billiard\reduction.py", line 123, in steal_handle
    return _winapi.DuplicateHandle(
PermissionError: [WinError 5] Access is denied
Run Code Online (Sandbox Code Playgroud)

示例存储库上出现相同的错误和回溯: https ://github.com/pallets/flask/tree/main/examples/celery

我该如何解决这个问题?有哪些替代方案可用?

尝试将其class FlaskTask(Task)移出函数的本地范围celery_init_app,但随后我显然失去了对该app变量的访问权限。

m25*_*m25 6

我遇到了同样的问题。将 celery_init_app 更改为像这样修复它

celery_app = Celery(app.name)
celery_app.config_from_object(app.config["CELERY"])
celery_app.Task = FlaskTask
Run Code Online (Sandbox Code Playgroud)

我希望我能解释一下“为什么”。如果有人知道请留言