最近,我开始使用Flask构建应用程序。它非常易于使用,但功能强大且有趣。不幸的是,我在理解应用程序上下文的工作方式时遇到了问题。我知道current_app
,app_context
和current_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_app
中routes.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']
)
答案 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