Spring Cloud 源码学习之 Feign

欢迎访问陈同学博客原文

Spring Cloud Doc: Declarative REST Client: Feign

本文学习了 Spring Cloud 中 openfeign 组件,代码基于 Finchley.SR1 版本。

什么是Feign

spring-cloud-openfeign 在 Github 描述了其特性:

Declarative REST Client: Feign creates a dynamic implementation of an interface decorated with JAX-RS or Spring MVC annotations

Feign 利用注解来描述接口,简化了 Java HTTP Client 的调用过程,隐藏了实现细节。

下面是个小例子,在A服务中调用User Service。

@FeignClient(name = "USER-SERVICE", fallbackFactory = UserServiceFallback.class)
public interface UserService {
    @GetMapping("/users/{id}")
    User getUser(@PathVariable("id") String id);
}

这种融合 Spring MVC 注解的声明式调用,结合Ribbon做客户端负载均衡,再加上Hystrix的安全守护,简单又强大。

Feign 的原理

在应用启动时,Feign 会自动为 @FeignClient 标记的接口动态创建实现类。在调用接口时,会根据接口上的注解信息来创建RequestTemplate(可参考附录),结合实际调用时的参数来创建Request,最后完成调用。

具体的Http Client可以自由选择,如:Apache Http Client、OkHttp等都可以。

源码学习

入口

Spring Cloud 相关组件的源码的学习入口一般在两个地方可以找到,一是 EnableXXX的注解,二是SPI机制中的spring.factories。

在使用 Feign时,通过 @EnableFeignClients 来启用。

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
@Documented
@Import(FeignClientsRegistrar.class)
public @interface EnableFeignClients {
    ...
    Class<?>[] defaultConfiguration() default {};
}

它以 @Import 的方式将FeignClientsRegistrar实例注入到Spring Ioc 容器中。

动态注册

FeignClientsRegistrar 用于处理 FeignClient 的全局配置和被 @FeignClient 标记的接口,为接口动态创建实现类并添加到Ioc容器。

class FeignClientsRegistrar implements ImportBeanDefinitionRegistrar,
      ResourceLoaderAware, EnvironmentAware {

   @Override
   public void registerBeanDefinitions(AnnotationMetadata metadata,
         BeanDefinitionRegistry registry) {
      // 处理默认配置类 
      registerDefaultConfiguration(metadata, registry);
      // 注册被 @FeignClient 标记的接口
      registerFeignClients(metadata, registry);
   }
}   

@EnableFeignClients 中有个属性 defaultConfiguration,可以用来配置Feign的属性。

public @interface EnableFeignClients {
    Class<?>[] defaultConfiguration() default {};
}

registerDefaultConfiguration() 方法就是获取defaultConfiguration属性值,如果有则将配置类注入到Ioc容器。

private void registerDefaultConfiguration(AnnotationMetadata metadata,
        BeanDefinitionRegistry registry) {
    Map<String, Object> defaultAttrs = metadata
            .getAnnotationAttributes(EnableFeignClients.class.getName(), true);

    if (defaultAttrs != null && defaultAttrs.containsKey("defaultConfiguration")) {
        String name;
        if (metadata.hasEnclosingClass()) {
            name = "default." + metadata.getEnclosingClassName();
        }
        else {
            name = "default." + metadata.getClassName();
        }
        registerClientConfiguration(registry, name,
                defaultAttrs.get("defaultConfiguration"));
    }
}

registerFeignClients()用来处理 @FeignClient 标记的接口。首先扫描了classpath中 @FeignClient 标记的接口,然后注册。

public void registerFeignClients(AnnotationMetadata metadata,
        BeanDefinitionRegistry registry) {
    // classpath scan工具
    ClassPathScanningCandidateComponentProvider scanner = getScanner();
    scanner.setResourceLoader(this.resourceLoader);
    ...
    // 利用FeignClient作为过滤条件
    AnnotationTypeFilter annotationTypeFilter = new AnnotationTypeFilter(
            FeignClient.class);
    for (String basePackage : basePackages) {
        Set<BeanDefinition> candidateComponents = scanner
                .findCandidateComponents(basePackage);
        for (BeanDefinition candidateComponent : candidateComponents) {
            if (candidateComponent instanceof AnnotatedBeanDefinition) {
                ...
                // 注册
                registerFeignClient(registry, annotationMetadata, attributes);
            }
        }
    }
}

由于 @FeignClient 标记的是接口,不是普通对象,因此 Feign 利用了 FeignClientFactoryBean 来特殊处理。

接着看 registerFeignClient(),最重要的是FeignClientFactoryBean.

private void registerFeignClient(BeanDefinitionRegistry registry,
        AnnotationMetadata annotationMetadata, Map<String, Object> attributes) {
    String className = annotationMetadata.getClassName();
    // 拿到FeignClientFactoryBean的BeanDefinitionBuilder
    BeanDefinitionBuilder definition = BeanDefinitionBuilder
            .genericBeanDefinition(FeignClientFactoryBean.class);
    validate(attributes);
    definition.addPropertyValue("url", getUrl(attributes));
    ...

    BeanDefinitionHolder holder = new BeanDefinitionHolder(beanDefinition, className,
            new String[] { alias });
    BeanDefinitionReaderUtils.registerBeanDefinition(holder, registry);
}

