zuul整合nacos_Spring Cloud Zuul的动态路由怎样做?集成Nacos实现很简单-阿里云开发者社区...

Spring Cloud Zuul的动态路由怎样做?集成Nacos实现很简单

一、说明

网关的核心概念就是路由配置和路由规则,而作为所有请求流量的入口,在实际生产环境中为了保证高可靠和高可用,是尽量要避免重启的,所以实现动态路由是非常有必要的;本文主要介绍实现的思路,并且以Nacos为数据源来讲解

二、实现要点

要实现动态路由只需关注下面4个点

网关启动时,动态路由的数据怎样加载进来

静态路由与动态路由以那个为准,ps:静态路由指的是配置文件里写死的路由配置

监听动态路由的数据源变化

数据有变化时怎样通知zuul刷新路由

三、具体实现

3.1. 实现动态路由的数据加载

重写SimpleRouteLocator类的locateRoutes方法,此方法是加载路由配置的,父类中是获取properties中的路由配置,可以通过扩展此方法,达到动态获取配置的目的

这里采用静态路由与动态路由共存,相同路由id以动态路由优先覆盖的实现方式

AbstractDynRouteLocator类可查看:AbstractDynRouteLocator.java

public abstract class AbstractDynRouteLocator extends SimpleRouteLocator implements RefreshableRouteLocator {

private ZuulProperties properties;

public AbstractDynRouteLocator(String servletPath, ZuulProperties properties) {

super(servletPath, properties);

this.properties = properties;

}

/**

* 刷新路由

*/

@Override

public void refresh() {

doRefresh();

}

@Override

protected Map locateRoutes() {

LinkedHashMap routesMap = new LinkedHashMap<>();

// 从application.properties中加载静态路由信息

routesMap.putAll(super.locateRoutes());

// 从数据源中加载动态路由信息

routesMap.putAll(loadDynamicRoute());

// 优化一下配置

LinkedHashMap values = new LinkedHashMap<>();

for (Map.Entry entry : routesMap.entrySet()) {

String path = entry.getKey();

// Prepend with slash if not already present.

if (!path.startsWith("/")) {

path = "/" + path;

}

if (StringUtils.hasText(this.properties.getPrefix())) {

path = this.properties.getPrefix() + path;

if (!path.startsWith("/")) {

path = "/" + path;

}

}

values.put(path, entry.getValue());

}

return values;

}

/**

* 加载路由配置,由子类去实现

*/

public abstract Map loadDynamicRoute();

}

由于动态路由的数据可以有很多种途径,如:Nacos、Redis、Zookeeper、DB等,所以这里定义一个抽象类,由具体的实现类去定义loadDynamicRoute方法

3.2. Nacos路由实现类

NacosDynRouteLocator类完整的代码实现可查看:NacosDynRouteLocator.java

3.2.1. 实现loadDynamicRoute方法获取动态数据

@Override

public Map loadDynamicRoute() {

Map routes = new LinkedHashMap<>();

if (zuulRouteEntities == null) {

zuulRouteEntities = getNacosConfig();

}

for (ZuulRouteEntity result : zuulRouteEntities) {

if (StrUtil.isBlank(result.getPath()) || !result.isEnabled()) {

continue;

}

ZuulRoute zuulRoute = new ZuulRoute();

BeanUtil.copyProperties(result, zuulRoute);

routes.put(zuulRoute.getPath(), zuulRoute);

}

return routes;

}

private List getNacosConfig() {

try {

String content = nacosConfigProperties.configServiceInstance().getConfig(ZUUL_DATA_ID, ZUUL_GROUP_ID,5000);

return getListByStr(content);

} catch (NacosException e) {

log.error("listenerNacos-error", e);

}

return new ArrayList<>(0);

}

3.2.2. 增加NacosListener监听路由数据变化

