在CLI命令中访问Flask应用上下文

时间:2018-08-13 12:13:43

标签: python flask click flask-cli

我想注册一个与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"])

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

1 个答案:

答案 0 :(得分:3)

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