springCloud之Gateway动态路由

        学习gateway网关时,是以产品应用为目的,打算做一个类似于SAAS平台,网关负责统一的鉴权,日志记录,对外屏蔽真实的访问地址。路由信息也不能是写死在配置文件的,必须是提供管理页面可维护的。所以就略过配置文件,直接开启动态路由的实现。

一、gateway动态路由需要的jar包

我的springboot及springCloud版本

<!-- SpringBoot的依赖配置-->
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-dependencies</artifactId>
                <version>2.6.11</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
<!-- SpringCloud的依赖配置-->
            <dependency>
                <groupId>org.springframework.cloud</groupId>
                <artifactId>spring-cloud-dependencies</artifactId>
                <version>2021.0.1</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>

依赖包

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

<!-- 负载均衡,路由的真实地址可以是一个负载均衡地址 -->
<dependency>
   <groupId>org.springframework.cloud</groupId>
   <artifactId>spring-cloud-starter-loadbalancer</artifactId>
 </dependency>

<!-- 服务注册与发现 -->
<dependency>
   <groupId>com.alibaba.cloud</groupId>
   <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>
</dependency>

<!-- 配置中心 -->
<dependency>
   <groupId>com.alibaba.cloud</groupId>
   <artifactId>spring-cloud-starter-alibaba-nacos-config</artifactId>
</dependency>
二、修改application.yml文件

开启服务与发现,路由的真实地址可设置为“lb:{应用名称}”格式的地址,可实现通过fegin进行负载均衡地址访问

spring:
  cloud:
    nacos:
      discovery: ##配置服务与发现
        server-addr: 127.0.0.1:8848
        namespace: d5afac56-78a0-48e5-ac76-c6e13c96f35f
    gateway:
      discovery:
        locator:
          ##指定是否启用服务发现定位器。当设置为true时,Gateway将通过服务发现来定位后端服务
          enabled: true
三、动态路由的代码实现

路由信息的持久化此处不做解释,无非就是将json格式的路由数据保存到数据库,以下代码有将JSON格式的路由数据转换为路由对象的方法。以下代码为维护路由信息的业务层,包含路由信息的增、删、改、刷新等方法。

package com.zhangzz.gateway.route.service.impl;

import com.alibaba.fastjson2.JSON;
import com.zhangzz.gateway.domain.CustomPredicateDefinition;
import com.zhangzz.gateway.domain.CustomRouteDefinition;
import com.zhangzz.gateway.domain.RouteInfo;
import com.zhangzz.core.entity.AjaxResult;
import com.zhangzz.gateway.route.service.IRouteService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.ApplicationArguments;
import org.springframework.boot.ApplicationRunner;
import org.springframework.cloud.gateway.event.RefreshRoutesEvent;
import org.springframework.cloud.gateway.filter.FilterDefinition;
import org.springframework.cloud.gateway.handler.predicate.PredicateDefinition;
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.Service;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.web.util.UriComponentsBuilder;
import reactor.core.publisher.Mono;

import java.net.URI;
import java.util.ArrayList;
import java.util.List;
import java.util.stream.Collectors;

@Service
@Transactional
public class DynamicRouteService implements ApplicationEventPublisherAware , ApplicationRunner {

    private final RouteDefinitionWriter routeDefinitionWriter;

    //路由信息持久化业务层,无需关注
    @Autowired
    private IRouteService routeService;

    //通过构造方法进行注入,此处通过跟踪代码,RouteDefinitionWriter的实现类是基于内存的,非redis的
    private ApplicationEventPublisher publisher;

    public DynamicRouteService(RouteDefinitionWriter routeDefinitionWriter) {
        this.routeDefinitionWriter = routeDefinitionWriter;
    }

    @Override
    public void setApplicationEventPublisher(ApplicationEventPublisher publisher) {
        this.publisher = publisher;
    }

