feign源码解析--@RequestHeader为空导致的异常

最近在生产环境遇到一个很奇怪的问题,客户端传递过来的值在写入数据库之后,竟然变成了“{name}”这种字段名的占位符形式。查看日志后发现传过来的值为null(不是空字符串),feign远程调用接口已经声明required =false。feign接口详情如下:

Map test(@RequestHeader(value = "name", required = false) String name, @RequestHeader("remark") String remark);

注:本文基于springboot2.0.6 + springcloud Finchley.SR1,使用的feign-core是9.5.1版本。
feign重写了@RequestHeader、@RequestParam、@PathVariable这三个spring-web注解,参数初始化实现类位于openfeign-core.jar的annotation包,实现了AnnotatedParameterProcessor接口。三个注解的处理方式类似,本文从@RequestHeader开始解析。
回到最初提到的问题。出现这种情况,我们可以对feign的参数值设定做出如下设想:
1.初始化feign接口时,对方法中的参数赋一个默认值。
2.调用接口时,把默认值修改为用户传参。
根据以上猜想,我们先从初始化的代码进行分析。RequestHeaderParameterProcessor的主要方法如下:

public boolean processArgument(AnnotatedParameterContext context, Annotation annotation, Method method) {
    int parameterIndex = context.getParameterIndex();
    Class<?> parameterType = method.getParameterTypes()[parameterIndex];
    MethodMetadata data = context.getMethodMetadata();
    //本文中的参数类型为String 不进入if
    if (Map.class.isAssignableFrom(parameterType)) {
        Util.checkState(data.headerMapIndex() == null, "Header map can only be present once.", new Object[0]);
        data.headerMapIndex(parameterIndex);
        return true;
    } else {
        String name = ((RequestHeader)ANNOTATION.cast(annotation)).value();
        Util.checkState(Util.emptyToNull(name) != null, "RequestHeader.value() was empty 
              on parameter %s", new Object[]{parameterIndex});
        context.setParameterName(name);
        //设置context中的参数初始值,以String.format("{%s}", name)方式填充
        Collection<String> header = context.setTemplateParameter(name, (Collection)data.template().headers().get(name));
        data.template().header(name, header);
        return true;
    }
}

至此可以证明,第一个猜想是正确的,参数默认值是{name}。那么,替换参数是什么逻辑呢?为何null值并没有被替换?
通过分析AnnotatedParameterContext的构造方法不难得出,我们追踪的header存放在RequestTemplate的headers字段,以LinkedHashMap存放。在RequestTemplate的resolve方法中发现headers的初始化:

Map<String, Collection<String>> resolvedHeaders = new LinkedHashMap<String, Collection<String>>();
for (String field : headers.keySet()) {
  Collection<String> resolvedValues = new ArrayList<String>();
  for (String value : valuesOrEmpty(headers, field)) {
    //header的值
    String resolved = expand(value, unencoded);
    resolvedValues.add(resolved);
  }
  resolvedHeaders.put(field, resolvedValues);
}
headers.clear();
headers.putAll(resolvedHeaders);

作为替换header值得关键,expand方法是什么样的呢?

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();
  }

终于找到了问题关键,其实解决方案也很简单,如果是null就替换为空字符串即可(是不是很惊喜~)。
Netflix应该也注意到了这个问题,所以在springcloud最新的RELEASE版本Greenwich.SR2中,已经更新至feign-core 10.2.3并修复了这个问题(springcloud更新实在是太快了,所以我也不知道具体哪个版本修复了这个问题)。最新版本中RequestTemplate已经修改的面目全非,其中resolve方法中对headers的初始化修改为如下方式:

if (!this.headers.isEmpty()) {
  resolved.headers(Collections.emptyMap());
  for (HeaderTemplate headerTemplate : this.headers.values()) {
        //这里的expand返回的已经不是{name},而是"name ",注意后面还有空格。
    String header = headerTemplate.expand(variables);
    if (!header.isEmpty()) {
          //因为测试用的name字段value是空,所以这里必然会是空的
      String headerValues = header.substring(header.indexOf(" ") + 1);
          //name这个header字段就这样被丢弃了。简单粗暴,所以现在如果header值为null,会直接抛异常。
      if (!headerValues.isEmpty()) {
        resolved.header(headerTemplate.getName(), headerValues);
      }
    }
  }
}

总结:解决本问题有两种方案:1.对于传入的值判空,如果为null则赋值空字符串。2.更新springcloud版本为最新。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352