首页 > 其他分享 >初始化安装后 Nacos 动态路由配置不生效

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

时间:2023-03-01 18:00:48浏览次数:67  
标签:初始化 Nacos springframework nacos 路由 import org cloud

一、问题描述

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依赖

		<!--gateway-->
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-gateway</artifactId>
        </dependency>

        <!--SpringCloud ailibaba nacos -->
        <dependency>
            <groupId>com.alibaba.cloud</groupId>
            <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>
        </dependency>

        <!-- SpringCloud Ailibaba Nacos Config -->
        <dependency>
            <groupId>com.alibaba.cloud</groupId>
            <artifactId>spring-cloud-starter-alibaba-nacos-config</artifactId>
        </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<String> ROUTE_LIST = new ArrayList<>();

    // 增加网关启动时,更新一次配置
    @PostConstruct
    public void init() {
        log.info("gateway route init...");
        try {
            List<RouteDefinition> 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<RouteDefinition> 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<RouteDefinition> 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 后解决!

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

标签:初始化,Nacos,springframework,nacos,路由,import,org,cloud
From: https://www.cnblogs.com/cherf/p/17169197.html

相关文章

  • 记一次CPU占用持续上升问题排查(Nacos动态路由引起)
    1、问题描述在整个项目安装完成测试后一个周的时间里,IC服务(我们自己的服务名称)从刚开始CPU占用百分之一涨到了百分之六十左右,而且在持续上升,只有重启IC服务才会降下去,但是......
  • 关于React-Router6 (React 路由)
    一、概要(1)每个单页应用其实是一系列的JS文件,当用户请求网站时,网站返回一整个(或一系列)的js文件和HTML,而当用户在某个页面内点击时,你需要告诉浏览器怎么加载另一个......
  • 思科交换机与路由器的小型局域网互通配置
    此次实验通过配置一个小型局域网来梳理细节实验拓扑: 实验目标:1.实现两个网段互通2.配置交换机和路由器的用户密码和特权密码(都为123)3.PC通过telnet和ssh实现......
  • springcloud之nacos(一)安装
    一、首先下载nacos:https://github.com/alibaba/nacos/releases二、打开修改conf文件夹下的配置文件新建一个数据库nacos_dev,执行nacos-mysql.sql  执行完后,可以看......
  • 关于在接入交换机上配置静态路由
    近期,在一台二层的接入交换机上看到了一条缺省的静态路由:iproute0.0.0.00.0.0.0192.168.1.1已知,该接入交换机有一个管理接口,管理IP为192.168.1.10/24经实验发现,这条缺省......
  • vue3 路由
      vue2在2023年12月份开始停止维护,说实话,很苦逼,很多人根本还停留在vue2思想的时代,包括我自己 。之前一直在写vue的服务端渲染,好长时间已经忘了vue-router创......
  • ubuntu iptables 做为路由转发
    实现功能,本地服务器的号段的192.168.8.0/24,而做为路由器的机器有2个ip,192.168.8.x和另一个ip,而另一个ip可以访问 192.168.2.0/24号段,为了让其它192.168.8.0/24上服务器......
  • Pig nacos 不支持 moxm/java:1.8-full 缺少 so 问题
    使用如下Dockerfile构建镜像#catDockerfileFROMmoxm/java:1.8-fullasbuilderWORKDIR/buildARGJAR_FILE=target/pig-register.jarCOPY${JAR_FILE}app.jarRU......
  • 以太网中静态路由不能只以出接口作为下一跳
    2023.02.28--11.14以太网中静态路由不能只以出接口作为下一跳 拓扑一:  查看华为交换机mac地址表:displaymac-address;查看华为路由器arp表:displayarp清空华为......
  • 从Linux Bridge引发的网桥、交换机与路由器区别探究
    背景最近接触docker的网络配置方式,发现其默认会创建一个docker0的LinuxBridge,宿主机上运行的容器可以通过连接该birdge实现与外网的通信,根据bridge这个命名很自然的认为......