java打印日志时,如何对字段进行脱敏?

在我们开发项目的时候,有些字段比较敏感,比如用户信息,这就需要在打印日志的时候对相关字段脱敏处理,本文提供的脱敏方案是使用conversionRule标签的方式,通过继承MessageConverter,在打印日志的时候对相关字段进行脱敏。
第一步,创建类继承MessageConverter,重写convert方法,如下:

public class SensitiveMessageConverter extends MessageConverter {
    /**
     * 正则匹配模式 
     */
    public static final Pattern REGEX_PATTERN = Pattern.compile("\\s*([\"]?[\\w]+[\"]?)(\\s*[:=]+[^\\u4e00-\\u9fa5@,.*{\\[\\w]*\\s*)([\\u4e00-\\u9fa5_\\-@.\\w]+)[\\W&&[^\\-@.]]?\\s*");

    @Override
    public String convert(ILoggingEvent event) {
        // 获取原始日志
        String formattedMessage = event.getFormattedMessage();
        return doConvert(formattedMessage);
    }

    private String doConvert(String formattedMessage) {
        if (StringUtils.isBlank(formattedMessage)) return formattedMessage;
        // 获取是否脱敏开关,可分环境配置是否开启
        Boolean convertSwitch = ApolloPropertyUtil.getBooleanProperty("log.sensitive.convert.switch", true);
        // 获取需要脱敏的字段。配置模板(key为脱敏模式,value为脱敏字段,可以有多个,other不脱敏):
        // {"name":["name","userName"],"mobilePhone":["phone"],"password":["password"],"idCard":["idcard"],"email":["email"],"other":["address"]}
        JSONObject jsonObject = ApolloPropertyUtil.getJSONObjectProperty("log.sensitive.convert.fields", "");
        // 是否开启脱敏开关
        if (!convertSwitch) return formattedMessage;
        if (Objects.isNull(jsonObject)) return formattedMessage;
        Matcher matcher = REGEX_PATTERN.matcher(formattedMessage);
        while (matcher.find()) {
            // 字段
            String key = matcher.group(1);
            key = key.replaceAll("\"", "");
            // 字段值
            String value = matcher.group(3);
            String mode = "";
            Iterator<String> iterator = jsonObject.keySet().iterator();
            while (iterator.hasNext()) {
                String next = iterator.next();
                List<String> list = (List<String>) jsonObject.get(next);
                if (list.contains(key)) {
                    // 命中当前属性属于哪一种脱敏模式(只能命中一种),目前支持 phone、password、idcard、other
                    mode = next;
                    break;
                }
            }
            // 没有命中脱敏模式,或者属性值为空,或者属性值为”null“,直接返回
            if (StringUtils.isAnyBlank(value, mode) || "null".equals(value)) continue;
            String afterValue = desensitize(mode, value);
            String origin = matcher.group(1) + matcher.group(2) + matcher.group(3);
            formattedMessage = formattedMessage.replace(origin, matcher.group(1) + matcher.group(2) + afterValue);
        }
        return formattedMessage;
    }

    /**
     * 脱敏
     *
     * @param mode 脱敏模式
     * @param value 脱敏的内容
     * @return 脱敏后的内容
     */
    private String desensitize(String mode, String value) {
        SensitiveModeEnum sensitiveModeEnum = SensitiveModeEnum.get(mode);
        // 获取不到相关脱敏模式,暂不支持,返回原日志
        if (Objects.isNull(sensitiveModeEnum)) return value;
        char[] chars = value.toCharArray();
        // 具体模式的策略可以再扩展(比如可以支持指定下标范围,指定正则,指定符号前面几位或者后面几位脱敏等等),这里暂不展开
        switch (sensitiveModeEnum) {
            case NAME:
                chars[0] = '*';
                break;
            case MOBILE_PHONE:
                for (int i = 3; i < 7; i++) {
                    chars[i] = '*';
                }
                break;
            case TELEPHONE:
                for (int i = 3; i < 5; i++) {
                    chars[i] = '*';
                }
                break;
            case ID_CARD:
                for (int i = 9; i < 13; i++) {
                    chars[i] = '*';
                }
                break;
            case PASSWORD:
                Arrays.fill(chars, '*');
                break;
            case EMAIL:
                for (int i = 3; i < 6; i++) {
                    chars[i] = '*';
                }
                break;
            case OTHER:
                break;
        }
        return new String(chars);
    }
}

第二步,在logback.xml中增加 conversionRule 标签

conversionWord是输入的日志信息,converterClass对应上面增加的类

<conversionRule conversionWord="msg" converterClass="xxx.xxx.log.SensitiveMessageConverter"/>

测试

@Test
    public void test() {
        HashMap<Object, Object> map = new HashMap<>();
        Map<String, String> map1 = new HashMap<>();
        map1.put("email", "1234567@qq.com");
        map1.put("address", "上海市详细地址");
        map.put("name", "张三分");
        map.put("sendPhone", "18912430987");
        map.put("password", "123456");
        map.put("detail", map1);
        log.info("结果:{}", JSON.toJSONString(map));
}

