Spring Cloud Config + Spring Cloud Bus + RabbitMq实现微服务高可用配置中心

32 篇文章 0 订阅
21 篇文章 0 订阅

        在一个微服务架构中会有很多微服务,每个微服务都有自己的配置文件,这会随着微服务的数量增多而变的难以管理。所以spring cloud 给我们提供了一个很强大的配置管理中心,spring cloud config,他可以将我们的配置文件集中于一处进行集中管理。并且可以通过spring cloud bus + rabbitMQ 进行广播,同步到所有节点,并且直接生效。

 

注意几点问题:

1,可以单独使用spring cloud config进行配置管理,但是无法实现广播推送到多节点,那么就感觉意义不大了;想要实现广播推送,那么应该还是需要使用spring cloud bus+rabbitMQ的。

2,rabbitMQ安装回头单独写一个吧,这里说一下我遇到的一个使用的问题,默认账户密码是guest/guest,并且这个账户的权限不足外部登录访问。需要操作设置,不然会启动项目时会出现socketException。 无法登录。

3,我用的spring boot版本较高,和spring boot1的很多依赖名称都改变了(比如feign->openfeign等等),部分其他博主的教程不太可行。这里使用时你们可以注意一下版本号。

 

建设配置文件的git仓库

学习阶段,我一般使用的是github,接下来我写一下我建立的git配置文件仓库。

登录github,点击右上角“+”号,new repository

因为这个仓库我已经存在,所以爆红,换个名称即可。最后点击create repository。新建仓库。然后进入仓库

进入新建文件页面。

这里注意的是如果你想将文件创建在一个文件夹下,再标了1的位置先写上你的文件夹名,再用/隔开即可。类似再IDEA下新建一个类的操作。在2处可以编写你的配置文件内容。最后commit。

这里我已经新建好了一个yml,以此为例。

 

搭建spring cloud config server端

我们使用sts自带的插件spring starter project新建一个spring boot 项目

然后选择依赖,这里我直接贴上我的POM,上面有注释。

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
	<modelVersion>4.0.0</modelVersion>
	<parent>
		<groupId>org.springframework.boot</groupId>
		<artifactId>spring-boot-starter-parent</artifactId>
		<version>2.1.2.RELEASE</version>
		<relativePath/> <!-- lookup parent from repository -->
	</parent>
	<groupId>com.travelsky.config</groupId>
	<artifactId>travelsky-config</artifactId>
	<version>0.0.1-SNAPSHOT</version>
	<name>travelsky-config</name>
	<description>Demo project for Spring Boot</description>

	<properties>
		<java.version>1.8</java.version>
		<spring-cloud.version>Greenwich.RC2</spring-cloud.version>
	</properties>

	<dependencies>
	
		<!-- spring cloud bus依赖,用于将配置文件推送 -->
	    <dependency>
	        <groupId>org.springframework.cloud</groupId>
	        <artifactId>spring-cloud-starter-bus-amqp</artifactId>
	    </dependency>

		<!-- eureka 客户端依赖,用于再eureka注册中心中注册自己
			并且从eureka中获取application name用于匹配配置文件
		 -->
		<dependency>
			<groupId>org.springframework.cloud</groupId>
			<artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
		</dependency>
	
		<!-- spring cloud config server端 依赖 -->
		<dependency>
			<groupId>org.springframework.cloud</groupId>
			<artifactId>spring-cloud-config-server</artifactId>
		</dependency>

		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-test</artifactId>
			<scope>test</scope>
		</dependency>
	</dependencies>

	<dependencyManagement>
		<dependencies>
			<dependency>
				<groupId>org.springframework.cloud</groupId>
				<artifactId>spring-cloud-dependencies</artifactId>
				<version>${spring-cloud.version}</version>
				<type>pom</type>
				<scope>import</scope>
			</dependency>
		</dependencies>
	</dependencyManagement>

	<build>
		<plugins>
			<plugin>
				<groupId>org.springframework.boot</groupId>
				<artifactId>spring-boot-maven-plugin</artifactId>
			</plugin>
		</plugins>
	</build>

	<repositories>
		<repository>
			<id>spring-milestones</id>
			<name>Spring Milestones</name>
			<url>https://repo.spring.io/milestone</url>
		</repository>
	</repositories>

