Phenomenon of the problem

One of the log directories will be named log.base_is_undefined

The structure of the project is a Spring-Boot project with spring-Cloud components introduced.

The log configuration is stored in bootstrap.yml

logging:
  appLevel: info
  path: ${LOG_PATH:logs/}
Copy the code

The logback-spring. XML configuration is as follows

<configuration scan="true" scanPeriod="30 seconds">
    <include resource="org/springframework/boot/logging/logback/defaults.xml"/>
    <include resource="org/springframework/boot/logging/logback/console-appender.xml"/>

    <springProperty scope="context" name="log.base" source="logging.path"/>
    <springProperty scope="context" name="logLevel" source="logging.appLevel"/>
    <springProperty scope="context" name="log.additivity" source="logging.additivity"/>

    <include resource="logback-core.xml" />

    <root level="${logLevel}">
        <appender-ref ref="other"/>
    </root>
</configuration>
Copy the code

Cause analysis,

I. Preliminary analysis

Yml. Since spring-Cloud is introduced, the Bootstrap Context will be created by spring-Cloud. As the parent Context of the Spring Application Context.

The Bootstrap Context reads the configuration written in bootstrap.yml. The log configuration cannot be read because it is written to application. IS_UNDEFINED log directory using logback

Now you have the second case, where the logging configuration is already in bootstrap.yml, and it doesn’t matter which side you put it on. However, after testing, two configuration files can be configured at the same time to solve this problem. But this is an inelegant way of handling the problem.

Second, detailed analysis

I initially suspected that it was due to the spring-Cloud context, so I tried to remove the spring-Cloud to see the effect. This is the only spring-cloud-related component in POM.xml. Comment it out.

<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-context</artifactId>
</dependency>
Copy the code

Since the spring-Cloud component is removed, the bootstrap.yml configuration is not read. Move the configuration inside to application.yml, and it does not generate log.base_IS_UNDEFINED after startup.

According to the design of the Spring context, the child context can read the configuration of the parent context, so the configuration written in bootstral. Yml should be available to see if they are parent context. Reference: Bootstrap context in Spring Cloud

Go to the SpringApplication class and debug the run method to see the Spring context initialization process:

The first is initializationBootstrap ContextIs initialized by reading the bootstrap.yml configuration file.And then we initialize itApplication ContextAnd you can see that it isBootstrap ContextParent-child context indeed.

Let’s see why logback didn’t get the configuration.

Debug logback:

<configuration scan="true" scanPeriod="30 seconds" debug="true">.</configuration>
Copy the code

According to the logback debug log, the logback is initialized twice. The configured value can be obtained once and the configured value cannot be obtained once. This is because both spring contexts initialize the logging component separately, which is normal, and now let’s see why the configuration value is not fetched the second time.

