Spring Cloud Gateway -动态路由篇

本文介绍了如何使用Spring Cloud Gateway实现动态路由,包括创建基本的路由模型、动态路由服务Service以及提供相应的API接口。通过这些接口,可以在不重启网关的情况下动态添加、更新和删除路由,提高了服务上线的灵活性。
摘要由CSDN通过智能技术生成

动态路由


​ Gateway配置路由主要有两种方式:一种是yml配置文件,另一种是写在代码里。而无论哪种方式,启动网关后将无法修改路由配置,如有新服务要上线,则需要先把网关下线,修改yml配置后,再重启网关。

​ Gateway提供了Endpoint断点,暴露路由信息,有获取所有路由,刷新路由,查看单个路由,删除路由等方法。要访问断点中的方法需要添加spring-boot-starter-actuator注解,并在配置文件中暴露所有端点。

# 暴露监控断点
management:
  endpoints:
    web:
      exposure:
        include:
        - '*'
      health:
        show-details: always

所需依赖:

	<dependency>
        <groupId>org.springframework.cloud</groupId>
        <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
    </dependency>
  	<dependency>
  		<groupId>org.springframework.cloud</groupId>
  		<artifactId>spring-cloud-starter-gateway</artifactId>
  	</dependency>
  	<dependency>
  		<groupId>org.springframework.boot</groupId>
  		<artifactId>spring-boot-starter-actuator</artifactId>
  	</dependency>

1 创建基本模型

根据路由的结构创建模型(路由模型,断言模型,过滤器模型)

路由模型:

/**
 * 路由模型
 * @author sw
 *
 */
public class GatewayRouteDefinition {

	/**
	 * 路由的ID
	 */
	private String id;
	/**
	 * 路由断言集合配置
	 */
	private List<GatewayPredicateDefinition> predicates = new ArrayList<>();
	/**
	 * 路由过滤器集合配置
	 */
	private List<GatewayFilterDefinition> filters = new ArrayList<>();
	/**
	 * 路由规则转发的目标uri
	 */
	private String uri;
	/**
	 * 路由执行的顺序
	 */
	private int order = 0;
}

断言模型:

/**
 * 路由断言
 * @author sw
 *
 */
public class GatewayPredicateDefinition {
	/**
	 * 断言对应的name
	 */
	private String name;
	/**
	 * 配置的断言规则
	 */
	private Map<String, String> args = new LinkedHashMap<String, String>();
}

过滤器模型:

/**
 * 过滤器模型
 * @author sw
 *
 */
public class GatewayFilterDefinition {
	/**
	 * 过滤器名称
	 */
	private String name;
	/**
	 * 对应的路由规则
	 */
	private Map<String, String> args = new LinkedHashMap<>();
}

2 创建动态路由服务Service

动态路由服务主要提供:增加路由,修改路由,删除路由这三个服务,需要实现ApplicationEventPublisherAware接口。

DynamicRouteServiceImpl:

package com.lmc.service;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.cloud.gateway.event.RefreshRoutesEvent;
import org.springframework.cloud.gateway.route.RouteDefinition;
import org.springframework.cloud.gateway.route.RouteDefinitionWriter;
import org.springframework.cloud.gateway.support.NotFoundException;
import org.springframework.context.ApplicationEventPublisher;
import org.springframework.context.ApplicationEventPublisherAware;
import org.springframework.http.ResponseEntity;
import org.springframework.stereotype.Service;

import reactor.core.publisher.Mono;

/**
 * 动态路由服务
 * @author sw
 *
 */
@Service
public class DynamicRouteServiceImpl implements ApplicationEventPublisherAware {
	
	@Autowired
	private RouteDefinitionWriter routeDefinitionWriter;
	private ApplicationEventPublisher applicationEventPublisher;

	@Override
	public void setApplicationEventPublisher(ApplicationEventPublisher applicationEventPublisher) {
		// TODO Auto-generated method stub
		this.applicationEventPublisher = applicationEventPublisher;
	}
	
	//增加路由
	public String add(RouteDefinition definition) {
		routeDefinitionWriter.save(Mono.just(definition)).subscribe();
		this.applicationEventPublisher.publishEvent(new RefreshRoutesEvent(this));
		return "success";
	}
	
