Files
golib/crypt/crypt.go
Nicolas JUHEL 984ba51587 Rework Error interface
Package Errors:
- add function to check & cast error interface into golib Error interface
- update CodeError type to simplify management & error creation
- add function to simplify call of Error function from a generic error interface
- remove some useless function from Error interface

All Other Packages:
- apply change of package Errors into all other packages
2023-08-28 11:22:08 +02:00

121 lines
3.4 KiB
Go

/*
* MIT License
*
* Copyright (c) 2019 Nicolas JUHEL
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*
*/
package crypt
import (
"crypto/aes"
"crypto/cipher"
"crypto/rand"
"encoding/hex"
"io"
"github.com/nabbar/golib/errors"
)
var (
cryptKey = make([]byte, 32)
cryptNonce = make([]byte, 12)
)
func SetKeyHex(key, nonce string) errors.Error {
var err error
// Load your secret key from a safe place and reuse it across multiple
// Seal/Open calls. (Obviously don't use this example key for anything
// real.) If you want to convert a passphrase to a key, use a suitable
// package like bcrypt or scrypt.
cryptKey, err = hex.DecodeString(key)
if err != nil {
return ErrorHexaKey.Error(err)
}
cryptNonce, err = hex.DecodeString(nonce)
if err != nil {
return ErrorHexaNonce.Error(err)
}
return nil
}
func SetKeyByte(key [32]byte, nonce [12]byte) {
cryptKey = key[:]
cryptNonce = nonce[:]
}
func GenKeyByte() ([]byte, []byte, errors.Error) {
// Never use more than 2^32 random key with a given key because of the risk of a repeat.
if _, err := io.ReadFull(rand.Reader, cryptKey); err != nil {
return make([]byte, 32), make([]byte, 12), ErrorByteKeygen.Error(err)
}
// Never use more than 2^32 random nonces with a given key because of the risk of a repeat.
if _, err := io.ReadFull(rand.Reader, cryptNonce); err != nil {
return make([]byte, 32), make([]byte, 12), ErrorByteNonceGen.Error(err)
}
return cryptKey, cryptNonce, nil
}
func Encrypt(clearValue []byte) (string, errors.Error) {
// When decoded the key should be 16 bytes (AES-128) or 32 (AES-256).
block, err := aes.NewCipher(cryptKey)
if err != nil {
return "", ErrorAESBlock.Error(err)
}
aesgcm, err := cipher.NewGCM(block)
if err != nil {
return "", ErrorAESGCM.Error(err)
}
return hex.EncodeToString(aesgcm.Seal(nil, cryptNonce, clearValue, nil)), nil
}
func Decrypt(hexaVal string) ([]byte, errors.Error) {
// When decoded the key should be 16 bytes (AES-128) or 32 (AES-256).
ciphertext, err := hex.DecodeString(hexaVal)
if err != nil {
return nil, ErrorHexaDecode.Error(err)
}
block, err := aes.NewCipher(cryptKey)
if err != nil {
return nil, ErrorAESBlock.Error(err)
}
aesgcm, err := cipher.NewGCM(block)
if err != nil {
return nil, ErrorAESGCM.Error(err)
}
if res, err := aesgcm.Open(nil, cryptNonce, ciphertext, nil); err != nil {
return res, ErrorAESDecrypt.Error(err)
} else {
return res, nil
}
}