    /**
     * 增加路由
     *
     * @param routeForm
     * @return
     */
    public AjaxResult add(CustomRouteDefinition routeForm) {
        RouteDefinition definition = convert(routeForm);
        routeDefinitionWriter.save(Mono.just(definition)).subscribe();
        enduranceRule(routeForm.getName(), routeForm.getDescription(), definition);
        publishRouteEvent();
//        System.out.println(JSON.toJSONString(definition));
        return AjaxResult.success(true);
    }

    /**
     * 更新路由
     *
     * @param routeForm
     * @return
     */
    public AjaxResult update(CustomRouteDefinition routeForm) {
        RouteDefinition definition = convert(routeForm);
        try {
            this.routeDefinitionWriter.delete(Mono.just(definition.getId())).subscribe();
        } catch (Exception e) {
            return AjaxResult.error("未知路由信息",500);
        }
        try {
            routeDefinitionWriter.save(Mono.just(definition)).subscribe();
            enduranceRule(routeForm.getName(), routeForm.getDescription(), definition);
            publishRouteEvent();
            return AjaxResult.success(true);
        } catch (Exception e) {
            return AjaxResult.error("路由信息修改失败!",500);
        }
    }

    /**
     * 删除路由
     *
     * @param id
     * @return
     */
    public AjaxResult delete(String id) {
        this.routeDefinitionWriter.delete(Mono.just(id)).subscribe();
        routeService.deleteByRouteId(id);
        publishRouteEvent();
        return AjaxResult.success();
    }

    private void publishRouteEvent() {
        this.publisher.publishEvent(new RefreshRoutesEvent(this));
    }


    /**
     * 刷新路由信息
     **/
    public AjaxResult flushRoute() {
        publishRouteEvent();
        return AjaxResult.success(true);
    }



    /**
     * 获取路由信息列表
     * @return
     */
    public List<CustomRouteDefinition> getRouteList(){

        List<RouteInfo> list = routeService.selectList();

        List<CustomRouteDefinition> routeList = new ArrayList<>();
        if(list != null && !list.isEmpty()){
            for (RouteInfo info:list) {
                routeList.add(this.convertCustomRouteDefinition(info));
            }
        }

        return routeList;

    }

    public CustomRouteDefinition getRouteById(String routeId){
        RouteInfo info = routeService.selectByRouteId(routeId);
        return this.convertCustomRouteDefinition(info);
    }

    /**
     * 转换为自定义路由
     * @param info 路由持久化对象
     * @return 自定义路由
     */
    private CustomRouteDefinition convertCustomRouteDefinition(RouteInfo info){
        CustomRouteDefinition routeDefinition = new CustomRouteDefinition();
        routeDefinition.setDescription(info.getDescription());
        routeDefinition.setFilters(JSON.parseArray(info.getFilters(),FilterDefinition.class));
        routeDefinition.setId(info.getRouteId());
        routeDefinition.setName(info.getRouteName());
        routeDefinition.setOrder(info.getOrderNum());
        routeDefinition.setPredicateDefinitions(JSON.parseArray(info.getPredicates(), CustomPredicateDefinition.class));
        routeDefinition.setUrl(info.getUrl());
        return routeDefinition;
    }

    /**
     * 把自定义请求模型转换为RouteDefinition
     *
     * @param form
     * @return
     */
    private RouteDefinition convert(CustomRouteDefinition form) {
        RouteDefinition definition = new RouteDefinition();
        definition.setId(form.getId());
        definition.setOrder(form.getOrder());
        //设置断言
        List<PredicateDefinition> predicateDefinitions = form.getPredicateDefinitions().stream()
                .distinct().map(predicateInfo -> {
                    PredicateDefinition predicate = new PredicateDefinition();
                    predicate.setArgs(predicateInfo.getArgs());
                    predicate.setName(predicateInfo.getName());
                    return predicate;
                }).collect(Collectors.toList());
        definition.setPredicates(predicateDefinitions);

        if(form.getFilters() != null) {
            // 设置过滤
            List<FilterDefinition> filterList = form.getFilters().stream().distinct().map(x -> {
                FilterDefinition filter = new FilterDefinition();
                filter.setName(x.getName());
                filter.setArgs(x.getArgs());
                return filter;
            }).collect(Collectors.toList());
            definition.setFilters(filterList);
        }
        // 设置URI,判断是否进行负载均衡
        URI uri;
        if (form.getUrl().startsWith("http")) {
            uri = UriComponentsBuilder.fromHttpUrl(form.getUrl()).build().toUri();
        } else {
            uri = URI.create(form.getUrl());
        }
        definition.setUri(uri);
        return definition;
    }


