10

我试图用来flask.g存储可以在其他函数中访问的变量,但我似乎没有正确地做某事。当我尝试访问时,应用程序会生成以下错误g.nameAttributeError: '_RequestGlobals' object has no attribute 'name'

文档flask.g说:

只需存储您想要的任何东西。例如数据库连接或当前登录的用户。

这是一个完整的最小示例,说明了我在尝试访问创建它的函数之外的变量时收到的错误。任何帮助将不胜感激。

#!/usr/bin/env python

from flask import Flask, render_template_string, request, redirect, url_for, g
from wtforms import Form, TextField

application = app = Flask('wsgi')

@app.route('/', methods=['GET', 'POST'])
def index():
    form = LoginForm(request.form)

    if request.method == 'POST' and form.validate():
        name =  form.name.data
        g.name = name

        # Need to create an instance of a class and access that in another route
        #g.api = CustomApi(name)

        return redirect(url_for('get_posts'))

    else:

        return render_template_string(template_form, form=form)

@app.route('/posts', methods=['GET'])
def get_posts():
    # Need to access the instance of CustomApi here
    #api = g.api
    name = g.name
    return render_template_string(name_template, name=name)


class LoginForm(Form):
    name = TextField('Name')

template_form = """
{% block content %}
<h1>Enter your name</h1>

<form method="POST" action="/">
    <div>{{ form.name.label }} {{ form.name() }}</div><br>
    <button type="submit" class="btn">Submit</button>    
</form>
{% endblock %}

"""

name_template = """
{% block content %}

    <div>"Hello {{ name }}"</div><br>

{% endblock %}

"""

if __name__ == '__main__':
    app.run(debug=True)
4

4 回答 4

19

g对象是基于请求的对象,不会在请求之间持续存在,即在您的请求和您的请求g之间重新创建。indexget_posts

Flask 中的应用程序全局变量

Flask 为您提供了一个特殊的对象,确保它只对活动请求有效,并且将为每个请求返回不同的值。简而言之:它做正确的事,就像它对请求和会话所做的那样。

对于请求之间微小数据的持久存储,请sessions改用。您可以(但不应该)将数据直接存储在app对象中以获得全局(所有会话)应用程序状态,类似于这样config做,如果您找到一个非常好的理由这样做。

对于更复杂的数据,请使用数据库。

于 2012-12-07T05:33:26.990 回答
3

如果您需要跟踪身份验证信息,我建议您使用 Flask 插件之一,例如Flask-LoginFlask-Principal

例如,我们使用 Flask-Principal。当有人进行身份验证(或检测到身份验证 cookie)时,它会发出身份加载信号。然后,我们将他们的登录身份映射到我们数据库中的用户。像这样的东西:

# not actual code
@identity_loaded.connect_via(app)
def on_identity_loaded(sender, identity):
  user = Person.query.filter(Person.username==identity.person.username).one()
  g.user = user

然后我们可以在任何控制器或模板中使用 g.user 。(实际上,我们已经删除了很多内容,这是一个简单、懒惰的 hack,但带来的麻烦多于其价值。)

如果您不想使用模块,则可以在每个请求开始时挂接一个内置信号:

http://flask.pocoo.org/docs/tutorial/dbcon/

# This runs before every request
@app.before_request
def before_request():
    g.user = your_magic_user_function()

然后 g.user 将神奇地随处可用。

我希望这会有所帮助!

于 2012-12-07T06:26:08.147 回答
1

只需在烧瓶中使用会话。在您的情况下,您只想在请求中保存用户/名称,最简单的方法是使用会话。

from flask import session
app.secret_key = 'some key for session'

然后,您的功能可以更改如下:

@app.route('/', methods=['GET', 'POST'])
def index():
    form = LoginForm(request.form)
    if request.method == 'POST' and form.validate():
        session['name'] =  form.name.data
        return redirect(url_for('get_posts'))
    else:
        return render_template_string(template_form, form=form)

@app.route('/posts', methods=['GET'])
def get_posts():
    if 'name' in session:
        name = session['name']
    else:
        name = "Unknown"
    return render_template_string(name_template, name=name)
于 2012-12-07T16:24:09.310 回答
0

我想更深入地了解 g global 在存储数据中的使用。g 仅存储带有请求的数据,当重定向到另一个路由时,g 全局设置回 null,即重置为空。这意味着在一个请求中设置为 g 的任何内容都不能在另一个请求中访问。使用会话存储将跨请求访问的数据。

使用 g global 的一个好处是在连接到数据库以获取用户时。例如,可能是数据库中的管理员。管理员可以使用以下方法存储在 g 全局中。

from flask import Flask, g
app = Flask(__name__)

@app.before_request
def text():
    g.a = User.query.filter_by(email='admin@gmail.com')
    

@app.route("/getTrue", methods=['GET', 'POST'])
def getTrue():
    form = UserForm()
    if form.validate_on_submit():
        if g.a == form.email.data:
              return "Admin is logged in"
        else:
              return "User is logged in"
    return render_template('login.html', form=form) 

在上面的示例中, g 可用于保存将在另一个请求中使用的数据。我希望这会有所帮助。谢谢

于 2021-08-11T15:59:57.563 回答