• 初始化安装后 Nacos 动态路由配置不生效


    一、问题描述

    1、每次初始化安装整套项目,包括安装 Nacos 和其他服务还有mysql,redis等其他中间件,安装后 Nacos 获取不到 nacos 路由信息(包括后续新写入动态路由配置)!只有手动重启 Nacos 服务后,才会生效,后续更新的动态路由配置也会正常;

    二、版本

    Nacos: 2.1.0

    spring-boot:2.6.14

    spring-cloud:2021.0.1

    spring-cloud-alibaba:2021.0.1.0

    三、动态路由实现代码(只是其中一种方式,也可用其他)

    1、需要引入的pom依赖

    		
            <dependency>
                <groupId>org.springframework.cloudgroupId>
                <artifactId>spring-cloud-starter-gatewayartifactId>
            dependency>
    
            
            <dependency>
                <groupId>com.alibaba.cloudgroupId>
                <artifactId>spring-cloud-starter-alibaba-nacos-discoveryartifactId>
            dependency>
    
            
            <dependency>
                <groupId>com.alibaba.cloudgroupId>
                <artifactId>spring-cloud-starter-alibaba-nacos-configartifactId>
            dependency>
    

    2、动态刷新路由service

    package com.cherf.flow.gateway.nacosconfig;
    
    import java.util.ArrayList;
    import java.util.List;
    import java.util.Properties;
    import java.util.concurrent.Executor;
    import javax.annotation.PostConstruct;
    
    import com.alibaba.nacos.api.PropertyKeyConst;
    import com.cherf.flow.gateway.config.RoutesConfig;
    import org.apache.commons.lang3.StringUtils;
    import org.slf4j.Logger;
    import org.slf4j.LoggerFactory;
    import org.springframework.beans.factory.annotation.Autowired;
    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 com.alibaba.fastjson.JSON;
    import com.alibaba.fastjson.JSONObject;
    import com.alibaba.nacos.api.NacosFactory;
    import com.alibaba.nacos.api.config.ConfigService;
    import com.alibaba.nacos.api.config.listener.Listener;
    import com.alibaba.nacos.api.exception.NacosException;
    import reactor.core.publisher.Mono;
    
    @Component
    public class NacosDynamicRouteService implements ApplicationEventPublisherAware {
    
        private static final Logger log = LoggerFactory.getLogger(NacosDynamicRouteService.class);
    
    
        public static final long DEFAULT_TIMEOUT = 30000;
    
        @Value("${spring.cloud.nacos.discovery.server-addr}")
        private String serverAddr;
    
        /**
         * 配置 ID
         */
        @Value("${flowGateway.dataId}")
        private String dataId;
    
        /**
         * 配置 分组
         */
        @Value("${flowGateway.group}")
        private String group;
    
        /**
         * 静态路由地址
         */
        @Autowired
        private RoutesConfig routesConfig;
    
        @Autowired
        private RouteDefinitionWriter routeDefinitionWriter;
    
        private ApplicationEventPublisher applicationEventPublisher;
    
        private static final List ROUTE_LIST = new ArrayList<>();
    
        // 增加网关启动时,更新一次配置
        @PostConstruct
        public void init() {
            log.info("gateway route init...");
            try {
                List definitionList = new ArrayList<>();
                definitionList.addAll(routesConfig.getRoutes());
                Properties properties = new Properties();
                // nacos服务器地址,127.0.0.1:8848
                properties.put(PropertyKeyConst.SERVER_ADDR, serverAddr);
                ConfigService configService = NacosFactory.createConfigService(properties);
                if (configService != null) {
                    String configInfo = configService.getConfig(dataId, group, DEFAULT_TIMEOUT);
                    if (StringUtils.isNotBlank(configInfo)) {
                        definitionList.addAll(JSON.parseArray(configInfo, RouteDefinition.class));
                    }
                }
                log.info("update route : {}", definitionList);
                for (RouteDefinition definition : definitionList) {
                    addRoute(definition);
                }
            } catch (Exception e) {
                log.error("初始化网关路由时发生错误", e);
            }
        }
    
        @PostConstruct
        public void dynamicRouteByNacosListener() {
            try {
                Properties properties = new Properties();
                // nacos服务器地址,127.0.0.1:8848
                properties.put(PropertyKeyConst.SERVER_ADDR, serverAddr);
                ConfigService configService = NacosFactory.createConfigService(properties);
                if (configService != null) {
                    configService.getConfig(dataId, group, 5000);
                    configService.addListener(dataId, group, new Listener() {
                        @Override
                        public void receiveConfigInfo(String configInfo) {
                            if (StringUtils.isNotBlank(configInfo)) {
                                // 手动新配置以后,先清除原来的配置
                                clearRoute();
                                try {
                                    List gatewayRouteDefinitions = JSONObject.parseArray(configInfo, RouteDefinition.class);
                                    gatewayRouteDefinitions.addAll(routesConfig.getRoutes());
                                    for (RouteDefinition routeDefinition : gatewayRouteDefinitions) {
                                        addRoute(routeDefinition);
                                    }
                                    publish();
                                } catch (Exception e) {
                                    log.error("加载网关路由时发生错误", e);
                                }
                            }
                        }
    
                        @Override
                        public Executor getExecutor() {
                            return null;
                        }
                    });
                }
            } catch (NacosException e) {
                log.error("加载网关路由时发生错误", e);
            }
        }
    
    
        private void clearRoute() {
            for (String id : ROUTE_LIST) {
                this.routeDefinitionWriter.delete(Mono.just(id)).subscribe();
            }
            ROUTE_LIST.clear();
        }
    
        private void addRoute(RouteDefinition definition) {
            try {
                routeDefinitionWriter.save(Mono.just(definition)).subscribe();
                ROUTE_LIST.add(definition.getId());
            } catch (Exception e) {
                log.error("初始化网关路由时发生错误", e);
            }
        }
    
        private void publish() {
            this.applicationEventPublisher.publishEvent(new RefreshRoutesEvent(this.routeDefinitionWriter));
        }
    
        @Override
        public void setApplicationEventPublisher(ApplicationEventPublisher applicationEventPublisher) {
            this.applicationEventPublisher = applicationEventPublisher;
        }
    }
    

    3、配置文件中的路由信息配置

    package com.cherf.flow.gateway.config;
    
    
    import lombok.Data;
    import org.springframework.boot.context.properties.ConfigurationProperties;
    import org.springframework.cloud.gateway.route.RouteDefinition;
    import org.springframework.stereotype.Component;
    import java.util.List;
    
    /**
     * @author
     */
    @ConfigurationProperties(prefix = "spring.cloud.gateway")
    @Data
    @Component
    public class RoutesConfig {
    
        /**
         * 所有的路由信息
         */
        private List routes;
    }
    
    

    4、yml配置信息

    spring:
      cloud:
        nacos:
          discovery:
            # 不使用nacos的配置
            # enabled: false
            server-addr: NACOS_HOST
          config:
            # 配置中心地址
            server-addr: NACOS_HOST
            # 配置文件格式
            file-extension: yml
            username:
            password:
        gateway:
          discovery:
            locator:
              enabled: true #开启从注册中心动态创建路由的功能,利用微服务名进行路由
          routes:
            - id: i-console-api #payment_route    #路由的ID,没有固定规则但要求唯一,建议配合服务名
              uri: lb://i-console-api #匹配后提供服务的路由地址,lb为负载均衡的其中一种模式
              predicates:
                # 断言,路径相匹配的进行路由
                - Path= /api/**,/oapi/**
              filters: #redis令牌桶限流功能
                - name: RequestRateLimiter
                  args:
                    # 令牌桶每秒填充平均速率
                    redis-rate-limiter.replenishRate: 100
                    # 令牌桶的总容量
                    redis-rate-limiter.burstCapacity: 100
    
            - id: i-system-api #payment_route    #路由的ID,没有固定规则但要求唯一,建议配合服务名
              uri: lb://i-system-api #匹配后提供服务的路由地址,lb为负载均衡的其中一种模式
              predicates:
                # 断言,路径相匹配的进行路由
                - Path= /sys/**,/user/**
              filters: #redis令牌桶限流功能
                - name: RequestRateLimiter
                  args:
                    # 令牌桶每秒填充平均速率
                    redis-rate-limiter.replenishRate: 100
                    # 令牌桶的总容量
                    redis-rate-limiter.burstCapacity: 100
    
    
    flowGateway:
      dataId: gateway-flow
      group: DEFAULT_GROUP
    

    5、Nacos页面

    新增配置
    nacos
    路由信息
    路由信息

    6、启动类修改

    ** 在启动类中需要添加注解@EeableDiscoveryClient**

    package com.cherf;
    
    import org.springframework.boot.SpringApplication;
    import org.springframework.boot.autoconfigure.SpringBootApplication;
    import org.springframework.boot.autoconfigure.security.servlet.SecurityAutoConfiguration;
    import org.springframework.cloud.client.discovery.EnableDiscoveryClient;
    
    @SpringBootApplication(exclude = {SecurityAutoConfiguration.class})
    @EnableDiscoveryClient
    public class FlowGatewayApplication {
        public static void main(String[] args) {
            SpringApplication.run(FlowGatewayApplication.class, args);
        }
    }
    
    

    PS:参考大佬实现代码,实现代码网上很多可自行百度!

    四、可能的原因

    配置文件不生效的把名字改为bootstrap.yml
    仔细看 Data Id 和 Group 配置;
    yml 中配置是否开启了;
    这个配置也需要开启
    还有很多可能的原因可以看官方issue去解决
    这些都不符合我的情况!

    五、定位与解决

    后续发现是安装后只有第一次自动更新配置有问题,重启Nacos后消失,不是一直有问题,所以就从nacos刷新配置的代码下手,最终定位问题出在时区上
    使用 show VARIABLES like '%time_zone%'; 查看数据库时区;
    数据库所在时区

    原因

    /nacos/conf/ 目录下 application.properties 中数据源 url 的时区为 UTC 和数据库不一致导致
    nacos中application.properties

    解决

    将 nacos 配置文件中的 serverTimezone=UTC 改成 serverTimezone=Asia/Shanghai 后解决!

    解决了我的问题,但是也可能不适用于其他场景!欢迎大家讨论!


    __EOF__

  • 本文作者: cherf
  • 本文链接: https://www.cnblogs.com/cherf/p/17169197.html
  • 关于博主: 评论和私信会在第一时间回复。或者直接私信我。
  • 版权声明: 本博客所有文章除特别声明外,均采用 BY-NC-SA 许可协议。转载请注明出处!
  • 声援博主: 如果您觉得文章对您有帮助,可以点击文章右下角推荐一下。
  • 相关阅读:
    Windows/Ubuntu安装frida和objection
    windows电脑彻底删除文件怎么恢复?可尝试这2种恢复办法!
    Python【多分支选择结构】
    池州控股集团财务共享项目启动啦!
    自定义picker滚动选择器样式
    快速掌握MyBatis
    号称史上最难618,淘宝数据盘点你做对了吗?
    【TypeScript】TypeScript 与 JavaScript 有何不同?
    SQLServer添加Oracle链接服务器
    C++23:多维视图(std::mdspan)
  • 原文地址:https://www.cnblogs.com/cherf/p/17169197.html