加密工具 (crypto.ts)
加密工具类,提供全面的加密、解密、哈希计算等安全功能,基于 CryptoJS 库实现。
📖 概述
加密工具库包含以下功能类别:
- 随机生成:生成随机字符串和AES密钥
- Base64处理:Base64编码与解码
- AES加密/解密:使用AES密钥加密和解密数据
- 高级加解密:自动密钥生成的加密和支持JSON解析的解密
- 哈希计算:计算数据的哈希值,用于唯一标识
🔧 依赖说明
本工具基于 crypto-js
库,使用前请确保已安装:
bash
npm install crypto-js
npm install @types/crypto-js # TypeScript类型定义
🎲 随机生成
generateRandomString
随机生成32位的字符串,包含大小写字母和数字。
typescript
generateRandomString(): string
返回值:
string
- 32位随机字符串
示例:
typescript
// 生成随机字符串
const randomStr = generateRandomString()
console.log(randomStr) // 输出类似:'aB3dE9fH2jK7mN1pQ5rS8tU4vW6xY0zC'
// 用于生成临时密码
const tempPassword = generateRandomString()
generateAesKey
随机生成AES密钥,用于AES加密操作。
typescript
generateAesKey(): CryptoJS.lib.WordArray
返回值:
CryptoJS.lib.WordArray
- AES密钥对象
示例:
typescript
// 生成AES密钥
const aesKey = generateAesKey()
// 使用密钥进行加密
const encrypted = encryptWithAes('Hello World', aesKey)
📄 Base64处理
encodeBase64
将 WordArray 数据编码为Base64字符串。
typescript
encodeBase64(data: CryptoJS.lib.WordArray): string
参数:
data
- 要编码的WordArray数据
返回值:
string
- Base64编码后的字符串
示例:
typescript
import CryptoJS from 'crypto-js'
// 将字符串转为WordArray并编码
const message = CryptoJS.enc.Utf8.parse('Hello World')
const encoded = encodeBase64(message)
console.log(encoded) // 输出:'SGVsbG8gV29ybGQ='
decodeBase64
将Base64字符串解码为 WordArray 数据。
typescript
decodeBase64(str: string): CryptoJS.lib.WordArray
参数:
str
- Base64编码的字符串
返回值:
CryptoJS.lib.WordArray
- 解码后的数据
示例:
typescript
// 解码Base64字符串
const decoded = decodeBase64('SGVsbG8gV29ybGQ=')
const text = CryptoJS.enc.Utf8.stringify(decoded)
console.log(text) // 输出:'Hello World'
🔐 AES加密/解密
encryptWithAes
使用AES密钥加密数据,采用ECB模式和PKCS7填充。
typescript
encryptWithAes(message: string, aesKey: CryptoJS.lib.WordArray): string
参数:
message
- 要加密的消息aesKey
- AES密钥
返回值:
string
- 加密后的字符串
示例:
typescript
// 生成密钥并加密
const key = generateAesKey()
const encrypted = encryptWithAes('敏感数据', key)
console.log(encrypted) // 输出加密后的字符串
decryptWithAes
使用AES密钥解密数据。
typescript
decryptWithAes(message: string, aesKey: CryptoJS.lib.WordArray): string
参数:
message
- 加密的消息aesKey
- AES密钥
返回值:
string
- 解密后的字符串
示例:
typescript
// 解密数据
const key = generateAesKey()
const encrypted = encryptWithAes('敏感数据', key)
const decrypted = decryptWithAes(encrypted, key)
console.log(decrypted) // 输出:'敏感数据'
⚡ 高级加解密
encryptWithAutoKey
一步完成加密过程:自动生成密钥、加密数据并返回结果。
typescript
encryptWithAutoKey(data: string | object): {
encryptedData: string
key: CryptoJS.lib.WordArray
}
参数:
data
- 要加密的数据(字符串或对象)
返回值:
object
- 包含加密数据和密钥的对象encryptedData
- 加密后的数据key
- 使用的AES密钥
示例:
typescript
// 加密字符串
const result1 = encryptWithAutoKey('Hello World')
console.log(result1.encryptedData) // 加密后的数据
console.log(result1.key) // 生成的密钥
// 加密对象
const userInfo = { name: 'John', age: 30, email: 'john@example.com' }
const result2 = encryptWithAutoKey(userInfo)
// 对象会自动转为JSON字符串后加密
decryptWithParsing
使用提供的密钥解密数据,支持JSON解析。
typescript
decryptWithParsing(
encryptedData: string,
key: CryptoJS.lib.WordArray,
parseJson?: boolean
): string | object
参数:
encryptedData
- 加密的数据key
- 解密密钥parseJson
- 是否将结果解析为JSON对象,默认为false
返回值:
string | object
- 解密后的数据
示例:
typescript
// 解密字符串
const { encryptedData, key } = encryptWithAutoKey('Hello World')
const decrypted1 = decryptWithParsing(encryptedData, key)
console.log(decrypted1) // 输出:'Hello World'
// 解密并解析为JSON
const userInfo = { name: 'John', age: 30 }
const result = encryptWithAutoKey(userInfo)
const decrypted2 = decryptWithParsing(result.encryptedData, result.key, true)
console.log(decrypted2) // 输出:{ name: 'John', age: 30 }
🔍 哈希计算
computeSha256Hash
计算字符串的SHA-256哈希值。
typescript
computeSha256Hash(data: string): string
参数:
data
- 要计算哈希的数据
返回值:
string
- 哈希值(十六进制表示)
示例:
typescript
// 计算SHA-256哈希
const hash = computeSha256Hash('Hello World')
console.log(hash) // 输出:'a591a6d40bf420404a011733cfb7b190d62c65bf0bcda32b57b277d9ad9f146e'
// 用于密码哈希(简单示例,实际应用建议加盐)
const passwordHash = computeSha256Hash('user_password')
computeMd5Hash
计算字符串的MD5哈希值。
typescript
computeMd5Hash(data: string): string
参数:
data
- 要计算哈希的数据
返回值:
string
- 哈希值(十六进制表示)
示例:
typescript
// 计算MD5哈希
const md5Hash = computeMd5Hash('Hello World')
console.log(md5Hash) // 输出:'b10a8db164e0754105b7a99be72e3fe5'
// 用于文件完整性校验
const fileContent = 'file content here'
const checksum = computeMd5Hash(fileContent)
generateImageHash
生成图片的唯一哈希标识,基于图片的Base64数据。
typescript
generateImageHash(base64Data: string): string
参数:
base64Data
- 图片的Base64编码数据
返回值:
string
- 图片的唯一哈希值
示例:
typescript
// 生成图片哈希
const imageBase64 = ''
const imageHash = generateImageHash(imageBase64)
console.log(imageHash)
// 用于图片去重
const uploadedImages = new Set()
if (uploadedImages.has(imageHash)) {
console.log('图片已存在')
} else {
uploadedImages.add(imageHash)
console.log('新图片')
}
generateFileHash
异步生成文件的哈希标识,支持多种文件类型。
typescript
generateFileHash(file: Blob | File | ArrayBuffer): Promise<string>
参数:
file
- 文件或数据流(Blob、File或ArrayBuffer)
返回值:
Promise<string>
- 文件的哈希值
示例:
typescript
// 处理用户上传的文件
const handleFileUpload = async (file: File) => {
try {
const fileHash = await generateFileHash(file)
console.log('文件哈希:', fileHash)
// 检查文件是否已存在
const existingFiles = await checkFileExists(fileHash)
if (existingFiles) {
console.log('文件已存在,跳过上传')
return
}
// 上传新文件
await uploadFile(file, fileHash)
} catch (error) {
console.error('生成文件哈希失败:', error)
}
}
// 处理ArrayBuffer
const buffer = new ArrayBuffer(1024)
const bufferHash = await generateFileHash(buffer)
💡 实际应用场景
1. 用户敏感信息加密
typescript
// 加密用户个人信息
const sensitiveInfo = {
idCard: '123456789012345678',
bankCard: '6222020111122220000',
phone: '13812345678'
}
const { encryptedData, key } = encryptWithAutoKey(sensitiveInfo)
// 存储加密数据和密钥(实际应用中密钥需要安全保存)
localStorage.setItem('encrypted_info', encryptedData)
localStorage.setItem('info_key', encodeBase64(key))
// 解密使用
const storedData = localStorage.getItem('encrypted_info')
const storedKey = decodeBase64(localStorage.getItem('info_key'))
const decryptedInfo = decryptWithParsing(storedData, storedKey, true)
2. API接口参数加密
typescript
// 加密API请求参数
const apiParams = {
userId: 12345,
action: 'transfer',
amount: 1000,
targetAccount: '6222020111122220000'
}
const encryptedParams = encryptWithAutoKey(apiParams)
// 发送加密请求
const response = await fetch('/api/secure-operation', {
method: 'POST',
headers: {
'Content-Type': 'application/json',
'Encryption-Key': encodeBase64(encryptedParams.key)
},
body: JSON.stringify({
data: encryptedParams.encryptedData
})
})
3. 文件上传去重
typescript
// 文件上传前检查重复
const handleFileUpload = async (files: FileList) => {
const uploadQueue = []
for (const file of files) {
const fileHash = await generateFileHash(file)
// 检查服务器是否已有该文件
const exists = await checkFileExists(fileHash)
if (!exists) {
uploadQueue.push({
file,
hash: fileHash
})
} else {
console.log(`文件 ${file.name} 已存在,跳过上传`)
}
}
// 批量上传不重复的文件
if (uploadQueue.length > 0) {
await uploadFiles(uploadQueue)
}
}
4. 数据完整性校验
typescript
// 下载文件时验证完整性
const downloadAndVerify = async (url: string, expectedHash: string) => {
try {
const response = await fetch(url)
const fileData = await response.arrayBuffer()
const actualHash = await generateFileHash(fileData)
if (actualHash === expectedHash) {
console.log('文件完整性验证通过')
return fileData
} else {
throw new Error('文件完整性验证失败')
}
} catch (error) {
console.error('下载或验证失败:', error)
throw error
}
}
5. 缓存键生成
typescript
// 根据查询参数生成缓存键
const generateCacheKey = (queryParams: object): string => {
const paramString = JSON.stringify(queryParams)
return computeSha256Hash(paramString)
}
// 使用示例
const query = {
page: 1,
size: 20,
filters: {
status: 'active',
category: 'electronics'
}
}
const cacheKey = generateCacheKey(query)
const cachedData = localStorage.getItem(cacheKey)
if (cachedData) {
console.log('使用缓存数据')
} else {
const data = await fetchData(query)
localStorage.setItem(cacheKey, JSON.stringify(data))
}
⚠️ 安全注意事项
1. 密钥管理
- 不要将密钥存储在客户端:密钥应该安全地存储在服务端
- 定期更换密钥:建议定期更换加密密钥
- 密钥传输安全:密钥传输时必须使用安全通道(HTTPS)
2. 加密强度
- AES-256:建议使用AES-256而不是AES-128
- 避免ECB模式:生产环境建议使用CBC或GCM模式
- 使用随机IV:每次加密使用不同的初始化向量
3. 哈希安全
- 密码哈希加盐:密码哈希必须使用随机盐值
- 避免MD5:MD5已不安全,推荐使用SHA-256或更强的算法
- 防止彩虹表攻击:使用足够复杂的盐值
4. 前端加密限制
- 前端加密非绝对安全:客户端代码可被逆向,仅用于基础保护
- 服务端验证必要:重要数据必须在服务端进行二次验证
- HTTPS传输:始终使用HTTPS传输敏感数据
🛡️ 最佳实践
1. 分层加密策略
typescript
// 多层加密保护敏感数据
const multiLayerEncrypt = (data: string) => {
// 第一层:AES加密
const firstLayer = encryptWithAutoKey(data)
// 第二层:Base64编码
const secondLayer = encodeBase64(
CryptoJS.enc.Utf8.parse(firstLayer.encryptedData)
)
return {
data: secondLayer,
key: firstLayer.key
}
}
2. 安全的密钥派生
typescript
// 从密码派生密钥(示例)
const deriveKeyFromPassword = (password: string, salt: string): CryptoJS.lib.WordArray => {
return CryptoJS.PBKDF2(password, salt, {
keySize: 256/32, // 256位密钥
iterations: 10000 // 迭代次数
})
}
3. 错误处理
typescript
const safeEncrypt = (data: string): string | null => {
try {
const result = encryptWithAutoKey(data)
return result.encryptedData
} catch (error) {
console.error('加密失败:', error)
return null
}
}
const safeDecrypt = (encryptedData: string, key: CryptoJS.lib.WordArray): string | null => {
try {
return decryptWithAes(encryptedData, key)
} catch (error) {
console.error('解密失败:', error)
return null
}
}