89 lines
		
	
	
		
			2.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			89 lines
		
	
	
		
			2.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package aeshelper
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"crypto/aes"
 | |
| 	"crypto/cipher"
 | |
| 	"crypto/md5"
 | |
| 	"crypto/rand"
 | |
| 	"encoding/base64"
 | |
| 	"io"
 | |
| )
 | |
| 
 | |
| // Encrypt text with the passphrase
 | |
| func Encrypt(text string, passphrase string) string {
 | |
| 	salt := make([]byte, 8)
 | |
| 	if _, err := io.ReadFull(rand.Reader, salt); err != nil {
 | |
| 		panic(err.Error())
 | |
| 	}
 | |
| 
 | |
| 	key, iv := DeriveKeyAndIv(passphrase, string(salt))
 | |
| 
 | |
| 	block, err := aes.NewCipher([]byte(key))
 | |
| 	if err != nil {
 | |
| 		panic(err)
 | |
| 	}
 | |
| 
 | |
| 	pad := PKCS7Padding([]byte(text), block.BlockSize())
 | |
| 	ecb := cipher.NewCBCEncrypter(block, []byte(iv))
 | |
| 	encrypted := make([]byte, len(pad))
 | |
| 	ecb.CryptBlocks(encrypted, pad)
 | |
| 
 | |
| 	return base64.StdEncoding.EncodeToString([]byte("Salted__" + string(salt) + string(encrypted)))
 | |
| }
 | |
| 
 | |
| // Decrypt encrypted text with the passphrase
 | |
| func Decrypt(encrypted string, passphrase string) string {
 | |
| 	ct, _ := base64.StdEncoding.DecodeString(encrypted)
 | |
| 	if len(ct) < 16 || string(ct[:8]) != "Salted__" {
 | |
| 		return ""
 | |
| 	}
 | |
| 
 | |
| 	salt := ct[8:16]
 | |
| 	ct = ct[16:]
 | |
| 	key, iv := DeriveKeyAndIv(passphrase, string(salt))
 | |
| 
 | |
| 	block, err := aes.NewCipher([]byte(key))
 | |
| 	if err != nil {
 | |
| 		panic(err)
 | |
| 	}
 | |
| 
 | |
| 	cbc := cipher.NewCBCDecrypter(block, []byte(iv))
 | |
| 	dst := make([]byte, len(ct))
 | |
| 	cbc.CryptBlocks(dst, ct)
 | |
| 
 | |
| 	return string(PKCS7Trimming(dst))
 | |
| }
 | |
| 
 | |
| // PKCS7Padding PKCS7Padding
 | |
| func PKCS7Padding(ciphertext []byte, blockSize int) []byte {
 | |
| 	padding := blockSize - len(ciphertext)%blockSize
 | |
| 	padtext := bytes.Repeat([]byte{byte(padding)}, padding)
 | |
| 	return append(ciphertext, padtext...)
 | |
| }
 | |
| 
 | |
| // PKCS7Trimming PKCS7Trimming
 | |
| func PKCS7Trimming(encrypt []byte) []byte {
 | |
| 	padding := encrypt[len(encrypt)-1]
 | |
| 	return encrypt[:len(encrypt)-int(padding)]
 | |
| }
 | |
| 
 | |
| // DeriveKeyAndIv DeriveKeyAndIv
 | |
| func DeriveKeyAndIv(passphrase string, salt string) (string, string) {
 | |
| 	salted := ""
 | |
| 	dI := ""
 | |
| 
 | |
| 	for len(salted) < 48 {
 | |
| 		md := md5.New()
 | |
| 		md.Write([]byte(dI + passphrase + salt))
 | |
| 		dM := md.Sum(nil)
 | |
| 		dI = string(dM[:16])
 | |
| 		salted = salted + dI
 | |
| 	}
 | |
| 
 | |
| 	key := salted[0:32]
 | |
| 	iv := salted[32:48]
 | |
| 
 | |
| 	return key, iv
 | |
| }
 |