使用Feign动态设置header和原理分析

网友投稿 1062 2022-08-23


使用Feign动态设置header和原理分析

目录Feign动态设置header和原理例如先说解决办法设置Feign的header信息(两种形式)1.在方法参数前面添加@RequestHeader注解2.实现RequestInterceptor接口

Feign动态设置header和原理

项目中用到了Feign做远程调用, 有部分场景需要动态配置header

开始的做法是通过 @RequestHeader 设置参数来实现动态的header配置

例如

@GetMapping(value = "/test", consumes = {MediaType.APPLICATION_jsON_UTF8_VALUE})

String access(@RequestHeader("Auth") String auth, @RequestBody Expression expression);

这种方式虽然可以达到header的动态配置, 但是当参数过多时会降低接口可用性, 所以想通过传递bean的方式来设置header

先说解决办法

public class HeaderInterceptor implements RequestInterceptor {

@Override

public void apply(RequestTemplate requestTemplate) {

byte[] bytes = requestTemplate.requestBody().asBytes();

Identity identity = JSONObject.parseObject(bytes, Identity.class);

requestTemplate.header("Auth", identity.getSecret());

}

}

/**

* configuration指定Interceptor

**/

@FeignClient(name = "test", url = "127.0.0.1:8300", configuration = HeaderInterceptor.class)

public interface golangTestHandle2 {

@GetMapping(value = "/handler", consumes = {MediaType.APPLICATION_JSON_UTF8_VALUE})

String handle(Identity identity);

}

自定义Interceptor实现RequestInterceptor接口, 回调方法apply提供了RequestTemplate对象, 对象内部封装了request的所有信息, 最后通过configuration指定接口, 之后就随便你怎么玩了(例如通过body获取接口参数并动态设置header)

值得注意的一点是HeaderInterceptor如果注入到Springboot容器的话会全局生效, 就是说及时没有指定configuration也会对全局feign接口生效, 为什么呢? 这里简单说明一下

首先Feign为每个feign class创建springcontext上下文

spring通过调用getObject获取feign工厂实例

@Override

public Object getObject() throws Exception {

return getTarget();

}

内部调用FeignClientFatoryBean.getTarget()方法

T getTarget() {

//获取feign上下文

FeignContext context = this.applicationContext.getBean(FeignContext.class);

//构建feign Builder

Feign.Builder builder = feign(context);

...

}

根据feign(FeignContext context)构建Builder

protected Feign.Builder feign(FeignContext context) {

FeignLoggerFactory loggerFactory = get(context, FeignLoggerFactory.class);

Logger logger = loggerFactory.create(this.type);

// @formatter:off

Feign.BuildePGZWHznr builder = get(context, Feign.Builder.class)

// required values

.logger(logger)

//默认springEncoder

.encoder(get(context, Encoder.class))

//默认OptionalDecoder

.decoder(get(context, Decoder.class))

//默认SpringMvcContrat

.contract(get(context, Contract.class));

// @formatter:on

//配置该feign的context

configureFeign(conPGZWHzntext, builder);

return builder;

}

在构建过程中通过FeignClientFactoryBean.configureUsingConfiguration为feign class注册基本的配置项, 其中也包括了Interceptor的注册

protected void configureUsingConfiguration(FeignContext context,

Feign.Builder builder) {

Logger.Level level = getOptional(context, Logger.Level.class);

if (level != null) {

builder.logLevel(level);

}

Retryer retryer = getOptional(context, Retryer.class);

if (retryer != null) {

builder.retryer(retryer);

}

ErrorDecoder errorDecoder = getOptional(context, ErrorDecoder.class);

if (errorDecoder != null) {

builder.errorDecoder(errorDecoder);

}

Request.Options options = getOptional(context, Request.Options.class);

if (options != null) {

builder.options(options);

}

//从feign context获取interceptors

Map requestInterceptors = context

.getInstances(this.contextId, RequestInterceptor.class);

if (requestInterceptors != null) {

builder.requestInterceptors(requestInterceptors.values());

}

if (this.decode404) {

builder.decode404();

}

}

contextId为具体的feign class id, RequestInterceptor为具体的接口, 即是说通过context.getInstances获取所有RequestInterceptor实例并注册到builder中.

public Map getInstances(String name, Class type) {

AnnotationConfigApplicationContext context = getContext(name);

//使用beanNamesForTypeIncludingAncestors

if (BeanFactoryUtils.beanNamesForTypeIncludingAncestors(context,

type).length > 0) {

return BeanFactoryUtils.beansOfTypeIncludingAncestors(context, type);

}

return null;

}

获取工厂中的实例使用的是beanNamesForTypeIncludingAncestors方法, 该方法不仅会从feign的factory中查找, 也会通过父级别spring工厂查找相应实例(类似于springmvc的工厂)

也是因为该方法, 即使你没有在FeignClient中配置configuration, 但是你的Interceptor通过@Component等方法注入容器的话也会全局生效的, 所以如果指向让你的Interceptor部分生效不让它注入到Spring容器就好

设置Feign的header信息(两种形式)

在使用微服务SpringCloud全家桶组件Fegin的时候,我们在进行远程服务之间调用的同时,为了防止客户端劫持信息,我们需要将一些敏感信息添加到我们的Fegin头部(Header)当中,今天朋友问起,总结一下:那么工作中常见的方式有两种

1.在方法参数前面添加@RequestHeader注解

@PostMapping(value = "/getPersonDetail")

public ServerResponse getPersonDetail(@RequestBody Map map,@RequestHeader(name = "id") String id);

使用@RequestHeader(name = "id")可以传递动态header属性

2.实现RequestInterceptor接口

设置Header(所有的Fegin请求)

import org.springframework.context.annotation.Configuration;

import org.springframework.web.context.request.RequestContextHolder;

import org.springframework.web.context.request.ServletRequestAttributes;

import feign.RequestInterceptor;

import feign.RequestTemplate;

@Configuration

public class FeignConfiguration implements RequestInterceptor {

@Override

public void apply(RequestTemplate template) {

ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();

HttpServletRequest request = attributes.getRequest();

Enumeration headerNames = request.getHeaderNames();

if (headerNames != null) {

while (headerNames.hasMoreElements()) {

String name = headerNames.nextElement();

String values = request.getHeader(name);

template.header(name, values);

}

}

}

}

@Component

@FeignClient(value = "abc",fallback = abcServiceHystric.class ,configuration = http://FeignConfiguration.class) public interface AbcService { }


版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:什么是Django(什么是django中间件)
下一篇:Pytest初体验(pytest视频教程)
相关文章

 发表评论

暂时没有评论,来抢沙发吧~