SpringCloud存储统一请求日志

GA666666 2022-12-02 AM 6414℃ 2条

save.png

1.前言

SpringCloud日志存储,大多数采用Sleuth + Zipkin,对于小公司来说感觉有点重

本次采用Sleuth + ControllerAdvice注解 + SpringEvent注解来完成这个任务。引入Sleuth来获取traceid,存储不采用zipkin,通过 ControllerAdvice注解来实现

2.实践

通常一个项目下有多个项目,为了统一管理我们将配置文件统一抽取到common中进行配置和管理,以下是文件的目录结构

demo
├── common
├── gateway
├── model
│   ├── base-model
│   └── demo-model
├── service
│   ├── demo-service  
│   └── system-service

common中包含拦截器等组件

2.1添加pom

<dependency>
      <groupId>org.springframework.cloud</groupId>
      <artifactId>spring-cloud-starter-sleuth</artifactId>
</dependency>

2.2添加配置文件

配置application.yml
spring:
  sleuth:
    sampler: #采样器
      probability: 1 #采样率,采样率是采集Trace的比率,默认0.1
      rate: 10000 #每秒数据采集量,最多n条/秒Trace
针对不同版本的sleuth,可能会在引入后无法启动项目,可以选填此项
/**
 * @Author By GaoXu
 * @Date 2022 11 30 10 06
 * @Description : 必须配置此文件,不配置会与其他组件冲突产生死锁
 **/
@Configuration
public class SleuthConfiguration {
    @Value("${spring.sleuth.sampler.probability:1}")
    private String probability;

    @Bean
    public Sampler defaultSampler() throws Exception {
        Float f = new Float(probability);
        SamplerProperties samplerProperties = new SamplerProperties();
        samplerProperties.setProbability(f);
        ProbabilityBasedSampler sampler = new ProbabilityBasedSampler(samplerProperties);
        return sampler;
    }
}

2.3添加ControllerAdvice

import brave.Tracer;
/**
 * @Author By GaoXu
 * @Date 2022 11 30 10 06
 * @Description : 操作日志收集器
 **/

@Slf4j
@ControllerAdvice
public class RequestLogCollect implements ResponseBodyAdvice<R> {

    @Autowired
    private Tracer tracer;
    @Resource
    private ApplicationEventPublisher context;

    @Value("${spring.application.name}")
    private String moduleName;

    @Override
    public boolean supports(MethodParameter methodParameter, Class<? extends HttpMessageConverter<?>> aClass) {
        return R.class.isAssignableFrom(methodParameter.getParameterType());
    }

    @Override
    public R beforeBodyWrite(R body, MethodParameter methodParameter, MediaType mediaType, Class<? extends HttpMessageConverter<?>> aClass, ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse) {
        requestLogSave(body, methodParameter, serverHttpRequest);
        return body;
    }

    @Async
    void requestLogSave(R body, MethodParameter methodParameter, ServerHttpRequest serverHttpRequest) {
        try {
            boolean check = checkIgnore(methodParameter);
            if (check) {
                return;
            }
            String traceId = tracer.currentSpan().context().traceIdString();
            UserVo user = RedisUtils.getUser();
            OperationLogVo operationLogVo = new OperationLogVo();
            operationLogVo.setTraceId(traceId);
            // 获取客户端IP地址
            String remoteAddr = serverHttpRequest.getRemoteAddress().getAddress().getHostAddress();
            // 获取请求方式
            String method = serverHttpRequest.getMethod().name();
            // 获取请求路径
            String servletPath = serverHttpRequest.getURI().getPath();
            operationLogVo.setUserAgent(HttpUtils.getDeviceInfo(serverHttpRequest));
            operationLogVo.setBasePath(methodParameter.getMethod().getDeclaringClass().getName() + "." + methodParameter.getMethod().getName());
            operationLogVo.setUrl(servletPath);
            operationLogVo.setModuleName(moduleName + ":" + getDescription(methodParameter));
            operationLogVo.setOplogType(OperationLogTypeEnum.getLogTypeEnumByKeyWord(servletPath).getCode());
            operationLogVo.setOperationContent(OperationLogTypeEnum.getLogTypeEnumByKeyWord(servletPath).getDescription() + ":" + (body.getCode() == 200 ? "成功" : "失败"));
            operationLogVo.setRequestType(method);
            operationLogVo.setRealIp(HttpUtils.getRealIP(serverHttpRequest));
            operationLogVo.setComId(user.getComId());
            operationLogVo.setTenantId(user.getTenantId());
            operationLogVo.setClientIp(remoteAddr);
            operationLogVo.setResult(body.getCode() == 200);
            context.publishEvent(operationLogVo);
        } catch (Exception e) {
            log.error(e.getMessage(), e);
        }
    }

