feign动态调用接口,feign header参数

  feign动态调用接口,feign header参数

  

目录

假装动态设置页眉和原理例如先说解决办法设置假装的页眉信息(两种形式)1.在方法参数前面添加@请求标题注解2.实现请求拦截器接口

 

  

Feign动态设置header和原理

项目中用到了假装做远程调用, 有部分场景需要动态配置页眉

 

  开始的做法是通过@请求标题设置参数来实现动态的页眉配置

  

例如

@ get映射(value=/test ,consumes={MediaType .应用程序_JSON_UTF8_VALUE})字符串访问(@RequestHeader(Auth )字符串验证,@RequestBody表达式表达式);这种方式虽然可以达到页眉的动态配置, 但是当参数过多时会降低接口可用性, 所以想通过传递豆的方式来设置页眉

 

  

先说解决办法

公共类头部感受器实现请求拦截器{ @ Override public void apply(请求模板请求模板){ byte[]bytes=请求模板。请求正文().as bytes();identity身份=JSON对象。解析对象(字节,标识。类);requestTemplate.header(Auth ,identity。getsecret());}}/***配置指定拦截器**/@FeignClient(name=test ,url=127.0.0.1:8300 ,configuration=header interceptor。类)公共接口golangtesthandle 2 { @ get mapping(value=/handler ,consumes={MediaType .应用程序_JSON_UTF8_VALUE})字符串句柄(身份标识);}自定义拦截机实现请求拦截器接口, 回调方法应用提供了请求模板对象, 对象内部封装了请求的所有信息, 最后通过配置指定接口, 之后就随便你怎么玩了(例如通过身体获取接口参数并动态设置标题)

 

  值得注意的一点是头部感受器如果注入到跳羚容器的话会全局生效, 就是说及时没有指定配置也会对全局假装接口生效, 为什么呢?这里简单说明一下

  首先假装为每个假装上课创建春天语境上下文

  弹簧通过调用关键字获取假装工厂实例

  @覆盖公共对象getObject()抛出异常{ return get target();} 内部调用feignclientfatorybean。获取目标()方法

  T T getTarget() { //获取假装上下文假装上下文上下文=this。应用程序上下文。获取bean(假装上下文。类);//构建假装建造者假装Builder builder=feign(上下文);}根据feign(FeignContext上下文)构建建设者

  受保护的假装builder feign(feign context context){ feignlogerfactory logger factory=get(context,FeignLogger

  Factory.class);        Logger logger = loggerFactory.create(this.type);        // @formatter:off        Feign.Builder 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(context, 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<String, RequestInterceptor> 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 <T> Map<String, T> getInstances(String name, Class<T> 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<String> 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 = FeignConfiguration.class) public interface AbcService { }

以上为个人经验,希望能给大家一个参考,也希望大家多多支持盛行IT。

 

郑重声明:本文由网友发布,不代表盛行IT的观点,版权归原作者所有,仅为传播更多信息之目的,如有侵权请联系,我们将第一时间修改或删除,多谢。

留言与评论(共有 条评论)
   
验证码: