先描述一下問題,多個服務(wù)器實現(xiàn)的負(fù)載均衡,每個服務(wù)器存儲在自己的硬盤里。但是現(xiàn)在需要對日志做統(tǒng)一的分析,在多個服務(wù)器上統(tǒng)計就麻煩了。思路是把日志統(tǒng)一到一臺日志服務(wù)器上,再統(tǒng)一做統(tǒng)計分析。怎么統(tǒng)一到一臺服務(wù)器上,說實話沒有特別好的思路,最后嘗試了log4j的SocketAppender。查了不少網(wǎng)絡(luò)資源,都說的有些不明了,還是得親自嘗試之后才見分曉。
1、客戶端的配置:
客戶端的配置比較簡單,只需要告訴log4j需要監(jiān)聽哪個遠(yuǎn)程服務(wù)器的哪個端口即可。直接在log4j.properties里直接配置就好。
- <span style="font-size:12px;">log4j.appender.logs=org.apache.log4j.DailyRollingFileAppender
- log4j.appender.logs.File = /data/logs/request/logs.log
- log4j.appender.logs.layout = org.apache.log4j.PatternLayout
- log4j.appender.logs.layout.ConversionPattern=%d [%t] - %m%n
- log4j.appender.logs.DatePattern='.'yyyy-MM-dd'.log'
-
- log4j.appender.socket=org.apache.log4j.net.SocketAppender
- log4j.appender.socket.RemoteHost=172.16.2.152
- log4j.appender.socket.Port=4560
- log4j.appender.socket.LocationInfo=true
- #下面這兩句感覺沒用
- log4j.appender.socket.layout=org.apache.log4j.PatternLayout
- log4j.appender.socket.layout.ConversionPattern=[%-5p] %d{yyyy-MM-dd HH:mm:ss,SSS} method:%l%n%t%m%n
-
- #將日志寫入本地和遠(yuǎn)程日志服務(wù)器
- log4j.logger.com.test.core.filter =DEBUG,socket,logs</span>
2、日志服務(wù)器的配置:
日志服務(wù)器需要單獨(dú)啟動一個java進(jìn)程,接收客戶端給自己發(fā)送的socket請求。Log4j提供了org.apache.log4j.net.SocketServer類,直接運(yùn)行其main函數(shù)就行了(當(dāng)然也可以自己寫啦)。
java -cp /log4jsocket/serverConfig/log4j-1.2.16.jarorg.apache.log4j.net.SocketServer 4560 /log4jsocket/log4jserver.properties /log4jsocket/clientConfig
/log4jsocket/serverConfig/log4j-1.2.16.jar是log4j jar包存放的位置,org.apache.log4j.net.SocketServer需要三個參數(shù):
1)4560 是監(jiān)聽的端口號
2)/log4jsocket/log4jserver.properties 是記錄日志服務(wù)器的日志的配置文件
3)/log4jsocket/clientConfig 是客戶端配置文件所在的目錄(注意是目錄)。
著重說一下org.apache.log4j.net.SocketServer的第三個參數(shù),這個文件夾下配置的是各個客戶端的日志的配置。配置文件以.lcf結(jié)尾,文件名可以用客戶端的IP命名,log4j會自己找發(fā)送請求的客戶端IP對應(yīng)的那個配置文件,如172.16.2.46服務(wù)器發(fā)送的socket請求會尋找172.16.2.46.lcf配置文件,并根據(jù)配置將日志寫入對應(yīng)的文件。
- <span style="font-size:12px;">#注意logger后面的值要與client的值相同
- log4j.logger.com.test.core.filter=DEBUG,localLogs
-
- log4j.appender.localLogs=org.apache.log4j.DailyRollingFileAppender
- log4j.appender.localLogs.File=/data/logs/request/172.16.2.46/logs.log
- log4j.appender.localLogs.layout=org.apache.log4j.PatternLayout
- log4j.appender.localLogs.layout.ConversionPattern=%d [%t] - %m%n
- log4j.appender.localLogs.DatePattern='.'yyyy-MM-dd'.log'
- </span>
這樣做的好處是可以根據(jù)不同客戶端,將日志寫入不同的文件夾下的。
其實,配置過程就這么簡單,但是當(dāng)你這么做之后,你會發(fā)現(xiàn)運(yùn)行org.apache.log4j.net.SocketServer后,客戶端向日志服務(wù)器發(fā)送請求時,會報找不到.lcf文件的錯誤,得不到想要的結(jié)果。原因出在org.apache.log4j.net.SocketServer代碼中的一個小bug。
- <span style="font-size:12px;">LoggerRepository configureHierarchy(InetAddress inetAddress)
- {
- cat.info("Locating configuration file for " + inetAddress);
-
- String s = inetAddress.toString();
- int i = s.indexOf("/");
- if (i == -1) {
- cat.warn("Could not parse the inetAddress [" + inetAddress + "]. Using default hierarchy.");
-
- return genericHierarchy();
- }
- String key = s.substring(0,i);
-
- File configFile = new File(this.dir, key + CONFIG_FILE_EXT);
- if (configFile.exists()) {
- Hierarchy h = new Hierarchy(new RootLogger(Level.DEBUG));
- this.hierarchyMap.put(inetAddress, h);
-
- new PropertyConfigurator().doConfigure(configFile.getAbsolutePath(), h);
-
- return h;
- }
- cat.warn("Could not find config file [" + configFile + "].");
- return genericHierarchy();
- }</span>
String key = s.substring(0, i);換成String key = s.substring(i+1);就好了。這段代碼是解析IP地址,然后尋找對應(yīng)IP命名的.lcf配置文件;如果找不到,則解析默認(rèn)的generic.lcf。由于截取的錯誤,導(dǎo)致找不到172.16.2.46.lcf,文件夾下又沒有g(shù)eneric.lcf,所以會拋異常。
org.apache.log4j.net.SocketServer代碼中的另外一個bug是,只能接收來自一臺客戶端的日志請求,一旦客戶端停止運(yùn)行,SocketServer也將關(guān)閉。查看代碼:
- public static void main(String[] argv)
- {
- if (argv.length == 3)
- init(argv[0], argv[1], argv[2]);
- else
- usage("Wrong number of arguments.");
- try
- {
- cat.info("Listening on port " + port);
- ServerSocket serverSocket = new ServerSocket(port);
- cat.info("Waiting to accept a new client.");
- Socket socket = serverSocket.accept();
- InetAddress inetAddress = socket.getInetAddress();
- cat.info("Connected to client at " + inetAddress);
-
- LoggerRepository h = (LoggerRepository)server.hierarchyMap.get(inetAddress);
- if (h == null) {
- h = server.configureHierarchy(inetAddress);
- }
-
- cat.info("Starting new socket node.");
- new Thread(new SocketNode(socket, h)).start();
- }
- catch (Exception e)
- {
- e.printStackTrace();
- }
- }
問題出在只建立了一個socket連接就不在accept了,加上while循環(huán)問題就解決了。
- ServerSocket serverSocket = new ServerSocket(port);
- while(true){
- cat.info("Waiting to accept a new client.");
- Socket socket = serverSocket.accept();
- InetAddress inetAddress = socket.getInetAddress();
- cat.info("Connected to client at " + inetAddress);
-
- LoggerRepository h = (LoggerRepository)server.hierarchyMap.get(inetAddress);
- if (h == null) {
- h = server.configureHierarchy(inetAddress);
- }
-
- cat.info("Starting new socket node.");
- new Thread(new SocketNode(socket, h)).start();
- }
好了。Log4j的配置到此結(jié)束。
最后一個問題,日志服務(wù)器是linux,需要有一個統(tǒng)一的start、shutdown命令來啟動和關(guān)閉org.apache.log4j.net.SocketServer。那就需要些shell命令了,下面這段代碼參考了http://www.cnblogs.com/baibaluo/archive/2011/08/31/2160934.html
catalina.sh
startup.sh- <span style="font-size:12px;">#!/bin/sh
- EXECUTABLE=/log4jsocket/catalina.sh
- exec "$EXECUTABLE" start "$@"</span>
shutdown.sh- <span style="font-size:12px;">EXECUTABLE=/log4jsocket/catalina.sh
- exec "$EXECUTABLE" stop "$@"</span>
PS: csdn博客啥時可以上傳附件啊
版權(quán)聲明:本文為博主原創(chuàng)文章,未經(jīng)博主允許不得轉(zhuǎn)載。