    /**
     * 获取方法描述
     *
     * @param methodParameter
     * @return
     */
    public String getDescription(MethodParameter methodParameter) {
        StringBuilder result = new StringBuilder();
        try {
            Class<?> declaringClass = methodParameter.getMethod().getDeclaringClass();
            Api annotation = declaringClass.getAnnotation(Api.class);
            if (!Objects.isNull(annotation)) {
                String[] tags = annotation.tags();
                result.append(tags[0]);
                ApiOperation methodAnnot = methodParameter.getMethod().getAnnotation(ApiOperation.class);
                if (!Objects.isNull(methodAnnot)) {
                    result.append("->" + methodAnnot.value());
                }
            }
            return result.toString();
        } catch (Exception e) {
            return result.toString();
        }
    }
}

2.4过滤不需要存储的请求

创建IgnoreLog注解
import java.lang.annotation.*;

/**
 * @Author By GaoXu
 * @Date 2022 11 30 16 09
 * @Description : 方法上添加此注解,请求后不会进行日志存储;类上添加此注解,类中所有方法日志都不会进行存储
 **/
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD, ElementType.TYPE})
public @interface IgnoreLog {
}
在RequestLogCollect添加注解检测方法,通过反射获取请求方法是否加了注解
 /**
     * 注解检测
     *
     * @param methodParameter
     * @return
     */
    private boolean checkIgnore(MethodParameter methodParameter) {
        boolean result = false;
        try {
            Class<?> declaringClass = methodParameter.getMethod().getDeclaringClass();
            IgnoreLog annotation = declaringClass.getAnnotation(IgnoreLog.class);
            if (Objects.isNull(annotation)) {
                IgnoreLog methodAnnot = methodParameter.getMethod().getAnnotation(IgnoreLog.class);
                if (!Objects.isNull(methodAnnot)) {
                    result = true;
                }
            } else {
                result = true;
            }
            return result;
        } catch (Exception e) {
            return result;
        }
    }

2.5优化

缓存
由于采用注解形式,在运行时会消耗大量IO来反射获取注解,这里可以采用@PostConstruct注解,在启动完成时,扫描注解后,将需要过滤的类名方法名,存储到内存,如果数据量大可以存储到redis,这里toB项目不追求性能,就不再优化这部分了
异步
虽说是toB,该优化也要优化下,我们在存储的时候采用@Async注解异步处理

2.6存储

因为我们将配置从service中抽出后统一进行配置,这里项目起步阶段还未采用MQ,那么如果我们在common中调用service中的方法,会产生循环依赖问题,这里采用SpringEvent来化解这个问题

推送:context.publishEvent(object);
public class RequestLogCollect implements ResponseBodyAdvice<R> {
  
    @Resource
    private ApplicationEventPublisher context;
  
    @Override
    public boolean supports(MethodParameter methodParameter, Class<? extends HttpMessageConverter<?>> aClass) {
        return R.class.isAssignableFrom(methodParameter.getParameterType());
    }

    @Override
    public R beforeBodyWrite(R body, MethodParameter methodParameter, MediaType mediaType, Class<? extends HttpMessageConverter<?>> aClass, ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse) {
        requestLogSave(body, methodParameter, serverHttpRequest);
        return body;
    }

    @Async
    void requestLogSave(R body, MethodParameter methodParameter, ServerHttpRequest serverHttpRequest) {
        try {
            context.publishEvent(object);
        } catch (Exception e) {
            log.error(e.getMessage(), e);
        }
    }
}

接收:@EventListener

@Service
public class RequestLogCollectService {

    @Resource
    private OperationLogClient operationLogClient;

    /**
     * 添加操作日志列表记录
     *
     * @param operationLogVo 实例对象
     * @return 对象列表
     */
    @EventListener(OperationLogVo.class)
    public R addOperationLog(OperationLogVo operationLogVo) {
        return operationLogClient.addOperationLog(operationLogVo);
    }
}
这样我们就实现了请求日志的存储
标签: none

非特殊说明,本博所有文章均为博主原创。

评论啦~



已有 2 条评论


  1. GA666666
    GA666666 博主

    {!{}!}

    回复 2022-12-02 09:33
    1. Advice
      Advice

      {!{}!}

      回复 2022-12-02 09:35