news 2026/4/23 14:32:31

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

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
一个注解搞定接口返回数据脱敏...

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; }
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/23 11:25:56

如何设计接口测试用例?

接口测试用例设计其实跟功能测试用例设计是类似的&#xff0c;从根本上来说都是验证产品功能的实现情况。但是功能测试偏向于测试前端数据的展示、业务逻辑&#xff0c;但是有些情况是前端无法模拟的&#xff08;如前端的支付金额无法为负值&#xff09;&#xff0c;此时就需要…

作者头像 李华
网站建设 2026/4/23 12:57:58

日化产品想贴牌?源头厂家直供,成本降一半

想要打造属于自己的洗发水、洗衣液、面膜或者洗手液品牌吗?贴牌生产&#xff08;OEM&#xff09;确实是一条最为快捷的路径。但许多刚开始涉足这个领域的新手&#xff0c;在第一步就遇到了麻烦&#xff1a;他们找不到真正的源头厂家&#xff0c;反而被贸易商一层又一层地加价&…

作者头像 李华
网站建设 2026/3/29 11:25:55

JAVA理发预约系统:同城服务秒速响应

JAVA理发预约系统&#xff1a;同城服务秒速响应&#xff0c;打造高效、智能、个性化的美发体验 在快节奏的都市生活中&#xff0c;理发作为高频刚需服务&#xff0c;用户对“快速预约、精准匹配、透明服务”的需求日益迫切。JAVA凭借其跨平台、高并发、安全稳定等特性&#xff…

作者头像 李华
网站建设 2026/4/23 12:57:10

一键预约理发,JAVA系统畅享同城便利

一键预约理发&#xff0c;JAVA系统畅享同城便利——打造高效、智能、无缝衔接的美发体验在快节奏的都市生活中&#xff0c;“一键预约理发”已成为用户对同城服务的核心需求。JAVA凭借其高并发处理能力、跨平台兼容性、安全稳定性&#xff0c;成为构建理发预约系统的理想选择。…

作者头像 李华
网站建设 2026/4/23 12:45:02

是德科技DSOX4054A DSOX4024A DSOX4034A示波器

是德DSOX4054A示波器说明&#xff1a;   DSOX4054A 示波器&#xff1a;500 MHz&#xff0c;4 个模拟通道。   4000 X 系列拥有一系列引以为傲的配置&#xff0c;包括采用了电容触摸屏技术的 12.1 英寸显示屏、InfiniiScan 区域触摸触发、100 万波形/秒捕获率、MegaZoom…

作者头像 李华
网站建设 2026/4/7 16:11:27

NMFC04处理器模块

NMFC04 处理器模块简介NMFC04 处理器模块用于工业控制系统中&#xff0c;负责核心运算、数据处理及系统运行协调。作为系统的核心运算单元&#xff0c;承担控制处理任务具备较强的数据处理与运算能力控制响应速度快&#xff0c;满足实时控制需求支持多任务并行处理与各类功能模…

作者头像 李华