zhb.dong 1 jaar geleden
bovenliggende
commit
9ba60e5b89

+ 7 - 1
zd-gateway/pom.xml

@@ -113,7 +113,13 @@
             <scope>compile</scope>
         </dependency>
 
-<!--        <dependency>-->
+        <dependency>
+            <groupId>org.springframework.boot</groupId>
+            <artifactId>spring-boot-starter-webflux</artifactId>
+        </dependency>
+
+
+        <!--        <dependency>-->
 <!--            <groupId>org.springframework.boot</groupId>-->
 <!--            <artifactId>spring-boot-starter-web</artifactId>-->
 <!--            <scope>compile</scope>-->

+ 29 - 0
zd-gateway/src/main/java/com/zd/gateway/filter/GatewayLogEventListener.java

@@ -0,0 +1,29 @@
+package com.zd.gateway.filter;
+
+import com.zd.gateway.log.GatewayLog;
+import com.zd.gateway.log.GatewayLogEvent;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.context.ApplicationListener;
+import org.springframework.stereotype.Component;
+
+@Component
+@Slf4j(topic = "[RequestLog]")
+public class GatewayLogEventListener implements ApplicationListener<GatewayLogEvent> {
+    @Override
+    public void onApplicationEvent(GatewayLogEvent event) {
+        GatewayLog gatewayLog = event.getGatewayLog();
+        log.info("============================================ begins ============================================");
+        log.info("Request Ip : {}",gatewayLog.getIp());
+        log.info("Target Server : {}",gatewayLog.getTargetServer());
+        log.info("Request Method : {}",gatewayLog.getRequestMethod());
+        log.info("Request Path: {}",gatewayLog.getRequestPath());
+        log.info("Request ContentType : {}",gatewayLog.getRequestContentType());
+        log.info("Request Headers : {}",gatewayLog.getHeaders());
+        log.info("Request Params : {}",gatewayLog.getRequestBody());
+        log.info("Response status : {}",gatewayLog.getStatus());
+        log.info("Response Data : {}",gatewayLog.getResponseData());
+        log.info("This request response time is  {} ms !",gatewayLog.getExecuteTime() );
+        log.info("============================================   end  ============================================");
+        //TODO es or db
+    }
+}

+ 315 - 0
zd-gateway/src/main/java/com/zd/gateway/filter/GatewayLogFilter.java

