• spring-gateway+nacos+schedule实现动态路由


    1. 首先声明: 

         本文不会去讲gateway的基础知识。

    2.  开发背景:

         spring-gateway路由配置通常有两种,一种通过yml文件中、另外一种是通过代码组件RouteDefinition的方式。这两种方式都是将路由写死,而我们的真实场景,网管是不允许重启的,网关部署后,需要动态感知业务服务的增删。因此在这一场景下,需要开发一种gateway动态感知服务的变化完成内存中路由的刷新。

    3.  方式1: 将配置写在nacos上,gateway监控nacos上的配置完成。源码见下:

    @Component
    public class NacosDynamicRouteService implements ApplicationEventPublisherAware {
     
        private String dataId = "gateway-router";
     
        private String group = "DEFAULT_GROUP";
     
        @Value("${spring.cloud.nacos.config.server-addr}")
        private String serverAddr;
     
        @Autowired
        private RouteDefinitionWriter routeDefinitionWriter;
     
        private ApplicationEventPublisher applicationEventPublisher;
     
        private static final List<String> ROUTE_LIST = new ArrayList<>();
     
        @PostConstruct
        public void dynamicRouteByNacosListener() {
            try {
                ConfigService configService = NacosFactory.createConfigService(serverAddr);
                configService.getConfig(dataId, group, 5000);
                configService.addListener(dataId, group, new Listener() {
                    @Override
                    public void receiveConfigInfo(String configInfo) {
                        clearRoute();
                        try {
                            List<RouteDefinition> gatewayRouteDefinitions = JSONObject.parseArray(configInfo, RouteDefinition.class);
                            for (RouteDefinition routeDefinition : gatewayRouteDefinitions) {
                                addRoute(routeDefinition);
                            }
                            publish();
                        } catch (Exception e) {
                            e.printStackTrace();
                        }
                    }
     
                    @Override
                    public Executor getExecutor() {
                        return null;
                    }
                });
            } catch (NacosException e) {
                e.printStackTrace();
            }
        }
     
        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) {
                e.printStackTrace();
            }
        }
     
        private void publish() {
            this.applicationEventPublisher.publishEvent(new RefreshRoutesEvent(this.routeDefinitionWriter));
        }
     
        @Override
        public void setApplicationEventPublisher(ApplicationEventPublisher applicationEventPublisher) {
            this.applicationEventPublisher = applicationEventPublisher;
        }
    }

    因此,此方式需要我们每次上线新的业务服务后,需要专业人员去操作gateway-router这一配置文件,按照一定的格式,去填写业务服务的路由及对应的predicate,filter等。格式例子如下:

    [{
        "id": "account-router",
        "order": 0,
        "predicates": [{
            "args": {
                "pattern": "/acc/**"
            },
            "name": "Path"
        }],
        "uri": "lb://account-service"
    },{
        "id": "payment-router",
        "order": 2,
        "predicates": [{
            "args": {
                "pattern": "/pay/**"
            },
            "name": "Path"
        }],
        "uri": "lb://payment-service"
    }]

    4. 方式 2:通过定时任务,全自动化的去读取注册到nacos上的服务列表进行注册。(代码待更新)

        我们需要编写带有Schedule的方法,此方法通过nacos暴露的api接口完成所有业务服务名称列表的获取,完成动态的完成路由的刷新。

    总结: 以上两种方式各有利弊,然而都不是实时的方式,第一种方式需要在config修改后才能生效,第二种方式需要在cron表达式设定的时间后才能刷新,如果设置短cpu压力过大,如果设置长,业务服务的路由又感知很慢,因此cron表达式也需要衡量。

    新的方式待研究。

  • 相关阅读:
    构建之法阅读笔记06
    钢镚儿开发前会议
    构建之法阅读笔记05
    4.11第8周学习总结
    人月神话阅读笔记01
    构建之法阅读笔记04
    4.4日学习总结
    构建之法阅读笔记03
    3.28第六周学习内容总结
    二人团队项目增删改查
  • 原文地址:https://www.cnblogs.com/dhName/p/15496194.html
Copyright © 2020-2023  润新知