也就是说,FeignClient 标记的接口实例会由 FeignClientFactoryBean.getObject() 来搞定。

调试时在 getObject() 加个断点,在创建具体对象时会进入该方法。getObject()时会根据 @FeignClient 注解的一些属性信息来创建bean。

@Override
public Object getObject() throws Exception {
    FeignContext context = applicationContext.getBean(FeignContext.class);
    Feign.Builder builder = feign(context);

    // 如果FeignClient没有指定URL(配置的是service)
    if (!StringUtils.hasText(this.url)) {
        String url;
        if (!this.name.startsWith("http")) {
            url = "http://" + this.name;
        }
        else {
            url = this.name;
        }
        url += cleanPath();
        // 结合ribbon使得客户端具备负载均衡的能力
        return loadBalance(builder, context, new HardCodedTarget<>(this.type,
                this.name, url));
    }
    ...
}

loadBalance()方法如下,注意 client 是 LoadBalancerFeignClient。

protected <T> T loadBalance(Feign.Builder builder, FeignContext context,
        HardCodedTarget<T> target) {
    // 得到的是 LoadBalancerFeignClient
    Client client = getOptional(context, Client.class);
    if (client != null) {
        builder.client(client);
        // HystrixTargeter
        Targeter targeter = get(context, Targeter.class);
        return targeter.target(this, builder, context, target);
    }
    ...
}

跟进targeter.target(),最后会发现调用了SynchronousMethodHandler.create()方法。也就是说,FeignClientFactoryBean.getObject() 返回的是一个SynchronousMethodHandler对象。

public MethodHandler create(Target<?> target, MethodMetadata md,
                            RequestTemplate.Factory buildTemplateFromArgs,
                            Options options, Decoder decoder, ErrorDecoder errorDecoder) {
  return new SynchronousMethodHandler(target, client, retryer, requestInterceptors, logger,
                                      logLevel, md, buildTemplateFromArgs, options, decoder,
                                      errorDecoder, decode404);
}

执行请求

SynchronousMethodHandler是核心类,负责根据参数创建RequestTemplate,然后使用具体的http client执行请求。

看一下SynchronousMethodHandler.invoke()方法。

@Override
public Object invoke(Object[] argv) throws Throwable {
  // 利用参数构建请求模板, argv 就是被MVC注解描述的各种参数
  RequestTemplate template = buildTemplateFromArgs.create(argv);
  Retryer retryer = this.retryer.clone();
  while (true) {
    try {
      // 执行请求  
      return executeAndDecode(template);
    } catch (RetryableException e) {
      retryer.continueOrPropagate(e);
      if (logLevel != Logger.Level.NONE) {
        logger.logRetry(metadata.configKey(), logLevel);
      }
      continue;
    }
  }
}   

具体执行由executeAndDecode()搞定,targetRequest()就是应用Feign的拦截器,decode()用于处理response,可以自定义Decoder.

Object executeAndDecode(RequestTemplate template) throws Throwable {
  // 应用Feign 的拦截器
  Request request = targetRequest(template);
  Response response;
  long start = System.nanoTime();
  try {
    // 真正发起请求  
    response = client.execute(request, options);
    // ensure the request is set. TODO: remove in Feign 10
    response.toBuilder().request(request).build();
  } catch (IOException e) {
    ...
  }
  try {
    // response 处理机制,可以自定义Decoder来处理response
    if (response.status() >= 200 && response.status() < 300) {
      if (void.class == metadata.returnType()) {
        return null;
      } else {
        return decode(response);
      }
    } else if (decode404 && response.status() == 404 && void.class != metadata.returnType()) {
      return decode(response);
    } else {
      throw errorDecoder.decode(metadata.configKey(), response);
    }
  } ...
}

拓展机制

Feign 提供了拓展机制用于定制request和response。

  • custom request

Feign的请求拦截器RequestInterceptor在targetRequest()实现。

// 应用所有自定义拦截器
Request targetRequest(RequestTemplate template) {
  for (RequestInterceptor interceptor : requestInterceptors) {
    interceptor.apply(template);
  }
  // 基于请求模板创建Request对象  
  return target.apply(new RequestTemplate(template));
}

拦截器结构如下:

public interface RequestInterceptor {
  void apply(RequestTemplate template);
}

RequestInterceptor 参数是 RequestTemplate,也就是说在发送具体的请求前,提供了拓展机制可以在每个自定义的拦截器中处理请求信息。

  • custom response

自行实现Decoder接口来处理response

public class MyResponseDecoder implements Decoder {
    // type 为 @FeignClient标记接口中方法的返回值类型
    @Override
    public Object decode(Response response, Type type) throws IOException, DecodeException, FeignException {
       // 通过 response.body().asInputStream() 获取返回的数据
       return 处理后的response.
    }
}

小结