@@ -0,0 +1,315 @@
+package com.zd.gateway.filter;
+
+import com.alibaba.fastjson.JSON;
+import com.zd.gateway.log.GatewayLog;
+import com.zd.gateway.log.GatewayLogEvent;
+import com.zd.gateway.log.IpUtil;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.cloud.gateway.filter.factory.rewrite.CachedBodyOutputMessage;
+import org.springframework.context.ApplicationEventPublisher;
+import org.springframework.util.ObjectUtils;
+import reactor.core.publisher.Flux;
+import reactor.core.publisher.Mono;
+import java.nio.charset.StandardCharsets;
+import java.util.*;
+import lombok.extern.slf4j.Slf4j;
+import org.apache.commons.lang3.StringUtils;
+import org.reactivestreams.Publisher;
+import org.springframework.cloud.gateway.filter.GatewayFilterChain;
+import org.springframework.cloud.gateway.filter.GlobalFilter;
+import org.springframework.cloud.gateway.route.Route;
+import org.springframework.cloud.gateway.support.BodyInserterContext;
+import org.springframework.cloud.gateway.support.ServerWebExchangeUtils;
+import org.springframework.core.Ordered;
+import org.springframework.core.io.buffer.DataBuffer;
+import org.springframework.core.io.buffer.DataBufferFactory;
+import org.springframework.core.io.buffer.DataBufferUtils;
+import org.springframework.core.io.buffer.DefaultDataBufferFactory;
+import org.springframework.http.HttpHeaders;
+import org.springframework.http.HttpStatus;
+import org.springframework.http.MediaType;
+import org.springframework.http.codec.HttpMessageReader;
+import org.springframework.http.server.reactive.ServerHttpRequest;
+import org.springframework.http.server.reactive.ServerHttpRequestDecorator;
+import org.springframework.http.server.reactive.ServerHttpResponse;
+import org.springframework.http.server.reactive.ServerHttpResponseDecorator;
+import org.springframework.stereotype.Component;
+import org.springframework.util.MultiValueMap;
+import org.springframework.web.reactive.function.BodyInserter;
+import org.springframework.web.reactive.function.BodyInserters;
+import org.springframework.web.reactive.function.server.HandlerStrategies;
+import org.springframework.web.reactive.function.server.ServerRequest;
+import org.springframework.web.server.ServerWebExchange;
+
+import java.util.*;
+
+@Slf4j
+@Component
+public class GatewayLogFilter  implements GlobalFilter, Ordered {
+
+    @Autowired
+    private ApplicationEventPublisher applicationEventPublisher;
+
+    private final List<HttpMessageReader<?>> messageReaders = HandlerStrategies.withDefaults().messageReaders();
+    private static final String CONTENT_TYPE = "application/json";
+
+    // 请求来源应用
+    private static final String REQUEST_ORIGIN_APP = "Request-Origin-App";
+
+    // 自定义请求头,转发之前删除自定义请求头
+    private static final List<String> CUSTOM_HEADERS = Arrays.asList("sign", "timestamp", "random", "Request-Origin-App");
+
+    @Override
+    public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
+
+        ServerHttpRequest request = exchange.getRequest();
+
+        // 请求路径
+        String requestPath = request.getPath().pathWithinApplication().value();
+
+        // 获取路由信息
+        Route route = getGatewayRoute(exchange);
+
+        String ipAddress = IpUtil.getIpAddr(request);
+
+        GatewayLog gatewayLog = new GatewayLog();
+        gatewayLog.setOrigin(request.getHeaders().getFirst(REQUEST_ORIGIN_APP));
+        gatewayLog.setSchema(request.getURI().getScheme());
+        gatewayLog.setRequestMethod(request.getMethodValue());
+        gatewayLog.setRequestPath(requestPath);
+        gatewayLog.setTargetServer(route.getUri().toString());
+        gatewayLog.setStartTime(new Date().getTime());
+        gatewayLog.setIp(ipAddress);
+        gatewayLog.setRouteConfig(JSON.toJSONString(route));
+        Map<String, Object> headers = new HashMap<>();
+        for (String key : request.getHeaders().keySet()) {
+            headers.put(key, request.getHeaders().getFirst(key));
+        }
+        gatewayLog.setHeaders(JSON.toJSONString(headers));
+
+
+        MediaType mediaType = request.getHeaders().getContentType();
+        if (request.getHeaders().getContentType() != null) {
+            gatewayLog.setRequestContentType(request.getHeaders().getContentType().toString());
+        }
+
+        if(mediaType != null && (MediaType.APPLICATION_FORM_URLENCODED.isCompatibleWith(mediaType) || MediaType.APPLICATION_JSON.isCompatibleWith(mediaType))){
+            return writeBodyLog(exchange, chain, gatewayLog);
+        }else{
+            return writeBasicLog(exchange, chain, gatewayLog);
+        }
+
+    }
+
+    @Override
+    public int getOrder() {
+        // 必须小于等于-2,否则无法获取相应结果
+        return -10;
+    }
+
+
+    /**
+     * 获取路由信息
+     * @param exchange
+     * @return
+     */
+    private Route getGatewayRoute(ServerWebExchange exchange) {
+        return exchange.getAttribute(ServerWebExchangeUtils.GATEWAY_ROUTE_ATTR);
+    }
+
+
+
+    private Mono<Void> writeBasicLog(ServerWebExchange exchange, GatewayFilterChain chain, GatewayLog gatewayLog) {
+        StringBuilder builder = new StringBuilder();
+        MultiValueMap<String, String> queryParams = exchange.getRequest().getQueryParams();
+        gatewayLog.setRequestBody(getUrlParamsByMap(queryParams));
+
+        // 修改Header
+        ServerHttpRequest mutableReq = exchange.getRequest().mutate().headers(httpHeaders -> {
+            // 删除自定义header
+            for (String customHeader : CUSTOM_HEADERS) {
+                httpHeaders.remove(customHeader);
+            }
+        }).build();
+
+        //获取响应体
+        ServerHttpResponseDecorator decoratedResponse = recordResponseLog(exchange, gatewayLog);
+
+        return chain.filter(exchange.mutate().request(mutableReq).response(decoratedResponse).build())
+                .then(Mono.fromRunnable(() -> {
+                    // 打印日志
+                    writeAccessLog(gatewayLog);
+                }));
+    }
+
+
+    /**
+     * 解决 request body 只能读取一次问题,
+     * 参考: org.springframework.cloud.gateway.filter.factory.rewrite.ModifyRequestBodyGatewayFilterFactory
+     * @param exchange
+     * @param chain
+     * @param gatewayLog
+     * @return
+     */
+    private Mono<Void> writeBodyLog(ServerWebExchange exchange, GatewayFilterChain chain, GatewayLog gatewayLog) {
+        ServerRequest serverRequest = ServerRequest.create(exchange,messageReaders);
+
+        Mono<String> modifiedBody = serverRequest.bodyToMono(String.class)
+                .flatMap(body ->{
+                    gatewayLog.setRequestBody(body);
+                    return Mono.just(body);
+                });
+
+        // 通过 BodyInserter 插入 body(支持修改body), 避免 request body 只能获取一次
+        BodyInserter bodyInserter = BodyInserters.fromPublisher(modifiedBody, String.class);
+        HttpHeaders headers = new HttpHeaders();
+        headers.putAll(exchange.getRequest().getHeaders());
+        // the new content type will be computed by bodyInserter
+        // and then set in the request decorator
+        headers.remove(HttpHeaders.CONTENT_LENGTH);
+
+        CachedBodyOutputMessage outputMessage = new CachedBodyOutputMessage(exchange, headers);
+
+        return bodyInserter.insert(outputMessage,new BodyInserterContext())
+                .then(Mono.defer(() -> {
+                    // 重新封装请求
+                    ServerHttpRequest decoratedRequest = requestDecorate(exchange, headers, outputMessage);
+
+                    // 记录响应日志
+                    ServerHttpResponseDecorator decoratedResponse = recordResponseLog(exchange, gatewayLog);
+
+                    // 记录普通的
+                    return chain.filter(exchange.mutate().request(decoratedRequest).response(decoratedResponse).build())
+                            .then(Mono.fromRunnable(() -> {
+                                // 打印日志
+                                writeAccessLog(gatewayLog);
+                            }));
+                }));
+    }
+
+
+
+    /**
+     * 打印日志
+     * @param gatewayLog 网关日志
+     */
+    private void writeAccessLog(GatewayLog gatewayLog) {
+        applicationEventPublisher.publishEvent(new GatewayLogEvent(this, gatewayLog));
+    }
+
+
+
+    /**
+     * 请求装饰器,重新计算 headers
+     * @param exchange
+     * @param headers
+     * @param outputMessage
+     * @return
+     */
+    private ServerHttpRequestDecorator requestDecorate(ServerWebExchange exchange, HttpHeaders headers,
+                                                       CachedBodyOutputMessage outputMessage) {
+        return new ServerHttpRequestDecorator(exchange.getRequest()) {
+            @Override
+            public HttpHeaders getHeaders() {
+                long contentLength = headers.getContentLength();
+                HttpHeaders httpHeaders = new HttpHeaders();
+                httpHeaders.putAll(super.getHeaders());
+                if (contentLength > 0) {
+                    httpHeaders.setContentLength(contentLength);
+                } else {
+                    // TODO: this causes a 'HTTP/1.1 411 Length Required' // on
+                    // httpbin.org
+                    httpHeaders.set(HttpHeaders.TRANSFER_ENCODING, "chunked");
+                }
+
+                // 删除自定义header
+                for (String customHeader : CUSTOM_HEADERS) {
+                    headers.remove(customHeader);
+                }
+
+                return httpHeaders;
+            }
+
+            @Override
+            public Flux<DataBuffer> getBody() {
+                return outputMessage.getBody();
+            }
+        };
+    }
+
+
+    /**
+     * 记录响应日志
+     * 通过 DataBufferFactory 解决响应体分段传输问题。
+     */
+    private ServerHttpResponseDecorator recordResponseLog(ServerWebExchange exchange, GatewayLog gatewayLog) {
+        ServerHttpResponse response = exchange.getResponse();
+        DataBufferFactory bufferFactory = response.bufferFactory();
+
+        return new ServerHttpResponseDecorator(response) {
+            @Override
+            public Mono<Void> writeWith(Publisher<? extends DataBuffer> body) {
+                if (body instanceof Flux) {
+                    Date responseTime = new Date();
+                    gatewayLog.setEndTime(responseTime.getTime());
+                    // 计算执行时间
+                    long executeTime = (responseTime.getTime() - gatewayLog.getStartTime());
+
+                    gatewayLog.setExecuteTime(executeTime);
+                    gatewayLog.setStatus(response.getStatusCode().value() == 200 ? "成功" : "失败");
+
+                    // 获取响应类型,如果是 json 就打印
+                    String originalResponseContentType = exchange.getAttribute(ServerWebExchangeUtils.ORIGINAL_RESPONSE_CONTENT_TYPE_ATTR);
+
+
+                    if (Objects.equals(this.getStatusCode(), HttpStatus.OK)
+                            && StringUtils.isNotBlank(originalResponseContentType)
+                            && originalResponseContentType.contains(CONTENT_TYPE)) {
+
+                        Flux<? extends DataBuffer> fluxBody = Flux.from(body);
+                        return super.writeWith(fluxBody.buffer().map(dataBuffers -> {
+
+                            // 合并多个流集合,解决返回体分段传输
+                            DataBufferFactory dataBufferFactory = new DefaultDataBufferFactory();
+                            DataBuffer join = dataBufferFactory.join(dataBuffers);
+                            byte[] content = new byte[join.readableByteCount()];
+                            join.read(content);
+
+                            // 释放掉内存
+                            DataBufferUtils.release(join);
+                            String responseResult = new String(content, StandardCharsets.UTF_8);
+
+                            gatewayLog.setResponseData(responseResult);
+
+                            return bufferFactory.wrap(content);
+                        }));
+                    }
+                }
+                // if body is not a flux. never got there.
+                return super.writeWith(body);
+            }
+        };
+    }
+
+
+    /**
+     * 将map参数转换成url参数
+     * @param map
+     * @return
+     */
+    private String getUrlParamsByMap(MultiValueMap<String, String> map) {
+        if (ObjectUtils.isEmpty(map)) {
+            return "";
+        }
+        StringBuilder sb = new StringBuilder();
+        for (Map.Entry<String, List<String>> entry : map.entrySet()) {
+            sb.append(entry.getKey()).append("=").append(entry.getValue().get(0));
+            sb.append("&");
+        }
+        String s = sb.toString();
+        if (s.endsWith("&")) {
+            s = StringUtils.substringBeforeLast(s, "&");
+        }
+        return s;
+    }
+}

+ 0 - 85
zd-gateway/src/main/java/com/zd/gateway/filter/RequestBodyFilter.java

@@ -1,85 +0,0 @@
-package com.zd.gateway.filter;
-
-import lombok.extern.slf4j.Slf4j;
-import org.springframework.cloud.gateway.filter.GatewayFilterChain;
-import org.springframework.cloud.gateway.filter.GlobalFilter;
-import org.springframework.core.Ordered;
-import org.springframework.core.io.buffer.DataBuffer;
-import org.springframework.core.io.buffer.DataBufferUtils;
-import org.springframework.http.HttpHeaders;
-
-import org.springframework.http.server.reactive.ServerHttpRequest;
-import org.springframework.http.server.reactive.ServerHttpRequestDecorator;
-import org.springframework.stereotype.Component;
-import org.springframework.util.MultiValueMap;
-import org.springframework.web.server.ServerWebExchange;
-import reactor.core.publisher.Flux;
-import reactor.core.publisher.Mono;
-
-import java.io.UnsupportedEncodingException;
-import java.net.URI;
-
-
-@Component
-@Slf4j(topic = "【gateway-http-body---】")
-public class RequestBodyFilter implements GlobalFilter, Ordered {
-
-    @Override
-    public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
-        ServerHttpRequest request = exchange.getRequest();
-        URI URIPath = request.getURI();
-        String path = request.getPath().value();
-        String method = request.getMethodValue();
-        HttpHeaders headers = request.getHeaders();
-        log.info("============================================Request begin============================================");
-        log.info("Request Information :URI = {}, path = {},method = {},header = {}。", URIPath, path, method, headers);
-        if ("POST".equals(method)) {
-            return DataBufferUtils.join(exchange.getRequest().getBody())
-                    .flatMap(dataBuffer -> {
-                        byte[] bytes = new byte[dataBuffer.readableByteCount()];
-                        dataBuffer.read(bytes);
-                        try {
-                            String bodyString = new String(bytes, "utf-8");
-                            log.info("Request params :" + bodyString);
-                            exchange.getAttributes().put("POST_BODY", bodyString);
-                        } catch (UnsupportedEncodingException e) {
-                            e.printStackTrace();
-                        }
-                        DataBufferUtils.release(dataBuffer);
-                        Flux<DataBuffer> cachedFlux = Flux.defer(() -> {
-                            DataBuffer buffer = exchange.getResponse().bufferFactory()
-                                    .wrap(bytes);
-                            return Mono.just(buffer);
-                        });
-                        ServerHttpRequest mutatedRequest = new ServerHttpRequestDecorator(
-                                exchange.getRequest()) {
-                            @Override
-                            public Flux<DataBuffer> getBody() {
-                                return cachedFlux;
-                            }
-                        };
-                        log.info("***********************************Request end**********************************");
-                        return chain.filter(exchange.mutate().request(mutatedRequest)
-                                .build()).then(Mono.fromRunnable(() -> {
-                            log.info("*********************************** Response begin **********************************");
-                            log.info("Response: code:{},headers:{}", exchange.getResponse().getStatusCode(),exchange.getResponse().getHeaders());
-                            log.info("============================================ Response end ============================================");
-                        }));
-                    });
-        } else {
-            MultiValueMap<String, String> queryParams = request.getQueryParams();
-            log.info("Request params :{} " ,queryParams);
-            log.info("***********************************Request end**********************************");
-            return chain.filter(exchange).then(Mono.fromRunnable(() -> {
-                log.info("*********************************** Response begin **********************************");
-                log.info("Response: code:{},headers:{}", exchange.getResponse().getStatusCode(),exchange.getResponse().getHeaders());
-                log.info("============================================ Response end ============================================");
-            }));
-        }
-    }
-
-    @Override
-    public int getOrder() {
-        return -10;
-    }
-}

+ 65 - 0
zd-gateway/src/main/java/com/zd/gateway/log/GatewayLog.java

@@ -0,0 +1,65 @@
+package com.zd.gateway.log;
+
+
+import lombok.*;
+
+@Data
+@Builder
+@NoArgsConstructor
+@AllArgsConstructor
+public class GatewayLog {
+
+    /**请求来源**/
+    private String origin;
+
+    /**访问实例*/
+    private String targetServer;
+
+    /**请求路径*/
+    private String requestPath;
+
+    /**请求方法*/
+    private String requestMethod;
+
+    /**协议 */
+    private String schema;
+
+    /**请求类型 */
+    private String requestContentType;
+
+    /**请求头 */
+    private String headers;
+
+    /**请求体*/
+    private String requestBody;
+
+    /**响应体*/
+    private String responseData;
+
+    /**请求ip*/
+    private String ip;
+
+    /**IP所属城市*/
+    private String city;
+
+    /**开始时间*/
+    private Long startTime;
+
+    /**结束时间*/
+    private Long endTime;
+
+    /**请求时间*/
+    private String requestTime;
+
+    /**响应时间*/
+    private String responseTime;
+
+    /**执行时间*/
+    private long executeTime;
+
+    /**路由配置*/
+    private String routeConfig;
+
+    /**响应状态*/
+    private String status;
+}

+ 17 - 0
zd-gateway/src/main/java/com/zd/gateway/log/GatewayLogEvent.java

@@ -0,0 +1,17 @@
+package com.zd.gateway.log;
+
+import org.springframework.context.ApplicationEvent;
+
+public class GatewayLogEvent extends ApplicationEvent {
+
+    private final GatewayLog gatewayLog;
+
+    public GatewayLogEvent(Object source, GatewayLog gatewayLog) {
+        super(source);
+        this.gatewayLog = gatewayLog;
+    }
+
+    public GatewayLog getGatewayLog() {
+        return gatewayLog;
+    }
+}

+ 59 - 0
zd-gateway/src/main/java/com/zd/gateway/log/IpUtil.java

@@ -0,0 +1,59 @@
+package com.zd.gateway.log;
+
+import com.zd.common.core.utils.StringUtils;
+import org.springframework.http.server.reactive.ServerHttpRequest;
+
+import java.net.InetSocketAddress;
+
+public class IpUtil {
+
+    /**
+     * 获取客户端真实IP地址
+     *
+     * @param request ServerHttpRequest对象
+     * @return 客户端IP地址
+     */
+    public static String getIpAddr(ServerHttpRequest request) {
+        String ipAddress = null;
+        if (request != null) {
+            // 优先从 X-Forwarded-For 头部获取
+            ipAddress = request.getHeaders().getFirst("X-Forwarded-For");
+            if (StringUtils.isEmpty(ipAddress) || "unknown".equalsIgnoreCase(ipAddress)) {
+                ipAddress = request.getHeaders().getFirst("Proxy-Client-IP");
+            }
+            if (StringUtils.isEmpty(ipAddress) || "unknown".equalsIgnoreCase(ipAddress)) {
+                ipAddress = request.getHeaders().getFirst("WL-Proxy-Client-IP");
+            }
+            if (StringUtils.isEmpty(ipAddress) || "unknown".equalsIgnoreCase(ipAddress)) {
+                ipAddress = request.getHeaders().getFirst("HTTP_X_FORWARDED_FOR");
+            }
+            if (StringUtils.isEmpty(ipAddress) || "unknown".equalsIgnoreCase(ipAddress)) {
+                ipAddress = request.getHeaders().getFirst("HTTP_X_FORWARDED");
+            }
+            if (StringUtils.isEmpty(ipAddress) || "unknown".equalsIgnoreCase(ipAddress)) {
+                ipAddress = request.getHeaders().getFirst("HTTP_X_CLUSTER_CLIENT_IP");
+            }
+            if (StringUtils.isEmpty(ipAddress) || "unknown".equalsIgnoreCase(ipAddress)) {
+                ipAddress = request.getHeaders().getFirst("HTTP_CLIENT_IP");
+            }
+            if (StringUtils.isEmpty(ipAddress) || "unknown".equalsIgnoreCase(ipAddress)) {
+                ipAddress = request.getHeaders().getFirst("HTTP_FORWARDED_FOR");
+            }
+            if (StringUtils.isEmpty(ipAddress) || "unknown".equalsIgnoreCase(ipAddress)) {
+                ipAddress = request.getHeaders().getFirst("HTTP_FORWARDED");
+            }
+            if (StringUtils.isEmpty(ipAddress) || "unknown".equalsIgnoreCase(ipAddress)) {
+                // 如果以上都未获取到,使用 RemoteAddress
+                InetSocketAddress address = request.getRemoteAddress();
+                if (address != null) {
+                    ipAddress = address.getAddress().getHostAddress();
+                }
+            }
+        }
+        // 如果通过多级代理,获取到的是一个逗号分隔的IP串,第一个IP为客户端真实IP
+        if (ipAddress != null && ipAddress.contains(",")) {
+            ipAddress = ipAddress.split(",")[0];
+        }
+        return ipAddress;
+    }
+}

+ 1 - 1
zd-gateway/src/main/resources/logback.xml

@@ -10,7 +10,7 @@
 	<property name="TRACE_ID" value="%X{traceId}"/>
 
 	<property name="TRACE_LOG_PATTERN"
-		value="%clr(%d{yyyy-MM-dd HH:mm:ss.SSS}){faint} %clr(${LOG_LEVEL_PATTERN:-%5p}) %clr(${PID:- }){magenta} %clr([TraceId - %X{traceId}]){faint} %clr(---){faint} %clr([%15.15t]){faint} %clr(%-40.40logger{39}){cyan} %clr(:){faint} %m%n${LOG_EXCEPTION_CONVERSION_WORD:-%wEx}" />
+		value="%clr(%d{yyyy-MM-dd HH:mm:ss.SSS}){faint} %clr(${LOG_LEVEL_PATTERN:-%5p}) %clr(${PID:- }){magenta} %clr([TID - %X{traceId}]){faint} %clr(---){faint} %clr([%15.15t]){faint} %clr(%-40.40logger{39}){cyan} %clr(:){faint} %m%n${LOG_EXCEPTION_CONVERSION_WORD:-%wEx}" />
 
 	<appender class="ch.qos.logback.core.ConsoleAppender"
 		name="STDOUT">

+ 1 - 1
zd-modules/zd-modules-laboratory/src/main/resources/logback.xml

@@ -10,7 +10,7 @@
 	<property name="TRACE_ID" value="%X{traceId}"/>
 
 	<property name="TRACE_LOG_PATTERN"
-		value="%clr(%d{yyyy-MM-dd HH:mm:ss.SSS}){faint} %clr(${LOG_LEVEL_PATTERN:-%5p}) %clr(${PID:- }){magenta} %clr([TraceId - %X{traceId}]){faint} %clr(---){faint} %clr([%15.15t]){faint} %clr(%-40.40logger{39}){cyan} %clr(:){faint} %m%n${LOG_EXCEPTION_CONVERSION_WORD:-%wEx}" />
+		value="%clr(%d{yyyy-MM-dd HH:mm:ss.SSS}){faint} %clr(${LOG_LEVEL_PATTERN:-%5p}) %clr(${PID:- }){magenta} %clr([TID - %X{traceId}]){faint} %clr(---){faint} %clr([%15.15t]){faint} %clr(%-40.40logger{39}){cyan} %clr(:){faint} %m%n${LOG_EXCEPTION_CONVERSION_WORD:-%wEx}" />
 
 	<appender class="ch.qos.logback.core.ConsoleAppender"
 		name="STDOUT">