/** * Copyright 2021 BrutalWizard (https://github.com/bru74lw1z4rd). All Rights Reserved. * * Licensed under the Apache License 2.0 (the "License"). You may not use * this file except in compliance with the License. You can obtain a copy * in the file LICENSE in the source distribution **/ #include "include/QRsa.h" QSimpleCrypto::QRsa::QRsa() { } /// /// \brief QSimpleCrypto::QRSA::generateRsaKeys - Function generate Rsa Keys and returns them in OpenSSL structure. /// \param bits - RSA key size. /// \param rsaBigNumber - The exponent is an odd number, typically 3, 17 or 65537. /// \return Returns 'OpenSSL RSA structure' or 'nullptr', if error happened. Returned value must be cleaned up with 'RSA_free()' to avoid memory leak. /// RSA* QSimpleCrypto::QRsa::generateRsaKeys(const int& bits, const int& rsaBigNumber) { try { /* Initialize big number */ std::unique_ptr bigNumber { BN_new(), BN_free }; if (bigNumber == nullptr) { throw std::runtime_error("Couldn't initialize \'bigNumber\'. BN_new(). Error: " + QByteArray(ERR_error_string(ERR_get_error(), nullptr))); return nullptr; } /* Set big number */ if (!BN_set_word(bigNumber.get(), rsaBigNumber)) { throw std::runtime_error("Couldn't set bigNumber. BN_set_word(). Error: " + QByteArray(ERR_error_string(ERR_get_error(), nullptr))); } /* Initialize RSA */ RSA* rsa = nullptr; if (!(rsa = RSA_new())) { throw std::runtime_error("Couldn't initialize x509. X509_new(). Error: " + QByteArray(ERR_error_string(ERR_get_error(), nullptr))); } /* Generate key pair and store it in RSA */ if (!RSA_generate_key_ex(rsa, bits, bigNumber.get(), nullptr)) { throw std::runtime_error("Couldn't generate RSA. RSA_generate_key_ex(). Error: " + QByteArray(ERR_error_string(ERR_get_error(), nullptr))); } return rsa; } catch (std::exception& exception) { QSimpleCrypto::QRsa::error.setError(1, exception.what()); return nullptr; } catch (...) { QSimpleCrypto::QRsa::error.setError(2, "Unknown error!"); return nullptr; } } /// /// \brief QSimpleCrypto::QRSA::savePublicKey - Saves to file RSA public key. /// \param rsa - OpenSSL RSA structure. /// \param publicKeyFileName - Public key file name. /// void QSimpleCrypto::QRsa::savePublicKey(RSA* rsa, const QByteArray& publicKeyFileName) { try { /* Initialize BIO */ std::unique_ptr bioPublicKey { BIO_new_file(publicKeyFileName.data(), "w+"), BIO_free_all }; if (bioPublicKey == nullptr) { throw std::runtime_error("Couldn't initialize \'bioPublicKey\'. BIO_new_file(). Error: " + QByteArray(ERR_error_string(ERR_get_error(), nullptr))); } /* Write public key on file */ if (!PEM_write_bio_RSA_PUBKEY(bioPublicKey.get(), rsa)) { throw std::runtime_error("Couldn't save public key. PEM_write_bio_RSAPublicKey(). Error: " + QByteArray(ERR_error_string(ERR_get_error(), nullptr))); } } catch (std::exception& exception) { QSimpleCrypto::QRsa::error.setError(1, exception.what()); return; } catch (...) { QSimpleCrypto::QRsa::error.setError(2, "Unknown error!"); return; } } /// /// \brief QSimpleCrypto::QRSA::savePrivateKey - Saves to file RSA private key. /// \param rsa - OpenSSL RSA structure. /// \param privateKeyFileName - Private key file name. /// \param password - Private key password. /// \param cipher - Can be used with 'OpenSSL EVP_CIPHER' (ecb, cbc, cfb, ofb, ctr) - 128, 192, 256. Example: EVP_aes_256_cbc(). /// void QSimpleCrypto::QRsa::savePrivateKey(RSA* rsa, const QByteArray& privateKeyFileName, QByteArray password, const EVP_CIPHER* cipher) { try { /* Initialize BIO */ std::unique_ptr bioPrivateKey { BIO_new_file(privateKeyFileName.data(), "w+"), BIO_free_all }; if (bioPrivateKey == nullptr) { throw std::runtime_error("Couldn't initialize bioPrivateKey. BIO_new_file(). Error: " + QByteArray(ERR_error_string(ERR_get_error(), nullptr))); } /* Write private key to file */ if (!PEM_write_bio_RSAPrivateKey(bioPrivateKey.get(), rsa, cipher, reinterpret_cast(password.data()), password.size(), nullptr, nullptr)) { throw std::runtime_error("Couldn't save private key. PEM_write_bio_RSAPrivateKey(). Error: " + QByteArray(ERR_error_string(ERR_get_error(), nullptr))); } } catch (std::exception& exception) { QSimpleCrypto::QRsa::error.setError(1, exception.what()); return; } catch (...) { QSimpleCrypto::QRsa::error.setError(2, "Unknown error!"); return; } } /// /// \brief QSimpleCrypto::QRSA::getPublicKeyFromFile - Gets RSA public key from a file. /// \param filePath - File path to public key file. /// \return Returns 'OpenSSL EVP_PKEY structure' or 'nullptr', if error happened. Returned value must be cleaned up with 'EVP_PKEY_free()' to avoid memory leak. /// EVP_PKEY* QSimpleCrypto::QRsa::getPublicKeyFromFile(const QByteArray& filePath) { try { /* Initialize BIO */ std::unique_ptr bioPublicKey { BIO_new_file(filePath.data(), "r"), BIO_free_all }; if (bioPublicKey == nullptr) { throw std::runtime_error("Couldn't initialize bioPublicKey. BIO_new_file(). Error: " + QByteArray(ERR_error_string(ERR_get_error(), nullptr))); } /* Initialize EVP_PKEY */ EVP_PKEY* keyStore = nullptr; if (!(keyStore = EVP_PKEY_new())) { throw std::runtime_error("Couldn't initialize keyStore. EVP_PKEY_new(). Error: " + QByteArray(ERR_error_string(ERR_get_error(), nullptr))); } /* Write private key to file */ if (!PEM_read_bio_PUBKEY(bioPublicKey.get(), &keyStore, nullptr, nullptr)) { throw std::runtime_error("Couldn't read private key. PEM_read_bio_PrivateKey(). Error: " + QByteArray(ERR_error_string(ERR_get_error(), nullptr))); } return keyStore; } catch (std::exception& exception) { QSimpleCrypto::QRsa::error.setError(1, exception.what()); return nullptr; } catch (...) { QSimpleCrypto::QRsa::error.setError(2, "Unknown error!"); return nullptr; } } /// /// \brief QSimpleCrypto::QRSA::getPrivateKeyFromFile - Gets RSA private key from a file. /// \param filePath - File path to private key file. /// \param password - Private key password. /// \return - Returns 'OpenSSL EVP_PKEY structure' or 'nullptr', if error happened. Returned value must be cleaned up with 'EVP_PKEY_free()' to avoid memory leak. /// EVP_PKEY* QSimpleCrypto::QRsa::getPrivateKeyFromFile(const QByteArray& filePath, const QByteArray& password) { try { /* Initialize BIO */ std::unique_ptr bioPrivateKey { BIO_new_file(filePath.data(), "r"), BIO_free_all }; if (bioPrivateKey == nullptr) { throw std::runtime_error("Couldn't initialize bioPrivateKey. BIO_new_file(). Error: " + QByteArray(ERR_error_string(ERR_get_error(), nullptr))); } /* Initialize EVP_PKEY */ EVP_PKEY* keyStore = nullptr; if (!(keyStore = EVP_PKEY_new())) { throw std::runtime_error("Couldn't initialize keyStore. EVP_PKEY_new(). Error: " + QByteArray(ERR_error_string(ERR_get_error(), nullptr))); } /* Write private key to file */ if (!PEM_read_bio_PrivateKey(bioPrivateKey.get(), &keyStore, nullptr, (void*)password.data())) { throw std::runtime_error("Couldn't read private key. PEM_read_bio_PrivateKey(). Error: " + QByteArray(ERR_error_string(ERR_get_error(), nullptr))); } return keyStore; } catch (std::exception& exception) { QSimpleCrypto::QRsa::error.setError(1, exception.what()); return nullptr; } catch (...) { QSimpleCrypto::QRsa::error.setError(2, "Unknown error!"); return nullptr; } } /// /// \brief QSimpleCrypto::QRSA::encrypt - Encrypt data with RSA algorithm. /// \param plaintext - Text that must be encrypted. /// \param rsa - OpenSSL RSA structure. /// \param encryptType - Public or private encrypt type. (PUBLIC_ENCRYPT, PRIVATE_ENCRYPT). /// \param padding - OpenSSL RSA padding can be used with: 'RSA_PKCS1_PADDING', 'RSA_NO_PADDING' and etc. /// \return Returns encrypted data or "", if error happened. /// QByteArray QSimpleCrypto::QRsa::encrypt(QByteArray plainText, RSA* rsa, const int& encryptType, const int& padding) { try { /* Initialize array. Here encrypted data will be saved */ std::unique_ptr cipherText { new unsigned char[RSA_size(rsa)]() }; if (cipherText == nullptr) { throw std::runtime_error("Couldn't allocate memory for 'cipherText'."); } /* Result of encryption operation */ short int result = 0; /* Execute encryption operation */ if (encryptType == PublicDecrypt) { result = RSA_public_encrypt(plainText.size(), reinterpret_cast(plainText.data()), cipherText.get(), rsa, padding); } else if (encryptType == PrivateDecrypt) { result = RSA_private_encrypt(plainText.size(), reinterpret_cast(plainText.data()), cipherText.get(), rsa, padding); } /* Check for result */ if (result <= -1) { throw std::runtime_error("Couldn't encrypt data. Error: " + QByteArray(ERR_error_string(ERR_get_error(), nullptr))); } /* Get encrypted data */ const QByteArray& encryptedData = QByteArray(reinterpret_cast(cipherText.get()), RSA_size(rsa)); return encryptedData; } catch (std::exception& exception) { QSimpleCrypto::QRsa::error.setError(1, exception.what()); return ""; } catch (...) { QSimpleCrypto::QRsa::error.setError(2, "Unknown error!"); return ""; } } /// /// \brief QSimpleCrypto::QRSA::decrypt - Decrypt data with RSA algorithm. /// \param cipherText - Text that must be decrypted. /// \param rsa - OpenSSL RSA structure. /// \param decryptType - Public or private type. (PUBLIC_DECRYPT, PRIVATE_DECRYPT). /// \param padding - RSA padding can be used with: 'RSA_PKCS1_PADDING', 'RSA_NO_PADDING' and etc. /// \return - Returns decrypted data or "", if error happened. /// QByteArray QSimpleCrypto::QRsa::decrypt(QByteArray cipherText, RSA* rsa, const int& decryptType, const int& padding) { try { /* Initialize array. Here decrypted data will be saved */ std::unique_ptr plainText { new unsigned char[cipherText.size()]() }; if (plainText == nullptr) { throw std::runtime_error("Couldn't allocate memory for 'plainText'."); } /* Result of decryption operation */ short int result = 0; /* Execute decryption operation */ if (decryptType == PublicDecrypt) { result = RSA_public_decrypt(RSA_size(rsa), reinterpret_cast(cipherText.data()), plainText.get(), rsa, padding); } else if (decryptType == PrivateDecrypt) { result = RSA_private_decrypt(RSA_size(rsa), reinterpret_cast(cipherText.data()), plainText.get(), rsa, padding); } /* Check for result */ if (result <= -1) { throw std::runtime_error("Couldn't decrypt data. Error: " + QByteArray(ERR_error_string(ERR_get_error(), nullptr))); } /* Get decrypted data */ const QByteArray& decryptedData = QByteArray(reinterpret_cast(plainText.get())); return decryptedData; } catch (std::exception& exception) { QSimpleCrypto::QRsa::error.setError(1, exception.what()); return ""; } catch (...) { QSimpleCrypto::QRsa::error.setError(2, "Unknown error!"); return ""; } }