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 ,这个最简单。

相关推荐
jason.zeng@15022071 分钟前
spring boot mqtt开发-原生 Paho 手动封装(最高灵活性,完全自定义)
java·spring boot·后端
xixi09244 分钟前
selenium的安装配置
开发语言·python
sunnyday04266 分钟前
Filter、Interceptor、Spring AOP 的执行顺序详解
java·spring boot·后端·spring
GIS之路9 分钟前
GDAL 实现影像合并
前端·python·信息可视化
想用offer打牌13 分钟前
一站式了解Spring AI Alibaba的Memory机制
java·人工智能·后端·spring·chatgpt·系统架构
打工的小王17 分钟前
Langchain4j(二)RAG知识库
java·后端·ai·语言模型
SR_shuiyunjian17 分钟前
Python第一次作业
开发语言·python·算法
薛定谔的猫喵喵27 分钟前
基于Python+PyGame实现的一款功能完整的数独游戏,支持多难度选择、实时验证、提示系统、成绩记录,并采用多线程优化加载体验。(文末附全部代码)
python·游戏·pygame
人工智能培训29 分钟前
如何持续、安全地向大模型注入新知识?
人工智能·python·算法·大模型·大模型学习·大模型应用工程师·大模型工程师证书