结果:

结果:{"password":"******","name":"*三分","sendPhone":"189****0987","detail":{"address":"上海市详细地址","email":"123***7@qq.com"}} 
  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在 Java 中,可以使用以下方法来实现对某些字段进行脱敏: 1. 创建一个脱敏类,包含脱敏所需的方法,如手机号脱敏、身份证号脱敏等。 2. 在需要脱敏的地方,调用该脱敏类中对应的方法进行脱敏。 例如,下面是一个简单的脱敏类的示例: ``` public class DesensitizationUtil { public static String desensitizeMobile(String mobile) { return mobile.replaceAll("(\\d{3})\\d{4}(\\d{4})", "$1****$2"); } public static String desensitizeIdCard(String idCard) { return idCard.replaceAll("(\\d{4})\\d{10}(\\w{4})", "$1**********$2"); } } ``` 然后,在需要脱敏的地方调用这些方法即可: ``` String mobile = "13812345678"; String desensitizedMobile = DesensitizationUtil.desensitizeMobile(mobile); // 输出:138****5678 String idCard = "420123198806251234"; String desensitizedIdCard = DesensitizationUtil.desensitizeIdCard(idCard); // 输出:4201**********1234 ``` 注意:这只是一个简单的示例,实际应用中可能需要更复杂的脱敏方法。 ### 回答2: 在Java中,可以通过一些方法实现对某些字段进行脱敏的操作。下面是一种常见的实现方式。 1. 首先,定义一个脱敏处理工具类,例如DesensitizationUtil,该类包含各种脱敏处理方法。 2. 对于需要脱敏字段,可以根据不同的需求选择不同的脱敏方法。以下是一些常见的脱敏方法: - 姓名脱敏:对于姓名字段,可以将姓氏保留,将名字部分用星号或其他字符替代,例如把"张三"脱敏为"张*" - 身份证号脱敏:对于身份证号字段,可以保留前几位和后几位,中间部分用星号或其他字符替代,例如把"110101199001011234"脱敏为"110***********234" - 手机号脱敏:对于手机号字段,可以保留前3位和后4位,中间部分用星号或其他字符替代,例如把"13812345678"脱敏为"138****5678" - 邮箱脱敏:对于邮箱字段,可以保留邮箱用户名的前几个字符,后面用星号或其他字符替代,例如把"example@example.com"脱敏为"exa**********m" 3. 在具体的业务类中,将需要脱敏字段通过调用脱敏工具类的方法进行处理,例如: ```java String name = "张三"; String desensitizedName = DesensitizationUtil.desensitizeName(name); System.out.println(desensitizedName); // 输出:张* String idCard = "110101199001011234"; String desensitizedIdCard = DesensitizationUtil.desensitizeIdCard(idCard); System.out.println(desensitizedIdCard); // 输出:110***********234 String phoneNumber = "13812345678"; String desensitizedPhoneNumber = DesensitizationUtil.desensitizePhoneNumber(phoneNumber); System.out.println(desensitizedPhoneNumber); // 输出:138****5678 String email = "example@example.com"; String desensitizedEmail = DesensitizationUtil.desensitizeEmail(email); System.out.println(desensitizedEmail); // 输出:exa**********m ``` 通过以上步骤,就可以方便地实现对某些字段脱敏操作。这样可以有效保护敏感信息,避免隐私泄露的风险。 ### 回答3: Java可以通过一些方法实现对某些字段进行脱敏脱敏是一种隐私保护技术,在保护敏感数据信息的同将其进行处理,以便在某些场景下限制敏感数据的可见性。 一种实现方式是利用字符串操作方法对字段进行脱敏。比如对于手机号码,可以将其中间的几位数字替换为特定的符号或者统一的数字。另外,可以使用正则表达式来匹配指定的字段进行替换。 另一种方法是利用加密算法对字段进行加密处理。通过使用一定的加密方法,将敏感字段进行加密,并在需要使用进行解密。这种方式可以保证数据的安全性,但对于某些需要明文数据的场景可能不适用。 还有一种方法是利用数据库存储过程对字段进行脱敏。通过在数据库中创建存储过程来对字段进行处理,可以实现对特定数据进行脱敏。这种方式可以在数据库层面进行数据保护,对于需要对数据库中所有数据进行脱敏的情况比较适用。 此外,可以使用开源库或框架来实现字段脱敏,如Apache Commons Lang库中的StringUtils类提供了丰富的字符串处理方法,也可以结合正则表达式进行字段匹配和替换。 总的来说,Java可以通过字符串操作、加密算法、数据库存储过程等多种方法实现对某些字段脱敏。具体的实现方式需根据实际需求和场景进行选择。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值