You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
OneAuth/libs/utils/crypto.go

174 lines
4.0 KiB
Go

package utils
//
// crypto.go
// Copyright (C) 2020 light <light@1870499383@qq.com>
//
// Distributed under terms of the MIT license.
//
import (
"bytes"
"crypto"
"crypto/aes"
"crypto/cipher"
"crypto/rand"
"crypto/rsa"
"crypto/sha256"
"crypto/x509"
"encoding/base64"
"encoding/pem"
"errors"
)
// PKCS7Padding 添加 PKCS#7 填充
func PKCS7Padding(ciphertext []byte, blockSize int) []byte {
padding := blockSize - len(ciphertext)%blockSize
padtext := bytes.Repeat([]byte{byte(padding)}, padding)
return append(ciphertext, padtext...)
}
// PKCS7UnPadding 移除 PKCS#7 填充
func PKCS7UnPadding(origData []byte) ([]byte, bool) {
length := len(origData)
unpadding := int(origData[length-1])
if unpadding >= length {
return nil, false
}
return origData[:(length - unpadding)], true
}
// AesEncrypt 使用 AES-256-CBC 进行加密
// key 256 bit / 32 Byte
// iv 128 bit / 16 Byte
func AesEncrypt(plaintext, key, iv []byte) (string, error) {
block, err := aes.NewCipher(key)
if err != nil {
return "", err
}
plaintext = PKCS7Padding(plaintext, block.BlockSize())
blockMode := cipher.NewCBCEncrypter(block, iv)
crypted := make([]byte, len(plaintext))
blockMode.CryptBlocks(crypted, plaintext)
return base64.StdEncoding.EncodeToString(crypted), nil
}
// AesDecrypt 使用 AES-256-CBC 进行解密
func AesDecrypt(encrypted, key, iv []byte) (string, error) {
block, err := aes.NewCipher(key)
if err != nil {
return "", err
}
encrypted, err = base64.StdEncoding.DecodeString(string(encrypted))
if err != nil {
return "", err
}
blockMode := cipher.NewCBCDecrypter(block, iv)
origData := make([]byte, len(encrypted))
blockMode.CryptBlocks(origData, encrypted)
origData, ok := PKCS7UnPadding(origData)
if !ok {
return "", errors.New("PKCS7UnPadding error")
}
return string(origData), nil
}
// rsa
func GetRsaKey(bits int) (public *rsa.PublicKey, private *rsa.PrivateKey, err error) {
private, err = rsa.GenerateKey(rand.Reader, bits)
if err != nil {
return
}
public = &private.PublicKey
return
}
func GetPublicStr(key *rsa.PublicKey) (string, error) {
der, err := x509.MarshalPKIXPublicKey(key)
if err != nil {
return "", err
}
publicBlock := &pem.Block{
Type: "PUBLIC KEY",
Bytes: der,
}
return string(pem.EncodeToMemory(publicBlock)), nil
}
func GetPrivateStr(key *rsa.PrivateKey) (string, error) {
derStream := x509.MarshalPKCS1PrivateKey(key)
priBlock := &pem.Block{
Type: "RSA PRIVATE KEY",
Bytes: derStream,
}
return string(pem.EncodeToMemory(priBlock)), nil
}
func GetPublicFromStr(key string) (*rsa.PublicKey, error) {
//解密pem格式的公钥
block, _ := pem.Decode([]byte(key))
if block == nil {
return nil, errors.New("public key error")
}
// 解析公钥
pubInterface, err := x509.ParsePKIXPublicKey(block.Bytes)
if err != nil {
return nil, err
}
// 类型断言
pub := pubInterface.(*rsa.PublicKey)
return pub, nil
}
func GetPrivateFromStr(key string) (*rsa.PrivateKey, error) {
block, _ := pem.Decode([]byte(key))
if block == nil {
return nil, errors.New("private key error")
}
//解析PKCS1格式的私钥
return x509.ParsePKCS1PrivateKey(block.Bytes)
}
func RsaEncode(msg string, key *rsa.PublicKey) (string, error) {
encryptedBytes, err := rsa.EncryptOAEP(
sha256.New(),
rand.Reader,
key,
[]byte(msg),
nil)
if err != nil {
return "", err
}
return ToBase64(encryptedBytes), nil
}
func RsaDecode(msg string, key *rsa.PrivateKey) (string, error) {
raw, err := FromBase64(msg)
if err != nil {
return "", err
}
decryptedBytes, err := key.Decrypt(nil, raw, &rsa.OAEPOptions{Hash: crypto.SHA256})
return string(decryptedBytes), err
}
func RsaSign(msg string, key *rsa.PrivateKey) (string, error) {
signature, err := rsa.SignPSS(rand.Reader, key, crypto.SHA256, HashSha256Byte([]byte(msg)), nil)
if err != nil {
return "", err
}
return ToBase64(signature), nil
}
func RsaCheckSign(msg string, sign string, key *rsa.PublicKey) error {
raw, err := FromBase64(sign)
if err != nil {
return err
}
return rsa.VerifyPSS(key, crypto.SHA256, HashSha256Byte([]byte(msg)), raw, nil)
}