转载

非spring boot 使用Spring cloud Config (老版 spring 集成 spring cloud config )

非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>

    <dependencies>

        <!--    junit 依赖    -->
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>${junit.version}</version>
            <!-- <scope>test</scope> -->
        </dependency>

        <!--    spring mvc 的 依赖-->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-webmvc</artifactId>
            <version>${spring.version}</version>
        </dependency>

        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>${spring.version}</version>
        </dependency>

        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-core</artifactId>
            <version>${spring.version}</version>
        </dependency>

        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-web</artifactId>
            <version>${spring.version}</version>
        </dependency>


        <!--    spring cloud config client 依赖    -->
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-config-client</artifactId>
            <version>${spring.cloud.version}</version>
        </dependency>

        <!--    logback -->
        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>jcl-over-slf4j</artifactId>
            <version>${jcl-over-slf4j.version}</version>
        </dependency>

        <dependency>
            <groupId>ch.qos.logback</groupId>
            <artifactId>logback-classic</artifactId>
            <version>${logback.version}</version>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.16.8</version>
        </dependency>

        <dependency>
            <groupId>javax.servlet</groupId>
            <artifactId>javax.servlet-api</artifactId>
            <version>3.1.0</version>
            <scope>provided</scope>
        </dependency>

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.properties
spring.application.name=config-client
demo=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")
@Data
public 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
@Component
public @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=8761
eureka.instance.hostname=localhost

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

(1)入口类:ConfigServerApplication

(2)配置:

端口:

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

gitlab :来源

spring.cloud.config.server.git.uri=https://github.com/Eric-ly/hconfig-properties.git
spring.cloud.config.server.git.searchPaths=configRepo
spring.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=localhost
spring.rabbitmq.port=5672
spring.rabbitmq.username=guest
spring.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-client
spring.cloud.config.label=master
spring.cloud.config.profile=dev
spring.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=true

spring.rabbitmq.host=localhost
spring.rabbitmq.port=5672
spring.rabbitmq.username=guest
spring.rabbitmq.password=guest

(4) 测试

(提交代码触发post请求给bus/refresh ----server端接收到请求并发送给Spring Cloud Bus
----Spring Cloud bus接到消息并通知给其它客户端 ----其它客户端接收到通知,请求Server端获取最新配置
----全部客户端均获取到最新的配置)
原文  https://juejin.im/post/5a3094d36fb9a045055e1cfe
正文到此结束
Loading...