    /**
     * 持久化至数据库
     */
    public void enduranceRule(String name, String description, RouteDefinition definition) {
        String id = definition.getId();
        List<PredicateDefinition> predicates = definition.getPredicates();
        List<FilterDefinition> filters = definition.getFilters();
        int order = definition.getOrder();
        URI uri = definition.getUri();
        RouteInfo routeInfo = new RouteInfo();
        routeInfo.setRouteName(name);
        routeInfo.setRouteId(id);
        routeInfo.setUrl(uri.toString());
        routeInfo.setPredicates(JSON.toJSONString(predicates));
        routeInfo.setFilters(JSON.toJSONString(filters));
        routeInfo.setDescription(description);
        routeInfo.setOrderNum(order);
        RouteInfo one = routeService.selectByRouteId(id);
        if (one == null) {
            routeService.add(routeInfo);
        } else {
            routeInfo.setId(one.getId());
            routeService.update(routeInfo);
        }

    }

    /**
     * 该方法会在网关启动时,从数据库读取路由信息,并加载至内存中。
     */
    @Override
    public void run(ApplicationArguments args) throws Exception {
        System.out.println("----------加载路由信息Start---------");
        List<RouteInfo> list = routeService.selectList();
        if(list != null && !list.isEmpty()){
            CustomRouteDefinition definition = null;
            for (RouteInfo info:list) {
                System.out.println("----------加载路由“"+info.getRouteName()+"”---------");
                definition = convertCustomRouteDefinition(info);
                RouteDefinition routedefinition = convert(definition);
                routeDefinitionWriter.save(Mono.just(routedefinition)).subscribe();
            }
            publishRouteEvent();
        }
        System.out.println("----------加载路由信息End---------");
    }
}

Controller层代码如下:

package com.zhangzz.gateway.route.controller;

import com.zhangzz.gateway.domain.CustomRouteDefinition;
import com.zhangzz.core.entity.AjaxResult;
import com.zhangzz.gateway.route.service.impl.DynamicRouteService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;

@RestController
@RequestMapping("/route")
public class RouteController {


    @Autowired
    private DynamicRouteService service;

    @PostMapping
    public AjaxResult addRoute(@RequestBody CustomRouteDefinition routeDefinition){

        return service.add(routeDefinition);
    }

    @PutMapping
    public AjaxResult updateRoute(@RequestBody CustomRouteDefinition routeDefinition){

        return service.update(routeDefinition);
    }

    @DeleteMapping("/{routeId}")
    public AjaxResult deleteRoute(@PathVariable String routeId){
        return service.delete(routeId);
    }

    /**
     * 刷新路由
     * @return
     */
    @GetMapping("/flushRoute")
    public AjaxResult flushRoute(){
        return service.flushRoute();
    }

    @GetMapping
    public AjaxResult getList(){
        return AjaxResult.success(service.getRouteList());
    }

    @GetMapping("/routeId/{routeId}")
    public AjaxResult getByRouteId(@PathVariable String routeId){
        return AjaxResult.success(service.getRouteById(routeId));
    }
}

四、网关的启动类

@SpringBootApplication
@EnableDiscoveryClient
public class GatewayApplication {

