Spring Cloud Config 流程源码详解

Server

  • 如何加载

    • 核心接口是EnvironmentRepository,提供配置的读取
    public interface EnvironmentRepository {
          Environment findOne(String application, String profile, String label);
      }
    
    • EnvironmentRepository有多种实现,基于JDBC、SVN、GIT等等
    • 默认情况下,使用的是 MultipleJGitEnvironmentRepository(可以配置多个地址的GIT数据源)
    • ConfigServerAutoConfiguration -> EnvironmentRepositoryConfiguration -> DefaultRepositoryConfiguration
    @Configuration
      @ConditionalOnMissingBean(value = EnvironmentRepository.class)
      class DefaultRepositoryConfiguration {
          ...
          @Bean
          public MultipleJGitEnvironmentRepository defaultEnvironmentRepository(...) throws Exception {
              return gitEnvironmentRepositoryFactory.build(environmentProperties);
          }
      }
    
    • MultipleJGitEnvironmentRepository 代理遍历每个 JGitEnvironmentRepository, JGitEnvironmentRepository 下使用 NativeEnvironmentRepository 代理读取本地文件。
    • AbstractScmEnvironmentRepository#findOne
    public synchronized Environment findOne(String application, String profile, String label) {
          NativeEnvironmentRepository delegate = new NativeEnvironmentRepository(getEnvironment(),
                  new NativeEnvironmentProperties());
          Locations locations = getLocations(application, profile, label);
          delegate.setSearchLocations(locations.getLocations());
          Environment result = delegate.findOne(application, profile, "");
          ...
                  getUri());
      }
    
  • 其中getLocations这部会从GIT远程仓库同步到本地

  • JGitEnvironmentRepository#getLocations -> JGitEnvironmentRepository.refresh

public String refresh(String label) {
       Git git = createGitClient();
       ...
       checkout(git, label);
       ...
       merge(git, label);
       ...
       resetHard(...)
       ...
}

  • 配置文件加载优先级顺序(上面的覆盖下面的)
模式 应用
{spring.application.name}-{profile}.properties/yml 应用-环境-配置
{spring.application.name}.properties/yml 应用-全局-配置
application-{profile}.properties/yml 公众-环境-配置
application.properties/yml 公众-全局-配置
  • 具体的逻辑参考NativeEnvironmentRepository和ConfigFileApplicationListener

  • 如何注入

    • ConfigServer本身也可以注入自己的读取的配置,使得其他服务可以和ConfigServer配置在一起,比如Eureka
    • Config的自身注入在BootStrap阶段
    • ConfigServerBootstrapConfiguration#LocalPropertySourceLocatorConfiguration
     @Configuration
      @ConditionalOnProperty("spring.cloud.config.server.bootstrap")
      public class ConfigServerBootstrapConfiguration {
          @Bean
          public EnvironmentRepositoryPropertySourceLocator environmentRepositoryPropertySourceLocator() {
              return new EnvironmentRepositoryPropertySourceLocator(this.repository, this.client.getName(),
                      this.client.getProfile(), getDefaultLabel());
          }
      }
    
    • EnvironmentRepositoryPropertySourceLocator会调用EnvironmentRepository获取配置
    public class EnvironmentRepositoryPropertySourceLocator implements PropertySourceLocator{
          @Override
          public PropertySource<?> locate(Environment environment) {
              CompositePropertySource composite = new CompositePropertySource("configService");
              for (PropertySource source : environmentRepository.findOne(name, profiles, label)
                      .getPropertySources()) {
                  composite.addPropertySource(...);
              }
              return composite;
          }
    }
    
    • PropertySourceBootstrapConfiguration#init负责所有BootStrap配置的加载,所有实现PropertySourceLocator接口的服务都会被调用
    CompositePropertySource composite = new CompositePropertySource(
                  BOOTSTRAP_PROPERTY_SOURCE_NAME);
      for (PropertySourceLocator locator : this.propertySourceLocators) {
          PropertySource<?> source = locator.locate(environment);
          composite.addPropertySource(source);
      }
    

Client

  • 2种高可用方式, Spring-Cloud-Config-Client配置存在两种策略

    1. 通过ConfigServer获取注册中心地址和其他配置
    2. 通过注册中心获取ConfigServer地址,然后获得其他配置
  • 参考国内比较成熟的分布式集中配置,比如百度Disconf,携程Apollo等基本都是第1种策略,好处是唯一需要本地写死的是配置中心的参数.

  • 但是Spring Cloud 1.X 并不支持ConfigServer集群的高可用配置。需要单独将spring-cloud-starter-config升级到2.x(可以和其他1.x组件混用)

  • 升级后就可以配多个地址了,如下:

spring:
  application:
    name: user
  cloud:
    config:
      fail-fast: true
      profile: dev
      label: master
      uri: http://localhost:8761/config, http://localhost:8762/config, http://localhost:8763/config
  • 但是网上更多介绍的是第二种方案,这里也贴出来。配置项更多了,需要同时指定configserver和eureka
spring:
  application:
    name: user
  cloud:
    config:
      fail-fast: true
      profile: dev
      label: master
      discovery:
        enabled: true
        serviceId: dashboard
      
eureka:
  client:
    register-with-eureka: true
    fetch-registry: true
    serviceUrl:
      defaultZone: http://localhost:8761/eureka/,http://localhost2:8762/eureka,http://localhost3:8763/eureka
  • 启动过程和ConfigServer注入过程类似
  • ConfigServiceBootstrapConfiguration -> ConfigServicePropertySourceLocator -> locate
public PropertySource<?> locate(Environment environment) {
    CompositePropertySource composite = new CompositePropertySource("configService");
    ...
    RestTemplate restTemplate = getSecureRestTemplate(properties);
    ...
    Environment result = getRemoteEnvironment(restTemplate...);
    ...
    for (PropertySource source : result.getPropertySources()) {
        ...
        composite.addPropertySource(source);
    }
    ...
}
  • 该服务实现了PropertySourceLocator接口,在PropertySourceBootstrapConfiguration#init中会被注入到bootStrap的context中

  • 方案2比方案1多一个环节,configServer 的 uri 是由注册中心获取的

  • DiscoveryClientConfigServiceBootstrapConfiguration#refresh

@ConditionalOnProperty(value = "spring.cloud.config.discovery.enabled", matchIfMissing = false)
@Configuration
@EnableDiscoveryClient
public class DiscoveryClientConfigServiceBootstrapConfiguration {
    void refresh() {
        String serviceId = this.config.getDiscovery().getServiceId();
        List<ServiceInstance> serviceInstances = this.instanceProvider
                    .getConfigServerInstances(serviceId);
        for (int i = 0; i < serviceInstances.size(); i++) {
                ServiceInstance server = serviceInstances.get(i);
                String url = getHomePage(server);
                listOfUrls.add(url);
        }
        ...
        configClientProperties.setUri(listOfUrls);
    }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,875评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,569评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,475评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,459评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,537评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,563评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,580评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,326评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,773评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,086评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,252评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,921评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,566评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,190评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,435评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,129评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,125评论 2 352

推荐阅读更多精彩内容