1 缘起

继gateway限流篇:https://blog.csdn.net/Xin_101/article/details/127890605
之后,继续补充网关统一鉴权的相关应用,
网关作为所有流量入口,承接所有请求,因此,可以在网关层统一做鉴权,
授权的放行,未授权的禁行,这里,可以添加黑白名单的功能,
白名单,无需鉴权,直接放行;
黑名单,直接禁行,
本文,即通过实战讲解网关鉴权以及黑白名单的使用。

2 架构

整体架构如下图所示,
由图可知,该体系共有三部分:
(1)网关层:做统一鉴权,保护后面的服务;
(2)注册中心层:管理所有服务;
(3)服务层:业务或其他功能性服务。

3 网关配置

网关配置的白名单是针对保护的资源,
当用户访问白名单的资源时,无需鉴权,即直接通过约定,即可获取正确的响应,
访问非白名单的资源时,需要鉴权成功后,才能获取正确的响应,否则,无法获取正确的响应。

3.1 白名单配置

白名单配置有两种方式,
(1)在启动的文件application.yml中配置;
(2)在数据库Redis或MySQL中配置,启动应用时,需要将数据加载到内存(JVM),
如何在SpringBoot启动时加载外部数据到内存参见:https://blog.csdn.net/Xin_101/article/details/127945236;
简单起见,本文直接在配置文件application.yml中配置白名单,
样例如下:

request:white-list:- /api/v1/get/test

3.2 白名单映射的实体

由上面的配置可知,映射的实体中数据类型为List,
白名单实体如下图所示,完整样例在图后面。

package com.monkey.gateway_template.config;import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.stereotype.Component;import java.util.List;/*** 请求白名单.** @author xindaqi* @since 2022-11-19 15:23*/
@Component
@ConfigurationProperties("request")
public class RequestWhiteList {/*** 白名单列表*/List<String> whiteList;public List<String> getWhiteList() {return whiteList;}public void setWhiteList(List<String> whiteList) {this.whiteList = whiteList;}
}

3.2 请求链配置

为了使网关具有统一鉴权的功能,就需要在网关服务中添加请求拦截功能,通过实现GlobalFilter接口来拦截请求。
测试样例核心如下图所示,核心部分均有注释,源码在图后面。
这里为了方便测试,在非白名单鉴权时,没有做token校验,只判断是否在请求头中携带token,
实际应用中开发者自定这部分认证逻辑,比如通过Redis、JWT、Oauth2.0等。

package com.monkey.gateway_template.config;import com.google.gson.Gson;
import com.monkey.gateway_template.response.Response;
import org.apache.http.HttpHeaders;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.cloud.gateway.filter.GatewayFilterChain;
import org.springframework.cloud.gateway.filter.GlobalFilter;
import org.springframework.core.Ordered;
import org.springframework.core.io.buffer.DataBuffer;
import org.springframework.http.HttpStatus;
import org.springframework.http.server.reactive.ServerHttpResponse;
import org.springframework.stereotype.Component;
import org.springframework.web.server.ServerWebExchange;
import reactor.core.publisher.Mono;import javax.annotation.Resource;
import javax.ws.rs.core.MediaType;
import java.nio.charset.StandardCharsets;
import java.util.List;
import java.util.Objects;/*** 请求拦截器(过滤器).** @author xindaqi* @since 2022-11-19 15:18*/
@Component
public class RequestFilter implements GlobalFilter, Ordered {private static final Logger logger = LoggerFactory.getLogger(RequestFilter.class);private static final String TOKEN = "token";// 获取配置的白名单@ResourceRequestWhiteList requestWhiteList;@Overridepublic Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {// 获取请求的URI,网关配置uri的lb时做了剪裁,所以,直接使用原生接口的URI,无需拼接其他内容String requestPath = exchange.getRequest().getPath().value();// 白名单接口直接放行if (Objects.nonNull(requestWhiteList) && requestWhiteList.getWhiteList().contains(requestPath)) {return chain.filter(exchange);}// 非白名单接口需要鉴权:请求头携带tokenList<String> tokenList = exchange.getRequest().getHeaders().get(TOKEN);// 这里为了方便测试,没有做token校验,只判断是否在请求头中携带token,// 实际应用中开发这自定这部分认证逻辑if (Objects.isNull(tokenList) || tokenList.size() == 0) {Gson gson = new Gson();// 这里开发者可以自定义响应的内容,我自己构建的对象Responsebyte[] data = gson.toJson(Response.invalidToken()).getBytes(StandardCharsets.UTF_8);ServerHttpResponse response = exchange.getResponse();DataBuffer buffer = response.bufferFactory().wrap(data);// 状态码配置:未授权401response.setStatusCode(HttpStatus.UNAUTHORIZED);response.getHeaders().add(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_JSON);return response.writeWith(Mono.just(buffer));}return chain.filter(exchange);}@Overridepublic int getOrder() {return Ordered.LOWEST_PRECEDENCE;}
}

