当前位置: 首页 > ds >正文

若依脱敏功能升级:接口返回想脱就脱,想不脱就不脱(实现灵活可控制的数据脱敏)

若依原生框架中的脱敏功能不够灵活(默认超级管理员不脱敏,其他则脱敏)。

有时候,我们有些接口想要脱敏,但是有些接口又不想脱敏。(例如列表查询的时候脱敏。修改的时候,不想数据脱敏)

1、在com.ruoyi.common.config.serializer 新建SensitiveConditionSerializer

package com.ruoyi.common.config.serializer;import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.databind.BeanProperty;
import com.fasterxml.jackson.databind.JsonMappingException;
import com.fasterxml.jackson.databind.JsonSerializer;
import com.fasterxml.jackson.databind.SerializerProvider;
import com.fasterxml.jackson.databind.ser.ContextualSerializer;
import com.ruoyi.common.annotation.SensitiveCondition;
import com.ruoyi.common.enums.DesensitizedType;
import org.springframework.util.ReflectionUtils;
import org.springframework.util.StringUtils;import java.io.IOException;
import java.lang.reflect.Field;
import java.lang.reflect.Method;/*** 条件数据脱敏序列化器(新版,支持DesensitizedType)*/
public class SensitiveConditionSerializer extends JsonSerializer<String> implements ContextualSerializer {private SensitiveCondition sensitive;private DesensitizedType desensitizedType;public SensitiveConditionSerializer() {}public SensitiveConditionSerializer(SensitiveCondition sensitive, DesensitizedType desensitizedType) {this.sensitive = sensitive;this.desensitizedType = desensitizedType;}@Overridepublic void serialize(String value, JsonGenerator gen, SerializerProvider serializers) throws IOException {if (value == null) {gen.writeNull();return;}boolean needDesensitize = false;Object currentObj = gen.getCurrentValue();// 判断是否需要脱敏if (sensitive != null) {if (StringUtils.hasLength(sensitive.conditionMethod())) {// 通过方法判断Method method = ReflectionUtils.findMethod(currentObj.getClass(), sensitive.conditionMethod());if (method != null) {try {needDesensitize = (boolean) ReflectionUtils.invokeMethod(method, currentObj);} catch (Exception e) {needDesensitize = false;}}} else if (StringUtils.hasLength(sensitive.conditionField())) {// 通过字段判断Field field = ReflectionUtils.findField(currentObj.getClass(), sensitive.conditionField());if (field != null) {field.setAccessible(true);try {needDesensitize = (boolean) field.get(currentObj);} catch (Exception e) {needDesensitize = false;}}}}// 根据条件决定是否脱敏if (needDesensitize && desensitizedType != null) {gen.writeString(desensitizedType.desensitizer().apply(value));} else {gen.writeString(value);}}@Overridepublic JsonSerializer<?> createContextual(SerializerProvider prov, BeanProperty property) throws JsonMappingException {if (property != null) {SensitiveCondition anno = property.getAnnotation(SensitiveCondition.class);if (anno == null) {anno = property.getContextAnnotation(SensitiveCondition.class);}if (anno != null) {// 从注解里拿typeDesensitizedType type = anno.type();return new SensitiveConditionSerializer(anno, type);}}return this;}
}

2、在 com.ruoyi.common.annotation新建注解SensitiveCondition

package com.ruoyi.common.annotation;import com.fasterxml.jackson.annotation.JacksonAnnotationsInside;
import com.fasterxml.jackson.databind.annotation.JsonSerialize;
import com.ruoyi.common.config.serializer.SensitiveConditionSerializer;
import com.ruoyi.common.enums.DesensitizedType;import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;/*** 条件数据脱敏注解*/
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@JacksonAnnotationsInside
@JsonSerialize(using = SensitiveConditionSerializer.class)
public @interface SensitiveCondition {/*** 脱敏类型*/DesensitizedType type();/*** 判断是否需要脱敏的方法名* 该方法需要定义在实体类中,返回boolean类型*/String conditionMethod() default "";/*** 判断是否需要脱敏的字段名* 该字段需要定义在实体类中,类型为boolean*/String conditionField() default "";
}

3、在 com.ruoyi.common.core.domain的基类BaseEntity中,添加属性needDesensitize

