我正在使用Flask + gevent,并希望在greenlet的目标函数内访问flask.g
应用程序全局。
我正在使用copy_current_request_context
装饰器,其情况与文档中给出的示例非常相似:
import gevent
from flask import copy_current_request_context, g
@app.route('/')
def index():
g.user_data = 'foobar'
g.more_user_data = 'baz'
@copy_current_request_context
def do_some_work():
some_func(g.user_data, g.more_user_data)
...
gevent.spawn(do_some_work)
return 'Regular response'
但是,我收到以下错误:
AttributeError: '_AppCtxGlobals' object has no attribute 'user_data'
我认为在复制请求上下文时会推送新的应用程序上下文?我在Flask代码here中设置了一个跟踪,似乎就是这种情况。所以错误并不令人惊讶,因为flask.g
对象的应用程序上下文范围为0.10(参见http://flask.pocoo.org/docs/0.12/api/#flask.Flask.app_ctx_globals_class)。
显然,我可以将用户数据作为参数传递给目标函数:
import gevent
from flask import g
@app.route('/')
def index():
g.user_data = 'foobar'
g.more_user_data = 'baz'
def do_some_work(user_data, more_user_data):
some_func(user_data, more_user_data)
...
gevent.spawn(do_some_work, g.user_data, g.more_user_data)
return 'Regular response'
这很好用,但我希望尽可能使用flask.g
。
答案 0 :(得分:7)
flask.g
为bound with the app context,不在请求上下文中,as the doc says:
从Flask 0.10开始,它存储在应用程序上下文中,不再存储在请求上下文中......
copy_current_request_context()
仅复制请求上下文,但为您提供新的应用上下文。您可以创建一个以通过闭包传递当前应用程序上下文:
def copy_current_app_context(f):
from flask.globals import _app_ctx_stack
appctx = _app_ctx_stack.top
def _(*args, **kwargs):
with appctx:
return f(*args, **kwargs)
return _
但是,我更喜欢通过参数明确地将数据传递给greenlet,这更清晰。