Spring Gateway自定义请求参数封装的实现示例

 更新时间:2020年09月25日 09:32:51   作者:乔丁  
这篇文章主要介绍了Spring Gateway自定义请求参数封装的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

一、需求

在使用spring gateway作为网关时,我们需要在经过网关的请求中添加一些需要传递给后续服务的公共参数,这个时候就可以用到spring gateway提供的自定义请求参数功能了。

二、寻找解决途径

 1、参考官方文档

我们可以猜测,spring gateway作为网关功能,肯定会提供很多处理请求参数的功能,于是我们查询文档得到如下内容:

 

2、探索GatewayFilterFactory实现规律

  • 通过查询spring官方文档可以看到,spring gateway为我们提供了很多xxxGatewayFilterFactory,而这些factory都有相同点,都是以GatewayFilterFactory结尾的。
  • 在类名中,我们可以根据类名进行大胆的猜测,前面的几个单词是描述他的功能的。
  • 右侧的yml配置文件可以看到,filter的配置也是呈现出一定的规律的。

3、从源码获取实现原理

既然我们是要解决自定义请求参数封装问题,那么我们通过上面描述规律,可以很大胆的猜测AddRequestParameteGatewayFilterFactory就是我们要找的目标。那么我们查看一下他的源码,看看他是如何实现的。

首先看看AddRequestParameteGatewayFilterFactory的继承关系,大概了解一下他的组成

从继承关系来看,还是比较复杂,有点懵逼,那直接进入代码看看他的实现。

我们只粘贴比较核心的代码进行分析,其他代码暂时不用关心。

