Can I manually load @ConfigurationProperties without the Spring AppContext?

前端 未结 5 1675
天命终不由人
天命终不由人 2021-01-02 00:28

Is there any way to load a class marked with @ConfigurationProperties without using a Spring Context directly? Basically I want to reuse all the smart logic tha

相关标签:
5条回答
  • 2021-01-02 00:56

    Here's an update to ctranxuan's answer for Spring Boot 2.x. In our situation, we avoid spinning up a Spring context for unit tests, but do like to test our configuration classes (which is called AppConfig in this example, and its settings are prefixed by app):

    public class AppConfigTest {
      private static AppConfig config;
    
      @BeforeClass
      public static void init() {
        YamlPropertiesFactoryBean factoryBean = new YamlPropertiesFactoryBean();
        factoryBean.setResources(new ClassPathResource("application.yaml"));
    
        Properties properties = factoryBean.getObject();
    
        ConfigurationPropertySource propertySource = new MapConfigurationPropertySource(properties);
        Binder binder = new Binder(propertySource);
    
        config = binder.bind("app", AppConfig.class).get(); // same prefix as @ConfigurationProperties
      }
    }
    
    0 讨论(0)
  • 2021-01-02 00:56
    import org.springframework.boot.context.properties.bind.Binder
    val binder = Binder.get(environment)
    binder.bind(prefix, MySettings.class).get
    
    0 讨论(0)
  • 2021-01-02 00:57

    This post is going into similar direction but extends the last answer with also validation and property placeholder resolutions.

    Spring Boot Binder API support for @Value Annotations

    @Value annotations in ConfigurationPropertys don't seem to bind properly though (at least if the referenced values are not part of the ConfigurationProperty's prefix namespace).

    0 讨论(0)
  • 2021-01-02 01:08

    The "magic" class you are looking for is PropertiesConfigurationFactory. But I would question your need for it - if you only need to bind once, then Spring should be able to do it for you, and if you have lifecycle issues it would be better to address those (in case they break something else).

    0 讨论(0)
  • 2021-01-02 01:10

    I had the same "issue". Here is how I solved it in SpringBoot version 1.3.xxx and 1.4.1.

    Let's say we have the following yaml configuration file:

    foo:
      apis:
          -
           name: Happy Api
           path: /happyApi.json?v=bar
          -
           name: Grumpy Api
           path: /grumpyApi.json?v=grrr
    

    and we have the following ConfigurationProperties:

    @ConfigurationProperties(prefix = "foo")
    public class ApisProperties {
        private List<ApiPath> apis = Lists.newArrayList();
    
        public ApisProperties() {
        }
    
        public List<ApiPath> getApis() {
            return apis;
        }
    
        public static class ApiPath {
            private String name;
            private String path;
    
            public String getName() {
                return name;
            }
    
            public void setName(final String aName) {
                name = aName;
            }
    
            public String getPath() {
                return path;
            }
    
            public void setPath(final String aPath) {
                path = aPath;
            }
        }
    } 
    

    Then, to do the "magic" things of Spring Boot programmatically (e.g. loading some properties in a static method), you can do:

    private static ApisProperties apiProperties() {
        try {
            ClassPathResource resource;
            resource = new ClassPathResource("/config/application.yml");
    
            YamlPropertiesFactoryBean factoryBean;
            factoryBean = new YamlPropertiesFactoryBean();
            factoryBean.setSingleton(true); // optional depends on your use-case
            factoryBean.setResources(resource);
    
            Properties properties;
            properties = factoryBean.getObject();
    
            MutablePropertySources propertySources;
            propertySources = new MutablePropertySources();
            propertySources.addLast(new PropertiesPropertySource("apis", properties));
    
            ApisProperties apisProperties;
            apisProperties = new ApisProperties();
    
            PropertiesConfigurationFactory<ApisProperties> configurationFactory;
            configurationFactory = new PropertiesConfigurationFactory<>(apisProperties);
            configurationFactory.setPropertySources(propertySources);
            configurationFactory.setTargetName("foo"); // it's the same prefix as the one defined in the @ConfigurationProperties
    
            configurationFactory.bindPropertiesToTarget();
            return apisProperties; // apiProperties are fed with the values defined in the application.yaml
    
        } catch (BindException e) {
            throw new IllegalArgumentException(e);
    
        }
    }
    
    0 讨论(0)
提交回复
热议问题