使用TypeScript和裝飾器實(shí)現(xiàn)前端數(shù)據(jù)脫敏
一、需求背景
雖然我經(jīng)常在很多地方都說過,“前端脫敏,脫褲子放屁。”
但在用戶需要截圖,或者是數(shù)據(jù)極其敏感的情況下,不管后端是否脫敏,前端都還是應(yīng)該給人脫敏一下的。
否則用戶截圖的話,就需要先截圖,再用截圖工具抹掉敏感信息后再發(fā)送,如果是極其敏感的數(shù)據(jù)的話。
所以需求來了:
要求在前端表格中的所有敏感數(shù)據(jù)列都需要脫敏,且需要支持用戶點(diǎn)擊查看原始數(shù)據(jù)。
二、需求目標(biāo)
2.1 實(shí)現(xiàn)效果

2.2 如何使用

2.3 自定義脫敏
只需要在需要脫敏的屬性上的 @Table() 裝飾器中配置脫敏方式即可。當(dāng)然,也支持直接自定義脫敏:

實(shí)現(xiàn)的效果就這樣咯:

三、實(shí)現(xiàn)過程
3.1 @Table參數(shù)聲明
我們需要在負(fù)責(zé)管理表格狀態(tài)的 @Table() 裝飾器的參數(shù)中添加下面一些如 2.3 截圖中的配置項(xiàng):
3.1.1 desensitize 脫敏類型
配置這個后才能自動脫敏,該配置是個枚舉選項(xiàng):
static TELEPHONE = new AirDesensitizeType(0, '座機(jī)號碼', 3, 4) static MOBILE = new AirDesensitizeType(1, '手機(jī)號碼', 3, 4) static ID_CARD = new AirDesensitizeType(2, '身份證號', 6, 4) static BANK_CARD = new AirDesensitizeType(3, '銀行卡號', 4, 4) static CAR_NUMBER = new AirDesensitizeType(4, '車牌號', 2, 1) static EMAIL = new AirDesensitizeType(5, '郵箱', 2, 2) static CHINESE_NAME = new AirDesensitizeType(6, '中文名', 1, 1) static ADDRESS = new AirDesensitizeType(7, '地址', 3, 0) static IP_V4 = new AirDesensitizeType(8, 'IPv4地址', 0, 0) static CUSTOM = new AirDesensitizeType(9, '自定義', 0, 0)
其中,末尾的兩個數(shù)字代表脫敏的頭部保留數(shù)量和尾部保留數(shù)量。
3.1.2 desensitizeHead
脫敏開始保留的數(shù)量,默認(rèn)為0,可覆蓋 3.1.1 中的 倒數(shù)第二 個參數(shù)。
3.1.3 desensitizeTail
脫敏結(jié)尾保留的數(shù)量,默認(rèn)為0,可覆蓋 3.1.1 中的 倒數(shù)第一 個參數(shù)。
3.1.4 desensitizeSymbol
脫敏符號,默認(rèn)為 *。
3.2 編寫脫敏助手類
我們定義了一個 AirDesensitize 助手類,用于實(shí)現(xiàn)脫敏邏輯。
/**
* # 脫敏助手類
*
* @author Hamm.cn
*/
export class AirDesensitize {
/**
* ## `IPv4` 的塊長度
*/
private static readonly IP_V4_PART_COUNT = 4
/**
* ## 字符串替換
*
* @param text 原始字符串
* @param head 頭部保留長度
* @param tail 尾部保留長度
* @param symbol 中間替換的單個符號
* @return 替換后的字符串
*/
public static replace(text: string, head: number, tail: number, symbol: string): string {
if (head < 0 || tail < 0 || head + tail >= text.length) {
return text
}
let str = ''
for (let i = 0; i < text.length; i += 1) {
if (i >= head && i <= text.length - tail - 1) {
str += symbol
} else {
str += text[i]
}
}
return str
}
/**
* ## `IPv4` 地址脫敏
*
* @param ipv4 `IPv4` 地址
* @param symbol [可選]符號
* @return 脫敏后的 `IPv4` 地址
*/
public static desensitizeIpv4Address(ipv4: string, symbol = AirConstant.ASTERISK): string {
const strings = ipv4.split(AirConstant.DOT)
if (strings.length !== AirDesensitize.IP_V4_PART_COUNT) {
return ipv4
}
const temp = symbol + symbol + symbol
strings[1] = temp
strings[2] = temp
return strings.join(AirConstant.DOT)
}
/**
* ## 文本脫敏
*
* @param valueString 原始文本
* @param type 脫敏類型
* @param head 頭部保留
* @param tail 尾部保留
* @param symbol 脫敏符號
* @return 脫敏后的文本
*/
public static desensitize(valueString: string, type: AirDesensitizeType, head = 0, tail = 0, symbol = AirConstant.ASTERISK): string {
switch (type.key) {
case AirDesensitizeType.BANK_CARD.key:
case AirDesensitizeType.ID_CARD.key:
case AirDesensitizeType.MOBILE.key:
case AirDesensitizeType.ADDRESS.key:
case AirDesensitizeType.CAR_NUMBER.key:
case AirDesensitizeType.EMAIL.key:
head = Math.max(type.head, head)
tail = Math.max(type.tail, tail)
break
case AirDesensitizeType.IP_V4.key:
return AirDesensitize.desensitizeIpv4Address(valueString, symbol)
case AirDesensitizeType.CHINESE_NAME.key:
head = Math.max(type.head, head)
tail = Math.max(type.tail, tail)
if (valueString.length <= head + tail) {
tail = 0
}
break
case AirDesensitizeType.TELEPHONE.key:
// 包含區(qū)號 前后各留4 不包含則各留2
// eslint-disable-next-line no-case-declarations
const isContainRegionCode = valueString.length > 8 ? 4 : 2
head = Math.max(isContainRegionCode, head)
tail = Math.max(isContainRegionCode, tail)
break
default:
}
return this.replace(valueString, head, tail, symbol)
}
}
3.3 給表格組件添加脫敏邏輯
<el-table-column>
<!-- ...省略... -->
<template v-if="columnConfig.desensitize">
{{ AirDesensitize.desensitize(getColumnRowData(row), columnConfig.desensitize, columnConfig.desensitizeHead, columnConfig.desensitizeTail, columnConfig.desensitizeSymbol) }}
</template>
<template v-else>
getColumnRowData(row)
</template>
<!-- ...省略... -->
</el-table-column>
我們添加一個 v-if 來判斷是否需要脫敏,然后使用 AirDesensitize 助手類提供的 desensitize 方法來脫敏即可。
好的,搞定!
接下來其他同事就只需要美滋滋的去給屬性的 @Table() 裝飾器標(biāo)參數(shù)即可~
到此這篇關(guān)于使用TypeScript和裝飾器實(shí)現(xiàn)前端數(shù)據(jù)脫敏的文章就介紹到這了,更多相關(guān)TypeScript數(shù)據(jù)脫敏內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
JS正則表達(dá)式實(shí)現(xiàn)字符串中連續(xù)在一起的字符去重
這篇文章主要給大家介紹了關(guān)于JS正則表達(dá)式實(shí)現(xiàn)字符串中連續(xù)在一起的字符去重的相關(guān)資料,學(xué)會正則表達(dá)式對開發(fā)者而言是個非常有用的技能,很多功能可以簡單的用一句正則來實(shí)現(xiàn),需要的朋友可以參考下2023-11-11
PHP自動加載autoload和命名空間的應(yīng)用小結(jié)
PHP的自動加載就是我們加載實(shí)例化類的時候,不需要手動去寫require來導(dǎo)入這個class.php文件,程序自動幫我們加載導(dǎo)入進(jìn)來這.篇文章主要介紹了PHP自動加載autoload和命名空的應(yīng)用,需要的朋友可以參考下2017-12-12
前端HTTP發(fā)POST請求攜帶參數(shù)與后端接口接收參數(shù)的實(shí)現(xiàn)
近期在學(xué)習(xí)的時候,碰到一個關(guān)于post的小問題,故拿出來分享一下,下面這篇文章主要給大家介紹了關(guān)于前端HTTP發(fā)POST請求攜帶參數(shù)與后端接口接收參數(shù)的相關(guān)資料,需要的朋友可以參考下2022-10-10
JavaScript之Date_動力節(jié)點(diǎn)Java學(xué)院整理
在JavaScript中,Date對象用來表示日期和時間。下面給大家介紹js中的date,需要的朋友參考下吧2017-06-06
前端JS,刪除JSON數(shù)據(jù)(JSON數(shù)組)中的指定元素方式
這篇文章主要介紹了前端JS,刪除JSON數(shù)據(jù)(JSON數(shù)組)中的指定元素方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-05-05

