记录一次:UT010034: Stream not in async mode

今天写了一个导出接口,很正常的导出:

@GetMapping(value = "/exportSalesOrder")
public void export(Param param, HttpServletResponse response) throws IOException {

}

就这样一个正常的导出,但是突然报错了。

{
"id": "a500fd6c-426f-478e-a8fa-969544c57ece",
"code": 500,
"message": "java.lang.IllegalStateException: UT010034: Stream not in async mode",
"subCode": null,
"subMessage": null,
"data": null,
"errors": null,
"stackTraceElementList": null,
"stackTraceInfo": null
}

百思不得其解,怎么就UT010034了,百度了一圈,都是在说异步问题,我也没用异步。只能靠自己了。
一步步排查:

首先,我们项目框架里有日志拦截打印的,

会判断 入参是不是HttpServletRequest或者HttpServletResponse的,如果是就不打印入参,如果是其他自定义参数会打印出来

 直接报错的点就是这里,打印入参。为什么导出的入参有HttpServletResponse,到这里是会过滤掉的,不会去jsonToString的,应该不会报错把。
接下来:

 

 上图是我找了另外的导出的HttpServletResponse,下图是我报错的导出的HttpServletResponse,可以看出来,上下两个图的HttpServletResponse实际对象是不一样的。上图能过滤的是HttpServletResponse,下图不能过滤的是CounterServletResponseWrapper对象。这就是为什么到了jsonToString的时候报错了,因为jsonToString不能String出CounterServletResponseWrapper这个对象,所以会报错。

那为什么下图的HttpServletResponse的实际对象是CounterServletResponseWrapper呢。
继续深入,

点开CounterServletResponseWrapper这个类,一直进去

 会发现CounterServletResponseWrapper这个实际实现了HttpServletResponse,所以CounterServletResponseWrapper是HttpServletResponse的子类实现类。

然后由于我们底层的日志拦截器没有对CounterServletResponseWrapper这个排除过滤,所以导致不能toString,导致报错。

继续排查,为什么会有CounterServletResponseWrapper这个类呢,这个类在哪呢

可以看到,这个类是在javamelody-spring-boot-starter这个包里。

net.bull.javamelody:javamelody-spring-boot-starter:1.87.0

这个包在项目里有引入,看记录是被引入了很久了,前人引入的,实际没用到,就是一个多余的包。

把这个包去掉,HttpServletResponse就回归正常了。就可以正常的被底层的日志拦截过滤了。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值