4 测试

准备的测试接口如下:
接口由其他服务提供,测试使用的为自建的producer服务,原生接口URI如下表所示

序号 状态 接口
1 白名单接口 /api/v1/get/test
2 非白名单接口 /api/v1/biz/test

producer是通过网关进行访问的,网关和producer均注册到同一个注册中心集群,
所以无需关注producer的IP和PORT,通过网关已经配置,参见gateway限流篇:https://blog.csdn.net/Xin_101/article/details/127890605
所以请求时,通过网关的IP和PORT以及配置的断言Path,这里配置的为producer-server,
所以通过网关请求produer接口时需要添加produer-server前缀。
为了使限流生效,断言Path切不可与服务名相同。

4.1 白名单

白名单接口无需进行授权,直接访问,按照约定数据即可获取正确的数据。
测试结果如下图所示。

4.2 非白名单

非白名单接口需要进行鉴权,
授权通过后,方可获取正确的结果,如果授权失败,则返回未授权相关信息,
测试结果如下图所示。

5 小结

  • 白名单配置有两种方式,
    (1)在启动的文件application.yml中配置;
    (2)在数据库Redis或MySQL中配置,启动应用时,需要将数据加载到内存(JVM),
    如何在SpringBoot启动时加载外部数据到内存参见:https://blog.csdn.net/Xin_101/article/details/127945236;
  • 网关拦截请求核心是实现GlobalFilter接口,重写方法filter(ServerWebExchange exchange, GatewayFilterChain chain)。