</project>

然后我们要写一下config的配置文件

server:
  port: 8084
spring:
  application:
    name: travelsky-config-server
  cloud:
    config:
      server:
        git:   
          uri:      # 配置git仓库的地址
          search-paths: #git仓库地址下的相对地址,可以配置多个,用,分割。
          username: # git仓库的账号
          password: # git仓库的密码
          
  rabbitmq:   #rabbitMQ配置
    host: localhost 
    port: 5672
    username: admin
    password: 123456
eureka:
  client:
    serviceUrl:
      defaultZone: http://127.0.0.1:8081/eureka/    ## 注册中心eurka地址

#关闭安全验证,和之前的校验写法不一样了,这里要注意。
management: 
    endpoints: 
        web: 
            exposure: 
                include: "*"
      

接下来最后一步,在启动类加上对spring cloud config的支持注解,和eureka客户端发现注解。

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.client.discovery.EnableDiscoveryClient;
import org.springframework.cloud.config.server.EnableConfigServer;

@SpringBootApplication
@EnableConfigServer
@EnableDiscoveryClient
public class TravelskyConfigApplication {

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

}

这里我们的项目已经搭建完成了。

 

spring cloud config clinet 

这里我们将之前做的spring cloud + eureka 搭建的生产者拿来改造一下。

为了完整性我就把全部的pom贴出来;里面有注释。

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
	<modelVersion>4.0.0</modelVersion>
	<parent>
		<groupId>org.springframework.boot</groupId>
		<artifactId>spring-boot-starter-parent</artifactId>
		<version>2.1.1.RELEASE</version>
		<relativePath /> <!-- lookup parent from repository -->
	</parent>
	<groupId>com.travelsky.producer.test</groupId>
	<artifactId>travelsky-producer-test</artifactId>
	<version>0.0.1-SNAPSHOT</version>
	<name>travelsky-producer-test</name>
	<description>Demo project for Spring Boot</description>

	<properties>
		<java.version>1.8</java.version>
		<spring-cloud.version>Greenwich.RC2</spring-cloud.version>
	</properties>

	<dependencies>

		<dependency>
			<groupId>org.springframework.cloud</groupId>
			<artifactId>spring-cloud-starter-bus-amqp</artifactId>
		</dependency>

		<!-- 监控,方便自动更新refresh -->
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-actuator</artifactId>
		</dependency>

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

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

		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-test</artifactId>
			<scope>test</scope>
		</dependency>
	</dependencies>

	<dependencyManagement>
		<dependencies>
			<dependency>
				<groupId>org.springframework.cloud</groupId>
				<artifactId>spring-cloud-dependencies</artifactId>
				<version>${spring-cloud.version}</version>
				<type>pom</type>
				<scope>import</scope>
			</dependency>
		</dependencies>
	</dependencyManagement>

	<build>
		<plugins>
			<plugin>
				<groupId>org.springframework.boot</groupId>
				<artifactId>spring-boot-maven-plugin</artifactId>
			</plugin>
		</plugins>
	</build>

	<repositories>
		<repository>
			<id>spring-milestones</id>
			<name>Spring Milestones</name>
			<url>https://repo.spring.io/milestone</url>
		</repository>
	</repositories>

</project>

然后我们新建一个bootstrap.yml配置文件

server: 
  port: 8050   #服务端口号