	//更新路由
	public String update(RouteDefinition definition) {
		try {
			delete(definition.getId());
		} catch (Exception e) {
			return "update fail, not find route routeId: " + definition.getId();
		}
		try {
			routeDefinitionWriter.save(Mono.just(definition)).subscribe();
			this.applicationEventPublisher.publishEvent(new RefreshRoutesEvent(this));
			return "success";
		} catch (Exception e) {
			return "update route fail";
		}
		
	}

	//删除路由
	public Mono<ResponseEntity<Object>> delete(String id) {
		return this.routeDefinitionWriter.delete(Mono.just(id)).then(Mono.defer(() -> {
			return Mono.just(ResponseEntity.ok().build());
		})).onErrorResume((t) -> {
			return t instanceof NotFoundException;
		}, (t) -> {
			return Mono.just(ResponseEntity.notFound().build());
		});
		
	}
}

3 提供动态路由服务接口API

提供对外接口,可以通过接口对gateway的路由进行操作

package com.lmc.controller;

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

import org.springframework.beans.factory.annotation.Autowired;
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.http.ResponseEntity;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.util.UriComponentsBuilder;

import com.lmc.definition.GatewayFilterDefinition;
import com.lmc.definition.GatewayPredicateDefinition;
import com.lmc.definition.GatewayRouteDefinition;
import com.lmc.service.DynamicRouteServiceImpl;

import reactor.core.publisher.Mono;

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

	@Autowired
	private DynamicRouteServiceImpl dynamicRouteService;
	
    /**
     * 添加路由
     **/
	@PostMapping("/add")
	public String add(@RequestBody GatewayRouteDefinition gwdefinition) {
		System.out.println("add = " + gwdefinition.toString());
		String flag = "fail";
		try {
			RouteDefinition definition = assembleRouteDefinition(gwdefinition);
			flag = this.dynamicRouteService.add(definition);
		} catch (Exception e) {
			e.printStackTrace();
		}
		return flag;
	}
	
    /**
     * 删除路由
     **/
	@PostMapping("/routes/{id}")
	public Mono<ResponseEntity<Object>> delete(@PathVariable String id) {
		try {
			return this.dynamicRouteService.delete(id);
		} catch (Exception e) {
			e.printStackTrace();
		}
		return null;
	}
	
    /**
     * 修改路由
     **/
	@PostMapping("/update")
	public String update(@RequestBody GatewayRouteDefinition gwdefinition) {
		System.out.println("update = " + gwdefinition.toString());
		RouteDefinition definition = assembleRouteDefinition(gwdefinition);
		return this.dynamicRouteService.update(definition);
	}

	private RouteDefinition assembleRouteDefinition(GatewayRouteDefinition gwdefinition) {
		RouteDefinition definition = new RouteDefinition();
		definition.setId(gwdefinition.getId());
		definition.setOrder(gwdefinition.getOrder());
		//设置断言
		List<PredicateDefinition> pdList = new ArrayList<PredicateDefinition>();
		List<GatewayPredicateDefinition> gatewayPredicateDefinitionList = gwdefinition.getPredicates();
		for (GatewayPredicateDefinition gPDefinition : gatewayPredicateDefinitionList) {
			PredicateDefinition predicate = new PredicateDefinition();
			predicate.setArgs(gPDefinition.getArgs());
			predicate.setName(gPDefinition.getName());
			pdList.add(predicate);
		}
		definition.setPredicates(pdList);
		
		//设置过滤器
		List<FilterDefinition> fdList = new ArrayList<>();
		List<GatewayFilterDefinition> gatewayFilterDefinitionList = gwdefinition.getFilters();
		for (GatewayFilterDefinition gFDefinition : gatewayFilterDefinitionList) {
			FilterDefinition filter = new FilterDefinition();
			filter.setArgs(gFDefinition.getArgs());
			filter.setName(gFDefinition.getName());
			fdList.add(filter);
		}
		definition.setFilters(fdList);
		System.out.println("gatewayFilterDefinitionList = " + gatewayFilterDefinitionList.toString());
		System.out.println("fdList = " + fdList.toString());
		URI uri = null;
		if (gwdefinition.getUri().startsWith("http")) {
			uri = UriComponentsBuilder.fromHttpUrl(gwdefinition.getUri()).build().toUri();
		} else {
			uri = URI.create(gwdefinition.getUri());
		}
		definition.setUri(uri);
		return definition;
	}
}

以上配置完成发布服务之后,以后添加新服务就可以通过接口来动态操作路由。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值