current_app并在Flask中导入配置设置

时间:2016-01-02 19:21:02

标签: python flask wtforms

最近,我开始使用Flask构建应用程序。它非常易于使用,但功能强大且有趣。不幸的是,我在理解应用程序上下文的工作方式时遇到了问题。我知道current_appapp_contextcurrent_app引用应该在请求中使用,但这可能是我问题的一部分。

我正在构建一个具有以下文件夹结构的应用程序:

app/
     main/
         __init__.py
         error.py
         forms.py
         routes.py
     static/
     templates/
     __init__.py
     email.py
     models.py
config.py
run.py

我正在from flask import current_approutes.py导入配置设置并按预期工作。

但是当我在forms.py中导入current_app时,我可以做任何我想做的事,但我总是得到这个错误:

Traceback (most recent call last):
  File "./run.py", line 6, in <module>
    app = create_app('development')
  File "/home/letmecode/Projects/python/flask/folder/app/__init__.py", line 34, in create_app
    from main import main as main_blueprint
  File "/home/letmecode/Projects/python/flask/folder/app/main/__init__.py", line 5, in <module>
    from . import routes, errors, forms
  File "/home/letmecode/Projects/python/flask/folder/app/main/routes.py", line 8, in <module>
    from .forms import (ChangePasswordForm, ChangeEmailForm, CreateItemForm, RetrievePasswordForm, LoginForm,
  File "/home/letmecode/Projects/python/flask/folder/app/main/forms.py", line 132, in <module>
    class CreateItemForm(Form):
  File "/home/letmecode/Projects/python/flask/folder/app/main/forms.py", line 133, in CreateItemForm
    print current_app
  File "/home/letmecode/.virtualenvs/folder/local/lib/python2.7/site-packages/werkzeug/local.py", line 362, in <lambda>
    __str__ = lambda x: str(x._get_current_object())
  File "/home/letmecode/.virtualenvs/folder/local/lib/python2.7/site-packages/werkzeug/local.py", line 302, in _get_current_object
    return self.__local()
  File "/home/letmecode/.virtualenvs/folder/local/lib/python2.7/site-packages/flask/globals.py", line 34, in _find_app
    raise RuntimeError('working outside of application context')
RuntimeError: working outside of application context

forms.py只是一个包含flask-wtf表单类和验证器函数的模块。

我的问题是,我想在那里引用应用程序上下文,以获取整个应用程序中使用的设置。

我在这里做错了什么?我猜在forms.py内发生的事情不是请求的一部分,因此以某种方式失败。但是,我还应该如何引用应用程序上下文呢?

修改

#forms.py

from flask import current_app

#[...]

class CreateItemForm(Form):
    title = TextField('Title', [
        validators.Length(min=3, max=140),
        validators.Regexp('^[a-zA-Z][a-zA-Z0-9 ]+$', 0, 'Item titles must start with a letter and consist of numbers and letters, only')
    ])
    description = TextAreaField('Description', [
        validators.Length(min=3, max=1000),
    ])
    tags = TextField('Tags')
    branch = SelectField(
        'Branch',
        choices=current_app.config['BRANCHES']
    )

2 个答案:

答案 0 :(得分:1)

我发现我是个白痴。您不必为每个表单模型提供选择。您可以(并且应该)通过视图/路由功能提供它们,例如:

form.select_field.choices = [('value1', 'label1'), ('value2', 'label2')]

这没有明确回答我的问题,但有些过时了。但是,我不会接受我自己的答案作为解决方案(还),因为我很好奇在这种情况下是否可以从via app / current_app导入配置选项,而不会产生hacky。如果我的好奇心不满意,我会接受自己的答案。

我希望wtforms / flask-wtf文档更少关注显而易见的事情,而更多关注不那么明显。

答案 1 :(得分:1)

根本问题是CreateItemForm并且首次导入forms模块时会创建所有属性。这意味着只要运行branch就会创建import .form字段,因此可以访问current_app

# app/main/routes.py
from .forms import ( ..., CreateItemForm, ...)

查看您的堆栈跟踪,这会在您的create_app来电中发生 - 最有可能的是,您的create_app看起来像这样:

def create_app(config_name):
    app = Flask('some-name-here')
    app.config.from_SOMETHING(config_name)

    # Register things here
    from main import main as main_blueprint

    app.register_blueprint(...)

    return app

# Register things here,您只需创建一个app_context并导入其中的所有内容:

# Register things here
# Creating an explicit application context to allow
# easy access to current_app.config, etc.
with app.app_context():
    from main import main as main_blueprint
    app.register_blueprint(...)

return app