Flask 会话不持久(Postman 有效,Javascript 无效)

2024-05-05

我正在开发一个 Flask 服务器,用于通过网络在一些后端 Python 功能和 Javascript 客户端之间进行通信。我正在尝试利用 Flask 的session变量来存储用户在与应用程序交互过程中的特定数据。我已经删除了下面大部分应用程序特定的代码,但我遇到的核心问题仍然存在。

这是我的(简化的)Flask 应用程序的代码:

import json
import os
from flask import Flask, jsonify, request, session

app = Flask(__name__)
app.secret_key = 'my_secret_key'

@app.route('/', methods=['GET'])
def run():
  session['hello'] = 'world'
  return jsonify(session['hello'])

@app.route('/update', methods=['POST'])
def update():
  return jsonify(session['hello'])

if __name__ == '__main__':
  app.run(host='0.0.0.0')

利用 Postman,我可以向我的服务器发出 GET 请求并收到预期的输出"world"。然后我可以使用任意主体发出 POST 请求并收到相同的预期输出"world"(再次使用邮递员)。

使用 Chrome 时,我可以访问我的服务器 IP 并查看预期的输出"world"在页面上。我还可以使用 Javascript(在 Chrome 的控制台中)手动发出 GET 请求,并收到与预期相同的响应。但是,当尝试使用 Javascript 向服务器发送 POST 请求时,出现了问题;服务器显示KeyError: 'hello'当尝试提出此请求时。

这是我用来发出 POST 请求的 Javascript:

var url = 'http://my_server_ip/update';
fetch(url, {
  method: 'POST',
  body: JSON.stringify('arbitrary_string'),
  headers: new Headers({
    'Content-Type': 'application/json'
  })
})
.then(response => response.json())
.then((data) => {
  console.log(data);
})

这里出了什么问题?为什么我可以使用 Postman 发出 GET/POST 请求,但使用 Javascript 发出相同的请求时会遇到错误?


The caveats https://github.com/github/fetch#caveats的部分fetch文档说:

默认情况下,fetch 不会从服务器发送或接收任何 cookie,如果站点依赖于维护用户会话,则会导致未经身份验证的请求。

推荐使用AJAX http://flask.pocoo.org/docs/0.12/patterns/jquery/与 Flask 视图交换信息。

同时,在 Flask 应用程序的代码中,session对象是一本字典。现在,如果您使用字典的键访问字典session['hello']如果该键不存在,则Keyerror被提出。要解决此错误,您可以使用get()字典的方法。

正在发生的事情是:fetch请求未找到helloFlask 会话中的 key(或从 Flask 视图获取会话值)。

user = session.get('hello')
return jsonify(session_info=user)

但这仍然会给你一个null会话的价值{ session_info: null }。为什么会这样?

当您向 Flask 服务器发送 GET/POST 请求时,会在 Flask 中初始化并查询会话。但是,当您发送 JavascriptfetchPOST 请求,您必须首先从 Flask 获取会话值,然后将其作为 POST 请求发送到 Flask 视图,其中return是会话信息。

在您的代码中,当 POST 请求被触发时fetch,当我将有效负载数据发送到 Flask 时,它被正确接收,您可以使用request.get_json()在烧瓶视图中:

@app.route('/update', methods=['POST'])
def update():
  user = session.get('hello')
  payload = request.get_json()
  return jsonify(session_info=user, payload=payload)

这将返回{ payload: 'arbitrary_string', session_info: null }。这也表明fetch没有收到会话信息,因为我们没有先调用 GET 从 Flask 获取会话信息。

Remember:Flask 会话位于 Flask 服务器上。要通过 Javascript 发送/接收信息,您必须进行单独的调用,除非有存储会话 cookie 的规定。

const fetch = require('node-fetch');

var url_get = 'http://my_server_ip';
var url_post = 'http://my_server_ip/update';
fetch(url_get, {
  method:'GET'
}).then((response)=>response.json()).then((data) =>fetch(url_post, {
  method: 'POST',
  body: JSON.stringify(data),
  dataType:'json',
  headers: {
    'Content-Type': 'application/json'
  }
})
.then(response => response.json())
.then((postdata) => {
  console.log(postdata);
}));

Flask 视图会略有变化:

@app.route('/', methods=['GET'])
def set_session():
    session['hello'] = 'world'
    return jsonify(session['hello'])

@app.route('/update', methods=['POST'])
def update():
    payload = request.get_json()
    return jsonify(session_info=payload)

当您现在触发 Javascript 请求时,输出将是:{ session_info: 'world' }

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Flask 会话不持久(Postman 有效,Javascript 无效) 的相关文章

随机推荐