【问题标题】:Access Flask app context in CLI command在 CLI 命令中访问 Flask 应用程序上下文
【发布时间】:2018-08-13 12:13:42
【问题描述】:

我想注册一个在 Flask 应用工厂的单独文件中定义的 CLI 命令。此命令需要访问app.config。但是,从命令访问 current_app.config 会引发 RuntimeError: Working outside of application context.

app/__init__.py

from flask import Flask
from app.commands import my_command

def create_app():
    app = Flask(__name__, instance_relative_config=True)
    app.config.from_pyfile("config.py")
    app.add_command(my_command)
    return app

instance/config.py

TEST_VARIABLE = "TESTVALUE"

app/commands.py

from flask import current_app

@click.command()
def my_command():
    click.echo(current_app.config["TEST_VARIABLE"])

我希望运行 flask my_command 以输出 TESTVALUE。但是,我收到以下错误:

RuntimeError: Working outside of application context.

This typically means that you attempted to use functionality that needed
to interface with the current application object in some way. To solve
this, set up an application context with app.app_context().  See the
documentation for more information.

我需要使用with app.app_context(): 才能使current_app 工作,但我无法访问app,因为它是在工厂中定义的。如果我在工厂使用@app.cli.command(),这不会有问题,因为我可以访问app 变量,甚至不需要推送应用上下文。

def create_app():
    ...
    @app.cli.command()
    def my_command():
        click.echo(current_app.config["TEST_VARIABLE"])
    ...

但是,我想在其他文件中定义我的命令并让它们使用应用配置中的值,而这需要将所有命令嵌套在工厂函数中。

我尝试使用命令中的工厂创建一个应用程序,这很有效,但我不认为这样做只是为了访问配置变量是一个好主意。

import click
import app

@click.command()
def my_command():
    app_config = app.create_app().config
    click.echo(app_config["TEST_VARIABLE"])

在使用应用程序工厂模式时,如何定义可以访问应用程序上下文的命令?

【问题讨论】:

    标签: python flask click flask-cli


    【解决方案1】:

    Flask's docs about the CLI discuss this.

    @app.cli.command() 在命令运行时自动确保存在应用上下文。但是,在使用应用工厂时使用起来不太方便,因为在定义命令时您无权访问应用。

    在使用应用工厂时,你使用@click.command()app.cli.add_command()分开定义和注册命令,但@click.command不知道应用上下文。使用 @with_appcontext 装饰器确保命令在应用上下文中运行。

    @click.command()
    @with_appcontext
    def my_command():
        config = current_app.config
        click.echo(config["TEST_VARIABLE"])
    
    def create_app():
        app = Flask(__name__)
        ...
        app.cli.add_command(my_command)
        ...
        return app
    
    $ flask my-command
    TESTVALUE
    

    【讨论】:

      猜你喜欢
      • 2021-02-15
      • 2016-05-16
      • 2011-05-25
      • 1970-01-01
      • 1970-01-01
      • 2017-06-08
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多