    /** 是否需要脱敏 */@JsonIgnoreprivate boolean needDesensitize = true;public boolean isNeedDesensitize() {return needDesensitize;}public void setNeedDesensitize(boolean needDesensitize) {this.needDesensitize = needDesensitize;}

4、在需要数据脱敏的实体类字段上方写注解

//eg: 用户类SysUser中/** 用户昵称 */@Excel(name = "用户名称")@SensitiveCondition(type = DesensitizedType.USERNAME, conditionField = "needDesensitize")private String nickName;/** 用户邮箱 */@Excel(name = "用户邮箱")@SensitiveCondition(type = DesensitizedType.EMAIL, conditionField = "needDesensitize")private String email;/** 手机号码 */@Excel(name = "手机号码", cellType = ColumnType.TEXT)@SensitiveCondition(type = DesensitizedType.PHONE, conditionField = "needDesensitize")private String phonenumber;

5、控制方式(使用方法)

// 默认情况下会脱敏(needDesensitize默认为true)
PersonInfo person = personInfoMapper.selectPersonInfoById(1L);
System.out.println(person.getName()); // 输出:张*丰// 不需要脱敏的场景
person.setNeedDesensitize(false);
System.out.println(person.getName()); // 输出:张三丰------------------------------------------------------1、 在需要控制脱敏的地方,可以通过以下方式控制【通过字段控制】:// 在Service层或Controller层public List<SysUser> getPersonInfoList() {List<SysUser> list = personInfoMapper.selectPersonInfoList();// 某些情况下不需要脱敏if (某些条件) {list.forEach(person -> person.setNeedDesensitize(false));}return list;}2、通过方法控制(如果您选择使用conditionMethod)@SensitiveCondition(type = DesensitizedType.USERNAME, conditionMethod = "shouldDesensitize")private String name;public boolean shouldDesensitize() {// 根据业务逻辑决定是否脱敏,例如判断是否管理员return !SecurityUtils.isAdmin();}
---------------------------------------------------------------------------------
优点:
完全兼容原有的脱敏功能
可以灵活控制是否脱敏
支持多种控制方式(字段、方法)
易于扩展和维护
性能影响小,只在序列化时进行处理
http://www.xdnf.cn/news/2548.html

相关文章:

  • 手撕——贪吃蛇小游戏(下)
  • 【quantity】1 创建 crates.io 账号并上传 Rust 库
  • 数据库查询艺术:从单表操作到多表联查的全面指南
  • Rollup、Webpack、Esbuild 和 Vite 前端打包工具
  • Redis01-基础-入门
  • 华为仓颉编程语言的实际用法与使用领域详解
  • OpenCV实验室工具的使用
  • 【银河麒麟高级服务器操作系统】在VMware虚拟机情况下出现软锁处理过程
  • C/C++死锁和活锁
  • k8s学习记录(五):Pod亲和性详解
  • 解决两个技术问题后小有感触-QZ Tray使用经验小总结
  • 分布式GPU上计算长向量模的方法
  • 数据一致性问题剖析与实践(四)——竞态条件竞争导致的一致性问题
  • 制作一款打飞机游戏26:精灵编辑器
  • streamlit实现非原生的按钮触发效果 + flask实现带信息的按钮触发
  • Pikachu靶场-PHP反序列化漏洞
  • 2024ICPC网络赛第二场题解
  • DeepSeek:重构人类文明的智能引擎
  • JVM——运行时数据区
  • NLP预处理:如何 处理表情符号
  • 基于物理信息的神经网络在异常检测Anomaly Detection中的应用:实践指南
  • 解决Cline的Shell Integration Unavailable问题
  • 软考:软件设计师考试数据结构知识点详解
  • 引领印尼 Web3 变革:Mandala Chain 如何助力 1 亿用户迈向数字未来?
  • .class文件是字节码吗还是二进制文件
  • 【首款Armv9开源芯片“星睿“O6测评】SVE2指令集介绍与测试
  • Android调试那些事儿
  • uniapp-商城-42-shop 后台管理 分包
  • 多视觉编码器协同与高低分辨率特征融合技术综述
  • JVM——垃圾收集策略