天天看点

feign.FeignException$BadRequest: status 400 reading 诡异错误记录

诡异事情描述:A客户端通过feign调用 服务端接口报错feign.FeignException$BadRequest: status 400 reading。B客户端通过feign调用服务端接口 又是正常的返回。 A和B 项目中导入的相关feign版本都一样。

代码调整前如下:

客户端代码:

@Data
public class ControlRequestDto implements Serializable {

    private static final long serialVersionUID = -5082851597684307746L;
    //设备点位名对应的水务集团的编码
    private String mappingId;

    //设备点位名
    private String tag;

    //控制命令下发时间
    private Date controlTime;

    //控制命令值
    private String controlValue;
}
           
/**
 * @author wangtingsong
 * @date 2022/5/9 16:15
 * @description 这里的注解FeignClient中的name属性值必须是 你调用的接口模块的spring.application.name 的值
 */
@FeignClient(name = "controlcenter")
public interface ControlCenterClient {
    @PostMapping("/controlcenter/control/controlDevice")
    String controlDevice(@RequestBody ControlRequestDto controlRequestDto);

}
           

服务端代码:

@RestController
@RequestMapping("/control")
@Slf4j
public class ControlCenterController {


    @Autowired
    private ControlCenterService controlCenterService;


    @PostMapping("/controlDevice")
    public BaseResult<Boolean> controlDevice(@RequestBody ControlRequestDto dto){

        log.info("控制设备入参:{}", JSONObject.toJSONString(dto));
        try {
            return controlCenterService.controlDevice(dto);
        }catch (Exception e){
            log.error("控制设备异常:{}", dto.getTag(),e);
            return BaseResult.<Boolean>builder().message("控制设备异常").code("500").build();
        }
    }
}
           

问题查找:根据A和B断点来一步步看,最终是哪里调用不一样。发现一个是A调用的是RetryableFeignLoadBalancer中的execute方法 而B调用的是FeignLoadBalancer 中的execute方法。

最后发现参数中的时间controlTime字段在两个方法中显示有点不一样,然后就想到估计是两个类中对 参数的时间类型处理有点不同。(这里到底为什么同样的客户端调用 服务端代码 会走两个不同的LoadBalancer,还在继续研究中)。

解决办法:最终解决是给时间字段加上 @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")

//控制命令下发时间

@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")

private Date controlTime;

总结:这种400错误有时候需要注意字段的类型