spring boot自帶logback作為其日志新系統,但是在實際工作中,常常需要對日志進行管理或分析,
如果只是單純的將日志導入文本文件,則在查詢時操作過於繁瑣,
如果將其導入mysql等關系型數據庫進行存儲,又太影響系統性能,同時由於Mysql其結構化的信息存儲結構,導致在存儲時不夠靈活。
因此,在此考慮將springboot系統中產出的日志(logback) 存入mongodb中
1.pom.xml 引入依賴
注意排除掉log4j的依賴
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-mongodb</artifactId>
<version>1.5.8.RELEASE</version>
</dependency>
<!-- https://mvnrepository.com/artifact/ch.qos.logback/logback-core -->
<dependency>
<groupId>ch.qos.logback</groupId>
<artifactId>logback-core</artifactId>
<version>1.2.3</version>
</dependency>
<!-- https://mvnrepository.com/artifact/ch.qos.logback/logback-classic -->
<dependency>
<groupId>ch.qos.logback</groupId>
<artifactId>logback-classic</artifactId>
<version>1.2.3</version>
</dependency>
<!-- log4j 記錄日志-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-log4j</artifactId>
<version>1.3.8.RELEASE</version>
<exclusions>
<exclusion>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-log4j12</artifactId>
</exclusion>
</exclusions>
</dependency>
<!--AOP-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
<version>1.5.7.RELEASE</version>
</dependency>

