1 /*---------------------------------------------------------------------\
3 | |__ / \ / / . \ . \ |
8 \---------------------------------------------------------------------*/
9 /** \file zypp/Digest.cc
11 * \todo replace by Blocxx
15 #include <cstdio> // snprintf
16 #include <openssl/evp.h>
17 #include <openssl/conf.h>
18 #include <openssl/engine.h>
25 #ifdef DIGEST_TESTSUITE
29 #include "zypp/Digest.h"
33 bool DigestReport::askUserToAcceptNoDigest( const zypp::Pathname &file )
36 bool DigestReport::askUserToAccepUnknownDigest( const Pathname &file, const std::string &name )
39 bool DigestReport::askUserToAcceptWrongDigest( const Pathname &file, const std::string &requested, const std::string &found )
43 const std::string & Digest::md5()
44 { static std::string _type( "md5" ); return _type; }
46 const std::string & Digest::sha1()
47 { static std::string _type( "sha1" ); return _type; }
49 const std::string & Digest::sha224()
50 { static std::string _type( "sha224" ); return _type; }
52 const std::string & Digest::sha256()
53 { static std::string _type( "sha256" ); return _type; }
55 const std::string & Digest::sha384()
56 { static std::string _type( "sha384" ); return _type; }
58 const std::string & Digest::sha512()
59 { static std::string _type( "sha512" ); return _type; }
65 const P& operator=(const P& p);
73 unsigned char md_value[EVP_MAX_MD_SIZE];
78 static bool openssl_digests_added;
82 inline bool maybeInit();
83 inline void cleanup();
89 bool Digest::P::openssl_digests_added = false;
103 bool Digest::P::maybeInit()
105 if(!openssl_digests_added)
107 OPENSSL_config(NULL);
108 ENGINE_load_builtin_engines();
109 ENGINE_register_all_complete();
110 OpenSSL_add_all_digests();
111 openssl_digests_added = true;
116 md = EVP_get_digestbyname(name.c_str());
120 EVP_MD_CTX_init(&mdctx);
122 if(!EVP_DigestInit_ex(&mdctx, md, NULL))
126 ::memset(md_value, 0, sizeof(md_value));
132 void Digest::P::cleanup()
136 EVP_MD_CTX_cleanup(&mdctx);
142 Digest::Digest() : _dp(new P())
151 bool Digest::create(const std::string& name)
153 if(name.empty()) return false;
160 return _dp->maybeInit();
163 const std::string& Digest::name()
170 if (!_dp->initialized)
174 (void)EVP_DigestFinal_ex(&_dp->mdctx, _dp->md_value, &_dp->md_len);
175 _dp->finalized = true;
177 if(!EVP_DigestInit_ex(&_dp->mdctx, _dp->md, NULL))
179 _dp->finalized = false;
183 std::string Digest::digest()
185 if(!_dp->maybeInit())
186 return std::string();
190 if(!EVP_DigestFinal_ex(&_dp->mdctx, _dp->md_value, &_dp->md_len))
191 return std::string();
193 _dp->finalized = true;
196 char mdtxt[_dp->md_len*2 + 1];
197 mdtxt[_dp->md_len*2] = '\0';
199 for(unsigned i = 0; i < _dp->md_len; ++i)
201 ::snprintf(mdtxt + i*2, 3, "%02hhx", _dp->md_value[i]);
204 return std::string(mdtxt);
207 std::vector<unsigned char> Digest::digestVector()
209 std::vector<unsigned char> r;
210 if(!_dp->maybeInit())
215 if(!EVP_DigestFinal_ex(&_dp->mdctx, _dp->md_value, &_dp->md_len))
217 _dp->finalized = true;
219 r.reserve(_dp->md_len);
220 for(unsigned i = 0; i < _dp->md_len; ++i)
221 r.push_back(_dp->md_value[i]);
225 bool Digest::update(const char* bytes, size_t len)
232 if(!_dp->maybeInit())
238 if(!_dp->maybeInit())
242 if(!EVP_DigestUpdate(&_dp->mdctx, reinterpret_cast<const unsigned char*>(bytes), len))
248 std::string Digest::digest(const std::string& name, std::istream& is, size_t bufsize)
250 if(name.empty() || !is)
256 if(!digest.create(name))
263 is.read(buf, bufsize);
264 readed = is.gcount();
265 if(readed && !digest.update(buf, readed))
269 return digest.digest();
272 std::string Digest::digest( const std::string & name, const std::string & input, size_t bufsize )
274 istringstream is( input );
275 return digest( name, is, bufsize );
278 #ifdef DIGEST_TESTSUITE
279 int main(int argc, char *argv[])
281 bool openssl = false;
284 if(argc > 1 && string(argv[argpos]) == "--openssl")
290 if(argc - argpos < 2)
292 cerr << "Usage: " << argv[0] << " <DIGESTNAME> <FILE>" << endl;
296 const char* digestname = argv[argpos++];
297 const char* fn = argv[argpos++];
301 string digest = Digest::digest(digestname, file);
304 cout << digestname << "(" << fn << ")= " << digest << endl;
306 cout << digest << " " << fn << endl;