/* **************************************************************************** * Copyright 2019 Open Systems Development BV * * * * 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. * * ***************************************************************************/ #include "crypter.h" // std #include #include #include #include #include // openssl #include // osdev::components #include "log.h" #include "scopeguard.h" using namespace osdev::components; namespace { const EVP_MD* getAlgo( Crypter::AlgorithmEnum algo ) { switch ( algo ) { case Crypter::AlgorithmEnum::MD0: return EVP_md_null(); case Crypter::AlgorithmEnum::MD2: // return EVP_md2(); case Crypter::AlgorithmEnum::MD5: return EVP_md5(); case Crypter::AlgorithmEnum::MDC2: // deprecated. // md = EVP_mdc2(); return EVP_md_null(); case Crypter::AlgorithmEnum::SHA1: return EVP_sha1(); case Crypter::AlgorithmEnum::SHA224: return EVP_sha224(); case Crypter::AlgorithmEnum::SHA256: return EVP_sha256(); case Crypter::AlgorithmEnum::SHA384: return EVP_sha384(); case Crypter::AlgorithmEnum::SHA512: return EVP_sha512(); case Crypter::AlgorithmEnum::RIPEMD160: return EVP_ripemd160(); } throw std::invalid_argument("Crypto algorithm not found."); } } // anonymous Crypter::Crypter() { } std::string Crypter::encrypt( const std::string& message, Crypter::AlgorithmEnum algo ) { // Create the environment auto md = getAlgo(algo); #if OPENSSL_VERSION_NUMBER >= 0x1010008fL auto mdContext = std::unique_ptr( EVP_MD_CTX_create(), [](EVP_MD_CTX* ptr){EVP_MD_CTX_free(ptr);}); // This will call EVP_cleanup if the guard goes out of scope. ScopeGuard oGuard( &EVP_PBE_cleanup ); #else auto mdContext = std::unique_ptr( EVP_MD_CTX_create(), [](EVP_MD_CTX* ptr){EVP_MD_CTX_destroy(ptr);}); // This will call EVP_cleanup if the guard goes out of scope. ScopeGuard oGuard( &EVP_cleanup ); #endif (void)oGuard; // Satisfy the compiler for unused variables. auto errorCode = EVP_DigestInit_ex( mdContext.get(), md, NULL ); if( 1 != errorCode ) { LogError( "[Crypter::encrypt]", QString( "No encryption digest environment created." ) ); throw std::system_error( errorCode, std::system_category(), "No encryption digest environment created." ); } // Update the environment with the message errorCode = EVP_DigestUpdate( mdContext.get(), message.c_str(), message.length() ); if( 1 != errorCode ) { LogError( "[Crypter::encrypt]", QString("Digest failed.") ); throw std::system_error( errorCode, std::system_category(), "Digest failed.." ); } // End the Digest so we can read the crypted message. unsigned int mdLen; unsigned char mdValue[EVP_MAX_MD_SIZE]; errorCode = EVP_DigestFinal_ex( mdContext.get(), mdValue, &mdLen ); if( 1 != errorCode ) { LogError( "[Crypter::encrypt]", QString("There was an error closing the digest environment.") ); throw std::system_error( errorCode, std::system_category(), "There was an error closing the digest environment." ); } // If we got here, all went well. We retrieve the crypted message // through a stringstream : convert to hex, padding and width and return the string. std::stringstream ss; for( unsigned int nIndex = 0; nIndex < mdLen; nIndex++ ) { ss << std::hex << std::setw( 2 ) << std::setfill( '0' ) << static_cast( mdValue[nIndex] ); } return ss.str(); }