    public static void main(String[] args) {
        SpringApplication.run(GatewayApplication.class,args);
    }
}

启动网关服务后,通过postman或apipost等工具添加路由信息,同时路由信息会保存入库,下次服务启动会重新装载。

请求地址(POST):http://localhost/route

请求报文:

{
    "id":"test1",
    "name":"routeTest1",
    "description":"路由测试1",
    "order":1,
    "predicateDefinitions":[
        {
            "name":"Path",
            "args":{
                "_genkey_0":"/b3/**",
                "_genkey_1":"/b4/**"
            }
        }
    ],
    "filters":[
        {
            "name":"StripPrefix",
            "args":{
                "_genkey_0":"1"
            }
        }
    ],
    "url":"lb://content"

}

浏览器中访问http://localhost/b3或http://localhost/b4,则会直接访问nacos服务中的content服务

  • 2
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
### 回答1: Spring Cloud Gateway动态路由是指在Spring Cloud Gateway网关中,路由规则可以在运行时动态地进行添加、删除、修改等操作。相比于静态路由,动态路由可以根据实际情况进行动态调整,从而更加灵活、方便地进行流量控制和负载均衡。例如,在服务上线、下线或者进行扩容缩容的时候,可以通过修改路由规则,动态地将流量引导到不同的服务实例中,从而实现动态负载均衡和容错能力。 ### 回答2: Spring Cloud Gateway动态路由是指在Spring Cloud Gateway网关中,根据某些条件动态地将请求路由到不同的目标服务实例上。传统的静态路由需要事先配置好路由规则,但是在微服务架构中,服务的实例会动态地增加、减少、更新,因此需要一种能够动态适应变化的路由机制。 Spring Cloud Gateway动态路由的实现需要依赖于服务注册与发现组件,比如Eureka或Consul。当服务实例注册到服务注册中心时,Spring Cloud Gateway会订阅服务注册中心的变化,当有新的服务实例上线或下线时会自动更新路由规则。 动态路由可以根据多种条件进行判断和匹配,如路径、域名、Header、请求参数等。可以根据业务需求动态地配置路由规则,使得请求能够被准确地路由到目标服务实例上。动态路由能够实现动态扩展和负载均衡,提高系统的灵活性和可伸缩性。 Spring Cloud Gateway动态路由的配置通常以YAML或JSON的形式进行,可以通过配置文件、配置中心或接口的方式进行配置。支持多种动态路由的配置方式,如断言(Predicate)、过滤器(Filter)、转发(Forwarding)、重定向(Redirecting)等,可以根据具体需求实现各种功能。 总之,Spring Cloud Gateway动态路由是一种能够根据条件动态路由请求到不同服务实例的机制,具有灵活、可扩展、高效的特点,是构建微服务架构中的网关的重要特性。 ### 回答3: Spring Cloud Gateway动态路由是一种基于Spring Cloud Gateway框架的动态路由功能。传统的静态路由是在网关的配置文件中预先定义好所有的路由规则,而动态路由可以在运行时根据业务需要实时插入、修改和删除路由规则,实现灵活的请求转发和负载均衡。通过动态路由,可以根据不同的路径或者请求头等匹配条件,将请求转发到指定的目标服务,从而实现微服务架构中的请求路由和负载均衡功能。动态路由的配置可以通过网关的API接口或者命令行工具进行管理,使得路由的配置更加灵活和方便。同时,动态路由还支持动态修改和重载路由规则,可以根据实际情况动态调整路由策略,提高系统的可用性和弹性。Spring Cloud Gateway动态路由的实现是基于Spring Framework中的路由器和过滤器的概念,通过使用reactive编程模型处理请求,并且支持使用各种插件来扩展网关的功能,例如服务发现、熔断器、限流等。总之,Spring Cloud Gateway动态路由提供了一种灵活、易用且高性能的路由解决方案,适用于构建微服务架构的API网关。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值