非spring boot (即spring) 使用/集成 Spring cloud Config 分布式配置中心

来源:互联网 发布:nginx 优化加速 编辑:程序博客网 时间:2024/05/01 05:56

非spring boot 使用Spring cloud Config (老版 spring 集成 spring cloud config ) || spring cloud config 搭建
个人博客:https://eric-ly.github.io/

背景:想自己搭建一个分布式的配置中心,发现spring Cloud Config 不错,(后台基于git,可以配置webhook 实现热更新、回滚,分环境配置等)与自己的期望相近。

虽然好用但是基于spring boot的,spring 与spring boot 还是有挺多巨别。为了能用在现有的 项目中,决定 想办法集成一下,百度 google 都没有现成的可以copy…

于是想自己实现下,查询之后 按照一个大神的想法实现了下。中间遇到许多问题,不过更加了解spring了 也会继续完善。java 新人 希望各位大神多多指导。

github:https://github.com/Eric-ly/hconfig

后面有项目的测试说明:

实现结果:

在spring 中 使用spring Cloud Config 功能,分项目/环境 获取配置文件,自动注入,自动更新 spring boot的注解无法兼容 待稍后解决。

前提:

一个简单的spring spring MVC 项目。

config server 配置好,github 配置好。 这个网上百度就有很多。

(在搭建spring cloud config (server and client)后,搭建注册应用中心 eureka 和 rabbitMq 进行自动热更新。)

实现方式:

1.创建自定义的ApplicationContext,先通过在web.xml中设置 contextClass 来指定应用使用自定义的ApplicationContext,

2.在自定义的applicationContext中 覆写 创建environment的方法,用来定制 environment,把config server的配置信息加入 应用

3.自定义的CloudEnvironment extends StandardServletEnvironment,扩展customizePropertySources方法,指定 配置的来源,然后放入 propertySource的list中,作为整个项目的 配置之一。

4.自定义propertySouce ,使用spring cloud client的ConfigClientProperties 作为 propertySouce的来源, 然后添加到PropertySource中

5.自定义配置类,将配置中心的配置按照单个/model 进行注入

后期实现:

(1)兼容@refresh的注解,使 自动更新。

(2)通过修改name 获取多个配置文件。优化config server的文件

(3)细化 回滚功能

(4)写 拦截器,打成jar包,成为一个工具包,通过自定义注解 来使用。隐藏实现。成为一个通用的功能

最近有点忙,这个项目先达到能基本用,慢慢完善。

具体实现步骤:

1.增加pom 依赖,spring mvc ,spring cloud,logback 等

    <properties>        <spring.version>4.2.9.RELEASE</spring.version>        <jstl.version>1.2</jstl.version>        <junit.version>4.11</junit.version>        <logback.version>1.0.13</logback.version>        <jcl-over-slf4j.version>1.7.5</jcl-over-slf4j.version>        <spring.cloud.version>1.2.0.RELEASE</spring.cloud.version>    </properties>

2.创建自定义的ApplicationContext,先通过在web.xml中设置 contextClass 来指定自己的ApplicationContext,

contextClass

org.lybm.hconfig.clientWosb.config.CustomWebApplicationContext

ApplicationContext 是spring ioc 的体现,spring 中的容器。一般有几种实现。FileSystemXmlApplicationContext/ClassPathXmlApplicationContext/WebXmlApplicationContex我们在搭建框架的时候一般通过注册ContextLoaderListener 监听 去自动获取 初始化(这里先去通过classContext的配置获取,而且必须可以强转为WebXmlApplicationContxt,如果没设置 创建默认的),这里为了使用spring boot的功能,我们需要自定义ApplicationContext

在新定义的ApplicationContext 中我们使用自定义的Environment,为了把配置放到 environment中来使用。

public class CustomWebApplicationContext extends XmlWebApplicationContext {    @Override    protected ConfigurableEnvironment createEnvironment() {        System.out.println("-------- loaded my CustomWebApplicationContext context");        return new CloudEnvironment();    } }

