java查询字段脱敏

场景1:查询出的字段页面需要脱敏显示

image.png

解决方式如下

1、通过sql实现字段脱敏

select CONCAT(LEFT(token, 6), '***', RIGHT(token, 6))  as token from device;

sql脱敏.png

2、通过ResponseBodyAdvice对@RestController返回值重写
借助的是AOP思想,对返回值进行一次修改,重写ResponseBodyAdvice接口这种方法适用于统一对请求前后进行一些操作的场景,比如说对数据的加解密等
这里脱敏采用的是hutool的公共方法(DesensitizedUtil)
信息脱敏工具-DesensitizedUtil参考链接

@Slf4j
@ControllerAdvice(basePackages = "xxx.application.controller")
public class DesensitizationFilter implements ResponseBodyAdvice {


    @Override
    public boolean supports(MethodParameter returnType, Class converterType) {
        return true;
    }

    @Override
    public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType, Class selectedConverterType, ServerHttpRequest request, ServerHttpResponse response) {
        try {
            Method method = (Method) ((MethodParameter) returnType).getMethod();
            switch (method.getName()) {
                case "searchDeviceInfo":
                    return searchDeviceInfo(body);
                default:
                    return body;
            }
        } catch (Throwable e) {
            log.info("返回对象转换出错, body = ", body, e);
            return body;
        }
    }
private Object searchDeviceInfo(Object body){
        Result<PageInfo<DeviceRespVO>> result = (Result<PageInfo<DeviceRespVO>>) body;
        List<DeviceRespVO> list = result.getData().getList();
        for(DeviceRespVO deviceRespVO :list){      
deviceRespVO.setToken(DesensitizedUtil.idCardNum(deviceRespVO.getToken(),3,3));
        }

        return result;
    }
}

看下请求返回结果

image.png

3、通过MapStruct转换

  @Mapping(expression = "java(cn.hutool.core.util.DesensitizedUtil.idCardNum(deviceRespInfoVO.getToken(),3,3))",target = "token")
  DeviceRespVO respInfo2vo(DeviceRespInfoVO deviceRespInfoVO);

转换结果同上


场景2:需要注意的是查询接口显示脱敏后,更新接口传入的数据就是脱敏后的数据,所以更新的时候需要过滤这种情况

image.png

解决方法如下:
1、通过MapStruct转换
自己写个过滤的方法,在MapStruct转换的时候转换下

@Mapper(componentModel = "spring", unmappedTargetPolicy = ReportingPolicy.IGNORE, uses = TypeConvertor.class)
public interface DeviceInfoMapStruct {

    @Mapping(expression = "java(xxx..tokenCheck(deviceUpdateDTO.getToken()))",target = "token")
    DeviceUpdateInfoDTO dto2InfoDto(DeviceUpdateDTO deviceUpdateDTO);
}

    /**
     * 自己写了一个简单的转化方法
     * @param token
     * @return
     */
public class MapStructUtils {

    public static String tokenCheck(String token){
        if(token != null && token.contains("*")){
            return null;
        }
        return token;
    }
}
更新的sql需要判空处理

2、通过ResponseBodyAdvice对@RestController返回值重写
新增注解

@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface DesensitizedReqField {
}

重写ResponseBodyAdvice方法,获取注解中字段进行判断

@Slf4j
@ControllerAdvice(basePackages = "xxx.application.controller")
public class RequestRewriteFilter implements RequestBodyAdvice {

    @Override
    public boolean supports(MethodParameter methodParameter, Type targetType, Class<? extends HttpMessageConverter<?>> converterType) {
        if (targetType.getTypeName().contains("DeviceUpdateDTO")) {
            return true;
        }
        return false;
    }

    @Override
    public HttpInputMessage beforeBodyRead(HttpInputMessage inputMessage, MethodParameter parameter, Type targetType, Class<? extends HttpMessageConverter<?>> converterType) throws IOException {
        return inputMessage;
    }

    @Override
    public Object afterBodyRead(Object body, HttpInputMessage inputMessage, MethodParameter parameter, Type targetType, Class<? extends HttpMessageConverter<?>> converterType) {
        try {
            if (targetType.getTypeName().contains("DeviceUpdateDTO")) {
                DeviceUpdateDTO DeviceUpdateDTO = (DeviceUpdateDTO) body;
                Field[] fields = ReflectUtil.getFields(DeviceUpdateDTO.class);
                for (Field field : fields) {
                    DesensitizedReqField desensitizationReqValue = field.getAnnotation(DesensitizedReqField.class);
                    if (String.class.equals(field.getType()) && desensitizationReqValue != null) {
                        String value = (String) ReflectUtil.getFieldValue(DeviceUpdateDTO, field);
                        // 如果有DesensitizationReqValue注解,并且内容包括*,则替换为null
                        if (value != null && value.contains("*")) {
                            ReflectUtil.setFieldValue(DeviceUpdateDTO, field, null);
                        }
                    }

                }
                return DeviceUpdateDTO;
            }

        } catch (Throwable e) {
            log.warn("请求转换异常", e);
        }

        return body;
    }

    @Override
    public Object handleEmptyBody(Object body, HttpInputMessage inputMessage, MethodParameter parameter, Type targetType, Class<? extends HttpMessageConverter<?>> converterType) {
        return body;
    }
}

使用时直接在需要检测的字段上添加注解即可


小结
本文介绍了两种查询数据脱敏的方式,以及避免更新接口会更新脱敏字段覆盖数据库原有字段的方式,数据脱敏的方法还有很多,比如自定义注解实现动态脱敏,可以自行了解


参考链接
1、RequestBodyAdvice用法详解-参数加解密示例
2、java自定义注解

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,761评论 5 460
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,953评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,998评论 0 320
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,248评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,130评论 4 356
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,145评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,550评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,236评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,510评论 1 291
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,601评论 2 310
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,376评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,247评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,613评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,911评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,191评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,532评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,739评论 2 335