2.創建實體類: logback.MyLog.java
package com.wutongshu.springboot.logback; import java.io.Serializable; import java.util.Date; public class MyLog implements Serializable { private String id; private String msg; private Date time; private String threadName; private String level; public String getId() { return id; } public void setId(String id) { this.id = id; } public String getMsg() { return msg; } public void setMsg(String msg) { this.msg = msg; } public Date getTime() { return time; } public void setTime(Date time) { this.time = time; } public String getThreadName() { return threadName; } public void setThreadName(String threadName) { this.threadName = threadName; } public String getLevel() { return level; } public void setLevel(String level) { this.level = level; } }
3.添加數據訪問接口: LogRepository.java
package com.wutongshu.springboot.logback; import org.springframework.data.mongodb.repository.MongoRepository; public interface LogRepository extends MongoRepository<MyLog,String> { }
4.Appender 類: MongoDBAppender.java
package com.wutongshu.springboot.logback; import ch.qos.logback.classic.spi.LoggingEvent; import ch.qos.logback.core.UnsynchronizedAppenderBase; import org.springframework.context.ApplicationContext; import org.springframework.context.ApplicationContextAware; import org.springframework.stereotype.Component; import java.util.Date; @Component public class MongoDBAppender extends UnsynchronizedAppenderBase<LoggingEvent> implements ApplicationContextAware { private static LogRepository logRepository; @Override public void start() { super.start(); } @Override public void stop() { super.stop(); } @Override protected void append(LoggingEvent e) { MyLog myLog = new MyLog(); myLog.setLevel(e.getLevel().toString()); myLog.setMsg(e.getFormattedMessage()); myLog.setThreadName(e.getThreadName()); myLog.setTime(new Date(e.getTimeStamp())); logRepository.save(myLog); } @Override public void setApplicationContext(ApplicationContext applicationContext) { if (applicationContext.getAutowireCapableBeanFactory().getBean(LogRepository.class) != null) { logRepository = (LogRepository) applicationContext.getAutowireCapableBeanFactory().getBean(LogRepository.class); } } }
5.創建切面類記錄日志信息
logger取名為MONGODB
通過getBasicDBObject函數從HttpServletRequest和JoinPoint對象中獲取請求信息,並組裝成BasicDBObject
getHeadersInfo函數從HttpServletRequest中獲取header信息
通過logger.info(),輸出BasicDBObject對象的信息到mongodb
package com.wutongshu.springboot.logback; import com.mongodb.BasicDBObject; import org.apache.log4j.Logger; import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.AfterReturning; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut; import org.springframework.core.annotation.Order; import org.springframework.stereotype.Component; import org.springframework.web.context.request.RequestContextHolder; import org.springframework.web.context.request.ServletRequestAttributes; import javax.servlet.http.HttpServletRequest; import java.util.Arrays; import java.util.Enumeration; import java.util.HashMap; import java.util.Map; /** * 日志切面類 */ @Aspect @Component @Order(5) public class WebLogAspect { private Logger logger=Logger.getLogger("MONGODB"); private ThreadLocal<Long> startTime=new ThreadLocal<>(); @Pointcut("execution(public * com.*.*.web.*.*(..))") public void webLog(){ } @Before(value = "webLog()") public void doBefore(JoinPoint point){ startTime.set(System.currentTimeMillis()); logger.info("WebLogAspect.doBefore............"); ServletRequestAttributes attributes= (ServletRequestAttributes) RequestContextHolder.getRequestAttributes(); HttpServletRequest request=attributes.getRequest(); // 獲取要記錄的日志內容 BasicDBObject logInfo = getBasicDBObject(request, point); logger.info(logInfo); } private BasicDBObject getBasicDBObject(HttpServletRequest request, JoinPoint point) { // 基本信息 BasicDBObject r = new BasicDBObject(); r.append("requestURL", request.getRequestURL().toString()); r.append("requestURI", request.getRequestURI()); r.append("queryString", request.getQueryString()); r.append("remoteAddr", request.getRemoteAddr()); r.append("remoteHost", request.getRemoteHost()); r.append("remotePort", request.getRemotePort()); r.append("localAddr", request.getLocalAddr()); r.append("localName", request.getLocalName()); r.append("method", request.getMethod()); r.append("headers", getHeadersInfo(request)); r.append("parameters", request.getParameterMap()); r.append("classMethod", point.getSignature().getDeclaringTypeName() + "." + point.getSignature().getName()); r.append("args", Arrays.toString(point.getArgs())); return r; } /** * 獲取頭信息 * * @param request * @return */ private Map<String, String> getHeadersInfo(HttpServletRequest request) { Map<String, String> map = new HashMap<>(); Enumeration headerNames = request.getHeaderNames(); while (headerNames.hasMoreElements()) { String key = (String) headerNames.nextElement(); String value = request.getHeader(key); map.put(key, value); } return map; } }
6.創建logback.xml
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE xml> <configuration scan="true" scanPeriod="3600 seconds" debug="false"> <property name="logDir" value="logs"/> <appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender"> <encoder> <charset>UTF-8</charset> <pattern>%d [%thread] %-5level %logger{68} %line - logId[[%X{client}][%X{request_id}]] - %msg%n</pattern> </encoder> </appender> <appender name="RollingFile" class="ch.qos.logback.core.rolling.RollingFileAppender"> <filter class="ch.qos.logback.classic.filter.ThresholdFilter"> <level>TRACE</level> </filter> <!-- 可讓每天產生一個日志文件,最多 10 個,自動回滾 --> <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy"> <fileNamePattern>${logDir}/file-%d{yyyy-MM-dd}.log</fileNamePattern> <maxHistory>10</maxHistory> </rollingPolicy> <encoder> <pattern>%d{HH:mm:ss.SSS} [%thread] %-5level %logger - %msg%n</pattern> <charset>UTF-8</charset> </encoder> </appender> <!--appender類的路徑--> <appender name="MONGODB" class="com.wutongshu.springboot.logback.MongoDBAppender"/> <root level="INFO"> <appender-ref ref="STDOUT"/> <appender-ref ref="RollingFile"/> <appender-ref ref="MONGODB"/> </root> </configuration>
7.在application.properties里添加mongodb的uri

啟動MongoDB數據庫,可看到多了一個名為logs的database

在java的項目結構上也多了個logs的文件夾