实战讲解网关接口统一认证SpringCloudGateway(图+文)相关推荐

  1. 实战讲解Spring定时任务:@Scheduled(图+文+源码)

    1 缘起 最近看到有些定时任务的项目, 使用了Spring自带的定时任务系统,通过添加@Scheduled注解的方式实现, 并且,使用了不只cron表达式的方式实现定时执行, 恍然大悟,原来@Sche ...

  2. 14-网关实战:网关层整合 Swagger 聚合API文档

    上节课介绍了网关层的认证鉴权,今天这节介绍一下网关层如何聚合API接口文文档. 为什么需要聚合API接口文档? 大型微服务系统模块众多,木谷博客系统就有9个,如果这些服务的接口地址没有一个统一,那么客 ...

  3. 实践讲解Spring配置中心config(图+文,本地文件方式)

    1 缘起 微服务的学习过程中,发现了许多服务的配置是相同的,并且项目稳定运行期间不会轻易变更, 于是,自己开始做实验,将这些相同的配置提取出来放在配置中心, 各个服务需要时,通过这个配置中心获取,Sp ...

  4. 实践讲解死锁及死锁检测(图+文+代码)

    1 缘起 在学习ArrayBlockingQueue(ABQ)的过程中, 实现队列阻塞相关的功能是通过java.util.concurrent.locks.Condition, 如enqueue的no ...

  5. 【.NET Core项目实战-统一认证平台】第三章 网关篇-数据库存储配置(1)

    [.NET Core项目实战-统一认证平台]第三章 网关篇-数据库存储配置(1) 原文:[.NET Core项目实战-统一认证平台]第三章 网关篇-数据库存储配置(1) [.NET Core项目实战- ...

  6. 【.NET Core项目实战-统一认证平台】第四章 网关篇-数据库存储配置(2)

    [.NET Core项目实战-统一认证平台]第四章 网关篇-数据库存储配置(2) 原文:[.NET Core项目实战-统一认证平台]第四章 网关篇-数据库存储配置(2) [.NET Core项目实战- ...

  7. 【.NET Core项目实战-统一认证平台】第七章 网关篇-自定义客户端限流

    上篇文章我介绍了如何在网关上增加自定义客户端授权功能,从设计到编码实现,一步一步详细讲解,相信大家也掌握了自定义中间件的开发技巧了,本篇我们将介绍如何实现自定义客户端的限流功能,来进一步完善网关的基础 ...

  8. 【.NET Core项目实战-统一认证平台】第六章 网关篇-自定义客户端授权

    上篇文章[.NET Core项目实战-统一认证平台]第五章 网关篇-自定义缓存Redis 我们介绍了网关使用Redis进行缓存,并介绍了如何进行缓存实现,缓存信息清理接口的使用.本篇我们将介绍如何实现 ...

  9. 【.NET Core项目实战-统一认证平台】第五章 网关篇-自定义缓存Redis

    上篇文章[.NET Core项目实战-统一认证平台]第四章 网关篇-数据库存储配置(2)我们介绍了2种网关配置信息更新的方法和扩展Mysql存储,本篇我们将介绍如何使用Redis来实现网关的所有缓存功 ...

  10. 【.NET Core项目实战-统一认证平台】第十二章 授权篇-深入理解JWT生成及验证流程...

    上篇文章介绍了基于Ids4密码授权模式,从使用场景.原理分析.自定义帐户体系集成完整的介绍了密码授权模式的内容,并最后给出了三个思考问题,本篇就针对第一个思考问题详细的讲解下Ids4是如何生成acce ...

最新文章

  1. 【AJAX】JavaScript的面向对象
  2. python表白源代码加音乐_python炫酷烟花表白源代码
  3. php链接数据库地址填写ip地址,PHP使用IP地址联接MySQL数据库
  4. 微信浏览器的html5页面显示配置等问题汇集 1,禁止微信浏览器分享页面链接 (定点更新)...
  5. 用scikit-learn研究局部线性嵌入(LLE)
  6. Linux学习之服务器搭建——DHCP服务器
  7. Mysql 学习笔记08
  8. 智能指针的释放_堆栈里的悄悄话——智能指针
  9. 拍摄女生人像,摄影师觉得最大的困难是什么?
  10. Java基础篇:算术赋值运算符
  11. jar 添加环境变量
  12. 破解金山ocr文字识别软件
  13. 如何完成一款游戏? | 独立游戏制作
  14. 图普科技李麟|当新零售遇上人工智能
  15. 我喜欢两个男人。。。
  16. 黑龙江伊春现受伤雄性东北虎 目前已被成功救治
  17. python中矩阵的表示_Python表示矩阵的方法分析
  18. A Survey on Deep Learning Techniques for Stereo-based Depth Estimation论文阅读
  19. android 距离传感器 api,Android传感器API之:近距离感应Proximity功能源码
  20. opencv图片保存0字节_openCV将8bit(1像素对应8bit)图像保存为1bit(1像素对应1bit)...

热门文章

  1. 基于JAVA春运出行铁路路线规划推荐系统计算机毕业设计源码+系统+lw文档+部署
  2. linux查看无线网卡漫游,linux 无线命令
  3. JMX 和 管理系统简介(一)
  4. Python-Django毕业设计小斌美食网站(程序+LW)
  5. Java 下载Excel打不开是什么鬼
  6. python-opencv标定相机内参(针孔+鱼目)
  7. vmware共享文件夹不显示
  8. OA流程 工作流设计
  9. VC++ 操作Word(使用微软office控件)
  10. Android SO文件保护加固——加密篇(一)