/* */ #include "MessageDigest.h" #include #include #include "MessageDigestImpl.h" #include "util.h" #include "array_fun.h" namespace aria2 { namespace { struct HashTypeEntry { std::string hashType; int strength; HashTypeEntry(std::string hashType, int strength) : hashType(std::move(hashType)), strength(strength) { } }; } // namespace namespace { HashTypeEntry hashTypes[] = { HashTypeEntry("sha-1", 1), HashTypeEntry("sha-224", 2), HashTypeEntry("sha-256", 3), HashTypeEntry("sha-384", 4), HashTypeEntry("sha-512", 5), HashTypeEntry("md5", 0), HashTypeEntry("adler32", 0), }; } // namespace aria2 MessageDigest::MessageDigest(std::unique_ptr impl) : pImpl_{std::move(impl)} { } MessageDigest::~MessageDigest() = default; std::unique_ptr MessageDigest::sha1() { return make_unique(MessageDigestImpl::sha1()); } std::unique_ptr MessageDigest::create(const std::string& hashType) { return make_unique(MessageDigestImpl::create(hashType)); } bool MessageDigest::supports(const std::string& hashType) { return MessageDigestImpl::supports(hashType); } std::vector MessageDigest::getSupportedHashTypes() { std::vector rv; for (const auto& i : hashTypes) { if (MessageDigestImpl::supports(i.hashType)) { rv.push_back(i.hashType); } } return rv; } std::string MessageDigest::getSupportedHashTypeString() { auto ht = getSupportedHashTypes(); std::stringstream ss; std::copy(std::begin(ht), std::end(ht), std::ostream_iterator(ss, ", ")); auto res = ss.str(); if (!res.empty()) { res.erase(ss.str().length() - 2); } return res; } size_t MessageDigest::getDigestLength(const std::string& hashType) { return MessageDigestImpl::getDigestLength(hashType); } bool MessageDigest::isStronger(const std::string& lhs, const std::string& rhs) { auto lEntry = std::find_if( std::begin(hashTypes), std::end(hashTypes), [&lhs](const HashTypeEntry& entry) { return lhs == entry.hashType; }); auto rEntry = std::find_if( std::begin(hashTypes), std::end(hashTypes), [&rhs](const HashTypeEntry& entry) { return rhs == entry.hashType; }); if (lEntry == std::end(hashTypes)) { return false; } if (rEntry == std::end(hashTypes)) { return true; } return lEntry->strength > rEntry->strength; } bool MessageDigest::isValidHash(const std::string& hashType, const std::string& hexDigest) { return util::isHexDigit(hexDigest) && supports(hashType) && getDigestLength(hashType) * 2 == hexDigest.size(); } std::string MessageDigest::getCanonicalHashType(const std::string& hashType) { // This is really backward compatibility for Metalink3. aria2 only // supported sha-1, sha-256 and md5 at Metalink3 era. So we don't // add alias for sha-224, sha-384 and sha-512. if ("sha1" == hashType) { return "sha-1"; } else if ("sha256" == hashType) { return "sha-256"; } else { return hashType; } } size_t MessageDigest::getDigestLength() const { return pImpl_->getDigestLength(); } void MessageDigest::reset() { pImpl_->reset(); } MessageDigest& MessageDigest::update(const void* data, size_t length) { pImpl_->update(data, length); return *this; } void MessageDigest::digest(unsigned char* md) { pImpl_->digest(md); } std::string MessageDigest::digest() { size_t length = pImpl_->getDigestLength(); auto buf = make_unique(length); pImpl_->digest(buf.get()); return std::string(&buf[0], &buf[length]); } } // namespace aria2