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);
}
}
这样我们就实现了请求日志的存储
{!{}!}
{!{}!}