自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Spring Boot 記錄請(qǐng)求響應(yīng)日志的常用手段

開(kāi)發(fā) 架構(gòu)
某些業(yè)務(wù)需求需要追蹤我們的接口訪問(wèn)情況,也就是把請(qǐng)求和響應(yīng)記錄下來(lái)?;镜挠涗浘S度包含了請(qǐng)求入?yún)?路徑query參數(shù),請(qǐng)求體)、請(qǐng)求路徑(uri)、請(qǐng)求方法(method)、請(qǐng)求頭(headers)以及響應(yīng)狀態(tài)、響應(yīng)頭、甚至包含了敏感的響應(yīng)體等等。今天總結(jié)了幾種方法,你可以按需選擇。

[[416714]]

某些業(yè)務(wù)需求需要追蹤我們的接口訪問(wèn)情況,也就是把請(qǐng)求和響應(yīng)記錄下來(lái)?;镜挠涗浘S度包含了請(qǐng)求入?yún)?路徑query參數(shù),請(qǐng)求體)、請(qǐng)求路徑(uri)、請(qǐng)求方法(method)、請(qǐng)求頭(headers)以及響應(yīng)狀態(tài)、響應(yīng)頭、甚至包含了敏感的響應(yīng)體等等。今天總結(jié)了幾種方法,你可以按需選擇。

請(qǐng)求追蹤的實(shí)現(xiàn)方式

網(wǎng)關(guān)層

很多網(wǎng)關(guān)設(shè)施都具有httptrace的功能,可以幫助我們集中記錄請(qǐng)求流量的情況。Orange、Kong、Apache Apisix這些基于Nginx的網(wǎng)關(guān)都具有該能力,就連Nginx本身也提供了記錄httptrace日志的能力。

優(yōu)點(diǎn)是可以集中的管理httptrace日志,免開(kāi)發(fā);缺點(diǎn)是技術(shù)要求高,需要配套的分發(fā)、存儲(chǔ)、查詢的設(shè)施。

Spring Boot Actuator

在Spring Boot中,其實(shí)提供了簡(jiǎn)單的追蹤功能。你只需要集成:

  1. <dependency> 
  2.     <groupId>org.springframework.boot</groupId> 
  3.     <artifactId>spring-boot-starter-actuator</artifactId> 
  4. </dependency> 

開(kāi)啟/actuator/httptrace:

  1. management: 
  2.   endpoints: 
  3.     web: 
  4.       exposure: 
  5.         include: 'httptrace' 

就可以通過(guò)http://server:port/actuator/httptrace獲取最近的Http請(qǐng)求信息了。

不過(guò)在最新的版本中可能需要顯式的聲明這些追蹤信息的存儲(chǔ)方式,也就是實(shí)現(xiàn)HttpTraceRepository接口并注入Spring IoC。

