在一个微服务架构中会有很多微服务,每个微服务都有自己的配置文件,这会随着微服务的数量增多而变的难以管理。所以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。