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

一个注解搞定接口返回数据脱敏...

01

背景

下午惬意时光,突然产品小姐姐走到我面前,打断我短暂的摸鱼 time,企图与我进行深入交流,还好我早有防备没有闪,打开瑞 star 的点单页面,暗示没有一杯 coffee 解决不了的需求。

需求是某些接口返回的信息,涉及到敏感数据的必须进行脱敏操作,我思考一反,表示某问题,马上安排。

02

思路

①要做成可配置多策略的脱敏操作,要不然一个个接口进行脱敏操作,重复的工作量太多,很显然违背了“多写一行算我输”的程序员规范。

思来想去,定义数据脱敏注解和数据脱敏逻辑的接口, 在返回类上,对需要进行脱敏的属性加上,并指定对应的脱敏策略操作。

②接下来我只需要拦截控制器返回的数据,找到带有脱敏注解的属性操作即可,一开始打算用 @ControllerAdvice 去实现,但发现需要自己去反射类获取注解。

当返回对象比较复杂,需要递归去反射,性能一下子就会降低,于是换种思路,我想到平时使用的 @JsonFormat,跟我现在的场景很类似,通过自定义注解跟字段解析器,对字段进行自定义解析,tql。

03

实现代码

①自定义数据注解,并可以配置数据脱敏策略:

@Target({ElementType.FIELD, ElementType.TYPE}) @Retention(RetentionPolicy.RUNTIME) @Documented public @interface DataMasking { DataMaskingFunc maskFunc() default DataMaskingFunc.NO_MASK; }

②自定义 Serializer,参考 jackson 的 StringSerializer,下面的示例只针对 String 类型进行脱敏。

public interface DataMaskingOperation { String MASK_CHAR = "*"; String mask(String content, String maskChar); } publicenum DataMaskingFunc { /** * 脱敏转换器 */ NO_MASK((str, maskChar) -> { return str; }), ALL_MASK((str, maskChar) -> { if (StringUtils.hasLength(str)) { StringBuilder sb = new StringBuilder(); for (int i = 0; i < str.length(); i++) { sb.append(StringUtils.hasLength(maskChar) ? maskChar : DataMaskingOperation.MASK_CHAR); } return sb.toString(); } else { return str; } }); privatefinal DataMaskingOperation operation; private DataMaskingFunc(DataMaskingOperation operation) { this.operation = operation; } public DataMaskingOperation operation() { returnthis.operation; } } publicfinalclass DataMaskingSerializer extends StdScalarSerializer<Object> { privatefinal DataMaskingOperation operation; public DataMaskingSerializer() { super(String.class, false); this.operation = null; } public DataMaskingSerializer(DataMaskingOperation operation) { super(String.class, false); this.operation = operation; } public boolean isEmpty(SerializerProvider prov, Object value) { String str = (String)value; return str.isEmpty(); } public void serialize(Object value, JsonGenerator gen, SerializerProvider provider) throws IOException { if (Objects.isNull(operation)) { String content = DataMaskingFunc.ALL_MASK.operation().mask((String) value, null); gen.writeString(content); } else { String content = operation.mask((String) value, null); gen.writeString(content); } } public final void serializeWithType(Object value, JsonGenerator gen, SerializerProvider provider, TypeSerializer typeSer) throws IOException { this.serialize(value, gen, provider); } public JsonNode getSchema(SerializerProvider provider, Type typeHint) { returnthis.createSchemaNode("string", true); } public void acceptJsonFormatVisitor(JsonFormatVisitorWrapper visitor, JavaType typeHint) throws JsonMappingException { this.visitStringFormat(visitor, typeHint); } }

③自定义 AnnotationIntrospector,适配我们自定义注解返回相应的 Serializer。

@Slf4j public class DataMaskingAnnotationIntrospector extends NopAnnotationIntrospector { @Override public Object findSerializer(Annotated am) { DataMasking annotation = am.getAnnotation(DataMasking.class); if (annotation != null) { return new DataMaskingSerializer(annotation.maskFunc().operation()); } return null; } }

④覆盖 ObjectMapper:

@Configuration( proxyBeanMethods = false ) publicclass DataMaskConfiguration { @Configuration( proxyBeanMethods = false ) @ConditionalOnClass({Jackson2ObjectMapperBuilder.class}) staticclass JacksonObjectMapperConfiguration { JacksonObjectMapperConfiguration() { } @Bean @Primary ObjectMapper jacksonObjectMapper(Jackson2ObjectMapperBuilder builder) { ObjectMapper objectMapper = builder.createXmlMapper(false).build(); AnnotationIntrospector ai = objectMapper.getSerializationConfig().getAnnotationIntrospector(); AnnotationIntrospector newAi = AnnotationIntrospectorPair.pair(ai, new DataMaskingAnnotationIntrospector()); objectMapper.setAnnotationIntrospector(newAi); return objectMapper; } } }

⑤返回对象加上注解:

public class User implements Serializable { /** * 主键ID */ private Long id; /** * 姓名 */ @DataMasking(maskFunc = DataMaskingFunc.ALL_MASK) private String name; /** * 年龄 */ private Integer age; /** * 邮箱 */ @DataMasking(maskFunc = DataMaskingFunc.ALL_MASK) private String email; }
http://www.jsqmd.com/news/321799/

相关文章:

  • 总有机碳(TOC)分析仪原理
  • 郑州留学中介排名解析,申请成功率高的关键因素揭晓
  • 【C#】求目标Vector2向量与Vector2.Right方向上的夹角 θ
  • Spring的反射与动态代理
  • 日化产品想贴牌?源头厂家直供,成本降一半
  • 接口性能优化的11个小技巧
  • RAG 深度实践系列(六):基于科大讯飞 RAG + 星火知识库的企业级实战指南
  • 基于Arduino智能家居环境监测系统
  • Spring的自定义注解与处理器
  • 超级浏览器是什么?RoxyBrowser浏览器怎么样?一篇文章看明白!
  • 如何设计接口测试用例?
  • Git 中的 Rebase 与 Merge:原理、区别与最佳实践
  • 实时音视频通信技术解析:WebRTC核心原理与实战
  • RISC-V IDE MRS2使用笔记(八):手动切换文件编码
  • 能为你加分的性能测试
  • B2B商城系统如何选择?解析千匠网络的三大核心优势
  • 前端框架演进史:从jQuery到Vue 3的架构变迁
  • 基于单片机的酒精监测系统
  • 口罩机通用程序 已经升级一拖一7,8,9,10 伺服口罩机通用程序架构, 程序高度模块化, 可...
  • 2026年4米2高栏货车经销商综合评估报告:重载运输场景首选品牌推荐
  • CrossFormer 实现图像分类以及视觉任务的骨干网络替换 它使用交替的局部和全局注意力击...
  • 计算机毕业设计之基于springboot的学测评系统设计与实现
  • 自动化测试框架搭建:Selenium与Pytest集成指南
  • 容器编排进阶:Kubernetes Operator设计与实现
  • 千匠网络助力省级龙头企业打造农产品供应链平台
  • 解锁飞行焊接:电芯顶盖封口的高效与精准密码
  • 普洱市英语雅思培训辅导机构推荐-2026权威出国雅思课程中心学校口碑排行榜
  • 2026年全国坚果炒货连锁批发巧克力生产厂家排行榜及全景解析与参考
  • 前端工程化实践:Webpack 5配置优化与插件开发
  • 别再瞎选场镜(F-Theta Lens)啦!焦距、光斑、景深关系大揭秘