Java请求进入Python FastAPI 后,请求体为空,参数不合法

问题

最近在做 Python FastAPI 和 Java spring boot 对接接口的事情。

FastAPI 接口写好后,postman 和 swage 测试都正常,但是 Java 服务使用 HttpExchange 怎么着都请求 400,参数不合法。

检查了请求参数,看着也没问题,也都有值。

Python 服务经过 debug 日志发现,请求体是空:

python 复制代码
@app.post("/test")
async def audit_product_raw_debug(request: Request):
    # 打印原始 body(bytes)
    body_bytes = await request.body()
    logger.info(f"Raw request body (bytes): {body_bytes}")

    # 尝试 decode 为字符串
    try:
        body_str = body_bytes.decode('utf-8')
        logger.info(f"Raw request body (str): {body_str}")
    except Exception as e:
        logger.error(f"Failed to decode body: {e}")

    # 尝试手动解析 JSON(用于调试)
    try:
        import json
        body_json = json.loads(body_str)
        logger.info(f"Parsed JSON: {body_json}")
    except Exception as e:
        logger.error(f"Failed to parse JSON: {e}")

    # 然后继续走正常逻辑(或返回)
    return {"message": "debug only"}

二,解决过程

我写一个极其简单的 fast api demo :uv add fastapi uvcorn

复制代码
# main.py
from fastapi import FastAPI
from pydantic import BaseModel

# 创建 FastAPI 应用
app = FastAPI()

# 定义一个请求体的 Pydantic 模型,表示接收的数据
class DataRequest(BaseModel):
    data: str


# 假设这是你的 Python 逻辑代码
def process_data(data):
    return {"message": f"Processed data: {data}"}


# 定义一个 POST 请求的接口
@app.post("/process")
async def process(request: DataRequest):
    # 调用你的 Python 业务逻辑
    result = process_data(request.data)
    return result


if __name__ == "__main__":
    import uvicorn
    uvicorn.run("main:app", host="192.168.0.103", port=8000, reload=True)

和一个简单的 Java 客户端请求:

java 复制代码
public class App 
{
    public static void main( String[] args ) throws IOException, InterruptedException {
        // 创建 HttpClient
        HttpClient client = HttpClient.newHttpClient();

        // 构造请求体
        String jsonRequest = "{\"data\": \"Hello, FastAPI!\"}";

        // 创建 POST 请求
        HttpRequest request = HttpRequest.newBuilder()
                .uri(URI.create("http://192.168.0.103:8000/process"))
                .header("Content-Type", "application/json")
                .POST(HttpRequest.BodyPublishers.ofString(jsonRequest))
                .build();

        // 发送请求并接收响应
        HttpResponse<String> response = client.send(request, HttpResponse.BodyHandlers.ofString());

        // 输出响应
        System.out.println("Response status code: " + response.statusCode());
        System.out.println("Response body: " + response.body());

        // Response status code: 400
        //Response body: Invalid HTTP request received.
    }
}

经过测试发现,Python web 可以正常获取到 Java 客户端的请求,而Java 客户端也可以正常请求到 Python 的 http 响应。

但关键来了,当 Python 处理 Java 请求时,打印了这么一个日志:

复制代码
WARNING:  Unsupported upgrade request.
WARNING:  No supported WebSocket library detected. Please use "pip install 'uvicorn[standard]'", or install 'websockets' or 'wsproto' manually.

于是我进行了如下测试:

测试 1 :执行 uv add 'uvicorn[standard]' ,运行 Python web 服务,然后 Java 客户端测试,结果真的复现了前面我提到的错误,Python 请求体是空的,日志:

复制代码
WARNING:  Unsupported upgrade request.
WARNING:  Invalid HTTP request received.
INFO:     192.168.0.103:60011 - "POST /process HTTP/1.1" 422 Unprocessable Content

Java 报错:

复制代码
Response status code: 422
Response body: {"detail":[{"type":"missing","loc":["body"],"msg":"Field required","input":null}]}

测试 2uv remove 'uvicorn[standard]' , uv add uvicorn , 重启 Python,Java客户端请求正常;

测试 3uv add websockets ,重启 Python,Java客户端请求正常。

三,问题的解释

关键日志应该是这个:

复制代码
WARNING:  Unsupported upgrade request.
WARNING:  No supported WebSocket library detected. Please use "pip install 'uvicorn[standard]'", or install 'websockets' or 'wsproto' manually.

Java 客户端发送请求时,会尝试使用 HTTP/2 协议,于是请求协议升级;而 uvicorn[standard] 的 http_tools 解析工具 将其视为 "不支持的协议升级请求" ,于是丢弃了请求体。

而 uvicorn 使用 h11 解析器,兼容性更好,依然保留了请求体。

下面通过抓包工具,看看 Java 请求头:

然后看看 postman 为什么能访问,它就没有 upgrade 请求头:

但是奇怪的是,postman 加了 upgrade 请求头依然能正常访问,可能 Java 底层 和 postman 底层还是有什么区别吧。

四,解决方案

方式1,Python 服务,使用 uvicorn + websockets(可选),而不是用 uvicorn[standard]

方式2,如果你想使用 uvicorn[standard] 高性能 http ,可以在 uvicorn 中强制使用 http1.1 ,这样 uvicorn[standard] 会直接忽略升级操作

py 复制代码
uvicorn.run(
        "app.main:app",
        host=settings.service_ip,
        port=settings.service_port,
        reload=settings.debug,
        log_level=settings.log_level.lower(),
        http="h11"
    )

方式3:Java 禁用升级请求头。但具体怎么操作,我还没有弄清楚,不同的客户端使用方式也不一样。至少,我在 Java 原生的 HttpClient 上强制 http1.1 ,还是没效果,还是发送了升级请求。

推荐使用方式 2 ,这个最简单。

相关推荐
94620164zwb52 小时前
外观设置模块 Cordova 与 OpenHarmony 混合开发实战
python
毕设源码-郭学长2 小时前
【开题答辩全过程】以 基于springboot的健身房信息管理为例,包含答辩的问题和答案
java·spring boot·后端
爱编码的傅同学2 小时前
【单例模式】深入理解懒汉与饿汉模式
java·javascript·单例模式
深蓝海拓2 小时前
PySide6从0开始学习的笔记(二十) qdarkstyle的深入应用
笔记·python·qt·学习·pyqt
better_liang2 小时前
每日Java面试场景题知识点之-ThreadLocal在Web项目中的实战应用
java· threadlocal· web开发· 多线程· 企业级开发
网络风云2 小时前
Flask 的 Docker 部署指南
python·docker·flask
Rysxt_2 小时前
Spring Boot 4.0 新特性深度解析与实战教程
java·spring boot·后端
飞天小蜈蚣2 小时前
django的模板渲染、for循环标签、继承模板
数据库·python·django
飞Link2 小时前
【Anaconda】Linux(CentOS7)下安装Anaconda教程
linux·运维·python