如何在Bean实例化之前记录弹簧引导应用程序的所有活动属性?

2022-09-02 09:37:00

已经有一个问题要求记录活动配置,有一个正确的答案,但问题是,只有当所有bean都正确实例化时,才会记录配置。我想记录所有属性,即使(主要是)如果应用程序在启动时崩溃。我的问题更具体:

如何在Bean实例化之前记录弹簧引导应用程序的所有活动属性?


答案 1

为此,您需要注册一个应用程序管理器。根据文档,要捕获的事件是 ApplicationPreparedEvent

应用程序准备事件是在 SpringApplication 启动并且 ApplicationContext 已完全准备就绪但未刷新时发布的事件。将加载 Bean 定义,并且环境在此阶段已准备就绪,可供使用。

main 方法如下所示:

public static void main(String[] args) {
        SpringApplication springApplication = new SpringApplication(MyApplication.class);
        springApplication.addListeners(new PropertiesLogger());
        springApplication.run(args);        
}

我已经重用了当前问题中引用的答案的代码,但我已经修改了它,因为您获得的上下文尚未刷新,并且环境的结构与应用程序启动后不完全相同。我还按属性源打印了属性:一个用于系统环境,一个用于系统属性,一个用于应用程序配置属性,等等...另请注意,可以多次触发 ,并且属性仅在第一次打印。有关详细信息,请参阅 Spring Boot 问题 #8899ApplicationPreparedEvent

package com.toto.myapp.util;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.boot.context.event.ApplicationPreparedEvent;
import org.springframework.context.ApplicationListener;
import org.springframework.core.env.ConfigurableEnvironment;
import org.springframework.core.env.EnumerablePropertySource;
import org.springframework.core.env.PropertySource;

import java.util.LinkedList;
import java.util.List;

public class PropertiesLogger implements ApplicationListener<ApplicationPreparedEvent> {
  private static final Logger log = LoggerFactory.getLogger(PropertiesLogger.class);

  private ConfigurableEnvironment environment;
  private boolean isFirstRun = true;

  @Override
  public void onApplicationEvent(ApplicationPreparedEvent event) {
    if (isFirstRun) {
      environment = event.getApplicationContext().getEnvironment();
      printProperties();
    }
    isFirstRun = false;
  }

  public void printProperties() {
    for (EnumerablePropertySource propertySource : findPropertiesPropertySources()) {
      log.info("******* " + propertySource.getName() + " *******");
      String[] propertyNames = propertySource.getPropertyNames();
      Arrays.sort(propertyNames);
      for (String propertyName : propertyNames) {
        String resolvedProperty = environment.getProperty(propertyName);
        String sourceProperty = propertySource.getProperty(propertyName).toString();
        if(resolvedProperty.equals(sourceProperty)) {
          log.info("{}={}", propertyName, resolvedProperty);
        }else {
          log.info("{}={} OVERRIDDEN to {}", propertyName, sourceProperty, resolvedProperty);
        }
      }
    }
  }

  private List<EnumerablePropertySource> findPropertiesPropertySources() {
    List<EnumerablePropertySource> propertiesPropertySources = new LinkedList<>();
    for (PropertySource<?> propertySource : environment.getPropertySources()) {
      if (propertySource instanceof EnumerablePropertySource) {
        propertiesPropertySources.add((EnumerablePropertySource) propertySource);
      }
    }
    return propertiesPropertySources;
  }
}

答案 2


推荐