ChatGPT解决这个技术问题 Extra ChatGPT

从 Flask 视图返回 JSON 响应

我有一个函数可以用 Pandas 分析 CSV 文件并生成带有摘要信息的字典。我想将结果作为 Flask 视图的响应返回。如何返回 JSON 响应?

@app.route("/summary")
def summary():
    d = make_summary()
    # send it back as json

c
codegeek

从 Flask 1.1.0 开始,视图可以直接返回 Python dict,Flask 会自动调用 jsonify

@app.route("/summary")
def summary():
    d = make_summary()
    return d

如果您的 Flask 版本低于 1.1.0 或要返回不同的 JSON 可序列化对象,请导入并使用 jsonify

from flask import jsonify

@app.route("/summary")
def summary():
    d = make_summary()
    return jsonify(d)

d
davidism

jsonify 将您传递给 JSON 的数据序列化。如果您想自己序列化数据,请通过使用 status=200mimetype='application/json' 构建响应来执行 jsonify 的操作。

from flask import json

@app.route('/summary')
def summary():
    data = make_summary()
    response = app.response_class(
        response=json.dumps(data),
        status=200,
        mimetype='application/json'
    )
    return response

d
davidism

将关键字参数传递给 flask.jsonify,它们将作为 JSON 对象输出。

@app.route('/_get_current_user')
def get_current_user():
    return jsonify(
        username=g.user.username,
        email=g.user.email,
        id=g.user.id
    )
{
    "username": "admin",
    "email": "admin@localhost",
    "id": 42
}

如果您已经有一个 dict,您可以直接将其作为 jsonify(d) 传递。


d
davidism

如果您出于某种原因不想使用 jsonify,您可以手动执行它所做的事情。调用 flask.json.dumps 创建 JSON 数据,然后返回具有 application/json 内容类型的响应。

from flask import json

@app.route('/summary')
def summary():
    data = make_summary()
    response = app.response_class(
        response=json.dumps(data),
        mimetype='application/json'
    )
    return response

flask.json 与内置 json 模块不同。如果可用,它将使用更快的 simplejson 模块,并启用与您的 Flask 应用程序的各种集成。


d
davidism

要返回 JSON 响应并设置状态代码,您可以使用 make_response

from flask import jsonify, make_response

@app.route('/summary')
def summary():
    d = make_summary()
    return make_response(jsonify(d), 200)

灵感来自 Flask 问题跟踪器中的 comment


d
davidism

从版本 1.1.0 Flask 开始,if a view returns a dict it will be turned into a JSON response

@app.route("/users", methods=['GET'])
def get_user():
    return {
        "user": "John Doe",
    }

d
davidism

如果您想分析用户上传的文件,Flask quickstart 显示了如何从用户那里获取文件并访问它们。从 request.files 获取文件并将其传递给汇总函数。

from flask import request, jsonify
from werkzeug import secure_filename

@app.route('/summary', methods=['GET', 'POST'])
def summary():
    if request.method == 'POST':
        csv = request.files['data']
        return jsonify(
            summary=make_summary(csv),
            csv_name=secure_filename(csv.filename)
        )

    return render_template('submit_data.html')

request.files'data' 键替换为 HTML 表单中输入的文件名。


d
davidism

Flask 1.1.x 支持在不调用 jsonify 的情况下返回 JSON dict。如果你想返回一个字典以外的东西,你仍然需要调用 jsonify

@app.route("/")
def index():
    return {
        "api_stuff": "values",
    }

相当于

@app.route("/")
def index():
    return jsonify({
        "api_stuff": "values",
    })

请参阅添加此内容的拉取请求:https://github.com/pallets/flask/pull/3111


d
davidism

我使用装饰器返回 jsonfiy 的结果。我认为当一个视图有多个返回时它更具可读性。这不支持返回像 content, status 这样的元组,但我用 app.errorhandler 处理返回错误状态。

import functools
from flask import jsonify

def return_json(f):
    @functools.wraps(f)
    def inner(**kwargs):
        return jsonify(f(**kwargs))

    return inner

@app.route('/test/<arg>')
@return_json
def test(arg):
    if arg == 'list':
        return [1, 2, 3]
    elif arg == 'dict':
        return {'a': 1, 'b': 2}
    elif arg == 'bool':
        return True
    return 'none of them'

d
davidism

在 Flask 0.11 之前,jsonfiy 不允许直接返回数组。相反,将列表作为关键字参数传递。

@app.route('/get_records')
def get_records():
    results = [
        {
          "rec_create_date": "12 Jun 2016",
          "rec_dietary_info": "nothing",
          "rec_dob": "01 Apr 1988",
          "rec_first_name": "New",
          "rec_last_name": "Guy",
        },
        {
          "rec_create_date": "1 Apr 2016",
          "rec_dietary_info": "Nut allergy",
          "rec_dob": "01 Feb 1988",
          "rec_first_name": "Old",
          "rec_last_name": "Guy",
        },
    ]
    return jsonify(results=list)

S
Steven C. Howell

在 Flask 1.1 中,如果你返回一个字典,它会自动转换成 JSON。因此,如果 make_summary() 返回字典,您可以

from flask import Flask

app = Flask(__name__)

@app.route('/summary')
def summary():
    d = make_summary()
    return d

SO that asks about including the status code 已关闭,与此重复。因此,为了回答这个问题,您可以通过返回 (dict, int) 形式的元组来包含状态代码。 dict 转换为 JSON,int 将是 HTTP 状态代码。在没有任何输入的情况下,Status 是默认的 200。因此在上面的示例中,代码将为 200。在下面的示例中,它更改为 201。

from flask import Flask

app = Flask(__name__)

@app.route('/summary')
def summary():
    d = make_summary()
    return d, 201  # 200 is the default

您可以使用检查状态代码

curl --request GET "http://127.0.0.1:5000/summary" -w "\ncode: %{http_code}\n\n"

d
davidism

使用 Flask 的基于类的视图时,答案是一样的。

from flask import Flask, request, jsonify
from flask.views import MethodView

app = Flask(__name__)

class Summary(MethodView):
    def get(self):
        d = make_summary()
        return jsonify(d)

app.add_url_rule('/summary/', view_func=Summary.as_view('summary'))

c
c8999c 3f964f64

如果它是一个字典,flask 可以直接返回它(版本 1.0.2)

def summary():
    d = make_summary()
    return d, 200

b
bkoiki950

要序列化对象,请使用 flask 模块中的 jsonify 对对象进行 json 化,字典默认被序列化。此外,如果您正在处理文件,您总是可以使用 make_response


这对现有答案并没有增加太多,这些答案展示了如何使用 jsonifymake_response 并提供更深入的见解。
A
Amin Shojaei

我喜欢这种方式:

    @app.route("/summary")
    def summary():
        responseBody = { "message": "bla bla bla", "summary": make_summary() }
        return make_response(jsonify(responseBody), 200)

你为什么喜欢这样?是什么让这比仅仅返回 responseBody 更好? (顺便说一句,每个 PEP-8 最好使用 snake_case)。
这对于代码审查来说更清楚,否则,返回行将是一个难以阅读的长行。另外,我想通过这种方式我们可以选择编辑 HTTP 状态代码。