例如放在內(nèi)存中并限制為最近的100條(不推薦生產(chǎn)使用):

  1. @Bean 
  2. public HttpTraceRepository httpTraceRepository(){ 
  3.     return new InMemoryHttpTraceRepository(); 

追蹤日志以json格式呈現(xiàn):

Spring Boot Actuator記錄的httptrace

記錄的維度不多,當(dāng)然如果夠用的話可以試試。

優(yōu)點(diǎn)在于集成起來(lái)簡(jiǎn)單,幾乎免除開(kāi)發(fā);缺點(diǎn)在于記錄的維度不多,而且需要搭建緩沖消費(fèi)這些日志信息的設(shè)施。

CommonsRequestLoggingFilter

Spring Web模塊還提供了一個(gè)過(guò)濾器CommonsRequestLoggingFilter,它可以對(duì)請(qǐng)求的細(xì)節(jié)進(jìn)行日志輸出。配置起來(lái)也比較簡(jiǎn)單:

  1. @Bean 
  2. CommonsRequestLoggingFilter  loggingFilter(){ 
  3.     CommonsRequestLoggingFilter loggingFilter = new CommonsRequestLoggingFilter(); 
  4.     // 記錄 客戶端 IP信息 
  5.     loggingFilter.setIncludeClientInfo(true); 
  6.     // 記錄請(qǐng)求頭 
  7.     loggingFilter.setIncludeHeaders(true); 
  8.     // 如果記錄請(qǐng)求頭的話,可以指定哪些記錄,哪些不記錄 
  9.     // loggingFilter.setHeaderPredicate(); 
  10.     // 記錄 請(qǐng)求體  特別是POST請(qǐng)求的body參數(shù) 
  11.     loggingFilter.setIncludePayload(true); 
  12.     // 請(qǐng)求體的大小限制 默認(rèn)50 
  13.     loggingFilter.setMaxPayloadLength(10000); 
  14.     //記錄請(qǐng)求路徑中的query參數(shù)  
  15.     loggingFilter.setIncludeQueryString(true); 
  16.     return loggingFilter; 

而且必須開(kāi)啟對(duì)CommonsRequestLoggingFilter的debug日志:

  1. logging: 
  2.   level
  3.     org: 
  4.       springframework: 
  5.         web: 
  6.           filter: 
  7.             CommonsRequestLoggingFilter: debug 

一次請(qǐng)求會(huì)輸出兩次日志,一次是在第一次經(jīng)過(guò)過(guò)濾器前;一次是完成過(guò)濾器鏈后。

CommonsRequestLoggingFilter記錄請(qǐng)求日志

這里多說(shuō)一句其實(shí)可以改造成輸出json格式的。

優(yōu)點(diǎn)是靈活配置、而且對(duì)請(qǐng)求追蹤的維度全面,缺點(diǎn)是只記錄請(qǐng)求而不記錄響應(yīng)。

ResponseBodyAdvice

Spring Boot統(tǒng)一返回體其實(shí)也能記錄,需要自行實(shí)現(xiàn)。這里借鑒了CommonsRequestLoggingFilter解析請(qǐng)求的方法。響應(yīng)體也可以獲取了,不過(guò)響應(yīng)頭和狀態(tài)因?yàn)樯芷谶€不清楚,這里獲取還不清楚是否合適,不過(guò)這是一個(gè)思路。

  1. /** 
  2.  * @author felord.cn 
  3.  * @since 1.0.8.RELEASE 
  4.  */ 
  5. @Slf4j 
  6. @RestControllerAdvice(basePackages = {"cn.felord.logging"}) 
  7. public class RestBodyAdvice implements ResponseBodyAdvice<Object> { 
  8.     private static final int DEFAULT_MAX_PAYLOAD_LENGTH = 10000; 
  9.     public static final String REQUEST_MESSAGE_PREFIX = "Request ["
  10.     public static final String REQUEST_MESSAGE_SUFFIX = "]"
  11.     private ObjectMapper objectMapper = new ObjectMapper(); 
  12.  
  13.     @Override 
  14.     public boolean supports(MethodParameter returnType, Class<? extends HttpMessageConverter<?>> converterType) { 
  15.         return true
  16.     } 
  17.  
  18.     @SneakyThrows 
  19.     @Override 
  20.     public Object beforeBodyWrite(Object body, 
  21.                                   MethodParameter returnType, 
  22.                                   MediaType selectedContentType, 
  23.                                   Class<? extends HttpMessageConverter<?>> selectedConverterType, 
  24.                                   ServerHttpRequest request, 
  25.                                   ServerHttpResponse response) { 
  26.  
  27.         ServletServerHttpRequest servletServerHttpRequest = (ServletServerHttpRequest) request; 
  28.  
  29.         log.debug(createRequestMessage(servletServerHttpRequest.getServletRequest(), REQUEST_MESSAGE_PREFIX, REQUEST_MESSAGE_SUFFIX)); 
  30.         Rest<Object> objectRest; 
  31.         if (body == null) { 
  32.             objectRest = RestBody.okData(Collections.emptyMap()); 
  33.         } else if (Rest.class.isAssignableFrom(body.getClass())) { 
  34.             objectRest = (Rest<Object>) body; 
  35.         } 
  36.         else if (checkPrimitive(body)) { 
  37.             return RestBody.okData(Collections.singletonMap("result", body)); 
  38.         }else { 
  39.             objectRest = RestBody.okData(body); 
  40.         } 
  41.         log.debug("Response Body ["+ objectMapper.writeValueAsString(objectRest) +"]"); 
  42.         return objectRest; 
  43.     } 
  44.  
  45.  
  46.     private boolean checkPrimitive(Object body) { 
  47.         Class<?> clazz = body.getClass(); 
  48.         return clazz.isPrimitive() 
  49.                 || clazz.isArray() 
  50.                 || Collection.class.isAssignableFrom(clazz) 
  51.                 || body instanceof Number 
  52.                 || body instanceof Boolean 
  53.                 || body instanceof Character 
  54.                 || body instanceof String; 
  55.     } 
  56.  
  57.  
  58.     protected String createRequestMessage(HttpServletRequest request, String prefix, String suffix) { 
  59.         StringBuilder msg = new StringBuilder(); 
  60.         msg.append(prefix); 
  61.         msg.append(request.getMethod()).append(" "); 
  62.         msg.append(request.getRequestURI()); 
  63.  
  64.  
  65.         String queryString = request.getQueryString(); 
  66.         if (queryString != null) { 
  67.             msg.append('?').append(queryString); 
  68.         } 
  69.  
  70.  
  71.         String client = request.getRemoteAddr(); 
  72.         if (StringUtils.hasLength(client)) { 
  73.             msg.append(", client=").append(client); 
  74.         } 
  75.         HttpSession session = request.getSession(false); 
  76.         if (session != null) { 
  77.             msg.append(", session=").append(session.getId()); 
  78.         } 
  79.         String user = request.getRemoteUser(); 
  80.         if (user != null) { 
  81.             msg.append(", user=").append(user); 
  82.         } 
  83.  
  84.         HttpHeaders headers = new ServletServerHttpRequest(request).getHeaders(); 
  85.         msg.append(", headers=").append(headers); 
  86.  
  87.         String payload = getMessagePayload(request); 
  88.         if (payload != null) { 
  89.             msg.append(", payload=").append(payload); 
  90.         } 
  91.  
  92.         msg.append(suffix); 
  93.         return msg.toString(); 
  94.     } 
  95.  
  96.     protected String getMessagePayload(HttpServletRequest request) { 
  97.         ContentCachingRequestWrapper wrapper = 
  98.                 WebUtils.getNativeRequest(request, ContentCachingRequestWrapper.class); 
  99.         if (wrapper != null) { 
  100.             byte[] buf = wrapper.getContentAsByteArray(); 
  101.             if (buf.length > 0) { 
  102.                 int length = Math.min(buf.length, DEFAULT_MAX_PAYLOAD_LENGTH); 
  103.                 try { 
  104.                     return new String(buf, 0, length, wrapper.getCharacterEncoding()); 
  105.                 } catch (UnsupportedEncodingException ex) { 
  106.                     return "[unknown]"
  107.                 } 
  108.             } 
  109.         } 
  110.         return null
  111.     } 

別忘記配置ResponseBodyAdvice的logging級(jí)別為DEBUG。

logstash-logback-encoder

這個(gè)是logstash的logback編碼器,可以結(jié)構(gòu)化輸出httptrace為json。引入:

  1. <dependency> 
  2.     <groupId>net.logstash.logback</groupId> 
  3.     <artifactId>logstash-logback-encoder</artifactId> 
  4.     <version>6.6</version> 
  5. </dependency> 

配置logback的ConsoleAppender為L(zhǎng)ogstashEncoder:

  1. <configuration> 
  2.     <appender name="jsonConsoleAppender" class="ch.qos.logback.core.ConsoleAppender"
  3.         <encoder class="net.logstash.logback.encoder.LogstashEncoder"/> 
  4.     </appender> 
  5.     <root level=" INFO"
  6.         <appender-ref ref="jsonConsoleAppender"/> 
  7.     </root> 
  8. </configuration> 

然后同樣實(shí)現(xiàn)一個(gè)解析的Filter:

  1. import org.slf4j.Logger; 
  2. import org.slf4j.LoggerFactory; 
  3. import org.slf4j.MDC; 
  4. import org.springframework.core.annotation.Order
  5. import org.springframework.stereotype.Component; 
  6.  
  7. import javax.servlet.*; 
  8. import javax.servlet.http.HttpServletRequest; 
  9. import javax.servlet.http.HttpServletResponse; 
  10. import java.io.IOException; 
  11. import java.util.UUID; 
  12.  
  13. /** 
  14.  * @author felord.cn 
  15.  * @since 1.0.8.RELEASE 
  16.  */ 
  17. @Order(1) 
  18. @Component 
  19. public class MDCFilter implements Filter { 
  20.  
  21.     private final Logger LOGGER = LoggerFactory.getLogger(MDCFilter.class); 
  22.     private final String X_REQUEST_ID = "X-Request-ID"
  23.  
  24.     @Override 
  25.     public void doFilter(ServletRequest request, 
  26.                          ServletResponse response, 
  27.                          FilterChain chain) throws IOException, ServletException { 
  28.         HttpServletRequest req = (HttpServletRequest) request; 
  29.         HttpServletResponse res = (HttpServletResponse) response; 
  30.         try { 
  31.             addXRequestId(req); 
  32.             LOGGER.info("path: {}, method: {}, query {}"
  33.                     req.getRequestURI(), req.getMethod(), req.getQueryString()); 
  34.             res.setHeader(X_REQUEST_ID, MDC.get(X_REQUEST_ID)); 
  35.             chain.doFilter(request, response); 
  36.         } finally { 
  37.             LOGGER.info("statusCode {}, path: {}, method: {}, query {}"
  38.                     res.getStatus(), req.getRequestURI(), req.getMethod(), req.getQueryString()); 
  39.             MDC.clear(); 
  40.         } 
  41.     } 
  42.  
  43.     private void addXRequestId(HttpServletRequest request) { 
  44.         String xRequestId = request.getHeader(X_REQUEST_ID); 
  45.         if (xRequestId == null) { 
  46.             MDC.put(X_REQUEST_ID, UUID.randomUUID().toString()); 
  47.         } else { 
  48.             MDC.put(X_REQUEST_ID, xRequestId); 
  49.         } 
  50.     } 
  51.  

這里解析方式其實(shí)還可以更加精細(xì)一些。

然后所有的日志都可以結(jié)構(gòu)化為json了:

  1. {"@timestamp":"2021-08-10T23:48:51.322+08:00","@version":"1","message":"statusCode 200, path: /log/get, method: GET, query foo=xxx&bar=ooo","logger_name":"cn.felord.logging.MDCFilter","thread_name":"http-nio-8080-exec-1","level":"INFO","level_value":20000,"X-Request-ID":"7c0db56c-b1f2-4d85-ad9a-7ead67660f96"

總結(jié)

今天介紹了不少記錄追蹤接口請(qǐng)求響應(yīng)的方法,總有一款適合你。

本文轉(zhuǎn)載自微信公眾號(hào)「碼農(nóng)小胖哥」,可以通過(guò)以下二維碼關(guān)注。轉(zhuǎn)載本文請(qǐng)聯(lián)系碼農(nóng)小胖哥公眾號(hào)。

 

責(zé)任編輯:武曉燕 來(lái)源: 碼農(nóng)小胖哥
相關(guān)推薦

2021-03-01 23:26:41

日志Spring BootAOP

2024-06-04 10:05:48

微服務(wù)網(wǎng)關(guān)日志

2022-02-08 17:07:54

Spring BooSpring Aop日志記錄

2018-11-19 14:29:17

Spring BootXML支持

2010-09-30 14:38:13

2020-10-08 14:52:37

數(shù)據(jù)網(wǎng)絡(luò)威脅安全

2022-11-18 08:31:56

Spring日志工具

2009-07-05 21:04:02

2009-11-27 09:34:38

VS2003命令

2023-03-01 13:54:53

Springpostion?繼承

2025-02-03 09:00:00

API接口性能

2022-05-12 11:38:26

Java日志Slf4j

2023-09-13 08:56:51

2019-04-15 08:32:25

Spring Boot日志門(mén)面模式

2020-09-27 11:35:16

Spring BootStarterJava

2024-08-01 09:10:03

2021-03-26 06:01:45

日志MongoDB存儲(chǔ)

2023-12-29 18:13:27

Spring日志應(yīng)用程序

2023-09-19 22:41:30

控制器HTTP

2021-05-18 07:30:36

開(kāi)發(fā)Spring Boot日志
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)