@Override
	public GatewayFilter apply(NameValueConfig config) {
		return new GatewayFilter() {
			@Override
			public Mono<Void> filter(ServerWebExchange exchange,
					GatewayFilterChain chain) {
				URI uri = exchange.getRequest().getURI();
				StringBuilder query = new StringBuilder();
				//获取请求uri的请求参数(GET请求参数通过拼接key=value形式进行传参)
    String originalQuery = uri.getRawQuery(); 
    
				//判断最后一个字符是否是&,如果不是则拼接一个&,以备后续的参数进行连接
				if (StringUtils.hasText(originalQuery)) {
					query.append(originalQuery);
					if (originalQuery.charAt(originalQuery.length() - 1) != '&') {
						query.append('&');
					}
				}
				//获取config中的key、value,然后拼接到uri请求参数后面
				String value = ServerWebExchangeUtils.expand(exchange, config.getValue());
				// TODO urlencode?
				query.append(config.getName());
				query.append('=');
				query.append(value);
				//把请求参数重新拼接回去,并放入request中传递到过滤链的下一个请求中去
				try {
					URI newUri = UriComponentsBuilder.fromUri(uri)
							.replaceQuery(query.toString()).build(true).toUri();

					ServerHttpRequest request = exchange.getRequest().mutate().uri(newUri)
							.build();

					return chain.filter(exchange.mutate().request(request).build());
				}
				catch (RuntimeException ex) {
					throw new IllegalStateException(
							"Invalid URI query: \"" + query.toString() + "\"");
				}
			}

此方法是用于封装请求参数的具体实现,代码的具体实现步骤已经通过注释进行说明。

config中的key:value又是如何传递进来的呢?

4、filter的配置和参数传递

代码实现中出现了config参数的封装,那这个参数是如何获得的呢?这时候我们就要去查看这个filter是如何使用的了。

  • 从图中可以看到,我们的filter就是在这个配置文件中配置使用的。那为什么配置文件中只有AddRequestParameter配置,而不是AddRequestParameterGatewayFilterFactory呢???
  • 其实这是spring的一种约定,实现了GatewayFilterFactory接口的类在配置使用的时候,需要省略掉后面的GatewayFilterFactory,仅配置前缀即可。
  • 同时,配置文件中的red,blue又是做什么的???
  • 这就是我们需要传递的请求参数了,他以key=red,value=blue的方式进行配置。
  • 然后spring会帮我们把这个键值传入到上面的apply方法的config中。我们通过config的源码就能看到究竟:

可以看到,config类的实现就是一个namevalue,分别对应了配置文件中的两个参数。

三、基于官方实现进行自定义拓展

我们看到,spring利用同种方式,实现了各种filter。但如果这些都不能满足我们的要求,那该怎么办???

1、自定义参数惨景预设

我们现在看到的参数信息都是写死在配置文件中的,无法进行动态参数的传递。我们可以设想一个很普通的场景: 我们需要把用户的登录信息封装到请求参数中,供其他服务使用。那这又该如何实现呢???

2、参考官方实现进行自定义

其实如果我们只要模仿官方实现,在uri上拼接我们要传递的动态参数就可以了。具体实现如下:

 @Override
   public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
    URI uri = exchange.getRequest().getURI();
    StringBuilder query = new StringBuilder();
    String originalQuery = uri.getRawQuery();

    if (StringUtils.hasText(originalQuery)) {
     query.append(originalQuery);
     if (originalQuery.charAt(originalQuery.length() - 1) != '&') {
      query.append('&');
     }
    }

    String value = ServerWebExchangeUtils.expand(exchange, config.getValue());
    // TODO urlencode?
    query.append(config.getName());
    query.append('=');
    query.append(value);

    //获取redis中用户的缓存信息,拼接到请求参数后面
    String token = exchange.getRequest().getHeaders().getFirst("token");
    if (StringUtils.hasText(token)) {
     AccountEntity accountEntity = accountAdminApiService.loginAccountAdmin(token);
     //通过发射拿到bean的属性和值,以备后面进行传递参数拼接
     Map<String, Object> beanMap = beanValue(accountEntity);
     if (!CollectionUtils.isEmpty(beanMap)) {
      for (String key : beanMap.keySet()) {
       query.append('&').append(key).append('=').append(beanMap.get(key));
      }
     }
    }

    //以下与官方代码一样,省略...
   }

在固定传参参数拼接后面,直接通过token去获取缓存中的用户登陆信息,然后依次拼接对应的属性和值即可。

四、自定义filter的应用。

网上看了很多人写的自定义,其实和本文差不多。但是有点没有提到,定义好了怎么用啊???

通过使用注解@bean的方式进行配置,具体可以自己百度。

这里用一个简易的配置,套用yml配置的规律,直接在配置文件中配置。

default-filters:
  - AccountRequestParameter=k, v

因为我是全局使用的,所以使用的是default-filters进行配置

此外还应该注意,此时自定的filter并不会被spring boot加载,所以在启动的时候会报错。

解决办法:此时还要在对应的类上加上注解@Component,把bean交给spring管理。

至此,我们自定义请求参数封装完成!!!

到此这篇关于Spring Gateway自定义请求参数封装的实现示例的文章就介绍到这了,更多相关Spring Gateway自定义请求参数封装内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 详解Spring Cloud Zuul 服务网关

    详解Spring Cloud Zuul 服务网关

    本篇文章主要介绍了详解Spring Cloud Zuul 服务网关,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-12-12
  • Java实现将类数据逐行写入CSV文件的方法详解

    Java实现将类数据逐行写入CSV文件的方法详解

    这篇文章主要为大家详细介绍了Java如何实现将类数据逐行写入CSV文件,文中的示例代码讲解详细,具有一定的参考价值,需要的可以借鉴一下
    2022-11-11
  • Java IO流之节点流与字符流的相关知识总结

    Java IO流之节点流与字符流的相关知识总结

    今天给大家带来的是关于Java的相关知识,文章围绕着Java节点流与字符流展开,文中有非常详细的介绍及代码示例,需要的朋友可以参考下
    2021-06-06
  • java实现IP地址转换

    java实现IP地址转换

    这篇文章主要为大家详细介绍了java实现IP地址转换,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-11-11
  • 解决elastic-job-ui在使用druid作为数据库连接池时作业维度报错问题

    解决elastic-job-ui在使用druid作为数据库连接池时作业维度报错问题

    这篇文章主要介绍了解决elastic-job-ui在使用druid作为数据库连接池时作业维度报错问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-04-04
  • java实现动态编译并动态加载

    java实现动态编译并动态加载

    这篇文章主要介绍了java实现动态编译并动态加载,需要的朋友可以参考下
    2021-04-04
  • java线程池核心API源码详细分析

    java线程池核心API源码详细分析

    大家好,本篇文章主要讲的是java线程池核心API源码详细分析,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下,方便下次浏览
    2022-01-01
  • 解决Spring boot 嵌入的tomcat不启动问题

    解决Spring boot 嵌入的tomcat不启动问题

    这篇文章主要介绍了解决Spring boot 嵌入的tomcat不启动问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-10-10
  • Java通过正则表达式获取字符串中数字的方法示例

    Java通过正则表达式获取字符串中数字的方法示例

    最近工作中遇到了一个需求,需要利用java获取字符串中的数字,尝试几种方法后发现利用正则表达式实现最为方法,下面这篇文章就主要介绍了Java通过正则表达式获取字符串中数字的方法,文中给出了详细的示例代码,需要的朋友可以参考下。
    2017-03-03
  • Spring Eureka 未授权访问漏洞修复问题小结

    Spring Eureka 未授权访问漏洞修复问题小结

    项目组使用的 Spring Boot 比较老,是 1.5.4.RELEASE ,最近被检测出 Spring Eureka 未授权访问漏洞,这篇文章主要介绍了Spring Eureka 未授权访问漏洞修复问题小结,需要的朋友可以参考下
    2024-04-04

最新评论