SpringCloud系列教程(十二):网关配置动态路由

server/2025/3/5 8:30:21/

除了token以外,还有一个很实用的功能就是把网关的路由配置放到nacos上,并且修改路由配置的时候,网关服务可以动态的更新,这样我们在调整网络配置的时候,就不用重启服务了。所以我们需要用到两个重要的类:NacosConfigManager和RouteDefinitionWriter,NacosConfigManager用来监听nacos上配置的变化,RouteDefinitionWriter会执行路由配置。

1、将原来写在application.yml中的路由配置信息迁移到nacos上,dataId定义为gateway-routes,这里要注意,原来在项目中yml文件解析的时候,会被SpringCloud层层解析后生成RouteDefinition,在这期间SpringCloud会去解析比如Path=/api/*这种形式,但是现在我们要自己解析,就要改成符合RouteDefinition的完全形式,这是一个小难点。

spring:cloud:gateway:routes:- id: nacos-client-demouri: lb://nacos-client-demopredicates: #断言,匹配访问的路径- name: Pathargs:pattern: /nacos-client-demo/api/**filters: # 过滤器- name: AddRequestHeaderargs:_genkey_0: headername_genkey_1: I am a header!- name: Myargs:_genkey_0: zhangsan_genkey_1: lisi_genkey_2: wangwu- id: open-feign-demouri: lb://open-feign-demopredicates:- name: Pathargs:pattern: /open-feign-demo/api/**

2、在application.yml中添加nacos的配置信息就可以了,并且我特意添加了两个参数,用来指明动态路由配置的dataId,这样以后改成其他的也方便。

spring:main:# gateway组件中的spring-boot-starter-webflux和springboot作为web项目启动必不可少的spring-boot-starter-web出现冲突web-application-type: reactiveapplication:name: gateway-democloud:gateway:nacos-data-id: gateway-routesnacos-group: devopsnacos:server-addr: 192.168.3.54:80username: nacospassword: nacosdiscovery:group: devopsnamespace: sitconfig:namespace: sitgroup: devopsconfig:import: nacos:${spring.application.name}?refresh=true
server:port: 8888

3、定义一个component类DynamicRouterLoader,继承ApplicationEventPublisherAware,网上很多教程没有继承,因为版本不同,我用新版的发现必须要用ApplicationEventPublisherAware发布一下才能让路由信息生效。

package com.mj.gateway.router;import cn.hutool.core.lang.Dict;
import cn.hutool.json.JSONUtil;
import cn.hutool.setting.yaml.YamlUtil;
import com.alibaba.cloud.nacos.NacosConfigManager;
import com.alibaba.fastjson2.JSONObject;
import com.alibaba.nacos.api.config.listener.Listener;
import com.alibaba.nacos.api.exception.NacosException;
import jakarta.annotation.PostConstruct;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.cloud.gateway.event.RefreshRoutesEvent;
import org.springframework.cloud.gateway.route.RouteDefinition;
import org.springframework.cloud.gateway.route.RouteDefinitionWriter;
import org.springframework.context.ApplicationEventPublisher;
import org.springframework.context.ApplicationEventPublisherAware;
import org.springframework.stereotype.Component;
import reactor.core.publisher.Mono;import java.io.BufferedReader;
import java.io.ByteArrayInputStream;
import java.io.InputStreamReader;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import java.util.concurrent.Executor;@Slf4j
@Component
@RequiredArgsConstructor
public class DynamicRouterLoader implements ApplicationEventPublisherAware {//nacos配置管理器private final NacosConfigManager nacosConfigManager;//gateway路由规则写入器private final RouteDefinitionWriter writer;private ApplicationEventPublisher publisher;@Value("${spring.cloud.gateway.nacos-data-id:test}")private String dataId;@Value("${spring.cloud.gateway.nacos-group:test}")private String group;private final int timeout = 5000;//保存所有的路由idprivate final Set<String> routeIds = new HashSet<>();//PostConstruct表示实例化之后就会执行@PostConstructpublic void initRouteConfigListener() throws NacosException {//获取配置,服务启动的时候就要加载一次配置String config = nacosConfigManager.getConfigService().getConfigAndSignListener(dataId, group, timeout, new Listener() {@Overridepublic Executor getExecutor() {return null;}@Overridepublic void receiveConfigInfo(String config) {//配置变化时,重新加载配置updateConfig(config);}});//服务启动,加载配置updateConfig(config);}private void updateConfig(String config) {log.info("更新路由信息");//读取nacos中yaml配置Dict dict = YamlUtil.load(new BufferedReader(new InputStreamReader(new ByteArrayInputStream(config.getBytes()))));//转换成Route对象List<RouteDefinition> routeDefinitions = JSONObject.parseObject(JSONUtil.toJsonStr(dict)).getJSONObject("spring").getJSONObject("cloud").getJSONObject("gateway").getJSONArray("routes").toList(RouteDefinition.class);//清理旧的路由规则routeIds.forEach(routeId -> writer.delete(Mono.just(routeId)).subscribe());//清空缓存routeIds.clear();// 遍历添加路由routeDefinitions.forEach(definition -> {//提交路由信息writer.save(Mono.just(definition)).subscribe();//缓存下记录,这样便于清除路由配置routeIds.add(definition.getId());});this.publisher.publishEvent(new RefreshRoutesEvent(this));}@Overridepublic void setApplicationEventPublisher(ApplicationEventPublisher applicationEventPublisher) {this.publisher = applicationEventPublisher;}
}

DynamicRouterLoader在服务启动的时候就会实例化并且注入spring中,所以我们给方法initRouteConfigListener添加注解@PostConstruct,这样实例化一完成就会执行该方法。方法里使用NacosConfigManager获取到nacos的数据并做一次更新路由,这次更新是用来启动服务时初始化路由配置,再添加一个监听器,每次nacos修改都会触发这个监听器再次更新路由配置。更新路由其实有两个思路,思路1是能准确判断每次修改、删除的路由信息,然后分别执行update和delete,这就需要我们去做新旧版本配置的区分工作了,工作量比较大,思路2就是每次都清空所有路由,然后把最新版配置信息加载进去。思路2更简单一点,不过我没有测试过路由信息特别大的情况下会不会导致延迟,通常我们的微服务不至于多到会产生延迟,但是这里依旧要警惕,工作中如果发现数量真的很大了,就要多做一些压力测试。

4、重启服务,做测试:http://127.0.0.1:8888/nacos-client-demo/api/login?userid=123&username=zhangsan

这时候是正常的:

5、修改一下nacos上路由信息,把nacos-client-demo的路由直接删掉,保存服务一直正常运行,观察是否生效。

如果debug能发现监听器被调用了,日志也会打印,再调用接口就已经404了。


http://www.ppmy.cn/server/172555.html

相关文章

中科大计算机网络笔记第一章1.8 互联网历史笔记

计算机网络与互联网历史概述 一、早期通信方式 1. 线路交换 ‌时间‌&#xff1a;1960年之前‌特点‌&#xff1a; 线路建立时间长&#xff0c;代价高。线路资源独享&#xff0c;不适合突发性强的计算机通信。可靠性不高&#xff0c;核心节点损毁影响大。 二、分组交换理论的…

物理竞赛中的线性代数

线性代数 1 行列式 1.1 n n n 阶行列式 定义 1.1.1&#xff1a;称以下的式子为一个 n n n 阶行列式&#xff1a; ∣ A ∣ ∣ a 11 a 12 ⋯ a 1 n a 21 a 22 ⋯ a 2 n ⋮ ⋮ ⋱ ⋮ a n 1 a n 2 ⋯ a n n ∣ \begin{vmatrix}\mathbf A\end{vmatrix} \begin{vmatrix} a_{11…

翻译: 深入分析LLMs like ChatGPT 二

监督微调&#xff08;SFT&#xff09; 使用人工标注的对话数据集&#xff08;如1M条"用户-助手"对话&#xff09;继续训练模型。 标注员遵循指导原则编写理想回答&#xff0c;使模型学习助手的回应风格。 示例对话格式&#xff1a; [系统] 你是一个有帮助的AI助手……

Apache Kafka单节点极速部署指南:10分钟搭建开发单节点环境

Apache Kafka单节点极速部署指南&#xff1a;10分钟搭建开发单节点环境 Kafka简介&#xff1a; Apache Kafka是由LinkedIn开发并捐赠给Apache基金会的分布式流处理平台&#xff0c;现已成为实时数据管道和流应用领域的行业标准。它基于高吞吐、低延迟的设计理念&#xff0c;能够…

uploadlabs经验总结

目录 一、基础上传漏洞&#xff08;太过简单目前环境不可能存在&#xff09; 1、抓包然后改后缀进行绕过 2、抓包然后改上传文件类型进行绕过 3、改后缀大小写绕过&#xff0c;以及收尾加空格&#xff0c;加::$DATA,加点等等 4、黑名单不完整绕过&#xff0c;复习后缀绕过&…

【网络安全】——二进制协议 vs 文本协议:从原理到实战的深度解析

目录 引言 一、协议的本质与分类 二、二进制协议详解 1. 核心特点 2. 典型结构示例 3. 常见应用场景 4. 详细介绍 三、文本协议详解 1. 核心特点 2. 典型结构示例 3. 常见应用场景 4.详细介绍 四、关键对比&#xff1a;二进制协议 vs 文本协议 五、实战案例&…

05 HarmonyOS NEXT高效编程秘籍:Arkts函数调用与声明优化深度解析

温馨提示&#xff1a;本篇博客的详细代码已发布到 git : https://gitcode.com/nutpi/HarmonyosNext 可以下载运行哦&#xff01; 目录 概述函数调用优化声明参数要和实际的参数一致反例正例 函数内部变量尽量使用参数传递反例正例 函数与类声明优化避免动态声明function与class…

费曼物理学讲义-对世界的认知以及方法

《费曼物理学讲义》&#xff08;The Feynman Lectures on Physics&#xff09;是理查德费曼&#xff08;Richard Feynman&#xff09;在加州理工学院为本科生讲授物理学的经典课程整理而成的著作。这套讲义以深刻的物理直觉、生动的语言和独特的视角著称&#xff0c;至今仍是物…