spring: 
  application: 
    name: travelsky-producer-test  #服务名称
  rabbitmq: 
    host: localhost
    port: 5672
    username: admin
    password: 123456
  cloud: 
    bus: 
      trace: 
        enabled: true
    config: 
          #配置文件名称application前缀
      name: travelsky-producer-test
            #配置文件分支名
      profile: dev
#            配置到了eureka管理,不直接使用URI了。
#            uri: http://localhost:8086/
      label: master
      discovery:
                #开始config服务发现支持
        enabled: true
                #指向sever端name
        service-id: travelsky-config-server


eureka: 
  client: 
    serviceUrl:  #注册中心的注册地址
      defaultZone: http://127.0.0.1:8081/eureka/ 
  instance: 
    lease-renewal-interval-in-seconds: 1  #每间隔1s,向服务端发送一次心跳,证明自己依然”存活“
    lease-expiration-duration-in-seconds: 2 #告诉服务端,如果我2s之内没有给你发心跳,就代表我“死”了,将我踢出掉。
#关闭安全验证
management: 
    endpoints: 
        web: 
            exposure: 
                include: "*"

这里有一个关于bootstrap.yml和applicatio.yml的关系。就是bootstrap是在application之前被运行的。他是运来加载一些不会变得配置,比如application.name;而当使用spring cloud config时,一般application被用来做一些应用级的配置,可以进行热加载。

这时候我们系统中就可以不适用application.yml了。

我们写一个controller测试一下

import org.springframework.beans.factory.annotation.Value;
import org.springframework.cloud.context.config.annotation.RefreshScope;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RestController;

/**
 * 生产者暴露接口服务
 * @ClassName: com.travelsky.ProducerTestController
 * @author: wm
 * @date: 2019年1月12日 上午11:14:45
 * @version : 1.0
 */
@RestController
//刷新地址http://localhost:8082/actuator/refresh/
@RefreshScope//自动更新配置文件
public class ProducerTestController {

	@Value("${travelsky.hello}")
	private String travelskyHello;
	
	@GetMapping("/getHello")
	public String getHello() {
		return this.travelskyHello;
	}
	

}

使用value注解从配置文件中取出travelsky.hello的值进行打印。

 

测试

这时候我们按顺序启动以下服务:1,eureka注册中心。2,config配置中心。3,travelsky-producer-test 8050。4,travelsky-producer-test 8051。

然后访问http://localhost:8050/getHello进行测试

此时已经证明我们已经通过config配置中心管理了我们的配置文件。

 

单一节点刷新配置文件

在生产过程中配置文件很可能需要改变,这时候我们就需要使用到spring cloud config 的refresh功能了。他可以让我们通过重新请求配置文件,得到最新的配置而无需重启项目。做到更改即生效。

我们 先来修改一下配置文件,再最后加11111111111

这时候我们再请求http://localhost:8050/getHello结果并没有改变。

接下来我们发送一个post请求

http://localhost:8050/actuator/refresh

再次请求http://localhost:8050/getHello,会发现结果已经是最新的了。

这就大大的方便了我们的使用,无需重启系统。

但这时候我们请求http://localhost:8051/getHello结果却没有改变。多节点情况该如何处理呢?

 

多节点刷新

 

真正的生产环境,肯定是要高可用性的,所以一般服务我们都不可能只部署一个节点。那么当配置文件更新时需要多节点同步更新。所以我们需用使用spring cloud bus 进行广播推送。其实很简单。

1,通过client端进行同步刷新

我们将上面的post请求变更一下,http://localhost:8050/actuator/bus-refresh

当请求结束后,我们请求8051,就会发现8050和8051被同步更新了。

2,通过server端进行广播推送

我们先将配置文件改成33333333

这时候我们请求config端的POST请求:http://localhost:8084/actuator/bus-refresh

同样的,8050和8051节点都得到了更新。

 

spring cloud config其实还是有他的缺点的,比如没有可视化界面,这一点好像携程的开源框架apollo做的还不错。回头单独再开一个了解一下apollo。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值