3.创建 自定义的Environment,用来把 配置中心 config server的配置 放到环境中。

(1)重写定制方法

自定义的CloudEnvironment extends StandardServletEnvironment,扩展customizePropertySources方法,指定 配置的来源,然后放入 propertySource的list中,作为整个项目的 配置之一。

这里说一下,spring 的所有配置,包括system配置 ,jdni,自定义的properties等 都会放到 spring的environtment的propertySource的list中,用来使用。

    @Override    protected void customizePropertySources(MutablePropertySources propertySources) {        super.customizePropertySources(propertySources);        try {            //用来添加应用名到environment中            propertySources.addLast( initResourcePropertySourceLocator() );            //添加config Server的配置            PropertySource<?> source = initConfigServicePropertySourceLocator(this);            propertySources.addLast(source);        } catch (Exception ex) {            logger.warn("failed to initialize cloud config environment", ex);        }    }

(2)获取 config server的 配置信息,生成propertySource ,创建一个来源。用到了spring cloud的方法

这里说明一下,在配置ConfigClientProperties 连接信息的时候, 后面的代码 会重新覆盖 applicationName 所以这里设置name是没有意义的。 会从environment 中 获取商品日嗯

    private PropertySource<?> initConfigServicePropertySourceLocator(Environment environment) {        ConfigClientProperties configClientProperties = new ConfigClientProperties(environment);        configClientProperties.setUri("http://localhost:9001/");//        configClientProperties.setName("config-client");        configClientProperties.setProfile("dev");        configClientProperties.setLabel("master");        logger.debug("will load the client configuration-------"+configClientProperties);        ConfigServicePropertySourceLocator configServicePropertySourceLocator =                new ConfigServicePropertySourceLocator(configClientProperties);        return configServicePropertySourceLocator.locate(environment);    }

重新设置name的代码,如下, 这样会导致我们连接获取到的配置信息不正确。

        override.setName(                environment.resolvePlaceholders("${" + ConfigClientProperties.PREFIX                        + ".name:${spring.application.name:application}}"));

于是,我在 进行这一步前 将正确的 name信息, 加入到environment中,就可以找到 正常使用。

(3)添加 应用相关配置 到environment中。

在查询相关源码后发现。我只需要 将一个resource 对象放入 environment的 资源 list中就可以,这里我新建一个properties,写入配置,然后封装成resource 就可以了。

cloud-config-context.propertiesspring.application.name=config-clientdemo=demo

具体实现:

    Resource resource = new DefaultResourceLoader(this.getClass().getClassLoader()).getResource("classpath:cloud-config-context.properties");    resourcePropertySource = new ResourcePropertySource(resource);

这里学习的过程 花费了一些时间。了解了 environment,propertySource 等spring 加载配置信息的代码。一些相关知识会在后续贴出来

4.现在environment 重写好了,创建配置信息类

我们目前可以通过两种方式获取配置,

(1)单个配置的获取。

    //单独一个字段的获取    @Value("${test}")    private String test; 这么写 如果没有test ,程序会报错,启动不了, 所以 不要用这种写法     //单独一个字段的获取    @Value("${xxx:x}")    private String test;单个字段 推荐这么写,如果xxx没有,则使用默认值x ,这样比较符合 业务场景

(2)按照配置信息类 获取

如下,将配置信息映射成一个model,可以将统一前缀 如wosb的配置信息 放到一个类里,这样方便管理,在写配置的时候 也好区分,就像一个一个配置文件。

@ConfigurationProperties(prefix = "wosb")@Datapublic class WosbProperties {    private String test;    private String demo;    private String name;    private String paht;}

(2.2)将配置类 注入,两种方式,

a. 通过注解,增加@Configuration,将这个类注入到 容器中

注解定义@Target(ElementType.TYPE)@Retention(RetentionPolicy.RUNTIME)@Documented@Componentpublic @interface Configuration {

b.通过配置PropertySourcesPlaceHolderConfigurer,不太推荐

@Configuration@EnableConfigurationProperties({WosbProperties.class})public class PropertiesConfigurer {    @Bean    public static PropertySourcesPlaceholderConfigurer propertySourcesPlaceholderConfigurer() {        return new PropertySourcesPlaceholderConfigurer();    }}

5.继承完成,使用/测试

(1)创建controller

(2)使用

    @Autowired    private SearchProperties searchProperties;    //单独一个字段的获取    @Value("${xxx:x}")    private String test;

(3)可以注入Environment env 来查看当前的environment信息。

项目说明:

github:https://github.com/Eric-ly/hconfig

github的配置文件地址:https://github.com/Eric-ly/hconfig-properties

cloudeureka: spring boot的应用注册中心

ConfligClientNoSpringBoot: 非spring boot 应用 使用 应用配置中心

hconfigclient:spring boot的 应用配置中心的 客户端

hconfigserver:spring boot的 应用配置中心的服务端

使用说明:

1.启动 应用注册中心

(1)入口类:EurekaServerApplication IDE debug 或者用命令

(2)配置:

server.port=8761eureka.instance.hostname=localhost

2.启动 配置中心的服务端

(1)入口类:ConfigServerApplication

(2)配置:

端口:

spring.application.name=config-serverserver.port=9001

gitlab :来源

spring.cloud.config.server.git.uri=https://github.com/Eric-ly/hconfig-properties.gitspring.cloud.config.server.git.searchPaths=configRepospring.cloud.config.server.git.username=spring.cloud.config.server.git.password=spring.cloud.config.server.git.basedir=src/main/resource/config# //加载到ConfigServer项目所在的本地目录的位置, 可以不用配置

消息总线,用来自动热更新,配置的是mq的地址,需要自己启

#   添加cloud bus 消息总线,配合webhook 实现 所有client的消息热更新spring.rabbitmq.host=localhostspring.rabbitmq.port=5672spring.rabbitmq.username=guestspring.rabbitmq.password=guest## 刷新时,关闭安全验证management.security.enabled=false## 开启消息跟踪spring.cloud.bus.trace.enabled=true

注册中心:

#注册中心地址eureka.client.serviceUrl.defaultZone=http://localhost:8761/eureka/

3.启动 非spring boot的 ConfligClientNoSpringBoot

(1)配置 应用名

cloud-config-context.properties

spring.application.name=config-client

(2)IDE添加tomcat,然后启动

(3)测试

http://localhost:8888/hello

结果: 配置文件都读取到了

search-pproperties : collection :{{engine}} demo: {{solr}} ||| wosb-properties: demo : {{ a}} name: {{ wosb }} path: {{ /}} single value : test?{{ abcdegggg }}

4.spring boot的 配置中新 客户端( 自动热更新)hconfigclient

(1)入口类 entryApplication

(2)配置:

(1) 应用名+ 配置中心的 服务端

spring.application.name=config-clientspring.cloud.config.label=masterspring.cloud.config.profile=devspring.cloud.config.uri= http://localhost:9001/server.port=8888

(2)

#注册中心地址eureka.client.serviceUrl.defaultZone=http://localhost:8761/eureka/

(3)

## 刷新时,关闭安全验证management.security.enabled=false## 开启消息跟踪spring.cloud.bus.trace.enabled=truespring.rabbitmq.host=localhostspring.rabbitmq.port=5672spring.rabbitmq.username=guestspring.rabbitmq.password=guest

(4) 测试

(提交代码触发post请求给bus/refresh ----server端接收到请求并发送给Spring Cloud Bus----Spring Cloud bus接到消息并通知给其它客户端 ----其它客户端接收到通知,请求Server端获取最新配置----全部客户端均获取到最新的配置)
阅读全文
0 0