private void addListener() {

try {

nacosConfigProperties.configServiceInstance().addListener(ZUUL_DATA_ID, ZUUL_GROUP_ID, new Listener() {

@Override

public Executor getExecutor() {

return null;

}

@Override

public void receiveConfigInfo(String configInfo) {

//赋值路由信息

locator.setZuulRouteEntities(getListByStr(configInfo));

RoutesRefreshedEvent routesRefreshedEvent = new RoutesRefreshedEvent(locator);

publisher.publishEvent(routesRefreshedEvent);

}

});

} catch (NacosException e) {

log.error("nacos-addListener-error", e);

}

}

注意路由数据变化后不需要自己手动刷新路由,只需要给zuul发送一个RoutesRefreshedEvent事件即可,zuul自己有个ZuulRefreshListener类会监听事件帮我们刷新路由

3.3. 配置类创建NacosDynRouteLocator的Bean

DynamicZuulRouteConfig可查看:NacosDynRouteLocator.java

@Configuration

@ConditionalOnProperty(prefix = "zlt.gateway.dynamicRoute", name = "enabled", havingValue = "true")

public class DynamicZuulRouteConfig {

@Autowired

private ZuulProperties zuulProperties;

@Autowired

private DispatcherServletPath dispatcherServletPath;

/**

* Nacos实现方式

*/

@Configuration

@ConditionalOnProperty(prefix = "zlt.gateway.dynamicRoute", name = "dataType", havingValue = "nacos", matchIfMissing = true)

public class NacosZuulRoute {

@Autowired

private NacosConfigProperties nacosConfigProperties;

@Autowired

private ApplicationEventPublisher publisher;

@Bean

public NacosDynRouteLocator nacosDynRouteLocator() {

return new NacosDynRouteLocator(nacosConfigProperties, publisher, dispatcherServletPath.getPrefix(), zuulProperties);

}

}

}

这里通过自定义配置来控制是否开启动态路由功能

3.4. 添加Nacos路由配置

file

新增配置项:

Data Id:zuul-routes

Group:ZUUL_GATEWAY

配置内容:

[

{

"enabled":true,

"id":"csdn",

"path":"/csdn/**",

"retryable":false,

"stripPrefix":true,

"url":"https://www.csdn.net/"

}, {

"enabled":true,

"id":"github",

"path":"/github/**",

"retryable":false,

"stripPrefix":true,

"url":"http://github.com/"

}

]

添加两条路由数据

四、测试

启动网关通过/actuator/routes端点查看当前路由信息

file

可以看到静态路由和Nacos里配置的两条路由信息并存显示

修改Nacos配置,关闭csdn路由

file

刷新查看网关的路由信息

file

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要在Zuul中使用Nacos,需要进行以下步骤: 1. 引入NacosZuul的依赖 在pom.xml文件中添加以下依赖: ```xml <dependency> <groupId>com.alibaba.cloud</groupId> <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId> </dependency> <dependency> <groupId>org.springframework.cloud</groupId> <artifactId>spring-cloud-starter-netflix-zuul</artifactId> </dependency> ``` 2. 配置Nacos注册中心 在application.yml文件中添加以下配置: ```yaml spring: cloud: nacos: discovery: server-addr: ${NACOS_SERVER_ADDR:localhost:8848} # Nacos服务端地址 # 下面是注册中心元数据配置,可以根据实际情况进行配置 metadata: management: context-path: ${server.context-path:/} version: ${project.version} group: ${project.groupId} # Zuul相关配置 application: name: zuul-gateway profiles: active: dev # 激活的profile zuul: routes: user-service: path: /user-service/** serviceId: user-service ``` 3. 启用Zuul 在启动类上添加@EnableZuulProxy注解,启用Zuul代理功能,代码如下: ```java @SpringBootApplication @EnableDiscoveryClient @EnableZuulProxy public class GatewayApplication { public static void main(String[] args) { SpringApplication.run(GatewayApplication.class, args); } } ``` 4. 测试 启动应用程序后,可以在浏览器中访问 http://localhost:port/user-service/ 来测试路由是否正常。如果user-service服务已经注册到Nacos中,Zuul将会代理所有请求并将其转发到user-service服务。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值