157 lines
4.6 KiB
Go
157 lines
4.6 KiB
Go
|
package test
|
|||
|
|
|||
|
import (
|
|||
|
"bytes"
|
|||
|
"crypto/aes"
|
|||
|
"crypto/cipher"
|
|||
|
crand "crypto/rand"
|
|||
|
"encoding/base64"
|
|||
|
"io"
|
|||
|
"math/rand"
|
|||
|
"unsafe"
|
|||
|
)
|
|||
|
|
|||
|
const lettersString = "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
|
|||
|
|
|||
|
// 字符串长度
|
|||
|
const number = 16
|
|||
|
|
|||
|
/*
|
|||
|
16位码,前15位随机字符串,最后一位通过前15位字符串计算校验生成
|
|||
|
*/
|
|||
|
|
|||
|
func LotteryEncryptEncode() string {
|
|||
|
b := make([]byte, number)
|
|||
|
var sum byte
|
|||
|
for i := 0; i < number-1; i++ {
|
|||
|
b[i] = lettersString[rand.Int63()%int64(len(lettersString))]
|
|||
|
sum += b[i]
|
|||
|
}
|
|||
|
b[number-1] = lettersString[sum%byte(len(lettersString))]
|
|||
|
return *(*string)(unsafe.Pointer(&b))
|
|||
|
}
|
|||
|
|
|||
|
func LotteryEncryptDecode(str string) bool {
|
|||
|
if len(str) != number {
|
|||
|
return false
|
|||
|
}
|
|||
|
var sum byte
|
|||
|
for i := 0; i < len(str)-1; i++ {
|
|||
|
sum += str[i]
|
|||
|
}
|
|||
|
if lettersString[sum%byte(len(lettersString))] != str[len(str)-1] {
|
|||
|
return false
|
|||
|
}
|
|||
|
return true
|
|||
|
}
|
|||
|
|
|||
|
// =================== CBC ======================
|
|||
|
func AesEncryptCBC(origData []byte, key []byte) (str string) {
|
|||
|
// 分组秘钥
|
|||
|
// NewCipher该函数限制了输入k的长度必须为16, 24或者32
|
|||
|
block, _ := aes.NewCipher(key)
|
|||
|
blockSize := block.BlockSize() // 获取秘钥块的长度
|
|||
|
origData = pkcs5Padding(origData, blockSize) // 补全码
|
|||
|
blockMode := cipher.NewCBCEncrypter(block, key[:blockSize]) // 加密模式
|
|||
|
encrypted := make([]byte, len(origData)) // 创建数组
|
|||
|
blockMode.CryptBlocks(encrypted, origData) // 加密
|
|||
|
|
|||
|
return base64.StdEncoding.EncodeToString(encrypted)
|
|||
|
}
|
|||
|
func AesDecryptCBC(data string, key []byte) (decrypted []byte) {
|
|||
|
encrypted, err := base64.StdEncoding.DecodeString(data)
|
|||
|
if err != nil {
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
block, _ := aes.NewCipher(key) // 分组秘钥
|
|||
|
blockSize := block.BlockSize() // 获取秘钥块的长度
|
|||
|
blockMode := cipher.NewCBCDecrypter(block, key[:blockSize]) // 加密模式
|
|||
|
decrypted = make([]byte, len(encrypted)) // 创建数组
|
|||
|
blockMode.CryptBlocks(decrypted, encrypted) // 解密
|
|||
|
decrypted = pkcs5UnPadding(decrypted) // 去除补全码
|
|||
|
return decrypted
|
|||
|
}
|
|||
|
func pkcs5Padding(ciphertext []byte, blockSize int) []byte {
|
|||
|
padding := blockSize - len(ciphertext)%blockSize
|
|||
|
padtext := bytes.Repeat([]byte{byte(padding)}, padding)
|
|||
|
return append(ciphertext, padtext...)
|
|||
|
}
|
|||
|
func pkcs5UnPadding(origData []byte) []byte {
|
|||
|
length := len(origData)
|
|||
|
unpadding := int(origData[length-1])
|
|||
|
return origData[:(length - unpadding)]
|
|||
|
}
|
|||
|
|
|||
|
// =================== ECB ======================
|
|||
|
func AesEncryptECB(origData []byte, key []byte) (encrypted []byte) {
|
|||
|
cipher, _ := aes.NewCipher(generateKey(key))
|
|||
|
length := (len(origData) + aes.BlockSize) / aes.BlockSize
|
|||
|
plain := make([]byte, length*aes.BlockSize)
|
|||
|
copy(plain, origData)
|
|||
|
pad := byte(len(plain) - len(origData))
|
|||
|
for i := len(origData); i < len(plain); i++ {
|
|||
|
plain[i] = pad
|
|||
|
}
|
|||
|
encrypted = make([]byte, len(plain))
|
|||
|
// 分组分块加密
|
|||
|
for bs, be := 0, cipher.BlockSize(); bs <= len(origData); bs, be = bs+cipher.BlockSize(), be+cipher.BlockSize() {
|
|||
|
cipher.Encrypt(encrypted[bs:be], plain[bs:be])
|
|||
|
}
|
|||
|
|
|||
|
return encrypted
|
|||
|
}
|
|||
|
func AesDecryptECB(encrypted []byte, key []byte) (decrypted []byte) {
|
|||
|
cipher, _ := aes.NewCipher(generateKey(key))
|
|||
|
decrypted = make([]byte, len(encrypted))
|
|||
|
//
|
|||
|
for bs, be := 0, cipher.BlockSize(); bs < len(encrypted); bs, be = bs+cipher.BlockSize(), be+cipher.BlockSize() {
|
|||
|
cipher.Decrypt(decrypted[bs:be], encrypted[bs:be])
|
|||
|
}
|
|||
|
|
|||
|
trim := 0
|
|||
|
if len(decrypted) > 0 {
|
|||
|
trim = len(decrypted) - int(decrypted[len(decrypted)-1])
|
|||
|
}
|
|||
|
|
|||
|
return decrypted[:trim]
|
|||
|
}
|
|||
|
func generateKey(key []byte) (genKey []byte) {
|
|||
|
genKey = make([]byte, 16)
|
|||
|
copy(genKey, key)
|
|||
|
for i := 16; i < len(key); {
|
|||
|
for j := 0; j < 16 && i < len(key); j, i = j+1, i+1 {
|
|||
|
genKey[j] ^= key[i]
|
|||
|
}
|
|||
|
}
|
|||
|
return genKey
|
|||
|
}
|
|||
|
|
|||
|
// =================== CFB ======================
|
|||
|
func AesEncryptCFB(origData []byte, key []byte) (encrypted []byte) {
|
|||
|
block, err := aes.NewCipher(key)
|
|||
|
if err != nil {
|
|||
|
panic(err)
|
|||
|
}
|
|||
|
encrypted = make([]byte, aes.BlockSize+len(origData))
|
|||
|
iv := encrypted[:aes.BlockSize]
|
|||
|
if _, err := io.ReadFull(crand.Reader, iv); err != nil {
|
|||
|
panic(err)
|
|||
|
}
|
|||
|
stream := cipher.NewCFBEncrypter(block, iv)
|
|||
|
stream.XORKeyStream(encrypted[aes.BlockSize:], origData)
|
|||
|
return encrypted
|
|||
|
}
|
|||
|
func AesDecryptCFB(encrypted []byte, key []byte) (decrypted []byte) {
|
|||
|
block, _ := aes.NewCipher(key)
|
|||
|
if len(encrypted) < aes.BlockSize {
|
|||
|
panic("ciphertext too short")
|
|||
|
}
|
|||
|
iv := encrypted[:aes.BlockSize]
|
|||
|
encrypted = encrypted[aes.BlockSize:]
|
|||
|
|
|||
|
stream := cipher.NewCFBDecrypter(block, iv)
|
|||
|
stream.XORKeyStream(encrypted, encrypted)
|
|||
|
return encrypted
|
|||
|
}
|