- for the first time loading 10:02:10, 904 | - INFO in ch. Qos. Logback. Core. Joran. Action. AppenderAction - Naming appenders as [access] 10:02:10, 912 | - INFO in C.Q.L.C ore. Rolling. The TimeBasedRollingPolicy @ 1115420993 - No compression will will be 2 10:02:10, 914 |-INFO in c.q.l.core.rolling.TimeBasedRollingPolicy@1115420993 - Will use the pattern Logs / / access/access. Log. % d - the dd - HH} {yyyy - MM. % I for the active file 10:02:10, | - INFO in 915 ch.qos.logback.core.rolling.SizeAndTimeBasedFNATP@26b1ded0 - The date pattern is 'yyyy-MM-dd-HH' from file name pattern 'logs / / access/access. Log. % d - the dd - HH} {yyyy - MM. % I. 10:02:10, | - INFO in 915 Ch. Qos. Logback. Core. Rolling. SizeAndTimeBasedFNATP @ 26 b1ded0 - Roll - over at the top of every hour. 10:02:10, | - INFO in 917 ch.qos.logback.core.rolling.SizeAndTimeBasedFNATP@26b1ded0 - Setting initial period to Fri May 14 09:50:35 CST 2021 10:02:10, 918 | - WARN in ch. Qos. Logback. Core. Rolling. The SizeAndTimeBasedFNATP @ 26 b1ded0 - SizeAndTimeBasedFNATP is deprecated. Home Use SizeAndTimeBasedRollingPolicy 10:02:10, | - WARN in 918 ch.qos.logback.core.rolling.SizeAndTimeBasedFNATP@26b1ded0 - For more information see http://logback.qos.ch/manual/appenders.html#SizeAndTimeBasedRollingPolicy 10:02:10, | - INFO in 919 ch.qos.logback.core.joran.action.NestedComplexPropertyIA - Assuming default type [ch. Qos. Logback. Classic. Encoder. PatternLayoutEncoder] for [encoder] property 10:02:10, | - INFO in 921 ch.qos.logback.core.rolling.RollingFileAppender[access] - Active log file name: Logs / / access/access log 10:02:10, 921 | - INFO in ch. Qos. Logback. Core. Rolling. The RollingFileAppender [access] - the File property Is set to [logs / / access/access log] - the second loading 10:02:11, 325 | - INFO in ch. Qos. Logback. Core. Joran. Action. AppenderAction - Naming appenders as [access] 10:02:11, 325 | - INFO in C.Q.L.C ore. Rolling. TimeBasedRollingPolicy @ 620091520 - No compression will Will be 2 10:02:11, 326 | - INFO in C.Q.L.C ore. Rolling. The TimeBasedRollingPolicy @ 620091520 - will use the pattern The base_IS_UNDEFINED/access/access. Log. % d - the dd - HH} {yyyy - MM. % I for the active file 10:02:11, | - INFO in 326 ch.qos.logback.core.rolling.SizeAndTimeBasedFNATP@7d573a89 - The date pattern is 'yyyy-MM-dd-HH' from file name pattern 'log. Base_IS_UNDEFINED/access/access. Enter the % d - the dd - HH} {yyyy - MM. % I. 10:02:11, | - INFO in 326 Ch. Qos. Logback. Core. Rolling. SizeAndTimeBasedFNATP @ 7 d573a89 - Roll - over at the top of every hour. 10:02:11, | - INFO in 326 ch.qos.logback.core.rolling.SizeAndTimeBasedFNATP@7d573a89 - Setting initial period to Fri May 14 09:51:52 CST 2021 10:02:11, 326 | - WARN in ch. Qos. Logback. Core. Rolling. The SizeAndTimeBasedFNATP @ 7 d573a89 - SizeAndTimeBasedFNATP is deprecated. Home Use SizeAndTimeBasedRollingPolicy 10:02:11, | - WARN in 326 ch.qos.logback.core.rolling.SizeAndTimeBasedFNATP@7d573a89 - For more information see http://logback.qos.ch/manual/appenders.html#SizeAndTimeBasedRollingPolicy 10:02:11, | - INFO in 326 ch.qos.logback.core.joran.action.NestedComplexPropertyIA - Assuming default type [ch. Qos. Logback. Classic. Encoder. PatternLayoutEncoder] for [encoder] property 10:02:11, | - INFO in 327 ch.qos.logback.core.rolling.RollingFileAppender[access] - Active log file name: Log. Base_IS_UNDEFINED/access/access. Log 10:02:11, 327 | - INFO in ch. Qos. Logback. Core. Rolling. The RollingFileAppender [access] -  File property is set to [log.base_IS_UNDEFINED/access/access.log]Copy the code

According to this line of log 10:02:11, 327 | – INFO in ch. Qos. Logback. Core. Rolling. The RollingFileAppender [access] – Active log file name: log.base_IS_UNDEFINED/access/access.log

The TimeBasedRollingPolicy class is retrieved from the SpringPropertyAction class, and the logback configuration is retrieved twice, once with a value and once with no value. I just know that I can’t get the configuration value, but I don’t know why, the code level is too deep, there is no clue.

At this point, the colleague realized that it was ok to remove the jasypt-spring-boot-starter reference.

<dependency>
    <groupId>com.github.ulisesbocchio</groupId>
    <artifactId>jasypt-spring-boot-starter</artifactId>
    <version>2.1.1</version>
</dependency>
Copy the code

So I went back to the previous breakpoint debugging, and compared the two cases with no breakpoint and found that there was a difference: The defaultProperties have one value and one value, and the bootstrap.yml configuration is in there.

Continue to tracking code, found that the class went to the BootstrapApplicationListener mergeDefaultProperties method, the method name should be defaultProperties no value associated with the above.

Go ahead and find that the configuration of the parent context is merged:

If jasypt-spring-boot-starter is used, it is false. If jasypt-spring-boot-starter is not used, it is true.

The jasypt-spring-boot-starter package changes the PropertySource type, causing the configuration merge failure.

The solution

The bootstrap.yml configuration is cleared by using jasypt-spring-boot-starter, which means that the bootstrap.yml configuration is not available in the application context. If there are critical configurations, troubleshooting can be difficult.

Github has a solution to this problem: bootstrap configurations are cleaned up after the starter integration

Add the following configuration to bootstrap.yml:

jasypt:
  encryptor:
    bootstrap: false
Copy the code