SpringBoot Logback無法獲取配置中心屬性


SpringBoot Logback無法獲取配置中心屬性

前言

最近在做項目中,需要把項目中的日志信息通過RabbitMQ將規定格式的消息發送到消息隊列中,然后ELK系統通過消息隊列拿日志並且保存起來,在日志的配置文件(logback-spring.xml)中我們需要加入RabbitMQ的配置信息,我們的RabbitMQ信息存在Nacos的配置中心,就出現項目啟動無法獲取到RabbitMQ的配置,導致出錯

如何解決

問題原因

在springboot官網 https://docs.spring.io/spring-boot/docs/2.0.4.RELEASE/reference/htmlsingle/#boot-features-logging
中對LogBack的描述中我們可以知道,項目啟動時,logback.xml或者logback-spring.xml加載早於applicaton.yml,所以我們在logback.xml中配置的RabbitMQ屬性無法獲取到

<property name="rabbitmq_host" source="spring.rabbitmq.host"/>
<property name="rabbitmq_vhost" source="spring.rabbitmq.virtual-host"/>
<property name="rabbitmq_username" source="spring.rabbitmq.username"/>
<property name="rabbitmq_password" source="spring.rabbitmq.password"/>

source指定的是application.yml配置文件的key

解決方案

將logback.xml或者logback-spring.xml文件自定義名稱,並在配置中心中指定該文件,這樣SpringBoot就不會在獲取配置中心配置之前加載日志配置了

配置中心的配置

#RabbitMQ配置
spring:
   rabbitmq:
     host: 127.0.0.1
     virtual-host: test
     username: admin
     password: 123

logging:
  config: classpath:logback-test.xml

日志配置

logback-test.xml

<?xml version="1.0" encoding="UTF-8"?>
<configuration scan="true" scanPeriod="60 seconds" debug="false">
    <!-- 日志存放路徑 -->
   <property name="log.path" value="./target/logs/system-service" />
    <!-- 參數 -->
    <property name="app_name" source="spring.application.name"/>
    <property name="app_instance_id" source="rabbitmq.instance"/>
    <property name="rabbitmq_host" source="spring.rabbitmq.host"/>
    <property name="rabbitmq_vhost" source="spring.rabbitmq.virtual-host"/>
    <property name="rabbitmq_username" source="spring.rabbitmq.username"/>
    <property name="rabbitmq_password" source="spring.rabbitmq.password"/>

   <!-- 日志輸出格式 -->
   <property name="log.pattern"
              value="%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{36} - [%method,%line] - %msg%n" />:ss} %-5level ${springAppName:-} %thread %logger %msg%n"/>

    <!-- 控制台輸出 -->
   <appender name="CONSOLE" class="ch.qos.logback.core.ConsoleAppender">
      <encoder>
         <pattern>${log.pattern}</pattern>
            <charset>UTF-8</charset>
      </encoder>
   </appender>

    <!-- 系統日志輸出 -->
   <appender name="FIFE" class="ch.qos.logback.core.rolling.RollingFileAppender">
       <file>${log.path}/${app_name}.log</file>
        <!-- 循環政策:基於時間創建日志文件 -->
      <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
            <!-- 日志文件名格式 -->
         <fileNamePattern>${log.path}/${app_name}.%d{yyyy-MM-dd}.%i.log</fileNamePattern>
         <!-- 日志最大的歷史 60天 -->
         <maxHistory>10</maxHistory>
            <maxFileSize>10MB</maxFileSize>
      </rollingPolicy>
        <append>true</append>
      <encoder>
         <pattern>${log.pattern}</pattern>
            <charset>UTF-8</charset>
      </encoder>
      <filter class="ch.qos.logback.classic.filter.LevelFilter">
            <!-- 過濾的級別 -->
            <level>INFO</level>
            <!-- 匹配時的操作:接收(記錄) -->
            <onMatch>ACCEPT</onMatch>
            <!-- 不匹配時的操作:拒絕(不記錄) -->
            <onMismatch>DENY</onMismatch>
        </filter>
   </appender>
    <!-- 日志發送到消息隊列RabbitMQ,接入ELK -->
    <appender name="RabbitMQ" class="org.springframework.amqp.rabbit.logback.AmqpAppender">
        <!-- 純文本,不是格式化的JSON -->
        <layout>
            <pattern>
                {
                    "appName":"${app_name}",
                    "appInstance":"${app_instance_id}",
                    "date":"%d{yyyy-MM-dd HH:mm:ss.SSS}",
                    "thread":"[%thread]",
                    "level":"%-5level",
                    "logger":"%logger{36}",
                    "msg":"%msg"
                }
            </pattern>
        </layout>
        <host>${rabbitmq_host}</host>
        <port>5672</port>
        <username>${rabbitmq_username}</username>
        <password>${rabbitmq_password}</password>
        <virtualHost>${rabbitmq_vhost}</virtualHost>
        <declareExchange>false</declareExchange>
        <exchangeType>direct</exchangeType>
        <exchangeName>logs.direct</exchangeName>
        <routingKeyPattern>logback</routingKeyPattern>
        <generateId>true</generateId>
        <durable>false</durable>
        <charset>UTF-8</charset>
        <deliveryModel>NON_PERSISTENT</deliveryModel>
        <filter class="ch.qos.logback.classic.filter.LevelFilter">
            <level>info</level>
            <onMatch>ACCEPT</onMatch>
            <onMismatch>DENY</onMismatch>
        </filter>
    </appender>

   <!--系統操作日志-->
    <root level="INFO">
        <appender-ref ref="CONSOLE" />
        <appender-ref ref="FILE" />
        <appender-ref ref="RabbitMQ" />
    </root>
</configuration>


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM