Springcloud Gateway网关日志这样打才完美

打印 上一主题 下一主题

主题 823|帖子 823|积分 2469

日志打印收集是开发中调试和定位线上问题的关键手段也是重中之重,gateway作为请求入口转发的核心模块,合理、规范的日志打印很重要。


  • 这里只进行日志的打印输出,不做收集,收集汇总工作可以结合ELK,监控日志文件进行同步。
  • 实现方式使用 gateway 的 GlobalFilter 过滤器。
  • 请求日志打印的过滤器排序尽量低一些。
  • 打印日志时,注意避免多次打印造成并发请求日志错乱,可以拼接一个大的日志串,一次打印输出。
RequestLogFilter

增加一个请求入参过滤器,用来打印入参信息。
  1. @Slf4j
  2. @Configuration
  3. @ConditionalOnProperty(value = "log.request.enabled", havingValue = "true", matchIfMissing = true)
  4. public class RequestLogFilter implements GlobalFilter, Ordered {
  5.    @Override
  6.    public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
  7.       ServerHttpRequest request = exchange.getRequest();
  8.       // 打印请求路径
  9.       String path = request.getPath().pathWithinApplication().value();
  10.       // 打印请求url
  11.       String requestUrl = this.getOriginalRequestUrl(exchange);
  12.       // **构建成一条长 日志,避免并发下日志错乱**
  13.       StringBuilder reqLog = new StringBuilder(200);
  14.       // 日志参数
  15.       List<Object> reqArgs = new ArrayList<>();
  16.       reqLog.append("\n\n================ Gateway Request Start  ================\n");
  17.       // 打印路由添加占位符
  18.       reqLog.append("===> {}: {}\n");
  19.       // 参数
  20.       String requestMethod = request.getMethodValue();
  21.       reqArgs.add(requestMethod);
  22.       reqArgs.add(requestUrl);
  23.       // 打印请求头
  24.       HttpHeaders headers = request.getHeaders();
  25.       headers.forEach((headerName, headerValue) -> {
  26.          reqLog.append("===Headers===  {}: {}\n");
  27.          reqArgs.add(headerName);
  28.          //如果有token,可以先把token解析后打印出原始报文,JwtUtil替换成自己项目里工具类
  29.          if (AUTH_KEY.toLowerCase().equals(headerName)) {
  30.             String value = headerValue.get(0);
  31.             String token = JwtUtil.getToken(value);
  32.             Claims claims = JwtUtil.parseJWT(token);
  33.             reqArgs.add((claims == null) ? "" : claims.toString());
  34.             reqLog.append("===Headers===  {}: {}\n");
  35.             reqArgs.add(headerName.concat("-original"));
  36.             reqArgs.add(StringUtils.join(headerValue.toArray()));
  37.          } else {
  38.             reqArgs.add(StringUtils.join(headerValue.toArray()));
  39.          }
  40.       });
  41.       reqLog.append("================  Gateway Request End  =================\n");
  42.       // 打印执行时间
  43.       log.info(reqLog.toString(), beforeReqArgs.toArray());
  44.       return chain.filter(exchange);
  45.    }
  46.    
  47.     private String getOriginalRequestUrl(ServerWebExchange exchange) {
  48.             ServerHttpRequest request = exchange.getRequest();
  49.             LinkedHashSet<URI> uris = exchange.getRequiredAttribute(ServerWebExchangeUtils.GATEWAY_ORIGINAL_REQUEST_URL_ATTR);
  50.             URI requestUri = uris.stream().findFirst().orElse(request.getURI());
  51.             MultiValueMap<String, String> queryParams = request.getQueryParams();
  52.             return UriComponentsBuilder.fromPath(requestUri.getRawPath()).queryParams(queryParams).build().toUriString();
  53.     }
  54.    @Override
  55.    public int getOrder() {
  56.       return Ordered.LOWEST_PRECEDENCE;
  57.    }
复制代码
ResponseLogFilter

网关请求的响应报文不在这里打印,post body 参数没有打印
[code]@Slf4j@Configuration@ConditionalOnProperty(value = "log.request.enabled", havingValue = "true", matchIfMissing = true)public class GlobalResponseLogFilter implements GlobalFilter, Ordered {   @Override   public Mono filter(ServerWebExchange exchange, GatewayFilterChain chain) {      ServerHttpRequest request = exchange.getRequest();      // 打印请求路径      String path = request.getPath().pathWithinApplication().value();      return chain.filter(exchange).then(         Mono.fromRunnable(() -> {            MultiValueMap queryParams = request.getQueryParams();            String requestUrl = UriComponentsBuilder.fromPath(path).queryParams(queryParams).build().toUriString();            // 构建成一条长日志            StringBuilder responseLog = new StringBuilder(200);            // 日志参数            List responseArgs = new ArrayList();            responseLog.append("\n\n================ Gateway Response Start  ================\n");            ServerHttpResponse response = exchange.getResponse();            // 状态码个path占位符: 200 get: /xxx/xxx/xxx?a=b            responseLog.append("
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

慢吞云雾缓吐愁

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表