Feign 的精华是一种设计思想,它设计了一种全新的HTTP调用方式,屏蔽了具体的调用细节,与Spring MVC 注解的结合更是极大提高了效率(没有重复造轮子,又设计一套新注解)。

其他的特性列举一下:

  • 利用Spring 的动态注入机制,实现了ImportBeanDefinitionRegistrar接口,为 @FeignClient 标记的接口创建实现类并添加到Ioc容器
  • 设计了良好拓展机制,开发者可以定制request和response。

Feign 其他知识

fallback 与 fallbackFactory

以下内容翻译自Spring Cloud Feign文档

Hystrix 支持 fallback(降级)的概念,在熔断器打开或发生异常时可以执行默认的代码。如果要对某个 @FeignClient 启用 fallback,只需要设置 fallback 属性即可。

@FeignClient(name = "hello", fallback = HystrixClientFallback.class)
protected interface HystrixClient {
    @RequestMapping(method = RequestMethod.GET, value = "/hello")
    Hello iFailSometimes();
}

static class HystrixClientFallback implements HystrixClient {
    @Override
    public Hello iFailSometimes() {
        return new Hello("fallback");
    }
}

如果需要访问fallback触发的原因,可以使用fallbackFactory属性,它可以提供Throwable对象。

@FeignClient(name = "hello", fallbackFactory = HystrixClientFallbackFactory.class)
protected interface HystrixClient {
    @RequestMapping(method = RequestMethod.GET, value = "/hello")
    Hello iFailSometimes();
}

@Component
static class HystrixClientFallbackFactory implements FallbackFactory<HystrixClient> {
    @Override
    public HystrixClient create(Throwable cause) {
        return new HystrixClient() {
            @Override
            public Hello iFailSometimes() {
                return new Hello("fallback; reason was: " + cause.getMessage());
            }
        };
    }
}

官方小提示:

There is a limitation with the implementation of fallbacks in Feign and how Hystrix fallbacks work. Fallbacks are currently not supported for methods that return com.netflix.hystrix.HystrixCommand and rx.Observable.

Decoder Demo

Add possibility to get body from feign.Response as a decoded object

上面提了一下利用Decoder来处理Response,这里再写个小Demo。

UserService提供了下面接口用于获取用户信息,返回的数据中包含 code(状态码,200表示成功)、message(提示信息)和data(实际的数据)

 @GetMapping("/users/{id}")
 public Map getUser(@PathVariable("id") String id) {
     Map<String, Object> result = new HashMap<>();
     result.put("code", "200");
     result.put("message", "ok");
     result.put("data", new User("1001", "张三"));
     return result;
 }

B服务利用Feign调用UserService,注意:返回值为 User,不是Map

@FeignClient(name = "USER-SERVICE")
public interface Userervice {
    @GetMapping("/users/{id}")
    User getUser(@PathVariable("id") String id) ;
}

自定义Decoder(仅用简单的代码做演示,不可实际使用)。

public class MyResponseDecoder implements Decoder {

    @Override
    public Object decode(Response response, Type type) throws IOException, DecodeException, FeignException {
        ObjectMapper objectMapper = new ObjectMapper();
        // 将response转为json
        JSONObject jsonObject = objectMapper.readValue(response.body().asInputStream(), JSONObject.class);

        // 如果UserSerivce处理失败, 抛出异常
        if (!"200".equals(jsonObject.getString("code"))) {
            throw new YourException("your error message");
        }
        // 将data转为目标类型
        return jsonObject.getJSONObject("data").toJavaObject(type);
    }
}

为了让Decoder生效,需要在配置文件中设置 feign.client.config.default.decoder=yourpackage.MyResponseDecoder

通过Decoder,完成了依赖服务返回的数据和目标数据结构的适配,将调用依赖服务变成了和调用应用内方法一样的效果

feign 的常用配置

参考:org.springframework.cloud.openfeign.FeignClientProperties、FeignClientEncodingProperties、FeignHttpClientProperties

以下仅列举一些配置,需要用时可直接查询。

feign:
  hystrix:
    enabled: true
  client:
    config:
      default:
        connectTimeout: 5000
        readTimeout: 5000
        loggerLevel: basic
   # 配置请求和响应压缩     
   compression:
    request:
      enabled: true
      # The list of supported mime types.
      mime-types: text/xml,application/xml,application/json
      # The minimum threshold content size.
      min-request-size: 2048
    response:
      enabled: true

RequestTemplate

RequestTemplate的数据结构如下:

public final class RequestTemplate implements Serializable {
  // 查询参数
  private final Map<String, Collection<String>> queries =
      new LinkedHashMap<String, Collection<String>>();
  // http headers
  private final Map<String, Collection<String>> headers =
      new LinkedHashMap<String, Collection<String>>();
  private String method;
  /* final to encourage mutable use vs replacing the object. */
  private StringBuilder url = new StringBuilder();
  private transient Charset charset;
  private byte[] body;
  private String bodyTemplate;
  private boolean decodeSlash = true;
}

欢迎关注陈同学的公众号,一起学习,一起成长

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

推荐阅读更多精彩内容