好久没写博客了,今天趁着周末把工作中遇到的问题梳理一下(在这个问题排查过程中,发现自己排查问题的能力还是太弱了,需要加强)。
最近在公司springCloud的项目里,通过feign远程调用其他服务,代码如下,可以看到,这里的RequestHeader里面我传了String类型的tenantId,测试的时候没有问题,但是项目上线后,发现部分用户在调用这个接口的时候返回的不是想要的结果,当然大部分用户使用的时候是正常的。然后我们就打印了日志,查看了调用链路,发现只要RequestHeader里面的tenantId为null的时候,就会出错,client传递的tenantId为null的时候,server收到的参数为“{tenantId}”,你没看错,就是在参数的key上面加了大括号。这个有点匪夷所思了。
/** * 查询用户动态提额(仅限卡面使用) * @param tenantId * @param paramJson * @return */ @RequestMapping(value = "${wk.url.quota}/api/quota/findDynamicQuotaByCustomerId", method = RequestMethod.POST) String findDynamicQuotaByCustomerId(@RequestHeader String tenantId, @RequestBody JSONObject paramJson);
通过排查,查资料,发现是由于feign底层的源码的问题,见下面红色字体部分。当时我们用的springCloud版本是Finchley.SR2。但是新版的springCloud里面已经修复了这个问题。
public static String expand(String template, Map<String, ?> variables) { // 如果没有设置有效变量,则跳过扩展。 if (checkNotNull(template, "template").length() < 3) { return template; } checkNotNull(variables, "variables for %s", template); boolean inVar = false; StringBuilder var = new StringBuilder(); StringBuilder builder = new StringBuilder(); for (char c : template.toCharArray()) { switch (c) { case '{': if (inVar) { // '{{' 是转义字符,不进行解析 builder.append("{"); inVar = false; break; } inVar = true; break; case '}': if (!inVar) { builder.append('}'); break; } inVar = false; String key = var.toString(); //这里的variables就是header的map,由于header中的name值为null,所以这里只有一个key-value Object value = variables.get(var.toString()); if (value != null) { builder.append(value); } else { //“罪魁祸首”就在这里,又把初始化时的默认值返回了 builder.append('{').append(key).append('}'); } var = new StringBuilder(); break; default: if (inVar) { var.append(c); } else { builder.append(c); } } } return builder.toString(); }
所以解决办法很简单:1)升级springCloud版本;2)使出入的参数不能为null。
参考博客:https://www.jianshu.com/p/550fb3b5f533