本篇內(nèi)容介紹了“如何理解集成apollo動態(tài)日志”的有關(guān)知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細(xì)閱讀,能夠?qū)W有所成!
創(chuàng)新互聯(lián)一直秉承“誠信做人,踏實做事”的原則,不欺瞞客戶,是我們最起碼的底線! 以服務(wù)為基礎(chǔ),以質(zhì)量求生存,以技術(shù)求發(fā)展,成交一個客戶多一個朋友!為您提供成都網(wǎng)站建設(shè)、成都網(wǎng)站設(shè)計、成都網(wǎng)頁設(shè)計、微信小程序定制開發(fā)、成都網(wǎng)站開發(fā)、成都網(wǎng)站制作、成都軟件開發(fā)、app軟件定制開發(fā)是成都本地專業(yè)的網(wǎng)站建設(shè)和網(wǎng)站設(shè)計公司,等你一起來見證!
動態(tài)調(diào)整線上日志級別是一個非常常見的場景,借助apollo這種配置中心組件非常容易實現(xiàn)。作為apollo的官方技術(shù)支持,博主經(jīng)常在技術(shù)群看到有使用者詢問apollo是否可以托管logback的配置文件,畢竟有了配置中心后,消滅所有的本地配置全部交給apollo管理是我們的最終目標(biāo)??墒?,apollo不具備直接托管logback-spring.xml配置文件能力,但是,我們可以基于spring和logback的裝載機(jī)制,完全取締logback-spring.xml配置,以apollo中的配置驅(qū)動。而且,改造后,大大提高了日志系統(tǒng)的靈活性和可擴(kuò)展性。
何為apollo動態(tài)日志?直接這樣說可能會有歧義,以為是apollo里的日志,其實不然。舉個簡單的例子,比如,我們項目很多地方使用了log.debug()打印日志,為了方便通過日志信息排查問題,但是一般情況下,生產(chǎn)環(huán)境的日志級別會配置成info。只有遇到需要排查線上問題的時候才會臨時打開debug級別日志。這個時候只能需改配置文件,將日志級別調(diào)整成debug,然后重新打包部署驗證。不僅流程繁瑣耗時,還會破壞當(dāng)時的"案發(fā)現(xiàn)場的環(huán)境",導(dǎo)致判斷不準(zhǔn)確。如果應(yīng)用具備了apollo動態(tài)日志這種能力,就只需在apollo修改下配置然后提交,就可以熱更新日志級別,馬上打印debug級別日志。這就是所謂的apollo動態(tài)日志。實現(xiàn)這個效果,需要具備兩個能力,分別由spring和apollo提供
spring應(yīng)用中,spring適配了主流的日志框架,如logback、log4j2等,在這些日志框架之上,又抽象了自己的日志系統(tǒng)服務(wù),這里我們用到了spring的LoggingSystem,用它來熱更新日志級別,這個類在日志系統(tǒng)初始化時就添加到了spring的容器中,所以只要在spring的上下文管理范圍內(nèi),就可以直接注入,以下為主要使用到的api描述:
/** * 設(shè)置給定日志記錄器的日志級別. * @param loggerName 要設(shè)置的日志記錄器的名稱({@code null}可用于根日志記錄器)。 * @param level 日志級別 */ public void setLogLevel(String loggerName, LogLevel level) { throw new UnsupportedOperationException("Unable to set log level"); }
apollo作為分布式配置中心,配置集中管理和配置熱更新是其最核心的功能,此外,apollo還提供了配置變更下發(fā)監(jiān)聽的功能?;谶@個配置監(jiān)聽的設(shè)計,實現(xiàn)動態(tài)日志就變得非常簡單了。而且不僅可以實現(xiàn)日志動態(tài)熱更,基于這個思路,連接池、數(shù)據(jù)源等都可以輕松實現(xiàn)。apollo實現(xiàn)監(jiān)聽配置變更有多種方式,可以通過Config實例手動添加,如:
@ApolloConfig public Config config; public void addConfigChangeListener(){ config.addChangeListener(changeEvent->{ System.out.println("config change keys" + changeEvent.changedKeys()); }); }
也可以通過注解直接驅(qū)動
@ApolloConfigChangeListener public void addConfigChangeListener(ConfigChangeEvent changeEvent){ System.out.println("config change keys" + changeEvent.changedKeys()); }
有了上述能力,在結(jié)合spring支持的日志加載配置方式,如:
logging.level.org.springframework.web=debug
logging.level.org.hibernate=error
可以實現(xiàn)如下代碼完成功能,遇到需要調(diào)整日志級別時,修改apollo里的配置,即可實時生效
@Configuration public class LogbackConfiguration { private static final Logger logger = LoggerFactory.getLogger(LoggerConfiguration.class); private static final String LOGGER_TAG = "logging.level."; private final LoggingSystem loggingSystem; public LogbackConfiguration(LoggingSystem loggingSystem) { this.loggingSystem = loggingSystem; } @ApolloConfigChangeListener private void onChange(ConfigChangeEvent changeEvent) { for (String key : changeEvent.changedKeys()) { if (this.containsIgnoreCase(key, LOGGER_TAG)) { String strLevel = changeEvent.getChange(key).getNewValue(); LogLevel level = LogLevel.valueOf(strLevel.toUpperCase()); loggingSystem.setLogLevel(key.replace(LOGGER_TAG, ""), level); logger.info("logging changed: {},oldValue:{},newValue:{}", key, changeEvent.getChange(key).getOldValue(), strLevel); } } } private boolean containsIgnoreCase(String str, String searchStr) { if (str == null || searchStr == null) { return false; } int len = searchStr.length(); int max = str.length() - len; for (int i = 0; i <= max; i++) { if (str.regionMatches(true, i, searchStr, 0, len)) { return true; } } return false; } }
在"消滅"logback-xml配置之前,先看下這個配置文件有哪些配置信息,起到了哪些作用,下面貼出一個典型的配置文件內(nèi)容:
ERROR
一個典型的logback配置文件里包含了Appender和日志級別設(shè)置的信息,Appender可以理解為日志的輸出源。如上貼出的這個配置,添加了兩個Appender信息,一個是spring中內(nèi)置的,將日志輸出到控制臺的Appender。一個是將error日志信息發(fā)送到Sentry應(yīng)用監(jiān)控平臺的Appender。其他的配置描述了每個包路徑不同的日志級別信息。到這里,我們很容易想到,上文已經(jīng)說過,spring已經(jīng)支持以logging.level.包名=info這種配置來設(shè)置日志系統(tǒng)的日志級別。那么剩下的只要解決Appender的配置就ok了。在這里,其實只需要解決SentryAppender的加載就行,因為consoleAppender spring自己會處理。有了目標(biāo)和方向,就好辦了。以logback-spring.xml配置的信息,最終都會加載成class對象。就和spring.xml配置一樣。所以研究的方向就變成了Logback的加載原理的問題。
在java的日志生態(tài)里,除了響當(dāng)當(dāng)?shù)膌ogback、log4j2、apache common log外,還有一個日志框架不得不提,就是sl4j。正因為java生態(tài)強(qiáng)大,日志框架層出不窮,所以sl4j出來了,不干實事,專門定義日志標(biāo)準(zhǔn)、規(guī)范定義接口。而且,在我們平時的編碼過程中,也建議使用sl4j的api,這樣,無論底層日志框架實現(xiàn)怎么切換,都不會影響。主流的日志框架都有實現(xiàn)sl4j的接口,spring中日志系統(tǒng)的加載也是面向的sl4j,而不是直接面向日志實現(xiàn),加載過程是一個自動化的過程,系統(tǒng)會自動掃描實現(xiàn)了sl4j的接口實現(xiàn),如:
public interface ILoggerFactory { public Logger getLogger(String name); }
每個日志框架都會實現(xiàn)這個接口,如Logback中的LoggerContext。Logback所有的功能都集成在了這個Context中,logback-spring.xml的配置也是為了配置LoggerContext中的屬性信息,所有我們只要拿到了LoggerContext實例,問題就解決了一大半。這涉及到sl4j的另一個接口,獲取ILoggerFactory實例的接口:
public interface LoggerFactoryBinder { public ILoggerFactory getLoggerFactory(); public String getLoggerFactoryClassStr(); }
Logback的實現(xiàn)類為StaticLoggerBinder,也就是說,我們可以通過StaticLoggerBinder的getLoggerFactory方法拿到LoggerContext實例了。
拿到Logback的LoggerContext后,就好辦了,見代碼:
@Configuration public class LogbackConfiguration { private final LoggerContext ctx = (LoggerContext) StaticLoggerBinder.getSingleton().getLoggerFactory(); @Bean @Profile(PROD_ENV) public void initSenTry() { SentryAppender sentryAppender = new SentryAppender(); sentryAppender.setContext(ctx); ThresholdFilter filter = new ThresholdFilter(); filter.setLevel(Level.ERROR.levelStr); filter.start(); sentryAppender.addFilter(filter); sentryAppender.start(); ctx.addTurboFilter(new TurboFilter() { @Override public FilterReply decide(Marker marker, ch.qos.logback.classic.Logger logger, Level level, String format, Object[] params, Throwable t) { logger.addAppender(sentryAppender); return FilterReply.NEUTRAL; } }); } }
看到這種代碼就非常有感覺了,配置文件中的xml其實就是描述了日志組成對象以及對象的屬性。在使用java bean的方式配置時需要注意,Logback的設(shè)計里,每個日志系統(tǒng)組成實例都有一個start狀態(tài)屬性,上面的start()方法其實不是動作,只是標(biāo)記了這個屬性為true。而在xml里這個屬性只要配置了就自動激活為true了,這里必須顯示的start()一下。解決了日志級別配置和Appender配置后,Logback-spring.xml文件就可以徹底的刪除了
“如何理解集成apollo動態(tài)日志”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識可以關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實用文章!