python-3.x - 如何在 Flask RESTful API 应用程序中访问 json POST 数据
问题描述
我目前正试图在烧瓶 RESTful API 应用程序中访问 POST json 数据。POST 数据只是使用该方法转换为 json 对象的 Pandas 数据帧pandas.to_json()
。
import pandas as pd
from flask import Flask, request, jsonify
from flask_restful import Resource, Api
app = Flask(__name__)
api = Api(app)
# Creation Of Main Endpoint Classes
class Test(Resource):
def post(self):
# Get POST data as json & read it as a DataFrame
new_x = request.get_json()
current_data = pd.read_json(new_x)
return {'message': 'POST data read successfully'}
# Addition of the Endpoint Classes As Endpoints For The RESTFul API
api.add_resource(Test, '/api/v1')
if __name__ == '__main__':
app.run(debug=True)
向此 Test 端点发出 POST 请求后,出现JSONDecodeError: Expecting value
错误。该帖子是使用 requests python 库制作的:
import requests
new_json = df.to_json()
post_url = 'http://127.0.0.1:5000/api/v1'
post_r = requests.post(url=post_url, data=new_json_orient)
print(post_r.json())
堆栈跟踪似乎指出request.get_json()
返回 None 和<class 'NoneType'>
. 结果,熊猫无法读取无。Pandas 和我缺少 POST 数据的位置。
127.0.0.1 - - [08/Mar/2019 13:06:56] "POST /api/v1 HTTP/1.1" 500 -
Traceback (most recent call last):
File "C:\Users\..\AppData\Local\Continuum\anaconda3\envs\..\lib\site-packages\flask\app.py", line 2309, in __call__
return self.wsgi_app(environ, start_response)
File "C:\Users\..\AppData\Local\Continuum\anaconda3\envs\..\lib\site-packages\flask\app.py", line 2295, in wsgi_app
response = self.handle_exception(e)
File "C:\Users\..\AppData\Local\Continuum\anaconda3\envs\..\lib\site-packages\flask_restful\__init__.py", line 273, in error_router
return original_handler(e)
File "C:\Users\..\AppData\Local\Continuum\anaconda3\envs\..\lib\site-packages\flask\app.py", line 1741, in handle_exception
reraise(exc_type, exc_value, tb)
File "C:\Users\..\AppData\Local\Continuum\anaconda3\envs\..\lib\site-packages\flask\_compat.py", line 34, in reraise
raise value.with_traceback(tb)
File "C:\Users\..\AppData\Local\Continuum\anaconda3\envs\..\lib\site-packages\flask\app.py", line 2292, in wsgi_app
response = self.full_dispatch_request()
File "C:\Users\..\AppData\Local\Continuum\anaconda3\envs\..\lib\site-packages\flask\app.py", line 1815, in full_dispatch_request
rv = self.handle_user_exception(e)
File "C:\Users\..\AppData\Local\Continuum\anaconda3\envs\..\lib\site-packages\flask_restful\__init__.py", line 273, in error_router
return original_handler(e)
File "C:\Users\..\AppData\Local\Continuum\anaconda3\envs\..\lib\site-packages\flask\app.py", line 1718, in handle_user_exception
reraise(exc_type, exc_value, tb)
File "C:\Users\..\AppData\Local\Continuum\anaconda3\envs\..\lib\site-packages\flask\_compat.py", line 34, in reraise
raise value.with_traceback(tb)
File "C:\Users\..\AppData\Local\Continuum\anaconda3\envs\..\lib\site-packages\flask\app.py", line 1813, in full_dispatch_request
rv = self.dispatch_request()
File "C:\Users\..\AppData\Local\Continuum\anaconda3\envs\..\lib\site-packages\flask\app.py", line 1799, in dispatch_request
return self.view_functions[rule.endpoint](**req.view_args)
File "C:\Users\..\AppData\Local\Continuum\anaconda3\envs\..\lib\site-packages\flask_restful\__init__.py", line 480, in wrapper
resp = resource(*args, **kwargs)
File "C:\Users\..\AppData\Local\Continuum\anaconda3\envs\..\lib\site-packages\flask\views.py", line 88, in view
return self.dispatch_request(*args, **kwargs)
File "C:\Users\..\AppData\Local\Continuum\anaconda3\envs\..\lib\site-packages\flask_restful\__init__.py", line 595, in dispatch_request
resp = meth(*args, **kwargs)
File "C:\Users\..\Documents\..\app.py", line 35, in post
current_data = pd.read_json(new_x)
File "C:\Users\..\AppData\Local\Continuum\anaconda3\envs\..\lib\site-packages\pandas\io\json\json.py", line 413, in read_json
path_or_buf, encoding=encoding, compression=compression,
File "C:\Users\..\AppData\Local\Continuum\anaconda3\envs\..\lib\site-packages\pandas\io\common.py", line 232, in get_filepath_or_buffer
raise ValueError(msg.format(_type=type(filepath_or_buffer)))
ValueError: Invalid file path or buffer object type: <class 'NoneType'>
我究竟做错了什么?如何访问 new_json POST 数据?提前致谢。
解决方案
这很可能是因为该POST
方法不包含Content-Type
它的任何信息。
- 如果
requests
版本比 2.4.2 新,您可以使用json
而不是data
用于帖子正文(请参阅此处的文档详细信息)。这将自动设置标题'Content-Type': 'application/json'
:
post_r = requests.post(url=post_url, json=new_json_orient)
- 如果您使用的请求较旧,则可以将标头值显式设置为:
headers = {'Content-type': 'application/json'}
post_r = requests.post(url=post_url, data=new_json_orient, headers=headers)
PS 有关Content-Type
这篇文章中提到的标题的更多信息,这里是MDN 文档的链接。
推荐阅读
- php - Sql如何使用位置参数限制查询?
- sql - 在 Oracle 中通过 SQL 查询更改 JSON 层次结构
- postgresql - Delete .. from .. join,postgresql 和 oracle 的语法相同
- java - Android Studio 移除 ActionBar
- java - 如何使同一域内的所有Android项目都以相同的结构创建?
- android - 相机意图打开相机但单击图像时不显示刻度和跨页
- reactjs - 在 React 中使用 HTML 解析包(如 html-react-parser vs dangerouslySetInnerHtml)的优缺点是什么
- c# - 在 C# 中结合泛型函数和泛型。转换类型时出现问题
- java - 读取/写入未定义的 unicode 字符到文件
- unity3d - Unity 2D - 将播放器对象保持在其父面板的边界内